Skip to content
Snippets Groups Projects
rewriteRemove.c 5.09 KiB
/*-------------------------------------------------------------------------
 *
 * rewriteRemove.c--
 *    routines for removing rewrite rules
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
 *    $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteRemove.c,v 1.2 1996/11/03 04:51:51 scrappy Exp $
 *
 *-------------------------------------------------------------------------
 */

#include "postgres.h"

#include "access/skey.h"
#include "catalog/pg_rewrite.h"
#include "catalog/catname.h"	/* for RewriteRelationName */
#include "utils/syscache.h"
#include "utils/elog.h"		/* for elog stuff */
#include "utils/palloc.h"
#include "utils/tqual.h"	/* 'NowTimeQual' defined here.. */
#include "access/heapam.h"	/* heap AM calls defined here */
#include "fmgr.h"		/* for CHAR_16_EQ */

#include "rewrite/rewriteRemove.h"	/* where the decls go */
#include "rewrite/rewriteSupport.h"

/*-----------------------------------------------------------------------
 * RewriteGetRuleEventRel
 *-----------------------------------------------------------------------
 */
char*
RewriteGetRuleEventRel(char *rulename)
{
    HeapTuple htp;
    Oid eventrel;
    
    htp = SearchSysCacheTuple(REWRITENAME, PointerGetDatum(rulename),
			      0,0,0);
    if (!HeapTupleIsValid(htp))
	elog(WARN, "RewriteGetRuleEventRel: rule \"%s\" not found",
	     rulename);
    eventrel = ((Form_pg_rewrite) GETSTRUCT(htp))->ev_class;
    htp = SearchSysCacheTuple(RELOID, PointerGetDatum(eventrel), 
			      0,0,0);
    if (!HeapTupleIsValid(htp))
	elog(WARN, "RewriteGetRuleEventRel: class %d not found",
	     eventrel);
    return ((Form_pg_class) GETSTRUCT(htp))->relname.data;
}

/* ----------------------------------------------------------------
 *
 * RemoveRewriteRule
 *
 * Delete a rule given its rulename.
 *
 * There are three steps.
 *   1) Find the corresponding tuple in 'pg_rewrite' relation.
 *      Find the rule Id (i.e. the Oid of the tuple) and finally delete
 *      the tuple.
 *   3) Delete the locks from the 'pg_class' relation.
 *
 *
 * ----------------------------------------------------------------
 */
void
RemoveRewriteRule(char *ruleName)
{
    Relation RewriteRelation	= NULL;
    HeapScanDesc scanDesc	= NULL;
    ScanKeyData scanKeyData;
    HeapTuple tuple		= NULL;
    Oid ruleId			= (Oid)0;
    Oid eventRelationOid	= (Oid)NULL;
    Datum eventRelationOidDatum	= (Datum)NULL;
    Buffer buffer		= (Buffer)NULL;
    bool isNull		= false;

    /*
     * Open the pg_rewrite relation. 
     */
    RewriteRelation = heap_openr(RewriteRelationName);
    
    /*
     * Scan the RuleRelation ('pg_rewrite') until we find a tuple
     */
    ScanKeyEntryInitialize(&scanKeyData, 0, Anum_pg_rewrite_rulename,
			   F_CHAR16EQ, NameGetDatum(ruleName));
    scanDesc = heap_beginscan(RewriteRelation,
			      0, NowTimeQual, 1, &scanKeyData);
    
    tuple = heap_getnext(scanDesc, 0, (Buffer *)NULL);
    
    /*
     * complain if no rule with such name existed
     */
    if (!HeapTupleIsValid(tuple)) {
	heap_close(RewriteRelation);
	elog(WARN, "No rule with name = '%s' was found.\n", ruleName);
    }
    
    /*
     * Store the OID of the rule (i.e. the tuple's OID)
     * and the event relation's OID
     */
    ruleId = tuple->t_oid;
    eventRelationOidDatum =
	PointerGetDatum(heap_getattr(tuple,
				     buffer,
				     Anum_pg_rewrite_ev_class,
		     RelationGetTupleDescriptor(RewriteRelation),
				     &isNull));
    if (isNull) {
	/* XXX strange!!! */
	elog(WARN, "RemoveRewriteRule: null event target relation!");
    }
    eventRelationOid = DatumGetObjectId(eventRelationOidDatum);
    
    /*
     * Now delete the relation level locks from the updated relation.
     * (Make sure we do this before we remove the rule from pg_rewrite.
     * Otherwise, heap_openr on eventRelationOid which reads pg_rwrite
     * for the rules will fail.)
     */
    prs2_deleteFromRelation(eventRelationOid, ruleId);

    /*
     * Now delete the tuple...
     */
    heap_delete(RewriteRelation, &(tuple->t_ctid));
    heap_close(RewriteRelation);
    heap_endscan(scanDesc);
}

/*
 * 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);
    
    /*
     * 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, NowTimeQual, 1, &scanKeyData);

    for(;;) {
	tuple = heap_getnext(scanDesc, 0, (Buffer *)NULL);
    
	if (!HeapTupleIsValid(tuple)) {
	    break; /* we're done */
	}
    
	/*
	 * delete the tuple...
	 */
	heap_delete(RewriteRelation, &(tuple->t_ctid));
    }

    heap_endscan(scanDesc);
    heap_close(RewriteRelation);
}