Rule rewriter was doing the wrong thing with conditional INSTEAD rules

whose conditions might yield NULL.  The negated qual to attach to the
original query is properly 'x IS NOT TRUE', not 'NOT x'.  This fix
produces correct behavior, but we may be taking a performance hit because
the planner is much stupider about IS NOT TRUE than it is about NOT
clauses.  Future TODO: teach prepqual, other parts of planner how to
cope with BooleanTest clauses more effectively.
This commit is contained in:
Tom Lane 2002-10-20 00:58:55 +00:00
parent 6d6b582850
commit a044e2abdd
3 changed files with 30 additions and 36 deletions

View File

@ -7,7 +7,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteHandler.c,v 1.112 2002/10/19 19:00:47 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteHandler.c,v 1.113 2002/10/20 00:58:55 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
@ -842,9 +842,11 @@ fireRIRrules(Query *parsetree)
/* /*
* Modify the given query by adding 'AND NOT rule_qual' to its qualification. * Modify the given query by adding 'AND rule_qual IS NOT TRUE' to its
* This is used to generate suitable "else clauses" for conditional INSTEAD * qualification. This is used to generate suitable "else clauses" for
* rules. * conditional INSTEAD rules. (Unfortunately we must use "x IS NOT TRUE",
* not just "NOT x" which the planner is much smarter about, else we will
* do the wrong thing when the qual evaluates to NULL.)
* *
* The rule_qual may contain references to OLD or NEW. OLD references are * The rule_qual may contain references to OLD or NEW. OLD references are
* replaced by references to the specified rt_index (the relation that the * replaced by references to the specified rt_index (the relation that the
@ -853,10 +855,10 @@ fireRIRrules(Query *parsetree)
* of the related entries in the query's own targetlist. * of the related entries in the query's own targetlist.
*/ */
static Query * static Query *
CopyAndAddQual(Query *parsetree, CopyAndAddInvertedQual(Query *parsetree,
Node *rule_qual, Node *rule_qual,
int rt_index, int rt_index,
CmdType event) CmdType event)
{ {
Query *new_tree = (Query *) copyObject(parsetree); Query *new_tree = (Query *) copyObject(parsetree);
Node *new_qual = (Node *) copyObject(rule_qual); Node *new_qual = (Node *) copyObject(rule_qual);
@ -872,7 +874,7 @@ CopyAndAddQual(Query *parsetree,
event, event,
rt_index); rt_index);
/* And attach the fixed qual */ /* And attach the fixed qual */
AddNotQual(new_tree, new_qual); AddInvertedQual(new_tree, new_qual);
return new_tree; return new_tree;
} }
@ -956,10 +958,10 @@ fireRules(Query *parsetree,
{ {
if (*qual_product == NULL) if (*qual_product == NULL)
*qual_product = parsetree; *qual_product = parsetree;
*qual_product = CopyAndAddQual(*qual_product, *qual_product = CopyAndAddInvertedQual(*qual_product,
event_qual, event_qual,
rt_index, rt_index,
event); event);
} }
} }

View File

@ -7,7 +7,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteManip.c,v 1.66 2002/09/11 14:48:54 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteManip.c,v 1.67 2002/10/20 00:58:55 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
@ -690,34 +690,26 @@ AddHavingQual(Query *parsetree, Node *havingQual)
parsetree->hasSubLinks |= checkExprHasSubLink(copy); parsetree->hasSubLinks |= checkExprHasSubLink(copy);
} }
#ifdef NOT_USED
/*
* Invert the given clause and add it to the WHERE qualifications of the
* given querytree. Inversion means "x IS NOT TRUE", not just "NOT x",
* else we will do the wrong thing when x evaluates to NULL.
*/
void void
AddNotHavingQual(Query *parsetree, Node *havingQual) AddInvertedQual(Query *parsetree, Node *qual)
{ {
Node *notqual; BooleanTest *invqual;
if (havingQual == NULL)
return;
/* Need not copy input qual, because AddHavingQual will... */
notqual = (Node *) make_notclause((Expr *) havingQual);
AddHavingQual(parsetree, notqual);
}
#endif
void
AddNotQual(Query *parsetree, Node *qual)
{
Node *notqual;
if (qual == NULL) if (qual == NULL)
return; return;
/* Need not copy input qual, because AddQual will... */ /* Need not copy input qual, because AddQual will... */
notqual = (Node *) make_notclause((Expr *) qual); invqual = makeNode(BooleanTest);
invqual->arg = qual;
invqual->booltesttype = IS_NOT_TRUE;
AddQual(parsetree, notqual); AddQual(parsetree, (Node *) invqual);
} }

View File

@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $Id: rewriteManip.h,v 1.31 2002/06/20 20:29:52 momjian Exp $ * $Id: rewriteManip.h,v 1.32 2002/10/20 00:58:55 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
@ -32,7 +32,7 @@ extern Query *getInsertSelectQuery(Query *parsetree, Query ***subquery_ptr);
extern void AddQual(Query *parsetree, Node *qual); extern void AddQual(Query *parsetree, Node *qual);
extern void AddHavingQual(Query *parsetree, Node *havingQual); extern void AddHavingQual(Query *parsetree, Node *havingQual);
extern void AddNotQual(Query *parsetree, Node *qual); extern void AddInvertedQual(Query *parsetree, Node *qual);
extern bool checkExprHasAggs(Node *node); extern bool checkExprHasAggs(Node *node);
extern bool checkExprHasSubLink(Node *node); extern bool checkExprHasSubLink(Node *node);