mirror of
https://github.com/postgres/postgres.git
synced 2025-08-11 04:22:52 +03:00
Sort dump objects independent of OIDs, for the 7 holdout object types.
pg_dump sorts objects by their logical names, e.g. (nspname, relname,
tgname), before dependency-driven reordering. That removes one source
of logically-identical databases differing in their schema-only dumps.
In other words, it helps with schema diffing. The logical name sort
ignored essential sort keys for constraints, operators, PUBLICATION
... FOR TABLE, PUBLICATION ... FOR TABLES IN SCHEMA, operator classes,
and operator families. pg_dump's sort then depended on object OID,
yielding spurious schema diffs. After this change, OIDs affect dump
order only in the event of catalog corruption. While pg_dump also
wrongly ignored pg_collation.collencoding, CREATE COLLATION restrictions
have been keeping that imperceptible in practical use.
Use techniques like we use for object types already having full sort key
coverage. Where the pertinent queries weren't fetching the ignored sort
keys, this adds columns to those queries and stores those keys in memory
for the long term.
The ignorance of sort keys became more problematic when commit
172259afb5
added a schema diff test
sensitive to it. Buildfarm member hippopotamus witnessed that.
However, dump order stability isn't a new goal, and this might avoid
other dump comparison failures. Hence, back-patch to v13 (all supported
versions).
Reviewed-by: Robert Haas <robertmhaas@gmail.com>
Discussion: https://postgr.es/m/20250707192654.9e.nmisch@google.com
Backpatch-through: 13
This commit is contained in:
@@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
|
|
||||||
|
#include "catalog/pg_am_d.h"
|
||||||
#include "catalog/pg_class_d.h"
|
#include "catalog/pg_class_d.h"
|
||||||
#include "catalog/pg_collation_d.h"
|
#include "catalog/pg_collation_d.h"
|
||||||
#include "catalog/pg_extension_d.h"
|
#include "catalog/pg_extension_d.h"
|
||||||
@@ -944,6 +945,24 @@ findOprByOid(Oid oid)
|
|||||||
return (OprInfo *) dobj;
|
return (OprInfo *) dobj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* findAccessMethodByOid
|
||||||
|
* finds the DumpableObject for the access method with the given oid
|
||||||
|
* returns NULL if not found
|
||||||
|
*/
|
||||||
|
AccessMethodInfo *
|
||||||
|
findAccessMethodByOid(Oid oid)
|
||||||
|
{
|
||||||
|
CatalogId catId;
|
||||||
|
DumpableObject *dobj;
|
||||||
|
|
||||||
|
catId.tableoid = AccessMethodRelationId;
|
||||||
|
catId.oid = oid;
|
||||||
|
dobj = findObjectByCatalogId(catId);
|
||||||
|
Assert(dobj == NULL || dobj->objType == DO_ACCESS_METHOD);
|
||||||
|
return (AccessMethodInfo *) dobj;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* findCollationByOid
|
* findCollationByOid
|
||||||
* finds the DumpableObject for the collation with the given oid
|
* finds the DumpableObject for the collation with the given oid
|
||||||
|
@@ -2169,6 +2169,13 @@ selectDumpableProcLang(ProcLangInfo *plang, Archive *fout)
|
|||||||
static void
|
static void
|
||||||
selectDumpableAccessMethod(AccessMethodInfo *method, Archive *fout)
|
selectDumpableAccessMethod(AccessMethodInfo *method, Archive *fout)
|
||||||
{
|
{
|
||||||
|
/* see getAccessMethods() comment about v9.6. */
|
||||||
|
if (fout->remoteVersion < 90600)
|
||||||
|
{
|
||||||
|
method->dobj.dump = DUMP_COMPONENT_NONE;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (checkExtensionMembership(&method->dobj, fout))
|
if (checkExtensionMembership(&method->dobj, fout))
|
||||||
return; /* extension membership overrides all else */
|
return; /* extension membership overrides all else */
|
||||||
|
|
||||||
@@ -6183,6 +6190,8 @@ getOperators(Archive *fout)
|
|||||||
int i_oprnamespace;
|
int i_oprnamespace;
|
||||||
int i_oprowner;
|
int i_oprowner;
|
||||||
int i_oprkind;
|
int i_oprkind;
|
||||||
|
int i_oprleft;
|
||||||
|
int i_oprright;
|
||||||
int i_oprcode;
|
int i_oprcode;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -6194,6 +6203,8 @@ getOperators(Archive *fout)
|
|||||||
"oprnamespace, "
|
"oprnamespace, "
|
||||||
"oprowner, "
|
"oprowner, "
|
||||||
"oprkind, "
|
"oprkind, "
|
||||||
|
"oprleft, "
|
||||||
|
"oprright, "
|
||||||
"oprcode::oid AS oprcode "
|
"oprcode::oid AS oprcode "
|
||||||
"FROM pg_operator");
|
"FROM pg_operator");
|
||||||
|
|
||||||
@@ -6209,6 +6220,8 @@ getOperators(Archive *fout)
|
|||||||
i_oprnamespace = PQfnumber(res, "oprnamespace");
|
i_oprnamespace = PQfnumber(res, "oprnamespace");
|
||||||
i_oprowner = PQfnumber(res, "oprowner");
|
i_oprowner = PQfnumber(res, "oprowner");
|
||||||
i_oprkind = PQfnumber(res, "oprkind");
|
i_oprkind = PQfnumber(res, "oprkind");
|
||||||
|
i_oprleft = PQfnumber(res, "oprleft");
|
||||||
|
i_oprright = PQfnumber(res, "oprright");
|
||||||
i_oprcode = PQfnumber(res, "oprcode");
|
i_oprcode = PQfnumber(res, "oprcode");
|
||||||
|
|
||||||
for (i = 0; i < ntups; i++)
|
for (i = 0; i < ntups; i++)
|
||||||
@@ -6222,6 +6235,8 @@ getOperators(Archive *fout)
|
|||||||
findNamespace(atooid(PQgetvalue(res, i, i_oprnamespace)));
|
findNamespace(atooid(PQgetvalue(res, i, i_oprnamespace)));
|
||||||
oprinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_oprowner));
|
oprinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_oprowner));
|
||||||
oprinfo[i].oprkind = (PQgetvalue(res, i, i_oprkind))[0];
|
oprinfo[i].oprkind = (PQgetvalue(res, i, i_oprkind))[0];
|
||||||
|
oprinfo[i].oprleft = atooid(PQgetvalue(res, i, i_oprleft));
|
||||||
|
oprinfo[i].oprright = atooid(PQgetvalue(res, i, i_oprright));
|
||||||
oprinfo[i].oprcode = atooid(PQgetvalue(res, i, i_oprcode));
|
oprinfo[i].oprcode = atooid(PQgetvalue(res, i, i_oprcode));
|
||||||
|
|
||||||
/* Decide whether we want to dump it */
|
/* Decide whether we want to dump it */
|
||||||
@@ -6250,6 +6265,7 @@ getCollations(Archive *fout)
|
|||||||
int i_collname;
|
int i_collname;
|
||||||
int i_collnamespace;
|
int i_collnamespace;
|
||||||
int i_collowner;
|
int i_collowner;
|
||||||
|
int i_collencoding;
|
||||||
|
|
||||||
query = createPQExpBuffer();
|
query = createPQExpBuffer();
|
||||||
|
|
||||||
@@ -6260,7 +6276,8 @@ getCollations(Archive *fout)
|
|||||||
|
|
||||||
appendPQExpBufferStr(query, "SELECT tableoid, oid, collname, "
|
appendPQExpBufferStr(query, "SELECT tableoid, oid, collname, "
|
||||||
"collnamespace, "
|
"collnamespace, "
|
||||||
"collowner "
|
"collowner, "
|
||||||
|
"collencoding "
|
||||||
"FROM pg_collation");
|
"FROM pg_collation");
|
||||||
|
|
||||||
res = ExecuteSqlQuery(fout, query->data, PGRES_TUPLES_OK);
|
res = ExecuteSqlQuery(fout, query->data, PGRES_TUPLES_OK);
|
||||||
@@ -6274,6 +6291,7 @@ getCollations(Archive *fout)
|
|||||||
i_collname = PQfnumber(res, "collname");
|
i_collname = PQfnumber(res, "collname");
|
||||||
i_collnamespace = PQfnumber(res, "collnamespace");
|
i_collnamespace = PQfnumber(res, "collnamespace");
|
||||||
i_collowner = PQfnumber(res, "collowner");
|
i_collowner = PQfnumber(res, "collowner");
|
||||||
|
i_collencoding = PQfnumber(res, "collencoding");
|
||||||
|
|
||||||
for (i = 0; i < ntups; i++)
|
for (i = 0; i < ntups; i++)
|
||||||
{
|
{
|
||||||
@@ -6285,6 +6303,7 @@ getCollations(Archive *fout)
|
|||||||
collinfo[i].dobj.namespace =
|
collinfo[i].dobj.namespace =
|
||||||
findNamespace(atooid(PQgetvalue(res, i, i_collnamespace)));
|
findNamespace(atooid(PQgetvalue(res, i, i_collnamespace)));
|
||||||
collinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_collowner));
|
collinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_collowner));
|
||||||
|
collinfo[i].collencoding = atoi(PQgetvalue(res, i, i_collencoding));
|
||||||
|
|
||||||
/* Decide whether we want to dump it */
|
/* Decide whether we want to dump it */
|
||||||
selectDumpableObject(&(collinfo[i].dobj), fout);
|
selectDumpableObject(&(collinfo[i].dobj), fout);
|
||||||
@@ -6375,16 +6394,28 @@ getAccessMethods(Archive *fout)
|
|||||||
int i_amhandler;
|
int i_amhandler;
|
||||||
int i_amtype;
|
int i_amtype;
|
||||||
|
|
||||||
/* Before 9.6, there are no user-defined access methods */
|
|
||||||
if (fout->remoteVersion < 90600)
|
|
||||||
return;
|
|
||||||
|
|
||||||
query = createPQExpBuffer();
|
query = createPQExpBuffer();
|
||||||
|
|
||||||
/* Select all access methods from pg_am table */
|
/*
|
||||||
appendPQExpBufferStr(query, "SELECT tableoid, oid, amname, amtype, "
|
* Select all access methods from pg_am table. v9.6 introduced CREATE
|
||||||
"amhandler::pg_catalog.regproc AS amhandler "
|
* ACCESS METHOD, so earlier versions usually have only built-in access
|
||||||
"FROM pg_am");
|
* methods. v9.6 also changed the access method API, replacing dozens of
|
||||||
|
* pg_am columns with amhandler. Even if a user created an access method
|
||||||
|
* by "INSERT INTO pg_am", we have no way to translate pre-v9.6 pg_am
|
||||||
|
* columns to a v9.6+ CREATE ACCESS METHOD. Hence, before v9.6, read
|
||||||
|
* pg_am just to facilitate findAccessMethodByOid() providing the
|
||||||
|
* OID-to-name mapping.
|
||||||
|
*/
|
||||||
|
appendPQExpBufferStr(query, "SELECT tableoid, oid, amname, ");
|
||||||
|
if (fout->remoteVersion >= 90600)
|
||||||
|
appendPQExpBufferStr(query,
|
||||||
|
"amtype, "
|
||||||
|
"amhandler::pg_catalog.regproc AS amhandler ");
|
||||||
|
else
|
||||||
|
appendPQExpBufferStr(query,
|
||||||
|
"'i'::pg_catalog.\"char\" AS amtype, "
|
||||||
|
"'-'::pg_catalog.regproc AS amhandler ");
|
||||||
|
appendPQExpBufferStr(query, "FROM pg_am");
|
||||||
|
|
||||||
res = ExecuteSqlQuery(fout, query->data, PGRES_TUPLES_OK);
|
res = ExecuteSqlQuery(fout, query->data, PGRES_TUPLES_OK);
|
||||||
|
|
||||||
@@ -6433,6 +6464,7 @@ getOpclasses(Archive *fout)
|
|||||||
OpclassInfo *opcinfo;
|
OpclassInfo *opcinfo;
|
||||||
int i_tableoid;
|
int i_tableoid;
|
||||||
int i_oid;
|
int i_oid;
|
||||||
|
int i_opcmethod;
|
||||||
int i_opcname;
|
int i_opcname;
|
||||||
int i_opcnamespace;
|
int i_opcnamespace;
|
||||||
int i_opcowner;
|
int i_opcowner;
|
||||||
@@ -6442,7 +6474,7 @@ getOpclasses(Archive *fout)
|
|||||||
* system-defined opclasses at dump-out time.
|
* system-defined opclasses at dump-out time.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
appendPQExpBufferStr(query, "SELECT tableoid, oid, opcname, "
|
appendPQExpBufferStr(query, "SELECT tableoid, oid, opcmethod, opcname, "
|
||||||
"opcnamespace, "
|
"opcnamespace, "
|
||||||
"opcowner "
|
"opcowner "
|
||||||
"FROM pg_opclass");
|
"FROM pg_opclass");
|
||||||
@@ -6455,6 +6487,7 @@ getOpclasses(Archive *fout)
|
|||||||
|
|
||||||
i_tableoid = PQfnumber(res, "tableoid");
|
i_tableoid = PQfnumber(res, "tableoid");
|
||||||
i_oid = PQfnumber(res, "oid");
|
i_oid = PQfnumber(res, "oid");
|
||||||
|
i_opcmethod = PQfnumber(res, "opcmethod");
|
||||||
i_opcname = PQfnumber(res, "opcname");
|
i_opcname = PQfnumber(res, "opcname");
|
||||||
i_opcnamespace = PQfnumber(res, "opcnamespace");
|
i_opcnamespace = PQfnumber(res, "opcnamespace");
|
||||||
i_opcowner = PQfnumber(res, "opcowner");
|
i_opcowner = PQfnumber(res, "opcowner");
|
||||||
@@ -6468,6 +6501,7 @@ getOpclasses(Archive *fout)
|
|||||||
opcinfo[i].dobj.name = pg_strdup(PQgetvalue(res, i, i_opcname));
|
opcinfo[i].dobj.name = pg_strdup(PQgetvalue(res, i, i_opcname));
|
||||||
opcinfo[i].dobj.namespace =
|
opcinfo[i].dobj.namespace =
|
||||||
findNamespace(atooid(PQgetvalue(res, i, i_opcnamespace)));
|
findNamespace(atooid(PQgetvalue(res, i, i_opcnamespace)));
|
||||||
|
opcinfo[i].opcmethod = atooid(PQgetvalue(res, i, i_opcmethod));
|
||||||
opcinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_opcowner));
|
opcinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_opcowner));
|
||||||
|
|
||||||
/* Decide whether we want to dump it */
|
/* Decide whether we want to dump it */
|
||||||
@@ -6493,6 +6527,7 @@ getOpfamilies(Archive *fout)
|
|||||||
OpfamilyInfo *opfinfo;
|
OpfamilyInfo *opfinfo;
|
||||||
int i_tableoid;
|
int i_tableoid;
|
||||||
int i_oid;
|
int i_oid;
|
||||||
|
int i_opfmethod;
|
||||||
int i_opfname;
|
int i_opfname;
|
||||||
int i_opfnamespace;
|
int i_opfnamespace;
|
||||||
int i_opfowner;
|
int i_opfowner;
|
||||||
@@ -6504,7 +6539,7 @@ getOpfamilies(Archive *fout)
|
|||||||
* system-defined opfamilies at dump-out time.
|
* system-defined opfamilies at dump-out time.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
appendPQExpBufferStr(query, "SELECT tableoid, oid, opfname, "
|
appendPQExpBufferStr(query, "SELECT tableoid, oid, opfmethod, opfname, "
|
||||||
"opfnamespace, "
|
"opfnamespace, "
|
||||||
"opfowner "
|
"opfowner "
|
||||||
"FROM pg_opfamily");
|
"FROM pg_opfamily");
|
||||||
@@ -6518,6 +6553,7 @@ getOpfamilies(Archive *fout)
|
|||||||
i_tableoid = PQfnumber(res, "tableoid");
|
i_tableoid = PQfnumber(res, "tableoid");
|
||||||
i_oid = PQfnumber(res, "oid");
|
i_oid = PQfnumber(res, "oid");
|
||||||
i_opfname = PQfnumber(res, "opfname");
|
i_opfname = PQfnumber(res, "opfname");
|
||||||
|
i_opfmethod = PQfnumber(res, "opfmethod");
|
||||||
i_opfnamespace = PQfnumber(res, "opfnamespace");
|
i_opfnamespace = PQfnumber(res, "opfnamespace");
|
||||||
i_opfowner = PQfnumber(res, "opfowner");
|
i_opfowner = PQfnumber(res, "opfowner");
|
||||||
|
|
||||||
@@ -6530,6 +6566,7 @@ getOpfamilies(Archive *fout)
|
|||||||
opfinfo[i].dobj.name = pg_strdup(PQgetvalue(res, i, i_opfname));
|
opfinfo[i].dobj.name = pg_strdup(PQgetvalue(res, i, i_opfname));
|
||||||
opfinfo[i].dobj.namespace =
|
opfinfo[i].dobj.namespace =
|
||||||
findNamespace(atooid(PQgetvalue(res, i, i_opfnamespace)));
|
findNamespace(atooid(PQgetvalue(res, i, i_opfnamespace)));
|
||||||
|
opfinfo[i].opfmethod = atooid(PQgetvalue(res, i, i_opfmethod));
|
||||||
opfinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_opfowner));
|
opfinfo[i].rolname = getRoleName(PQgetvalue(res, i, i_opfowner));
|
||||||
|
|
||||||
/* Decide whether we want to dump it */
|
/* Decide whether we want to dump it */
|
||||||
|
@@ -260,6 +260,8 @@ typedef struct _oprInfo
|
|||||||
DumpableObject dobj;
|
DumpableObject dobj;
|
||||||
const char *rolname;
|
const char *rolname;
|
||||||
char oprkind;
|
char oprkind;
|
||||||
|
Oid oprleft;
|
||||||
|
Oid oprright;
|
||||||
Oid oprcode;
|
Oid oprcode;
|
||||||
} OprInfo;
|
} OprInfo;
|
||||||
|
|
||||||
@@ -273,12 +275,14 @@ typedef struct _accessMethodInfo
|
|||||||
typedef struct _opclassInfo
|
typedef struct _opclassInfo
|
||||||
{
|
{
|
||||||
DumpableObject dobj;
|
DumpableObject dobj;
|
||||||
|
Oid opcmethod;
|
||||||
const char *rolname;
|
const char *rolname;
|
||||||
} OpclassInfo;
|
} OpclassInfo;
|
||||||
|
|
||||||
typedef struct _opfamilyInfo
|
typedef struct _opfamilyInfo
|
||||||
{
|
{
|
||||||
DumpableObject dobj;
|
DumpableObject dobj;
|
||||||
|
Oid opfmethod;
|
||||||
const char *rolname;
|
const char *rolname;
|
||||||
} OpfamilyInfo;
|
} OpfamilyInfo;
|
||||||
|
|
||||||
@@ -286,6 +290,7 @@ typedef struct _collInfo
|
|||||||
{
|
{
|
||||||
DumpableObject dobj;
|
DumpableObject dobj;
|
||||||
const char *rolname;
|
const char *rolname;
|
||||||
|
int collencoding;
|
||||||
} CollInfo;
|
} CollInfo;
|
||||||
|
|
||||||
typedef struct _convInfo
|
typedef struct _convInfo
|
||||||
@@ -759,6 +764,7 @@ extern TableInfo *findTableByOid(Oid oid);
|
|||||||
extern TypeInfo *findTypeByOid(Oid oid);
|
extern TypeInfo *findTypeByOid(Oid oid);
|
||||||
extern FuncInfo *findFuncByOid(Oid oid);
|
extern FuncInfo *findFuncByOid(Oid oid);
|
||||||
extern OprInfo *findOprByOid(Oid oid);
|
extern OprInfo *findOprByOid(Oid oid);
|
||||||
|
extern AccessMethodInfo *findAccessMethodByOid(Oid oid);
|
||||||
extern CollInfo *findCollationByOid(Oid oid);
|
extern CollInfo *findCollationByOid(Oid oid);
|
||||||
extern NamespaceInfo *findNamespaceByOid(Oid oid);
|
extern NamespaceInfo *findNamespaceByOid(Oid oid);
|
||||||
extern ExtensionInfo *findExtensionByOid(Oid oid);
|
extern ExtensionInfo *findExtensionByOid(Oid oid);
|
||||||
|
@@ -162,6 +162,8 @@ static DumpId postDataBoundId;
|
|||||||
|
|
||||||
|
|
||||||
static int DOTypeNameCompare(const void *p1, const void *p2);
|
static int DOTypeNameCompare(const void *p1, const void *p2);
|
||||||
|
static int pgTypeNameCompare(Oid typid1, Oid typid2);
|
||||||
|
static int accessMethodNameCompare(Oid am1, Oid am2);
|
||||||
static bool TopoSort(DumpableObject **objs,
|
static bool TopoSort(DumpableObject **objs,
|
||||||
int numObjs,
|
int numObjs,
|
||||||
DumpableObject **ordering,
|
DumpableObject **ordering,
|
||||||
@@ -228,12 +230,39 @@ DOTypeNameCompare(const void *p1, const void *p2)
|
|||||||
else if (obj2->namespace)
|
else if (obj2->namespace)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
/* Sort by name */
|
/*
|
||||||
|
* Sort by name. With a few exceptions, names here are single catalog
|
||||||
|
* columns. To get a fuller picture, grep pg_dump.c for "dobj.name = ".
|
||||||
|
* Names here don't match "Name:" in plain format output, which is a
|
||||||
|
* _tocEntry.tag. For example, DumpableObject.name of a constraint is
|
||||||
|
* pg_constraint.conname, but _tocEntry.tag of a constraint is relname and
|
||||||
|
* conname joined with a space.
|
||||||
|
*/
|
||||||
cmpval = strcmp(obj1->name, obj2->name);
|
cmpval = strcmp(obj1->name, obj2->name);
|
||||||
if (cmpval != 0)
|
if (cmpval != 0)
|
||||||
return cmpval;
|
return cmpval;
|
||||||
|
|
||||||
/* To have a stable sort order, break ties for some object types */
|
/*
|
||||||
|
* Sort by type. This helps types that share a type priority without
|
||||||
|
* sharing a unique name constraint, e.g. opclass and opfamily.
|
||||||
|
*/
|
||||||
|
cmpval = obj1->objType - obj2->objType;
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* To have a stable sort order, break ties for some object types. Most
|
||||||
|
* catalogs have a natural key, e.g. pg_proc_proname_args_nsp_index. Where
|
||||||
|
* the above "namespace" and "name" comparisons don't cover all natural
|
||||||
|
* key columns, compare the rest here.
|
||||||
|
*
|
||||||
|
* The natural key usually refers to other catalogs by surrogate keys.
|
||||||
|
* Hence, this translates each of those references to the natural key of
|
||||||
|
* the referenced catalog. That may descend through multiple levels of
|
||||||
|
* catalog references. For example, to sort by pg_proc.proargtypes,
|
||||||
|
* descend to each pg_type and then further to its pg_namespace, for an
|
||||||
|
* overall sort by (nspname, typname).
|
||||||
|
*/
|
||||||
if (obj1->objType == DO_FUNC || obj1->objType == DO_AGG)
|
if (obj1->objType == DO_FUNC || obj1->objType == DO_AGG)
|
||||||
{
|
{
|
||||||
FuncInfo *fobj1 = *(FuncInfo *const *) p1;
|
FuncInfo *fobj1 = *(FuncInfo *const *) p1;
|
||||||
@@ -246,22 +275,10 @@ DOTypeNameCompare(const void *p1, const void *p2)
|
|||||||
return cmpval;
|
return cmpval;
|
||||||
for (i = 0; i < fobj1->nargs; i++)
|
for (i = 0; i < fobj1->nargs; i++)
|
||||||
{
|
{
|
||||||
TypeInfo *argtype1 = findTypeByOid(fobj1->argtypes[i]);
|
cmpval = pgTypeNameCompare(fobj1->argtypes[i],
|
||||||
TypeInfo *argtype2 = findTypeByOid(fobj2->argtypes[i]);
|
fobj2->argtypes[i]);
|
||||||
|
if (cmpval != 0)
|
||||||
if (argtype1 && argtype2)
|
return cmpval;
|
||||||
{
|
|
||||||
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)
|
||||||
@@ -273,6 +290,57 @@ DOTypeNameCompare(const void *p1, const void *p2)
|
|||||||
cmpval = (oobj2->oprkind - oobj1->oprkind);
|
cmpval = (oobj2->oprkind - oobj1->oprkind);
|
||||||
if (cmpval != 0)
|
if (cmpval != 0)
|
||||||
return cmpval;
|
return cmpval;
|
||||||
|
/* Within an oprkind, sort by argument type names */
|
||||||
|
cmpval = pgTypeNameCompare(oobj1->oprleft, oobj2->oprleft);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
cmpval = pgTypeNameCompare(oobj1->oprright, oobj2->oprright);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
else if (obj1->objType == DO_OPCLASS)
|
||||||
|
{
|
||||||
|
OpclassInfo *opcobj1 = *(OpclassInfo *const *) p1;
|
||||||
|
OpclassInfo *opcobj2 = *(OpclassInfo *const *) p2;
|
||||||
|
|
||||||
|
/* Sort by access method name, per pg_opclass_am_name_nsp_index */
|
||||||
|
cmpval = accessMethodNameCompare(opcobj1->opcmethod,
|
||||||
|
opcobj2->opcmethod);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
else if (obj1->objType == DO_OPFAMILY)
|
||||||
|
{
|
||||||
|
OpfamilyInfo *opfobj1 = *(OpfamilyInfo *const *) p1;
|
||||||
|
OpfamilyInfo *opfobj2 = *(OpfamilyInfo *const *) p2;
|
||||||
|
|
||||||
|
/* Sort by access method name, per pg_opfamily_am_name_nsp_index */
|
||||||
|
cmpval = accessMethodNameCompare(opfobj1->opfmethod,
|
||||||
|
opfobj2->opfmethod);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
else if (obj1->objType == DO_COLLATION)
|
||||||
|
{
|
||||||
|
CollInfo *cobj1 = *(CollInfo *const *) p1;
|
||||||
|
CollInfo *cobj2 = *(CollInfo *const *) p2;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Sort by encoding, per pg_collation_name_enc_nsp_index. Technically,
|
||||||
|
* this is not necessary, because wherever this changes dump order,
|
||||||
|
* restoring the dump fails anyway. CREATE COLLATION can't create a
|
||||||
|
* tie for this to break, because it imposes restrictions to make
|
||||||
|
* (nspname, collname) uniquely identify a collation within a given
|
||||||
|
* DatabaseEncoding. While pg_import_system_collations() can create a
|
||||||
|
* tie, pg_dump+restore fails after
|
||||||
|
* pg_import_system_collations('my_schema') does so. However, there's
|
||||||
|
* little to gain by ignoring one natural key column on the basis of
|
||||||
|
* those limitations elsewhere, so respect the full natural key like
|
||||||
|
* we do for other object types.
|
||||||
|
*/
|
||||||
|
cmpval = cobj1->collencoding - cobj2->collencoding;
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
}
|
}
|
||||||
else if (obj1->objType == DO_ATTRDEF)
|
else if (obj1->objType == DO_ATTRDEF)
|
||||||
{
|
{
|
||||||
@@ -317,11 +385,143 @@ DOTypeNameCompare(const void *p1, const void *p2)
|
|||||||
if (cmpval != 0)
|
if (cmpval != 0)
|
||||||
return cmpval;
|
return cmpval;
|
||||||
}
|
}
|
||||||
|
else if (obj1->objType == DO_CONSTRAINT)
|
||||||
|
{
|
||||||
|
ConstraintInfo *robj1 = *(ConstraintInfo *const *) p1;
|
||||||
|
ConstraintInfo *robj2 = *(ConstraintInfo *const *) p2;
|
||||||
|
|
||||||
/* Usually shouldn't get here, but if we do, sort by OID */
|
/*
|
||||||
|
* Sort domain constraints before table constraints, for consistency
|
||||||
|
* with our decision to sort CREATE DOMAIN before CREATE TABLE.
|
||||||
|
*/
|
||||||
|
if (robj1->condomain)
|
||||||
|
{
|
||||||
|
if (robj2->condomain)
|
||||||
|
{
|
||||||
|
/* Sort by domain name (domain namespace was considered) */
|
||||||
|
cmpval = strcmp(robj1->condomain->dobj.name,
|
||||||
|
robj2->condomain->dobj.name);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return PRIO_TYPE - PRIO_TABLE;
|
||||||
|
}
|
||||||
|
else if (robj2->condomain)
|
||||||
|
return PRIO_TABLE - PRIO_TYPE;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Sort by table name (table namespace was considered already) */
|
||||||
|
cmpval = strcmp(robj1->contable->dobj.name,
|
||||||
|
robj2->contable->dobj.name);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (obj1->objType == DO_PUBLICATION_REL)
|
||||||
|
{
|
||||||
|
PublicationRelInfo *probj1 = *(PublicationRelInfo *const *) p1;
|
||||||
|
PublicationRelInfo *probj2 = *(PublicationRelInfo *const *) p2;
|
||||||
|
|
||||||
|
/* Sort by publication name, since (namespace, name) match the rel */
|
||||||
|
cmpval = strcmp(probj1->publication->dobj.name,
|
||||||
|
probj2->publication->dobj.name);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
else if (obj1->objType == DO_PUBLICATION_TABLE_IN_SCHEMA)
|
||||||
|
{
|
||||||
|
PublicationSchemaInfo *psobj1 = *(PublicationSchemaInfo *const *) p1;
|
||||||
|
PublicationSchemaInfo *psobj2 = *(PublicationSchemaInfo *const *) p2;
|
||||||
|
|
||||||
|
/* Sort by publication name, since ->name is just nspname */
|
||||||
|
cmpval = strcmp(psobj1->publication->dobj.name,
|
||||||
|
psobj2->publication->dobj.name);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Shouldn't get here except after catalog corruption, but if we do, sort
|
||||||
|
* by OID. This may make logically-identical databases differ in the
|
||||||
|
* order of objects in dump output. Users will get spurious schema diffs.
|
||||||
|
* Expect flaky failures of 002_pg_upgrade.pl test 'dump outputs from
|
||||||
|
* original and restored regression databases match' if the regression
|
||||||
|
* database contains objects allowing that test to reach here. That's a
|
||||||
|
* consequence of the test using "pg_restore -j", which doesn't fully
|
||||||
|
* constrain OID assignment order.
|
||||||
|
*/
|
||||||
|
Assert(false);
|
||||||
return oidcmp(obj1->catId.oid, obj2->catId.oid);
|
return oidcmp(obj1->catId.oid, obj2->catId.oid);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Compare two OID-identified pg_type values by nspname, then by typname. */
|
||||||
|
static int
|
||||||
|
pgTypeNameCompare(Oid typid1, Oid typid2)
|
||||||
|
{
|
||||||
|
TypeInfo *typobj1;
|
||||||
|
TypeInfo *typobj2;
|
||||||
|
int cmpval;
|
||||||
|
|
||||||
|
if (typid1 == typid2)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
typobj1 = findTypeByOid(typid1);
|
||||||
|
typobj2 = findTypeByOid(typid2);
|
||||||
|
|
||||||
|
if (!typobj1 || !typobj2)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* getTypes() didn't find some OID. Assume catalog corruption, e.g.
|
||||||
|
* an oprright value without the corresponding OID in a pg_type row.
|
||||||
|
* Report as "equal", so the caller uses the next available basis for
|
||||||
|
* comparison, e.g. the next function argument.
|
||||||
|
*
|
||||||
|
* Unary operators have InvalidOid in oprleft (if oprkind='r') or in
|
||||||
|
* oprright (if oprkind='l'). Caller already sorted by oprkind,
|
||||||
|
* calling us only for like-kind operators. Hence, "typid1 == typid2"
|
||||||
|
* took care of InvalidOid. (v14 removed postfix operator support.
|
||||||
|
* Hence, when dumping from v14+, only oprleft can be InvalidOid.)
|
||||||
|
*/
|
||||||
|
Assert(false);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!typobj1->dobj.namespace || !typobj2->dobj.namespace)
|
||||||
|
Assert(false); /* catalog corruption */
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cmpval = strcmp(typobj1->dobj.namespace->dobj.name,
|
||||||
|
typobj2->dobj.namespace->dobj.name);
|
||||||
|
if (cmpval != 0)
|
||||||
|
return cmpval;
|
||||||
|
}
|
||||||
|
return strcmp(typobj1->dobj.name, typobj2->dobj.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Compare two OID-identified pg_am values by amname. */
|
||||||
|
static int
|
||||||
|
accessMethodNameCompare(Oid am1, Oid am2)
|
||||||
|
{
|
||||||
|
AccessMethodInfo *amobj1;
|
||||||
|
AccessMethodInfo *amobj2;
|
||||||
|
|
||||||
|
if (am1 == am2)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
amobj1 = findAccessMethodByOid(am1);
|
||||||
|
amobj2 = findAccessMethodByOid(am2);
|
||||||
|
|
||||||
|
if (!amobj1 || !amobj2)
|
||||||
|
{
|
||||||
|
/* catalog corruption: handle like pgTypeNameCompare() does */
|
||||||
|
Assert(false);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return strcmp(amobj1->dobj.name, amobj2->dobj.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Sort the given objects into a safe dump order using dependency
|
* Sort the given objects into a safe dump order using dependency
|
||||||
|
@@ -1934,3 +1934,24 @@ RESET client_min_messages;
|
|||||||
RESET SESSION AUTHORIZATION;
|
RESET SESSION AUTHORIZATION;
|
||||||
DROP ROLE regress_publication_user, regress_publication_user2;
|
DROP ROLE regress_publication_user, regress_publication_user2;
|
||||||
DROP ROLE regress_publication_user_dummy;
|
DROP ROLE regress_publication_user_dummy;
|
||||||
|
-- stage objects for pg_dump tests
|
||||||
|
CREATE SCHEMA pubme CREATE TABLE t0 (c int, d int) CREATE TABLE t1 (c int);
|
||||||
|
CREATE SCHEMA pubme2 CREATE TABLE t0 (c int, d int);
|
||||||
|
SET client_min_messages = 'ERROR';
|
||||||
|
CREATE PUBLICATION dump_pub_qual_1ct FOR
|
||||||
|
TABLE ONLY pubme.t0 (c, d) WHERE (c > 0);
|
||||||
|
CREATE PUBLICATION dump_pub_qual_2ct FOR
|
||||||
|
TABLE ONLY pubme.t0 (c) WHERE (c > 0),
|
||||||
|
TABLE ONLY pubme.t1 (c);
|
||||||
|
CREATE PUBLICATION dump_pub_nsp_1ct FOR
|
||||||
|
TABLES IN SCHEMA pubme;
|
||||||
|
CREATE PUBLICATION dump_pub_nsp_2ct FOR
|
||||||
|
TABLES IN SCHEMA pubme,
|
||||||
|
TABLES IN SCHEMA pubme2;
|
||||||
|
CREATE PUBLICATION dump_pub_all FOR
|
||||||
|
TABLE ONLY pubme.t0,
|
||||||
|
TABLE ONLY pubme.t1 WHERE (c < 0),
|
||||||
|
TABLES IN SCHEMA pubme,
|
||||||
|
TABLES IN SCHEMA pubme2
|
||||||
|
WITH (publish_via_partition_root = true);
|
||||||
|
RESET client_min_messages;
|
||||||
|
@@ -1229,3 +1229,25 @@ RESET client_min_messages;
|
|||||||
RESET SESSION AUTHORIZATION;
|
RESET SESSION AUTHORIZATION;
|
||||||
DROP ROLE regress_publication_user, regress_publication_user2;
|
DROP ROLE regress_publication_user, regress_publication_user2;
|
||||||
DROP ROLE regress_publication_user_dummy;
|
DROP ROLE regress_publication_user_dummy;
|
||||||
|
|
||||||
|
-- stage objects for pg_dump tests
|
||||||
|
CREATE SCHEMA pubme CREATE TABLE t0 (c int, d int) CREATE TABLE t1 (c int);
|
||||||
|
CREATE SCHEMA pubme2 CREATE TABLE t0 (c int, d int);
|
||||||
|
SET client_min_messages = 'ERROR';
|
||||||
|
CREATE PUBLICATION dump_pub_qual_1ct FOR
|
||||||
|
TABLE ONLY pubme.t0 (c, d) WHERE (c > 0);
|
||||||
|
CREATE PUBLICATION dump_pub_qual_2ct FOR
|
||||||
|
TABLE ONLY pubme.t0 (c) WHERE (c > 0),
|
||||||
|
TABLE ONLY pubme.t1 (c);
|
||||||
|
CREATE PUBLICATION dump_pub_nsp_1ct FOR
|
||||||
|
TABLES IN SCHEMA pubme;
|
||||||
|
CREATE PUBLICATION dump_pub_nsp_2ct FOR
|
||||||
|
TABLES IN SCHEMA pubme,
|
||||||
|
TABLES IN SCHEMA pubme2;
|
||||||
|
CREATE PUBLICATION dump_pub_all FOR
|
||||||
|
TABLE ONLY pubme.t0,
|
||||||
|
TABLE ONLY pubme.t1 WHERE (c < 0),
|
||||||
|
TABLES IN SCHEMA pubme,
|
||||||
|
TABLES IN SCHEMA pubme2
|
||||||
|
WITH (publish_via_partition_root = true);
|
||||||
|
RESET client_min_messages;
|
||||||
|
Reference in New Issue
Block a user