mirror of
https://github.com/postgres/postgres.git
synced 2025-07-08 11:42:09 +03:00
Avoid failure when altering state of partitioned foreign-key triggers.
Beginning in v15, if you apply ALTER TABLE ENABLE/DISABLE TRIGGER to a partitioned table, it also affects the partitions' cloned versions of the affected trigger(s). The initial implementation of this located the clones by name, but that fails on foreign-key triggers which have names incorporating their own OIDs. We can fix that, and also make the behavior more bulletproof in the face of user-initiated trigger renames, by identifying the cloned triggers by tgparentid. Following the lead of earlier commits in this area, I took care not to break ABI in the v15 branch, even though I rather doubt there are any external callers of EnableDisableTrigger. While here, update the documentation, which was not touched when the semantics were changed. Per bug #17817 from Alan Hodgson. Back-patch to v15; older versions do not have this behavior. Discussion: https://postgr.es/m/17817-31dfb7c2100d9f3d@postgresql.org
This commit is contained in:
@ -1715,7 +1715,8 @@ renametrig_partition(Relation tgrel, Oid partitionId, Oid parentTriggerOid,
|
||||
* to change 'tgenabled' field for the specified trigger(s)
|
||||
*
|
||||
* rel: relation to process (caller must hold suitable lock on it)
|
||||
* tgname: trigger to process, or NULL to scan all triggers
|
||||
* tgname: name of trigger to process, or NULL to scan all triggers
|
||||
* tgparent: if not zero, process only triggers with this tgparentid
|
||||
* fires_when: new value for tgenabled field. In addition to generic
|
||||
* enablement/disablement, this also defines when the trigger
|
||||
* should be fired in session replication roles.
|
||||
@ -1727,7 +1728,7 @@ renametrig_partition(Relation tgrel, Oid partitionId, Oid parentTriggerOid,
|
||||
* system triggers
|
||||
*/
|
||||
void
|
||||
EnableDisableTrigger(Relation rel, const char *tgname,
|
||||
EnableDisableTrigger(Relation rel, const char *tgname, Oid tgparent,
|
||||
char fires_when, bool skip_system, bool recurse,
|
||||
LOCKMODE lockmode)
|
||||
{
|
||||
@ -1766,6 +1767,9 @@ EnableDisableTrigger(Relation rel, const char *tgname,
|
||||
{
|
||||
Form_pg_trigger oldtrig = (Form_pg_trigger) GETSTRUCT(tuple);
|
||||
|
||||
if (OidIsValid(tgparent) && tgparent != oldtrig->tgparentid)
|
||||
continue;
|
||||
|
||||
if (oldtrig->tgisinternal)
|
||||
{
|
||||
/* system trigger ... ok to process? */
|
||||
@ -1816,7 +1820,8 @@ EnableDisableTrigger(Relation rel, const char *tgname,
|
||||
Relation part;
|
||||
|
||||
part = relation_open(partdesc->oids[i], lockmode);
|
||||
EnableDisableTrigger(part, NameStr(oldtrig->tgname),
|
||||
/* Match on child triggers' tgparentid, not their name */
|
||||
EnableDisableTrigger(part, NULL, oldtrig->oid,
|
||||
fires_when, skip_system, recurse,
|
||||
lockmode);
|
||||
table_close(part, NoLock); /* keep lock till commit */
|
||||
|
Reference in New Issue
Block a user