mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Postpone calls of unsafe server-side functions in pg_dump.
Avoid calling pg_get_partkeydef(), pg_get_expr(relpartbound),
and regtypeout until we have lock on the relevant tables.
The existing coding is at serious risk of failure if there
are any concurrent DROP TABLE commands going on --- including
drops of other sessions' temp tables.
Back-patch of commit e3fcbbd62.  That's been in v15/HEAD long enough
to have some confidence about it, so now let's fix the problem in
older branches.
Original patch by me; thanks to Gilles Darold for back-patching
legwork.
Discussion: https://postgr.es/m/2273648.1634764485@sss.pgh.pa.us
Discussion: https://postgr.es/m/7d7eb6128f40401d81b3b7a898b6b4de@W2012-02.nidsa.loc
Discussion: https://postgr.es/m/45c93d57-9973-248e-d2df-e02ca9af48d4@darold.net
			
			
This commit is contained in:
		| @@ -6129,14 +6129,15 @@ getTables(Archive *fout, int *numTables) | |||||||
| 	int			i_foreignserver; | 	int			i_foreignserver; | ||||||
| 	int			i_is_identity_sequence; | 	int			i_is_identity_sequence; | ||||||
| 	int			i_changed_acl; | 	int			i_changed_acl; | ||||||
| 	int			i_partkeydef; |  | ||||||
| 	int			i_ispartition; | 	int			i_ispartition; | ||||||
| 	int			i_partbound; |  | ||||||
| 	int			i_amname; | 	int			i_amname; | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * Find all the tables and table-like objects. | 	 * Find all the tables and table-like objects. | ||||||
| 	 * | 	 * | ||||||
|  | 	 * We must fetch all tables in this phase because otherwise we cannot | ||||||
|  | 	 * correctly identify inherited columns, owned sequences, etc. | ||||||
|  | 	 * | ||||||
| 	 * We include system catalogs, so that we can work if a user table is | 	 * We include system catalogs, so that we can work if a user table is | ||||||
| 	 * defined to inherit from a system catalog (pretty weird, but...) | 	 * defined to inherit from a system catalog (pretty weird, but...) | ||||||
| 	 * | 	 * | ||||||
| @@ -6150,8 +6151,10 @@ getTables(Archive *fout, int *numTables) | |||||||
| 	 * | 	 * | ||||||
| 	 * Note: in this phase we should collect only a minimal amount of | 	 * Note: in this phase we should collect only a minimal amount of | ||||||
| 	 * information about each table, basically just enough to decide if it is | 	 * information about each table, basically just enough to decide if it is | ||||||
| 	 * interesting. We must fetch all tables in this phase because otherwise | 	 * interesting.  In particular, since we do not yet have lock on any user | ||||||
| 	 * we cannot correctly identify inherited columns, owned sequences, etc. | 	 * table, we MUST NOT invoke any server-side data collection functions | ||||||
|  | 	 * (for instance, pg_get_partkeydef()).  Those are likely to fail or give | ||||||
|  | 	 * wrong answers if any concurrent DDL is happening. | ||||||
| 	 * | 	 * | ||||||
| 	 * We purposefully ignore toast OIDs for partitioned tables; the reason is | 	 * We purposefully ignore toast OIDs for partitioned tables; the reason is | ||||||
| 	 * that versions 10 and 11 have them, but 12 does not, so emitting them | 	 * that versions 10 and 11 have them, but 12 does not, so emitting them | ||||||
| @@ -6160,9 +6163,7 @@ getTables(Archive *fout, int *numTables) | |||||||
|  |  | ||||||
| 	if (fout->remoteVersion >= 90600) | 	if (fout->remoteVersion >= 90600) | ||||||
| 	{ | 	{ | ||||||
| 		char	   *partkeydef = "NULL"; |  | ||||||
| 		char	   *ispartition = "false"; | 		char	   *ispartition = "false"; | ||||||
| 		char	   *partbound = "NULL"; |  | ||||||
| 		char	   *relhasoids = "c.relhasoids"; | 		char	   *relhasoids = "c.relhasoids"; | ||||||
|  |  | ||||||
| 		PQExpBuffer acl_subquery = createPQExpBuffer(); | 		PQExpBuffer acl_subquery = createPQExpBuffer(); | ||||||
| @@ -6179,13 +6180,8 @@ getTables(Archive *fout, int *numTables) | |||||||
| 		 * Collect the information about any partitioned tables, which were | 		 * Collect the information about any partitioned tables, which were | ||||||
| 		 * added in PG10. | 		 * added in PG10. | ||||||
| 		 */ | 		 */ | ||||||
|  |  | ||||||
| 		if (fout->remoteVersion >= 100000) | 		if (fout->remoteVersion >= 100000) | ||||||
| 		{ |  | ||||||
| 			partkeydef = "pg_get_partkeydef(c.oid)"; |  | ||||||
| 			ispartition = "c.relispartition"; | 			ispartition = "c.relispartition"; | ||||||
| 			partbound = "pg_get_expr(c.relpartbound, c.oid)"; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		/* In PG12 upwards WITH OIDS does not exist anymore. */ | 		/* In PG12 upwards WITH OIDS does not exist anymore. */ | ||||||
| 		if (fout->remoteVersion >= 120000) | 		if (fout->remoteVersion >= 120000) | ||||||
| @@ -6226,7 +6222,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "CASE WHEN c.relkind = 'f' THEN " | 						  "CASE WHEN c.relkind = 'f' THEN " | ||||||
| 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | ||||||
| 						  "ELSE 0 END AS foreignserver, " | 						  "ELSE 0 END AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| @@ -6246,9 +6242,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "OR %s IS NOT NULL" | 						  "OR %s IS NOT NULL" | ||||||
| 						  "))" | 						  "))" | ||||||
| 						  "AS changed_acl, " | 						  "AS changed_acl, " | ||||||
| 						  "%s AS partkeydef, " | 						  "%s AS ispartition " | ||||||
| 						  "%s AS ispartition, " |  | ||||||
| 						  "%s AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6274,9 +6268,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  attracl_subquery->data, | 						  attracl_subquery->data, | ||||||
| 						  attinitacl_subquery->data, | 						  attinitacl_subquery->data, | ||||||
| 						  attinitracl_subquery->data, | 						  attinitracl_subquery->data, | ||||||
| 						  partkeydef, |  | ||||||
| 						  ispartition, | 						  ispartition, | ||||||
| 						  partbound, |  | ||||||
| 						  RELKIND_SEQUENCE, | 						  RELKIND_SEQUENCE, | ||||||
| 						  RELKIND_PARTITIONED_TABLE, | 						  RELKIND_PARTITIONED_TABLE, | ||||||
| 						  RELKIND_RELATION, RELKIND_SEQUENCE, | 						  RELKIND_RELATION, RELKIND_SEQUENCE, | ||||||
| @@ -6319,7 +6311,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "CASE WHEN c.relkind = 'f' THEN " | 						  "CASE WHEN c.relkind = 'f' THEN " | ||||||
| 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | ||||||
| 						  "ELSE 0 END AS foreignserver, " | 						  "ELSE 0 END AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| @@ -6328,9 +6320,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6372,7 +6362,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "CASE WHEN c.relkind = 'f' THEN " | 						  "CASE WHEN c.relkind = 'f' THEN " | ||||||
| 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | ||||||
| 						  "ELSE 0 END AS foreignserver, " | 						  "ELSE 0 END AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| @@ -6381,9 +6371,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6425,7 +6413,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "CASE WHEN c.relkind = 'f' THEN " | 						  "CASE WHEN c.relkind = 'f' THEN " | ||||||
| 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | ||||||
| 						  "ELSE 0 END AS foreignserver, " | 						  "ELSE 0 END AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| @@ -6434,9 +6422,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | 						  "WHEN 'check_option=cascaded' = ANY (c.reloptions) THEN 'CASCADED'::text ELSE NULL END AS checkoption, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6478,16 +6464,14 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "CASE WHEN c.relkind = 'f' THEN " | 						  "CASE WHEN c.relkind = 'f' THEN " | ||||||
| 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | 						  "(SELECT ftserver FROM pg_catalog.pg_foreign_table WHERE ftrelid = c.oid) " | ||||||
| 						  "ELSE 0 END AS foreignserver, " | 						  "ELSE 0 END AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| 						  "c.reloptions AS reloptions, " | 						  "c.reloptions AS reloptions, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6527,16 +6511,14 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "'d' AS relreplident, c.relpages, " | 						  "'d' AS relreplident, c.relpages, " | ||||||
| 						  "NULL AS amname, " | 						  "NULL AS amname, " | ||||||
| 						  "NULL AS foreignserver, " | 						  "NULL AS foreignserver, " | ||||||
| 						  "CASE WHEN c.reloftype <> 0 THEN c.reloftype::pg_catalog.regtype ELSE NULL END AS reloftype, " | 						  "c.reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| 						  "c.reloptions AS reloptions, " | 						  "c.reloptions AS reloptions, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6575,16 +6557,14 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "'d' AS relreplident, c.relpages, " | 						  "'d' AS relreplident, c.relpages, " | ||||||
| 						  "NULL AS amname, " | 						  "NULL AS amname, " | ||||||
| 						  "NULL AS foreignserver, " | 						  "NULL AS foreignserver, " | ||||||
| 						  "NULL AS reloftype, " | 						  "0 AS reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| 						  "c.reloptions AS reloptions, " | 						  "c.reloptions AS reloptions, " | ||||||
| 						  "tc.reloptions AS toast_reloptions, " | 						  "tc.reloptions AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6623,16 +6603,14 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "'d' AS relreplident, c.relpages, " | 						  "'d' AS relreplident, c.relpages, " | ||||||
| 						  "NULL AS amname, " | 						  "NULL AS amname, " | ||||||
| 						  "NULL AS foreignserver, " | 						  "NULL AS foreignserver, " | ||||||
| 						  "NULL AS reloftype, " | 						  "0 AS reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| 						  "c.reloptions AS reloptions, " | 						  "c.reloptions AS reloptions, " | ||||||
| 						  "NULL AS toast_reloptions, " | 						  "NULL AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6670,16 +6648,14 @@ getTables(Archive *fout, int *numTables) | |||||||
| 						  "'d' AS relreplident, relpages, " | 						  "'d' AS relreplident, relpages, " | ||||||
| 						  "NULL AS amname, " | 						  "NULL AS amname, " | ||||||
| 						  "NULL AS foreignserver, " | 						  "NULL AS foreignserver, " | ||||||
| 						  "NULL AS reloftype, " | 						  "0 AS reloftype, " | ||||||
| 						  "d.refobjid AS owning_tab, " | 						  "d.refobjid AS owning_tab, " | ||||||
| 						  "d.refobjsubid AS owning_col, " | 						  "d.refobjsubid AS owning_col, " | ||||||
| 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | 						  "(SELECT spcname FROM pg_tablespace t WHERE t.oid = c.reltablespace) AS reltablespace, " | ||||||
| 						  "NULL AS reloptions, " | 						  "NULL AS reloptions, " | ||||||
| 						  "NULL AS toast_reloptions, " | 						  "NULL AS toast_reloptions, " | ||||||
| 						  "NULL AS changed_acl, " | 						  "NULL AS changed_acl, " | ||||||
| 						  "NULL AS partkeydef, " | 						  "false AS ispartition " | ||||||
| 						  "false AS ispartition, " |  | ||||||
| 						  "NULL AS partbound " |  | ||||||
| 						  "FROM pg_class c " | 						  "FROM pg_class c " | ||||||
| 						  "LEFT JOIN pg_depend d ON " | 						  "LEFT JOIN pg_depend d ON " | ||||||
| 						  "(c.relkind = '%c' AND " | 						  "(c.relkind = '%c' AND " | ||||||
| @@ -6747,9 +6723,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 	i_reloftype = PQfnumber(res, "reloftype"); | 	i_reloftype = PQfnumber(res, "reloftype"); | ||||||
| 	i_is_identity_sequence = PQfnumber(res, "is_identity_sequence"); | 	i_is_identity_sequence = PQfnumber(res, "is_identity_sequence"); | ||||||
| 	i_changed_acl = PQfnumber(res, "changed_acl"); | 	i_changed_acl = PQfnumber(res, "changed_acl"); | ||||||
| 	i_partkeydef = PQfnumber(res, "partkeydef"); |  | ||||||
| 	i_ispartition = PQfnumber(res, "ispartition"); | 	i_ispartition = PQfnumber(res, "ispartition"); | ||||||
| 	i_partbound = PQfnumber(res, "partbound"); |  | ||||||
| 	i_amname = PQfnumber(res, "amname"); | 	i_amname = PQfnumber(res, "amname"); | ||||||
|  |  | ||||||
| 	if (dopt->lockWaitTimeout) | 	if (dopt->lockWaitTimeout) | ||||||
| @@ -6798,10 +6772,7 @@ getTables(Archive *fout, int *numTables) | |||||||
| 		tblinfo[i].toast_oid = atooid(PQgetvalue(res, i, i_toastoid)); | 		tblinfo[i].toast_oid = atooid(PQgetvalue(res, i, i_toastoid)); | ||||||
| 		tblinfo[i].toast_frozenxid = atooid(PQgetvalue(res, i, i_toastfrozenxid)); | 		tblinfo[i].toast_frozenxid = atooid(PQgetvalue(res, i, i_toastfrozenxid)); | ||||||
| 		tblinfo[i].toast_minmxid = atooid(PQgetvalue(res, i, i_toastminmxid)); | 		tblinfo[i].toast_minmxid = atooid(PQgetvalue(res, i, i_toastminmxid)); | ||||||
| 		if (PQgetisnull(res, i, i_reloftype)) | 		tblinfo[i].reloftype = atooid(PQgetvalue(res, i, i_reloftype)); | ||||||
| 			tblinfo[i].reloftype = NULL; |  | ||||||
| 		else |  | ||||||
| 			tblinfo[i].reloftype = pg_strdup(PQgetvalue(res, i, i_reloftype)); |  | ||||||
| 		tblinfo[i].ncheck = atoi(PQgetvalue(res, i, i_relchecks)); | 		tblinfo[i].ncheck = atoi(PQgetvalue(res, i, i_relchecks)); | ||||||
| 		if (PQgetisnull(res, i, i_owning_tab)) | 		if (PQgetisnull(res, i, i_owning_tab)) | ||||||
| 		{ | 		{ | ||||||
| @@ -6857,10 +6828,8 @@ getTables(Archive *fout, int *numTables) | |||||||
| 		tblinfo[i].is_identity_sequence = (i_is_identity_sequence >= 0 && | 		tblinfo[i].is_identity_sequence = (i_is_identity_sequence >= 0 && | ||||||
| 										   strcmp(PQgetvalue(res, i, i_is_identity_sequence), "t") == 0); | 										   strcmp(PQgetvalue(res, i, i_is_identity_sequence), "t") == 0); | ||||||
|  |  | ||||||
| 		/* Partition key string or NULL */ | 		/* Partition? */ | ||||||
| 		tblinfo[i].partkeydef = pg_strdup(PQgetvalue(res, i, i_partkeydef)); |  | ||||||
| 		tblinfo[i].ispartition = (strcmp(PQgetvalue(res, i, i_ispartition), "t") == 0); | 		tblinfo[i].ispartition = (strcmp(PQgetvalue(res, i, i_ispartition), "t") == 0); | ||||||
| 		tblinfo[i].partbound = pg_strdup(PQgetvalue(res, i, i_partbound)); |  | ||||||
|  |  | ||||||
| 		/* foreign server */ | 		/* foreign server */ | ||||||
| 		tblinfo[i].foreign_server = atooid(PQgetvalue(res, i, i_foreignserver)); | 		tblinfo[i].foreign_server = atooid(PQgetvalue(res, i, i_foreignserver)); | ||||||
| @@ -15934,12 +15903,34 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 	} | 	} | ||||||
| 	else | 	else | ||||||
| 	{ | 	{ | ||||||
|  | 		char	   *partkeydef = NULL; | ||||||
| 		char	   *ftoptions = NULL; | 		char	   *ftoptions = NULL; | ||||||
| 		char	   *srvname = NULL; | 		char	   *srvname = NULL; | ||||||
| 		char	   *foreign = ""; | 		char	   *foreign = ""; | ||||||
|  |  | ||||||
|  | 		/* | ||||||
|  | 		 * Set reltypename, and collect any relkind-specific data that we | ||||||
|  | 		 * didn't fetch during getTables(). | ||||||
|  | 		 */ | ||||||
| 		switch (tbinfo->relkind) | 		switch (tbinfo->relkind) | ||||||
| 		{ | 		{ | ||||||
|  | 			case RELKIND_PARTITIONED_TABLE: | ||||||
|  | 				{ | ||||||
|  | 					PQExpBuffer query = createPQExpBuffer(); | ||||||
|  | 					PGresult   *res; | ||||||
|  |  | ||||||
|  | 					reltypename = "TABLE"; | ||||||
|  |  | ||||||
|  | 					/* retrieve partition key definition */ | ||||||
|  | 					appendPQExpBuffer(query, | ||||||
|  | 									  "SELECT pg_get_partkeydef('%u')", | ||||||
|  | 									  tbinfo->dobj.catId.oid); | ||||||
|  | 					res = ExecuteSqlQueryForSingleRow(fout, query->data); | ||||||
|  | 					partkeydef = pg_strdup(PQgetvalue(res, 0, 0)); | ||||||
|  | 					PQclear(res); | ||||||
|  | 					destroyPQExpBuffer(query); | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
| 			case RELKIND_FOREIGN_TABLE: | 			case RELKIND_FOREIGN_TABLE: | ||||||
| 				{ | 				{ | ||||||
| 					PQExpBuffer query = createPQExpBuffer(); | 					PQExpBuffer query = createPQExpBuffer(); | ||||||
| @@ -15979,6 +15970,7 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 				break; | 				break; | ||||||
| 			default: | 			default: | ||||||
| 				reltypename = "TABLE"; | 				reltypename = "TABLE"; | ||||||
|  | 				break; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		numParents = tbinfo->numParents; | 		numParents = tbinfo->numParents; | ||||||
| @@ -16000,8 +15992,10 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 		 * Attach to type, if reloftype; except in case of a binary upgrade, | 		 * Attach to type, if reloftype; except in case of a binary upgrade, | ||||||
| 		 * we dump the table normally and attach it to the type afterward. | 		 * we dump the table normally and attach it to the type afterward. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (tbinfo->reloftype && !dopt->binary_upgrade) | 		if (OidIsValid(tbinfo->reloftype) && !dopt->binary_upgrade) | ||||||
| 			appendPQExpBuffer(q, " OF %s", tbinfo->reloftype); | 			appendPQExpBuffer(q, " OF %s", | ||||||
|  | 							  getFormattedTypeName(fout, tbinfo->reloftype, | ||||||
|  | 												   zeroIsError)); | ||||||
|  |  | ||||||
| 		if (tbinfo->relkind != RELKIND_MATVIEW) | 		if (tbinfo->relkind != RELKIND_MATVIEW) | ||||||
| 		{ | 		{ | ||||||
| @@ -16039,7 +16033,8 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 					 * Skip column if fully defined by reloftype, except in | 					 * Skip column if fully defined by reloftype, except in | ||||||
| 					 * binary upgrade | 					 * binary upgrade | ||||||
| 					 */ | 					 */ | ||||||
| 					if (tbinfo->reloftype && !print_default && !print_notnull && | 					if (OidIsValid(tbinfo->reloftype) && | ||||||
|  | 						!print_default && !print_notnull && | ||||||
| 						!dopt->binary_upgrade) | 						!dopt->binary_upgrade) | ||||||
| 						continue; | 						continue; | ||||||
|  |  | ||||||
| @@ -16072,7 +16067,7 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 					 * table ('OF type_name'), but in binary-upgrade mode, | 					 * table ('OF type_name'), but in binary-upgrade mode, | ||||||
| 					 * print it in that case too. | 					 * print it in that case too. | ||||||
| 					 */ | 					 */ | ||||||
| 					if (dopt->binary_upgrade || !tbinfo->reloftype) | 					if (dopt->binary_upgrade || !OidIsValid(tbinfo->reloftype)) | ||||||
| 					{ | 					{ | ||||||
| 						appendPQExpBuffer(q, " %s", | 						appendPQExpBuffer(q, " %s", | ||||||
| 										  tbinfo->atttypnames[j]); | 										  tbinfo->atttypnames[j]); | ||||||
| @@ -16135,7 +16130,7 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
|  |  | ||||||
| 			if (actual_atts) | 			if (actual_atts) | ||||||
| 				appendPQExpBufferStr(q, "\n)"); | 				appendPQExpBufferStr(q, "\n)"); | ||||||
| 			else if (!(tbinfo->reloftype && !dopt->binary_upgrade)) | 			else if (!(OidIsValid(tbinfo->reloftype) && !dopt->binary_upgrade)) | ||||||
| 			{ | 			{ | ||||||
| 				/* | 				/* | ||||||
| 				 * No attributes? we must have a parenthesized attribute list, | 				 * No attributes? we must have a parenthesized attribute list, | ||||||
| @@ -16164,7 +16159,7 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if (tbinfo->relkind == RELKIND_PARTITIONED_TABLE) | 			if (tbinfo->relkind == RELKIND_PARTITIONED_TABLE) | ||||||
| 				appendPQExpBuffer(q, "\nPARTITION BY %s", tbinfo->partkeydef); | 				appendPQExpBuffer(q, "\nPARTITION BY %s", partkeydef); | ||||||
|  |  | ||||||
| 			if (tbinfo->relkind == RELKIND_FOREIGN_TABLE) | 			if (tbinfo->relkind == RELKIND_FOREIGN_TABLE) | ||||||
| 				appendPQExpBuffer(q, "\nSERVER %s", fmtId(srvname)); | 				appendPQExpBuffer(q, "\nSERVER %s", fmtId(srvname)); | ||||||
| @@ -16347,12 +16342,13 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if (tbinfo->reloftype) | 			if (OidIsValid(tbinfo->reloftype)) | ||||||
| 			{ | 			{ | ||||||
| 				appendPQExpBufferStr(q, "\n-- For binary upgrade, set up typed tables this way.\n"); | 				appendPQExpBufferStr(q, "\n-- For binary upgrade, set up typed tables this way.\n"); | ||||||
| 				appendPQExpBuffer(q, "ALTER TABLE ONLY %s OF %s;\n", | 				appendPQExpBuffer(q, "ALTER TABLE ONLY %s OF %s;\n", | ||||||
| 								  qualrelname, | 								  qualrelname, | ||||||
| 								  tbinfo->reloftype); | 								  getFormattedTypeName(fout, tbinfo->reloftype, | ||||||
|  | 													   zeroIsError)); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @@ -16365,16 +16361,34 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 		 */ | 		 */ | ||||||
| 		if (tbinfo->ispartition) | 		if (tbinfo->ispartition) | ||||||
| 		{ | 		{ | ||||||
|  | 			PGresult   *ares; | ||||||
|  | 			char	   *partbound; | ||||||
|  | 			PQExpBuffer q2; | ||||||
|  |  | ||||||
| 			/* With partitions there can only be one parent */ | 			/* With partitions there can only be one parent */ | ||||||
| 			if (tbinfo->numParents != 1) | 			if (tbinfo->numParents != 1) | ||||||
| 				fatal("invalid number of parents %d for table \"%s\"", | 				fatal("invalid number of parents %d for table \"%s\"", | ||||||
| 					  tbinfo->numParents, tbinfo->dobj.name); | 					  tbinfo->numParents, tbinfo->dobj.name); | ||||||
|  |  | ||||||
|  | 			q2 = createPQExpBuffer(); | ||||||
|  |  | ||||||
|  | 			/* Fetch the partition's partbound */ | ||||||
|  | 			appendPQExpBuffer(q2, | ||||||
|  | 							  "SELECT pg_get_expr(c.relpartbound, c.oid) " | ||||||
|  | 							  "FROM pg_class c " | ||||||
|  | 							  "WHERE c.oid = '%u'", | ||||||
|  | 							  tbinfo->dobj.catId.oid); | ||||||
|  | 			ares = ExecuteSqlQueryForSingleRow(fout, q2->data); | ||||||
|  | 			partbound = PQgetvalue(ares, 0, 0); | ||||||
|  |  | ||||||
| 			/* Perform ALTER TABLE on the parent */ | 			/* Perform ALTER TABLE on the parent */ | ||||||
| 			appendPQExpBuffer(q, | 			appendPQExpBuffer(q, | ||||||
| 							  "ALTER TABLE ONLY %s ATTACH PARTITION %s %s;\n", | 							  "ALTER TABLE ONLY %s ATTACH PARTITION %s %s;\n", | ||||||
| 							  fmtQualifiedDumpable(parents[0]), | 							  fmtQualifiedDumpable(parents[0]), | ||||||
| 							  qualrelname, tbinfo->partbound); | 							  qualrelname, partbound); | ||||||
|  |  | ||||||
|  | 			PQclear(ares); | ||||||
|  | 			destroyPQExpBuffer(q2); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		/* | 		/* | ||||||
| @@ -16519,6 +16533,8 @@ dumpTableSchema(Archive *fout, TableInfo *tbinfo) | |||||||
| 								  tbinfo->attfdwoptions[j]); | 								  tbinfo->attfdwoptions[j]); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if (partkeydef) | ||||||
|  | 			free(partkeydef); | ||||||
| 		if (ftoptions) | 		if (ftoptions) | ||||||
| 			free(ftoptions); | 			free(ftoptions); | ||||||
| 		if (srvname) | 		if (srvname) | ||||||
|   | |||||||
| @@ -284,7 +284,7 @@ typedef struct _tableInfo | |||||||
| 	uint32		toast_frozenxid;	/* toast table's relfrozenxid, if any */ | 	uint32		toast_frozenxid;	/* toast table's relfrozenxid, if any */ | ||||||
| 	uint32		toast_minmxid;	/* toast table's relminmxid */ | 	uint32		toast_minmxid;	/* toast table's relminmxid */ | ||||||
| 	int			ncheck;			/* # of CHECK expressions */ | 	int			ncheck;			/* # of CHECK expressions */ | ||||||
| 	char	   *reloftype;		/* underlying type for typed table */ | 	Oid			reloftype;		/* underlying type for typed table */ | ||||||
| 	Oid			foreign_server; /* foreign server oid, if applicable */ | 	Oid			foreign_server; /* foreign server oid, if applicable */ | ||||||
| 	/* these two are set only if table is a sequence owned by a column: */ | 	/* these two are set only if table is a sequence owned by a column: */ | ||||||
| 	Oid			owning_tab;		/* OID of table owning sequence */ | 	Oid			owning_tab;		/* OID of table owning sequence */ | ||||||
| @@ -322,8 +322,6 @@ typedef struct _tableInfo | |||||||
| 	bool	   *inhNotNull;		/* true if NOT NULL is inherited */ | 	bool	   *inhNotNull;		/* true if NOT NULL is inherited */ | ||||||
| 	struct _attrDefInfo **attrdefs; /* DEFAULT expressions */ | 	struct _attrDefInfo **attrdefs; /* DEFAULT expressions */ | ||||||
| 	struct _constraintInfo *checkexprs; /* CHECK constraints */ | 	struct _constraintInfo *checkexprs; /* CHECK constraints */ | ||||||
| 	char	   *partkeydef;		/* partition key definition */ |  | ||||||
| 	char	   *partbound;		/* partition bound definition */ |  | ||||||
| 	bool		needs_override; /* has GENERATED ALWAYS AS IDENTITY */ | 	bool		needs_override; /* has GENERATED ALWAYS AS IDENTITY */ | ||||||
| 	char	   *amname;			/* relation access method */ | 	char	   *amname;			/* relation access method */ | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user