From 96bd67f61dac3d383fe775d25213d0c1d3d3a179 Mon Sep 17 00:00:00 2001 From: Tom Lane Date: Sat, 6 Jan 2001 04:14:35 +0000 Subject: [PATCH] Bring CREATE RULE reference page into some semblance of agreement with what's actually implemented. --- doc/src/sgml/ref/create_rule.sgml | 235 ++++++++++-------------------- 1 file changed, 81 insertions(+), 154 deletions(-) diff --git a/doc/src/sgml/ref/create_rule.sgml b/doc/src/sgml/ref/create_rule.sgml index 52b2e9117a..aeaddd9a20 100644 --- a/doc/src/sgml/ref/create_rule.sgml +++ b/doc/src/sgml/ref/create_rule.sgml @@ -1,5 +1,5 @@ @@ -20,17 +20,27 @@ Postgres documentation - 1999-07-20 + 2001-01-05 CREATE RULE name AS ON event TO object [ WHERE condition ] - DO [ INSTEAD ] [ action | NOTHING ] + DO [ INSTEAD ] action + +where action can be: + +NOTHING +| +query +| +( query ; query ... ) +| +[ query ; query ... ] - 1998-09-11 + 2001-01-05 Inputs @@ -50,9 +60,9 @@ CREATE RULE <replaceable class="parameter">name</replaceable> AS ON <replaceable <term><replaceable class="parameter">event</replaceable></term> <listitem> <para> - Event is one of <literal>select</literal>, - <literal>update</literal>, <literal>delete</literal> - or <literal>insert</literal>. + Event is one of <literal>SELECT</literal>, + <literal>UPDATE</literal>, <literal>DELETE</literal> + or <literal>INSERT</literal>. </para> </listitem> </varlistentry> @@ -62,7 +72,9 @@ CREATE RULE <replaceable class="parameter">name</replaceable> AS ON <replaceable <para> Object is either <replaceable class="parameter">table</replaceable> or <replaceable class="parameter">table</replaceable>.<replaceable - class="parameter">column</replaceable>. + class="parameter">column</replaceable>. (Currently, only the + <replaceable class="parameter">table</replaceable> form is + actually implemented.) </para> </listitem> </varlistentry> @@ -70,24 +82,38 @@ CREATE RULE <replaceable class="parameter">name</replaceable> AS ON <replaceable <term><replaceable class="parameter">condition</replaceable></term> <listitem> <para> - Any SQL WHERE clause, <literal>new</literal> or - <literal>old</literal>, can appear instead of an instance - variable whenever an instance variable is permissible in SQL. + Any SQL boolean-condition expression. The condition expression may not + refer to any tables except <literal>new</literal> and + <literal>old</literal>. </para> </listitem> </varlistentry> <varlistentry> - <term><replaceable class="parameter">action</replaceable></term> + <term><replaceable class="parameter">query</replaceable></term> <listitem> <para> - Any SQL statement, <literal>new</literal> or - <literal>old</literal>, can appear instead of an instance - variable whenever an instance variable is permissible in SQL. + The query or queries making up the + <replaceable class="PARAMETER">action</replaceable> + can be any SQL <literal>SELECT</literal>, <literal>INSERT</literal>, + <literal>UPDATE</literal>, <literal>DELETE</literal>, or + <literal>NOTIFY</literal> statement. </para> </listitem> </varlistentry> </variablelist> </para> + + <para> + Within the <replaceable class="parameter">condition</replaceable> + and <replaceable class="PARAMETER">action</replaceable>, the special + table names <literal>new</literal> and <literal>old</literal> may be + used to refer to values in the referenced table (the + <replaceable class="parameter">object</replaceable>). + <literal>new</literal> is valid in ON INSERT and ON UPDATE rules + to refer to the new row being inserted or updated. + <literal>old</literal> is valid in ON SELECT, ON UPDATE, and ON DELETE + rules to refer to the existing row being selected, updated, or deleted. + </para> </refsect2> <refsect2 id="R2-SQL-CREATERULE-2"> @@ -127,26 +153,42 @@ CREATE The <productname>Postgres</productname> <firstterm>rule system</firstterm> allows one to define an alternate action to be performed on inserts, updates, or deletions - from database tables or classes. Currently, rules are used to - implement table views. + from database tables. Rules are used to + implement table views as well. </para> <para> - The semantics of a rule is that at the time an individual instance is + The semantics of a rule is that at the time an individual instance (row) + is accessed, inserted, updated, or deleted, there is an old instance (for selects, updates and deletes) and a new instance (for inserts and - updates). - If the <replaceable class="parameter">event</replaceable> - specified in the ON clause and the + updates). All the rules for the given event type and the given target + object (table) are examined, in an unspecified order. If the <replaceable class="parameter">condition</replaceable> specified in the - WHERE clause are true for the old instance, the + WHERE clause (if any) is true, the <replaceable class="parameter">action</replaceable> part of the rule is - executed. First, however, values from fields in the old instance - and/or the new instance are substituted for + executed. The <replaceable class="parameter">action</replaceable> is + done instead of the original query if INSTEAD is specified; otherwise + it is done before the original query is performed. + Within both the <replaceable class="parameter">condition</replaceable> + and <replaceable class="parameter">action</replaceable>, values from + fields in the old instance and/or the new instance are substituted for <literal>old.</literal><replaceable class="parameter">attribute-name</replaceable> and <literal>new.</literal><replaceable class="parameter">attribute-name</replaceable>. </para> + <para> + The <replaceable class="parameter">action</replaceable> part of the rule + can consist of one or more queries. To write multiple queries, surround + them with either parentheses or square brackets. Such queries will be + performed in the specified order (whereas there are no guarantees about + the execution order of multiple rules for an object). The + <replaceable class="parameter">action</replaceable> can also be NOTHING + indicating no action. Thus, a DO INSTEAD NOTHING rule suppresses the + original query from executing (when its condition is true); a DO NOTHING + rule is useless. + </para> + <para> The <replaceable class="parameter">action</replaceable> part of the rule executes with the same command and transaction identifier as the user @@ -155,47 +197,29 @@ CREATE <refsect2 id="R2-SQL-CREATERULE-3"> <refsect2info> - <date>1998-09-11</date> + <date>2001-01-05</date> </refsect2info> <title> Notes - A caution about SQL rules is in order. If the same class name - or instance variable appears in the - event, - condition and - action parts of a rule, - they are all considered different tuple variables. More accurately, - new and old are the only tuple - variables that are shared between these clauses. For example, the following - two rules have the same semantics: - -ON UPDATE TO emp.salary WHERE emp.name = "Joe" - DO - UPDATE emp SET ... WHERE ... - - - -ON UPDATE TO emp-1.salary WHERE emp-2.name = "Joe" - DO - UPDATE emp-3 SET ... WHERE ... - + Presently, ON SELECT rules must be unconditional INSTEAD rules and must + have actions that consist of a single SELECT query. Thus, an ON SELECT + rule effectively turns the object table into a view, whose visible + contents are the rows returned by the rule's SELECT query rather than + whatever had been stored in the table (if anything). It is considered + better style to write a CREATE VIEW command than to create a table and + define an ON SELECT rule for it. + - Each rule can have the optional tag INSTEAD. - Without - this tag, action will be - performed in addition to the user command when the - event in the - condition part of the rule - occurs. Alternately, the - action part will be done - instead of the user command. In this latter case, the - action can be the keyword - NOTHING. + + You must have rule definition access to a class in order + to define a rule on it. Use GRANT + and REVOKE to change permissions. + - It is very important to note to avoid circular rules. + It is very important to take care to avoid circular rules. For example, though each of the following two rule definitions are accepted by Postgres, the @@ -226,105 +250,8 @@ SELECT * FROM emp; - - - You must have rule definition access to a class in order - to define a rule on it. Use GRANT - and REVOKE to change permissions. - - - - The object in a SQL rule cannot be an array reference and - cannot have parameters. - - - - Aside from the "oid" field, system attributes cannot be - referenced anywhere in a rule. Among other things, this - means that functions of instances (e.g., foo(emp) where - emp is a class) cannot be called anywhere in a rule. - - - - The rule system stores the rule text and query plans as - text attributes. This implies that creation of rules may - fail if the rule plus its various internal representations - exceed some value that is on the order of one page (8KB). - - - - - Usage - - - Make Sam get the same salary adjustment as Joe: - - -CREATE RULE example_1 AS - ON UPDATE emp.salary WHERE old.name = "Joe" - DO - UPDATE emp - SET salary = new.salary - WHERE emp.name = "Sam"; - - - At the time Joe receives a salary adjustment, the event - will become true and Joe's old instance and proposed - new instance are available to the execution routines. - Hence, his new salary is substituted into the action part - of the rule which is subsequently executed. This propagates - Joe's salary on to Sam. - - - Make Bill get Joe's salary when it is accessed: - -CREATE RULE example_2 AS - ON SELECT TO EMP.salary - WHERE old.name = "Bill" - DO INSTEAD - SELECT emp.salary - FROM emp - WHERE emp.name = "Joe"; - - - - Deny Joe access to the salary of employees in the shoe - department (current_user returns the name of - the current user): - -CREATE RULE example_3 AS - ON - SELECT TO emp.salary - WHERE old.dept = "shoe" AND current_user = "Joe" - DO INSTEAD NOTHING; - - - - Create a view of the employees working in the toy department: - -CREATE toyemp(name = char16, salary = int4); - -CREATE RULE example_4 AS - ON SELECT TO toyemp - DO INSTEAD - SELECT emp.name, emp.salary - FROM emp - WHERE emp.dept = "toy"; - - - - All new employees must make 5,000 or less: - -CREATE RULE example_5 AS - ON INERT TO emp WHERE new.salary > 5000 - DO - UPDATE emp SET salary = 5000 - WHERE emp.oid = new.oid; - - - -- 2.40.0