mirror of
https://github.com/postgres/postgres.git
synced 2025-07-05 07:21:24 +03:00
Correct attach/detach logic for FKs in partitions
There was no code to handle foreign key constraints on partitioned tables in the case of ALTER TABLE DETACH; and if you happened to ATTACH a partition that already had an equivalent constraint, that one was ignored and a new constraint was created. Adding this to the fact that foreign key cloning reuses the constraint name on the partition instead of generating a new name (as it probably should, to cater to SQL standard rules about constraint naming within schemas), the result was a pretty poor user experience -- the most visible failure was that just detaching a partition and re-attaching it failed with an error such as ERROR: duplicate key value violates unique constraint "pg_constraint_conrelid_contypid_conname_index" DETAIL: Key (conrelid, contypid, conname)=(26702, 0, test_result_asset_id_fkey) already exists. because it would try to create an identically-named constraint in the partition. To make matters worse, if you tried to drop the constraint in the now-independent partition, that would fail because the constraint was still seen as dependent on the constraint in its former parent partitioned table: ERROR: cannot drop inherited constraint "test_result_asset_id_fkey" of relation "test_result_cbsystem_0001_0050_monthly_2018_09" This fix attacks the problem from two angles: first, when the partition is detached, the constraint is also marked as independent, so the drop now works. Second, when the partition is re-attached, we scan existing constraints searching for one matching the FK in the parent, and if one exists, we link that one to the parent constraint. So we don't end up with a duplicate -- and better yet, we don't need to scan the referenced table to verify that the constraint holds. To implement this I made a small change to previously planner-only struct ForeignKeyCacheInfo to contain the constraint OID; also relcache now maintains the list of FKs for partitioned tables too. Backpatch to 11. Reported-by: Michael Vitale (bug #15425) Discussion: https://postgr.es/m/15425-2dbc9d2aa999f816@postgresql.org
This commit is contained in:
6
src/backend/utils/cache/relcache.c
vendored
6
src/backend/utils/cache/relcache.c
vendored
@ -4108,8 +4108,9 @@ RelationGetFKeyList(Relation relation)
|
||||
if (relation->rd_fkeyvalid)
|
||||
return relation->rd_fkeylist;
|
||||
|
||||
/* Fast path: if it doesn't have any triggers, it can't have FKs */
|
||||
if (!relation->rd_rel->relhastriggers)
|
||||
/* Fast path: non-partitioned tables without triggers can't have FKs */
|
||||
if (!relation->rd_rel->relhastriggers &&
|
||||
relation->rd_rel->relkind != RELKIND_PARTITIONED_TABLE)
|
||||
return NIL;
|
||||
|
||||
/*
|
||||
@ -4144,6 +4145,7 @@ RelationGetFKeyList(Relation relation)
|
||||
continue;
|
||||
|
||||
info = makeNode(ForeignKeyCacheInfo);
|
||||
info->conoid = HeapTupleGetOid(htup);
|
||||
info->conrelid = constraint->conrelid;
|
||||
info->confrelid = constraint->confrelid;
|
||||
|
||||
|
Reference in New Issue
Block a user