I do not see the bug in this implementation:
CREATE FUNCTION foo(anyelement) RETURNS SETOF int AS $f$
SELECT id FROM unnest(array[1,2,3]) t(id)
WHERE CASE WHEN (pg_typeof($1)::text)='integer' THEN $1::int>2 ELSE true END
$f$ LANGUAGE SQL IMMUTABLE;
SELECT * FROM foo(123); -- OK!
SELECT * FROM foo('test'::text); -- BUG
Is this some kind of PostgreSQL bug or a non-documented restriction on anyelement
datatype?
Interesting: when isolated the CASE
clause works fine:
CREATE FUNCTION bar(anyelement) RETURNS boolean AS $f$
SELECT CASE WHEN (pg_typeof($1)::text)='integer' THEN $1::int>2;
$f$ LANGUAGE SQL IMMUTABLE;
SELECT bar('test'::text), bar(123), bar(1); -- works fine!
It's definitely related to SQL planner/optimizer. Since the function is declared as
IMMUTABLE
, the optimizer tries to pre-evaluate the query parts. For some reason, it evaluates the expression$1::int>2
even if you call the function withtext
parameter.If you change your
foo
function toVOLATILE
it will work fine, because the query optimizer will not try to optimize/pre-evalute it.But why
bar
function works fine even if it'sIMMUTABLE
? I guess the optimizer decides not to pre-evalute it as it does not use expressions in loops. What I mean is that$1::int>2
is evaluated only once, whereas infoo
function it's evaluated multiple times.Seems like there are some differences how SQL planner works for
SQL
andPLPGSQL
language. The same function inPLPGSQL
works fine.