Commit 2378d79a authored by Tom Lane's avatar Tom Lane

Make LATERAL implicit for functions in FROM.

The SQL standard does not have general functions-in-FROM, but it does
allow UNNEST() there (see the <collection derived table> production),
and the semantics of that are defined to include lateral references.
So spec compliance requires allowing lateral references within UNNEST()
even without an explicit LATERAL keyword.  Rather than making UNNEST()
a special case, it seems best to extend this flexibility to any
function-in-FROM.  We'll still allow LATERAL to be written explicitly
for clarity's sake, but it's now a noise word in this context.

In theory this change could result in a change in behavior of existing
queries, by allowing what had been an outer reference in a function-in-FROM
to be captured by an earlier FROM-item at the same level.  However, all
pre-9.3 PG releases have a bug that causes them to match variable
references to earlier FROM-items in preference to outer references (and
then throw an error).  So no previously-working query could contain the
type of ambiguity that would risk a change of behavior.

Per a suggestion from Andrew Gierth, though I didn't use his patch.
parent 8865fe0a
...@@ -717,14 +717,24 @@ SELECT * ...@@ -717,14 +717,24 @@ SELECT *
</indexterm> </indexterm>
<para> <para>
Subqueries and table functions appearing in <literal>FROM</> can be Subqueries appearing in <literal>FROM</> can be
preceded by the key word <literal>LATERAL</>. This allows them to preceded by the key word <literal>LATERAL</>. This allows them to
reference columns provided by preceding <literal>FROM</> items. reference columns provided by preceding <literal>FROM</> items.
(Without <literal>LATERAL</literal>, each <literal>FROM</> item is (Without <literal>LATERAL</literal>, each subquery is
evaluated independently and so cannot cross-reference any other evaluated independently and so cannot cross-reference any other
<literal>FROM</> item.) <literal>FROM</> item.)
</para>
<para>
Table functions appearing in <literal>FROM</> can also be
preceded by the key word <literal>LATERAL</>, but for functions the
key word is optional; the function's arguments can contain references
to columns provided by preceding <literal>FROM</> items in any case.
</para>
<para>
A <literal>LATERAL</literal> item can appear at top level in the A <literal>LATERAL</literal> item can appear at top level in the
<literal>FROM</> list, or within a <literal>JOIN</> tree; in the latter <literal>FROM</> list, or within a <literal>JOIN</> tree. In the latter
case it can also refer to any items that are on the left-hand side of a case it can also refer to any items that are on the left-hand side of a
<literal>JOIN</> that it is on the right-hand side of. <literal>JOIN</> that it is on the right-hand side of.
</para> </para>
...@@ -770,7 +780,9 @@ FROM polygons p1 CROSS JOIN LATERAL vertices(p1.poly) v1, ...@@ -770,7 +780,9 @@ FROM polygons p1 CROSS JOIN LATERAL vertices(p1.poly) v1,
polygons p2 CROSS JOIN LATERAL vertices(p2.poly) v2 polygons p2 CROSS JOIN LATERAL vertices(p2.poly) v2
WHERE (v1 &lt;-&gt; v2) &lt; 10 AND p1.id != p2.id; WHERE (v1 &lt;-&gt; v2) &lt; 10 AND p1.id != p2.id;
</programlisting> </programlisting>
or in several other equivalent formulations. or in several other equivalent formulations. (As already mentioned,
the <literal>LATERAL</> key word is unnecessary in this example, but
we use it for clarity.)
</para> </para>
<para> <para>
......
...@@ -504,18 +504,28 @@ TABLE [ ONLY ] <replaceable class="parameter">table_name</replaceable> [ * ] ...@@ -504,18 +504,28 @@ TABLE [ ONLY ] <replaceable class="parameter">table_name</replaceable> [ * ]
<varlistentry> <varlistentry>
<term><literal>LATERAL</literal></term> <term><literal>LATERAL</literal></term>
<listitem> <listitem>
<para>The <literal>LATERAL</literal> key word can precede a <para>
sub-<command>SELECT</command> or function-call <literal>FROM</> The <literal>LATERAL</literal> key word can precede a
item. This allows the sub-<command>SELECT</command> or function sub-<command>SELECT</command> <literal>FROM</> item. This allows the
expression to refer to columns of <literal>FROM</> items that appear sub-<command>SELECT</command> to refer to columns of <literal>FROM</>
before it in the <literal>FROM</> list. (Without items that appear before it in the <literal>FROM</> list. (Without
<literal>LATERAL</literal>, each <literal>FROM</> item is evaluated <literal>LATERAL</literal>, each sub-<command>SELECT</command> is
independently and so cannot cross-reference any other evaluated independently and so cannot cross-reference any other
<literal>FROM</> item.) A <literal>LATERAL</literal> item can <literal>FROM</> item.)
appear at top level in the <literal>FROM</> list, or within a </para>
<literal>JOIN</> tree; in the latter case it can also refer to any
items that are on the left-hand side of a <literal>JOIN</> that it is <para>
on the right-hand side of. <literal>LATERAL</literal> can also precede a function-call
<literal>FROM</> item, but in this case it is a noise word, because
the function expression can refer to earlier <literal>FROM</> items
in any case.
</para>
<para>
A <literal>LATERAL</literal> item can appear at top level in the
<literal>FROM</> list, or within a <literal>JOIN</> tree. In the
latter case it can also refer to any items that are on the left-hand
side of a <literal>JOIN</> that it is on the right-hand side of.
</para> </para>
<para> <para>
...@@ -1738,7 +1748,12 @@ SELECT distributors.* WHERE distributors.name = 'Westward'; ...@@ -1738,7 +1748,12 @@ SELECT distributors.* WHERE distributors.name = 'Westward';
sub-<command>SELECT</command>; that is, the syntax sub-<command>SELECT</command>; that is, the syntax
<literal>FROM <replaceable>func</>(...) <replaceable>alias</></literal> <literal>FROM <replaceable>func</>(...) <replaceable>alias</></literal>
is approximately equivalent to is approximately equivalent to
<literal>FROM (SELECT <replaceable>func</>(...)) <replaceable>alias</></literal>. <literal>FROM LATERAL (SELECT <replaceable>func</>(...)) <replaceable>alias</></literal>.
Note that <literal>LATERAL</> is considered to be implicit; this is
because the standard requires <literal>LATERAL</> semantics for an
<literal>UNNEST()</> item in <literal>FROM</>.
<productname>PostgreSQL</productname> treats <literal>UNNEST()</> the
same as other set-returning functions.
</para> </para>
</refsect2> </refsect2>
......
...@@ -503,6 +503,7 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r) ...@@ -503,6 +503,7 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r)
{ {
Node *funcexpr; Node *funcexpr;
char *funcname; char *funcname;
bool is_lateral;
RangeTblEntry *rte; RangeTblEntry *rte;
/* /*
...@@ -514,12 +515,16 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r) ...@@ -514,12 +515,16 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r)
funcname = FigureColname(r->funccallnode); funcname = FigureColname(r->funccallnode);
/* /*
* If the function is LATERAL, make lateral_only names of this level * We make lateral_only names of this level visible, whether or not the
* visible to it. (LATERAL can't nest within a single pstate level, so we * function is explicitly marked LATERAL. This is needed for SQL spec
* don't need save/restore logic here.) * compliance in the case of UNNEST(), and seems useful on convenience
* grounds for all functions in FROM.
*
* (LATERAL can't nest within a single pstate level, so we don't need
* save/restore logic here.)
*/ */
Assert(!pstate->p_lateral_active); Assert(!pstate->p_lateral_active);
pstate->p_lateral_active = r->lateral; pstate->p_lateral_active = true;
/* /*
* Transform the raw expression. * Transform the raw expression.
...@@ -533,11 +538,17 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r) ...@@ -533,11 +538,17 @@ transformRangeFunction(ParseState *pstate, RangeFunction *r)
*/ */
assign_expr_collations(pstate, funcexpr); assign_expr_collations(pstate, funcexpr);
/*
* Mark the RTE as LATERAL if the user said LATERAL explicitly, or if
* there are any lateral cross-references in it.
*/
is_lateral = r->lateral || contain_vars_of_level(funcexpr, 0);
/* /*
* OK, build an RTE for the function. * OK, build an RTE for the function.
*/ */
rte = addRangeTableEntryForFunction(pstate, funcname, funcexpr, rte = addRangeTableEntryForFunction(pstate, funcname, funcexpr,
r, r->lateral, true); r, is_lateral, true);
/* /*
* If a coldeflist was supplied, ensure it defines a legal set of names * If a coldeflist was supplied, ensure it defines a legal set of names
......
...@@ -3157,7 +3157,7 @@ select *, (select r from (select q1 as q2) x, lateral (select q2 as r) y) from i ...@@ -3157,7 +3157,7 @@ select *, (select r from (select q1 as q2) x, lateral (select q2 as r) y) from i
4567890123456789 | -4567890123456789 | 4567890123456789 4567890123456789 | -4567890123456789 | 4567890123456789
(5 rows) (5 rows)
-- lateral SRF -- lateral with function in FROM
select count(*) from tenk1 a, lateral generate_series(1,two) g; select count(*) from tenk1 a, lateral generate_series(1,two) g;
count count
------- -------
...@@ -3184,6 +3184,17 @@ explain (costs off) ...@@ -3184,6 +3184,17 @@ explain (costs off)
-> Function Scan on generate_series g -> Function Scan on generate_series g
(4 rows) (4 rows)
-- don't need the explicit LATERAL keyword for functions
explain (costs off)
select count(*) from tenk1 a, generate_series(1,two) g;
QUERY PLAN
------------------------------------------------
Aggregate
-> Nested Loop
-> Seq Scan on tenk1 a
-> Function Scan on generate_series g
(4 rows)
-- lateral with UNION ALL subselect -- lateral with UNION ALL subselect
explain (costs off) explain (costs off)
select * from generate_series(100,200) g, select * from generate_series(100,200) g,
...@@ -3578,25 +3589,25 @@ select * from ...@@ -3578,25 +3589,25 @@ select * from
(26 rows) (26 rows)
-- test some error cases where LATERAL should have been used but wasn't -- test some error cases where LATERAL should have been used but wasn't
select f1,g from int4_tbl a, generate_series(0, f1) g; select f1,g from int4_tbl a, (select f1 as g) ss;
ERROR: column "f1" does not exist ERROR: column "f1" does not exist
LINE 1: select f1,g from int4_tbl a, generate_series(0, f1) g; LINE 1: select f1,g from int4_tbl a, (select f1 as g) ss;
^ ^
HINT: There is a column named "f1" in table "a", but it cannot be referenced from this part of the query. HINT: There is a column named "f1" in table "a", but it cannot be referenced from this part of the query.
select f1,g from int4_tbl a, generate_series(0, a.f1) g; select f1,g from int4_tbl a, (select a.f1 as g) ss;
ERROR: invalid reference to FROM-clause entry for table "a" ERROR: invalid reference to FROM-clause entry for table "a"
LINE 1: select f1,g from int4_tbl a, generate_series(0, a.f1) g; LINE 1: select f1,g from int4_tbl a, (select a.f1 as g) ss;
^ ^
HINT: There is an entry for table "a", but it cannot be referenced from this part of the query. HINT: There is an entry for table "a", but it cannot be referenced from this part of the query.
select f1,g from int4_tbl a cross join generate_series(0, f1) g; select f1,g from int4_tbl a cross join (select f1 as g) ss;
ERROR: column "f1" does not exist ERROR: column "f1" does not exist
LINE 1: ...ct f1,g from int4_tbl a cross join generate_series(0, f1) g; LINE 1: select f1,g from int4_tbl a cross join (select f1 as g) ss;
^ ^
HINT: There is a column named "f1" in table "a", but it cannot be referenced from this part of the query. HINT: There is a column named "f1" in table "a", but it cannot be referenced from this part of the query.
select f1,g from int4_tbl a cross join generate_series(0, a.f1) g; select f1,g from int4_tbl a cross join (select a.f1 as g) ss;
ERROR: invalid reference to FROM-clause entry for table "a" ERROR: invalid reference to FROM-clause entry for table "a"
LINE 1: ... f1,g from int4_tbl a cross join generate_series(0, a.f1) g; LINE 1: select f1,g from int4_tbl a cross join (select a.f1 as g) ss...
^ ^
HINT: There is an entry for table "a", but it cannot be referenced from this part of the query. HINT: There is an entry for table "a", but it cannot be referenced from this part of the query.
-- SQL:2008 says the left table is in scope but illegal to access here -- SQL:2008 says the left table is in scope but illegal to access here
select f1,g from int4_tbl a right join lateral generate_series(0, a.f1) g on true; select f1,g from int4_tbl a right join lateral generate_series(0, a.f1) g on true;
......
...@@ -19,12 +19,15 @@ INSERT INTO foo2 VALUES(1, 11); ...@@ -19,12 +19,15 @@ INSERT INTO foo2 VALUES(1, 11);
INSERT INTO foo2 VALUES(2, 22); INSERT INTO foo2 VALUES(2, 22);
INSERT INTO foo2 VALUES(1, 111); INSERT INTO foo2 VALUES(1, 111);
CREATE FUNCTION foot(int) returns setof foo2 as 'SELECT * FROM foo2 WHERE fooid = $1;' LANGUAGE SQL; CREATE FUNCTION foot(int) returns setof foo2 as 'SELECT * FROM foo2 WHERE fooid = $1;' LANGUAGE SQL;
-- supposed to fail with ERROR -- function with implicit LATERAL
select * from foo2, foot(foo2.fooid) z where foo2.f2 = z.f2; select * from foo2, foot(foo2.fooid) z where foo2.f2 = z.f2;
ERROR: invalid reference to FROM-clause entry for table "foo2" fooid | f2 | fooid | f2
LINE 1: select * from foo2, foot(foo2.fooid) z where foo2.f2 = z.f2; -------+-----+-------+-----
^ 1 | 11 | 1 | 11
HINT: There is an entry for table "foo2", but it cannot be referenced from this part of the query. 2 | 22 | 2 | 22
1 | 111 | 1 | 111
(3 rows)
-- function in subselect -- function in subselect
select * from foo2 where f2 in (select f2 from foot(foo2.fooid) z where z.fooid = foo2.fooid) ORDER BY 1,2; select * from foo2 where f2 in (select f2 from foot(foo2.fooid) z where z.fooid = foo2.fooid) ORDER BY 1,2;
fooid | f2 fooid | f2
......
...@@ -901,12 +901,15 @@ explain (costs off) ...@@ -901,12 +901,15 @@ explain (costs off)
select *, (select r from (select q1 as q2) x, (select q2 as r) y) from int8_tbl; select *, (select r from (select q1 as q2) x, (select q2 as r) y) from int8_tbl;
select *, (select r from (select q1 as q2) x, lateral (select q2 as r) y) from int8_tbl; select *, (select r from (select q1 as q2) x, lateral (select q2 as r) y) from int8_tbl;
-- lateral SRF -- lateral with function in FROM
select count(*) from tenk1 a, lateral generate_series(1,two) g; select count(*) from tenk1 a, lateral generate_series(1,two) g;
explain (costs off) explain (costs off)
select count(*) from tenk1 a, lateral generate_series(1,two) g; select count(*) from tenk1 a, lateral generate_series(1,two) g;
explain (costs off) explain (costs off)
select count(*) from tenk1 a cross join lateral generate_series(1,two) g; select count(*) from tenk1 a cross join lateral generate_series(1,two) g;
-- don't need the explicit LATERAL keyword for functions
explain (costs off)
select count(*) from tenk1 a, generate_series(1,two) g;
-- lateral with UNION ALL subselect -- lateral with UNION ALL subselect
explain (costs off) explain (costs off)
...@@ -987,10 +990,10 @@ select * from ...@@ -987,10 +990,10 @@ select * from
lateral (select ss2.y) ss3; lateral (select ss2.y) ss3;
-- test some error cases where LATERAL should have been used but wasn't -- test some error cases where LATERAL should have been used but wasn't
select f1,g from int4_tbl a, generate_series(0, f1) g; select f1,g from int4_tbl a, (select f1 as g) ss;
select f1,g from int4_tbl a, generate_series(0, a.f1) g; select f1,g from int4_tbl a, (select a.f1 as g) ss;
select f1,g from int4_tbl a cross join generate_series(0, f1) g; select f1,g from int4_tbl a cross join (select f1 as g) ss;
select f1,g from int4_tbl a cross join generate_series(0, a.f1) g; select f1,g from int4_tbl a cross join (select a.f1 as g) ss;
-- SQL:2008 says the left table is in scope but illegal to access here -- SQL:2008 says the left table is in scope but illegal to access here
select f1,g from int4_tbl a right join lateral generate_series(0, a.f1) g on true; select f1,g from int4_tbl a right join lateral generate_series(0, a.f1) g on true;
select f1,g from int4_tbl a full join lateral generate_series(0, a.f1) g on true; select f1,g from int4_tbl a full join lateral generate_series(0, a.f1) g on true;
......
...@@ -7,7 +7,7 @@ INSERT INTO foo2 VALUES(1, 111); ...@@ -7,7 +7,7 @@ INSERT INTO foo2 VALUES(1, 111);
CREATE FUNCTION foot(int) returns setof foo2 as 'SELECT * FROM foo2 WHERE fooid = $1;' LANGUAGE SQL; CREATE FUNCTION foot(int) returns setof foo2 as 'SELECT * FROM foo2 WHERE fooid = $1;' LANGUAGE SQL;
-- supposed to fail with ERROR -- function with implicit LATERAL
select * from foo2, foot(foo2.fooid) z where foo2.f2 = z.f2; select * from foo2, foot(foo2.fooid) z where foo2.f2 = z.f2;
-- function in subselect -- function in subselect
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment