Commit 08361cea authored by Tom Lane's avatar Tom Lane

Fix null-pointer-deref crash while doing COPY IN with check constraints.

In commit bf7ca158 I introduced an
assumption that an RTE referenced by a whole-row Var must have a valid eref
field.  This is false for RTEs constructed by DoCopy, and there are other
places taking similar shortcuts.  Perhaps we should make all those places
go through addRangeTableEntryForRelation or its siblings instead of having
ad-hoc logic, but the most reliable fix seems to be to make the new code in
ExecEvalWholeRowVar cope if there's no eref.  We can reasonably assume that
there's no need to insert column aliases if no aliases were provided.

Add a regression test case covering this, and also verifying that a sane
column name is in fact available in this situation.

Although the known case only crashes in 9.4 and HEAD, it seems prudent to
back-patch the code change to 9.2, since all the ingredients for a similar
failure exist in the variant patch applied to 9.3 and 9.2.

Per report from Jean-Pierre Pelletier.
parent c5b31e29
...@@ -900,7 +900,9 @@ ExecEvalWholeRowVar(WholeRowVarExprState *wrvstate, ExprContext *econtext, ...@@ -900,7 +900,9 @@ ExecEvalWholeRowVar(WholeRowVarExprState *wrvstate, ExprContext *econtext,
* If we can't locate the RTE, assume the column names we've got are OK. * If we can't locate the RTE, assume the column names we've got are OK.
* (As of this writing, the only cases where we can't locate the RTE are * (As of this writing, the only cases where we can't locate the RTE are
* in execution of trigger WHEN clauses, and then the Var will have the * in execution of trigger WHEN clauses, and then the Var will have the
* trigger's relation's rowtype, so its names are fine.) * trigger's relation's rowtype, so its names are fine.) Also, if the
* creator of the RTE didn't bother to fill in an eref field, assume our
* column names are OK. (This happens in COPY, and perhaps other places.)
*/ */
if (econtext->ecxt_estate && if (econtext->ecxt_estate &&
variable->varno <= list_length(econtext->ecxt_estate->es_range_table)) variable->varno <= list_length(econtext->ecxt_estate->es_range_table))
...@@ -908,7 +910,8 @@ ExecEvalWholeRowVar(WholeRowVarExprState *wrvstate, ExprContext *econtext, ...@@ -908,7 +910,8 @@ ExecEvalWholeRowVar(WholeRowVarExprState *wrvstate, ExprContext *econtext,
RangeTblEntry *rte = rt_fetch(variable->varno, RangeTblEntry *rte = rt_fetch(variable->varno,
econtext->ecxt_estate->es_range_table); econtext->ecxt_estate->es_range_table);
ExecTypeSetColNames(output_tupdesc, rte->eref->colnames); if (rte->eref)
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 */
......
...@@ -429,6 +429,40 @@ COPY forcetest (d, e) FROM STDIN WITH (FORMAT csv, FORCE_NULL(b)); ...@@ -429,6 +429,40 @@ COPY forcetest (d, e) FROM STDIN WITH (FORMAT csv, FORCE_NULL(b));
ERROR: FORCE NULL column "b" not referenced by COPY ERROR: FORCE NULL column "b" not referenced by COPY
ROLLBACK; ROLLBACK;
\pset null '' \pset null ''
-- test case with whole-row Var in a check constraint
create table check_con_tbl (f1 int);
create function check_con_function(check_con_tbl) returns bool as $$
begin
raise notice 'input = %', row_to_json($1);
return $1.f1 > 0;
end $$ language plpgsql immutable;
alter table check_con_tbl add check (check_con_function(check_con_tbl.*));
\d+ check_con_tbl
Table "public.check_con_tbl"
Column | Type | Modifiers | Storage | Stats target | Description
--------+---------+-----------+---------+--------------+-------------
f1 | integer | | plain | |
Check constraints:
"check_con_tbl_check" CHECK (check_con_function(check_con_tbl.*))
copy check_con_tbl from stdin;
NOTICE: input = {"f1":1}
CONTEXT: COPY check_con_tbl, line 1: "1"
NOTICE: input = {"f1":null}
CONTEXT: COPY check_con_tbl, line 2: "\N"
copy check_con_tbl from stdin;
NOTICE: input = {"f1":0}
CONTEXT: COPY check_con_tbl, line 1: "0"
ERROR: new row for relation "check_con_tbl" violates check constraint "check_con_tbl_check"
DETAIL: Failing row contains (0).
CONTEXT: COPY check_con_tbl, line 1: "0"
select * from check_con_tbl;
f1
----
1
(2 rows)
DROP TABLE forcetest; DROP TABLE forcetest;
DROP TABLE vistest; DROP TABLE vistest;
DROP FUNCTION truncate_in_subxact(); DROP FUNCTION truncate_in_subxact();
......
...@@ -308,6 +308,25 @@ BEGIN; ...@@ -308,6 +308,25 @@ BEGIN;
COPY forcetest (d, e) FROM STDIN WITH (FORMAT csv, FORCE_NULL(b)); COPY forcetest (d, e) FROM STDIN WITH (FORMAT csv, FORCE_NULL(b));
ROLLBACK; ROLLBACK;
\pset null '' \pset null ''
-- test case with whole-row Var in a check constraint
create table check_con_tbl (f1 int);
create function check_con_function(check_con_tbl) returns bool as $$
begin
raise notice 'input = %', row_to_json($1);
return $1.f1 > 0;
end $$ language plpgsql immutable;
alter table check_con_tbl add check (check_con_function(check_con_tbl.*));
\d+ check_con_tbl
copy check_con_tbl from stdin;
1
\N
\.
copy check_con_tbl from stdin;
0
\.
select * from check_con_tbl;
DROP TABLE forcetest; DROP TABLE forcetest;
DROP TABLE vistest; DROP TABLE vistest;
DROP FUNCTION truncate_in_subxact(); DROP FUNCTION truncate_in_subxact();
......
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