Commit a620d500 authored by Tom Lane's avatar Tom Lane

Fix a bug introduced when set-returning SQL functions were made inline-able:

we have to cope with the possibility that the declared result rowtype contains
dropped columns.  This fails in 8.4, as per bug #5240.

While at it, be more paranoid about inserting binary coercions when inlining.
The pre-8.4 code did not really need to worry about that because it could not
inline at all in any case where an added coercion could change the behavior
of the function's statement.  However, when inlining a SRF we allow sorting,
grouping, and set-ops such as UNION.  In these cases, modifying one of the
targetlist entries that the sort/group/setop depends on could conceivably
change the behavior of the function's statement --- so don't inline when
such a case applies.
parent 84f910a7
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/catalog/pg_proc.c,v 1.168 2009/10/08 02:39:18 tgl Exp $ * $PostgreSQL: pgsql/src/backend/catalog/pg_proc.c,v 1.169 2009/12/14 02:15:49 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -810,7 +810,7 @@ fmgr_sql_validator(PG_FUNCTION_ARGS) ...@@ -810,7 +810,7 @@ fmgr_sql_validator(PG_FUNCTION_ARGS)
proc->pronargs); proc->pronargs);
(void) check_sql_fn_retval(funcoid, proc->prorettype, (void) check_sql_fn_retval(funcoid, proc->prorettype,
querytree_list, querytree_list,
false, NULL); NULL, NULL);
} }
else else
querytree_list = pg_parse_query(prosrc); querytree_list = pg_parse_query(prosrc);
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.255 2009/11/20 20:38:10 tgl Exp $ * $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.256 2009/12/14 02:15:49 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -47,6 +47,7 @@ ...@@ -47,6 +47,7 @@
#include "nodes/makefuncs.h" #include "nodes/makefuncs.h"
#include "nodes/nodeFuncs.h" #include "nodes/nodeFuncs.h"
#include "optimizer/planner.h" #include "optimizer/planner.h"
#include "parser/parse_coerce.h"
#include "pgstat.h" #include "pgstat.h"
#include "utils/acl.h" #include "utils/acl.h"
#include "utils/builtins.h" #include "utils/builtins.h"
...@@ -1382,7 +1383,7 @@ tupledesc_match(TupleDesc dst_tupdesc, TupleDesc src_tupdesc) ...@@ -1382,7 +1383,7 @@ tupledesc_match(TupleDesc dst_tupdesc, TupleDesc src_tupdesc)
Form_pg_attribute dattr = dst_tupdesc->attrs[i]; Form_pg_attribute dattr = dst_tupdesc->attrs[i];
Form_pg_attribute sattr = src_tupdesc->attrs[i]; Form_pg_attribute sattr = src_tupdesc->attrs[i];
if (dattr->atttypid == sattr->atttypid) if (IsBinaryCoercible(sattr->atttypid, dattr->atttypid))
continue; /* no worries */ continue; /* no worries */
if (!dattr->attisdropped) if (!dattr->attisdropped)
ereport(ERROR, ereport(ERROR,
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/functions.c,v 1.136 2009/11/04 22:26:05 tgl Exp $ * $PostgreSQL: pgsql/src/backend/executor/functions.c,v 1.137 2009/12/14 02:15:51 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -338,7 +338,7 @@ init_sql_fcache(FmgrInfo *finfo, bool lazyEvalOK) ...@@ -338,7 +338,7 @@ init_sql_fcache(FmgrInfo *finfo, bool lazyEvalOK)
fcache->returnsTuple = check_sql_fn_retval(foid, fcache->returnsTuple = check_sql_fn_retval(foid,
rettype, rettype,
queryTree_list, queryTree_list,
false, NULL,
&fcache->junkFilter); &fcache->junkFilter);
if (fcache->returnsTuple) if (fcache->returnsTuple)
...@@ -1003,14 +1003,27 @@ ShutdownSQLFunction(Datum arg) ...@@ -1003,14 +1003,27 @@ ShutdownSQLFunction(Datum arg)
* function definition of a polymorphic function.) * function definition of a polymorphic function.)
* *
* This function returns true if the sql function returns the entire tuple * This function returns true if the sql function returns the entire tuple
* result of its final statement, and false otherwise. Note that because we * result of its final statement, or false if it returns just the first column
* allow "SELECT rowtype_expression", this may be false even when the declared * result of that statement. It throws an error if the final statement doesn't
* function return type is a rowtype. * return the right type at all.
* *
* If insertRelabels is true, then binary-compatible cases are dealt with * Note that because we allow "SELECT rowtype_expression", the result can be
* by actually inserting RelabelType nodes into the output targetlist; * false even when the declared function return type is a rowtype.
* obviously the caller must pass a parsetree that it's okay to modify in this *
* case. * If modifyTargetList isn't NULL, the function will modify the final
* statement's targetlist in two cases:
* (1) if the tlist returns values that are binary-coercible to the expected
* type rather than being exactly the expected type. RelabelType nodes will
* be inserted to make the result types match exactly.
* (2) if there are dropped columns in the declared result rowtype. NULL
* output columns will be inserted in the tlist to match them.
* (Obviously the caller must pass a parsetree that is okay to modify when
* using this flag.) Note that this flag does not affect whether the tlist is
* considered to be a legal match to the result type, only how we react to
* allowed not-exact-match cases. *modifyTargetList will be set true iff
* we had to make any "dangerous" changes that could modify the semantics of
* the statement. If it is set true, the caller should not use the modified
* statement, but for simplicity we apply the changes anyway.
* *
* If junkFilter isn't NULL, then *junkFilter is set to a JunkFilter defined * If junkFilter isn't NULL, then *junkFilter is set to a JunkFilter defined
* to convert the function's tuple result to the correct output tuple type. * to convert the function's tuple result to the correct output tuple type.
...@@ -1019,10 +1032,11 @@ ShutdownSQLFunction(Datum arg) ...@@ -1019,10 +1032,11 @@ ShutdownSQLFunction(Datum arg)
*/ */
bool bool
check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
bool insertRelabels, bool *modifyTargetList,
JunkFilter **junkFilter) JunkFilter **junkFilter)
{ {
Query *parse; Query *parse;
List **tlist_ptr;
List *tlist; List *tlist;
int tlistlen; int tlistlen;
char fn_typtype; char fn_typtype;
...@@ -1031,6 +1045,8 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1031,6 +1045,8 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
AssertArg(!IsPolymorphicType(rettype)); AssertArg(!IsPolymorphicType(rettype));
if (modifyTargetList)
*modifyTargetList = false; /* initialize for no change */
if (junkFilter) if (junkFilter)
*junkFilter = NULL; /* initialize in case of VOID result */ *junkFilter = NULL; /* initialize in case of VOID result */
...@@ -1064,6 +1080,7 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1064,6 +1080,7 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
parse->utilityStmt == NULL && parse->utilityStmt == NULL &&
parse->intoClause == NULL) parse->intoClause == NULL)
{ {
tlist_ptr = &parse->targetList;
tlist = parse->targetList; tlist = parse->targetList;
} }
else if (parse && else if (parse &&
...@@ -1072,6 +1089,7 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1072,6 +1089,7 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
parse->commandType == CMD_DELETE) && parse->commandType == CMD_DELETE) &&
parse->returningList) parse->returningList)
{ {
tlist_ptr = &parse->returningList;
tlist = parse->returningList; tlist = parse->returningList;
} }
else else
...@@ -1132,11 +1150,16 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1132,11 +1150,16 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
format_type_be(rettype)), format_type_be(rettype)),
errdetail("Actual return type is %s.", errdetail("Actual return type is %s.",
format_type_be(restype)))); format_type_be(restype))));
if (insertRelabels && restype != rettype) if (modifyTargetList && restype != rettype)
{
tle->expr = (Expr *) makeRelabelType(tle->expr, tle->expr = (Expr *) makeRelabelType(tle->expr,
rettype, rettype,
-1, -1,
COERCE_DONTCARE); COERCE_DONTCARE);
/* Relabel is dangerous if TLE is a sort/group or setop column */
if (tle->ressortgroupref != 0 || parse->setOperations)
*modifyTargetList = true;
}
/* Set up junk filter if needed */ /* Set up junk filter if needed */
if (junkFilter) if (junkFilter)
...@@ -1149,6 +1172,8 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1149,6 +1172,8 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
int tupnatts; /* physical number of columns in tuple */ int tupnatts; /* physical number of columns in tuple */
int tuplogcols; /* # of nondeleted columns in tuple */ int tuplogcols; /* # of nondeleted columns in tuple */
int colindex; /* physical column index */ int colindex; /* physical column index */
List *newtlist; /* new non-junk tlist entries */
List *junkattrs; /* new junk tlist entries */
/* /*
* If the target list is of length 1, and the type of the varnode in * If the target list is of length 1, and the type of the varnode in
...@@ -1165,11 +1190,16 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1165,11 +1190,16 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
restype = exprType((Node *) tle->expr); restype = exprType((Node *) tle->expr);
if (IsBinaryCoercible(restype, rettype)) if (IsBinaryCoercible(restype, rettype))
{ {
if (insertRelabels && restype != rettype) if (modifyTargetList && restype != rettype)
{
tle->expr = (Expr *) makeRelabelType(tle->expr, tle->expr = (Expr *) makeRelabelType(tle->expr,
rettype, rettype,
-1, -1,
COERCE_DONTCARE); COERCE_DONTCARE);
/* Relabel is dangerous if sort/group or setop column */
if (tle->ressortgroupref != 0 || parse->setOperations)
*modifyTargetList = true;
}
/* Set up junk filter if needed */ /* Set up junk filter if needed */
if (junkFilter) if (junkFilter)
*junkFilter = ExecInitJunkFilter(tlist, false, NULL); *junkFilter = ExecInitJunkFilter(tlist, false, NULL);
...@@ -1193,11 +1223,14 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1193,11 +1223,14 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
/* /*
* Verify that the targetlist matches the return tuple type. We scan * Verify that the targetlist matches the return tuple type. We scan
* the non-deleted attributes to ensure that they match the datatypes * the non-deleted attributes to ensure that they match the datatypes
* of the non-resjunk columns. * of the non-resjunk columns. For deleted attributes, insert NULL
* result columns if the caller asked for that.
*/ */
tupnatts = tupdesc->natts; tupnatts = tupdesc->natts;
tuplogcols = 0; /* we'll count nondeleted cols as we go */ tuplogcols = 0; /* we'll count nondeleted cols as we go */
colindex = 0; colindex = 0;
newtlist = NIL; /* these are only used if modifyTargetList */
junkattrs = NIL;
foreach(lc, tlist) foreach(lc, tlist)
{ {
...@@ -1207,7 +1240,11 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1207,7 +1240,11 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
Oid atttype; Oid atttype;
if (tle->resjunk) if (tle->resjunk)
{
if (modifyTargetList)
junkattrs = lappend(junkattrs, tle);
continue; continue;
}
do do
{ {
...@@ -1219,6 +1256,26 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1219,6 +1256,26 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
format_type_be(rettype)), format_type_be(rettype)),
errdetail("Final statement returns too many columns."))); errdetail("Final statement returns too many columns.")));
attr = tupdesc->attrs[colindex - 1]; attr = tupdesc->attrs[colindex - 1];
if (attr->attisdropped && modifyTargetList)
{
Expr *null_expr;
/* The type of the null we insert isn't important */
null_expr = (Expr *) makeConst(INT4OID,
-1,
sizeof(int32),
(Datum) 0,
true, /* isnull */
true /* byval */ );
newtlist = lappend(newtlist,
makeTargetEntry(null_expr,
colindex,
NULL,
false));
/* NULL insertion is dangerous in a setop */
if (parse->setOperations)
*modifyTargetList = true;
}
} while (attr->attisdropped); } while (attr->attisdropped);
tuplogcols++; tuplogcols++;
...@@ -1233,28 +1290,66 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList, ...@@ -1233,28 +1290,66 @@ check_sql_fn_retval(Oid func_id, Oid rettype, List *queryTreeList,
format_type_be(tletype), format_type_be(tletype),
format_type_be(atttype), format_type_be(atttype),
tuplogcols))); tuplogcols)));
if (insertRelabels && tletype != atttype) if (modifyTargetList)
tle->expr = (Expr *) makeRelabelType(tle->expr, {
atttype, if (tletype != atttype)
-1, {
COERCE_DONTCARE); tle->expr = (Expr *) makeRelabelType(tle->expr,
atttype,
-1,
COERCE_DONTCARE);
/* Relabel is dangerous if sort/group or setop column */
if (tle->ressortgroupref != 0 || parse->setOperations)
*modifyTargetList = true;
}
tle->resno = colindex;
newtlist = lappend(newtlist, tle);
}
} }
for (;;) /* remaining columns in tupdesc had better all be dropped */
for (colindex++; colindex <= tupnatts; colindex++)
{ {
colindex++;
if (colindex > tupnatts)
break;
if (!tupdesc->attrs[colindex - 1]->attisdropped) if (!tupdesc->attrs[colindex - 1]->attisdropped)
tuplogcols++; ereport(ERROR,
(errcode(ERRCODE_INVALID_FUNCTION_DEFINITION),
errmsg("return type mismatch in function declared to return %s",
format_type_be(rettype)),
errdetail("Final statement returns too few columns.")));
if (modifyTargetList)
{
Expr *null_expr;
/* The type of the null we insert isn't important */
null_expr = (Expr *) makeConst(INT4OID,
-1,
sizeof(int32),
(Datum) 0,
true, /* isnull */
true /* byval */ );
newtlist = lappend(newtlist,
makeTargetEntry(null_expr,
colindex,
NULL,
false));
/* NULL insertion is dangerous in a setop */
if (parse->setOperations)
*modifyTargetList = true;
}
} }
if (tlistlen != tuplogcols) if (modifyTargetList)
ereport(ERROR, {
(errcode(ERRCODE_INVALID_FUNCTION_DEFINITION), /* ensure resjunk columns are numbered correctly */
errmsg("return type mismatch in function declared to return %s", foreach(lc, junkattrs)
format_type_be(rettype)), {
errdetail("Final statement returns too few columns."))); TargetEntry *tle = (TargetEntry *) lfirst(lc);
tle->resno = colindex++;
}
/* replace the tlist with the modified one */
*tlist_ptr = list_concat(newtlist, junkattrs);
}
/* Set up junk filter if needed */ /* Set up junk filter if needed */
if (junkFilter) if (junkFilter)
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/util/clauses.c,v 1.279 2009/10/08 02:39:21 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/util/clauses.c,v 1.280 2009/12/14 02:15:52 tgl Exp $
* *
* HISTORY * HISTORY
* AUTHOR DATE MAJOR EVENT * AUTHOR DATE MAJOR EVENT
...@@ -3673,6 +3673,7 @@ inline_function(Oid funcid, Oid result_type, List *args, ...@@ -3673,6 +3673,7 @@ inline_function(Oid funcid, Oid result_type, List *args,
char *src; char *src;
Datum tmp; Datum tmp;
bool isNull; bool isNull;
bool modifyTargetList;
MemoryContext oldcxt; MemoryContext oldcxt;
MemoryContext mycxt; MemoryContext mycxt;
ErrorContextCallback sqlerrcontext; ErrorContextCallback sqlerrcontext;
...@@ -3792,7 +3793,7 @@ inline_function(Oid funcid, Oid result_type, List *args, ...@@ -3792,7 +3793,7 @@ inline_function(Oid funcid, Oid result_type, List *args,
* needed; that's probably not important, but let's be careful. * needed; that's probably not important, but let's be careful.
*/ */
if (check_sql_fn_retval(funcid, result_type, list_make1(querytree), if (check_sql_fn_retval(funcid, result_type, list_make1(querytree),
true, NULL)) &modifyTargetList, NULL))
goto fail; /* reject whole-tuple-result cases */ goto fail; /* reject whole-tuple-result cases */
/* Now we can grab the tlist expression */ /* Now we can grab the tlist expression */
...@@ -3800,6 +3801,8 @@ inline_function(Oid funcid, Oid result_type, List *args, ...@@ -3800,6 +3801,8 @@ inline_function(Oid funcid, Oid result_type, List *args,
/* Assert that check_sql_fn_retval did the right thing */ /* Assert that check_sql_fn_retval did the right thing */
Assert(exprType(newexpr) == result_type); Assert(exprType(newexpr) == result_type);
/* It couldn't have made any dangerous tlist changes, either */
Assert(!modifyTargetList);
/* /*
* Additional validity checks on the expression. It mustn't return a set, * Additional validity checks on the expression. It mustn't return a set,
...@@ -4088,6 +4091,7 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte) ...@@ -4088,6 +4091,7 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte)
char *src; char *src;
Datum tmp; Datum tmp;
bool isNull; bool isNull;
bool modifyTargetList;
MemoryContext oldcxt; MemoryContext oldcxt;
MemoryContext mycxt; MemoryContext mycxt;
ErrorContextCallback sqlerrcontext; ErrorContextCallback sqlerrcontext;
...@@ -4253,8 +4257,8 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte) ...@@ -4253,8 +4257,8 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte)
* Make sure the function (still) returns what it's declared to. This * Make sure the function (still) returns what it's declared to. This
* will raise an error if wrong, but that's okay since the function would * will raise an error if wrong, but that's okay since the function would
* fail at runtime anyway. Note that check_sql_fn_retval will also insert * fail at runtime anyway. Note that check_sql_fn_retval will also insert
* RelabelType(s) if needed to make the tlist expression(s) match the * RelabelType(s) and/or NULL columns if needed to make the tlist
* declared type of the function. * expression(s) match the declared type of the function.
* *
* If the function returns a composite type, don't inline unless the check * If the function returns a composite type, don't inline unless the check
* shows it's returning a whole tuple result; otherwise what it's * shows it's returning a whole tuple result; otherwise what it's
...@@ -4262,11 +4266,19 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte) ...@@ -4262,11 +4266,19 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte)
*/ */
if (!check_sql_fn_retval(func_oid, fexpr->funcresulttype, if (!check_sql_fn_retval(func_oid, fexpr->funcresulttype,
querytree_list, querytree_list,
true, NULL) && &modifyTargetList, NULL) &&
(get_typtype(fexpr->funcresulttype) == TYPTYPE_COMPOSITE || (get_typtype(fexpr->funcresulttype) == TYPTYPE_COMPOSITE ||
fexpr->funcresulttype == RECORDOID)) fexpr->funcresulttype == RECORDOID))
goto fail; /* reject not-whole-tuple-result cases */ goto fail; /* reject not-whole-tuple-result cases */
/*
* If we had to modify the tlist to make it match, and the statement is
* one in which changing the tlist contents could change semantics, we
* have to punt and not inline.
*/
if (modifyTargetList)
goto fail;
/* /*
* If it returns RECORD, we have to check against the column type list * If it returns RECORD, we have to check against the column type list
* provided in the RTE; check_sql_fn_retval can't do that. (If no match, * provided in the RTE; check_sql_fn_retval can't do that. (If no match,
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2009, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2009, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/functions.h,v 1.33 2009/01/01 17:23:59 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/functions.h,v 1.34 2009/12/14 02:15:54 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -22,7 +22,7 @@ extern Datum fmgr_sql(PG_FUNCTION_ARGS); ...@@ -22,7 +22,7 @@ extern Datum fmgr_sql(PG_FUNCTION_ARGS);
extern bool check_sql_fn_retval(Oid func_id, Oid rettype, extern bool check_sql_fn_retval(Oid func_id, Oid rettype,
List *queryTreeList, List *queryTreeList,
bool insertRelabels, bool *modifyTargetList,
JunkFilter **junkFilter); JunkFilter **junkFilter);
extern DestReceiver *CreateSQLFunctionDestReceiver(void); extern DestReceiver *CreateSQLFunctionDestReceiver(void);
......
...@@ -836,3 +836,64 @@ ERROR: a column definition list is required for functions returning "record" ...@@ -836,3 +836,64 @@ ERROR: a column definition list is required for functions returning "record"
LINE 1: select * from testfoo(); LINE 1: select * from testfoo();
^ ^
drop function testfoo(); drop function testfoo();
--
-- Check some cases involving dropped columns in a rowtype result
--
create temp table users (userid text, email text, todrop bool, enabled bool);
insert into users values ('id','email',true,true);
insert into users values ('id2','email2',true,true);
alter table users drop column todrop;
create or replace function get_first_user() returns users as
$$ SELECT * FROM users ORDER BY userid LIMIT 1; $$
language sql stable;
SELECT get_first_user();
get_first_user
----------------
(id,email,t)
(1 row)
SELECT * FROM get_first_user();
userid | email | enabled
--------+-------+---------
id | email | t
(1 row)
create or replace function get_users() returns setof users as
$$ SELECT * FROM users ORDER BY userid; $$
language sql stable;
SELECT get_users();
get_users
----------------
(id,email,t)
(id2,email2,t)
(2 rows)
SELECT * FROM get_users();
userid | email | enabled
--------+--------+---------
id | email | t
id2 | email2 | t
(2 rows)
drop function get_first_user();
drop function get_users();
drop table users;
-- this won't get inlined because of type coercion, but it shouldn't fail
create or replace function foobar() returns setof text as
$$ select 'foo'::varchar union all select 'bar'::varchar ; $$
language sql stable;
select foobar();
foobar
--------
foo
bar
(2 rows)
select * from foobar();
foobar
--------
foo
bar
(2 rows)
drop function foobar();
...@@ -391,3 +391,41 @@ select * from testfoo() as t(f1 int8,f2 int8); ...@@ -391,3 +391,41 @@ select * from testfoo() as t(f1 int8,f2 int8);
select * from testfoo(); -- fail select * from testfoo(); -- fail
drop function testfoo(); drop function testfoo();
--
-- Check some cases involving dropped columns in a rowtype result
--
create temp table users (userid text, email text, todrop bool, enabled bool);
insert into users values ('id','email',true,true);
insert into users values ('id2','email2',true,true);
alter table users drop column todrop;
create or replace function get_first_user() returns users as
$$ SELECT * FROM users ORDER BY userid LIMIT 1; $$
language sql stable;
SELECT get_first_user();
SELECT * FROM get_first_user();
create or replace function get_users() returns setof users as
$$ SELECT * FROM users ORDER BY userid; $$
language sql stable;
SELECT get_users();
SELECT * FROM get_users();
drop function get_first_user();
drop function get_users();
drop table users;
-- this won't get inlined because of type coercion, but it shouldn't fail
create or replace function foobar() returns setof text as
$$ select 'foo'::varchar union all select 'bar'::varchar ; $$
language sql stable;
select foobar();
select * from foobar();
drop function foobar();
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment