]> granicus.if.org Git - postgresql/commitdiff
Fix droppability of constraints upon partition detach
authorAlvaro Herrera <alvherre@alvh.no-ip.org>
Thu, 24 Jan 2019 17:09:56 +0000 (14:09 -0300)
committerAlvaro Herrera <alvherre@alvh.no-ip.org>
Thu, 24 Jan 2019 17:09:56 +0000 (14:09 -0300)
We were failing to set conislocal correctly for constraints in
partitions after partition detach, leading to those constraints becoming
undroppable.  Fix by setting the flag correctly.  Existing databases
might contain constraints with the conislocal wrongly set to false, for
partitions that were detached; this situation should be fixable by
applying an UPDATE on pg_constraint to set conislocal true.  This
problem should otherwise be innocuous and should disappear across a
dump/restore or pg_upgrade.

Secondarily, when constraint drop was attempted in a partitioned table,
ATExecDropConstraint would try to recurse to partitions after doing
performDeletion() of the constraint in the partitioned table itself; but
since the constraint in the partitions are dropped by the initial call
of performDeletion() (because of following dependencies), the recursion
step would fail since it would not find the constraint, causing the
whole operation to fail.  Fix by preventing recursion.

Reported-by: Amit Langote
Diagnosed-by: Amit Langote
Author: Amit Langote, Álvaro Herrera
Discussion: https://postgr.es/m/f2b8ead5-4131-d5a8-8016-2ea0a31250af@lab.ntt.co.jp

src/backend/catalog/pg_constraint.c
src/backend/commands/tablecmds.c
src/test/regress/expected/foreign_key.out
src/test/regress/sql/foreign_key.sql

index 446b54b9ffa3a75e5caa87f878cd0dc34053a50c..698b493fc409e410b7ad534d32a5f30c0804d7b4 100644 (file)
@@ -783,6 +783,12 @@ ConstraintSetParentConstraint(Oid childConstrId, Oid parentConstrId)
        constrForm = (Form_pg_constraint) GETSTRUCT(newtup);
        if (OidIsValid(parentConstrId))
        {
+               /* don't allow setting parent for a constraint that already has one */
+               Assert(constrForm->coninhcount == 0);
+               if (constrForm->conparentid != InvalidOid)
+                       elog(ERROR, "constraint %u already has a parent constraint",
+                                childConstrId);
+
                constrForm->conislocal = false;
                constrForm->coninhcount++;
                constrForm->conparentid = parentConstrId;
@@ -797,10 +803,12 @@ ConstraintSetParentConstraint(Oid childConstrId, Oid parentConstrId)
        else
        {
                constrForm->coninhcount--;
-               if (constrForm->coninhcount <= 0)
-                       constrForm->conislocal = true;
+               constrForm->conislocal = true;
                constrForm->conparentid = InvalidOid;
 
+               /* Make sure there's no further inheritance. */
+               Assert(constrForm->coninhcount == 0);
+
                deleteDependencyRecordsForClass(ConstraintRelationId, childConstrId,
                                                                                ConstraintRelationId,
                                                                                DEPENDENCY_INTERNAL_AUTO);
index 887c19c3eff9c79916d579d3ce09454678cfc0ab..e010586dd67e8c376336923e9148295999ca3545 100644 (file)
@@ -7243,6 +7243,7 @@ ATAddForeignKeyConstraint(List **wqueue, AlteredTableInfo *tab, Relation rel,
        Oid                     pfeqoperators[INDEX_MAX_KEYS];
        Oid                     ppeqoperators[INDEX_MAX_KEYS];
        Oid                     ffeqoperators[INDEX_MAX_KEYS];
+       bool            connoinherit;
        int                     i;
        int                     numfks,
                                numpks;
@@ -7586,6 +7587,12 @@ ATAddForeignKeyConstraint(List **wqueue, AlteredTableInfo *tab, Relation rel,
                ffeqoperators[i] = ffeqop;
        }
 
+       /*
+        * FKs always inherit for partitioned tables, and never for legacy
+        * inheritance.
+        */
+       connoinherit = rel->rd_rel->relkind != RELKIND_PARTITIONED_TABLE;
+
        /*
         * Record the FK constraint in pg_constraint.
         */
@@ -7616,7 +7623,7 @@ ATAddForeignKeyConstraint(List **wqueue, AlteredTableInfo *tab, Relation rel,
                                                                          NULL,
                                                                          true, /* islocal */
                                                                          0,    /* inhcount */
-                                                                         true, /* isnoinherit */
+                                                                         connoinherit, /* conNoInherit */
                                                                          false);       /* is_internal */
        ObjectAddressSet(address, ConstraintRelationId, constrOid);
 
@@ -7937,7 +7944,7 @@ CloneFkReferencing(Relation pg_constraint, Relation parentRel,
                        }
 
                        systable_endscan(scan);
-                       heap_close(trigrel, RowExclusiveLock);
+                       table_close(trigrel, RowExclusiveLock);
 
                        ConstraintSetParentConstraint(fk->conoid, parentConstrOid);
                        CommandCounterIncrement();
@@ -9131,6 +9138,7 @@ ATExecDropConstraint(Relation rel, const char *constrName,
        HeapTuple       tuple;
        bool            found = false;
        bool            is_no_inherit_constraint = false;
+       char            contype;
 
        /* At top level, permission check was done in ATPrepCmd, else do it */
        if (recursing)
@@ -9171,6 +9179,7 @@ ATExecDropConstraint(Relation rel, const char *constrName,
                                                        constrName, RelationGetRelationName(rel))));
 
                is_no_inherit_constraint = con->connoinherit;
+               contype = con->contype;
 
                /*
                 * If it's a foreign-key constraint, we'd better lock the referenced
@@ -9179,7 +9188,7 @@ ATExecDropConstraint(Relation rel, const char *constrName,
                 * that has unfired events).  But we can/must skip that in the
                 * self-referential case.
                 */
-               if (con->contype == CONSTRAINT_FOREIGN &&
+               if (contype == CONSTRAINT_FOREIGN &&
                        con->confrelid != RelationGetRelid(rel))
                {
                        Relation        frel;
@@ -9223,6 +9232,17 @@ ATExecDropConstraint(Relation rel, const char *constrName,
                }
        }
 
+       /*
+        * For partitioned tables, non-CHECK inherited constraints are dropped via
+        * the dependency mechanism, so we're done here.
+        */
+       if (contype != CONSTRAINT_CHECK &&
+               rel->rd_rel->relkind == RELKIND_PARTITIONED_TABLE)
+       {
+               table_close(conrel, RowExclusiveLock);
+               return;
+       }
+
        /*
         * Propagate to children as appropriate.  Unlike most other ALTER
         * routines, we have to do this one level of recursion at a time; we can't
index 76040a76011223b4ce2e96fae16fcfeb78879050..bf2c91d9f0ea0a00b2d2cfb7812d7358a36c932a 100644 (file)
@@ -1945,7 +1945,23 @@ DETAIL:  Key (a)=(2) is not present in table "pkey".
 delete from fkpart1.pkey where a = 1;
 ERROR:  update or delete on table "pkey" violates foreign key constraint "fk_part_a_fkey" on table "fk_part_1"
 DETAIL:  Key (a)=(1) is still referenced from table "fk_part_1".
+-- verify that attaching and detaching partitions manipulates the inheritance
+-- properties of their FK constraints correctly
+create schema fkpart2
+  create table pkey (a int primary key)
+  create table fk_part (a int, constraint fkey foreign key (a) references fkpart2.pkey) partition by list (a)
+  create table fk_part_1 partition of fkpart2.fk_part for values in (1) partition by list (a)
+  create table fk_part_1_1 (a int, constraint my_fkey foreign key (a) references fkpart2.pkey);
+alter table fkpart2.fk_part_1 attach partition fkpart2.fk_part_1_1 for values in (1);
+alter table fkpart2.fk_part_1 drop constraint fkey;    -- should fail
+ERROR:  cannot drop inherited constraint "fkey" of relation "fk_part_1"
+alter table fkpart2.fk_part_1_1 drop constraint my_fkey;       -- should fail
+ERROR:  cannot drop inherited constraint "my_fkey" of relation "fk_part_1_1"
+alter table fkpart2.fk_part detach partition fkpart2.fk_part_1;
+alter table fkpart2.fk_part_1 drop constraint fkey;    -- ok
+alter table fkpart2.fk_part_1_1 drop constraint my_fkey;       -- doesn't exist
+ERROR:  constraint "my_fkey" of relation "fk_part_1_1" does not exist
 \set VERBOSITY terse   \\ -- suppress cascade details
-drop schema fkpart0, fkpart1 cascade;
-NOTICE:  drop cascades to 5 other objects
+drop schema fkpart0, fkpart1, fkpart2 cascade;
+NOTICE:  drop cascades to 8 other objects
 \set VERBOSITY default
index 9ed1166c6666b8832496ddee2206ea900f12345c..c8d1214d02c621fb39f8dcbb8d3a0c5ecdda7198 100644 (file)
@@ -1392,6 +1392,20 @@ create table fkpart1.fk_part_1_2 partition of fkpart1.fk_part_1 for values in (2
 insert into fkpart1.fk_part_1 values (2);      -- should fail
 delete from fkpart1.pkey where a = 1;
 
+-- verify that attaching and detaching partitions manipulates the inheritance
+-- properties of their FK constraints correctly
+create schema fkpart2
+  create table pkey (a int primary key)
+  create table fk_part (a int, constraint fkey foreign key (a) references fkpart2.pkey) partition by list (a)
+  create table fk_part_1 partition of fkpart2.fk_part for values in (1) partition by list (a)
+  create table fk_part_1_1 (a int, constraint my_fkey foreign key (a) references fkpart2.pkey);
+alter table fkpart2.fk_part_1 attach partition fkpart2.fk_part_1_1 for values in (1);
+alter table fkpart2.fk_part_1 drop constraint fkey;    -- should fail
+alter table fkpart2.fk_part_1_1 drop constraint my_fkey;       -- should fail
+alter table fkpart2.fk_part detach partition fkpart2.fk_part_1;
+alter table fkpart2.fk_part_1 drop constraint fkey;    -- ok
+alter table fkpart2.fk_part_1_1 drop constraint my_fkey;       -- doesn't exist
+
 \set VERBOSITY terse   \\ -- suppress cascade details
-drop schema fkpart0, fkpart1 cascade;
+drop schema fkpart0, fkpart1, fkpart2 cascade;
 \set VERBOSITY default