Commit 3666260f authored by Tom Lane's avatar Tom Lane

Fix old pg_dump oversight: default values for domains really need to be dumped

by decompiling the typdefaultbin expression, not just printing the typdefault
text which may be out-of-date or assume the wrong schema search path.  (It's
the same hazard as for adbin vs adsrc in column defaults.)  The catalogs.sgml
spec for pg_type implies that the correct procedure is to look to
typdefaultbin first and consider typdefault only if typdefaultbin is NULL.
I made dumping of both domains and base types do that, even though in the
current backend code typdefaultbin is always correct for domains and
typdefault for base types --- might as well try to future-proof it a little.
Per bug report from Alexander Galler.
parent af49a163
...@@ -12,7 +12,7 @@ ...@@ -12,7 +12,7 @@
* by PostgreSQL * by PostgreSQL
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/bin/pg_dump/pg_dump.c,v 1.429 2006/02/12 06:11:51 momjian Exp $ * $PostgreSQL: pgsql/src/bin/pg_dump/pg_dump.c,v 1.430 2006/02/21 18:01:32 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -4690,10 +4690,11 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4690,10 +4690,11 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
Oid typsendoid; Oid typsendoid;
Oid typanalyzeoid; Oid typanalyzeoid;
char *typdelim; char *typdelim;
char *typdefault;
char *typbyval; char *typbyval;
char *typalign; char *typalign;
char *typstorage; char *typstorage;
char *typdefault;
bool typdefault_is_literal = false;
/* Set proper schema search path so regproc references list correctly */ /* Set proper schema search path so regproc references list correctly */
selectSourceSchema(tinfo->dobj.namespace->dobj.name); selectSourceSchema(tinfo->dobj.namespace->dobj.name);
...@@ -4709,8 +4710,8 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4709,8 +4710,8 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
"typreceive::pg_catalog.oid as typreceiveoid, " "typreceive::pg_catalog.oid as typreceiveoid, "
"typsend::pg_catalog.oid as typsendoid, " "typsend::pg_catalog.oid as typsendoid, "
"typanalyze::pg_catalog.oid as typanalyzeoid, " "typanalyze::pg_catalog.oid as typanalyzeoid, "
"typdelim, typdefault, typbyval, typalign, " "typdelim, typbyval, typalign, typstorage, "
"typstorage " "pg_catalog.pg_get_expr(typdefaultbin, 'pg_catalog.pg_type'::pg_catalog.regclass) as typdefaultbin, typdefault "
"FROM pg_catalog.pg_type " "FROM pg_catalog.pg_type "
"WHERE oid = '%u'::pg_catalog.oid", "WHERE oid = '%u'::pg_catalog.oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
...@@ -4725,8 +4726,8 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4725,8 +4726,8 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
"typreceive::pg_catalog.oid as typreceiveoid, " "typreceive::pg_catalog.oid as typreceiveoid, "
"typsend::pg_catalog.oid as typsendoid, " "typsend::pg_catalog.oid as typsendoid, "
"0 as typanalyzeoid, " "0 as typanalyzeoid, "
"typdelim, typdefault, typbyval, typalign, " "typdelim, typbyval, typalign, typstorage, "
"typstorage " "pg_catalog.pg_get_expr(typdefaultbin, 'pg_catalog.pg_type'::pg_catalog.regclass) as typdefaultbin, typdefault "
"FROM pg_catalog.pg_type " "FROM pg_catalog.pg_type "
"WHERE oid = '%u'::pg_catalog.oid", "WHERE oid = '%u'::pg_catalog.oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
...@@ -4741,13 +4742,13 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4741,13 +4742,13 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
"typoutput::pg_catalog.oid as typoutputoid, " "typoutput::pg_catalog.oid as typoutputoid, "
"0 as typreceiveoid, 0 as typsendoid, " "0 as typreceiveoid, 0 as typsendoid, "
"0 as typanalyzeoid, " "0 as typanalyzeoid, "
"typdelim, typdefault, typbyval, typalign, " "typdelim, typbyval, typalign, typstorage, "
"typstorage " "pg_catalog.pg_get_expr(typdefaultbin, 'pg_catalog.pg_type'::pg_catalog.regclass) as typdefaultbin, typdefault "
"FROM pg_catalog.pg_type " "FROM pg_catalog.pg_type "
"WHERE oid = '%u'::pg_catalog.oid", "WHERE oid = '%u'::pg_catalog.oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
} }
else if (fout->remoteVersion >= 70100) else if (fout->remoteVersion >= 70200)
{ {
/* /*
* Note: although pre-7.3 catalogs contain typreceive and typsend, * Note: although pre-7.3 catalogs contain typreceive and typsend,
...@@ -4761,8 +4762,28 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4761,8 +4762,28 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
"typoutput::oid as typoutputoid, " "typoutput::oid as typoutputoid, "
"0 as typreceiveoid, 0 as typsendoid, " "0 as typreceiveoid, 0 as typsendoid, "
"0 as typanalyzeoid, " "0 as typanalyzeoid, "
"typdelim, typdefault, typbyval, typalign, " "typdelim, typbyval, typalign, typstorage, "
"typstorage " "NULL as typdefaultbin, typdefault "
"FROM pg_type "
"WHERE oid = '%u'::oid",
tinfo->dobj.catId.oid);
}
else if (fout->remoteVersion >= 70100)
{
/*
* Ignore pre-7.2 typdefault; the field exists but has an unusable
* representation.
*/
appendPQExpBuffer(query, "SELECT typlen, "
"typinput, typoutput, "
"'-' as typreceive, '-' as typsend, "
"'-' as typanalyze, "
"typinput::oid as typinputoid, "
"typoutput::oid as typoutputoid, "
"0 as typreceiveoid, 0 as typsendoid, "
"0 as typanalyzeoid, "
"typdelim, typbyval, typalign, typstorage, "
"NULL as typdefaultbin, NULL as typdefault "
"FROM pg_type " "FROM pg_type "
"WHERE oid = '%u'::oid", "WHERE oid = '%u'::oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
...@@ -4777,8 +4798,9 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4777,8 +4798,9 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
"typoutput::oid as typoutputoid, " "typoutput::oid as typoutputoid, "
"0 as typreceiveoid, 0 as typsendoid, " "0 as typreceiveoid, 0 as typsendoid, "
"0 as typanalyzeoid, " "0 as typanalyzeoid, "
"typdelim, typdefault, typbyval, typalign, " "typdelim, typbyval, typalign, "
"'p'::char as typstorage " "'p'::char as typstorage, "
"NULL as typdefaultbin, NULL as typdefault "
"FROM pg_type " "FROM pg_type "
"WHERE oid = '%u'::oid", "WHERE oid = '%u'::oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
...@@ -4808,13 +4830,18 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4808,13 +4830,18 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
typsendoid = atooid(PQgetvalue(res, 0, PQfnumber(res, "typsendoid"))); typsendoid = atooid(PQgetvalue(res, 0, PQfnumber(res, "typsendoid")));
typanalyzeoid = atooid(PQgetvalue(res, 0, PQfnumber(res, "typanalyzeoid"))); typanalyzeoid = atooid(PQgetvalue(res, 0, PQfnumber(res, "typanalyzeoid")));
typdelim = PQgetvalue(res, 0, PQfnumber(res, "typdelim")); typdelim = PQgetvalue(res, 0, PQfnumber(res, "typdelim"));
if (PQgetisnull(res, 0, PQfnumber(res, "typdefault")))
typdefault = NULL;
else
typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefault"));
typbyval = PQgetvalue(res, 0, PQfnumber(res, "typbyval")); typbyval = PQgetvalue(res, 0, PQfnumber(res, "typbyval"));
typalign = PQgetvalue(res, 0, PQfnumber(res, "typalign")); typalign = PQgetvalue(res, 0, PQfnumber(res, "typalign"));
typstorage = PQgetvalue(res, 0, PQfnumber(res, "typstorage")); typstorage = PQgetvalue(res, 0, PQfnumber(res, "typstorage"));
if (!PQgetisnull(res, 0, PQfnumber(res, "typdefaultbin")))
typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefaultbin"));
else if (!PQgetisnull(res, 0, PQfnumber(res, "typdefault")))
{
typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefault"));
typdefault_is_literal = true; /* it needs quotes */
}
else
typdefault = NULL;
/* /*
* DROP must be fully qualified in case same name appears in pg_catalog * DROP must be fully qualified in case same name appears in pg_catalog
...@@ -4854,7 +4881,10 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo) ...@@ -4854,7 +4881,10 @@ dumpBaseType(Archive *fout, TypeInfo *tinfo)
if (typdefault != NULL) if (typdefault != NULL)
{ {
appendPQExpBuffer(q, ",\n DEFAULT = "); appendPQExpBuffer(q, ",\n DEFAULT = ");
appendStringLiteral(q, typdefault, true); if (typdefault_is_literal)
appendStringLiteral(q, typdefault, true);
else
appendPQExpBufferStr(q, typdefault);
} }
if (tinfo->isArray) if (tinfo->isArray)
...@@ -4936,6 +4966,7 @@ dumpDomain(Archive *fout, TypeInfo *tinfo) ...@@ -4936,6 +4966,7 @@ dumpDomain(Archive *fout, TypeInfo *tinfo)
char *typnotnull; char *typnotnull;
char *typdefn; char *typdefn;
char *typdefault; char *typdefault;
bool typdefault_is_literal = false;
/* Set proper schema search path so type references list correctly */ /* Set proper schema search path so type references list correctly */
selectSourceSchema(tinfo->dobj.namespace->dobj.name); selectSourceSchema(tinfo->dobj.namespace->dobj.name);
...@@ -4944,7 +4975,7 @@ dumpDomain(Archive *fout, TypeInfo *tinfo) ...@@ -4944,7 +4975,7 @@ dumpDomain(Archive *fout, TypeInfo *tinfo)
/* We assume here that remoteVersion must be at least 70300 */ /* We assume here that remoteVersion must be at least 70300 */
appendPQExpBuffer(query, "SELECT typnotnull, " appendPQExpBuffer(query, "SELECT typnotnull, "
"pg_catalog.format_type(typbasetype, typtypmod) as typdefn, " "pg_catalog.format_type(typbasetype, typtypmod) as typdefn, "
"typdefault " "pg_catalog.pg_get_expr(typdefaultbin, 'pg_catalog.pg_type'::pg_catalog.regclass) as typdefaultbin, typdefault "
"FROM pg_catalog.pg_type " "FROM pg_catalog.pg_type "
"WHERE oid = '%u'::pg_catalog.oid", "WHERE oid = '%u'::pg_catalog.oid",
tinfo->dobj.catId.oid); tinfo->dobj.catId.oid);
...@@ -4963,10 +4994,15 @@ dumpDomain(Archive *fout, TypeInfo *tinfo) ...@@ -4963,10 +4994,15 @@ dumpDomain(Archive *fout, TypeInfo *tinfo)
typnotnull = PQgetvalue(res, 0, PQfnumber(res, "typnotnull")); typnotnull = PQgetvalue(res, 0, PQfnumber(res, "typnotnull"));
typdefn = PQgetvalue(res, 0, PQfnumber(res, "typdefn")); typdefn = PQgetvalue(res, 0, PQfnumber(res, "typdefn"));
if (PQgetisnull(res, 0, PQfnumber(res, "typdefault"))) if (!PQgetisnull(res, 0, PQfnumber(res, "typdefaultbin")))
typdefault = NULL; typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefaultbin"));
else else if (!PQgetisnull(res, 0, PQfnumber(res, "typdefault")))
{
typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefault")); typdefault = PQgetvalue(res, 0, PQfnumber(res, "typdefault"));
typdefault_is_literal = true; /* it needs quotes */
}
else
typdefault = NULL;
appendPQExpBuffer(q, appendPQExpBuffer(q,
"CREATE DOMAIN %s AS %s", "CREATE DOMAIN %s AS %s",
...@@ -4976,8 +5012,14 @@ dumpDomain(Archive *fout, TypeInfo *tinfo) ...@@ -4976,8 +5012,14 @@ dumpDomain(Archive *fout, TypeInfo *tinfo)
if (typnotnull[0] == 't') if (typnotnull[0] == 't')
appendPQExpBuffer(q, " NOT NULL"); appendPQExpBuffer(q, " NOT NULL");
if (typdefault) if (typdefault != NULL)
appendPQExpBuffer(q, " DEFAULT %s", typdefault); {
appendPQExpBuffer(q, " DEFAULT ");
if (typdefault_is_literal)
appendStringLiteral(q, typdefault, true);
else
appendPQExpBufferStr(q, typdefault);
}
PQclear(res); PQclear(res);
......
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