mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Rethink function argument sorting in pg_dump.
Commit 7b583b20b1 created an unnecessary
dump failure hazard by applying pg_get_function_identity_arguments()
to every function in the database, even those that won't get dumped.
This could result in snapshot-related problems if concurrent sessions are,
for example, creating and dropping temporary functions, as noted by Marko
Tiikkaja in bug #12832.  While this is by no means pg_dump's only such
issue with concurrent DDL, it's unfortunate that we added a new failure
mode for cases that used to work, and even more so that the failure was
created for basically cosmetic reasons (ie, to sort overloaded functions
more deterministically).
To fix, revert that patch and instead sort function arguments using
information that pg_dump has available anyway, namely the names of the
argument types.  This will produce a slightly different sort ordering for
overloaded functions than the previous coding; but applying strcmp
directly to the output of pg_get_function_identity_arguments really was
a bit odd anyway.  The sorting will still be name-based and hence
independent of possibly-installation-specific OID assignments.  A small
additional benefit is that sorting now works regardless of server version.
Back-patch to 9.3, where the previous commit appeared.
			
			
This commit is contained in:
		| @@ -3951,7 +3951,6 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 	int			i_proargtypes; | 	int			i_proargtypes; | ||||||
| 	int			i_rolname; | 	int			i_rolname; | ||||||
| 	int			i_aggacl; | 	int			i_aggacl; | ||||||
| 	int			i_proiargs; |  | ||||||
|  |  | ||||||
| 	/* Make sure we are in proper schema */ | 	/* Make sure we are in proper schema */ | ||||||
| 	selectSourceSchema(fout, "pg_catalog"); | 	selectSourceSchema(fout, "pg_catalog"); | ||||||
| @@ -3961,12 +3960,11 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 	 * rationale behind the filtering logic. | 	 * rationale behind the filtering logic. | ||||||
| 	 */ | 	 */ | ||||||
|  |  | ||||||
| 	if (fout->remoteVersion >= 80400) | 	if (fout->remoteVersion >= 80200) | ||||||
| 	{ | 	{ | ||||||
| 		appendPQExpBuffer(query, "SELECT tableoid, oid, proname AS aggname, " | 		appendPQExpBuffer(query, "SELECT tableoid, oid, proname AS aggname, " | ||||||
| 						  "pronamespace AS aggnamespace, " | 						  "pronamespace AS aggnamespace, " | ||||||
| 						  "pronargs, proargtypes, " | 						  "pronargs, proargtypes, " | ||||||
| 			"pg_catalog.pg_get_function_identity_arguments(oid) AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname, " | 						  "(%s proowner) AS rolname, " | ||||||
| 						  "proacl AS aggacl " | 						  "proacl AS aggacl " | ||||||
| 						  "FROM pg_proc p " | 						  "FROM pg_proc p " | ||||||
| @@ -3984,28 +3982,12 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 								 "deptype = 'e')"); | 								 "deptype = 'e')"); | ||||||
| 		appendPQExpBufferChar(query, ')'); | 		appendPQExpBufferChar(query, ')'); | ||||||
| 	} | 	} | ||||||
| 	else if (fout->remoteVersion >= 80200) |  | ||||||
| 	{ |  | ||||||
| 		appendPQExpBuffer(query, "SELECT tableoid, oid, proname AS aggname, " |  | ||||||
| 						  "pronamespace AS aggnamespace, " |  | ||||||
| 						  "pronargs, proargtypes, " |  | ||||||
| 						  "NULL::text AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname, " |  | ||||||
| 						  "proacl AS aggacl " |  | ||||||
| 						  "FROM pg_proc p " |  | ||||||
| 						  "WHERE proisagg AND (" |  | ||||||
| 						  "pronamespace != " |  | ||||||
| 						  "(SELECT oid FROM pg_namespace " |  | ||||||
| 						  "WHERE nspname = 'pg_catalog'))", |  | ||||||
| 						  username_subquery); |  | ||||||
| 	} |  | ||||||
| 	else if (fout->remoteVersion >= 70300) | 	else if (fout->remoteVersion >= 70300) | ||||||
| 	{ | 	{ | ||||||
| 		appendPQExpBuffer(query, "SELECT tableoid, oid, proname AS aggname, " | 		appendPQExpBuffer(query, "SELECT tableoid, oid, proname AS aggname, " | ||||||
| 						  "pronamespace AS aggnamespace, " | 						  "pronamespace AS aggnamespace, " | ||||||
| 						  "CASE WHEN proargtypes[0] = 'pg_catalog.\"any\"'::pg_catalog.regtype THEN 0 ELSE 1 END AS pronargs, " | 						  "CASE WHEN proargtypes[0] = 'pg_catalog.\"any\"'::pg_catalog.regtype THEN 0 ELSE 1 END AS pronargs, " | ||||||
| 						  "proargtypes, " | 						  "proargtypes, " | ||||||
| 						  "NULL::text AS proiargs, " |  | ||||||
| 						  "(%s proowner) AS rolname, " | 						  "(%s proowner) AS rolname, " | ||||||
| 						  "proacl AS aggacl " | 						  "proacl AS aggacl " | ||||||
| 						  "FROM pg_proc " | 						  "FROM pg_proc " | ||||||
| @@ -4020,7 +4002,6 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 						  "0::oid AS aggnamespace, " | 						  "0::oid AS aggnamespace, " | ||||||
| 				  "CASE WHEN aggbasetype = 0 THEN 0 ELSE 1 END AS pronargs, " | 				  "CASE WHEN aggbasetype = 0 THEN 0 ELSE 1 END AS pronargs, " | ||||||
| 						  "aggbasetype AS proargtypes, " | 						  "aggbasetype AS proargtypes, " | ||||||
| 						  "NULL::text AS proiargs, " |  | ||||||
| 						  "(%s aggowner) AS rolname, " | 						  "(%s aggowner) AS rolname, " | ||||||
| 						  "'{=X}' AS aggacl " | 						  "'{=X}' AS aggacl " | ||||||
| 						  "FROM pg_aggregate " | 						  "FROM pg_aggregate " | ||||||
| @@ -4036,7 +4017,6 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 						  "0::oid AS aggnamespace, " | 						  "0::oid AS aggnamespace, " | ||||||
| 				  "CASE WHEN aggbasetype = 0 THEN 0 ELSE 1 END AS pronargs, " | 				  "CASE WHEN aggbasetype = 0 THEN 0 ELSE 1 END AS pronargs, " | ||||||
| 						  "aggbasetype AS proargtypes, " | 						  "aggbasetype AS proargtypes, " | ||||||
| 						  "NULL::text AS proiargs, " |  | ||||||
| 						  "(%s aggowner) AS rolname, " | 						  "(%s aggowner) AS rolname, " | ||||||
| 						  "'{=X}' AS aggacl " | 						  "'{=X}' AS aggacl " | ||||||
| 						  "FROM pg_aggregate " | 						  "FROM pg_aggregate " | ||||||
| @@ -4060,7 +4040,6 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 	i_proargtypes = PQfnumber(res, "proargtypes"); | 	i_proargtypes = PQfnumber(res, "proargtypes"); | ||||||
| 	i_rolname = PQfnumber(res, "rolname"); | 	i_rolname = PQfnumber(res, "rolname"); | ||||||
| 	i_aggacl = PQfnumber(res, "aggacl"); | 	i_aggacl = PQfnumber(res, "aggacl"); | ||||||
| 	i_proiargs = PQfnumber(res, "proiargs"); |  | ||||||
|  |  | ||||||
| 	for (i = 0; i < ntups; i++) | 	for (i = 0; i < ntups; i++) | ||||||
| 	{ | 	{ | ||||||
| @@ -4080,7 +4059,6 @@ getAggregates(Archive *fout, int *numAggs) | |||||||
| 		agginfo[i].aggfn.lang = InvalidOid;		/* not currently interesting */ | 		agginfo[i].aggfn.lang = InvalidOid;		/* not currently interesting */ | ||||||
| 		agginfo[i].aggfn.prorettype = InvalidOid;		/* not saved */ | 		agginfo[i].aggfn.prorettype = InvalidOid;		/* not saved */ | ||||||
| 		agginfo[i].aggfn.proacl = pg_strdup(PQgetvalue(res, i, i_aggacl)); | 		agginfo[i].aggfn.proacl = pg_strdup(PQgetvalue(res, i, i_aggacl)); | ||||||
| 		agginfo[i].aggfn.proiargs = pg_strdup(PQgetvalue(res, i, i_proiargs)); |  | ||||||
| 		agginfo[i].aggfn.nargs = atoi(PQgetvalue(res, i, i_pronargs)); | 		agginfo[i].aggfn.nargs = atoi(PQgetvalue(res, i, i_pronargs)); | ||||||
| 		if (agginfo[i].aggfn.nargs == 0) | 		if (agginfo[i].aggfn.nargs == 0) | ||||||
| 			agginfo[i].aggfn.argtypes = NULL; | 			agginfo[i].aggfn.argtypes = NULL; | ||||||
| @@ -4132,7 +4110,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 	int			i_proargtypes; | 	int			i_proargtypes; | ||||||
| 	int			i_prorettype; | 	int			i_prorettype; | ||||||
| 	int			i_proacl; | 	int			i_proacl; | ||||||
| 	int			i_proiargs; |  | ||||||
|  |  | ||||||
| 	/* Make sure we are in proper schema */ | 	/* Make sure we are in proper schema */ | ||||||
| 	selectSourceSchema(fout, "pg_catalog"); | 	selectSourceSchema(fout, "pg_catalog"); | ||||||
| @@ -4153,13 +4130,12 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 	 * doesn't have; otherwise we might not get creation ordering correct. | 	 * doesn't have; otherwise we might not get creation ordering correct. | ||||||
| 	 */ | 	 */ | ||||||
|  |  | ||||||
| 	if (fout->remoteVersion >= 80400) | 	if (fout->remoteVersion >= 70300) | ||||||
| 	{ | 	{ | ||||||
| 		appendPQExpBuffer(query, | 		appendPQExpBuffer(query, | ||||||
| 						  "SELECT tableoid, oid, proname, prolang, " | 						  "SELECT tableoid, oid, proname, prolang, " | ||||||
| 						  "pronargs, proargtypes, prorettype, proacl, " | 						  "pronargs, proargtypes, prorettype, proacl, " | ||||||
| 						  "pronamespace, " | 						  "pronamespace, " | ||||||
| 			"pg_catalog.pg_get_function_identity_arguments(oid) AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname " | 						  "(%s proowner) AS rolname " | ||||||
| 						  "FROM pg_proc p " | 						  "FROM pg_proc p " | ||||||
| 						  "WHERE NOT proisagg AND (" | 						  "WHERE NOT proisagg AND (" | ||||||
| @@ -4181,21 +4157,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 								 "deptype = 'e')"); | 								 "deptype = 'e')"); | ||||||
| 		appendPQExpBufferChar(query, ')'); | 		appendPQExpBufferChar(query, ')'); | ||||||
| 	} | 	} | ||||||
| 	else if (fout->remoteVersion >= 70300) |  | ||||||
| 	{ |  | ||||||
| 		appendPQExpBuffer(query, |  | ||||||
| 						  "SELECT tableoid, oid, proname, prolang, " |  | ||||||
| 						  "pronargs, proargtypes, prorettype, proacl, " |  | ||||||
| 						  "pronamespace, " |  | ||||||
| 						  "NULL::text AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname " |  | ||||||
| 						  "FROM pg_proc p " |  | ||||||
| 						  "WHERE NOT proisagg AND (" |  | ||||||
| 						  "pronamespace != " |  | ||||||
| 						  "(SELECT oid FROM pg_namespace " |  | ||||||
| 						  "WHERE nspname = 'pg_catalog'))", |  | ||||||
| 						  username_subquery); |  | ||||||
| 	} |  | ||||||
| 	else if (fout->remoteVersion >= 70100) | 	else if (fout->remoteVersion >= 70100) | ||||||
| 	{ | 	{ | ||||||
| 		appendPQExpBuffer(query, | 		appendPQExpBuffer(query, | ||||||
| @@ -4203,7 +4164,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 						  "pronargs, proargtypes, prorettype, " | 						  "pronargs, proargtypes, prorettype, " | ||||||
| 						  "'{=X}' AS proacl, " | 						  "'{=X}' AS proacl, " | ||||||
| 						  "0::oid AS pronamespace, " | 						  "0::oid AS pronamespace, " | ||||||
| 						  "NULL::text AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname " | 						  "(%s proowner) AS rolname " | ||||||
| 						  "FROM pg_proc " | 						  "FROM pg_proc " | ||||||
| 						  "WHERE pg_proc.oid > '%u'::oid", | 						  "WHERE pg_proc.oid > '%u'::oid", | ||||||
| @@ -4220,7 +4180,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 						  "pronargs, proargtypes, prorettype, " | 						  "pronargs, proargtypes, prorettype, " | ||||||
| 						  "'{=X}' AS proacl, " | 						  "'{=X}' AS proacl, " | ||||||
| 						  "0::oid AS pronamespace, " | 						  "0::oid AS pronamespace, " | ||||||
| 						  "NULL::text AS proiargs," |  | ||||||
| 						  "(%s proowner) AS rolname " | 						  "(%s proowner) AS rolname " | ||||||
| 						  "FROM pg_proc " | 						  "FROM pg_proc " | ||||||
| 						  "where pg_proc.oid > '%u'::oid", | 						  "where pg_proc.oid > '%u'::oid", | ||||||
| @@ -4246,7 +4205,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 	i_proargtypes = PQfnumber(res, "proargtypes"); | 	i_proargtypes = PQfnumber(res, "proargtypes"); | ||||||
| 	i_prorettype = PQfnumber(res, "prorettype"); | 	i_prorettype = PQfnumber(res, "prorettype"); | ||||||
| 	i_proacl = PQfnumber(res, "proacl"); | 	i_proacl = PQfnumber(res, "proacl"); | ||||||
| 	i_proiargs = PQfnumber(res, "proiargs"); |  | ||||||
|  |  | ||||||
| 	for (i = 0; i < ntups; i++) | 	for (i = 0; i < ntups; i++) | ||||||
| 	{ | 	{ | ||||||
| @@ -4262,7 +4220,6 @@ getFuncs(Archive *fout, int *numFuncs) | |||||||
| 		finfo[i].rolname = pg_strdup(PQgetvalue(res, i, i_rolname)); | 		finfo[i].rolname = pg_strdup(PQgetvalue(res, i, i_rolname)); | ||||||
| 		finfo[i].lang = atooid(PQgetvalue(res, i, i_prolang)); | 		finfo[i].lang = atooid(PQgetvalue(res, i, i_prolang)); | ||||||
| 		finfo[i].prorettype = atooid(PQgetvalue(res, i, i_prorettype)); | 		finfo[i].prorettype = atooid(PQgetvalue(res, i, i_prorettype)); | ||||||
| 		finfo[i].proiargs = pg_strdup(PQgetvalue(res, i, i_proiargs)); |  | ||||||
| 		finfo[i].proacl = pg_strdup(PQgetvalue(res, i, i_proacl)); | 		finfo[i].proacl = pg_strdup(PQgetvalue(res, i, i_proacl)); | ||||||
| 		finfo[i].nargs = atoi(PQgetvalue(res, i, i_pronargs)); | 		finfo[i].nargs = atoi(PQgetvalue(res, i, i_pronargs)); | ||||||
| 		if (finfo[i].nargs == 0) | 		if (finfo[i].nargs == 0) | ||||||
|   | |||||||
| @@ -184,7 +184,6 @@ typedef struct _funcInfo | |||||||
| 	Oid		   *argtypes; | 	Oid		   *argtypes; | ||||||
| 	Oid			prorettype; | 	Oid			prorettype; | ||||||
| 	char	   *proacl; | 	char	   *proacl; | ||||||
| 	char	   *proiargs; |  | ||||||
| } FuncInfo; | } FuncInfo; | ||||||
|  |  | ||||||
| /* AggInfo is a superset of FuncInfo */ | /* AggInfo is a superset of FuncInfo */ | ||||||
|   | |||||||
| @@ -287,13 +287,30 @@ DOTypeNameCompare(const void *p1, const void *p2) | |||||||
| 	{ | 	{ | ||||||
| 		FuncInfo   *fobj1 = *(FuncInfo *const *) p1; | 		FuncInfo   *fobj1 = *(FuncInfo *const *) p1; | ||||||
| 		FuncInfo   *fobj2 = *(FuncInfo *const *) p2; | 		FuncInfo   *fobj2 = *(FuncInfo *const *) p2; | ||||||
|  | 		int			i; | ||||||
|  |  | ||||||
| 		cmpval = fobj1->nargs - fobj2->nargs; | 		cmpval = fobj1->nargs - fobj2->nargs; | ||||||
| 		if (cmpval != 0) | 		if (cmpval != 0) | ||||||
| 			return cmpval; | 			return cmpval; | ||||||
| 		cmpval = strcmp(fobj1->proiargs, fobj2->proiargs); | 		for (i = 0; i < fobj1->nargs; i++) | ||||||
| 		if (cmpval != 0) | 		{ | ||||||
| 			return cmpval; | 			TypeInfo   *argtype1 = findTypeByOid(fobj1->argtypes[i]); | ||||||
|  | 			TypeInfo   *argtype2 = findTypeByOid(fobj2->argtypes[i]); | ||||||
|  |  | ||||||
|  | 			if (argtype1 && argtype2) | ||||||
|  | 			{ | ||||||
|  | 				if (argtype1->dobj.namespace && argtype2->dobj.namespace) | ||||||
|  | 				{ | ||||||
|  | 					cmpval = strcmp(argtype1->dobj.namespace->dobj.name, | ||||||
|  | 									argtype2->dobj.namespace->dobj.name); | ||||||
|  | 					if (cmpval != 0) | ||||||
|  | 						return cmpval; | ||||||
|  | 				} | ||||||
|  | 				cmpval = strcmp(argtype1->dobj.name, argtype2->dobj.name); | ||||||
|  | 				if (cmpval != 0) | ||||||
|  | 					return cmpval; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	else if (obj1->objType == DO_OPERATOR) | 	else if (obj1->objType == DO_OPERATOR) | ||||||
| 	{ | 	{ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user