<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="cs">
	<id>http://postgres.cz/index.php?action=history&amp;feed=atom&amp;title=Polymorfn%C3%AD_funkce_assert_equal</id>
	<title>Polymorfní funkce assert equal - Historie editací</title>
	<link rel="self" type="application/atom+xml" href="http://postgres.cz/index.php?action=history&amp;feed=atom&amp;title=Polymorfn%C3%AD_funkce_assert_equal"/>
	<link rel="alternate" type="text/html" href="http://postgres.cz/index.php?title=Polymorfn%C3%AD_funkce_assert_equal&amp;action=history"/>
	<updated>2026-05-12T22:32:25Z</updated>
	<subtitle>Historie editací této stránky</subtitle>
	<generator>MediaWiki 1.43.3</generator>
	<entry>
		<id>http://postgres.cz/index.php?title=Polymorfn%C3%AD_funkce_assert_equal&amp;diff=402&amp;oldid=prev</id>
		<title>imported&gt;Pavel: Zrušena verze 3186 od uživatele Pavel (diskuse)</title>
		<link rel="alternate" type="text/html" href="http://postgres.cz/index.php?title=Polymorfn%C3%AD_funkce_assert_equal&amp;diff=402&amp;oldid=prev"/>
		<updated>2015-06-02T06:14:17Z</updated>

		<summary type="html">&lt;p&gt;Zrušena verze 3186 od uživatele &lt;a href=&quot;/wiki/Speci%C3%A1ln%C3%AD:P%C5%99%C3%ADsp%C4%9Bvky/Pavel&quot; title=&quot;Speciální:Příspěvky/Pavel&quot;&gt;Pavel&lt;/a&gt; (&lt;a href=&quot;/index.php?title=Diskuse_s_u%C5%BEivatelem:Pavel&amp;amp;action=edit&amp;amp;redlink=1&quot; class=&quot;new&quot; title=&quot;Diskuse s uživatelem:Pavel (stránka neexistuje)&quot;&gt;diskuse&lt;/a&gt;)&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Nová stránka&lt;/b&gt;&lt;/p&gt;&lt;div&gt;Funkce &amp;#039;&amp;#039;plunit_assert_equals&amp;#039;&amp;#039; je ukázkou polymorfní funkce sloužící k porovnání dvou hodnot stejného typu. O přetypování se postará zajištění polymorfních typů. Funkce řeší pouze porovnání dvou hodnot. Jelikož dopředu nevíme o jaké typy se bude jednat, nelze použít statické řešení, tj. převod na C typy a klasické porovnání. Místo toho se dohledává funkce obsluhující operátor &amp;#039;&amp;#039;=&amp;#039;&amp;#039;. Jelikož tato funkce může být linkována k PostgreSQL dynamicky nelze ji volat staticky ale pouze použitím id (v případě zákaznických typů).&lt;br /&gt;
&lt;br /&gt;
Funkce &amp;#039;&amp;#039;assert_equals_base&amp;#039;&amp;#039; je opakovaně použita. Použití parametru typu &amp;#039;&amp;#039;FunctionCallInfo&amp;#039;&amp;#039; jednoduše umožňuje předat externí (SQL) parametry. Pokud by se nepoužil tento přístup, pak by se parametry funkce, přístup do cache funkce zpracovávali pouze v &amp;#039;&amp;#039;PG_FUNCTION_INFO_V1&amp;#039;&amp;#039; funkcích.&lt;br /&gt;
&amp;lt;pre&amp;gt;&lt;br /&gt;
static bool &lt;br /&gt;
assert_equals_base(FunctionCallInfo fcinfo) &lt;br /&gt;
{ &lt;br /&gt;
	Datum	value1 = PG_GETARG_DATUM(0); &lt;br /&gt;
 	Datum	value2 = PG_GETARG_DATUM(1); &lt;br /&gt;
 	Oid	*ptr; &lt;br /&gt;
 &lt;br /&gt;
	ptr = (Oid *) fcinfo-&amp;gt;flinfo-&amp;gt;fn_extra; &lt;br /&gt;
	if (ptr == NULL) &lt;br /&gt;
	{ &lt;br /&gt;
		Oid	valtype = get_fn_expr_argtype(fcinfo-&amp;gt;flinfo, 0); &lt;br /&gt;
  		Oid eqopfcid; &lt;br /&gt;
 &lt;br /&gt;
		if (!OidIsValid(valtype)) &lt;br /&gt;
			elog(ERROR, &amp;quot;could not determine data type of input&amp;quot;); &lt;br /&gt;
 &lt;br /&gt;
		eqopfcid = equality_oper_funcid(valtype); &lt;br /&gt;
  &lt;br /&gt;
		if (!OidIsValid(eqopfcid)) &lt;br /&gt;
			ereport(ERROR,. &lt;br /&gt;
    				(errcode(ERRCODE_INVALID_PARAMETER_VALUE), &lt;br /&gt;
				 errmsg(&amp;quot;unknown equal operand for datatype&amp;quot;)));&lt;br /&gt;
 &lt;br /&gt;
    		/* First time calling for current query: allocate storage */ &lt;br /&gt;
         	fcinfo-&amp;gt;flinfo-&amp;gt;fn_extra = MemoryContextAlloc(fcinfo-&amp;gt;flinfo-&amp;gt;fn_mcxt, &lt;br /&gt;
									sizeof(Oid));&lt;br /&gt;
 &lt;br /&gt;
		ptr = (Oid *) fcinfo-&amp;gt;flinfo-&amp;gt;fn_extra; &lt;br /&gt;
		*ptr = eqopfcid; &lt;br /&gt;
	} &lt;br /&gt;
	return DatumGetBool(OidFunctionCall2(*ptr, value1, value2));&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/pre&amp;gt;&lt;br /&gt;
Dohledání Oid funkce porovnání pro specifikovaný typ je relativně náročná operace, dohledaní Oid se ukládá do cache funkce. Funkce se aktivuje voláním &amp;#039;&amp;#039;OidFunctionCall2&amp;#039;&amp;#039;. Dvojka na konci určuje, že funkci porovnání se předávají dva Datum parametry.&lt;br /&gt;
&lt;br /&gt;
Funkce &amp;#039;&amp;#039;assert_equals&amp;#039;&amp;#039; existuje ve dvou variantách - s explicitní chybovým výpisem a s defaultním chybovým výpisem. Proto také existují dvě C funkce: &amp;#039;&amp;#039;plunit_assert_equals&amp;#039;&amp;#039; a &amp;#039;&amp;#039;plunit_assert_equals_message&amp;#039;&amp;#039;. Teoreticky by stačila pouze jedna C funkce, která by sloužila ve dvou SQL funkcích. Prakticky sdílení C funkcí není zvykem, takže existují také dvě C funkce. &amp;#039;&amp;#039;plunit_assert_equals&amp;#039;&amp;#039; je pouze wrapper, který přesměrovává volání na funkci &amp;#039;&amp;#039;plunit_assert_equals_message&amp;#039;&amp;#039;. Parametry se předávají v default proměnné fcinfo. Tento parametr je definován makrem &amp;#039;&amp;#039;PG_FUNCTION_ARGS&amp;#039;&amp;#039;.&lt;br /&gt;
&amp;lt;pre&amp;gt;&lt;br /&gt;
Datum.&lt;br /&gt;
plunit_assert_equals(PG_FUNCTION_ARGS)&lt;br /&gt;
{&lt;br /&gt;
	return plunit_assert_equals_message(fcinfo);&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/pre&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Test zda-li je chybová hláška zadaná je ve funkci &amp;#039;&amp;#039;assert_get_message&amp;#039;&amp;#039;. Tato funkce zároveň testuje, že pokud je chybová hláška zadaná explicitně, pak není NULL:&lt;br /&gt;
&amp;lt;pre&amp;gt;&lt;br /&gt;
static char *&lt;br /&gt;
assert_get_message(FunctionCallInfo fcinfo, int nargs, char *message)&lt;br /&gt;
{&lt;br /&gt;
	char *result;&lt;br /&gt;
&lt;br /&gt;
	if (PG_NARGS() == nargs)&lt;br /&gt;
	{&lt;br /&gt;
  		text	*msg;&lt;br /&gt;
  &lt;br /&gt;
		if (PG_ARGISNULL(nargs - 1))&lt;br /&gt;
			ereport(ERROR,&lt;br /&gt;
				(errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),&lt;br /&gt;
				 errmsg(&amp;quot;message is NULL&amp;quot;),&lt;br /&gt;
				 errdetail(&amp;quot;Message may not be NULL.&amp;quot;)));&lt;br /&gt;
&lt;br /&gt;
		msg = PG_GETARG_TEXT_P(nargs - 1);&lt;br /&gt;
		result = TextPGetCString(msg);&lt;br /&gt;
	}&lt;br /&gt;
	else&lt;br /&gt;
		result = message;  &lt;br /&gt;
&lt;br /&gt;
	return result;&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/pre&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Vlastní funkce &amp;#039;&amp;#039;plunit_assert_equals_message&amp;#039;&amp;#039; pouze spojuje výše uvedené funkce:&lt;br /&gt;
&amp;lt;pre&amp;gt;&lt;br /&gt;
Datum.&lt;br /&gt;
plunit_assert_equals_message(PG_FUNCTION_ARGS)&lt;br /&gt;
{&lt;br /&gt;
	char *message = assert_get_message(fcinfo, 3, &amp;quot;plunit.assert_equal excep&lt;br /&gt;
&lt;br /&gt;
	/* skip all tests for NULL value */&lt;br /&gt;
	if (PG_ARGISNULL(0) || PG_ARGISNULL(1))&lt;br /&gt;
		ereport(ERROR,&lt;br /&gt;
				(errcode(ERRCODE_CHECK_VIOLATION),&lt;br /&gt;
 				 errmsg(message),&lt;br /&gt;
				 errdetail(&amp;quot;Plunit.assertation fails (assert_equals).&amp;quot;)));&lt;br /&gt;
&lt;br /&gt;
	if (!assert_equals_base(fcinfo))&lt;br /&gt;
		ereport(ERROR,&lt;br /&gt;
			(errcode(ERRCODE_CHECK_VIOLATION),&lt;br /&gt;
			 errmsg(message),&lt;br /&gt;
			 errdetail(&amp;quot;Plunit.assertation fails (assert_equals).&amp;quot;)));&lt;br /&gt;
&lt;br /&gt;
	PG_RETURN_VOID();&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/pre&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Přetížené funkce jsou registrovány příkazy:&lt;br /&gt;
&amp;lt;pre&amp;gt;&lt;br /&gt;
CREATE FUNCTION plunit.assert_equals(expected anyelement, actual anyelement)&lt;br /&gt;
RETURNS void&lt;br /&gt;
AS &amp;#039;MODULE_PATHNAME&amp;#039;,&amp;#039;plunit_assert_equals&amp;#039;&lt;br /&gt;
LANGUAGE C IMMUTABLE;&lt;br /&gt;
COMMENT ON FUNCTION plunit.assert_equals(expected anyelement, actual anyelement) IS &amp;#039;Asserts that expected and actual are equal&amp;#039;;&lt;br /&gt;
&lt;br /&gt;
CREATE FUNCTION plunit.assert_equals(expected anyelement, actual anyelement, message varchar)&lt;br /&gt;
RETURNS void&lt;br /&gt;
AS &amp;#039;MODULE_PATHNAME&amp;#039;,&amp;#039;plunit_assert_equals_message&amp;#039;&lt;br /&gt;
LANGUAGE C IMMUTABLE;&lt;br /&gt;
COMMENT ON FUNCTION plunit.assert_equals(expected anyelement, actual anyelement, message varchar) IS &amp;#039;Asserts that expected and actual are equal&amp;#039;;&lt;br /&gt;
&amp;lt;/pre&amp;gt;&lt;/div&gt;</summary>
		<author><name>imported&gt;Pavel</name></author>
	</entry>
</feed>