view.c 7.8 KB
Newer Older
1 2 3
/*-------------------------------------------------------------------------
 *
 * view.c--
4
 *	  use rewrite rules to construct views
5 6 7
 *
 * Copyright (c) 1994, Regents of the University of California
 *
8
 *  $Id: view.c,v 1.31 1999/02/03 21:16:06 momjian Exp $
9 10 11
 *
 *-------------------------------------------------------------------------
 */
M
 
Marc G. Fournier 已提交
12
#include <stdio.h>			
13
#include <string.h>
M
Marc G. Fournier 已提交
14 15 16 17 18 19 20 21

#include <postgres.h>

#include <catalog/heap.h>
#include <access/heapam.h>
#include <access/xact.h>
#include <utils/builtins.h>
#include <nodes/relation.h>
22 23
#include <parser/parse_relation.h>
#include <parser/parse_type.h>
M
Marc G. Fournier 已提交
24 25 26 27 28
#include <rewrite/rewriteDefine.h>
#include <rewrite/rewriteHandler.h>
#include <rewrite/rewriteManip.h>
#include <rewrite/rewriteRemove.h>
#include <commands/creatinh.h>
29
#include <commands/view.h>
30 31 32 33 34 35 36 37 38 39 40 41 42

/*---------------------------------------------------------------------
 * DefineVirtualRelation
 *
 * Create the "view" relation.
 * `DefineRelation' does all the work, we just provide the correct
 * arguments!
 *
 * If the relation already exists, then 'DefineRelation' will abort
 * the xact...
 *---------------------------------------------------------------------
 */
static void
43
DefineVirtualRelation(char *relname, List *tlist)
44
{
45 46 47 48 49 50 51
	CreateStmt	createStmt;
	List	   *attrList,
			   *t;
	TargetEntry *entry;
	Resdom	   *res;
	char	   *resname;
	char	   *restypename;
52 53 54 55 56 57 58 59 60 61 62 63

	/*
	 * create a list with one entry per attribute of this relation. Each
	 * entry is a two element list. The first element is the name of the
	 * attribute (a string) and the second the name of the type (NOTE: a
	 * string, not a type id!).
	 */
	attrList = NIL;
	if (tlist != NIL)
	{
		foreach(t, tlist)
		{
64 65
			ColumnDef  *def = makeNode(ColumnDef);
			TypeName   *typename;
66 67 68 69 70 71 72

			/*
			 * find the names of the attribute & its type
			 */
			entry = lfirst(t);
			res = entry->resdom;
			resname = res->resname;
73
			restypename = typeidTypeName(res->restype);
74 75 76 77

			typename = makeNode(TypeName);

			typename->name = pstrdup(restypename);
78 79
			typename->typmod = res->restypmod;

80 81 82 83 84 85 86 87 88
			def->colname = pstrdup(resname);

			def->typename = typename;

			def->is_not_null = false;
			def->defval = (char *) NULL;

			attrList = lappend(attrList, def);
		}
89
	}
90
	else
91
		elog(ERROR, "attempted to define virtual relation with no attrs");
92 93 94 95 96 97

	/*
	 * now create the parametesr for keys/inheritance etc. All of them are
	 * nil...
	 */
	createStmt.relname = relname;
98
	createStmt.istemp = false;
99 100 101 102 103 104 105 106
	createStmt.tableElts = attrList;
/*	  createStmt.tableType = NULL;*/
	createStmt.inhRelnames = NIL;
	createStmt.constraints = NIL;

	/*
	 * finally create the relation...
	 */
107
	DefineRelation(&createStmt, RELKIND_RELATION);
108
}
109 110 111 112 113 114 115 116 117

/*------------------------------------------------------------------
 * makeViewRetrieveRuleName
 *
 * Given a view name, returns the name for the 'on retrieve to "view"'
 * rule.
 * This routine is called when defining/removing a view.
 *------------------------------------------------------------------
 */
118
char *
119 120
MakeRetrieveViewRuleName(char *viewName)
{
121
	char	   *buf;
122 123

	buf = palloc(strlen(viewName) + 5);
M
 
Marc G. Fournier 已提交
124
	snprintf(buf, strlen(viewName) + 5, "_RET%s", viewName);
125

126
	return buf;
127 128 129
}

static RuleStmt *
130
FormViewRetrieveRule(char *viewName, Query *viewParse)
131
{
132 133 134
	RuleStmt   *rule;
	char	   *rname;
	Attr	   *attr;
135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153

	/*
	 * Create a RuleStmt that corresponds to the suitable rewrite rule
	 * args for DefineQueryRewrite();
	 */
	rule = makeNode(RuleStmt);
	rname = MakeRetrieveViewRuleName(viewName);

	attr = makeNode(Attr);
	attr->relname = pstrdup(viewName);
/*	  attr->refname = pstrdup(viewName);*/
	rule->rulename = pstrdup(rname);
	rule->whereClause = NULL;
	rule->event = CMD_SELECT;
	rule->object = attr;
	rule->instead = true;
	rule->actions = lcons(viewParse, NIL);

	return rule;
154 155 156
}

static void
157
DefineViewRules(char *viewName, Query *viewParse)
158
{
159
	RuleStmt   *retrieve_rule = NULL;
160

161
#ifdef NOTYET
162 163 164
	RuleStmt   *replace_rule = NULL;
	RuleStmt   *append_rule = NULL;
	RuleStmt   *delete_rule = NULL;
165

166
#endif
167

168
	retrieve_rule = FormViewRetrieveRule(viewName, viewParse);
169

170
#ifdef NOTYET
171

172 173 174
	replace_rule = FormViewReplaceRule(viewName, viewParse);
	append_rule = FormViewAppendRule(viewName, viewParse);
	delete_rule = FormViewDeleteRule(viewName, viewParse);
175

176
#endif
177 178

	DefineQueryRewrite(retrieve_rule);
179 180

#ifdef NOTYET
181 182 183
	DefineQueryRewrite(replace_rule);
	DefineQueryRewrite(append_rule);
	DefineQueryRewrite(delete_rule);
184
#endif
185 186

}
187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206

/*---------------------------------------------------------------
 * UpdateRangeTableOfViewParse
 *
 * Update the range table of the given parsetree.
 * This update consists of adding two new entries IN THE BEGINNING
 * of the range table (otherwise the rule system will die a slow,
 * horrible and painful death, and we do not want that now, do we?)
 * one for the CURRENT relation and one for the NEW one (both of
 * them refer in fact to the "view" relation).
 *
 * Of course we must also increase the 'varnos' of all the Var nodes
 * by 2...
 *
 * NOTE: these are destructive changes. It would be difficult to
 * make a complete copy of the parse tree and make the changes
 * in the copy.
 *---------------------------------------------------------------
 */
static void
207
UpdateRangeTableOfViewParse(char *viewName, Query *viewParse)
208
{
209 210 211 212
	List	   *old_rt;
	List	   *new_rt;
	RangeTblEntry *rt_entry1,
			   *rt_entry2;
213 214 215 216

	/*
	 * first offset all var nodes by 2
	 */
B
Bruce Momjian 已提交
217 218
	OffsetVarNodes((Node *) viewParse->targetList, 2, 0);
	OffsetVarNodes(viewParse->qual, 2, 0);
219

B
Bruce Momjian 已提交
220
	OffsetVarNodes(viewParse->havingQual, 2, 0);
221

222

223 224 225 226 227 228 229 230 231
	/*
	 * find the old range table...
	 */
	old_rt = viewParse->rtable;

	/*
	 * create the 2 new range table entries and form the new range
	 * table... CURRENT first, then NEW....
	 */
232
	rt_entry1 = addRangeTableEntry(NULL, (char *) viewName, "*CURRENT*",
233
						   FALSE, FALSE);
234
	rt_entry2 = addRangeTableEntry(NULL, (char *) viewName, "*NEW*",
235
						   FALSE, FALSE);
236 237 238 239 240 241 242 243
	new_rt = lcons(rt_entry2, old_rt);
	new_rt = lcons(rt_entry1, new_rt);

	/*
	 * Now the tricky part.... Update the range table in place... Be
	 * careful here, or hell breaks loooooooooooooOOOOOOOOOOOOOOOOOOSE!
	 */
	viewParse->rtable = new_rt;
244 245 246 247 248
}

/*-------------------------------------------------------------------
 * DefineView
 *
249 250 251 252 253 254 255
 *		- takes a "viewname", "parsetree" pair and then
 *		1)		construct the "virtual" relation
 *		2)		commit the command but NOT the transaction,
 *				so that the relation exists
 *				before the rules are defined.
 *		2)		define the "n" rules specified in the PRS2 paper
 *				over the "virtual" relation
256 257 258
 *-------------------------------------------------------------------
 */
void
259
DefineView(char *viewName, Query *viewParse)
260
{
261
	List	   *viewTlist;
262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285

	viewTlist = viewParse->targetList;

	/*
	 * Create the "view" relation NOTE: if it already exists, the xaxt
	 * will be aborted.
	 */
	DefineVirtualRelation(viewName, viewTlist);

	/*
	 * The relation we have just created is not visible to any other
	 * commands running with the same transaction & command id. So,
	 * increment the command id counter (but do NOT pfree any memory!!!!)
	 */
	CommandCounterIncrement();

	/*
	 * The range table of 'viewParse' does not contain entries for the
	 * "CURRENT" and "NEW" relations. So... add them! NOTE: we make the
	 * update in place! After this call 'viewParse' will never be what it
	 * used to be...
	 */
	UpdateRangeTableOfViewParse(viewName, viewParse);
	DefineViewRules(viewName, viewParse);
286 287 288 289 290 291 292 293 294 295 296
}

/*------------------------------------------------------------------
 * RemoveView
 *
 * Remove a view given its name
 *------------------------------------------------------------------
 */
void
RemoveView(char *viewName)
{
297
	char	   *rname;
298 299 300 301 302 303 304 305 306 307 308 309 310 311 312

	/*
	 * first remove all the "view" rules... Currently we only have one!
	 */
	rname = MakeRetrieveViewRuleName(viewName);
	RemoveRewriteRule(rname);

	/*
	 * we don't really need that, but just in case...
	 */
	CommandCounterIncrement();

	/*
	 * now remove the relation.
	 */
313
	heap_destroy_with_catalog(viewName);
314
	pfree(rname);
315
}