summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTom Lane2010-10-03 00:02:41 +0000
committerTom Lane2010-10-03 00:02:41 +0000
commit9e718e6116d2ace2faff8cce3d907fcf69bb6416 (patch)
treed4b80712492df38397fda1699a145b7097f57389
parent4b1501cb119f90892c0e0c2fabbd368baf7a106e (diff)
Behave correctly if INSERT ... VALUES is decorated with additional clauses.
In versions 8.2 and up, the grammar allows attaching ORDER BY, LIMIT, FOR UPDATE, or WITH to VALUES, and hence to INSERT ... VALUES. But the special-case code for VALUES in transformInsertStmt() wasn't expecting any of those, and just ignored them, leading to unexpected results. Rather than complicate the special-case path, just ensure that the presence of any of those clauses makes us treat the query as if it had a general SELECT. Per report from Hitoshi Harada.
-rw-r--r--src/backend/parser/analyze.c11
1 files changed, 10 insertions, 1 deletions
diff --git a/src/backend/parser/analyze.c b/src/backend/parser/analyze.c
index 244414a61f3..c1868664ff2 100644
--- a/src/backend/parser/analyze.c
+++ b/src/backend/parser/analyze.c
@@ -355,8 +355,17 @@ transformInsertStmt(ParseState *pstate, InsertStmt *stmt)
* We have three cases to deal with: DEFAULT VALUES (selectStmt == NULL),
* VALUES list, or general SELECT input. We special-case VALUES, both for
* efficiency and so we can handle DEFAULT specifications.
+ *
+ * The grammar allows attaching ORDER BY, LIMIT, FOR UPDATE, or WITH to a
+ * VALUES clause. If we have any of those, treat it as a general SELECT;
+ * so it will work, but you can't use DEFAULT items together with those.
*/
- isGeneralSelect = (selectStmt && selectStmt->valuesLists == NIL);
+ isGeneralSelect = (selectStmt && (selectStmt->valuesLists == NIL ||
+ selectStmt->sortClause != NIL ||
+ selectStmt->limitOffset != NULL ||
+ selectStmt->limitCount != NULL ||
+ selectStmt->lockingClause != NIL ||
+ selectStmt->withClause != NULL));
/*
* If a non-nil rangetable/namespace was passed in, and we are doing