Commit 1d072bd2 authored by Tom Lane's avatar Tom Lane

Revert applying column aliases to the output of whole-row Vars.

In commit bf7ca158, I had the bright idea that we could make the
result of a whole-row Var (that is, foo.*) track any column aliases
that had been applied to the FROM entry the Var refers to.  However,
that's not terribly logically consistent, because now the output of
the Var is no longer of the named composite type that the Var claims
to emit.  bf7ca158 tried to handle that by changing the output
tuple values to be labeled with a blessed RECORD type, but that's
really pretty disastrous: we can wind up storing such tuples onto
disk, whereupon they're not readable by other sessions.

The only practical fix I can see is to give up on what bf7ca158
tried to do, and say that the column names of tuples produced by
a whole-row Var are always those of the underlying named composite
type, query aliases or no.  While this introduces some inconsistencies,
it removes others, so it's not that awful in the abstract.  What *is*
kind of awful is to make such a behavioral change in a back-patched
bug fix.  But corrupt data is worse, so back-patched it will be.

(A workaround available to anyone who's unhappy about this is to
introduce an extra level of sub-SELECT, so that the whole-row Var is
referring to the sub-SELECT's output and not to a named table type.
Then the Var is of type RECORD to begin with and there's no issue.)

Per report from Miles Delahunty.  The faulty commit dates to 9.5,
so back-patch to all supported branches.

Discussion: https://postgr.es/m/2950001.1638729947@sss.pgh.pa.us
parent 677a1dc0
...@@ -1897,16 +1897,16 @@ ExecInitExprRec(Expr *node, ExprState *state, ...@@ -1897,16 +1897,16 @@ ExecInitExprRec(Expr *node, ExprState *state,
{ {
/* generic record, use types of given expressions */ /* generic record, use types of given expressions */
tupdesc = ExecTypeFromExprList(rowexpr->args); tupdesc = ExecTypeFromExprList(rowexpr->args);
/* ... but adopt RowExpr's column aliases */
ExecTypeSetColNames(tupdesc, rowexpr->colnames);
/* Bless the tupdesc so it can be looked up later */
BlessTupleDesc(tupdesc);
} }
else else
{ {
/* it's been cast to a named type, use that */ /* it's been cast to a named type, use that */
tupdesc = lookup_rowtype_tupdesc_copy(rowexpr->row_typeid, -1); tupdesc = lookup_rowtype_tupdesc_copy(rowexpr->row_typeid, -1);
} }
/* In either case, adopt RowExpr's column aliases */
ExecTypeSetColNames(tupdesc, rowexpr->colnames);
/* Bless the tupdesc in case it's now of type RECORD */
BlessTupleDesc(tupdesc);
/* /*
* In the named-type case, the tupdesc could have more columns * In the named-type case, the tupdesc could have more columns
......
...@@ -4008,12 +4008,8 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext) ...@@ -4008,12 +4008,8 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
* generates an INT4 NULL regardless of the dropped column type). * generates an INT4 NULL regardless of the dropped column type).
* If we find a dropped column and cannot verify that case (1) * If we find a dropped column and cannot verify that case (1)
* holds, we have to use the slow path to check (2) for each row. * holds, we have to use the slow path to check (2) for each row.
*
* If vartype is a domain over composite, just look through that
* to the base composite type.
*/ */
var_tupdesc = lookup_rowtype_tupdesc_domain(variable->vartype, var_tupdesc = lookup_rowtype_tupdesc(variable->vartype, -1);
-1, false);
slot_tupdesc = slot->tts_tupleDescriptor; slot_tupdesc = slot->tts_tupleDescriptor;
...@@ -4050,9 +4046,8 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext) ...@@ -4050,9 +4046,8 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
/* /*
* Use the variable's declared rowtype as the descriptor for the * Use the variable's declared rowtype as the descriptor for the
* output values, modulo possibly assigning new column names * output values. In particular, we *must* absorb any
* below. In particular, we *must* absorb any attisdropped * attisdropped markings.
* markings.
*/ */
oldcontext = MemoryContextSwitchTo(econtext->ecxt_per_query_memory); oldcontext = MemoryContextSwitchTo(econtext->ecxt_per_query_memory);
output_tupdesc = CreateTupleDescCopy(var_tupdesc); output_tupdesc = CreateTupleDescCopy(var_tupdesc);
...@@ -4070,30 +4065,28 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext) ...@@ -4070,30 +4065,28 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
oldcontext = MemoryContextSwitchTo(econtext->ecxt_per_query_memory); oldcontext = MemoryContextSwitchTo(econtext->ecxt_per_query_memory);
output_tupdesc = CreateTupleDescCopy(slot->tts_tupleDescriptor); output_tupdesc = CreateTupleDescCopy(slot->tts_tupleDescriptor);
MemoryContextSwitchTo(oldcontext); MemoryContextSwitchTo(oldcontext);
}
/* /*
* Construct a tuple descriptor for the composite values we'll * It's possible that the input slot is a relation scan slot and
* produce, and make sure its record type is "blessed". The main * so is marked with that relation's rowtype. But we're supposed
* reason to do this is to be sure that operations such as * to be returning RECORD, so reset to that.
* row_to_json() will see the desired column names when they look up */
* the descriptor from the type information embedded in the composite output_tupdesc->tdtypeid = RECORDOID;
* values. output_tupdesc->tdtypmod = -1;
*
* We already got the correct physical datatype info above, but now we /*
* should try to find the source RTE and adopt its column aliases, in * We already got the correct physical datatype info above, but
* case they are different from the original rowtype's names. For * now we should try to find the source RTE and adopt its column
* example, in "SELECT foo(t) FROM tab t(x,y)", the first two columns * aliases, since it's unlikely that the input slot has the
* in the composite output should be named "x" and "y" regardless of * desired names.
* tab's column names.
* *
* If we can't locate the RTE, assume the column names we've got are * If we can't locate the RTE, assume the column names we've got
* OK. (As of this writing, the only cases where we can't locate the * are OK. (As of this writing, the only cases where we can't
* RTE are in execution of trigger WHEN clauses, and then the Var will * locate the RTE are in execution of trigger WHEN clauses, and
* have the trigger's relation's rowtype, so its names are fine.) * then the Var will have the trigger's relation's rowtype, so its
* Also, if the creator of the RTE didn't bother to fill in an eref * names are fine.) Also, if the creator of the RTE didn't bother
* field, assume our column names are OK. (This happens in COPY, and * to fill in an eref field, assume our column names are OK. (This
* perhaps other places.) * happens in COPY, and perhaps other places.)
*/ */
if (econtext->ecxt_estate && if (econtext->ecxt_estate &&
variable->varno <= econtext->ecxt_estate->es_range_table_size) variable->varno <= econtext->ecxt_estate->es_range_table_size)
...@@ -4104,6 +4097,7 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext) ...@@ -4104,6 +4097,7 @@ ExecEvalWholeRowVar(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
if (rte->eref) if (rte->eref)
ExecTypeSetColNames(output_tupdesc, rte->eref->colnames); ExecTypeSetColNames(output_tupdesc, rte->eref->colnames);
} }
}
/* Bless the tupdesc if needed, and save it in the execution state */ /* Bless the tupdesc if needed, and save it in the execution state */
op->d.wholerow.tupdesc = BlessTupleDesc(output_tupdesc); op->d.wholerow.tupdesc = BlessTupleDesc(output_tupdesc);
......
...@@ -2022,51 +2022,40 @@ ExecTypeFromExprList(List *exprList) ...@@ -2022,51 +2022,40 @@ ExecTypeFromExprList(List *exprList)
} }
/* /*
* ExecTypeSetColNames - set column names in a TupleDesc * ExecTypeSetColNames - set column names in a RECORD TupleDesc
* *
* Column names must be provided as an alias list (list of String nodes). * Column names must be provided as an alias list (list of String nodes).
*
* For some callers, the supplied tupdesc has a named rowtype (not RECORD)
* and it is moderately likely that the alias list matches the column names
* already present in the tupdesc. If we do change any column names then
* we must reset the tupdesc's type to anonymous RECORD; but we avoid doing
* so if no names change.
*/ */
void void
ExecTypeSetColNames(TupleDesc typeInfo, List *namesList) ExecTypeSetColNames(TupleDesc typeInfo, List *namesList)
{ {
bool modified = false;
int colno = 0; int colno = 0;
ListCell *lc; ListCell *lc;
/* It's only OK to change col names in a not-yet-blessed RECORD type */
Assert(typeInfo->tdtypeid == RECORDOID);
Assert(typeInfo->tdtypmod < 0);
foreach(lc, namesList) foreach(lc, namesList)
{ {
char *cname = strVal(lfirst(lc)); char *cname = strVal(lfirst(lc));
Form_pg_attribute attr; Form_pg_attribute attr;
/* Guard against too-long names list */ /* Guard against too-long names list (probably can't happen) */
if (colno >= typeInfo->natts) if (colno >= typeInfo->natts)
break; break;
attr = TupleDescAttr(typeInfo, colno); attr = TupleDescAttr(typeInfo, colno);
colno++; colno++;
/* Ignore empty aliases (these must be for dropped columns) */ /*
if (cname[0] == '\0') * Do nothing for empty aliases or dropped columns (these cases
* probably can't arise in RECORD types, either)
*/
if (cname[0] == '\0' || attr->attisdropped)
continue; continue;
/* Change tupdesc only if alias is actually different */ /* OK, assign the column name */
if (strcmp(cname, NameStr(attr->attname)) != 0)
{
namestrcpy(&(attr->attname), cname); namestrcpy(&(attr->attname), cname);
modified = true;
}
}
/* If we modified the tupdesc, it's now a new record type */
if (modified)
{
typeInfo->tdtypeid = RECORDOID;
typeInfo->tdtypmod = -1;
} }
} }
......
...@@ -1010,19 +1010,9 @@ select row_to_json(i) from int8_tbl i; ...@@ -1010,19 +1010,9 @@ select row_to_json(i) from int8_tbl i;
{"q1":4567890123456789,"q2":-4567890123456789} {"q1":4567890123456789,"q2":-4567890123456789}
(5 rows) (5 rows)
-- since "i" is of type "int8_tbl", attaching aliases doesn't change anything:
select row_to_json(i) from int8_tbl i(x,y); select row_to_json(i) from int8_tbl i(x,y);
row_to_json row_to_json
----------------------------------------------
{"x":123,"y":456}
{"x":123,"y":4567890123456789}
{"x":4567890123456789,"y":123}
{"x":4567890123456789,"y":4567890123456789}
{"x":4567890123456789,"y":-4567890123456789}
(5 rows)
create temp view vv1 as select * from int8_tbl;
select row_to_json(i) from vv1 i;
row_to_json
------------------------------------------------ ------------------------------------------------
{"q1":123,"q2":456} {"q1":123,"q2":456}
{"q1":123,"q2":4567890123456789} {"q1":123,"q2":4567890123456789}
...@@ -1031,16 +1021,7 @@ select row_to_json(i) from vv1 i; ...@@ -1031,16 +1021,7 @@ select row_to_json(i) from vv1 i;
{"q1":4567890123456789,"q2":-4567890123456789} {"q1":4567890123456789,"q2":-4567890123456789}
(5 rows) (5 rows)
select row_to_json(i) from vv1 i(x,y); -- in these examples, we'll report the exposed column names of the subselect:
row_to_json
----------------------------------------------
{"x":123,"y":456}
{"x":123,"y":4567890123456789}
{"x":4567890123456789,"y":123}
{"x":4567890123456789,"y":4567890123456789}
{"x":4567890123456789,"y":-4567890123456789}
(5 rows)
select row_to_json(ss) from select row_to_json(ss) from
(select q1, q2 from int8_tbl) as ss; (select q1, q2 from int8_tbl) as ss;
row_to_json row_to_json
......
...@@ -417,12 +417,10 @@ select longname(f) from fullname f; ...@@ -417,12 +417,10 @@ select longname(f) from fullname f;
-- --
select row_to_json(i) from int8_tbl i; select row_to_json(i) from int8_tbl i;
-- since "i" is of type "int8_tbl", attaching aliases doesn't change anything:
select row_to_json(i) from int8_tbl i(x,y); select row_to_json(i) from int8_tbl i(x,y);
create temp view vv1 as select * from int8_tbl; -- in these examples, we'll report the exposed column names of the subselect:
select row_to_json(i) from vv1 i;
select row_to_json(i) from vv1 i(x,y);
select row_to_json(ss) from select row_to_json(ss) from
(select q1, q2 from int8_tbl) as ss; (select q1, q2 from int8_tbl) as ss;
select row_to_json(ss) from select row_to_json(ss) from
......
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