Make getObjectDescription robust against dangling amproc type links.
authorTom Lane
Sat, 7 Dec 2024 19:28:16 +0000 (14:28 -0500)
committerTom Lane
Sat, 7 Dec 2024 19:28:16 +0000 (14:28 -0500)
Yoran Heling reported a case where a data type could be dropped
while references to its OID remain behind in pg_amproc.  This
causes getObjectDescription to fail, which blocks dropping the
operator family (since our DROP code likes to construct descriptions
of everything it's dropping).  The proper fix for this requires
adding more pg_depend entries.  But to allow DROP to go through with
already-corrupt catalogs, tweak getObjectDescription to print "???"
for the type instead of failing when it processes such an entry.

I changed the logic for pg_amop similarly, for consistency,
although it is not known that the problem can manifest in pg_amop.

Per report from Yoran Heling.  Back-patch to all supported
branches (although the problem may be unreachable in v13).

Discussion: https://postgr.es/m/Z1MVCOh1hprjK5Sf@gmai021

src/backend/catalog/objectaddress.c

index 91c722adf628cccded072a923c76a7012b7e7008..738ce65372d40727ba176fff10a38168681fa67e 100644 (file)
@@ -3052,6 +3052,12 @@ getObjectDescription(const ObjectAddress *object)
                initStringInfo(&opfam);
                getOpFamilyDescription(&opfam, amopForm->amopfamily);
 
+               /*
+                * We use FORMAT_TYPE_ALLOW_INVALID here so as not to fail
+                * completely if the type links are dangling, which is a form
+                * of catalog corruption that could occur due to old bugs.
+                */
+
                /*------
                   translator: %d is the operator strategy (a number), the
                   first two %s's are data type names, the third %s is the
@@ -3059,8 +3065,10 @@ getObjectDescription(const ObjectAddress *object)
                   textual form of the operator with arguments.  */
                appendStringInfo(&buffer, _("operator %d (%s, %s) of %s: %s"),
                                 amopForm->amopstrategy,
-                                format_type_be(amopForm->amoplefttype),
-                                format_type_be(amopForm->amoprighttype),
+                                format_type_extended(amopForm->amoplefttype,
+                                                     -1, FORMAT_TYPE_ALLOW_INVALID),
+                                format_type_extended(amopForm->amoprighttype,
+                                                     -1, FORMAT_TYPE_ALLOW_INVALID),
                                 opfam.data,
                                 format_operator(amopForm->amopopr));
 
@@ -3102,6 +3110,12 @@ getObjectDescription(const ObjectAddress *object)
                initStringInfo(&opfam);
                getOpFamilyDescription(&opfam, amprocForm->amprocfamily);
 
+               /*
+                * We use FORMAT_TYPE_ALLOW_INVALID here so as not to fail
+                * completely if the type links are dangling, which is a form
+                * of catalog corruption that could occur due to old bugs.
+                */
+
                /*------
                   translator: %d is the function number, the first two %s's
                   are data type names, the third %s is the description of the
@@ -3109,8 +3123,10 @@ getObjectDescription(const ObjectAddress *object)
                   function with arguments.  */
                appendStringInfo(&buffer, _("function %d (%s, %s) of %s: %s"),
                                 amprocForm->amprocnum,
-                                format_type_be(amprocForm->amproclefttype),
-                                format_type_be(amprocForm->amprocrighttype),
+                                format_type_extended(amprocForm->amproclefttype,
+                                                     -1, FORMAT_TYPE_ALLOW_INVALID),
+                                format_type_extended(amprocForm->amprocrighttype,
+                                                     -1, FORMAT_TYPE_ALLOW_INVALID),
                                 opfam.data,
                                 format_procedure(amprocForm->amproc));