/*------------------------------------------------------------------------- * * rewriteRemove.c * routines for removing rewrite rules * * Portions Copyright (c) 1996-2000, PostgreSQL, Inc * Portions Copyright (c) 1994, Regents of the University of California * * * IDENTIFICATION * $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteRemove.c,v 1.38 2000/06/30 07:04:23 tgl Exp $ * *------------------------------------------------------------------------- */ #include "postgres.h" #include "utils/builtins.h" #include "access/heapam.h" #include "catalog/catname.h" #include "catalog/pg_rewrite.h" #include "commands/comment.h" #include "rewrite/rewriteRemove.h" #include "rewrite/rewriteSupport.h" #include "utils/fmgroids.h" #include "utils/syscache.h" /*----------------------------------------------------------------------- * RewriteGetRuleEventRel *----------------------------------------------------------------------- */ char * RewriteGetRuleEventRel(char *rulename) { HeapTuple htup; Oid eventrel; htup = SearchSysCacheTuple(RULENAME, PointerGetDatum(rulename), 0, 0, 0); if (!HeapTupleIsValid(htup)) elog(ERROR, "Rule or view '%s' not found", ((!strncmp(rulename, "_RET", 4)) ? (rulename + 4) : rulename)); eventrel = ((Form_pg_rewrite) GETSTRUCT(htup))->ev_class; htup = SearchSysCacheTuple(RELOID, PointerGetDatum(eventrel), 0, 0, 0); if (!HeapTupleIsValid(htup)) elog(ERROR, "Class '%u' not found", eventrel); return NameStr(((Form_pg_class) GETSTRUCT(htup))->relname); } /* * RemoveRewriteRule * * Delete a rule given its rulename. */ void RemoveRewriteRule(char *ruleName) { Relation RewriteRelation; Relation event_relation; HeapTuple tuple; Oid ruleId; Oid eventRelationOid; bool hasMoreRules; /* * Open the pg_rewrite relation. */ RewriteRelation = heap_openr(RewriteRelationName, RowExclusiveLock); /* * Find the tuple for the target rule. */ tuple = SearchSysCacheTupleCopy(RULENAME, PointerGetDatum(ruleName), 0, 0, 0); /* * complain if no rule with such name existed */ if (!HeapTupleIsValid(tuple)) { heap_close(RewriteRelation, RowExclusiveLock); elog(ERROR, "Rule '%s' not found\n", ruleName); } /* * Save the OID of the rule (i.e. the tuple's OID) and the event * relation's OID */ ruleId = tuple->t_data->t_oid; eventRelationOid = ((Form_pg_rewrite) GETSTRUCT(tuple))->ev_class; /* * We had better grab AccessExclusiveLock so that we know no other * rule additions/deletions are going on for this relation. Else * we cannot set relhasrules correctly. Besides, we don't want to * be changing the ruleset while queries are executing on the rel. */ event_relation = heap_open(eventRelationOid, AccessExclusiveLock); hasMoreRules = event_relation->rd_rules != NULL && event_relation->rd_rules->numLocks > 1; /* * Delete any comments associated with this rule */ DeleteComments(ruleId); /* * Now delete the pg_rewrite tuple for the rule */ heap_delete(RewriteRelation, &tuple->t_self, NULL); heap_freetuple(tuple); heap_close(RewriteRelation, RowExclusiveLock); /* * Set pg_class 'relhasrules' field correctly for event relation. * * Important side effect: an SI notice is broadcast to force all * backends (including me!) to update relcache entries with the * new rule set. Therefore, must do this even if relhasrules is * still true! */ setRelhasrulesInRelation(eventRelationOid, hasMoreRules); /* Close rel, but keep lock till commit... */ heap_close(event_relation, NoLock); } /* * RelationRemoveRules - * removes all rules associated with the relation when the relation is * being removed. */ void RelationRemoveRules(Oid relid) { Relation RewriteRelation = NULL; HeapScanDesc scanDesc = NULL; ScanKeyData scanKeyData; HeapTuple tuple = NULL; /* * Open the pg_rewrite relation. */ RewriteRelation = heap_openr(RewriteRelationName, RowExclusiveLock); /* * Scan the RuleRelation ('pg_rewrite') for all the tuples that has * the same ev_class as relid (the relation to be removed). */ ScanKeyEntryInitialize(&scanKeyData, 0, Anum_pg_rewrite_ev_class, F_OIDEQ, ObjectIdGetDatum(relid)); scanDesc = heap_beginscan(RewriteRelation, 0, SnapshotNow, 1, &scanKeyData); while (HeapTupleIsValid(tuple = heap_getnext(scanDesc, 0))) { /*** Delete any comments associated with this relation ***/ DeleteComments(tuple->t_data->t_oid); heap_delete(RewriteRelation, &tuple->t_self, NULL); } heap_endscan(scanDesc); heap_close(RewriteRelation, RowExclusiveLock); }