mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Repair bug reported by Huxton, 1/24/01. We need to include a rule's
original table ('OLD' table) in its join tree if OLD is referenced by either the rule action, the rule qual, or the original query qual that will be added to the rule action. However, we only want one instance of the original table to be included; so beware of the possibility that the rule action already has a jointree entry for OLD.
This commit is contained in:
@@ -7,7 +7,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteHandler.c,v 1.88 2001/01/24 19:43:05 momjian Exp $
|
* $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteHandler.c,v 1.89 2001/01/27 04:40:59 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@@ -37,7 +37,7 @@ static RewriteInfo *gatherRewriteMeta(Query *parsetree,
|
|||||||
int rt_index,
|
int rt_index,
|
||||||
CmdType event,
|
CmdType event,
|
||||||
bool instead_flag);
|
bool instead_flag);
|
||||||
static List *adjustJoinTreeList(Query *parsetree, int rt_index, bool *found);
|
static List *adjustJoinTreeList(Query *parsetree, bool removert, int rt_index);
|
||||||
static void markQueryForUpdate(Query *qry, bool skipOldNew);
|
static void markQueryForUpdate(Query *qry, bool skipOldNew);
|
||||||
static List *matchLocks(CmdType event, RuleLock *rulelocks,
|
static List *matchLocks(CmdType event, RuleLock *rulelocks,
|
||||||
int varno, Query *parsetree);
|
int varno, Query *parsetree);
|
||||||
@@ -119,18 +119,25 @@ gatherRewriteMeta(Query *parsetree,
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Each rule action's jointree should be the main parsetree's jointree
|
* Each rule action's jointree should be the main parsetree's jointree
|
||||||
* plus that rule's jointree, but *without* the original rtindex
|
* plus that rule's jointree, but usually *without* the original rtindex
|
||||||
* that we're replacing (if present, which it won't be for INSERT).
|
* that we're replacing (if present, which it won't be for INSERT).
|
||||||
* Note that if the rule refers to OLD, its jointree will add back
|
* Note that if the rule action refers to OLD, its jointree will add
|
||||||
* a reference to rt_index.
|
* a reference to rt_index. If the rule action doesn't refer to OLD,
|
||||||
|
* but either the rule_qual or the user query quals do, then we need to
|
||||||
|
* keep the original rtindex in the jointree to provide data for the
|
||||||
|
* quals. We don't want the original rtindex to be joined twice,
|
||||||
|
* however, so avoid keeping it if the rule action mentions it.
|
||||||
*/
|
*/
|
||||||
if (sub_action->jointree != NULL)
|
if (sub_action->jointree != NULL)
|
||||||
{
|
{
|
||||||
bool found;
|
bool keeporig;
|
||||||
List *newjointree = adjustJoinTreeList(parsetree,
|
List *newjointree;
|
||||||
rt_index,
|
|
||||||
&found);
|
|
||||||
|
|
||||||
|
keeporig = (! rangeTableEntry_used((Node *) sub_action->jointree,
|
||||||
|
rt_index, 0)) &&
|
||||||
|
(rangeTableEntry_used(info->rule_qual, rt_index, 0) ||
|
||||||
|
rangeTableEntry_used(parsetree->jointree->quals, rt_index, 0));
|
||||||
|
newjointree = adjustJoinTreeList(parsetree, !keeporig, rt_index);
|
||||||
sub_action->jointree->fromlist =
|
sub_action->jointree->fromlist =
|
||||||
nconc(newjointree, sub_action->jointree->fromlist);
|
nconc(newjointree, sub_action->jointree->fromlist);
|
||||||
}
|
}
|
||||||
@@ -181,29 +188,30 @@ gatherRewriteMeta(Query *parsetree,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Copy the query's jointree list, and attempt to remove any occurrence
|
* Copy the query's jointree list, and optionally attempt to remove any
|
||||||
* of the given rt_index as a top-level join item (we do not look for it
|
* occurrence of the given rt_index as a top-level join item (we do not look
|
||||||
* within join items; this is OK because we are only expecting to find it
|
* for it within join items; this is OK because we are only expecting to find
|
||||||
* as an UPDATE or DELETE target relation, which will be at the top level
|
* it as an UPDATE or DELETE target relation, which will be at the top level
|
||||||
* of the join). Returns modified jointree list --- original list
|
* of the join). Returns modified jointree list --- original list is not
|
||||||
* is not changed. *found is set to indicate if we found the rt_index.
|
* changed.
|
||||||
*/
|
*/
|
||||||
static List *
|
static List *
|
||||||
adjustJoinTreeList(Query *parsetree, int rt_index, bool *found)
|
adjustJoinTreeList(Query *parsetree, bool removert, int rt_index)
|
||||||
{
|
{
|
||||||
List *newjointree = listCopy(parsetree->jointree->fromlist);
|
List *newjointree = listCopy(parsetree->jointree->fromlist);
|
||||||
List *jjt;
|
List *jjt;
|
||||||
|
|
||||||
*found = false;
|
if (removert)
|
||||||
foreach(jjt, newjointree)
|
|
||||||
{
|
{
|
||||||
RangeTblRef *rtr = lfirst(jjt);
|
foreach(jjt, newjointree)
|
||||||
|
|
||||||
if (IsA(rtr, RangeTblRef) && rtr->rtindex == rt_index)
|
|
||||||
{
|
{
|
||||||
newjointree = lremove(rtr, newjointree);
|
RangeTblRef *rtr = lfirst(jjt);
|
||||||
*found = true;
|
|
||||||
break;
|
if (IsA(rtr, RangeTblRef) && rtr->rtindex == rt_index)
|
||||||
|
{
|
||||||
|
newjointree = lremove(rtr, newjointree);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return newjointree;
|
return newjointree;
|
||||||
|
Reference in New Issue
Block a user