Tracing FK delete dependencies in PostgreSQL

TL;DR #

WITH RECURSIVE fk_chain AS (SELECT c.oid         AS constraint_oid,
c.conname AS constraint_name,
c.conrelid AS table_oid,
c.confrelid AS referenced_table_oid,
c.confdeltype,
ARRAY [c.oid] AS path -- prevent cycles
FROM pg_constraint c
-- THIS IS A PLACE WHERE YOU ADJUST THE TABLE
WHERE c.confrelid = 'TODO_TABLE_NAME'::regclass
AND c.contype = 'f'
-- THIS IS A PLACE WHERE YOU ADJUST THE DESIRED RELATIONS
and c.confdeltype in ('a', 'r', 'n')

UNION ALL

SELECT child_c.oid,
child_c.conname,
child_c.conrelid,
child_c.confrelid,
child_c.confdeltype,
path || child_c.oid
FROM pg_constraint child_c
JOIN fk_chain parent ON child_c.confrelid = parent.table_oid
WHERE child_c.contype = 'f'
AND NOT child_c.oid = ANY (path)
and child_c.confdeltype in ('a', 'r', 'n'))
SELECT child_table.relname AS table_name,
referenced_table.relname AS parent_table,
CASE confdeltype
WHEN 'a' THEN 'NO ACTION'
WHEN 'r' THEN 'RESTRICT'
WHEN 'c' THEN 'CASCADE'
WHEN 'n' THEN 'SET NULL'
WHEN 'd' THEN 'SET DEFAULT'
ELSE 'UNKNOWN'
END AS on_delete_action
FROM fk_chain c
JOIN pg_class child_table ON child_table.oid = c.table_oid
JOIN pg_class referenced_table ON referenced_table.oid = c.referenced_table_oid;

Use Cases #

What It Does #

Output Columns #

Customization #


Since you've made it this far, sharing this article on your favorite social media network would be highly appreciated 💖! For feedback, please ping me on Twitter.

Published