1 /*-------------------------------------------------------------------------
5 * Copyright (c) 1994, Regents of the University of California
9 * $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteHandler.c,v 1.27 1998/12/14 00:02:16 thomas Exp $
11 *-------------------------------------------------------------------------
15 #include "miscadmin.h"
16 #include "utils/palloc.h"
17 #include "utils/elog.h"
18 #include "utils/rel.h"
19 #include "nodes/pg_list.h"
20 #include "nodes/primnodes.h"
21 #include "nodes/relation.h"
23 #include "parser/parsetree.h" /* for parsetree manipulation */
24 #include "parser/parse_relation.h"
25 #include "nodes/parsenodes.h"
27 #include "rewrite/rewriteSupport.h"
28 #include "rewrite/rewriteHandler.h"
29 #include "rewrite/rewriteManip.h"
30 #include "rewrite/locks.h"
32 #include "commands/creatinh.h"
33 #include "access/heapam.h"
35 #include "utils/lsyscache.h"
36 #include "utils/syscache.h"
37 #include "utils/acl.h"
38 #include "catalog/pg_shadow.h"
39 #include "catalog/pg_type.h"
42 static RewriteInfo *gatherRewriteMeta(Query *parsetree,
48 static bool rangeTableEntry_used(Node *node, int rt_index, int sublevels_up);
49 static bool attribute_used(Node *node, int rt_index, int attno, int sublevels_up);
50 static void modifyAggregUplevel(Node *node);
51 static void modifyAggregChangeVarnodes(Node **nodePtr, int rt_index, int new_index, int sublevels_up);
52 static void modifyAggregDropQual(Node **nodePtr, Node *orignode, Expr *expr);
53 static SubLink *modifyAggregMakeSublink(Expr *origexp, Query *parsetree);
54 static void modifyAggregQual(Node **nodePtr, Query *parsetree);
57 static Query *fireRIRrules(Query *parsetree);
62 * Gather meta information about parsetree, and rule. Fix rule body
63 * and qualifier so that they can be mixed with the parsetree and
64 * maintain semantic validity
67 gatherRewriteMeta(Query *parsetree,
78 info = (RewriteInfo *) palloc(sizeof(RewriteInfo));
79 info->rt_index = rt_index;
81 info->instead_flag = *instead_flag;
82 info->rule_action = (Query *) copyObject(rule_action);
83 info->rule_qual = (Node *) copyObject(rule_qual);
84 if (info->rule_action == NULL)
88 info->nothing = FALSE;
89 info->action = info->rule_action->commandType;
90 info->current_varno = rt_index;
91 info->rt = parsetree->rtable;
92 rt_length = length(info->rt);
93 info->rt = append(info->rt, info->rule_action->rtable);
95 info->new_varno = PRS2_NEW_VARNO + rt_length;
96 OffsetVarNodes(info->rule_action->qual, rt_length, 0);
97 OffsetVarNodes((Node *) info->rule_action->targetList, rt_length, 0);
98 OffsetVarNodes(info->rule_qual, rt_length, 0);
99 ChangeVarNodes((Node *) info->rule_action->qual,
100 PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
101 ChangeVarNodes((Node *) info->rule_action->targetList,
102 PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
103 ChangeVarNodes(info->rule_qual,
104 PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
107 * bug here about replace CURRENT -- sort of replace current is
108 * deprecated now so this code shouldn't really need to be so
111 if (info->action != CMD_SELECT)
112 { /* i.e update XXXXX */
113 int new_result_reln = 0;
115 result_reln = info->rule_action->resultRelation;
118 case PRS2_CURRENT_VARNO:
119 new_result_reln = rt_index;
121 case PRS2_NEW_VARNO: /* XXX */
123 new_result_reln = result_reln + rt_length;
126 info->rule_action->resultRelation = new_result_reln;
134 * rangeTableEntry_used -
135 * we need to process a RTE for RIR rules only if it is
136 * referenced somewhere in var nodes of the query.
139 rangeTableEntry_used(Node *node, int rt_index, int sublevels_up)
144 switch(nodeTag(node)) {
147 TargetEntry *tle = (TargetEntry *)node;
149 return rangeTableEntry_used(
158 Aggreg *agg = (Aggreg *)node;
160 return rangeTableEntry_used(
161 (Node *)(agg->target),
169 GroupClause *grp = (GroupClause *)node;
171 return rangeTableEntry_used(
172 (Node *)(grp->entry),
180 Expr *exp = (Expr *)node;
182 return rangeTableEntry_used(
191 Iter *iter = (Iter *)node;
193 return rangeTableEntry_used(
194 (Node *)(iter->iterexpr),
202 ArrayRef *ref = (ArrayRef *)node;
204 if (rangeTableEntry_used(
205 (Node *)(ref->refupperindexpr),
210 if (rangeTableEntry_used(
211 (Node *)(ref->reflowerindexpr),
216 if (rangeTableEntry_used(
217 (Node *)(ref->refexpr),
222 if (rangeTableEntry_used(
223 (Node *)(ref->refassgnexpr),
234 Var *var = (Var *)node;
236 if (var->varlevelsup == sublevels_up)
237 return var->varno == rt_index;
253 foreach (l, (List *)node) {
254 if (rangeTableEntry_used(
266 SubLink *sub = (SubLink *)node;
268 if (rangeTableEntry_used(
269 (Node *)(sub->lefthand),
274 if (rangeTableEntry_used(
275 (Node *)(sub->subselect),
286 CaseExpr *exp = (CaseExpr *)node;
288 if (rangeTableEntry_used(
294 if (rangeTableEntry_used(
295 (Node *)(exp->defresult),
306 CaseWhen *when = (CaseWhen *)node;
308 if (rangeTableEntry_used(
309 (Node *)(when->expr),
314 if (rangeTableEntry_used(
315 (Node *)(when->result),
326 Query *qry = (Query *)node;
328 if (rangeTableEntry_used(
329 (Node *)(qry->targetList),
334 if (rangeTableEntry_used(
340 if (rangeTableEntry_used(
341 (Node *)(qry->havingQual),
346 if (rangeTableEntry_used(
347 (Node *)(qry->groupClause),
357 elog(NOTICE, "unknown node tag %d in rangeTableEntry_used()", nodeTag(node));
358 elog(NOTICE, "Node is: %s", nodeToString(node));
370 * Check if a specific attribute number of a RTE is used
371 * somewhere in the query
374 attribute_used(Node *node, int rt_index, int attno, int sublevels_up)
379 switch(nodeTag(node)) {
382 TargetEntry *tle = (TargetEntry *)node;
384 return attribute_used(
394 Aggreg *agg = (Aggreg *)node;
396 return attribute_used(
397 (Node *)(agg->target),
406 GroupClause *grp = (GroupClause *)node;
408 return attribute_used(
409 (Node *)(grp->entry),
418 Expr *exp = (Expr *)node;
420 return attribute_used(
430 Iter *iter = (Iter *)node;
432 return attribute_used(
433 (Node *)(iter->iterexpr),
442 ArrayRef *ref = (ArrayRef *)node;
445 (Node *)(ref->refupperindexpr),
452 (Node *)(ref->reflowerindexpr),
459 (Node *)(ref->refexpr),
466 (Node *)(ref->refassgnexpr),
478 Var *var = (Var *)node;
480 if (var->varlevelsup == sublevels_up)
481 return var->varno == rt_index;
497 foreach (l, (List *)node) {
511 SubLink *sub = (SubLink *)node;
514 (Node *)(sub->lefthand),
521 (Node *)(sub->subselect),
533 Query *qry = (Query *)node;
536 (Node *)(qry->targetList),
550 (Node *)(qry->havingQual),
557 (Node *)(qry->groupClause),
568 elog(NOTICE, "unknown node tag %d in attribute_used()", nodeTag(node));
569 elog(NOTICE, "Node is: %s", nodeToString(node));
580 * modifyAggregUplevel -
581 * In the newly created sublink for an aggregate column used in
582 * the qualification, we must adjust the varlevelsup in all the
586 modifyAggregUplevel(Node *node)
591 switch(nodeTag(node)) {
594 TargetEntry *tle = (TargetEntry *)node;
597 (Node *)(tle->expr));
603 Aggreg *agg = (Aggreg *)node;
606 (Node *)(agg->target));
612 Expr *exp = (Expr *)node;
615 (Node *)(exp->args));
621 Iter *iter = (Iter *)node;
624 (Node *)(iter->iterexpr));
630 ArrayRef *ref = (ArrayRef *)node;
633 (Node *)(ref->refupperindexpr));
635 (Node *)(ref->reflowerindexpr));
637 (Node *)(ref->refexpr));
639 (Node *)(ref->refassgnexpr));
645 Var *var = (Var *)node;
661 foreach (l, (List *)node)
669 SubLink *sub = (SubLink *)node;
672 (Node *)(sub->lefthand));
675 (Node *)(sub->oper));
678 (Node *)(sub->subselect));
684 Query *qry = (Query *)node;
687 (Node *)(qry->targetList));
690 (Node *)(qry->qual));
693 (Node *)(qry->havingQual));
696 (Node *)(qry->groupClause));
701 elog(NOTICE, "unknown node tag %d in modifyAggregUplevel()", nodeTag(node));
702 elog(NOTICE, "Node is: %s", nodeToString(node));
711 * modifyAggregChangeVarnodes -
712 * Change the var nodes in a sublink created for an aggregate column
713 * used in the qualification that is subject of the aggregate
714 * function to point to the correct local RTE.
717 modifyAggregChangeVarnodes(Node **nodePtr, int rt_index, int new_index, int sublevels_up)
719 Node *node = *nodePtr;
724 switch(nodeTag(node)) {
727 TargetEntry *tle = (TargetEntry *)node;
729 modifyAggregChangeVarnodes(
730 (Node **)(&(tle->expr)),
739 Aggreg *agg = (Aggreg *)node;
741 modifyAggregChangeVarnodes(
742 (Node **)(&(agg->target)),
751 GroupClause *grp = (GroupClause *)node;
753 modifyAggregChangeVarnodes(
754 (Node **)(&(grp->entry)),
763 Expr *exp = (Expr *)node;
765 modifyAggregChangeVarnodes(
766 (Node **)(&(exp->args)),
775 Iter *iter = (Iter *)node;
777 modifyAggregChangeVarnodes(
778 (Node **)(&(iter->iterexpr)),
787 ArrayRef *ref = (ArrayRef *)node;
789 modifyAggregChangeVarnodes(
790 (Node **)(&(ref->refupperindexpr)),
794 modifyAggregChangeVarnodes(
795 (Node **)(&(ref->reflowerindexpr)),
799 modifyAggregChangeVarnodes(
800 (Node **)(&(ref->refexpr)),
804 modifyAggregChangeVarnodes(
805 (Node **)(&(ref->refassgnexpr)),
814 Var *var = (Var *)node;
816 if (var->varlevelsup == sublevels_up &&
817 var->varno == rt_index) {
818 var = copyObject(var);
819 var->varno = new_index;
820 var->varnoold = new_index;
821 var->varlevelsup = 0;
823 *nodePtr = (Node *)var;
838 foreach (l, (List *)node)
839 modifyAggregChangeVarnodes(
840 (Node **)(&lfirst(l)),
849 SubLink *sub = (SubLink *)node;
851 modifyAggregChangeVarnodes(
852 (Node **)(&(sub->lefthand)),
857 modifyAggregChangeVarnodes(
858 (Node **)(&(sub->oper)),
863 modifyAggregChangeVarnodes(
864 (Node **)(&(sub->subselect)),
873 Query *qry = (Query *)node;
875 modifyAggregChangeVarnodes(
876 (Node **)(&(qry->targetList)),
881 modifyAggregChangeVarnodes(
882 (Node **)(&(qry->qual)),
887 modifyAggregChangeVarnodes(
888 (Node **)(&(qry->havingQual)),
893 modifyAggregChangeVarnodes(
894 (Node **)(&(qry->groupClause)),
902 elog(NOTICE, "unknown node tag %d in modifyAggregChangeVarnodes()", nodeTag(node));
903 elog(NOTICE, "Node is: %s", nodeToString(node));
912 * modifyAggregDropQual -
913 * remove the pure aggreg clase from a qualification
916 modifyAggregDropQual(Node **nodePtr, Node *orignode, Expr *expr)
918 Node *node = *nodePtr;
923 switch(nodeTag(node)) {
929 Aggreg *agg = (Aggreg *)node;
930 Aggreg *oagg = (Aggreg *)orignode;
932 modifyAggregDropQual(
933 (Node **)(&(agg->target)),
934 (Node *)(oagg->target),
950 Expr *this_expr = (Expr *)node;
951 Expr *orig_expr = (Expr *)orignode;
953 if (orig_expr == expr) {
956 if (expr->typeOid != BOOLOID)
958 "aggregate expression in qualification isn't of type bool");
959 ctrue = makeNode(Const);
960 ctrue->consttype = BOOLOID;
962 ctrue->constisnull = FALSE;
963 ctrue->constvalue = (Datum)TRUE;
964 ctrue->constbyval = TRUE;
966 *nodePtr = (Node *)ctrue;
969 modifyAggregDropQual(
970 (Node **)(&(this_expr->args)),
971 (Node *)(orig_expr->args),
978 Iter *iter = (Iter *)node;
979 Iter *oiter = (Iter *)orignode;
981 modifyAggregDropQual(
982 (Node **)(&(iter->iterexpr)),
983 (Node *)(oiter->iterexpr),
990 ArrayRef *ref = (ArrayRef *)node;
991 ArrayRef *oref = (ArrayRef *)orignode;
993 modifyAggregDropQual(
994 (Node **)(&(ref->refupperindexpr)),
995 (Node *)(oref->refupperindexpr),
997 modifyAggregDropQual(
998 (Node **)(&(ref->reflowerindexpr)),
999 (Node *)(oref->reflowerindexpr),
1001 modifyAggregDropQual(
1002 (Node **)(&(ref->refexpr)),
1003 (Node *)(oref->refexpr),
1005 modifyAggregDropQual(
1006 (Node **)(&(ref->refassgnexpr)),
1007 (Node *)(oref->refassgnexpr),
1015 List *ol = (List *)orignode;
1018 foreach (l, (List *)node) {
1019 modifyAggregDropQual(
1020 (Node **)(&(lfirst(l))),
1021 (Node *)nth(li, ol),
1030 SubLink *sub = (SubLink *)node;
1031 SubLink *osub = (SubLink *)orignode;
1033 modifyAggregDropQual(
1034 (Node **)(&(sub->subselect)),
1035 (Node *)(osub->subselect),
1042 Query *qry = (Query *)node;
1043 Query *oqry = (Query *)orignode;
1045 modifyAggregDropQual(
1046 (Node **)(&(qry->qual)),
1047 (Node *)(oqry->qual),
1050 modifyAggregDropQual(
1051 (Node **)(&(qry->havingQual)),
1052 (Node *)(oqry->havingQual),
1058 elog(NOTICE, "unknown node tag %d in modifyAggregDropQual()", nodeTag(node));
1059 elog(NOTICE, "Node is: %s", nodeToString(node));
1068 * modifyAggregMakeSublink -
1069 * Create a sublink node for a qualification expression that
1070 * uses an aggregate column of a view
1073 modifyAggregMakeSublink(Expr *origexp, Query *parsetree)
1083 Expr *exp = copyObject(origexp);
1085 if (nodeTag(nth(0, exp->args)) == T_Aggreg)
1087 if (nodeTag(nth(1, exp->args)) == T_Aggreg)
1088 elog(ERROR, "rewrite: comparision of 2 aggregate columns not supported");
1090 elog(ERROR, "rewrite: aggregate column of view must be at rigth side in qual");
1093 aggreg = (Aggreg *)nth(1, exp->args);
1094 target = (Var *)(aggreg->target);
1095 rte = (RangeTblEntry *)nth(target->varno - 1, parsetree->rtable);
1096 tle = makeNode(TargetEntry);
1097 resdom = makeNode(Resdom);
1099 aggreg->usenulls = TRUE;
1102 resdom->restype = ((Oper *)(exp->oper))->opresulttype;
1103 resdom->restypmod = -1;
1104 resdom->resname = pstrdup("<noname>");
1106 resdom->reskeyop = 0;
1107 resdom->resjunk = 0;
1109 tle->resdom = resdom;
1110 tle->expr = (Node *)aggreg;
1112 subqual = copyObject(parsetree->qual);
1113 modifyAggregDropQual((Node **)&subqual, (Node *)parsetree->qual, origexp);
1115 sublink = makeNode(SubLink);
1116 sublink->subLinkType = EXPR_SUBLINK;
1117 sublink->useor = FALSE;
1118 sublink->lefthand = lappend(NIL, copyObject(lfirst(exp->args)));
1119 sublink->oper = lappend(NIL, copyObject(exp));
1120 sublink->subselect = NULL;
1122 subquery = makeNode(Query);
1123 sublink->subselect = (Node *)subquery;
1125 subquery->commandType = CMD_SELECT;
1126 subquery->utilityStmt = NULL;
1127 subquery->resultRelation = 0;
1128 subquery->into = NULL;
1129 subquery->isPortal = FALSE;
1130 subquery->isBinary = FALSE;
1131 subquery->unionall = FALSE;
1132 subquery->uniqueFlag = NULL;
1133 subquery->sortClause = NULL;
1134 subquery->rtable = lappend(NIL, rte);
1135 subquery->targetList = lappend(NIL, tle);
1136 subquery->qual = subqual;
1137 subquery->groupClause = NIL;
1138 subquery->havingQual = NULL;
1139 subquery->hasAggs = TRUE;
1140 subquery->hasSubLinks = FALSE;
1141 subquery->unionClause = NULL;
1144 modifyAggregUplevel((Node *)sublink);
1146 modifyAggregChangeVarnodes((Node **)&(sublink->lefthand), target->varno,
1147 1, target->varlevelsup);
1148 modifyAggregChangeVarnodes((Node **)&(sublink->oper), target->varno,
1149 1, target->varlevelsup);
1150 modifyAggregChangeVarnodes((Node **)&(sublink->subselect), target->varno,
1151 1, target->varlevelsup);
1158 * modifyAggregQual -
1159 * Search for qualification expressions that contain aggregate
1160 * functions and substiture them by sublinks. These expressions
1161 * originally come from qualifications that use aggregate columns
1165 modifyAggregQual(Node **nodePtr, Query *parsetree)
1167 Node *node = *nodePtr;
1172 switch(nodeTag(node)) {
1184 GroupClause *grp = (GroupClause *)node;
1187 (Node **)(&(grp->entry)),
1194 Expr *exp = (Expr *)node;
1198 if (length(exp->args) != 2) {
1200 (Node **)(&(exp->args)),
1205 if (nodeTag(nth(0, exp->args)) != T_Aggreg &&
1206 nodeTag(nth(1, exp->args)) != T_Aggreg) {
1209 (Node **)(&(exp->args)),
1214 sub = modifyAggregMakeSublink(exp,
1217 *nodePtr = (Node *)sub;
1218 parsetree->hasSubLinks = TRUE;
1224 /* We're calling recursively,
1225 * and this routine knows how to handle lists
1226 * so let it do the work to handle the WHEN clauses... */
1228 (Node **)(&(((CaseExpr *)node)->args)),
1232 (Node **)(&(((CaseExpr *)node)->defresult)),
1240 (Node **)(&(((CaseWhen *)node)->expr)),
1244 (Node **)(&(((CaseWhen *)node)->result)),
1251 Iter *iter = (Iter *)node;
1254 (Node **)(&(iter->iterexpr)),
1261 ArrayRef *ref = (ArrayRef *)node;
1264 (Node **)(&(ref->refupperindexpr)),
1267 (Node **)(&(ref->reflowerindexpr)),
1270 (Node **)(&(ref->refexpr)),
1273 (Node **)(&(ref->refassgnexpr)),
1282 foreach (l, (List *)node)
1284 (Node **)(&(lfirst(l))),
1291 SubLink *sub = (SubLink *)node;
1294 (Node **)(&(sub->subselect)),
1295 (Query *)(sub->subselect));
1301 Query *qry = (Query *)node;
1304 (Node **)(&(qry->qual)),
1308 (Node **)(&(qry->havingQual)),
1314 elog(NOTICE, "unknown node tag %d in modifyAggregQual()", nodeTag(node));
1315 elog(NOTICE, "Node is: %s", nodeToString(node));
1324 FindMatchingTLEntry(List *tlist, char *e_attname)
1330 TargetEntry *tle = lfirst(i);
1333 resname = tle->resdom->resname;
1334 if (!strcmp(e_attname, resname))
1344 Const *c = makeNode(Const);
1346 c->consttype = type;
1347 c->constlen = get_typlen(type);
1348 c->constvalue = PointerGetDatum(NULL);
1349 c->constisnull = true;
1350 c->constbyval = get_typbyval(type);
1356 apply_RIR_adjust_sublevel(Node *node, int sublevels_up)
1361 switch(nodeTag(node)) {
1364 TargetEntry *tle = (TargetEntry *)node;
1366 apply_RIR_adjust_sublevel(
1367 (Node *)(tle->expr),
1374 Aggreg *agg = (Aggreg *)node;
1376 apply_RIR_adjust_sublevel(
1377 (Node *)(agg->target),
1384 GroupClause *grp = (GroupClause *)node;
1386 apply_RIR_adjust_sublevel(
1387 (Node *)(grp->entry),
1394 Expr *exp = (Expr *)node;
1396 apply_RIR_adjust_sublevel(
1397 (Node *)(exp->args),
1404 Iter *iter = (Iter *)node;
1406 apply_RIR_adjust_sublevel(
1407 (Node *)(iter->iterexpr),
1414 ArrayRef *ref = (ArrayRef *)node;
1416 apply_RIR_adjust_sublevel(
1417 (Node *)(ref->refupperindexpr),
1420 apply_RIR_adjust_sublevel(
1421 (Node *)(ref->reflowerindexpr),
1424 apply_RIR_adjust_sublevel(
1425 (Node *)(ref->refexpr),
1428 apply_RIR_adjust_sublevel(
1429 (Node *)(ref->refassgnexpr),
1436 Var *var = (Var *)node;
1438 var->varlevelsup = sublevels_up;
1452 foreach (l, (List *)node) {
1453 apply_RIR_adjust_sublevel(
1462 CaseExpr *exp = (CaseExpr *)node;
1464 apply_RIR_adjust_sublevel(
1465 (Node *)(exp->args),
1468 apply_RIR_adjust_sublevel(
1469 (Node *)(exp->defresult),
1476 CaseWhen *exp = (CaseWhen *)node;
1478 apply_RIR_adjust_sublevel(
1479 (Node *)(exp->expr),
1482 apply_RIR_adjust_sublevel(
1483 (Node *)(exp->result),
1489 elog(NOTICE, "unknown node tag %d in attribute_used()", nodeTag(node));
1490 elog(NOTICE, "Node is: %s", nodeToString(node));
1499 apply_RIR_view(Node **nodePtr, int rt_index, RangeTblEntry *rte, List *tlist, int *modified, int sublevels_up)
1501 Node *node = *nodePtr;
1506 switch(nodeTag(node)) {
1509 TargetEntry *tle = (TargetEntry *)node;
1512 (Node **)(&(tle->expr)),
1523 Aggreg *agg = (Aggreg *)node;
1526 (Node **)(&(agg->target)),
1537 GroupClause *grp = (GroupClause *)node;
1540 (Node **)(&(grp->entry)),
1551 Expr *exp = (Expr *)node;
1554 (Node **)(&(exp->args)),
1565 Iter *iter = (Iter *)node;
1568 (Node **)(&(iter->iterexpr)),
1579 ArrayRef *ref = (ArrayRef *)node;
1582 (Node **)(&(ref->refupperindexpr)),
1589 (Node **)(&(ref->reflowerindexpr)),
1596 (Node **)(&(ref->refexpr)),
1603 (Node **)(&(ref->refassgnexpr)),
1614 Var *var = (Var *)node;
1616 if (var->varlevelsup == sublevels_up &&
1617 var->varno == rt_index) {
1620 if (var->varattno < 0)
1621 elog(ERROR, "system column %s not available - %s is a view", get_attname(rte->relid, var->varattno), rte->relname);
1622 exp = FindMatchingTLEntry(
1624 get_attname(rte->relid,
1628 *nodePtr = make_null(var->vartype);
1632 exp = copyObject(exp);
1633 if (var->varlevelsup > 0)
1634 apply_RIR_adjust_sublevel(exp, var->varlevelsup);
1651 foreach (l, (List *)node)
1653 (Node **)(&(lfirst(l))),
1664 SubLink *sub = (SubLink *)node;
1667 (Node **)(&(sub->lefthand)),
1675 (Node **)(&(sub->subselect)),
1686 Query *qry = (Query *)node;
1689 (Node **)(&(qry->targetList)),
1697 (Node **)(&(qry->qual)),
1705 (Node **)(&(qry->havingQual)),
1713 (Node **)(&(qry->groupClause)),
1724 CaseExpr *exp = (CaseExpr *)node;
1727 (Node **)(&(exp->args)),
1735 (Node **)(&(exp->defresult)),
1746 CaseWhen *exp = (CaseWhen *)node;
1749 (Node **)(&(exp->expr)),
1757 (Node **)(&(exp->result)),
1767 elog(NOTICE, "unknown node tag %d in apply_RIR_view()", nodeTag(node));
1768 elog(NOTICE, "Node is: %s", nodeToString(node));
1775 ApplyRetrieveRule(Query *parsetree,
1782 Query *rule_action = NULL;
1790 rule_qual = rule->qual;
1793 if (length(rule->actions) > 1) /* ??? because we don't handle
1794 * rules with more than one
1798 rule_action = copyObject(lfirst(rule->actions));
1804 rtable = copyObject(parsetree->rtable);
1807 RangeTblEntry *rte = lfirst(rt);
1810 * this is to prevent add_missing_vars_to_base_rels() from adding
1811 * a bogus entry to the new target list.
1813 rte->inFromCl = false;
1815 rt_length = length(rtable);
1817 rtable = nconc(rtable, copyObject(rule_action->rtable));
1818 parsetree->rtable = rtable;
1820 rule_action->rtable = rtable;
1821 OffsetVarNodes((Node *) rule_qual, rt_length, 0);
1822 OffsetVarNodes((Node *) rule_action, rt_length, 0);
1824 ChangeVarNodes((Node *) rule_qual,
1825 PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
1826 ChangeVarNodes((Node *) rule_action,
1827 PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
1831 apply_RIR_view((Node **) &parsetree, rt_index,
1832 (RangeTblEntry *)nth(rt_index - 1, rtable),
1833 rule_action->targetList, modified, 0);
1834 apply_RIR_view((Node **) &rule_action, rt_index,
1835 (RangeTblEntry *)nth(rt_index - 1, rtable),
1836 rule_action->targetList, modified, 0);
1840 HandleRIRAttributeRule(parsetree, rtable, rule_action->targetList,
1841 rt_index, rule->attrno, modified, &badsql);
1843 if (*modified && !badsql) {
1844 AddQual(parsetree, rule_action->qual);
1845 /* This will only work if the query made to the view defined by the following
1846 * groupClause groups by the same attributes or does not use group at all! */
1847 if (parsetree->groupClause == NULL)
1848 parsetree->groupClause=rule_action->groupClause;
1849 AddHavingQual(parsetree, rule_action->havingQual);
1850 parsetree->hasAggs = (rule_action->hasAggs || parsetree->hasAggs);
1851 parsetree->hasSubLinks = (rule_action->hasSubLinks || parsetree->hasSubLinks);
1857 fireRIRonSubselect(Node *node)
1862 switch(nodeTag(node)) {
1865 TargetEntry *tle = (TargetEntry *)node;
1868 (Node *)(tle->expr));
1874 Aggreg *agg = (Aggreg *)node;
1877 (Node *)(agg->target));
1883 GroupClause *grp = (GroupClause *)node;
1886 (Node *)(grp->entry));
1892 Expr *exp = (Expr *)node;
1895 (Node *)(exp->args));
1901 Iter *iter = (Iter *)node;
1904 (Node *)(iter->iterexpr));
1910 ArrayRef *ref = (ArrayRef *)node;
1913 (Node *)(ref->refupperindexpr));
1915 (Node *)(ref->reflowerindexpr));
1917 (Node *)(ref->refexpr));
1919 (Node *)(ref->refassgnexpr));
1936 foreach (l, (List *)node)
1938 (Node *)(lfirst(l)));
1944 SubLink *sub = (SubLink *)node;
1948 (Node *)(sub->lefthand));
1950 qry = fireRIRrules((Query *)(sub->subselect));
1955 sub->subselect = (Node *) qry;
1961 CaseExpr *exp = (CaseExpr *)node;
1964 (Node *)(exp->args));
1967 (Node *)(exp->defresult));
1973 CaseWhen *exp = (CaseWhen *)node;
1976 (Node *)(exp->expr));
1979 (Node *)(exp->result));
1985 Query *qry = (Query *)node;
1988 (Node *)(qry->targetList));
1991 (Node *)(qry->qual));
1994 (Node *)(qry->havingQual));
1997 (Node *)(qry->groupClause));
2002 elog(NOTICE, "unknown node tag %d in fireRIRonSubselect()", nodeTag(node));
2003 elog(NOTICE, "Node is: %s", nodeToString(node));
2013 * Apply all RIR rules on each rangetable entry in a query
2016 fireRIRrules(Query *parsetree)
2024 RewriteRule RIRonly;
2030 while(rt_index < length(parsetree->rtable)) {
2033 if (!rangeTableEntry_used((Node *)parsetree, rt_index, 0))
2036 rte = nth(rt_index - 1, parsetree->rtable);
2037 rel = heap_openr(rte->relname);
2038 if (rel->rd_rules == NULL) {
2043 rules = rel->rd_rules;
2047 * Collect the RIR rules that we must apply
2049 for (i = 0; i < rules->numLocks; i++) {
2050 rule = rules->rules[i];
2051 if (rule->event != CMD_SELECT)
2054 if (rule->attrno > 0 &&
2055 !attribute_used((Node *)parsetree,
2060 locks = lappend(locks, rule);
2066 checkLockPerms(locks, parsetree, rt_index);
2071 foreach (l, locks) {
2074 RIRonly.event = rule->event;
2075 RIRonly.attrno = rule->attrno;
2076 RIRonly.qual = rule->qual;
2077 RIRonly.actions = rule->actions;
2079 ApplyRetrieveRule(parsetree,
2082 RIRonly.attrno == -1,
2090 fireRIRonSubselect((Node *) parsetree);
2091 modifyAggregQual((Node **) &(parsetree->qual), parsetree);
2098 * idea is to fire regular rules first, then qualified instead
2099 * rules and unqualified instead rules last. Any lemming is counted for.
2102 orderRules(List *locks)
2104 List *regular = NIL;
2105 List *instead_rules = NIL;
2106 List *instead_qualified = NIL;
2111 RewriteRule *rule_lock = (RewriteRule *) lfirst(i);
2113 if (rule_lock->isInstead)
2115 if (rule_lock->qual == NULL)
2116 instead_rules = lappend(instead_rules, rule_lock);
2118 instead_qualified = lappend(instead_qualified, rule_lock);
2121 regular = lappend(regular, rule_lock);
2123 regular = nconc(regular, instead_qualified);
2124 return nconc(regular, instead_rules);
2130 CopyAndAddQual(Query *parsetree,
2136 Query *new_tree = (Query *) copyObject(parsetree);
2137 Node *new_qual = NULL;
2138 Query *rule_action = NULL;
2141 rule_action = lfirst(actions);
2142 if (rule_qual != NULL)
2143 new_qual = (Node *) copyObject(rule_qual);
2144 if (rule_action != NULL)
2149 rtable = new_tree->rtable;
2150 rt_length = length(rtable);
2151 rtable = append(rtable, listCopy(rule_action->rtable));
2152 new_tree->rtable = rtable;
2153 OffsetVarNodes(new_qual, rt_length, 0);
2154 ChangeVarNodes(new_qual, PRS2_CURRENT_VARNO + rt_length, rt_index, 0);
2156 /* XXX -- where current doesn't work for instead nothing.... yet */
2157 AddNotQual(new_tree, new_qual);
2166 * Iterate through rule locks applying rules.
2167 * All rules create their own parsetrees. Instead rules
2168 * with rule qualification save the original parsetree
2169 * and add their negated qualification to it. Real instead
2170 * rules finally throw away the original parsetree.
2172 * remember: reality is for dead birds -- glass
2176 fireRules(Query *parsetree,
2181 List **qual_products)
2184 List *results = NIL;
2187 /* choose rule to fire from list of rules */
2193 locks = orderRules(locks); /* real instead rules last */
2196 RewriteRule *rule_lock = (RewriteRule *) lfirst(i);
2203 * Instead rules change the resultRelation of the query. So the
2204 * permission checks on the initial resultRelation would never be
2205 * done (this is normally done in the executor deep down). So we
2206 * must do it here. The result relations resulting from earlier
2207 * rewrites are already checked against the rules eventrelation
2208 * owner (during matchLocks) and have the skipAcl flag set.
2210 if (rule_lock->isInstead &&
2211 parsetree->commandType != CMD_SELECT)
2217 switch (parsetree->commandType)
2227 rte = (RangeTblEntry *) nth(parsetree->resultRelation - 1,
2231 acl_rc = pg_aclcheck(rte->relname,
2232 GetPgUserName(), reqperm);
2233 if (acl_rc != ACLCHECK_OK)
2235 elog(ERROR, "%s: %s",
2237 aclcheck_error_strings[acl_rc]);
2242 /* multiple rule action time */
2243 *instead_flag = rule_lock->isInstead;
2244 event_qual = rule_lock->qual;
2245 actions = rule_lock->actions;
2246 if (event_qual != NULL && *instead_flag)
2248 Query *qual_product;
2249 RewriteInfo qual_info;
2252 * If there are instead rules with qualifications,
2253 * the original query is still performed. But all
2254 * the negated rule qualifications of the instead
2255 * rules are added so it does it's actions only
2256 * in cases where the rule quals of all instead
2257 * rules are false. Think of it as the default
2258 * action in a case. We save this in *qual_products
2259 * so deepRewriteQuery() can add it to the query
2260 * list after we mangled it up enough.
2263 if (*qual_products == NIL)
2264 qual_product = parsetree;
2266 qual_product = (Query *) nth(0, *qual_products);
2268 qual_info.event = qual_product->commandType;
2269 qual_info.new_varno = length(qual_product->rtable) + 2;
2270 qual_product = CopyAndAddQual(qual_product,
2276 qual_info.rule_action = qual_product;
2278 if (event == CMD_INSERT || event == CMD_UPDATE)
2279 FixNew(&qual_info, qual_product);
2281 *qual_products = lappend(NIL, qual_product);
2286 Query *rule_action = lfirst(r);
2287 Node *rule_qual = copyObject(event_qual);
2289 if (rule_action->commandType == CMD_NOTHING)
2292 /*--------------------------------------------------
2293 * We copy the qualifications of the parsetree
2294 * to the action and vice versa. So force
2295 * hasSubLinks if one of them has it.
2297 * As of 6.4 only parsetree qualifications can
2298 * have sublinks. If this changes, we must make
2299 * this a node lookup at the end of rewriting.
2302 *--------------------------------------------------
2304 if (parsetree->hasSubLinks && !rule_action->hasSubLinks)
2306 rule_action = copyObject(rule_action);
2307 rule_action->hasSubLinks = TRUE;
2309 if (!parsetree->hasSubLinks && rule_action->hasSubLinks)
2311 parsetree->hasSubLinks = TRUE;
2314 /*--------------------------------------------------
2316 * Rewrite current.attribute or current to tuple variable
2317 * this appears to be done in parser?
2318 *--------------------------------------------------
2320 info = gatherRewriteMeta(parsetree, rule_action, rule_qual,
2321 rt_index, event, instead_flag);
2323 /* handle escapable cases, or those handled by other code */
2332 if (info->action == info->event &&
2333 info->event == CMD_SELECT)
2337 * Event Qualification forces copying of parsetree and
2338 * splitting into two queries one w/rule_qual, one w/NOT
2339 * rule_qual. Also add user query qual onto rule action
2341 qual = parsetree->qual;
2342 AddQual(info->rule_action, qual);
2344 if (info->rule_qual != NULL)
2345 AddQual(info->rule_action, info->rule_qual);
2347 /*--------------------------------------------------
2349 * Rewrite new.attribute w/ right hand side of target-list
2350 * entry for appropriate field name in insert/update
2351 *--------------------------------------------------
2353 if ((info->event == CMD_INSERT) || (info->event == CMD_UPDATE))
2354 FixNew(info, parsetree);
2356 /*--------------------------------------------------
2358 * rewriting due to retrieve rules
2359 *--------------------------------------------------
2361 info->rule_action->rtable = info->rt;
2363 ProcessRetrieveQuery(info->rule_action, info->rt,
2364 &orig_instead_flag, TRUE);
2367 /*--------------------------------------------------
2369 * Simplify? hey, no algorithm for simplification... let
2370 * the planner do it.
2371 *--------------------------------------------------
2373 results = lappend(results, info->rule_action);
2379 * If this was an unqualified instead rule,
2380 * throw away an eventually saved 'default' parsetree
2383 if (event_qual == NULL && *instead_flag)
2384 *qual_products = NIL;
2392 RewriteQuery(Query *parsetree, bool *instead_flag, List **qual_products)
2395 List *product_queries = NIL;
2396 int result_relation = 0;
2397 RangeTblEntry *rt_entry;
2398 Relation rt_entry_relation = NULL;
2399 RuleLock *rt_entry_locks = NULL;
2401 Assert(parsetree != NULL);
2403 event = parsetree->commandType;
2406 * SELECT rules are handled later when we have all the
2407 * queries that should get executed
2409 if (event == CMD_SELECT)
2413 * Utilities aren't rewritten at all - why is this here?
2415 if (event == CMD_UTILITY)
2419 * only for a delete may the targetlist be NULL
2421 if (event != CMD_DELETE)
2422 Assert(parsetree->targetList != NULL);
2424 result_relation = parsetree->resultRelation;
2427 * the statement is an update, insert or delete - fire rules
2430 rt_entry = rt_fetch(result_relation, parsetree->rtable);
2431 rt_entry_relation = heap_openr(rt_entry->relname);
2432 rt_entry_locks = rt_entry_relation->rd_rules;
2433 heap_close(rt_entry_relation);
2435 if (rt_entry_locks != NULL)
2438 matchLocks(event, rt_entry_locks, result_relation, parsetree);
2441 fireRules(parsetree,
2449 return product_queries;
2455 * to avoid infinite recursion, we restrict the number of times a query
2456 * can be rewritten. Detecting cycles is left for the reader as an excercise.
2458 #ifndef REWRITE_INVOKE_MAX
2459 #define REWRITE_INVOKE_MAX 10
2462 static int numQueryRewriteInvoked = 0;
2465 * deepRewriteQuery -
2466 * rewrites the query and apply the rules again on the queries rewritten
2469 deepRewriteQuery(Query *parsetree)
2472 List *rewritten = NIL;
2475 List *qual_products = NIL;
2479 if (++numQueryRewriteInvoked > REWRITE_INVOKE_MAX)
2481 elog(ERROR, "query rewritten %d times, may contain cycles",
2482 numQueryRewriteInvoked - 1);
2486 result = RewriteQuery(parsetree, &instead, &qual_products);
2490 Query *pt = lfirst(n);
2491 List *newstuff = NIL;
2493 newstuff = deepRewriteQuery(pt);
2494 if (newstuff != NIL)
2495 rewritten = nconc(rewritten, newstuff);
2499 * qual_products are the original query with the negated
2500 * rule qualification of an instead rule
2503 if (qual_products != NIL)
2504 rewritten = nconc(rewritten, qual_products);
2507 * The original query is appended last if not instead
2508 * because update and delete rule actions might not do
2509 * anything if they are invoked after the update or
2510 * delete is performed. The command counter increment
2511 * between the query execution makes the deleted (and
2512 * maybe the updated) tuples disappear so the scans
2513 * for them in the rule actions cannot find them.
2517 rewritten = lappend(rewritten, parsetree);
2528 QueryRewriteOne(Query *parsetree)
2530 numQueryRewriteInvoked = 0;
2533 * take a deep breath and apply all the rewrite rules - ay
2535 return deepRewriteQuery(parsetree);
2540 * RewritePreprocessQuery -
2541 * adjust details in the parsetree, the rule system
2546 RewritePreprocessQuery(Query *parsetree)
2549 * if the query has a resultRelation, reassign the
2550 * result domain numbers to the attribute numbers in the
2551 * target relation. FixNew() depends on it when replacing
2552 * *new* references in a rule action by the expressions
2553 * from the rewritten query.
2556 if (parsetree->resultRelation > 0)
2564 rte = (RangeTblEntry *) nth(parsetree->resultRelation - 1,
2566 rd = heap_openr(rte->relname);
2568 foreach(tl, parsetree->targetList)
2570 tle = (TargetEntry *) lfirst(tl);
2571 resdomno = attnameAttNum(rd, tle->resdom->resname);
2572 tle->resdom->resno = resdomno;
2582 * rewrite one query via query rewrite system, possibly returning 0
2586 QueryRewrite(Query *parsetree)
2589 List *results = NIL;
2596 * There still seems something broken with the resdom numbers
2597 * so we reassign them first.
2599 RewritePreprocessQuery(parsetree);
2604 * Apply all non-SELECT rules possibly getting 0 or many queries
2606 querylist = QueryRewriteOne(parsetree);
2611 * Apply all the RIR rules on each query
2613 foreach (l, querylist) {
2614 query = (Query *)lfirst(l);
2615 results = lappend(results, fireRIRrules(query));