view.c 7.8 KB
Newer Older
1 2 3
/*-------------------------------------------------------------------------
 *
 * view.c--
4
 *	  use rewrite rules to construct views
5 6 7 8 9
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
10
 *	  $Header: /cvsroot/pgsql/src/backend/commands/view.c,v 1.22 1998/06/15 19:28:17 momjian Exp $
11 12 13
 *
 *-------------------------------------------------------------------------
 */
14
#include <stdio.h>				/* for sprintf() */
15
#include <string.h>
M
Marc G. Fournier 已提交
16 17 18 19 20 21 22 23

#include <postgres.h>

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

/*---------------------------------------------------------------------
 * 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
45
DefineVirtualRelation(char *relname, List *tlist)
46
{
47 48 49 50 51 52 53
	CreateStmt	createStmt;
	List	   *attrList,
			   *t;
	TargetEntry *entry;
	Resdom	   *res;
	char	   *resname;
	char	   *restypename;
54 55 56 57 58 59 60 61 62 63 64 65

	/*
	 * 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)
		{
66 67
			ColumnDef  *def = makeNode(ColumnDef);
			TypeName   *typename;
68 69 70 71 72 73 74

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

			typename = makeNode(TypeName);

			typename->name = pstrdup(restypename);
80 81
			typename->typmod = res->restypmod;

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

			def->typename = typename;

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

			attrList = lappend(attrList, def);
		}
91
	}
92
	else
93
		elog(ERROR, "attempted to define virtual relation with no attrs");
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109

	/*
	 * now create the parametesr for keys/inheritance etc. All of them are
	 * nil...
	 */
	createStmt.relname = relname;
	createStmt.tableElts = attrList;
/*	  createStmt.tableType = NULL;*/
	createStmt.inhRelnames = NIL;
	createStmt.constraints = NIL;

	/*
	 * finally create the relation...
	 */
	DefineRelation(&createStmt);
}
110 111 112 113 114 115 116 117 118

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

	buf = palloc(strlen(viewName) + 5);
	sprintf(buf, "_RET%s", viewName);
126

127
	return buf;
128 129 130
}

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

	/*
	 * 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;
155 156 157
}

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

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

167
#endif
168 169 170 171

	retrieve_rule =
		FormViewRetrieveRule(viewName, viewParse);

172
#ifdef NOTYET
173 174 175 176 177 178 179 180

	replace_rule =
		FormViewReplaceRule(viewName, viewParse);
	append_rule =
		FormViewAppendRule(viewName, viewParse);
	delete_rule =
		FormViewDeleteRule(viewName, viewParse);

181
#endif
182 183

	DefineQueryRewrite(retrieve_rule);
184 185

#ifdef NOTYET
186 187 188
	DefineQueryRewrite(replace_rule);
	DefineQueryRewrite(append_rule);
	DefineQueryRewrite(delete_rule);
189
#endif
190 191

}
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211

/*---------------------------------------------------------------
 * 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
212
UpdateRangeTableOfViewParse(char *viewName, Query *viewParse)
213
{
214 215 216 217
	List	   *old_rt;
	List	   *new_rt;
	RangeTblEntry *rt_entry1,
			   *rt_entry2;
218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235

	/*
	 * first offset all var nodes by 2
	 */
	OffsetVarNodes((Node *) viewParse->targetList, 2);
	OffsetVarNodes(viewParse->qual, 2);

	/*
	 * 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....
	 */
	rt_entry1 =
		addRangeTableEntry(NULL, (char *) viewName, "*CURRENT*",
236
						   FALSE, FALSE);
237 238
	rt_entry2 =
		addRangeTableEntry(NULL, (char *) viewName, "*NEW*",
239
						   FALSE, FALSE);
240 241 242 243 244 245 246 247
	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;
248 249 250 251 252
}

/*-------------------------------------------------------------------
 * DefineView
 *
253 254 255 256 257 258 259
 *		- 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
260 261 262
 *-------------------------------------------------------------------
 */
void
263
DefineView(char *viewName, Query *viewParse)
264
{
265
	List	   *viewTlist;
266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289

	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);
290 291 292 293 294 295 296 297 298 299 300
}

/*------------------------------------------------------------------
 * RemoveView
 *
 * Remove a view given its name
 *------------------------------------------------------------------
 */
void
RemoveView(char *viewName)
{
301
	char	   *rname;
302 303 304 305 306 307 308 309 310 311 312 313 314 315 316

	/*
	 * 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.
	 */
317
	heap_destroy_with_catalog(viewName);
318
	pfree(rname);
319
}