1
0
mirror of https://github.com/postgres/postgres.git synced 2025-10-19 15:49:24 +03:00

Adjust ExecMakeTableFunctionResult to produce a single all-nulls row

when a function that returns a single tuple (not a setof tuple) returns
NULL.  This seems to be the most consistent behavior.  It would have
taken a bit less code to make it return an empty table (zero rows) but
ISTM a non-SETOF function ought always return exactly one row.  Per
bug report from Ivan-Sun1.
This commit is contained in:
Tom Lane
2004-09-22 17:41:51 +00:00
parent b84788debc
commit bebaf70613
2 changed files with 78 additions and 50 deletions

View File

@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.168 2004/08/29 05:06:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.169 2004/09/22 17:41:50 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
@@ -1115,7 +1115,8 @@ ExecMakeFunctionResultNoSets(FuncExprState *fcache,
* ExecMakeTableFunctionResult * ExecMakeTableFunctionResult
* *
* Evaluate a table function, producing a materialized result in a Tuplestore * Evaluate a table function, producing a materialized result in a Tuplestore
* object. (If function returns an empty set, we just return NULL instead.) * object. *returnDesc is set to the tupledesc actually returned by the
* function, or NULL if it didn't provide one.
*/ */
Tuplestorestate * Tuplestorestate *
ExecMakeTableFunctionResult(ExprState *funcexpr, ExecMakeTableFunctionResult(ExprState *funcexpr,
@@ -1127,6 +1128,7 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
TupleDesc tupdesc = NULL; TupleDesc tupdesc = NULL;
Oid funcrettype; Oid funcrettype;
bool returnsTuple; bool returnsTuple;
bool returnsSet = false;
FunctionCallInfoData fcinfo; FunctionCallInfoData fcinfo;
ReturnSetInfo rsinfo; ReturnSetInfo rsinfo;
HeapTupleData tmptup; HeapTupleData tmptup;
@@ -1135,6 +1137,31 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
bool direct_function_call; bool direct_function_call;
bool first_time = true; bool first_time = true;
callerContext = CurrentMemoryContext;
funcrettype = exprType((Node *) funcexpr->expr);
returnsTuple = (funcrettype == RECORDOID ||
get_typtype(funcrettype) == 'c');
/*
* Prepare a resultinfo node for communication. We always do this
* even if not expecting a set result, so that we can pass
* expectedDesc. In the generic-expression case, the expression
* doesn't actually get to see the resultinfo, but set it up anyway
* because we use some of the fields as our own state variables.
*/
MemSet(&fcinfo, 0, sizeof(fcinfo));
fcinfo.resultinfo = (Node *) &rsinfo;
rsinfo.type = T_ReturnSetInfo;
rsinfo.econtext = econtext;
rsinfo.expectedDesc = expectedDesc;
rsinfo.allowedModes = (int) (SFRM_ValuePerCall | SFRM_Materialize);
rsinfo.returnMode = SFRM_ValuePerCall;
/* isDone is filled below */
rsinfo.setResult = NULL;
rsinfo.setDesc = NULL;
/* /*
* Normally the passed expression tree will be a FuncExprState, since * Normally the passed expression tree will be a FuncExprState, since
* the grammar only allows a function call at the top level of a table * the grammar only allows a function call at the top level of a table
@@ -1165,6 +1192,7 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
init_fcache(func->funcid, fcache, econtext->ecxt_per_query_memory); init_fcache(func->funcid, fcache, econtext->ecxt_per_query_memory);
} }
returnsSet = fcache->func.fn_retset;
/* /*
* Evaluate the function's argument list. * Evaluate the function's argument list.
@@ -1174,7 +1202,6 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
* the inner loop. So do it in caller context. Perhaps we should * the inner loop. So do it in caller context. Perhaps we should
* make a separate context just to hold the evaluated arguments? * make a separate context just to hold the evaluated arguments?
*/ */
MemSet(&fcinfo, 0, sizeof(fcinfo));
fcinfo.flinfo = &(fcache->func); fcinfo.flinfo = &(fcache->func);
argDone = ExecEvalFuncArgs(&fcinfo, fcache->args, econtext); argDone = ExecEvalFuncArgs(&fcinfo, fcache->args, econtext);
/* We don't allow sets in the arguments of the table function */ /* We don't allow sets in the arguments of the table function */
@@ -1185,7 +1212,8 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
/* /*
* If function is strict, and there are any NULL arguments, skip * If function is strict, and there are any NULL arguments, skip
* calling the function and return NULL (actually an empty set). * calling the function and act like it returned NULL (or an empty
* set, in the returns-set case).
*/ */
if (fcache->func.fn_strict) if (fcache->func.fn_strict)
{ {
@@ -1194,10 +1222,7 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
for (i = 0; i < fcinfo.nargs; i++) for (i = 0; i < fcinfo.nargs; i++)
{ {
if (fcinfo.argnull[i]) if (fcinfo.argnull[i])
{ goto no_function_result;
*returnDesc = NULL;
return NULL;
}
} }
} }
} }
@@ -1207,33 +1232,11 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
direct_function_call = false; direct_function_call = false;
} }
funcrettype = exprType((Node *) funcexpr->expr);
returnsTuple = (funcrettype == RECORDOID ||
get_typtype(funcrettype) == 'c');
/*
* Prepare a resultinfo node for communication. We always do this
* even if not expecting a set result, so that we can pass
* expectedDesc. In the generic-expression case, the expression
* doesn't actually get to see the resultinfo, but set it up anyway
* because we use some of the fields as our own state variables.
*/
fcinfo.resultinfo = (Node *) &rsinfo;
rsinfo.type = T_ReturnSetInfo;
rsinfo.econtext = econtext;
rsinfo.expectedDesc = expectedDesc;
rsinfo.allowedModes = (int) (SFRM_ValuePerCall | SFRM_Materialize);
rsinfo.returnMode = SFRM_ValuePerCall;
/* isDone is filled below */
rsinfo.setResult = NULL;
rsinfo.setDesc = NULL;
/* /*
* Switch to short-lived context for calling the function or * Switch to short-lived context for calling the function or
* expression. * expression.
*/ */
callerContext = MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory); MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
/* /*
* Loop to handle the ValuePerCall protocol (which is also the same * Loop to handle the ValuePerCall protocol (which is also the same
@@ -1269,23 +1272,26 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
{ {
/* /*
* Check for end of result set. * Check for end of result set.
*
* Note: if function returns an empty set, we don't build a
* tupdesc or tuplestore (since we can't get a tupdesc in the
* function-returning-tuple case)
*/ */
if (rsinfo.isDone == ExprEndResult) if (rsinfo.isDone == ExprEndResult)
break; break;
/* /*
* Can't do anything useful with NULL rowtype values. * Can't do anything very useful with NULL rowtype values.
* Currently we raise an error, but another alternative is to * For a function returning set, we consider this a protocol
* just ignore the result and "continue" to get another row. * violation (but another alternative would be to just ignore
* the result and "continue" to get another row). For a function
* not returning set, we fall out of the loop; we'll cons up
* an all-nulls result row below.
*/ */
if (returnsTuple && fcinfo.isnull) if (returnsTuple && fcinfo.isnull)
{
if (!returnsSet)
break;
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED), (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
errmsg("function returning row cannot return null value"))); errmsg("function returning set of rows cannot return null value")));
}
/* /*
* If first time through, build tupdesc and tuplestore for * If first time through, build tupdesc and tuplestore for
@@ -1381,6 +1387,35 @@ ExecMakeTableFunctionResult(ExprState *funcexpr,
first_time = false; first_time = false;
} }
no_function_result:
/*
* If we got nothing from the function (ie, an empty-set or NULL result),
* we have to create the tuplestore to return, and if it's a
* non-set-returning function then insert a single all-nulls row.
*/
if (rsinfo.setResult == NULL)
{
MemoryContextSwitchTo(econtext->ecxt_per_query_memory);
tupstore = tuplestore_begin_heap(true, false, work_mem);
rsinfo.setResult = tupstore;
if (!returnsSet)
{
int natts = expectedDesc->natts;
Datum *nulldatums;
char *nullflags;
HeapTuple tuple;
MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
nulldatums = (Datum *) palloc0(natts * sizeof(Datum));
nullflags = (char *) palloc(natts * sizeof(char));
memset(nullflags, 'n', natts * sizeof(char));
tuple = heap_formtuple(expectedDesc, nulldatums, nullflags);
MemoryContextSwitchTo(econtext->ecxt_per_query_memory);
tuplestore_puttuple(tupstore, tuple);
}
}
MemoryContextSwitchTo(callerContext); MemoryContextSwitchTo(callerContext);
/* The returned pointers are those in rsinfo */ /* The returned pointers are those in rsinfo */

View File

@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeFunctionscan.c,v 1.26 2004/08/29 04:12:31 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeFunctionscan.c,v 1.27 2004/09/22 17:41:51 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
@@ -96,17 +96,10 @@ FunctionNext(FunctionScanState *node)
/* /*
* Get the next tuple from tuplestore. Return NULL if no more tuples. * Get the next tuple from tuplestore. Return NULL if no more tuples.
*/ */
slot = node->ss.ss_ScanTupleSlot;
if (tuplestorestate)
heapTuple = tuplestore_getheaptuple(tuplestorestate, heapTuple = tuplestore_getheaptuple(tuplestorestate,
ScanDirectionIsForward(direction), ScanDirectionIsForward(direction),
&should_free); &should_free);
else slot = node->ss.ss_ScanTupleSlot;
{
heapTuple = NULL;
should_free = false;
}
return ExecStoreTuple(heapTuple, slot, InvalidBuffer, should_free); return ExecStoreTuple(heapTuple, slot, InvalidBuffer, should_free);
} }