rules.sgml
上传用户:blenddy
上传日期:2007-01-07
资源大小:6495k
文件大小:76k
源码类别:

数据库系统

开发平台:

Unix_Linux

  1. <Chapter Id="rules">
  2. <Title>The <ProductName>Postgres</ProductName> Rule System</Title>
  3. <Para>
  4.      Production rule systems are conceptually simple, but
  5.      there are many subtle points involved in actually using
  6.      them. Some of these points and
  7.      the theoretical foundations of the <ProductName>Postgres</ProductName>
  8.      rule system can be found in
  9. [<XRef LinkEnd="STON90b" EndTerm="STON90b">].
  10. </Para>
  11. <Para>
  12.      Some other database systems define active database rules. These
  13.      are usually stored procedures and triggers and are implemented
  14.      in <ProductName>Postgres</ProductName> as functions and triggers.
  15. </Para>
  16. <Para>
  17.      The query rewrite rule system (the "rule system" from now on) 
  18.      is totally different from stored procedures and triggers.
  19.      It modifies  queries  to
  20.      take rules into consideration, and then passes the modified 
  21.      query to the query optimizer for  execution.   It
  22.      is  very powerful, and can be used for many things such
  23.      as query language procedures, views, and versions.  The
  24.      power  of  this  rule system is discussed in 
  25. [<XRef LinkEnd="ONG90" EndTerm="ONG90">]
  26.  as well as
  27. [<XRef LinkEnd="STON90b" EndTerm="STON90b">].
  28. </para>
  29. <Sect1>
  30. <Title>What is a Querytree?</Title>
  31. <Para>
  32.     To understand how the rule system works it is necessary to know
  33.     when it is invoked and what it's input and results are.
  34. </Para>
  35. <Para>
  36.     The rule system is located between the query parser and the optimizer.
  37.     It takes the output of the parser, one querytree, and the rewrite
  38.     rules from the <FileName>pg_rewrite</FileName> catalog, which are
  39.     querytrees too with some extra information, and creates zero or many
  40.     querytrees as result. So it's input and output are always things
  41.     the parser itself could have produced and thus, anything it sees
  42.     is basically representable as an <Acronym>SQL</Acronym> statement.
  43. </Para>
  44. <Para>
  45.     Now what is a querytree? It is an internal representation of an
  46.     <Acronym>SQL</Acronym> statement where the single parts that built
  47.     it are stored separately. These querytrees are visible when starting
  48.     the <ProductName>Postgres</ProductName> backend with debuglevel 4
  49.     and typing queries into the interactive backend interface. The rule
  50.     actions in the <FileName>pg_rewrite</FileName> system catalog are
  51.     also stored as querytrees. They are not formatted like the debug
  52.     output, but they contain exactly the same information.
  53. </Para>
  54. <Para>
  55.     Reading a querytree requires some experience and it was a hard
  56.     time when I started to work on the rule system. I can remember
  57.     that I was standing at the coffee machine and I saw the cup
  58.     in a targetlist, water and coffee powder in a rangetable and all
  59.     the buttons in a qualification expression. Since 
  60.     <Acronym>SQL</Acronym> representations of querytrees are
  61.     sufficient to understand the rule system, this document will
  62.     not teach how to read them. It might help to learn
  63.     it and the naming conventions are required in the later following
  64.     descriptions.
  65. </Para>
  66. <Sect2>
  67. <Title>The Parts of a Querytree</Title>
  68. <Para>
  69.     When reading the <Acronym>SQL</Acronym> representations of the 
  70.     querytrees in this document it is necessary to be able to identify
  71.     the parts the statement is broken into when it is in the querytree
  72.     structure. The parts of a querytree are
  73. </Para>
  74. <Para>
  75. <VariableList>
  76.     <VarListEntry>
  77.     <Term>
  78.         the commandtype
  79.     </Term>
  80.     <ListItem>
  81.     <Para>
  82.         This is a simple value telling which command
  83. (SELECT, INSERT, UPDATE, DELETE) produced the parsetree.
  84.     </Para>
  85.     </ListItem>
  86.     </VarListEntry>
  87.     <VarListEntry>
  88.     <Term>
  89.         the rangetable
  90.     </Term>
  91.     <ListItem>
  92.     <Para>
  93.         The rangtable is a list of relations that are used in the query.
  94. In a SELECT statement that are the relations given after
  95. the FROM keyword.
  96.     </Para>
  97.     <Para>
  98.         Every rangetable entry identifies a table or view and tells
  99. by which name it is called in the other parts of the query.
  100. In the querytree the rangetable entries are referenced by
  101. index rather than by name, so here it doesn't matter if there
  102. are duplicate names as it would in an <Acronym>SQL</Acronym>
  103. statement. This can happen after the rangetables of rules
  104. have been merged in. The examples in this document will not have
  105. this situation.
  106.     </Para>
  107.     </ListItem>
  108.     </VarListEntry>
  109.     <VarListEntry>
  110.     <Term>
  111.         the resultrelation
  112.     </Term>
  113.     <ListItem>
  114.     <Para>
  115.         This is an index into the rangetable that identifies the
  116. relation where the results of the query go.
  117.     </Para>
  118.     <Para>
  119. SELECT queries
  120. normally don't have a result relation. The special case
  121. of a SELECT INTO is mostly identical to a CREATE TABLE,
  122. INSERT ... SELECT sequence and is not discussed separately
  123. here.
  124.     </Para>
  125.     <Para>
  126.         On INSERT, UPDATE and DELETE queries the resultrelation
  127. is the table (or view!) where the changes take effect.
  128.     </Para>
  129.     </ListItem>
  130.     </VarListEntry>
  131.     <VarListEntry>
  132.     <Term>
  133.         the targetlist
  134.     </Term>
  135.     <ListItem>
  136.     <Para>
  137.         The targetlist is a list of expressions that define the result
  138. of the query. In the case of a SELECT, the expressions are what
  139. builds the final output of the query. They are the expressions
  140. between the SELECT and the FROM keywords (* is just an
  141. abbreviation for all the attribute names of a relation). 
  142.     </Para>
  143.     <Para>
  144.         DELETE queries don't need a targetlist because they don't
  145. produce any result. In fact the optimizer will add a special
  146. entry to the empty targetlist. But this is after the rule
  147. system and will be discussed later. For the rule system the
  148. targetlist is empty.
  149.     </Para>
  150.     <Para>
  151.         In INSERT queries the targetlist describes the new rows that
  152. should go into the resultrelation. Missing columns of the 
  153. resultrelation will be added by the optimizer with a constant
  154. NULL expression. It is the expressions in the VALUES clause
  155. or the ones from the SELECT clause on INSERT ... SELECT.
  156.     </Para>
  157.     <Para>
  158.         On UPDATE queries, it describes the new rows that should
  159. replace the old ones. Here now the optimizer will add missing
  160. columns by inserting expressions that put the values from the
  161. old rows into the new one. And it will add the special entry
  162. like for DELETE too. It is the expressions from the
  163. SET attribute = expression part of the query.
  164.     </Para>
  165.     <Para>
  166.         Every entry in the targetlist contains an expression that can
  167. be a constant value, a variable pointing to an attribute of one
  168. of the relations in the rangetable, a parameter or an expression
  169. tree made of function calls, constants, variables, operators etc.
  170.     </Para>
  171.     </ListItem>
  172.     </VarListEntry>
  173.     <VarListEntry>
  174.     <Term>
  175.         the qualification
  176.     </Term>
  177.     <ListItem>
  178.     <Para>
  179.         The queries qualification is an expression much like one of those
  180. contained in the targetlist entries. The result value of this
  181. expression is a boolean that tells if the operation
  182. (INSERT, UPDATE, DELETE or SELECT) for the final result row should be
  183. executed or not. It is the WHERE clause of an
  184. <Acronym>SQL</Acronym> statement.
  185.     </Para>
  186.     </ListItem>
  187.     </VarListEntry>
  188.     <VarListEntry>
  189.     <Term>
  190.         the others
  191.     </Term>
  192.     <ListItem>
  193.     <Para>
  194.         The other parts of the querytree like the ORDER BY 
  195. clause arent of interest here. The rule system
  196. substitutes entries there while applying rules, but that
  197. doesn't have much to do with the fundamentals of the rule
  198. system. GROUP BY is a special thing when it appears in
  199.         a view definition and still needs to be documented.
  200.     </Para>
  201.     </ListItem>
  202.     </VarListEntry>
  203. </VariableList>
  204. </para>
  205. </Sect2>
  206. </Sect1>
  207. <Sect1>
  208. <Title>Views and the Rule System</Title>
  209. <Sect2>
  210. <Title>Implementation of Views in <ProductName>Postgres</ProductName></Title>
  211. <Para>
  212.     Views in <ProductName>Postgres</ProductName> are implemented
  213.     using the rule system. In fact there is absolutely no difference
  214.     between a
  215. <ProgramListing>
  216.     CREATE VIEW myview AS SELECT * FROM mytab;
  217. </ProgramListing>
  218.     
  219.     compared against the two commands
  220. <ProgramListing>
  221.     CREATE TABLE myview (<Replaceable>same attribute list as for mytab</Replaceable>);
  222.     CREATE RULE "_RETmyview" AS ON SELECT TO myview DO INSTEAD
  223.         SELECT * FROM mytab;
  224. </ProgramListing>
  225.     
  226.     because this is exactly what the CREATE VIEW command does internally.
  227.     This has some side effects. One of them is that
  228.     the information about a view in the <ProductName>Postgres</ProductName>
  229.     system catalogs is exactly the same as it is for a table. So for the
  230.     query parsers, there is absolutely no difference between
  231.     a table and a view. They are the same thing - relations. That is the
  232.     important one for now.
  233. </Para>
  234. </Sect2>
  235. <Sect2>
  236. <Title>How SELECT Rules Work</Title>
  237. <Para>
  238.     Rules ON SELECT are applied to all queries as the
  239.     last step, even if the command
  240.     given is an INSERT, UPDATE or DELETE. And they have different
  241.     semantics from the others in that they modify the parsetree in
  242.     place instead of creating a new one.
  243.     So SELECT rules are described first.
  244. </Para>
  245. <Para>
  246.     Currently, there could be only one action and it must be a
  247.     SELECT action that is INSTEAD. This restriction was required
  248.     to make rules safe enough to open them for ordinary users and
  249.     it restricts rules ON SELECT to real view rules.
  250. </Para>
  251. <Para>
  252.     The example for this document are two join views that do some calculations
  253.     and some more views using them in turn.
  254.     One of the two first views is customized later by adding rules for
  255.     INSERT, UPDATE and DELETE operations so that the final result will
  256.     be a view that behaves like a real table with some magic functionality.
  257.     It is not such a simple example to start from and this makes things
  258.     harder to get into. But it's better to have one example that covers
  259.     all the points discussed step by step rather than having many
  260.     different ones that might mix up in mind.
  261. </Para>
  262. <Para>
  263.     The database needed to play on the examples is named al_bundy.
  264.     You'll see soon why this is the database name. And it needs the
  265.     procedural language PL/pgSQL installed, because
  266.     we need a little min() function returning the lower of 2
  267.     integer values. We create that as
  268. <ProgramListing>
  269.     CREATE FUNCTION min(integer, integer) RETURNS integer AS
  270.         'BEGIN
  271.             IF $1 < $2 THEN
  272.                 RETURN $1;
  273.             END IF;
  274.             RETURN $2;
  275.         END;'
  276.     LANGUAGE 'plpgsql';
  277. </ProgramListing>
  278. </Para>
  279. <Para>
  280.     The real tables we need in the first two rule system descripitons
  281.     are these:
  282. <ProgramListing>
  283.     CREATE TABLE shoe_data (
  284.         shoename   char(10),      -- primary key
  285.         sh_avail   integer,       -- available # of pairs
  286.         slcolor    char(10),      -- preferred shoelace color
  287.         slminlen   float,         -- miminum shoelace length
  288.         slmaxlen   float,         -- maximum shoelace length
  289.         slunit     char(8)        -- length unit
  290.     );
  291.     CREATE TABLE shoelace_data (
  292.         sl_name    char(10),      -- primary key
  293.         sl_avail   integer,       -- available # of pairs
  294.         sl_color   char(10),      -- shoelace color
  295.         sl_len     float,         -- shoelace length
  296.         sl_unit    char(8)        -- length unit
  297.     );
  298.     CREATE TABLE unit (
  299.         un_name    char(8),       -- the primary key
  300.         un_fact    float          -- factor to transform to cm
  301.     );
  302. </ProgramListing>
  303.     I think most of us wear shoes and can realize that this is
  304.     really useful data. Well there are shoes out in the world
  305.     that don't require shoelaces, but this doesn't make Al's
  306.     life easier and so we ignore it.
  307. </Para>
  308. <Para>
  309.     The views are created as
  310. <ProgramListing>
  311.     CREATE VIEW shoe AS
  312.         SELECT sh.shoename,
  313.                sh.sh_avail,
  314.                sh.slcolor,
  315.                sh.slminlen,
  316.                sh.slminlen * un.un_fact AS slminlen_cm,
  317.                sh.slmaxlen,
  318.                sh.slmaxlen * un.un_fact AS slmaxlen_cm,
  319.                sh.slunit
  320.           FROM shoe_data sh, unit un
  321.          WHERE sh.slunit = un.un_name;
  322.     CREATE VIEW shoelace AS
  323.         SELECT s.sl_name,
  324.                s.sl_avail,
  325.                s.sl_color,
  326.                s.sl_len,
  327.                s.sl_unit,
  328.                s.sl_len * u.un_fact AS sl_len_cm
  329.           FROM shoelace_data s, unit u
  330.          WHERE s.sl_unit = u.un_name;
  331.     CREATE VIEW shoe_ready AS
  332.         SELECT rsh.shoename,
  333.                rsh.sh_avail,
  334.                rsl.sl_name,
  335.                rsl.sl_avail,
  336.                min(rsh.sh_avail, rsl.sl_avail) AS total_avail
  337.           FROM shoe rsh, shoelace rsl
  338.          WHERE rsl.sl_color = rsh.slcolor
  339.            AND rsl.sl_len_cm >= rsh.slminlen_cm
  340.            AND rsl.sl_len_cm <= rsh.slmaxlen_cm;
  341. </ProgramListing>
  342.     The CREATE VIEW command for the <Filename>shoelace</Filename> 
  343.     view (which is the simplest one we have)
  344.     will create a relation shoelace and an entry 
  345.     in <FileName>pg_rewrite</FileName>
  346.     that tells that there is a rewrite rule that must be applied
  347.     whenever the relation shoelace is referenced in a queries rangetable.
  348.     The rule has no rule qualification (discussed in the
  349.     non SELECT rules since SELECT rules currently cannot have them) and
  350.     it is INSTEAD. Note that rule qualifications are not the same as
  351.     query qualifications! The rules action has a qualification.
  352. </Para>
  353. <Para>
  354.     The rules action is one querytree that is an exact copy of the
  355.     SELECT statement in the view creation command.
  356.     
  357.     <Note>
  358.     <Title>Note</Title>
  359.     <Para>
  360.     The two extra range
  361.     table entries for NEW and OLD (named *NEW* and *CURRENT* for
  362.     historical reasons in the printed querytree) you can see in
  363.     the <Filename>pg_rewrite</Filename> entry aren't of interest
  364.     for SELECT rules.
  365.     </Para>
  366.     </Note>
  367.     Now we populate <Filename>unit</Filename>, <Filename>shoe_data</Filename>
  368.     and <Filename>shoelace_data</Filename> and Al types the first 
  369.     SELECT in his life:
  370. <ProgramListing>
  371.     al_bundy=> INSERT INTO unit VALUES ('cm', 1.0);
  372.     al_bundy=> INSERT INTO unit VALUES ('m', 100.0);
  373.     al_bundy=> INSERT INTO unit VALUES ('inch', 2.54);
  374.     al_bundy=> 
  375.     al_bundy=> INSERT INTO shoe_data VALUES 
  376.     al_bundy->     ('sh1', 2, 'black', 70.0, 90.0, 'cm');
  377.     al_bundy=> INSERT INTO shoe_data VALUES 
  378.     al_bundy->     ('sh2', 0, 'black', 30.0, 40.0, 'inch');
  379.     al_bundy=> INSERT INTO shoe_data VALUES 
  380.     al_bundy->     ('sh3', 4, 'brown', 50.0, 65.0, 'cm');
  381.     al_bundy=> INSERT INTO shoe_data VALUES 
  382.     al_bundy->     ('sh4', 3, 'brown', 40.0, 50.0, 'inch');
  383.     al_bundy=> 
  384.     al_bundy=> INSERT INTO shoelace_data VALUES 
  385.     al_bundy->     ('sl1', 5, 'black', 80.0, 'cm');
  386.     al_bundy=> INSERT INTO shoelace_data VALUES 
  387.     al_bundy->     ('sl2', 6, 'black', 100.0, 'cm');
  388.     al_bundy=> INSERT INTO shoelace_data VALUES 
  389.     al_bundy->     ('sl3', 0, 'black', 35.0 , 'inch');
  390.     al_bundy=> INSERT INTO shoelace_data VALUES 
  391.     al_bundy->     ('sl4', 8, 'black', 40.0 , 'inch');
  392.     al_bundy=> INSERT INTO shoelace_data VALUES 
  393.     al_bundy->     ('sl5', 4, 'brown', 1.0 , 'm');
  394.     al_bundy=> INSERT INTO shoelace_data VALUES 
  395.     al_bundy->     ('sl6', 0, 'brown', 0.9 , 'm');
  396.     al_bundy=> INSERT INTO shoelace_data VALUES 
  397.     al_bundy->     ('sl7', 7, 'brown', 60 , 'cm');
  398.     al_bundy=> INSERT INTO shoelace_data VALUES 
  399.     al_bundy->     ('sl8', 1, 'brown', 40 , 'inch');
  400.     al_bundy=> 
  401.     al_bundy=> SELECT * FROM shoelace;
  402.     sl_name   |sl_avail|sl_color  |sl_len|sl_unit |sl_len_cm
  403.     ----------+--------+----------+------+--------+---------
  404.     sl1       |       5|black     |    80|cm      |       80
  405.     sl2       |       6|black     |   100|cm      |      100
  406.     sl7       |       7|brown     |    60|cm      |       60
  407.     sl3       |       0|black     |    35|inch    |     88.9
  408.     sl4       |       8|black     |    40|inch    |    101.6
  409.     sl8       |       1|brown     |    40|inch    |    101.6
  410.     sl5       |       4|brown     |     1|m       |      100
  411.     sl6       |       0|brown     |   0.9|m       |       90
  412.     (8 rows)
  413. </ProgramListing>
  414.     It's the simplest SELECT Al can do on our views, so we take this
  415.     to explain the basics of view rules.
  416.     The 'SELECT * FROM shoelace' was interpreted by the parser and
  417.     produced the parsetree
  418. <ProgramListing>
  419.     SELECT shoelace.sl_name, shoelace.sl_avail,
  420.            shoelace.sl_color, shoelace.sl_len,
  421.            shoelace.sl_unit, shoelace.sl_len_cm
  422.       FROM shoelace shoelace;
  423. </ProgramListing>
  424.     and this is given to the rule system. The rule system walks through the
  425.     rangetable and checks if there are rules in <Filename>pg_rewrite</Filename>
  426.     for any relation. When processing the rangetable entry for
  427.     <Filename>shoelace</Filename> (the only one up to now) it finds the
  428.     rule '_RETshoelace' with the parsetree
  429. <ProgramListing>
  430.     <FirstTerm>SELECT s.sl_name, s.sl_avail,
  431.            s.sl_color, s.sl_len, s.sl_unit,
  432.            float8mul(s.sl_len, u.un_fact) AS sl_len_cm
  433.       FROM shoelace *OLD*, shoelace *NEW*,
  434.            shoelace_data s, unit u
  435.      WHERE bpchareq(s.sl_unit, u.un_name);</FirstTerm>
  436. </ProgramListing>
  437.     Note that the parser changed the calculation and qualification into
  438.     calls to the appropriate functions. But
  439.     in fact this changes nothing.
  440.     The first step in rewriting is merging the two rangetables. The resulting
  441.     parsetree then reads
  442. <ProgramListing>
  443.     SELECT shoelace.sl_name, shoelace.sl_avail,
  444.            shoelace.sl_color, shoelace.sl_len,
  445.            shoelace.sl_unit, shoelace.sl_len_cm
  446.       FROM shoelace shoelace, <FirstTerm>shoelace *OLD*</FirstTerm>,
  447.            <FirstTerm>shoelace *NEW*</FirstTerm>, <FirstTerm>shoelace_data s</FirstTerm>,
  448.            <FirstTerm>unit u</FirstTerm>;
  449. </ProgramListing>
  450.     In step 2 it adds the qualification from the rule action to the
  451.     parsetree resulting in
  452. <ProgramListing>
  453.     SELECT shoelace.sl_name, shoelace.sl_avail,
  454.            shoelace.sl_color, shoelace.sl_len,
  455.            shoelace.sl_unit, shoelace.sl_len_cm
  456.       FROM shoelace shoelace, shoelace *OLD*,
  457.            shoelace *NEW*, shoelace_data s,
  458.            unit u
  459.      <FirstTerm>WHERE bpchareq(s.sl_unit, u.un_name)</FirstTerm>;
  460. </ProgramListing>
  461.     And in step 3 it replaces all the variables in the parsetree, that
  462.     reference the rangetable entry (the one for
  463.     <Filename>shoelace</Filename> that is currently processed)
  464.     by the corresponding targetlist expressions
  465.     from the rule action. This results in the final query
  466. <ProgramListing>
  467.     SELECT <FirstTerm>s.sl_name</FirstTerm>, <FirstTerm>s.sl_avail</FirstTerm>, 
  468.            <FirstTerm>s.sl_color</FirstTerm>, <FirstTerm>s.sl_len</FirstTerm>, 
  469.            <FirstTerm>s.sl_unit</FirstTerm>, <FirstTerm>float8mul(s.sl_len, u.un_fact)</FirstTerm> AS sl_len_cm
  470.       FROM shoelace shoelace, shoelace *OLD*,
  471.            shoelace *NEW*, shoelace_data s,
  472.            unit u
  473.      WHERE bpchareq(s.sl_unit, u.un_name);
  474. </ProgramListing>
  475.     Turning this back into a real <Acronym>SQL</Acronym> statement a human
  476.     user would type reads
  477. <ProgramListing>
  478.     SELECT s.sl_name, s.sl_avail,
  479.            s.sl_color, s.sl_len,
  480.            s.sl_unit, s.sl_len * u.un_fact AS sl_len_cm
  481.       FROM shoelace_data s, unit u
  482.      WHERE s.sl_unit = u.un_name;
  483. </ProgramListing>
  484.     That was the first rule applied. While this was done, the rangetable has
  485.     grown. So the rule system continues checking the range table entries.
  486.     The next one is number 2 (shoelace *OLD*).
  487.     Relation <Filename>shoelace</Filename>
  488.     has a rule, but this rangetable entry isn't referenced
  489.     in any of the variables of the parsetree, so it is ignored. Since all the
  490.     remaining rangetable entries either have no rules in
  491.     <Filename>pg_rewrite</Filename> or aren't referenced,
  492.     it reaches the end of the rangetable.
  493.     Rewriting is complete and the above is the final result given into
  494.     the optimizer.
  495.     The optimizer ignores the extra rangetable entries that aren't
  496.     referenced by variables in the parsetree and the plan produced
  497.     by the planner/optimizer would be exactly the same as if Al had typed
  498.     the above SELECT query instead of the view selection.
  499. </Para>
  500. <Para>
  501.     Now we face Al with the problem that the Blues Brothers appear
  502.     in his shop and
  503.     want to buy some new shoes, and as the Blues Brothers are,
  504.     they want to wear the same shoes. And they want to wear them
  505.     immediately, so they need shoelaces too.
  506. </Para>
  507. <Para>
  508.     Al needs to know for which shoes currently in the store
  509.     he has the matching shoelaces (color and size) and where the
  510.     total number of exactly matching pairs is greater or equal to two.
  511.     We theach him how to do and he asks his database:
  512. <ProgramListing>
  513.     al_bundy=> SELECT * FROM shoe_ready WHERE total_avail >= 2;
  514.     shoename  |sh_avail|sl_name   |sl_avail|total_avail
  515.     ----------+--------+----------+--------+-----------
  516.     sh1       |       2|sl1       |       5|          2
  517.     sh3       |       4|sl7       |       7|          4
  518.     (2 rows)
  519. </ProgramListing>
  520.     Al is a shoe guru and so he knows that only shoes of type sh1
  521.     would fit (shoelace sl7 is brown and shoes that need brown shoelaces
  522.     aren't shoes the Blues Brothers would ever wear).
  523. </Para>
  524. <Para>
  525.     The output of the parser this time is the parsetree
  526. <ProgramListing>
  527.     SELECT shoe_ready.shoename, shoe_ready.sh_avail,
  528.            shoe_ready.sl_name, shoe_ready.sl_avail,
  529.            shoe_ready.total_avail
  530.       FROM shoe_ready shoe_ready
  531.      WHERE int4ge(shoe_ready.total_avail, 2);
  532. </ProgramListing>
  533.     The first rule applied will be that one for the 
  534.     <Filename>shoe_ready</Filename> relation and it results in the
  535.     parsetree
  536. <ProgramListing>
  537.     SELECT <FirstTerm>rsh.shoename</FirstTerm>, <FirstTerm>rsh.sh_avail</FirstTerm>,
  538.            <FirstTerm>rsl.sl_name</FirstTerm>, <FirstTerm>rsl.sl_avail</FirstTerm>,
  539.            <FirstTerm>min(rsh.sh_avail, rsl.sl_avail) AS total_avail</FirstTerm>
  540.       FROM shoe_ready shoe_ready, <FirstTerm>shoe_ready *OLD*</FirstTerm>,
  541.            <FirstTerm>shoe_ready *NEW*</FirstTerm>, <FirstTerm>shoe rsh</FirstTerm>,
  542.            <FirstTerm>shoelace rsl</FirstTerm>
  543.      WHERE int4ge(<FirstTerm>min(rsh.sh_avail, rsl.sl_avail)</FirstTerm>, 2)
  544.        <FirstTerm>AND (bpchareq(rsl.sl_color, rsh.slcolor)
  545.             AND float8ge(rsl.sl_len_cm, rsh.slminlen_cm)
  546.             AND float8le(rsl.sl_len_cm, rsh.slmaxlen_cm)
  547.            )</FirstTerm>;
  548. </ProgramListing>
  549.     In reality the AND clauses in the qualification will be
  550.     operator nodes of type AND with a left and right expression. But
  551.     that makes it lesser readable as it already is, and there are more
  552.     rules to apply. So I only put them into some parantheses to group
  553.     them into logical units in the order they where added and we continue
  554.     with the rule for relation
  555.     <Filename>shoe</Filename> as it is the next rangetable entry
  556.     that is referenced and has a rule. The result of applying it is
  557. <ProgramListing>
  558.     SELECT <FirstTerm>sh.shoename</FirstTerm>, <FirstTerm>sh.sh_avail</FirstTerm>,
  559.            rsl.sl_name, rsl.sl_avail,
  560.            min(<FirstTerm>sh.sh_avail</FirstTerm>, rsl.sl_avail) AS total_avail,
  561.       FROM shoe_ready shoe_ready, shoe_ready *OLD*,
  562.            shoe_ready *NEW*, shoe rsh,
  563.            shoelace rsl, <FirstTerm>shoe *OLD*</FirstTerm>,
  564.            <FirstTerm>shoe *NEW*</FirstTerm>, <FirstTerm>shoe_data sh</FirstTerm>,
  565.            <FirstTerm>unit un</FirstTerm>
  566.      WHERE (int4ge(min(<FirstTerm>sh.sh_avail</FirstTerm>, rsl.sl_avail), 2)
  567.             AND (bpchareq(rsl.sl_color, <FirstTerm>sh.slcolor</FirstTerm>)
  568.                  AND float8ge(rsl.sl_len_cm, 
  569.                               <FirstTerm>float8mul(sh.slminlen, un.un_fact)</FirstTerm>)
  570.                  AND float8le(rsl.sl_len_cm, 
  571.                               <FirstTerm>float8mul(sh.slmaxlen, un.un_fact)</FirstTerm>)
  572.                 )
  573.            )
  574.        <FirstTerm>AND bpchareq(sh.slunit, un.un_name)</FirstTerm>;
  575. </ProgramListing>
  576.     And finally we apply the already well known rule for
  577.     <Filename>shoelace</Filename> (this time on a parsetree that is
  578.     a little more complex) and get
  579. <ProgramListing>
  580.     SELECT sh.shoename, sh.sh_avail,
  581.            <FirstTerm>s.sl_name</FirstTerm>, <FirstTerm>s.sl_avail</FirstTerm>,
  582.            min(sh.sh_avail, <FirstTerm>s.sl_avail</FirstTerm>) AS total_avail
  583.       FROM shoe_ready shoe_ready, shoe_ready *OLD*,
  584.            shoe_ready *NEW*, shoe rsh,
  585.            shoelace rsl, shoe *OLD*,
  586.            shoe *NEW*, shoe_data sh,
  587.            unit un, <FirstTerm>shoelace *OLD*</FirstTerm>,
  588.            <FirstTerm>shoelace *NEW*</FirstTerm>, <FirstTerm>shoelace_data s</FirstTerm>,
  589.            <FirstTerm>unit u</FirstTerm>
  590.      WHERE (    (int4ge(min(sh.sh_avail, <FirstTerm>s.sl_avail</FirstTerm>), 2)
  591.                  AND (bpchareq(<FirstTerm>s.sl_color</FirstTerm>, sh.slcolor)
  592.                       AND float8ge(<FirstTerm>float8mul(s.sl_len, u.un_fact)</FirstTerm>, 
  593.                                    float8mul(sh.slminlen, un.un_fact))
  594.                       AND float8le(<FirstTerm>float8mul(s.sl_len, u.un_fact)</FirstTerm>, 
  595.                                    float8mul(sh.slmaxlen, un.un_fact))
  596.                      )
  597.                 )
  598.             AND bpchareq(sh.slunit, un.un_name)
  599.            )
  600.        <FirstTerm>AND bpchareq(s.sl_unit, u.un_name)</FirstTerm>;
  601. </ProgramListing>
  602.     Again we reduce it to a real <Acronym>SQL</Acronym> statement
  603.     that is equivalent to the final output of the rule system:
  604. <ProgramListing>
  605.     SELECT sh.shoename, sh.sh_avail,
  606.            s.sl_name, s.sl_avail,
  607.            min(sh.sh_avail, s.sl_avail) AS total_avail
  608.       FROM shoe_data sh, shoelace_data s, unit u, unit un
  609.      WHERE min(sh.sh_avail, s.sl_avail) >= 2
  610.        AND s.sl_color = sh.slcolor
  611.        AND s.sl_len * u.un_fact >= sh.slminlen * un.un_fact
  612.        AND s.sl_len * u.un_fact <= sh.slmaxlen * un.un_fact
  613.        AND sh.sl_unit = un.un_name
  614.        AND s.sl_unit = u.un_name;
  615. </ProgramListing>
  616.     Recursive processing of rules rewrote one SELECT from a view
  617.     into a parsetree, that is equivalent to exactly that what Al
  618.     had to type if there would be no views at all.
  619.     <Note>
  620.     <Title>Note</Title>
  621.     <Para>
  622.     There is currently no recursion stopping mechanism for view
  623.     rules in the rule system (only for the other rules).
  624.     This doesn't hurt much, because the only way to push this
  625.     into an endless loop (blowing up the
  626.     backend until it reaches the memory limit)
  627.     is to create tables and then setup the
  628.     view rules by hand with CREATE RULE in such a way, that
  629.     one selects from the other that selects from the one.
  630.     This could never happen if CREATE VIEW is used because
  631.     on the first CREATE VIEW, the second relation does not exist
  632.     and thus the first view cannot select from the second.
  633.     </Para>
  634.     </Note>
  635. </Para>
  636. </Sect2>
  637. <Sect2>
  638. <Title>View Rules in Non-SELECT Statements</Title>
  639. <Para>
  640.     Two details of the parsetree aren't touched in the description of
  641.     view rules above. These are the commandtype and the resultrelation.
  642.     In fact, view rules don't need these informations.
  643. </Para>
  644. <Para>
  645.     There are only a few differences between a parsetree for a SELECT
  646.     and one for any other command. Obviously they have another commandtype
  647.     and this time the resultrelation points to the rangetable entry where
  648.     the result should go. Anything else is absolutely the same. 
  649.     So having two tables t1 and t2 with attributes
  650.     a and b, the parsetrees for the two statements
  651. <ProgramListing>
  652.     SELECT t2.b FROM t1, t2 WHERE t1.a = t2.a;
  653.     UPDATE t1 SET b = t2.b WHERE t1.a = t2.a;
  654. </ProgramListing>
  655.     are nearly identical.
  656.     <ItemizedList>
  657.         <ListItem>
  658. <Para>
  659.     The rangetables contain entries for the tables t1 and t2.
  660. </Para>
  661.         </ListItem>
  662.         <ListItem>
  663. <Para>
  664.     The targetlists contain one variable that points to attribute
  665.     b of the rangetable entry for table t2.
  666. </Para>
  667.         </ListItem>
  668.         <ListItem>
  669. <Para>
  670.     The qualification expressions compare the attributes a of both
  671.     ranges for equality.
  672. </Para>
  673.         </ListItem>
  674.     </ItemizedList>
  675.     The consequence is, that both parsetrees result in similar execution
  676.     plans. They are both joins over the two tables. For the UPDATE
  677.     the missing columns from t1 are added to the targetlist by the optimizer 
  678.     and the final parsetree will read as
  679. <ProgramListing>
  680.     UPDATE t1 SET a = t1.a, b = t2.b WHERE t1.a = t2.a;
  681. </ProgramListing>
  682.     and thus the executor run over the join will produce exactly the
  683.     same result set as a
  684. <ProgramListing>
  685.     SELECT t1.a, t2.b FROM t1, t2 WHERE t1.a = t2.a;
  686. </ProgramListing>
  687.     
  688.     will do. But there is a little problem in UPDATE. The executor does
  689.     not care what the results from the join it is doing are meant
  690.     for. It just produces a result set of rows. The difference that one
  691.     is a SELECT command and the other is an UPDATE is handled in the
  692.     caller of the executor. The caller still knows (looking at the
  693.     parsetree) that this is an UPDATE, and he knows that this result
  694.     should go into table t1. But which of the 666 rows that are there
  695.     has to be replaced by the new row? The plan executed is a join
  696.     with a qualification that potentially could produce any number of
  697.     rows between 0 and 666 in unknown order.
  698. </Para>
  699. <Para>
  700.     To resolve this problem, another entry is added to the targetlist
  701.     in UPDATE and DELETE statements. The current tuple ID (ctid). This
  702.     is a system attribute with a special feature. It contains the
  703.     block and position in the block for the row. Knowing the table,
  704.     the ctid can be used to find one specific row in a 1.5GB sized table
  705.     containing millions of rows by fetching one single data block.
  706.     After adding the ctid to the targetlist, the final result set
  707.     could be defined as
  708. <ProgramListing>
  709.     SELECT t1.a, t2.b, t1.ctid FROM t1, t2 WHERE t1.a = t2.a;
  710. </ProgramListing>
  711.     
  712.     Now another detail of <ProductName>Postgres</ProductName> enters the
  713.     stage. At this moment, table rows aren't overwritten and this is why
  714.     ABORT TRANSACTION is fast. In an UPDATE, the new result row is inserted
  715.     into the table (after stripping ctid) and in the tuple header of the row 
  716.     that ctid pointed to the cmax and xmax entries are set to the current
  717.     command counter and current transaction ID. Thus the old row is hidden
  718.     and after the transaction commited the vacuum cleaner can really move
  719.     it out.
  720. </Para>
  721. <Para>
  722.     Knowing that all, we can simply apply view rules in absolutely
  723.     the same way to any command. There is no difference.
  724. </Para>
  725. </Sect2>
  726. <Sect2>
  727. <Title>The Power of Views in <ProductName>Postgres</ProductName></Title>
  728. <Para>
  729.     The above demonstrates how the rule system incorporates
  730.     view definitions into the original parsetree. In the second example
  731.     a simple SELECT from one view created a final parsetree that is
  732.     a join of 4 tables (unit is used twice with different names).
  733. </Para>
  734. <Sect3>
  735. <Title>Benefits</Title>
  736. <Para>
  737.     The benefit of implementing views with the rule system is,
  738.     that the optimizer has all
  739.     the information about which tables have to be scanned plus the
  740.     relationships between these tables plus the restrictive
  741.     qualifications from the views plus the qualifications from
  742.     the original query
  743.     in one single parsetree. And this is still the situation
  744.     when the original query is already a join over views.
  745.     Now the optimizer has to decide which is
  746.     the best path to execute the query. The more information
  747.     the optimizer has, the better this decision can be. And
  748.     the rule system as implemented in <ProductName>Postgres</ProductName>
  749.     ensures, that this is all information available about the query
  750.     up to now.
  751. </Para>
  752. </Sect3>
  753. <Sect3>
  754. <Title>Concerns</Title>
  755. <Para>
  756.     There was a long time where the <ProductName>Postgres</ProductName> 
  757.     rule system was considered broken. The use of rules was not
  758.     recommended and the only part working where view rules. And also
  759.     these view rules made problems because the rule system wasn't able
  760.     to apply them properly on other statements than a SELECT (for
  761.     example an UPDATE
  762.     that used data from a view didn't work).
  763. </Para>
  764. <Para>
  765.     During that time, development moved on and many features where
  766.     added to the parser and optimizer. The rule system got more and more
  767.     out of sync with their capabilities and it became harder and harder
  768.     to start fixing it. Thus, noone did.
  769. </Para>
  770. <Para>
  771.     For 6.4, someone locked the door, took a deep breath and shuffled
  772.     that damned thing up. What came out was a rule system with the
  773.     capabilities described in this document. But there are still some
  774.     constructs not handled and some where it fails due to 
  775.     things that are currently not
  776.     supported by the <ProductName>Postgres</ProductName> query
  777.     optimizer.
  778.     <ItemizedList>
  779.         <ListItem>
  780. <Para>
  781.     Views with aggregate columns have bad problems. Aggregate
  782.     expressions in qualifications must be used in subselects.
  783.     Currently it is not possible to do a join of two views,
  784.     each having an aggregate column, and compare the two aggregate values
  785.     in the qualification. In the meantime it is possible to
  786.     put these aggregate expressions into functions with
  787.     the appropriate arguments and use
  788.     them in the view definition.
  789. </Para>
  790.         </ListItem>
  791.         <ListItem>
  792. <Para>
  793.     Views of unions are currently not supported. Well it's easy
  794.     to rewrite a simple SELECT into a union. But it is a little
  795.     difficult if the view is part of a join doing an update.
  796. </Para>
  797.         </ListItem>
  798.         <ListItem>
  799. <Para>
  800.     ORDER BY clauses in view definitions aren't supported.
  801. </Para>
  802.         </ListItem>
  803.         <ListItem>
  804. <Para>
  805.     DISTINCT isn't supported in view definitions.
  806. </Para>
  807.         </ListItem>
  808.     </ItemizedList>
  809.     There is no good reason why the optimizer should not
  810.     handle parsetree constructs that the parser could never produce
  811.     due to limitations in the <Acronym>SQL</Acronym> syntax.
  812.     The author hopes that these items disappear in the future.
  813. </Para>
  814. </Sect3>
  815. </Sect2>
  816. <Sect2>
  817. <Title>Implementation Side Effects</Title>
  818. <Para>
  819.     Using the described rule system to implement views has a funny
  820.     side effect. The following does not seem to work:
  821. <ProgramListing>
  822.     al_bundy=> INSERT INTO shoe (shoename, sh_avail, slcolor)
  823.     al_bundy->     VALUES ('sh5', 0, 'black');
  824.     INSERT 20128 1
  825.     al_bundy=> SELECT shoename, sh_avail, slcolor FROM shoe_data;
  826.     shoename  |sh_avail|slcolor   
  827.     ----------+--------+----------
  828.     sh1       |       2|black     
  829.     sh3       |       4|brown     
  830.     sh2       |       0|black     
  831.     sh4       |       3|brown     
  832.     (4 rows)
  833. </ProgramListing>
  834.     The interesting thing is that the return code for INSERT gave
  835.     us an object ID and told that 1 row has been inserted.
  836.     But it doesn't appear in <Filename>shoe_data</Filename>.
  837.     Looking into the database
  838.     directory we can see, that the database file for the
  839.     view relation <Filename>shoe</Filename> seems now to have
  840.     a data block. And that is definitely the case.
  841. </Para>
  842. <Para>
  843.     We can also issue a DELETE and if it does not have
  844.     a qualification, it tells us that rows have been deleted
  845.     and the next vacuum run will reset the file to zero size.
  846. </Para>
  847. <Para>
  848.     The reason for that behaviour is, that the parsetree for the
  849.     INSERT does not reference the <Filename>shoe</Filename> relation
  850.     in any variable. The targetlist contains only constant values.
  851.     So there is no rule to apply and it goes
  852.     down unchanged into execution and the row is inserted. And
  853.     so for the DELETE.
  854. </Para>
  855. <Para>
  856.     To change this we can define rules that modify the behaviour
  857.     of non-SELECT queries. This is the topic of the next section.
  858. </Para>
  859. </Sect2>
  860. </Sect1>
  861. <Sect1>
  862. <Title>Rules on INSERT, UPDATE and DELETE</Title>
  863. <Sect2>
  864. <Title>Differences to View Rules</Title>
  865. <Para>
  866.     Rules that are defined ON INSERT, UPDATE and DELETE are
  867.     totally different from the view rules described
  868.     in the previous section. First, their CREATE RULE
  869.     command allows more:
  870.     <ItemizedList>
  871.         <ListItem>
  872. <Para>
  873.     They can have no action.
  874. </Para>
  875. </ListItem>
  876.         <ListItem>
  877. <Para>
  878.     They can have multiple actions.
  879. </Para>
  880. </ListItem>
  881.         <ListItem>
  882. <Para>
  883.     The keyword INSTEAD is optional.
  884. </Para>
  885. </ListItem>
  886.         <ListItem>
  887. <Para>
  888.     The pseudo relations NEW and OLD become useful.
  889. </Para>
  890. </ListItem>
  891.         <ListItem>
  892. <Para>
  893.     They can have rule qualifications.
  894. </Para>
  895. </ListItem>
  896.     </ItemizedList>
  897.     Second, they don't modify the parsetree in place. Instead they
  898.     create zero or many new parsetrees and can throw away the
  899.     original one.
  900. </Para>
  901. </sect2>
  902. <Sect2>
  903. <Title>How These Rules Work</Title>
  904. <Para>
  905.     Keep the syntax
  906. <ProgramListing>
  907.     CREATE RULE rule_name AS ON event
  908.         TO object [WHERE rule_qualification]
  909.         DO [INSTEAD] [action | (actions) | NOTHING];
  910. </ProgramListing>
  911.     in mind.
  912.     In the following, "update rules" means rules that are defined
  913.     ON INSERT, UPDATE or DELETE.
  914. </Para>
  915. <Para>
  916.     Update rules get applied by the rule system when the result
  917.     relation and the commandtype of a parsetree are equal to the
  918.     object and event given in the CREATE RULE command.
  919.     For update rules, the rule system creates a list of parsetrees.
  920.     Initially the parsetree list is empty.
  921.     There can be zero (NOTHING keyword), one or multiple actions.
  922.     To simplify, we look at a rule with one action. This rule
  923.     can have a qualification or not and it can be INSTEAD or not.
  924. </Para>
  925. <Para>
  926.     What is a rule qualification? It is a restriction that tells
  927.     when the actions of the rule should be done and when not. This
  928.     qualification can only reference the NEW and/or OLD pseudo relations
  929.     which are basically the relation given as object (but with a
  930.     special meaning).
  931. </Para>
  932. <Para>
  933.     So we have four cases that produce the following parsetrees for
  934.     a one-action rule.
  935. </Para>
  936. <Para>
  937.     <ItemizedList>
  938.     <ListItem>
  939.     <Para>
  940.         No qualification and not INSTEAD:
  941.         <ItemizedList>
  942.     <ListItem>
  943.     <Para>
  944.         The parsetree from the rule action where the
  945. original parsetrees qualification has been added.
  946.     </Para>
  947.     </ListItem>
  948. </ItemizedList>
  949.     </Para>
  950.     </ListItem>
  951.     <ListItem>
  952.     <Para>
  953.         No qualification but INSTEAD:
  954.         <ItemizedList>
  955.     <ListItem>
  956.     <Para>
  957.         The parsetree from the rule action where the
  958. original parsetrees qualification has been added.
  959.     </Para>
  960.     </ListItem>
  961. </ItemizedList>
  962.     </Para>
  963.     </ListItem>
  964.     <ListItem>
  965.     <Para>
  966.         Qualification given and not INSTEAD:
  967.         <ItemizedList>
  968.     <ListItem>
  969.     <Para>
  970.         The parsetree from the rule action where the rule
  971. qualification and the original parsetrees 
  972. qualification have been added.
  973.     </Para>
  974.     </ListItem>
  975. </ItemizedList>
  976.     </Para>
  977.     </ListItem>
  978.     <ListItem>
  979.     <Para>
  980.         Qualification given and INSTEAD:
  981.         <ItemizedList>
  982.     <ListItem>
  983.     <Para>
  984.         The parsetree from the rule action where the rule
  985. qualification and the original parsetrees 
  986. qualification have been added.
  987.     </Para>
  988.     </ListItem>
  989.     <ListItem>
  990.     <Para>
  991.         The original parsetree where the negated rule
  992. qualification has been added.
  993.     </Para>
  994.     </ListItem>
  995. </ItemizedList>
  996.     </Para>
  997.     </ListItem>
  998.     </ItemizedList>
  999. </para>
  1000. <Para>
  1001.     Finally, if the rule is not INSTEAD, the unchanged original parsetree is
  1002.     added to the list. Since only qualified INSTEAD rules already add the
  1003.     original parsetree, we end up with a total maximum of two parsetrees
  1004.     for a rule with one action.
  1005. </Para>
  1006. <Para>
  1007.     The parsetrees generated from rule actions are thrown into the
  1008.     rewrite system again and maybe more rules get applied resulting
  1009.     in more or less parsetrees.
  1010.     So the parsetrees in the rule actions must have either another commandtype
  1011.     or another resultrelation. Otherwise this recursive process will end up in a loop.
  1012.     There is a compiled in recursion limit of currently 10 iterations.
  1013.     If after 10 iterations there are still update rules to apply the
  1014.     rule system assumes a loop over multiple rule definitions and aborts the
  1015.     transaction.
  1016. </Para>
  1017. <Para>
  1018.     The parsetrees found in the actions of the <Filename>pg_rewrite</Filename>
  1019.     system catalog are only templates. Since they can reference the
  1020.     rangetable entries for NEW and OLD, some substitutions have to be made
  1021.     before they can be used. For any reference to NEW, the targetlist of
  1022.     the original query is searched for a corresponding entry. If found,
  1023.     that entries expression is placed into the reference. Otherwise
  1024.     NEW means the same as OLD. Any reference to OLD is replaced by a
  1025.     reference to the rangetable entry which is the resultrelation.
  1026. </Para>
  1027. <Sect3>
  1028. <Title>A First Rule Step by Step</Title>
  1029. <Para>
  1030.     We want to trace changes to the sl_avail column in the
  1031.     <Filename>shoelace_data</Filename> relation. So we setup a
  1032.     log table and a rule that writes us entries every time
  1033.     and UPDATE is performed on <Filename>shoelace_data</Filename>.
  1034. <ProgramListing>
  1035.     CREATE TABLE shoelace_log (
  1036.         sl_name    char(10),      -- shoelace changed
  1037.         sl_avail   integer,       -- new available value
  1038.         log_who    name,          -- who did it
  1039.         log_when   datetime       -- when
  1040.     );
  1041.     CREATE RULE log_shoelace AS ON UPDATE TO shoelace_data
  1042.         WHERE NEW.sl_avail != OLD.sl_avail
  1043.         DO INSERT INTO shoelace_log VALUES (
  1044.                                         NEW.sl_name,
  1045.                                         NEW.sl_avail,
  1046.                                         getpgusername(),
  1047.                                         'now'::text
  1048.                                     );
  1049. </ProgramListing>
  1050.     One interesting detail is the casting of 'now' in the rules
  1051.     INSERT action to type text. Without that, the parser would see
  1052.     at CREATE RULE time, that the target type in <Filename>shoelace_log</Filename>
  1053.     is a datetime and tries to make a constant from it - with success.
  1054.     So a constant datetime value would be stored in the rule action
  1055.     and all log entries would have the time of the CREATE RULE statement.
  1056.     Not exactly what we want. The casting causes that the parser
  1057.     constructs a datetime('now'::text) from it and this will be 
  1058.     evaluated when the rule is executed.
  1059. </Para>
  1060. <Para>
  1061.     Now Al does
  1062. <ProgramListing>
  1063.     al_bundy=> UPDATE shoelace_data SET sl_avail = 6                       
  1064.     al_bundy->     WHERE sl_name = 'sl7';
  1065. </ProgramListing>
  1066.     and we look at the logtable.
  1067. <ProgramListing>
  1068.     al_bundy=> SELECT * FROM shoelace_log;
  1069.     sl_name   |sl_avail|log_who|log_when                        
  1070.     ----------+--------+-------+--------------------------------
  1071.     sl7       |       6|Al     |Tue Oct 20 16:14:45 1998 MET DST
  1072.     (1 row)
  1073. </ProgramListing>
  1074.     That's what we expected. What happened in the background is the following.
  1075.     The parser created the parsetree (this time the parts of the original
  1076.     parsetree are highlighted because the base of operations is the
  1077.     rule action for update rules).
  1078. <ProgramListing>
  1079.     <FirstTerm>UPDATE shoelace_data SET sl_avail = 6
  1080.       FROM shoelace_data shoelace_data
  1081.      WHERE bpchareq(shoelace_data.sl_name, 'sl7');</FirstTerm>
  1082. </ProgramListing>
  1083.     There is a rule 'log_shoelace' that is ON UPDATE with the rule
  1084.     qualification expression
  1085. <ProgramListing>
  1086.     int4ne(NEW.sl_avail, OLD.sl_avail)
  1087. </ProgramListing>
  1088.     and one action
  1089. <ProgramListing>
  1090.     INSERT INTO shoelace_log SELECT 
  1091.            *NEW*.sl_name, *NEW*.sl_avail,
  1092.            getpgusername(), datetime('now'::text)
  1093.       FROM shoelace_data *NEW*, shoelace_data *OLD*,
  1094.            shoelace_log shoelace_log;
  1095. </ProgramListing>
  1096.     Don't trust the output of the pg_rules system view. It specially
  1097.     handles the situation that there are only references to NEW
  1098.     and OLD in the INSERT and outputs the VALUES format of INSERT.
  1099.     In fact there is no difference between an INSERT ... VALUES
  1100.     and an INSERT ... SELECT on parsetree level. They both have
  1101.     rangetables, targetlists and maybe qualifications etc. The
  1102.     optimizer later decides, if to create an execution plan of
  1103.     type result, seqscan, indexscan, join or whatever for that
  1104.     parsetree. If there are no references to
  1105.     rangetable entries leftin the parsetree , it becomes
  1106.     a result execution plan
  1107.     (the INSERT ... VALUES version). The rule action above can
  1108.     truely result in both variants.
  1109. </Para>
  1110. <Para>
  1111.     The rule is a qualified non-INSTEAD rule, so the rule system
  1112.     has to return two parsetrees. The modified rule action and the original
  1113.     parsetree. In the first step the rangetable of the original query is
  1114.     incorporated into the rules action parsetree. This results in
  1115. <ProgramListing>
  1116.     INSERT INTO shoelace_log SELECT 
  1117.            *NEW*.sl_name, *NEW*.sl_avai,
  1118.            getpgusername(), datetime('now'::text)
  1119.       FROM <FirstTerm>shoelace_data shoelace_data</FirstTerm>, shoelace_data *NEW*,
  1120.            shoelace_data *OLD*, shoelace_log shoelace_log;
  1121. </ProgramListing>
  1122.     In step 2 the rule qualification is added to it, so the result set
  1123.     is restricted to rows where sl_avail changes.
  1124. <ProgramListing>
  1125.     INSERT INTO shoelace_log SELECT 
  1126.            *NEW*.sl_name, *NEW*.sl_avai,
  1127.            getpgusername(), datetime('now'::text)
  1128.       FROM shoelace_data shoelace_data, shoelace_data *NEW*,
  1129.            shoelace_data *OLD*, shoelace_log shoelace_log
  1130.      <FirstTerm>WHERE int4ne(*NEW*.sl_avail, *OLD*.sl_avail)</FirstTerm>;
  1131. </ProgramListing>
  1132.     In step 3 the original parsetrees qualification is added,
  1133.     restricting the resultset further to only the rows touched
  1134.     by the original parsetree.
  1135. <ProgramListing>
  1136.     INSERT INTO shoelace_log SELECT 
  1137.            *NEW*.sl_name, *NEW*.sl_avai,
  1138.            getpgusername(), datetime('now'::text)
  1139.       FROM shoelace_data shoelace_data, shoelace_data *NEW*,
  1140.            shoelace_data *OLD*, shoelace_log shoelace_log
  1141.      WHERE int4ne(*NEW*.sl_avail, *OLD*.sl_avail)
  1142.        <FirstTerm>AND bpchareq(shoelace_data.sl_name, 'sl7')</FirstTerm>;
  1143. </ProgramListing>
  1144.     Step 4 substitutes NEW references by the targetlist entries from the
  1145.     original parsetree or with the matching variable references
  1146.     from the result relation.
  1147. <ProgramListing>
  1148.     INSERT INTO shoelace_log SELECT 
  1149.            <FirstTerm>shoelace_data.sl_name</FirstTerm>, <FirstTerm>6</FirstTerm>,
  1150.            getpgusername(), datetime('now'::text)
  1151.       FROM shoelace_data shoelace_data, shoelace_data *NEW*,
  1152.            shoelace_data *OLD*, shoelace_log shoelace_log
  1153.      WHERE int4ne(<FirstTerm>6</FirstTerm>, *OLD*.sl_avail)
  1154.        AND bpchareq(shoelace_data.sl_name, 'sl7');
  1155. </ProgramListing>
  1156.     Step 5 replaces OLD references into resultrelation references.
  1157. <ProgramListing>
  1158.     INSERT INTO shoelace_log SELECT 
  1159.            shoelace_data.sl_name, 6,
  1160.            getpgusername(), datetime('now'::text)
  1161.       FROM shoelace_data shoelace_data, shoelace_data *NEW*,
  1162.            shoelace_data *OLD*, shoelace_log shoelace_log
  1163.      WHERE int4ne(6, <FirstTerm>shoelace_data.sl_avail</FirstTerm>)
  1164.        AND bpchareq(shoelace_data.sl_name, 'sl7');
  1165. </ProgramListing>
  1166.     That's it. So reduced to the max the return from the rule system
  1167.     is a list of two parsetrees that are the same as the statements:
  1168. <ProgramListing>
  1169.     INSERT INTO shoelace_log SELECT
  1170.            shoelace_data.sl_name, 6,
  1171.            getpgusername(), 'now'
  1172.       FROM shoelace_data
  1173.      WHERE 6 != shoelace_data.sl_avail
  1174.        AND shoelace_data.sl_name = 'sl7';
  1175.     UPDATE shoelace_data SET sl_avail = 6
  1176.      WHERE sl_name = 'sl7';
  1177. </ProgramListing>
  1178.     These are executed in this order and that is exactly what
  1179.     the rule defines. The subtitutions and the qualifications
  1180.     added ensure, that if the original query would be an
  1181. <ProgramListing>
  1182.     UPDATE shoelace_data SET sl_color = 'green'
  1183.      WHERE sl_name = 'sl7';
  1184. </ProgramListing>
  1185.     No log entry would get written because due to the fact that this
  1186.     time the original parsetree does not contain a targetlist
  1187.     entry for sl_avail, NEW.sl_avail will get replaced by
  1188.     shoelace_data.sl_avail resulting in the extra query
  1189. <ProgramListing>
  1190.     INSERT INTO shoelace_log SELECT
  1191.            shoelace_data.sl_name, <FirstTerm>shoelace_data.sl_avail</FirstTerm>,
  1192.            getpgusername(), 'now'
  1193.       FROM shoelace_data
  1194.      WHERE <FirstTerm>shoelace_data.sl_avail</FirstTerm> != shoelace_data.sl_avail
  1195.        AND shoelace_data.sl_name = 'sl7';
  1196. </ProgramListing>
  1197.     and that qualification will never be true. Since the is no
  1198.     difference on parsetree level between an INSERT ... SELECT,
  1199.     and an INSERT ... VALUES, it will also
  1200.     work if the original query modifies multiple rows. So if Al
  1201.     would issue the command
  1202. <ProgramListing>
  1203.     UPDATE shoelace_data SET sl_avail = 0
  1204.      WHERE sl_color = 'black';
  1205. </ProgramListing>
  1206.     four rows in fact get updated (sl1, sl2, sl3 and sl4).
  1207.     But sl3 already has sl_avail = 0. This time, the original
  1208.     parsetrees qualification is different and that results
  1209.     in the extra parsetree
  1210. <ProgramListing>
  1211.     INSERT INTO shoelace_log SELECT
  1212.            shoelace_data.sl_name, 0,
  1213.            getpgusername(), 'now'
  1214.       FROM shoelace_data
  1215.      WHERE 0 != shoelace_data.sl_avail
  1216.        AND <FirstTerm>shoelace_data.sl_color = 'black'</FirstTerm>;
  1217. </ProgramListing>
  1218.     This parsetree will surely insert three new log entries. And
  1219.     that's absolutely correct.
  1220. </Para>
  1221. <Para>
  1222.     It is important, that the original parsetree is executed last.
  1223.     The <ProductName>Postgres</ProductName> "traffic cop" does
  1224.     a command counter increment between the execution of the two
  1225.     parsetrees so the second one can see changes made by the first.
  1226.     If the UPDATE would have been executed first, all the rows
  1227.     are already set to zero, so the logging INSERT
  1228.     would not find any row where 0 != shoelace_data.sl_avail.
  1229. </Para>
  1230. </Sect3>
  1231. </Sect2>
  1232. <Sect2>
  1233. <Title>Cooperation with Views</Title>
  1234. <Para>
  1235.     A simple way to protect view relations from the mentioned
  1236.     possibility that someone can INSERT, UPDATE and DELETE
  1237.     invisible data on them is to let those parsetrees get
  1238.     thrown away. We create the rules
  1239. <ProgramListing>
  1240.     CREATE RULE shoe_ins_protect AS ON INSERT TO shoe
  1241.         DO INSTEAD NOTHING;
  1242.     CREATE RULE shoe_upd_protect AS ON UPDATE TO shoe
  1243.         DO INSTEAD NOTHING;
  1244.     CREATE RULE shoe_del_protect AS ON DELETE TO shoe
  1245.         DO INSTEAD NOTHING;
  1246. </ProgramListing>
  1247.     If Al now tries to do any of these operations on the view
  1248.     relation <Filename>shoe</Filename>, the rule system will
  1249.     apply the rules. Since the rules have
  1250.     no actions and are INSTEAD, the resulting list of
  1251.     parsetrees will be empty and the whole query will become
  1252.     nothing because there is nothing left to be optimized or
  1253.     executed after the rule system is done with it.
  1254.     <Note>
  1255.     <Title>Note</Title>
  1256.     <Para>
  1257.     This fact might irritate frontend applications because
  1258.     absolutely nothing happened on the database and thus, the
  1259.     backend will not return anything for the query. Not
  1260.     even a PGRES_EMPTY_QUERY or so will be available in libpq.
  1261.     In psql, nothing happens. This might change in the future.
  1262.     </Para>
  1263.     </Note>
  1264. </Para>
  1265. <Para>
  1266.     A more sophisticated way to use the rule system is to
  1267.     create rules that rewrite the parsetree into one that
  1268.     does the right operation on the real tables. To do that
  1269.     on the <Filename>shoelace</Filename> view, we create
  1270.     the following rules:
  1271. <ProgramListing>
  1272.     CREATE RULE shoelace_ins AS ON INSERT TO shoelace
  1273.         DO INSTEAD
  1274.         INSERT INTO shoelace_data VALUES (
  1275.                NEW.sl_name,
  1276.                NEW.sl_avail,
  1277.                NEW.sl_color,
  1278.                NEW.sl_len,
  1279.                NEW.sl_unit);
  1280.     CREATE RULE shoelace_upd AS ON UPDATE TO shoelace
  1281.         DO INSTEAD
  1282.         UPDATE shoelace_data SET
  1283.                sl_name = NEW.sl_name,
  1284.                sl_avail = NEW.sl_avail,
  1285.                sl_color = NEW.sl_color,
  1286.                sl_len = NEW.sl_len,
  1287.                sl_unit = NEW.sl_unit
  1288.          WHERE sl_name = OLD.sl_name;
  1289.     CREATE RULE shoelace_del AS ON DELETE TO shoelace
  1290.         DO INSTEAD
  1291.         DELETE FROM shoelace_data
  1292.          WHERE sl_name = OLD.sl_name;
  1293. </ProgramListing>
  1294.     Now there is a pack of shoelaces arriving in Al's shop and it has
  1295.     a big partlist. Al is not that good in calculating and so
  1296.     we don't want him to manually update the shoelace view.
  1297.     Instead we setup two little tables, one where he can
  1298.     insert the items from the partlist and one with a special
  1299.     trick. The create commands for anything are:
  1300. <ProgramListing>
  1301.     CREATE TABLE shoelace_arrive (
  1302.         arr_name    char(10),
  1303.         arr_quant   integer
  1304.     );
  1305.     CREATE TABLE shoelace_ok (
  1306.         ok_name     char(10),
  1307.         ok_quant    integer
  1308.     );
  1309.     CREATE RULE shoelace_ok_ins AS ON INSERT TO shoelace_ok
  1310.         DO INSTEAD
  1311.         UPDATE shoelace SET
  1312.                sl_avail = sl_avail + NEW.ok_quant
  1313.          WHERE sl_name = NEW.ok_name;
  1314. </ProgramListing>
  1315.     Now Al can sit down and do whatever until
  1316. <ProgramListing>
  1317.     al_bundy=> SELECT * FROM shoelace_arrive;
  1318.     arr_name  |arr_quant
  1319.     ----------+---------
  1320.     sl3       |       10
  1321.     sl6       |       20
  1322.     sl8       |       20
  1323.     (3 rows)
  1324. </ProgramListing>
  1325.     is exactly that what's on the part list. We take a quick look
  1326.     at the current data,
  1327.     
  1328. <ProgramListing>
  1329.     al_bundy=> SELECT * FROM shoelace ORDER BY sl_name;
  1330.     sl_name   |sl_avail|sl_color  |sl_len|sl_unit |sl_len_cm
  1331.     ----------+--------+----------+------+--------+---------
  1332.     sl1       |       5|black     |    80|cm      |       80
  1333.     sl2       |       6|black     |   100|cm      |      100
  1334.     sl7       |       6|brown     |    60|cm      |       60
  1335.     sl3       |       0|black     |    35|inch    |     88.9
  1336.     sl4       |       8|black     |    40|inch    |    101.6
  1337.     sl8       |       1|brown     |    40|inch    |    101.6
  1338.     sl5       |       4|brown     |     1|m       |      100
  1339.     sl6       |       0|brown     |   0.9|m       |       90
  1340.     (8 rows)
  1341. </ProgramListing>
  1342.     move the arrived shoelaces in
  1343. <ProgramListing>
  1344.     al_bundy=> INSERT INTO shoelace_ok SELECT * FROM shoelace_arrive;
  1345. </ProgramListing>
  1346.     and check the results
  1347. <ProgramListing>
  1348.     al_bundy=> SELECT * FROM shoelace ORDER BY sl_name;
  1349.     sl_name   |sl_avail|sl_color  |sl_len|sl_unit |sl_len_cm
  1350.     ----------+--------+----------+------+--------+---------
  1351.     sl1       |       5|black     |    80|cm      |       80
  1352.     sl2       |       6|black     |   100|cm      |      100
  1353.     sl7       |       6|brown     |    60|cm      |       60
  1354.     sl4       |       8|black     |    40|inch    |    101.6
  1355.     sl3       |      10|black     |    35|inch    |     88.9
  1356.     sl8       |      21|brown     |    40|inch    |    101.6
  1357.     sl5       |       4|brown     |     1|m       |      100
  1358.     sl6       |      20|brown     |   0.9|m       |       90
  1359.     (8 rows)
  1360.     al_bundy=> SELECT * FROM shoelace_log;
  1361.     sl_name   |sl_avail|log_who|log_when                        
  1362.     ----------+--------+-------+--------------------------------
  1363.     sl7       |       6|Al     |Tue Oct 20 19:14:45 1998 MET DST
  1364.     sl3       |      10|Al     |Tue Oct 20 19:25:16 1998 MET DST
  1365.     sl6       |      20|Al     |Tue Oct 20 19:25:16 1998 MET DST
  1366.     sl8       |      21|Al     |Tue Oct 20 19:25:16 1998 MET DST
  1367.     (4 rows)
  1368. </ProgramListing>
  1369.     It's a long way from the one INSERT ... SELECT to these
  1370.     results. And it's description will be the last in this
  1371.     document (but not the last example :-). First there was the parsers output
  1372. <ProgramListing>
  1373.     INSERT INTO shoelace_ok SELECT
  1374.            shoelace_arrive.arr_name, shoelace_arrive.arr_quant
  1375.       FROM shoelace_arrive shoelace_arrive, shoelace_ok shoelace_ok;
  1376. </ProgramListing>
  1377.     Now the first rule 'shoelace_ok_ins' is applied and turns it
  1378.     into
  1379. <ProgramListing>
  1380.     UPDATE shoelace SET
  1381.            sl_avail = int4pl(shoelace.sl_avail, shoelace_arrive.arr_quant)
  1382.       FROM shoelace_arrive shoelace_arrive, shoelace_ok shoelace_ok,
  1383.            shoelace_ok *OLD*, shoelace_ok *NEW*,
  1384.            shoelace shoelace
  1385.      WHERE bpchareq(shoelace.sl_name, showlace_arrive.arr_name);
  1386. </ProgramListing>
  1387.     and throws away the original INSERT on <Filename>shoelace_ok</Filename>.
  1388.     This rewritten query is passed to the rule system again and
  1389.     the second applied rule 'shoelace_upd' produced
  1390. <ProgramListing>
  1391.     UPDATE shoelace_data SET
  1392.            sl_name = shoelace.sl_name,
  1393.            sl_avail = int4pl(shoelace.sl_avail, shoelace_arrive.arr_quant),
  1394.            sl_color = shoelace.sl_color,
  1395.            sl_len = shoelace.sl_len,
  1396.            sl_unit = shoelace.sl_unit
  1397.       FROM shoelace_arrive shoelace_arrive, shoelace_ok shoelace_ok,
  1398.            shoelace_ok *OLD*, shoelace_ok *NEW*,
  1399.            shoelace shoelace, shoelace *OLD*,
  1400.            shoelace *NEW*, shoelace_data showlace_data
  1401.      WHERE bpchareq(shoelace.sl_name, showlace_arrive.arr_name)
  1402.        AND bpchareq(shoelace_data.sl_name, shoelace.sl_name);
  1403. </ProgramListing>
  1404.     Again it's an INSTEAD rule and the previous parsetree is trashed.
  1405.     Note that this query sill uses the view <Filename>shoelace</Filename>
  1406.     But the rule system isn't finished with this loop so it continues
  1407.     and applies the rule '_RETshoelace' on it and we get
  1408. <ProgramListing>
  1409.     UPDATE shoelace_data SET
  1410.            sl_name = s.sl_name,
  1411.            sl_avail = int4pl(s.sl_avail, shoelace_arrive.arr_quant),
  1412.            sl_color = s.sl_color,
  1413.            sl_len = s.sl_len,
  1414.            sl_unit = s.sl_unit
  1415.       FROM shoelace_arrive shoelace_arrive, shoelace_ok shoelace_ok,
  1416.            shoelace_ok *OLD*, shoelace_ok *NEW*,
  1417.            shoelace shoelace, shoelace *OLD*,
  1418.            shoelace *NEW*, shoelace_data showlace_data,
  1419.            shoelace *OLD*, shoelace *NEW*,
  1420.            shoelace_data s, unit u
  1421.      WHERE bpchareq(s.sl_name, showlace_arrive.arr_name)
  1422.        AND bpchareq(shoelace_data.sl_name, s.sl_name);
  1423. </ProgramListing>
  1424.     Again an update rule has been applied and so the wheel
  1425.     turns on and we are in rewrite round 3. This time rule
  1426.     'log_shoelace' gets applied what produces the extra
  1427.     parsetree
  1428. <ProgramListing>
  1429.     INSERT INTO shoelace_log SELECT
  1430.            s.sl_name,
  1431.            int4pl(s.sl_avail, shoelace_arrive.arr_quant),
  1432.            getpgusername(),
  1433.            datetime('now'::text)
  1434.       FROM shoelace_arrive shoelace_arrive, shoelace_ok shoelace_ok,
  1435.            shoelace_ok *OLD*, shoelace_ok *NEW*,
  1436.            shoelace shoelace, shoelace *OLD*,
  1437.            shoelace *NEW*, shoelace_data showlace_data,
  1438.            shoelace *OLD*, shoelace *NEW*,
  1439.            shoelace_data s, unit u,
  1440.            shoelace_data *OLD*, shoelace_data *NEW*
  1441.            shoelace_log shoelace_log
  1442.      WHERE bpchareq(s.sl_name,  showlace_arrive.arr_name)
  1443.        AND bpchareq(shoelace_data.sl_name, s.sl_name);
  1444.        AND int4ne(int4pl(s.sl_avail, shoelace_arrive.arr_quant),
  1445.                                                     s.sl_avail);
  1446. </ProgramListing>
  1447.     
  1448.     After that the rule system runs out of rules and returns the 
  1449.     generated parsetrees.
  1450.     So we end up with two final parsetrees that are equal to the
  1451.     <Acronym>SQL</Acronym> statements
  1452. <ProgramListing>
  1453.     INSERT INTO shoelace_log SELECT
  1454.            s.sl_name,
  1455.            s.sl_avail + shoelace_arrive.arr_quant,
  1456.            getpgusername(),
  1457.            'now'
  1458.       FROM shoelace_arrive shoelace_arrive, shoelace_data shoelace_data,
  1459.            shoelace_data s
  1460.      WHERE s.sl_name = shoelace_arrive.arr_name
  1461.        AND shoelace_data.sl_name = s.sl_name
  1462.        AND s.sl_avail + shoelace_arrive.arr_quant != s.sl_avail;
  1463.            
  1464.     UPDATE shoelace_data SET
  1465.            sl_avail = shoelace_data.sl_avail + shoelace_arrive.arr_quant
  1466.      FROM shoelace_arrive shoelace_arrive,
  1467.           shoelace_data shoelace_data,
  1468.           shoelace_data s
  1469.     WHERE s.sl_name = shoelace_arrive.sl_name
  1470.       AND shoelace_data.sl_name = s.sl_name;
  1471. </ProgramListing>
  1472.     The result is that data coming from one relation inserted into another,
  1473.     changed into updates on a third, changed into updating
  1474.     a fourth plus logging that final update in a fifth
  1475.     gets reduced into two queries.
  1476. </Para>
  1477. <Para>
  1478.     There is a little detail that's a bit ugly. Looking at
  1479.     the two queries turns out, that the <Filename>shoelace_data</Filename>
  1480.     relation appears twice in the rangetable where it could definitely
  1481.     be reduced to one. The optimizer does not handle it and so the
  1482.     execution plan for the rule systems output of the INSERT will be
  1483. <ProgramListing>
  1484. Nested Loop
  1485.   ->  Merge Join
  1486.         ->  Seq Scan
  1487.               ->  Sort
  1488.                     ->  Seq Scan on s
  1489.         ->  Seq Scan
  1490.               ->  Sort
  1491.                     ->  Seq Scan on shoelace_arrive
  1492.   ->  Seq Scan on shoelace_data
  1493. </ProgramListing>
  1494.     while omitting the extra rangetable entry would result in a
  1495. <ProgramListing>
  1496. Merge Join
  1497.   ->  Seq Scan
  1498.         ->  Sort
  1499.               ->  Seq Scan on s
  1500.   ->  Seq Scan
  1501.         ->  Sort
  1502.               ->  Seq Scan on shoelace_arrive
  1503. </ProgramListing>
  1504.     that totally produces the same entries in the log relation.
  1505.     Thus, the rule system caused one extra scan on the
  1506.     <Filename>shoelace_data</Filename> relation that is
  1507.     absolutely not necessary. And the same obsolete scan
  1508.     is done once more in the UPDATE. But it was a really hard
  1509.     job to make that all possible at all.
  1510. </Para>
  1511. <Para>
  1512.     A final demonstration of the <ProductName>Postgres</ProductName>
  1513.     rule system and it's power. There is a cute blonde that
  1514.     sells shoelaces. And what Al could never realize, she's not
  1515.     only cute, she's smart too - a little too smart. Thus, it
  1516.     happens from time to time that Al orders shoelaces that
  1517.     are absolutely not sellable. This time he ordered 1000 pairs
  1518.     of magenta shoelaces and since another kind is currently not
  1519.     available but he committed to buy some, he also prepared
  1520.     his database for pink ones.
  1521. <ProgramListing>
  1522.     al_bundy=> INSERT INTO shoelace VALUES 
  1523.     al_bundy->     ('sl9', 0, 'pink', 35.0, 'inch', 0.0);
  1524.     al_bundy=> INSERT INTO shoelace VALUES 
  1525.     al_bundy->     ('sl10', 1000, 'magenta', 40.0, 'inch', 0.0);
  1526. </ProgramListing>
  1527.     Since this happens often, we must lookup for shoelace entries,
  1528.     that fit for absolutely no shoe sometimes. We could do that in
  1529.     a complicated statement every time, or we can setup a view
  1530.     for it. The view for this is
  1531. <ProgramListing>
  1532.     CREATE VIEW shoelace_obsolete AS
  1533.         SELECT * FROM shoelace WHERE NOT EXISTS
  1534.             (SELECT shoename FROM shoe WHERE slcolor = sl_color);
  1535. </ProgramListing>
  1536.     It's output is
  1537. <ProgramListing>
  1538.     al_bundy=> SELECT * FROM shoelace_obsolete;
  1539.     sl_name   |sl_avail|sl_color  |sl_len|sl_unit |sl_len_cm
  1540.     ----------+--------+----------+------+--------+---------
  1541.     sl9       |       0|pink      |    35|inch    |     88.9
  1542.     sl10      |    1000|magenta   |    40|inch    |    101.6
  1543. </ProgramListing>
  1544.     For the 1000 magenta shoelaces we must debt Al before we can
  1545.     throw 'em away, but that's another problem. The pink entry we delete.
  1546.     To make it a little harder for <ProductName>Postgres</ProductName>,
  1547.     we don't delete it directly. Instead we create one more view
  1548. <ProgramListing>
  1549.     CREATE VIEW shoelace_candelete AS
  1550.         SELECT * FROM shoelace_obsolete WHERE sl_avail = 0;
  1551. </ProgramListing>
  1552.     and do it this way:
  1553. <ProgramListing>
  1554.     DELETE FROM shoelace WHERE EXISTS
  1555.         (SELECT * FROM shoelace_candelete
  1556.                  WHERE sl_name = shoelace.sl_name);
  1557. </ProgramListing>
  1558.     Voila:
  1559. <ProgramListing>
  1560.     al_bundy=> SELECT * FROM shoelace;
  1561.     sl_name   |sl_avail|sl_color  |sl_len|sl_unit |sl_len_cm
  1562.     ----------+--------+----------+------+--------+---------
  1563.     sl1       |       5|black     |    80|cm      |       80
  1564.     sl2       |       6|black     |   100|cm      |      100
  1565.     sl7       |       6|brown     |    60|cm      |       60
  1566.     sl4       |       8|black     |    40|inch    |    101.6
  1567.     sl3       |      10|black     |    35|inch    |     88.9
  1568.     sl8       |      21|brown     |    40|inch    |    101.6
  1569.     sl10      |    1000|magenta   |    40|inch    |    101.6
  1570.     sl5       |       4|brown     |     1|m       |      100
  1571.     sl6       |      20|brown     |   0.9|m       |       90
  1572.     (9 rows)
  1573. </ProgramListing>
  1574.     A DELETE on a view, with a subselect qualification that
  1575.     in total uses 4 nesting/joined views, where one of them
  1576.     itself has a subselect qualification containing a view
  1577.     and where calculated view columns are used,
  1578.     gets rewritten into 
  1579.     one single parsetree that deletes the requested data
  1580.     from a real table.
  1581. </Para>
  1582. <Para>
  1583.     I think there are only a few situations out in the real
  1584.     world, where such a construct is necessary. But
  1585.     it makes me feel comfortable that it works.
  1586.     <Note>
  1587.     <Title>The truth is</Title>
  1588.     <Para>
  1589.     Doing this I found one more bug while writing this document.
  1590.     But after fixing that I was a little amazed that it works at all.
  1591.     </Para>
  1592.     </Note>
  1593. </Para>
  1594. </Sect2>
  1595. </Sect1>
  1596. <Sect1>
  1597. <Title>Rules and Permissions</Title>
  1598. <Para>
  1599.     Due to rewriting of queries by the <ProductName>Postgres</ProductName>
  1600.     rule system, other tables/views than those used in the original
  1601.     query get accessed. Using update rules, this can include write access
  1602.     to tables.
  1603. </Para>
  1604. <Para>
  1605.     Rewrite rules don't have a separate owner. The owner of
  1606.     a relation (table or view) is automatically the owner of the
  1607.     rewrite rules that are defined for it.
  1608.     The <ProductName>Postgres</ProductName> rule system changes the
  1609.     behaviour of the default access control system. Relations that
  1610.     are used due to rules get checked during the rewrite against the
  1611.     permissions of the relation owner, the rule is defined on.
  1612.     This means, that a user does only need the required permissions
  1613.     for the tables/views he names in his queries. 
  1614. </Para>
  1615. <Para>
  1616.     For example: A user has a list of phone numbers where some of
  1617.     them are private, the others are of interest for the secretary of the office.
  1618.     He can construct the following:
  1619. <ProgramListing>
  1620.     CREATE TABLE phone_data (person text, phone text, private bool);
  1621.     CREATE VIEW phone_number AS
  1622.         SELECT person, phone FROM phone_data WHERE NOT private;
  1623.     GRANT SELECT ON phone_number TO secretary;
  1624. </ProgramListing>
  1625.     
  1626.     Nobody except him (and the database superusers) can access the
  1627.     phone_data table. But due to the GRANT, the secretary can SELECT from the
  1628.     phone_number view. The rule system will rewrite
  1629.     the SELECT from phone_number into a SELECT from phone_data and add the qualification
  1630.     that only entries where private is false are wanted. Since the
  1631.     user is the owner of phone_number, the read access to phone_data
  1632.     is now checked against his permissions and the query is considered
  1633.     granted. The check for accessing phone_number is still performed,
  1634.     so nobody than the secretary can use it.
  1635. </Para>
  1636. <Para>
  1637.     The permissions are checked rule by rule. So the secretary is for now the
  1638.     only one who can see the public phone numbers. But the secretary can setup
  1639.     another view and grant access to that to public. Then, anyone
  1640.     can see the phone_number data through the secretaries view.
  1641.     What the secretary cannot do is to create a view that directly
  1642.     accesses phone_data (actually he can, but it will not work since
  1643.     every access aborts the transaction during the permission checks).
  1644.     And as soon as the user will notice, that the secretary opened
  1645.     his phone_number view, he can REVOKE his access. Immediately any
  1646.     access to the secretaries view will fail.
  1647. </Para>
  1648. <Para>
  1649.     Someone might think that this rule by rule checking is a security
  1650.     hole, but in fact it isn't. If this would not work, the secretary
  1651.     could setup a table with the same columns as phone_number and
  1652.     copy the data to there once per day. Then it's his own data and
  1653.     he can grant access to everyone he wants. A GRANT means "I trust you".
  1654.     If someone you trust does the thing above, it's time to
  1655.     think it over and then REVOKE.
  1656. </Para>
  1657. <Para>
  1658.     This mechanism does also work for update rules. In the examples
  1659.     of the previous section, the owner of the tables in Al's database
  1660.     could GRANT SELECT, INSERT, UPDATE and DELETE on the shoelace view to al.
  1661.     But only SELECT on shoelace_log. The rule action to write log entries
  1662.     will still be executed successfull. And Al could see the log entries.
  1663.     But he cannot create fake entries, nor could he manipulate or remove
  1664.     existing ones.
  1665.     <Note>
  1666.     <Title>Warning</Title>
  1667.     <Para>
  1668.     GRANT ALL currently includes RULE permission. This means the granted
  1669.     user could drop the rule, do the changes and reinstall it. I think
  1670.     this should get changed quickly.
  1671.     </Para>
  1672.     </Note>
  1673. </Para>
  1674. </Sect1>
  1675. <Sect1>
  1676. <Title>Rules versus Triggers</Title>
  1677. <Para>
  1678.     Many things that can be done using triggers can also be
  1679.     implemented using the <ProductName>Postgres</ProductName>
  1680.     rule system. What currently cannot be implemented by
  1681.     rules are some kinds of constraints. It is possible,
  1682.     to place a qualified rule that rewrites a query to NOTHING
  1683.     if the value of a column does not appear in another table.
  1684.     But then the data is silently thrown away and that's
  1685.     not a good idea. If checks for valid values are required,
  1686.     and in the case of an invalid value an error message should
  1687.     be generated, it must be done by a trigger for now.
  1688. </Para>
  1689. <Para>
  1690.     On the other hand a trigger that is fired on INSERT 
  1691.     on a view can do the same as a rule, put the data somewhere
  1692.     else and suppress the insert in the view. But it cannot
  1693.     do the same thing on UPDATE or DELETE, because there is
  1694.     no real data in the view relation that could be scanned
  1695.     and thus the trigger would never get called. Only a rule
  1696.     will help.
  1697. </Para>
  1698. <Para>
  1699.     For the things that can be implemented by both,
  1700.     it depends on the usage of the database, which is the best.
  1701.     A trigger is fired for any row affected once. A rule manipulates
  1702.     the parsetree or generates an additional one. So if many
  1703.     rows are affected in one statement, a rule issuing one extra
  1704.     query would usually do a better job than a trigger that is
  1705.     called for any single row and must execute his operations
  1706.     this many times.
  1707. </Para>
  1708. <Para>
  1709.     For example: There are two tables
  1710. <ProgramListing>
  1711.     CREATE TABLE computer (
  1712.         hostname        text     -- indexed
  1713. manufacturer    text     -- indexed
  1714.     );
  1715.     CREATE TABLE software (
  1716.         software        text,    -- indexed
  1717.         hostname        text     -- indexed
  1718.     );
  1719. </ProgramListing>
  1720.     Both tables have many
  1721.     thousands of rows and the index on hostname is unique.
  1722.     The hostname column contains the full qualified domain
  1723.     name of the computer. The rule/trigger should constraint
  1724.     delete rows from software that reference the deleted host.
  1725.     Since the trigger is called for each individual row
  1726.     deleted from computer, it can use the statement
  1727. <ProgramListing>
  1728.     DELETE FROM software WHERE hostname = $1;
  1729. </ProgramListing>
  1730.     in a prepared and saved plan and pass the hostname in
  1731.     the parameter. The rule would be written as
  1732. <ProgramListing>
  1733.     CREATE RULE computer_del AS ON DELETE TO computer
  1734.         DO DELETE FROM software WHERE hostname = OLD.hostname;
  1735. </ProgramListing>
  1736.     Now we look at different types of deletes. In the case of a 
  1737.     
  1738. <ProgramListing>
  1739.     DELETE FROM computer WHERE hostname = 'mypc.local.net';
  1740. </ProgramListing>
  1741.     the table computer is scanned by index (fast) and the
  1742.     query issued by the trigger would also be an index scan (fast too).
  1743.     The extra query from the rule would be a
  1744. <ProgramListing>
  1745.     DELETE FROM software WHERE computer.hostname = 'mypc.local.net'
  1746.                            AND software.hostname = computer.hostname;
  1747. </ProgramListing>
  1748.     Since there are appropriate indices setup, the optimizer
  1749.     will create a plan of
  1750. <ProgramListing>
  1751.     Nestloop
  1752.       ->  Index Scan using comp_hostidx on computer
  1753.       ->  Index Scan using soft_hostidx on software
  1754. </ProgramListing>
  1755.     So there would be not that much difference in speed between
  1756.     the trigger and the rule implementation. With the next delete
  1757.     we want to get rid of all the 2000 computers where the hostname starts
  1758.     with 'old'. There are two possible queries to do that. One is
  1759. <ProgramListing>
  1760.     DELETE FROM computer WHERE hostname >= 'old'
  1761.                            AND hostname <  'ole'
  1762. </ProgramListing>
  1763.     Where the plan for the rule query will be a 
  1764. <ProgramListing>
  1765.     Hash Join
  1766.       ->  Seq Scan on software
  1767.       ->  Hash
  1768.     ->  Index Scan using comp_hostidx on computer
  1769. </ProgramListing>
  1770.     The other possible query is a
  1771. <ProgramListing>
  1772.     DELETE FROM computer WHERE hostname ~ '^old';
  1773. </ProgramListing>
  1774.     with the execution plan
  1775. <ProgramListing>
  1776.     Nestloop
  1777.       ->  Index Scan using comp_hostidx on computer
  1778.       ->  Index Scan using soft_hostidx on software
  1779. </ProgramListing>
  1780.     This shows, that the optimizer does not realize that the
  1781.     qualification for the hostname on computer could also be
  1782.     used for an index scan on software when there are
  1783.     multiple qualification expressions combined with AND, what
  1784.     he does in the regexp version of the query. The trigger will
  1785.     get invoked once for any of the 2000 old computers that
  1786.     have to be deleted and that will result in one index scan
  1787.     over computer and 2000 index scans for the software. The
  1788.     rule implementation will do it with two queries over indices.
  1789.     And it depends on the overall size of the software table if
  1790.     the rule will still be faster in the seqscan situation. 2000
  1791.     query executions over the SPI manager take some time, even
  1792.     if all the index blocks to look them up will soon appear in
  1793.     the cache.
  1794. </Para>
  1795. <Para>
  1796.     The last query we look at is a
  1797. <ProgramListing>
  1798.     DELETE FROM computer WHERE manufacurer = 'bim';
  1799. </ProgramListing>
  1800.     Again this could result in many rows to be deleted from
  1801.     computer. So the trigger will again fire many queries into
  1802.     the executor. But the rule plan will again be the Nestloop over
  1803.     two IndexScan's. Only using another index on computer:
  1804. <ProgramListing>
  1805.     Nestloop
  1806.       ->  Index Scan using comp_manufidx on computer
  1807.       ->  Index Scan using soft_hostidx on software
  1808. </ProgramListing>
  1809.     resulting from the rules query
  1810. <ProgramListing>
  1811.     DELETE FROM software WHERE computer.manufacurer = 'bim'
  1812.                            AND software.hostname = computer.hostname;
  1813. </ProgramListing>
  1814.     In any of these cases, the extra queries from the rule system will be
  1815.     more or less independent from the number of affected rows
  1816.     in a query. 
  1817. </Para>
  1818. <Para>
  1819.     Another situation are cases on UPDATE where it depends on the
  1820.     change of an attribute if an action should be performed or
  1821.     not. In <ProductName>Postgres</ProductName> version 6.4, the
  1822.     attribute specification for rule events is disabled (it will have
  1823.     it's comeback latest in 6.5, maybe earlier 
  1824.     - stay tuned). So for now the only way to
  1825.     create a rule as in the shoelace_log example is to do it with
  1826.     a rule qualification. That results in an extra query that is
  1827.     performed allways, even if the attribute of interest cannot
  1828.     change at all because it does not appear in the targetlist
  1829.     of the initial query. When this is enabled again, it will be
  1830.     one more advantage of rules over triggers. Optimization of
  1831.     a trigger must fail by definition in this case, because the
  1832.     fact that it's actions will only be done when a specific attribute
  1833.     is updated is hidden in it's functionality. The definition of
  1834.     a trigger only allows to specify it on row level, so whenever a
  1835.     row is touched, the trigger must be called to make it's
  1836.     decision. The rule system will know it by looking up the
  1837.     targetlist and will suppress the additional query completely
  1838.     if the attribute isn't touched. So the rule, qualified or not,
  1839.     will only do it's scan's if there ever could be something to do.
  1840. </Para>
  1841. <Para>
  1842.     Rules will only be significant slower than triggers if
  1843.     their actions result in large and bad qualified joins, a situation
  1844.     where the optimizer fails. They are a big hammer.
  1845.     Using a big hammer without caution can cause big damage. But
  1846.     used with the right touch, they can hit any nail on the head.
  1847. </Para>
  1848. </Sect1>
  1849. </Chapter>