Commit 1b4f7f93 authored by Tom Lane's avatar Tom Lane

Allow empty target list in SELECT.

This fixes a problem noted as a followup to bug #8648: if a query has a
semantically-empty target list, e.g. SELECT * FROM zero_column_table,
ruleutils.c will dump it as a syntactically-empty target list, which was
not allowed.  There doesn't seem to be any reliable way to fix this by
hacking ruleutils (note in particular that the originally zero-column table
might since have had columns added to it); and even if we had such a fix,
it would do nothing for existing dump files that might contain bad syntax.
The best bet seems to be to relax the syntactic restriction.

Also, add parse-analysis errors for SELECT DISTINCT with no columns (after
*-expansion) and RETURNING with no columns.  These cases previously
produced unexpected behavior because the parsed Query looked like it had
no DISTINCT or RETURNING clause, respectively.  If anyone ever offers
a plausible use-case for this, we could work a bit harder on making the
situation distinguishable.

Arguably this is a bug fix that should be back-patched, but I'm worried
that there may be client apps or PLs that expect "SELECT ;" to throw a
syntax error.  The issue doesn't seem important enough to risk changing
behavior in minor releases.
parent c03ad560
...@@ -34,7 +34,7 @@ PostgreSQL documentation ...@@ -34,7 +34,7 @@ PostgreSQL documentation
<synopsis> <synopsis>
[ WITH [ RECURSIVE ] <replaceable class="parameter">with_query</replaceable> [, ...] ] [ WITH [ RECURSIVE ] <replaceable class="parameter">with_query</replaceable> [, ...] ]
SELECT [ ALL | DISTINCT [ ON ( <replaceable class="parameter">expression</replaceable> [, ...] ) ] ] SELECT [ ALL | DISTINCT [ ON ( <replaceable class="parameter">expression</replaceable> [, ...] ) ] ]
* | <replaceable class="parameter">expression</replaceable> [ [ AS ] <replaceable class="parameter">output_name</replaceable> ] [, ...] [ * | <replaceable class="parameter">expression</replaceable> [ [ AS ] <replaceable class="parameter">output_name</replaceable> ] [, ...] ]
[ FROM <replaceable class="parameter">from_item</replaceable> [, ...] ] [ FROM <replaceable class="parameter">from_item</replaceable> [, ...] ]
[ WHERE <replaceable class="parameter">condition</replaceable> ] [ WHERE <replaceable class="parameter">condition</replaceable> ]
[ GROUP BY <replaceable class="parameter">expression</replaceable> [, ...] ] [ GROUP BY <replaceable class="parameter">expression</replaceable> [, ...] ]
...@@ -1740,13 +1740,27 @@ SELECT 2+2; ...@@ -1740,13 +1740,27 @@ SELECT 2+2;
following query is invalid: following query is invalid:
<programlisting> <programlisting>
SELECT distributors.* WHERE distributors.name = 'Westward'; SELECT distributors.* WHERE distributors.name = 'Westward';
</programlisting><productname>PostgreSQL</productname> releases prior to </programlisting>
<productname>PostgreSQL</productname> releases prior to
8.1 would accept queries of this form, and add an implicit entry 8.1 would accept queries of this form, and add an implicit entry
to the query's <literal>FROM</literal> clause for each table to the query's <literal>FROM</literal> clause for each table
referenced by the query. This is no longer allowed. referenced by the query. This is no longer allowed.
</para> </para>
</refsect2> </refsect2>
<refsect2>
<title>Empty <literal>SELECT</literal> Lists</title>
<para>
The list of output expressions after <literal>SELECT</literal> can be
empty, producing a zero-column result table.
This is not valid syntax according to the SQL standard.
<productname>PostgreSQL</productname> allows it to be consistent with
allowing zero-column tables.
However, an empty list is not allowed when <literal>DISTINCT</> is used.
</para>
</refsect2>
<refsect2> <refsect2>
<title>Omitting the <literal>AS</literal> Key Word</title> <title>Omitting the <literal>AS</literal> Key Word</title>
...@@ -1809,10 +1823,6 @@ SELECT distributors.* WHERE distributors.name = 'Westward'; ...@@ -1809,10 +1823,6 @@ SELECT distributors.* WHERE distributors.name = 'Westward';
<productname>PostgreSQL</productname> treats <literal>UNNEST()</> the <productname>PostgreSQL</productname> treats <literal>UNNEST()</> the
same as other set-returning functions. same as other set-returning functions.
</para> </para>
<para>
<literal>ROWS FROM( ... )</> is an extension of the SQL standard.
</para>
</refsect2> </refsect2>
<refsect2> <refsect2>
...@@ -1910,9 +1920,13 @@ SELECT distributors.* WHERE distributors.name = 'Westward'; ...@@ -1910,9 +1920,13 @@ SELECT distributors.* WHERE distributors.name = 'Westward';
<title>Nonstandard Clauses</title> <title>Nonstandard Clauses</title>
<para> <para>
The clause <literal>DISTINCT ON</literal> is not defined in the <literal>DISTINCT ON ( ... )</literal> is an extension of the
SQL standard. SQL standard.
</para> </para>
<para>
<literal>ROWS FROM( ... )</> is an extension of the SQL standard.
</para>
</refsect2> </refsect2>
</refsect1> </refsect1>
</refentry> </refentry>
...@@ -2018,6 +2018,19 @@ transformReturningList(ParseState *pstate, List *returningList) ...@@ -2018,6 +2018,19 @@ transformReturningList(ParseState *pstate, List *returningList)
/* transform RETURNING identically to a SELECT targetlist */ /* transform RETURNING identically to a SELECT targetlist */
rlist = transformTargetList(pstate, returningList, EXPR_KIND_RETURNING); rlist = transformTargetList(pstate, returningList, EXPR_KIND_RETURNING);
/*
* Complain if the nonempty tlist expanded to nothing (which is possible
* if it contains only a star-expansion of a zero-column table). If we
* allow this, the parsed Query will look like it didn't have RETURNING,
* with results that would probably surprise the user.
*/
if (rlist == NIL)
ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR),
errmsg("RETURNING must have at least one column"),
parser_errposition(pstate,
exprLocation(linitial(returningList)))));
/* mark column origins */ /* mark column origins */
markTargetListOrigins(pstate, rlist); markTargetListOrigins(pstate, rlist);
......
...@@ -334,7 +334,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query); ...@@ -334,7 +334,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query);
name_list from_clause from_list opt_array_bounds name_list from_clause from_list opt_array_bounds
qualified_name_list any_name any_name_list qualified_name_list any_name any_name_list
any_operator expr_list attrs any_operator expr_list attrs
target_list insert_column_list set_target_list target_list opt_target_list insert_column_list set_target_list
set_clause_list set_clause multiple_set_clause set_clause_list set_clause multiple_set_clause
ctext_expr_list ctext_row def_list indirection opt_indirection ctext_expr_list ctext_row def_list indirection opt_indirection
reloption_list group_clause TriggerFuncArgs select_limit reloption_list group_clause TriggerFuncArgs select_limit
...@@ -9259,7 +9259,7 @@ select_clause: ...@@ -9259,7 +9259,7 @@ select_clause:
* However, this is not checked by the grammar; parse analysis must check it. * However, this is not checked by the grammar; parse analysis must check it.
*/ */
simple_select: simple_select:
SELECT opt_distinct target_list SELECT opt_distinct opt_target_list
into_clause from_clause where_clause into_clause from_clause where_clause
group_clause having_clause window_clause group_clause having_clause window_clause
{ {
...@@ -12215,6 +12215,10 @@ ctext_row: '(' ctext_expr_list ')' { $$ = $2; } ...@@ -12215,6 +12215,10 @@ ctext_row: '(' ctext_expr_list ')' { $$ = $2; }
* *
*****************************************************************************/ *****************************************************************************/
opt_target_list: target_list { $$ = $1; }
| /* EMPTY */ { $$ = NIL; }
;
target_list: target_list:
target_el { $$ = list_make1($1); } target_el { $$ = list_make1($1); }
| target_list ',' target_el { $$ = lappend($1, $3); } | target_list ',' target_el { $$ = lappend($1, $3); }
......
...@@ -2011,6 +2011,20 @@ transformDistinctClause(ParseState *pstate, ...@@ -2011,6 +2011,20 @@ transformDistinctClause(ParseState *pstate,
true); true);
} }
/*
* Complain if we found nothing to make DISTINCT. Returning an empty list
* would cause the parsed Query to look like it didn't have DISTINCT, with
* results that would probably surprise the user. Note: this case is
* presently impossible for aggregates because of grammar restrictions,
* but we check anyway.
*/
if (result == NIL)
ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR),
is_agg ?
errmsg("an aggregate with DISTINCT must have at least one argument") :
errmsg("SELECT DISTINCT must have at least one column")));
return result; return result;
} }
...@@ -2115,6 +2129,11 @@ transformDistinctOnClause(ParseState *pstate, List *distinctlist, ...@@ -2115,6 +2129,11 @@ transformDistinctOnClause(ParseState *pstate, List *distinctlist,
true); true);
} }
/*
* An empty result list is impossible here because of grammar restrictions.
*/
Assert(result != NIL);
return result; return result;
} }
......
...@@ -15,26 +15,24 @@ select 1; ...@@ -15,26 +15,24 @@ select 1;
-- --
-- --
-- SELECT -- SELECT
-- missing relation name -- this used to be a syntax error, but now we allow an empty target list
select; select;
ERROR: syntax error at or near ";" --
LINE 1: select; (1 row)
^
-- no such relation -- no such relation
select * from nonesuch; select * from nonesuch;
ERROR: relation "nonesuch" does not exist ERROR: relation "nonesuch" does not exist
LINE 1: select * from nonesuch; LINE 1: select * from nonesuch;
^ ^
-- missing target list
select from pg_database;
ERROR: syntax error at or near "from"
LINE 1: select from pg_database;
^
-- bad name in target list -- bad name in target list
select nonesuch from pg_database; select nonesuch from pg_database;
ERROR: column "nonesuch" does not exist ERROR: column "nonesuch" does not exist
LINE 1: select nonesuch from pg_database; LINE 1: select nonesuch from pg_database;
^ ^
-- empty distinct list isn't OK
select distinct from pg_database;
ERROR: SELECT DISTINCT must have at least one column
-- bad attribute name on lhs of operator -- bad attribute name on lhs of operator
select * from pg_database where nonesuch = pg_database.datname; select * from pg_database where nonesuch = pg_database.datname;
ERROR: column "nonesuch" does not exist ERROR: column "nonesuch" does not exist
...@@ -45,12 +43,7 @@ select * from pg_database where pg_database.datname = nonesuch; ...@@ -45,12 +43,7 @@ select * from pg_database where pg_database.datname = nonesuch;
ERROR: column "nonesuch" does not exist ERROR: column "nonesuch" does not exist
LINE 1: ...ect * from pg_database where pg_database.datname = nonesuch; LINE 1: ...ect * from pg_database where pg_database.datname = nonesuch;
^ ^
-- bad select distinct on syntax, distinct attribute missing -- bad attribute name in select distinct on
select distinct on (foobar) from pg_database;
ERROR: syntax error at or near "from"
LINE 1: select distinct on (foobar) from pg_database;
^
-- bad select distinct on syntax, distinct attribute not in target list
select distinct on (foobar) * from pg_database; select distinct on (foobar) * from pg_database;
ERROR: column "foobar" does not exist ERROR: column "foobar" does not exist
LINE 1: select distinct on (foobar) * from pg_database; LINE 1: select distinct on (foobar) * from pg_database;
......
...@@ -16,28 +16,25 @@ select 1; ...@@ -16,28 +16,25 @@ select 1;
-- --
-- SELECT -- SELECT
-- missing relation name -- this used to be a syntax error, but now we allow an empty target list
select; select;
-- no such relation -- no such relation
select * from nonesuch; select * from nonesuch;
-- missing target list
select from pg_database;
-- bad name in target list -- bad name in target list
select nonesuch from pg_database; select nonesuch from pg_database;
-- empty distinct list isn't OK
select distinct from pg_database;
-- bad attribute name on lhs of operator -- bad attribute name on lhs of operator
select * from pg_database where nonesuch = pg_database.datname; select * from pg_database where nonesuch = pg_database.datname;
-- bad attribute name on rhs of operator -- bad attribute name on rhs of operator
select * from pg_database where pg_database.datname = nonesuch; select * from pg_database where pg_database.datname = nonesuch;
-- bad attribute name in select distinct on
-- bad select distinct on syntax, distinct attribute missing
select distinct on (foobar) from pg_database;
-- bad select distinct on syntax, distinct attribute not in target list
select distinct on (foobar) * from pg_database; select distinct on (foobar) * from pg_database;
......
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