PostgreSQL
 sql >> Database >  >> RDS >> PostgreSQL

Come elencare le tabelle interessate dall'eliminazione a catena

Usa pg_depend . Esempio:

create table master (id int primary key);
create table detail_1 (id int, master_id int references master(id) on delete restrict);
create table detail_2 (id int, master_id int references master(id) on delete cascade);

select pg_describe_object(classid, objid, objsubid)
from pg_depend 
where refobjid = 'master'::regclass and deptype = 'n';

                  pg_describe_object
------------------------------------------------------
 constraint detail_1_master_id_fkey on table detail_1
 constraint detail_2_master_id_fkey on table detail_2
(2 rows)

deptype = 'n' significa:

DIPENDENZA NORMALE - Una relazione normale tra oggetti creati separatamente. L'oggetto dipendente può essere eliminato senza influire sull'oggetto referenziato. L'oggetto di riferimento può essere eliminato solo specificando CASCADE, nel qual caso viene eliminato anche l'oggetto dipendente.

Usa pg_get_constraintdef() per ottenere le definizioni dei vincoli:

select 
    pg_describe_object(classid, objid, objsubid), 
    pg_get_constraintdef(objid)
from pg_depend 
where refobjid = 'master'::regclass and deptype = 'n';


                  pg_describe_object                  |                       pg_get_constraintdef
------------------------------------------------------+------------------------------------------------------------------
 constraint detail_1_master_id_fkey on table detail_1 | FOREIGN KEY (master_id) REFERENCES master(id) ON DELETE RESTRICT
 constraint detail_2_master_id_fkey on table detail_2 | FOREIGN KEY (master_id) REFERENCES master(id) ON DELETE CASCADE
(2 rows)

Per trovare l'intera catena di dipendenze a cascata dovremmo usare la ricorsione e guardare nel catalogo pg_constraint per ottenere id di una tabella dipendente.

with recursive chain as (
    select classid, objid, objsubid, conrelid
    from pg_depend d
    join pg_constraint c on c.oid = objid
    where refobjid = 'the_table'::regclass and deptype = 'n'
union all
    select d.classid, d.objid, d.objsubid, c.conrelid
    from pg_depend d
    join pg_constraint c on c.oid = objid
    join chain on d.refobjid = chain.conrelid and d.deptype = 'n'
    )
select pg_describe_object(classid, objid, objsubid), pg_get_constraintdef(objid)
from chain;