Commit 9e5f1f21 authored by Tom Lane's avatar Tom Lane

Rethink recent fix for pg_dump's handling of extension config tables.

Commit 3eb3d3e7 was a few bricks shy of a load: while it correctly
set the table's "interesting" flag when deciding to dump the data of
an extension config table, it was not correct to clear that flag
if we concluded we shouldn't dump the data.  This led to the crash
reported in bug #16655, because in fact we'll traverse dumpTableSchema
anyway for all extension tables (to see if they have user-added
seclabels or RLS policies).

The right thing to do is to force "interesting" true in makeTableDataInfo,
and otherwise leave the flag alone.  (Doing it there is more future-proof
in case additional calls are added, and it also avoids setting the flag
unnecessarily if that function decides the table is non-dumpable.)

This investigation also showed that while only the --inserts code path
had an obvious failure in the case considered by 3eb3d3e7, the COPY
code path also has a problem with not having loaded table subsidiary
data.  That causes fmtCopyColumnList to silently return an empty string
instead of the correct column list.  That accidentally mostly works,
which perhaps is why we didn't notice this before.  It would only fail
if the restore column order is different from the dump column order,
which only happens in weird inheritance cases, so it's not surprising
nobody had hit the case with an extension config table.  Nonetheless,
it's a bug, and it goes a long way back, not just to v12 where the
--inserts code path started to have a problem with this.

In hopes of catching such cases a bit sooner in future, add some
Asserts that "interesting" has been set in both dumpTableData and
dumpTableSchema.  Adjust the test case added by 3eb3d3e7 so that it
checks the COPY rather than INSERT form of that bug, allowing it to
detect the longer-standing symptom.

Per bug #16655 from Cameron Daniel.  Back-patch to all supported
branches.

Discussion: https://postgr.es/m/16655-5c92d6b3a9438137@postgresql.org
Discussion: https://postgr.es/m/18048b44-3414-b983-8c7c-9165b177900d@2ndQuadrant.com
parent f0770709
...@@ -2098,8 +2098,6 @@ dumpTableData_insert(Archive *fout, void *dcontext) ...@@ -2098,8 +2098,6 @@ dumpTableData_insert(Archive *fout, void *dcontext)
if (nfields == 0) if (nfields == 0)
continue; continue;
Assert(tbinfo->attgenerated);
/* Emit a row heading */ /* Emit a row heading */
if (rows_per_statement == 1) if (rows_per_statement == 1)
archputs(" (", fout); archputs(" (", fout);
...@@ -2260,6 +2258,9 @@ dumpTableData(Archive *fout, TableDataInfo *tdinfo) ...@@ -2260,6 +2258,9 @@ dumpTableData(Archive *fout, TableDataInfo *tdinfo)
char *copyStmt; char *copyStmt;
const char *copyFrom; const char *copyFrom;
/* We had better have loaded per-column details about this table */
Assert(tbinfo->interesting);
if (dopt->dump_inserts == 0) if (dopt->dump_inserts == 0)
{ {
/* Dump/restore using COPY */ /* Dump/restore using COPY */
...@@ -2451,6 +2452,9 @@ makeTableDataInfo(DumpOptions *dopt, TableInfo *tbinfo) ...@@ -2451,6 +2452,9 @@ makeTableDataInfo(DumpOptions *dopt, TableInfo *tbinfo)
addObjectDependency(&tdinfo->dobj, tbinfo->dobj.dumpId); addObjectDependency(&tdinfo->dobj, tbinfo->dobj.dumpId);
tbinfo->dataObj = tdinfo; tbinfo->dataObj = tdinfo;
/* Make sure that we'll collect per-column info for this table. */
tbinfo->interesting = true;
} }
/* /*
...@@ -8728,7 +8732,7 @@ getTableAttrs(Archive *fout, TableInfo *tblinfo, int numTables) ...@@ -8728,7 +8732,7 @@ getTableAttrs(Archive *fout, TableInfo *tblinfo, int numTables)
* Get info about table CHECK constraints. This is skipped for a * Get info about table CHECK constraints. This is skipped for a
* data-only dump, as it is only needed for table schemas. * data-only dump, as it is only needed for table schemas.
*/ */
if (!dopt->dataOnly && tbinfo->ncheck > 0) if (tbinfo->ncheck > 0 && !dopt->dataOnly)
{ {
ConstraintInfo *constrs; ConstraintInfo *constrs;
int numConstrs; int numConstrs;
...@@ -15538,10 +15542,12 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) ...@@ -15538,10 +15542,12 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo)
int j, int j,
k; k;
/* We had better have loaded per-column details about this table */
Assert(tbinfo->interesting);
qrelname = pg_strdup(fmtId(tbinfo->dobj.name)); qrelname = pg_strdup(fmtId(tbinfo->dobj.name));
qualrelname = pg_strdup(fmtQualifiedDumpable(tbinfo)); qualrelname = pg_strdup(fmtQualifiedDumpable(tbinfo));
if (tbinfo->hasoids) if (tbinfo->hasoids)
pg_log_warning("WITH OIDS is not supported anymore (table \"%s\")", pg_log_warning("WITH OIDS is not supported anymore (table \"%s\")",
qrelname); qrelname);
...@@ -17912,8 +17918,6 @@ processExtensionTables(Archive *fout, ExtensionInfo extinfo[], ...@@ -17912,8 +17918,6 @@ processExtensionTables(Archive *fout, ExtensionInfo extinfo[],
configtbl->dataObj->filtercond = pg_strdup(extconditionarray[j]); configtbl->dataObj->filtercond = pg_strdup(extconditionarray[j]);
} }
} }
configtbl->interesting = dumpobj;
} }
} }
if (extconfigarray) if (extconfigarray)
......
...@@ -135,9 +135,17 @@ my %pgdump_runs = ( ...@@ -135,9 +135,17 @@ my %pgdump_runs = (
"$tempdir/defaults_tar_format.tar", "$tempdir/defaults_tar_format.tar",
], ],
}, },
exclude_table => {
dump_cmd => [
'pg_dump',
'--exclude-table=regress_table_dumpable',
"--file=$tempdir/exclude_table.sql",
'postgres',
],
},
extension_schema => { extension_schema => {
dump_cmd => [ dump_cmd => [
'pg_dump', '--schema=public', '--inserts', 'pg_dump', '--schema=public',
"--file=$tempdir/extension_schema.sql", 'postgres', "--file=$tempdir/extension_schema.sql", 'postgres',
], ],
}, },
...@@ -225,6 +233,7 @@ my %full_runs = ( ...@@ -225,6 +233,7 @@ my %full_runs = (
clean_if_exists => 1, clean_if_exists => 1,
createdb => 1, createdb => 1,
defaults => 1, defaults => 1,
exclude_table => 1,
no_privs => 1, no_privs => 1,
no_owner => 1,); no_owner => 1,);
...@@ -317,11 +326,28 @@ my %tests = ( ...@@ -317,11 +326,28 @@ my %tests = (
regexp => qr/^ regexp => qr/^
\QCREATE TABLE public.regress_pg_dump_table (\E \QCREATE TABLE public.regress_pg_dump_table (\E
\n\s+\Qcol1 integer NOT NULL,\E \n\s+\Qcol1 integer NOT NULL,\E
\n\s+\Qcol2 integer\E \n\s+\Qcol2 integer,\E
\n\s+\QCONSTRAINT regress_pg_dump_table_col2_check CHECK ((col2 > 0))\E
\n\);\n/xm, \n\);\n/xm,
like => { binary_upgrade => 1, }, like => { binary_upgrade => 1, },
}, },
'COPY public.regress_table_dumpable (col1)' => {
regexp => qr/^
\QCOPY public.regress_table_dumpable (col1) FROM stdin;\E
\n/xm,
like => {
%full_runs,
data_only => 1,
section_data => 1,
extension_schema => 1,
},
unlike => {
binary_upgrade => 1,
exclude_table => 1,
},
},
'CREATE ACCESS METHOD regress_test_am' => { 'CREATE ACCESS METHOD regress_test_am' => {
regexp => qr/^ regexp => qr/^
\QCREATE ACCESS METHOD regress_test_am TYPE INDEX HANDLER bthandler;\E \QCREATE ACCESS METHOD regress_test_am TYPE INDEX HANDLER bthandler;\E
...@@ -443,7 +469,8 @@ my %tests = ( ...@@ -443,7 +469,8 @@ my %tests = (
regexp => qr/^ regexp => qr/^
\QCREATE TABLE regress_pg_dump_schema.test_table (\E \QCREATE TABLE regress_pg_dump_schema.test_table (\E
\n\s+\Qcol1 integer,\E \n\s+\Qcol1 integer,\E
\n\s+\Qcol2 integer\E \n\s+\Qcol2 integer,\E
\n\s+\QCONSTRAINT test_table_col2_check CHECK ((col2 > 0))\E
\n\);\n/xm, \n\);\n/xm,
like => { binary_upgrade => 1, }, like => { binary_upgrade => 1, },
}, },
...@@ -578,17 +605,6 @@ my %tests = ( ...@@ -578,17 +605,6 @@ my %tests = (
schema_only => 1, schema_only => 1,
section_pre_data => 1, section_pre_data => 1,
}, },
},
# Dumpable object inside specific schema
'INSERT INTO public.regress_table_dumpable VALUES (1);' => {
create_sql => 'INSERT INTO public.regress_table_dumpable VALUES (1);',
regexp => qr/^
\QINSERT INTO public.regress_table_dumpable VALUES (1);\E
\n/xm,
like => {
extension_schema => 1,
},
},); },);
######################################### #########################################
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
CREATE TABLE regress_pg_dump_table ( CREATE TABLE regress_pg_dump_table (
col1 serial, col1 serial,
col2 int col2 int check (col2 > 0)
); );
CREATE SEQUENCE regress_pg_dump_seq; CREATE SEQUENCE regress_pg_dump_seq;
...@@ -14,7 +14,7 @@ CREATE SEQUENCE regress_seq_dumpable; ...@@ -14,7 +14,7 @@ CREATE SEQUENCE regress_seq_dumpable;
SELECT pg_catalog.pg_extension_config_dump('regress_seq_dumpable', ''); SELECT pg_catalog.pg_extension_config_dump('regress_seq_dumpable', '');
CREATE TABLE regress_table_dumpable ( CREATE TABLE regress_table_dumpable (
col1 int col1 int check (col1 > 0)
); );
SELECT pg_catalog.pg_extension_config_dump('regress_table_dumpable', ''); SELECT pg_catalog.pg_extension_config_dump('regress_table_dumpable', '');
...@@ -34,7 +34,7 @@ CREATE ACCESS METHOD regress_test_am TYPE INDEX HANDLER bthandler; ...@@ -34,7 +34,7 @@ CREATE ACCESS METHOD regress_test_am TYPE INDEX HANDLER bthandler;
-- this extension. -- this extension.
CREATE TABLE regress_pg_dump_schema.test_table ( CREATE TABLE regress_pg_dump_schema.test_table (
col1 int, col1 int,
col2 int col2 int check (col2 > 0)
); );
GRANT SELECT ON regress_pg_dump_schema.test_table TO regress_dump_test_role; GRANT SELECT ON regress_pg_dump_schema.test_table TO regress_dump_test_role;
......
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