Commit 357889eb authored by Alvaro Herrera's avatar Alvaro Herrera

Support FETCH FIRST WITH TIES

WITH TIES is an option to the FETCH FIRST N ROWS clause (the SQL
standard's spelling of LIMIT), where you additionally get rows that
compare equal to the last of those N rows by the columns in the
mandatory ORDER BY clause.

There was a proposal by Andrew Gierth to implement this functionality in
a more powerful way that would yield more features, but the other patch
had not been finished at this time, so we decided to use this one for
now in the spirit of incremental development.

Author: Surafel Temesgen <surafel3000@gmail.com>
Reviewed-by: default avatarÁlvaro Herrera <alvherre@alvh.no-ip.org>
Reviewed-by: default avatarTomas Vondra <tomas.vondra@2ndquadrant.com>
Discussion: https://postgr.es/m/CALAY4q9ky7rD_A4vf=FVQvCGngm3LOes-ky0J6euMrg=_Se+ag@mail.gmail.com
Discussion: https://postgr.es/m/87o8wvz253.fsf@news-spur.riddles.org.uk
parent 26a944cf
...@@ -44,7 +44,7 @@ SELECT [ ALL | DISTINCT [ ON ( <replaceable class="parameter">expression</replac ...@@ -44,7 +44,7 @@ SELECT [ ALL | DISTINCT [ ON ( <replaceable class="parameter">expression</replac
[ ORDER BY <replaceable class="parameter">expression</replaceable> [ ASC | DESC | USING <replaceable class="parameter">operator</replaceable> ] [ NULLS { FIRST | LAST } ] [, ...] ] [ ORDER BY <replaceable class="parameter">expression</replaceable> [ ASC | DESC | USING <replaceable class="parameter">operator</replaceable> ] [ NULLS { FIRST | LAST } ] [, ...] ]
[ LIMIT { <replaceable class="parameter">count</replaceable> | ALL } ] [ LIMIT { <replaceable class="parameter">count</replaceable> | ALL } ]
[ OFFSET <replaceable class="parameter">start</replaceable> [ ROW | ROWS ] ] [ OFFSET <replaceable class="parameter">start</replaceable> [ ROW | ROWS ] ]
[ FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] { ROW | ROWS } ONLY ] [ FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] { ROW | ROWS } { ONLY | WITH TIES } ]
[ FOR { UPDATE | NO KEY UPDATE | SHARE | KEY SHARE } [ OF <replaceable class="parameter">table_name</replaceable> [, ...] ] [ NOWAIT | SKIP LOCKED ] [...] ] [ FOR { UPDATE | NO KEY UPDATE | SHARE | KEY SHARE } [ OF <replaceable class="parameter">table_name</replaceable> [, ...] ] [ NOWAIT | SKIP LOCKED ] [...] ]
<phrase>where <replaceable class="parameter">from_item</replaceable> can be one of:</phrase> <phrase>where <replaceable class="parameter">from_item</replaceable> can be one of:</phrase>
...@@ -1438,7 +1438,7 @@ OFFSET <replaceable class="parameter">start</replaceable> ...@@ -1438,7 +1438,7 @@ OFFSET <replaceable class="parameter">start</replaceable>
which <productname>PostgreSQL</productname> also supports. It is: which <productname>PostgreSQL</productname> also supports. It is:
<synopsis> <synopsis>
OFFSET <replaceable class="parameter">start</replaceable> { ROW | ROWS } OFFSET <replaceable class="parameter">start</replaceable> { ROW | ROWS }
FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] { ROW | ROWS } ONLY FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] { ROW | ROWS } { ONLY | WITH TIES }
</synopsis> </synopsis>
In this syntax, the <replaceable class="parameter">start</replaceable> In this syntax, the <replaceable class="parameter">start</replaceable>
or <replaceable class="parameter">count</replaceable> value is required by or <replaceable class="parameter">count</replaceable> value is required by
...@@ -1448,10 +1448,13 @@ FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] { ...@@ -1448,10 +1448,13 @@ FETCH { FIRST | NEXT } [ <replaceable class="parameter">count</replaceable> ] {
ambiguity. ambiguity.
If <replaceable class="parameter">count</replaceable> is If <replaceable class="parameter">count</replaceable> is
omitted in a <literal>FETCH</literal> clause, it defaults to 1. omitted in a <literal>FETCH</literal> clause, it defaults to 1.
<literal>ROW</literal> The <literal>WITH TIES</literal> option is used to return any additional
and <literal>ROWS</literal> as well as <literal>FIRST</literal> rows that tie for the last place in the result set according to
and <literal>NEXT</literal> are noise words that don't influence <literal>ORDER BY</literal> clause; <literal>ORDER BY</literal>
the effects of these clauses. is mandatory in this case.
<literal>ROW</literal> and <literal>ROWS</literal> as well as
<literal>FIRST</literal> and <literal>NEXT</literal> are noise
words that don't influence the effects of these clauses.
According to the standard, the <literal>OFFSET</literal> clause must come According to the standard, the <literal>OFFSET</literal> clause must come
before the <literal>FETCH</literal> clause if both are present; but before the <literal>FETCH</literal> clause if both are present; but
<productname>PostgreSQL</productname> is laxer and allows either order. <productname>PostgreSQL</productname> is laxer and allows either order.
......
...@@ -345,7 +345,7 @@ F863 Nested <result offset clause> in <query expression> YES ...@@ -345,7 +345,7 @@ F863 Nested <result offset clause> in <query expression> YES
F864 Top-level <result offset clause> in views YES F864 Top-level <result offset clause> in views YES
F865 <offset row count> in <result offset clause> YES F865 <offset row count> in <result offset clause> YES
F866 FETCH FIRST clause: PERCENT option NO F866 FETCH FIRST clause: PERCENT option NO
F867 FETCH FIRST clause: WITH TIES option NO F867 FETCH FIRST clause: WITH TIES option YES
R010 Row pattern recognition: FROM clause NO R010 Row pattern recognition: FROM clause NO
R020 Row pattern recognition: WINDOW clause NO R020 Row pattern recognition: WINDOW clause NO
R030 Row pattern recognition: full aggregate support NO R030 Row pattern recognition: full aggregate support NO
......
...@@ -41,6 +41,7 @@ static TupleTableSlot * /* return: a tuple or NULL */ ...@@ -41,6 +41,7 @@ static TupleTableSlot * /* return: a tuple or NULL */
ExecLimit(PlanState *pstate) ExecLimit(PlanState *pstate)
{ {
LimitState *node = castNode(LimitState, pstate); LimitState *node = castNode(LimitState, pstate);
ExprContext *econtext = node->ps.ps_ExprContext;
ScanDirection direction; ScanDirection direction;
TupleTableSlot *slot; TupleTableSlot *slot;
PlanState *outerPlan; PlanState *outerPlan;
...@@ -102,6 +103,16 @@ ExecLimit(PlanState *pstate) ...@@ -102,6 +103,16 @@ ExecLimit(PlanState *pstate)
node->lstate = LIMIT_EMPTY; node->lstate = LIMIT_EMPTY;
return NULL; return NULL;
} }
/*
* Tuple at limit is needed for comparation in subsequent
* execution to detect ties.
*/
if (node->limitOption == LIMIT_OPTION_WITH_TIES &&
node->position - node->offset == node->count - 1)
{
ExecCopySlot(node->last_slot, slot);
}
node->subSlot = slot; node->subSlot = slot;
if (++node->position > node->offset) if (++node->position > node->offset)
break; break;
...@@ -125,10 +136,13 @@ ExecLimit(PlanState *pstate) ...@@ -125,10 +136,13 @@ ExecLimit(PlanState *pstate)
if (ScanDirectionIsForward(direction)) if (ScanDirectionIsForward(direction))
{ {
/* /*
* Forwards scan, so check for stepping off end of window. If * Forwards scan, so check for stepping off end of window. At
* we are at the end of the window, return NULL without * the end of the window, the behavior depends on whether WITH
* advancing the subplan or the position variable; but change * TIES was specified: in that case, we need to change the
* the state machine state to record having done so. * state machine to LIMIT_WINDOWTIES. If not (nothing was
* specified, or ONLY was) return NULL without advancing the
* subplan or the position variable but change the state
* machine to record having done so
* *
* Once at the end, ideally, we can shut down parallel * Once at the end, ideally, we can shut down parallel
* resources but that would destroy the parallel context which * resources but that would destroy the parallel context which
...@@ -138,11 +152,20 @@ ExecLimit(PlanState *pstate) ...@@ -138,11 +152,20 @@ ExecLimit(PlanState *pstate)
*/ */
if (!node->noCount && if (!node->noCount &&
node->position - node->offset >= node->count) node->position - node->offset >= node->count)
{
if (node->limitOption == LIMIT_OPTION_COUNT)
{ {
node->lstate = LIMIT_WINDOWEND; node->lstate = LIMIT_WINDOWEND;
return NULL; return NULL;
} }
else
{
node->lstate = LIMIT_WINDOWEND_TIES;
/* fall-through */
}
}
else
{
/* /*
* Get next tuple from subplan, if any. * Get next tuple from subplan, if any.
*/ */
...@@ -152,8 +175,20 @@ ExecLimit(PlanState *pstate) ...@@ -152,8 +175,20 @@ ExecLimit(PlanState *pstate)
node->lstate = LIMIT_SUBPLANEOF; node->lstate = LIMIT_SUBPLANEOF;
return NULL; return NULL;
} }
/*
* Tuple at limit is needed for comparation in subsequent
* execution to detect ties.
*/
if (node->limitOption == LIMIT_OPTION_WITH_TIES &&
node->position - node->offset == node->count - 1)
{
ExecCopySlot(node->last_slot, slot);
}
node->subSlot = slot; node->subSlot = slot;
node->position++; node->position++;
break;
}
} }
else else
{ {
...@@ -175,6 +210,62 @@ ExecLimit(PlanState *pstate) ...@@ -175,6 +210,62 @@ ExecLimit(PlanState *pstate)
elog(ERROR, "LIMIT subplan failed to run backwards"); elog(ERROR, "LIMIT subplan failed to run backwards");
node->subSlot = slot; node->subSlot = slot;
node->position--; node->position--;
break;
}
case LIMIT_WINDOWEND_TIES:
if (ScanDirectionIsForward(direction))
{
/*
* Advance the subplan until we find the first row with
* different ORDER BY pathkeys.
*/
slot = ExecProcNode(outerPlan);
if (TupIsNull(slot))
{
node->lstate = LIMIT_SUBPLANEOF;
return NULL;
}
/*
* Test if the new tuple and the last tuple match. If so we
* return the tuple.
*/
econtext->ecxt_innertuple = slot;
econtext->ecxt_outertuple = node->last_slot;
if (ExecQualAndReset(node->eqfunction, econtext))
{
node->subSlot = slot;
node->position++;
}
else
{
node->lstate = LIMIT_WINDOWEND;
return NULL;
}
}
else
{
/*
* Backwards scan, so check for stepping off start of window.
* Change only state-machine status if so.
*/
if (node->position <= node->offset + 1)
{
node->lstate = LIMIT_WINDOWSTART;
return NULL;
}
/*
* Get previous tuple from subplan; there should be one! And
* change state-machine status.
*/
slot = ExecProcNode(outerPlan);
if (TupIsNull(slot))
elog(ERROR, "LIMIT subplan failed to run backwards");
node->subSlot = slot;
node->position--;
node->lstate = LIMIT_INWINDOW;
} }
break; break;
...@@ -198,6 +289,21 @@ ExecLimit(PlanState *pstate) ...@@ -198,6 +289,21 @@ ExecLimit(PlanState *pstate)
if (ScanDirectionIsForward(direction)) if (ScanDirectionIsForward(direction))
return NULL; return NULL;
/*
* We already past one position to detect ties so re-fetch
* previous tuple; there should be one! Note previous tuple must
* be in window.
*/
if (node->limitOption == LIMIT_OPTION_WITH_TIES)
{
slot = ExecProcNode(outerPlan);
if (TupIsNull(slot))
elog(ERROR, "LIMIT subplan failed to run backwards");
node->subSlot = slot;
node->lstate = LIMIT_INWINDOW;
}
else
{
/* /*
* Backing up from window end: simply re-return the last tuple * Backing up from window end: simply re-return the last tuple
* fetched from the subplan. * fetched from the subplan.
...@@ -205,6 +311,7 @@ ExecLimit(PlanState *pstate) ...@@ -205,6 +311,7 @@ ExecLimit(PlanState *pstate)
slot = node->subSlot; slot = node->subSlot;
node->lstate = LIMIT_INWINDOW; node->lstate = LIMIT_INWINDOW;
/* position does not change 'cause we didn't advance it before */ /* position does not change 'cause we didn't advance it before */
}
break; break;
case LIMIT_WINDOWSTART: case LIMIT_WINDOWSTART:
...@@ -319,7 +426,7 @@ recompute_limits(LimitState *node) ...@@ -319,7 +426,7 @@ recompute_limits(LimitState *node)
static int64 static int64
compute_tuples_needed(LimitState *node) compute_tuples_needed(LimitState *node)
{ {
if (node->noCount) if ((node->noCount) || (node->limitOption == LIMIT_OPTION_WITH_TIES))
return -1; return -1;
/* Note: if this overflows, we'll return a negative value, which is OK */ /* Note: if this overflows, we'll return a negative value, which is OK */
return node->count + node->offset; return node->count + node->offset;
...@@ -372,6 +479,7 @@ ExecInitLimit(Limit *node, EState *estate, int eflags) ...@@ -372,6 +479,7 @@ ExecInitLimit(Limit *node, EState *estate, int eflags)
(PlanState *) limitstate); (PlanState *) limitstate);
limitstate->limitCount = ExecInitExpr((Expr *) node->limitCount, limitstate->limitCount = ExecInitExpr((Expr *) node->limitCount,
(PlanState *) limitstate); (PlanState *) limitstate);
limitstate->limitOption = node->limitOption;
/* /*
* Initialize result type. * Initialize result type.
...@@ -388,6 +496,26 @@ ExecInitLimit(Limit *node, EState *estate, int eflags) ...@@ -388,6 +496,26 @@ ExecInitLimit(Limit *node, EState *estate, int eflags)
*/ */
limitstate->ps.ps_ProjInfo = NULL; limitstate->ps.ps_ProjInfo = NULL;
/*
* Initialize the equality evaluation, to detect ties.
*/
if (node->limitOption == LIMIT_OPTION_WITH_TIES)
{
TupleDesc desc;
const TupleTableSlotOps *ops;
desc = ExecGetResultType(outerPlanState(limitstate));
ops = ExecGetResultSlotOps(outerPlanState(limitstate), NULL);
limitstate->last_slot = ExecInitExtraTupleSlot(estate, desc, ops);
limitstate->eqfunction = execTuplesMatchPrepare(desc,
node->uniqNumCols,
node->uniqColIdx,
node->uniqOperators,
node->uniqCollations,
&limitstate->ps);
}
return limitstate; return limitstate;
} }
......
...@@ -1185,6 +1185,11 @@ _copyLimit(const Limit *from) ...@@ -1185,6 +1185,11 @@ _copyLimit(const Limit *from)
*/ */
COPY_NODE_FIELD(limitOffset); COPY_NODE_FIELD(limitOffset);
COPY_NODE_FIELD(limitCount); COPY_NODE_FIELD(limitCount);
COPY_SCALAR_FIELD(limitOption);
COPY_SCALAR_FIELD(uniqNumCols);
COPY_POINTER_FIELD(uniqColIdx, from->uniqNumCols * sizeof(AttrNumber));
COPY_POINTER_FIELD(uniqOperators, from->uniqNumCols * sizeof(Oid));
COPY_POINTER_FIELD(uniqCollations, from->uniqNumCols * sizeof(Oid));
return newnode; return newnode;
} }
...@@ -3079,6 +3084,7 @@ _copyQuery(const Query *from) ...@@ -3079,6 +3084,7 @@ _copyQuery(const Query *from)
COPY_NODE_FIELD(sortClause); COPY_NODE_FIELD(sortClause);
COPY_NODE_FIELD(limitOffset); COPY_NODE_FIELD(limitOffset);
COPY_NODE_FIELD(limitCount); COPY_NODE_FIELD(limitCount);
COPY_SCALAR_FIELD(limitOption);
COPY_NODE_FIELD(rowMarks); COPY_NODE_FIELD(rowMarks);
COPY_NODE_FIELD(setOperations); COPY_NODE_FIELD(setOperations);
COPY_NODE_FIELD(constraintDeps); COPY_NODE_FIELD(constraintDeps);
...@@ -3163,6 +3169,7 @@ _copySelectStmt(const SelectStmt *from) ...@@ -3163,6 +3169,7 @@ _copySelectStmt(const SelectStmt *from)
COPY_NODE_FIELD(sortClause); COPY_NODE_FIELD(sortClause);
COPY_NODE_FIELD(limitOffset); COPY_NODE_FIELD(limitOffset);
COPY_NODE_FIELD(limitCount); COPY_NODE_FIELD(limitCount);
COPY_SCALAR_FIELD(limitOption);
COPY_NODE_FIELD(lockingClause); COPY_NODE_FIELD(lockingClause);
COPY_NODE_FIELD(withClause); COPY_NODE_FIELD(withClause);
COPY_SCALAR_FIELD(op); COPY_SCALAR_FIELD(op);
......
...@@ -981,6 +981,7 @@ _equalQuery(const Query *a, const Query *b) ...@@ -981,6 +981,7 @@ _equalQuery(const Query *a, const Query *b)
COMPARE_NODE_FIELD(sortClause); COMPARE_NODE_FIELD(sortClause);
COMPARE_NODE_FIELD(limitOffset); COMPARE_NODE_FIELD(limitOffset);
COMPARE_NODE_FIELD(limitCount); COMPARE_NODE_FIELD(limitCount);
COMPARE_SCALAR_FIELD(limitOption);
COMPARE_NODE_FIELD(rowMarks); COMPARE_NODE_FIELD(rowMarks);
COMPARE_NODE_FIELD(setOperations); COMPARE_NODE_FIELD(setOperations);
COMPARE_NODE_FIELD(constraintDeps); COMPARE_NODE_FIELD(constraintDeps);
...@@ -1055,6 +1056,7 @@ _equalSelectStmt(const SelectStmt *a, const SelectStmt *b) ...@@ -1055,6 +1056,7 @@ _equalSelectStmt(const SelectStmt *a, const SelectStmt *b)
COMPARE_NODE_FIELD(sortClause); COMPARE_NODE_FIELD(sortClause);
COMPARE_NODE_FIELD(limitOffset); COMPARE_NODE_FIELD(limitOffset);
COMPARE_NODE_FIELD(limitCount); COMPARE_NODE_FIELD(limitCount);
COMPARE_SCALAR_FIELD(limitOption);
COMPARE_NODE_FIELD(lockingClause); COMPARE_NODE_FIELD(lockingClause);
COMPARE_NODE_FIELD(withClause); COMPARE_NODE_FIELD(withClause);
COMPARE_SCALAR_FIELD(op); COMPARE_SCALAR_FIELD(op);
......
...@@ -931,6 +931,11 @@ _outLimit(StringInfo str, const Limit *node) ...@@ -931,6 +931,11 @@ _outLimit(StringInfo str, const Limit *node)
WRITE_NODE_FIELD(limitOffset); WRITE_NODE_FIELD(limitOffset);
WRITE_NODE_FIELD(limitCount); WRITE_NODE_FIELD(limitCount);
WRITE_ENUM_FIELD(limitOption, LimitOption);
WRITE_INT_FIELD(uniqNumCols);
WRITE_ATTRNUMBER_ARRAY(uniqColIdx, node->uniqNumCols);
WRITE_OID_ARRAY(uniqOperators, node->uniqNumCols);
WRITE_OID_ARRAY(uniqCollations, node->uniqNumCols);
} }
static void static void
...@@ -2741,6 +2746,7 @@ _outSelectStmt(StringInfo str, const SelectStmt *node) ...@@ -2741,6 +2746,7 @@ _outSelectStmt(StringInfo str, const SelectStmt *node)
WRITE_NODE_FIELD(sortClause); WRITE_NODE_FIELD(sortClause);
WRITE_NODE_FIELD(limitOffset); WRITE_NODE_FIELD(limitOffset);
WRITE_NODE_FIELD(limitCount); WRITE_NODE_FIELD(limitCount);
WRITE_ENUM_FIELD(limitOption, LimitOption);
WRITE_NODE_FIELD(lockingClause); WRITE_NODE_FIELD(lockingClause);
WRITE_NODE_FIELD(withClause); WRITE_NODE_FIELD(withClause);
WRITE_ENUM_FIELD(op, SetOperation); WRITE_ENUM_FIELD(op, SetOperation);
...@@ -2952,6 +2958,7 @@ _outQuery(StringInfo str, const Query *node) ...@@ -2952,6 +2958,7 @@ _outQuery(StringInfo str, const Query *node)
WRITE_NODE_FIELD(sortClause); WRITE_NODE_FIELD(sortClause);
WRITE_NODE_FIELD(limitOffset); WRITE_NODE_FIELD(limitOffset);
WRITE_NODE_FIELD(limitCount); WRITE_NODE_FIELD(limitCount);
WRITE_ENUM_FIELD(limitOption, LimitOption);
WRITE_NODE_FIELD(rowMarks); WRITE_NODE_FIELD(rowMarks);
WRITE_NODE_FIELD(setOperations); WRITE_NODE_FIELD(setOperations);
WRITE_NODE_FIELD(constraintDeps); WRITE_NODE_FIELD(constraintDeps);
......
...@@ -278,6 +278,7 @@ _readQuery(void) ...@@ -278,6 +278,7 @@ _readQuery(void)
READ_NODE_FIELD(sortClause); READ_NODE_FIELD(sortClause);
READ_NODE_FIELD(limitOffset); READ_NODE_FIELD(limitOffset);
READ_NODE_FIELD(limitCount); READ_NODE_FIELD(limitCount);
READ_ENUM_FIELD(limitOption, LimitOption);
READ_NODE_FIELD(rowMarks); READ_NODE_FIELD(rowMarks);
READ_NODE_FIELD(setOperations); READ_NODE_FIELD(setOperations);
READ_NODE_FIELD(constraintDeps); READ_NODE_FIELD(constraintDeps);
...@@ -2398,6 +2399,11 @@ _readLimit(void) ...@@ -2398,6 +2399,11 @@ _readLimit(void)
READ_NODE_FIELD(limitOffset); READ_NODE_FIELD(limitOffset);
READ_NODE_FIELD(limitCount); READ_NODE_FIELD(limitCount);
READ_ENUM_FIELD(limitOption, LimitOption);
READ_INT_FIELD(uniqNumCols);
READ_ATTRNUMBER_ARRAY(uniqColIdx, local_node->uniqNumCols);
READ_OID_ARRAY(uniqOperators, local_node->uniqNumCols);
READ_OID_ARRAY(uniqCollations, local_node->uniqNumCols);
READ_DONE(); READ_DONE();
} }
......
...@@ -2378,7 +2378,9 @@ create_minmaxagg_plan(PlannerInfo *root, MinMaxAggPath *best_path) ...@@ -2378,7 +2378,9 @@ create_minmaxagg_plan(PlannerInfo *root, MinMaxAggPath *best_path)
plan = (Plan *) make_limit(plan, plan = (Plan *) make_limit(plan,
subparse->limitOffset, subparse->limitOffset,
subparse->limitCount); subparse->limitCount,
subparse->limitOption,
0, NULL, NULL, NULL);
/* Must apply correct cost/width data to Limit node */ /* Must apply correct cost/width data to Limit node */
plan->startup_cost = mminfo->path->startup_cost; plan->startup_cost = mminfo->path->startup_cost;
...@@ -2688,13 +2690,43 @@ create_limit_plan(PlannerInfo *root, LimitPath *best_path, int flags) ...@@ -2688,13 +2690,43 @@ create_limit_plan(PlannerInfo *root, LimitPath *best_path, int flags)
{ {
Limit *plan; Limit *plan;
Plan *subplan; Plan *subplan;
int numUniqkeys = 0;
AttrNumber *uniqColIdx = NULL;
Oid *uniqOperators = NULL;
Oid *uniqCollations = NULL;
/* Limit doesn't project, so tlist requirements pass through */ /* Limit doesn't project, so tlist requirements pass through */
subplan = create_plan_recurse(root, best_path->subpath, flags); subplan = create_plan_recurse(root, best_path->subpath, flags);
/* Extract information necessary for comparing rows for WITH TIES. */
if (best_path->limitOption == LIMIT_OPTION_WITH_TIES)
{
Query *parse = root->parse;
ListCell *l;
numUniqkeys = list_length(parse->sortClause);
uniqColIdx = (AttrNumber *) palloc(numUniqkeys * sizeof(AttrNumber));
uniqOperators = (Oid *) palloc(numUniqkeys * sizeof(Oid));
uniqCollations = (Oid *) palloc(numUniqkeys * sizeof(Oid));
numUniqkeys = 0;
foreach(l, parse->sortClause)
{
SortGroupClause *sortcl = (SortGroupClause *) lfirst(l);
TargetEntry *tle = get_sortgroupclause_tle(sortcl, parse->targetList);
uniqColIdx[numUniqkeys] = tle->resno;
uniqOperators[numUniqkeys] = sortcl->eqop;
uniqCollations[numUniqkeys] = exprCollation((Node *) tle->expr);
numUniqkeys++;
}
}
plan = make_limit(subplan, plan = make_limit(subplan,
best_path->limitOffset, best_path->limitOffset,
best_path->limitCount); best_path->limitCount,
best_path->limitOption,
numUniqkeys, uniqColIdx, uniqOperators, uniqCollations);
copy_generic_path_info(&plan->plan, (Path *) best_path); copy_generic_path_info(&plan->plan, (Path *) best_path);
...@@ -6672,7 +6704,9 @@ make_lockrows(Plan *lefttree, List *rowMarks, int epqParam) ...@@ -6672,7 +6704,9 @@ make_lockrows(Plan *lefttree, List *rowMarks, int epqParam)
* Build a Limit plan node * Build a Limit plan node
*/ */
Limit * Limit *
make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount) make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount,
LimitOption limitOption, int uniqNumCols, AttrNumber *uniqColIdx,
Oid *uniqOperators, Oid *uniqCollations)
{ {
Limit *node = makeNode(Limit); Limit *node = makeNode(Limit);
Plan *plan = &node->plan; Plan *plan = &node->plan;
...@@ -6684,6 +6718,11 @@ make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount) ...@@ -6684,6 +6718,11 @@ make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount)
node->limitOffset = limitOffset; node->limitOffset = limitOffset;
node->limitCount = limitCount; node->limitCount = limitCount;
node->limitOption = limitOption;
node->uniqNumCols = uniqNumCols;
node->uniqColIdx = uniqColIdx;
node->uniqOperators = uniqOperators;
node->uniqCollations = uniqCollations;
return node; return node;
} }
......
...@@ -2319,6 +2319,7 @@ grouping_planner(PlannerInfo *root, bool inheritance_update, ...@@ -2319,6 +2319,7 @@ grouping_planner(PlannerInfo *root, bool inheritance_update,
path = (Path *) create_limit_path(root, final_rel, path, path = (Path *) create_limit_path(root, final_rel, path,
parse->limitOffset, parse->limitOffset,
parse->limitCount, parse->limitCount,
parse->limitOption,
offset_est, count_est); offset_est, count_est);
} }
......
...@@ -3602,6 +3602,7 @@ LimitPath * ...@@ -3602,6 +3602,7 @@ LimitPath *
create_limit_path(PlannerInfo *root, RelOptInfo *rel, create_limit_path(PlannerInfo *root, RelOptInfo *rel,
Path *subpath, Path *subpath,
Node *limitOffset, Node *limitCount, Node *limitOffset, Node *limitCount,
LimitOption limitOption,
int64 offset_est, int64 count_est) int64 offset_est, int64 count_est)
{ {
LimitPath *pathnode = makeNode(LimitPath); LimitPath *pathnode = makeNode(LimitPath);
...@@ -3623,6 +3624,7 @@ create_limit_path(PlannerInfo *root, RelOptInfo *rel, ...@@ -3623,6 +3624,7 @@ create_limit_path(PlannerInfo *root, RelOptInfo *rel,
pathnode->subpath = subpath; pathnode->subpath = subpath;
pathnode->limitOffset = limitOffset; pathnode->limitOffset = limitOffset;
pathnode->limitCount = limitCount; pathnode->limitCount = limitCount;
pathnode->limitOption = limitOption;
/* /*
* Adjust the output rows count and costs according to the offset/limit. * Adjust the output rows count and costs according to the offset/limit.
......
...@@ -1281,9 +1281,12 @@ transformSelectStmt(ParseState *pstate, SelectStmt *stmt) ...@@ -1281,9 +1281,12 @@ transformSelectStmt(ParseState *pstate, SelectStmt *stmt)
/* transform LIMIT */ /* transform LIMIT */
qry->limitOffset = transformLimitClause(pstate, stmt->limitOffset, qry->limitOffset = transformLimitClause(pstate, stmt->limitOffset,
EXPR_KIND_OFFSET, "OFFSET"); EXPR_KIND_OFFSET, "OFFSET",
stmt->limitOption);
qry->limitCount = transformLimitClause(pstate, stmt->limitCount, qry->limitCount = transformLimitClause(pstate, stmt->limitCount,
EXPR_KIND_LIMIT, "LIMIT"); EXPR_KIND_LIMIT, "LIMIT",
stmt->limitOption);
qry->limitOption = stmt->limitOption;
/* transform window clauses after we have seen all window functions */ /* transform window clauses after we have seen all window functions */
qry->windowClause = transformWindowDefinitions(pstate, qry->windowClause = transformWindowDefinitions(pstate,
...@@ -1524,9 +1527,12 @@ transformValuesClause(ParseState *pstate, SelectStmt *stmt) ...@@ -1524,9 +1527,12 @@ transformValuesClause(ParseState *pstate, SelectStmt *stmt)
false /* allow SQL92 rules */ ); false /* allow SQL92 rules */ );
qry->limitOffset = transformLimitClause(pstate, stmt->limitOffset, qry->limitOffset = transformLimitClause(pstate, stmt->limitOffset,
EXPR_KIND_OFFSET, "OFFSET"); EXPR_KIND_OFFSET, "OFFSET",
stmt->limitOption);
qry->limitCount = transformLimitClause(pstate, stmt->limitCount, qry->limitCount = transformLimitClause(pstate, stmt->limitCount,
EXPR_KIND_LIMIT, "LIMIT"); EXPR_KIND_LIMIT, "LIMIT",
stmt->limitOption);
qry->limitOption = stmt->limitOption;
if (stmt->lockingClause) if (stmt->lockingClause)
ereport(ERROR, ereport(ERROR,
...@@ -1775,9 +1781,12 @@ transformSetOperationStmt(ParseState *pstate, SelectStmt *stmt) ...@@ -1775,9 +1781,12 @@ transformSetOperationStmt(ParseState *pstate, SelectStmt *stmt)
exprLocation(list_nth(qry->targetList, tllen))))); exprLocation(list_nth(qry->targetList, tllen)))));
qry->limitOffset = transformLimitClause(pstate, limitOffset, qry->limitOffset = transformLimitClause(pstate, limitOffset,
EXPR_KIND_OFFSET, "OFFSET"); EXPR_KIND_OFFSET, "OFFSET",
stmt->limitOption);
qry->limitCount = transformLimitClause(pstate, limitCount, qry->limitCount = transformLimitClause(pstate, limitCount,
EXPR_KIND_LIMIT, "LIMIT"); EXPR_KIND_LIMIT, "LIMIT",
stmt->limitOption);
qry->limitOption = stmt->limitOption;
qry->rtable = pstate->p_rtable; qry->rtable = pstate->p_rtable;
qry->jointree = makeFromExpr(pstate->p_joinlist, NULL); qry->jointree = makeFromExpr(pstate->p_joinlist, NULL);
......
...@@ -127,6 +127,14 @@ typedef struct ImportQual ...@@ -127,6 +127,14 @@ typedef struct ImportQual
List *table_names; List *table_names;
} ImportQual; } ImportQual;
/* Private struct for the result of opt_select_limit production */
typedef struct SelectLimit
{
Node *limitOffset;
Node *limitCount;
LimitOption limitOption;
} SelectLimit;
/* ConstraintAttributeSpec yields an integer bitmask of these flags: */ /* ConstraintAttributeSpec yields an integer bitmask of these flags: */
#define CAS_NOT_DEFERRABLE 0x01 #define CAS_NOT_DEFERRABLE 0x01
#define CAS_DEFERRABLE 0x02 #define CAS_DEFERRABLE 0x02
...@@ -164,7 +172,7 @@ static List *makeOrderedSetArgs(List *directargs, List *orderedargs, ...@@ -164,7 +172,7 @@ static List *makeOrderedSetArgs(List *directargs, List *orderedargs,
core_yyscan_t yyscanner); core_yyscan_t yyscanner);
static void insertSelectOptions(SelectStmt *stmt, static void insertSelectOptions(SelectStmt *stmt,
List *sortClause, List *lockingClause, List *sortClause, List *lockingClause,
Node *limitOffset, Node *limitCount, SelectLimit *limitClause,
WithClause *withClause, WithClause *withClause,
core_yyscan_t yyscanner); core_yyscan_t yyscanner);
static Node *makeSetOp(SetOperation op, bool all, Node *larg, Node *rarg); static Node *makeSetOp(SetOperation op, bool all, Node *larg, Node *rarg);
...@@ -242,6 +250,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query); ...@@ -242,6 +250,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query);
PartitionSpec *partspec; PartitionSpec *partspec;
PartitionBoundSpec *partboundspec; PartitionBoundSpec *partboundspec;
RoleSpec *rolespec; RoleSpec *rolespec;
struct SelectLimit *selectlimit;
} }
%type <node> stmt schema_stmt %type <node> stmt schema_stmt
...@@ -374,6 +383,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query); ...@@ -374,6 +383,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query);
%type <ival> import_qualification_type %type <ival> import_qualification_type
%type <importqual> import_qualification %type <importqual> import_qualification
%type <node> vacuum_relation %type <node> vacuum_relation
%type <selectlimit> opt_select_limit select_limit limit_clause
%type <list> stmtblock stmtmulti %type <list> stmtblock stmtmulti
OptTableElementList TableElementList OptInherit definition OptTableElementList TableElementList OptInherit definition
...@@ -394,8 +404,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query); ...@@ -394,8 +404,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query);
target_list opt_target_list insert_column_list set_target_list target_list opt_target_list insert_column_list set_target_list
set_clause_list set_clause set_clause_list set_clause
def_list operator_def_list indirection opt_indirection def_list operator_def_list indirection opt_indirection
reloption_list group_clause TriggerFuncArgs select_limit reloption_list group_clause TriggerFuncArgs opclass_item_list opclass_drop_list
opt_select_limit opclass_item_list opclass_drop_list
opclass_purpose opt_opfamily transaction_mode_list_or_empty opclass_purpose opt_opfamily transaction_mode_list_or_empty
OptTableFuncElementList TableFuncElementList opt_type_modifiers OptTableFuncElementList TableFuncElementList opt_type_modifiers
prep_type_clause prep_type_clause
...@@ -458,7 +467,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query); ...@@ -458,7 +467,7 @@ static Node *makeRecursiveViewSelect(char *relname, List *aliases, Node *query);
comment_type_any_name comment_type_name comment_type_any_name comment_type_name
security_label_type_any_name security_label_type_name security_label_type_any_name security_label_type_name
%type <node> fetch_args limit_clause select_limit_value %type <node> fetch_args select_limit_value
offset_clause select_offset_value offset_clause select_offset_value
select_fetch_first_value I_or_F_const select_fetch_first_value I_or_F_const
%type <ival> row_or_rows first_or_next %type <ival> row_or_rows first_or_next
...@@ -11378,14 +11387,14 @@ select_no_parens: ...@@ -11378,14 +11387,14 @@ select_no_parens:
| select_clause sort_clause | select_clause sort_clause
{ {
insertSelectOptions((SelectStmt *) $1, $2, NIL, insertSelectOptions((SelectStmt *) $1, $2, NIL,
NULL, NULL, NULL, NULL, NULL,
yyscanner); yyscanner);
$$ = $1; $$ = $1;
} }
| select_clause opt_sort_clause for_locking_clause opt_select_limit | select_clause opt_sort_clause for_locking_clause opt_select_limit
{ {
insertSelectOptions((SelectStmt *) $1, $2, $3, insertSelectOptions((SelectStmt *) $1, $2, $3,
list_nth($4, 0), list_nth($4, 1), $4,
NULL, NULL,
yyscanner); yyscanner);
$$ = $1; $$ = $1;
...@@ -11393,7 +11402,7 @@ select_no_parens: ...@@ -11393,7 +11402,7 @@ select_no_parens:
| select_clause opt_sort_clause select_limit opt_for_locking_clause | select_clause opt_sort_clause select_limit opt_for_locking_clause
{ {
insertSelectOptions((SelectStmt *) $1, $2, $4, insertSelectOptions((SelectStmt *) $1, $2, $4,
list_nth($3, 0), list_nth($3, 1), $3,
NULL, NULL,
yyscanner); yyscanner);
$$ = $1; $$ = $1;
...@@ -11401,7 +11410,7 @@ select_no_parens: ...@@ -11401,7 +11410,7 @@ select_no_parens:
| with_clause select_clause | with_clause select_clause
{ {
insertSelectOptions((SelectStmt *) $2, NULL, NIL, insertSelectOptions((SelectStmt *) $2, NULL, NIL,
NULL, NULL, NULL,
$1, $1,
yyscanner); yyscanner);
$$ = $2; $$ = $2;
...@@ -11409,7 +11418,7 @@ select_no_parens: ...@@ -11409,7 +11418,7 @@ select_no_parens:
| with_clause select_clause sort_clause | with_clause select_clause sort_clause
{ {
insertSelectOptions((SelectStmt *) $2, $3, NIL, insertSelectOptions((SelectStmt *) $2, $3, NIL,
NULL, NULL, NULL,
$1, $1,
yyscanner); yyscanner);
$$ = $2; $$ = $2;
...@@ -11417,7 +11426,7 @@ select_no_parens: ...@@ -11417,7 +11426,7 @@ select_no_parens:
| with_clause select_clause opt_sort_clause for_locking_clause opt_select_limit | with_clause select_clause opt_sort_clause for_locking_clause opt_select_limit
{ {
insertSelectOptions((SelectStmt *) $2, $3, $4, insertSelectOptions((SelectStmt *) $2, $3, $4,
list_nth($5, 0), list_nth($5, 1), $5,
$1, $1,
yyscanner); yyscanner);
$$ = $2; $$ = $2;
...@@ -11425,7 +11434,7 @@ select_no_parens: ...@@ -11425,7 +11434,7 @@ select_no_parens:
| with_clause select_clause opt_sort_clause select_limit opt_for_locking_clause | with_clause select_clause opt_sort_clause select_limit opt_for_locking_clause
{ {
insertSelectOptions((SelectStmt *) $2, $3, $5, insertSelectOptions((SelectStmt *) $2, $3, $5,
list_nth($4, 0), list_nth($4, 1), $4,
$1, $1,
yyscanner); yyscanner);
$$ = $2; $$ = $2;
...@@ -11719,20 +11728,44 @@ sortby: a_expr USING qual_all_Op opt_nulls_order ...@@ -11719,20 +11728,44 @@ sortby: a_expr USING qual_all_Op opt_nulls_order
select_limit: select_limit:
limit_clause offset_clause { $$ = list_make2($2, $1); } limit_clause offset_clause
| offset_clause limit_clause { $$ = list_make2($1, $2); } {
| limit_clause { $$ = list_make2(NULL, $1); } $$ = $1;
| offset_clause { $$ = list_make2($1, NULL); } ($$)->limitOffset = $2;
}
| offset_clause limit_clause
{
$$ = $2;
($$)->limitOffset = $1;
}
| limit_clause
{
$$ = $1;
}
| offset_clause
{
SelectLimit *n = (SelectLimit *) palloc(sizeof(SelectLimit));
n->limitOffset = $1;
n->limitCount = NULL;
n->limitOption = LIMIT_OPTION_COUNT;
$$ = n;
}
; ;
opt_select_limit: opt_select_limit:
select_limit { $$ = $1; } select_limit { $$ = $1; }
| /* EMPTY */ { $$ = list_make2(NULL,NULL); } | /* EMPTY */ { $$ = NULL; }
; ;
limit_clause: limit_clause:
LIMIT select_limit_value LIMIT select_limit_value
{ $$ = $2; } {
SelectLimit *n = (SelectLimit *) palloc(sizeof(SelectLimit));
n->limitOffset = NULL;
n->limitCount = $2;
n->limitOption = LIMIT_OPTION_COUNT;
$$ = n;
}
| LIMIT select_limit_value ',' select_offset_value | LIMIT select_limit_value ',' select_offset_value
{ {
/* Disabled because it was too confusing, bjm 2002-02-18 */ /* Disabled because it was too confusing, bjm 2002-02-18 */
...@@ -11750,9 +11783,29 @@ limit_clause: ...@@ -11750,9 +11783,29 @@ limit_clause:
* we can see the ONLY token in the lookahead slot. * we can see the ONLY token in the lookahead slot.
*/ */
| FETCH first_or_next select_fetch_first_value row_or_rows ONLY | FETCH first_or_next select_fetch_first_value row_or_rows ONLY
{ $$ = $3; } {
SelectLimit *n = (SelectLimit *) palloc(sizeof(SelectLimit));
n->limitOffset = NULL;
n->limitCount = $3;
n->limitOption = LIMIT_OPTION_COUNT;
$$ = n;
}
| FETCH first_or_next select_fetch_first_value row_or_rows WITH TIES
{
SelectLimit *n = (SelectLimit *) palloc(sizeof(SelectLimit));
n->limitOffset = NULL;
n->limitCount = $3;
n->limitOption = LIMIT_OPTION_WITH_TIES;
$$ = n;
}
| FETCH first_or_next row_or_rows ONLY | FETCH first_or_next row_or_rows ONLY
{ $$ = makeIntConst(1, -1); } {
SelectLimit *n = (SelectLimit *) palloc(sizeof(SelectLimit));
n->limitOffset = NULL;
n->limitCount = makeIntConst(1, -1);
n->limitOption = LIMIT_OPTION_COUNT;
$$ = n;
}
; ;
offset_clause: offset_clause:
...@@ -16047,7 +16100,7 @@ makeOrderedSetArgs(List *directargs, List *orderedargs, ...@@ -16047,7 +16100,7 @@ makeOrderedSetArgs(List *directargs, List *orderedargs,
static void static void
insertSelectOptions(SelectStmt *stmt, insertSelectOptions(SelectStmt *stmt,
List *sortClause, List *lockingClause, List *sortClause, List *lockingClause,
Node *limitOffset, Node *limitCount, SelectLimit *limitClause,
WithClause *withClause, WithClause *withClause,
core_yyscan_t yyscanner) core_yyscan_t yyscanner)
{ {
...@@ -16068,23 +16121,35 @@ insertSelectOptions(SelectStmt *stmt, ...@@ -16068,23 +16121,35 @@ insertSelectOptions(SelectStmt *stmt,
} }
/* We can handle multiple locking clauses, though */ /* We can handle multiple locking clauses, though */
stmt->lockingClause = list_concat(stmt->lockingClause, lockingClause); stmt->lockingClause = list_concat(stmt->lockingClause, lockingClause);
if (limitOffset) if (limitClause && limitClause->limitOffset)
{ {
if (stmt->limitOffset) if (stmt->limitOffset)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR), (errcode(ERRCODE_SYNTAX_ERROR),
errmsg("multiple OFFSET clauses not allowed"), errmsg("multiple OFFSET clauses not allowed"),
parser_errposition(exprLocation(limitOffset)))); parser_errposition(exprLocation(limitClause->limitOffset))));
stmt->limitOffset = limitOffset; stmt->limitOffset = limitClause->limitOffset;
} }
if (limitCount) if (limitClause && limitClause->limitCount)
{ {
if (stmt->limitCount) if (stmt->limitCount)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR), (errcode(ERRCODE_SYNTAX_ERROR),
errmsg("multiple LIMIT clauses not allowed"), errmsg("multiple LIMIT clauses not allowed"),
parser_errposition(exprLocation(limitCount)))); parser_errposition(exprLocation(limitClause->limitCount))));
stmt->limitCount = limitCount; stmt->limitCount = limitClause->limitCount;
}
if (limitClause && limitClause->limitOption != LIMIT_OPTION_DEFAULT)
{
if (stmt->limitOption)
ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR),
errmsg("multiple limit options not allowed")));
if (!stmt->sortClause && limitClause->limitOption == LIMIT_OPTION_WITH_TIES)
ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR),
errmsg("WITH TIES options can not be specified without ORDER BY clause")));
stmt->limitOption = limitClause->limitOption;
} }
if (withClause) if (withClause)
{ {
......
...@@ -1745,7 +1745,8 @@ transformWhereClause(ParseState *pstate, Node *clause, ...@@ -1745,7 +1745,8 @@ transformWhereClause(ParseState *pstate, Node *clause,
*/ */
Node * Node *
transformLimitClause(ParseState *pstate, Node *clause, transformLimitClause(ParseState *pstate, Node *clause,
ParseExprKind exprKind, const char *constructName) ParseExprKind exprKind, const char *constructName,
LimitOption limitOption)
{ {
Node *qual; Node *qual;
...@@ -1759,6 +1760,18 @@ transformLimitClause(ParseState *pstate, Node *clause, ...@@ -1759,6 +1760,18 @@ transformLimitClause(ParseState *pstate, Node *clause,
/* LIMIT can't refer to any variables of the current query */ /* LIMIT can't refer to any variables of the current query */
checkExprIsVarFree(pstate, qual, constructName); checkExprIsVarFree(pstate, qual, constructName);
/*
* Don't allow NULLs in FETCH FIRST .. WITH TIES. This test is ugly and
* extremely simplistic, in that you can pass a NULL anyway by hiding it
* inside an expression -- but this protects ruleutils against emitting an
* unadorned NULL that's not accepted back by the grammar.
*/
if (exprKind == EXPR_KIND_LIMIT && limitOption == LIMIT_OPTION_WITH_TIES &&
IsA(clause, A_Const) && ((A_Const *) clause)->val.type == T_Null)
ereport(ERROR,
(errcode(ERRCODE_INVALID_ROW_COUNT_IN_LIMIT_CLAUSE),
errmsg("row count cannot be NULL in FETCH FIRST ... WITH TIES clause")));
return qual; return qual;
} }
......
...@@ -5243,7 +5243,10 @@ get_select_query_def(Query *query, deparse_context *context, ...@@ -5243,7 +5243,10 @@ get_select_query_def(Query *query, deparse_context *context,
force_colno, context); force_colno, context);
} }
/* Add the LIMIT clause if given */ /*
* Add the LIMIT/OFFSET clauses if given. If non-default options, use the
* standard spelling of LIMIT.
*/
if (query->limitOffset != NULL) if (query->limitOffset != NULL)
{ {
appendContextKeyword(context, " OFFSET ", appendContextKeyword(context, " OFFSET ",
...@@ -5251,6 +5254,15 @@ get_select_query_def(Query *query, deparse_context *context, ...@@ -5251,6 +5254,15 @@ get_select_query_def(Query *query, deparse_context *context,
get_rule_expr(query->limitOffset, context, false); get_rule_expr(query->limitOffset, context, false);
} }
if (query->limitCount != NULL) if (query->limitCount != NULL)
{
if (query->limitOption == LIMIT_OPTION_WITH_TIES)
{
appendContextKeyword(context, " FETCH FIRST ",
-PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
get_rule_expr(query->limitCount, context, false);
appendStringInfo(buf, " ROWS WITH TIES");
}
else
{ {
appendContextKeyword(context, " LIMIT ", appendContextKeyword(context, " LIMIT ",
-PRETTYINDENT_STD, PRETTYINDENT_STD, 0); -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
...@@ -5260,6 +5272,7 @@ get_select_query_def(Query *query, deparse_context *context, ...@@ -5260,6 +5272,7 @@ get_select_query_def(Query *query, deparse_context *context,
else else
get_rule_expr(query->limitCount, context, false); get_rule_expr(query->limitCount, context, false);
} }
}
/* Add FOR [KEY] UPDATE/SHARE clauses if present */ /* Add FOR [KEY] UPDATE/SHARE clauses if present */
if (query->hasForUpdate) if (query->hasForUpdate)
......
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 202004071 #define CATALOG_VERSION_NO 202004072
#endif #endif
...@@ -2449,6 +2449,7 @@ typedef enum ...@@ -2449,6 +2449,7 @@ typedef enum
LIMIT_RESCAN, /* rescan after recomputing parameters */ LIMIT_RESCAN, /* rescan after recomputing parameters */
LIMIT_EMPTY, /* there are no returnable rows */ LIMIT_EMPTY, /* there are no returnable rows */
LIMIT_INWINDOW, /* have returned a row in the window */ LIMIT_INWINDOW, /* have returned a row in the window */
LIMIT_WINDOWEND_TIES, /* have returned a tied row */
LIMIT_SUBPLANEOF, /* at EOF of subplan (within window) */ LIMIT_SUBPLANEOF, /* at EOF of subplan (within window) */
LIMIT_WINDOWEND, /* stepped off end of window */ LIMIT_WINDOWEND, /* stepped off end of window */
LIMIT_WINDOWSTART /* stepped off beginning of window */ LIMIT_WINDOWSTART /* stepped off beginning of window */
...@@ -2459,12 +2460,16 @@ typedef struct LimitState ...@@ -2459,12 +2460,16 @@ typedef struct LimitState
PlanState ps; /* its first field is NodeTag */ PlanState ps; /* its first field is NodeTag */
ExprState *limitOffset; /* OFFSET parameter, or NULL if none */ ExprState *limitOffset; /* OFFSET parameter, or NULL if none */
ExprState *limitCount; /* COUNT parameter, or NULL if none */ ExprState *limitCount; /* COUNT parameter, or NULL if none */
LimitOption limitOption; /* limit specification type */
int64 offset; /* current OFFSET value */ int64 offset; /* current OFFSET value */
int64 count; /* current COUNT, if any */ int64 count; /* current COUNT, if any */
bool noCount; /* if true, ignore count */ bool noCount; /* if true, ignore count */
LimitStateCond lstate; /* state machine status, as above */ LimitStateCond lstate; /* state machine status, as above */
int64 position; /* 1-based index of last tuple returned */ int64 position; /* 1-based index of last tuple returned */
TupleTableSlot *subSlot; /* tuple last obtained from subplan */ TupleTableSlot *subSlot; /* tuple last obtained from subplan */
ExprState *eqfunction; /* tuple equality qual in case of WITH TIES
* option */
TupleTableSlot *last_slot; /* slot for evaluation of ties */
} LimitState; } LimitState;
#endif /* EXECNODES_H */ #endif /* EXECNODES_H */
...@@ -826,4 +826,17 @@ typedef enum OnConflictAction ...@@ -826,4 +826,17 @@ typedef enum OnConflictAction
ONCONFLICT_UPDATE /* ON CONFLICT ... DO UPDATE */ ONCONFLICT_UPDATE /* ON CONFLICT ... DO UPDATE */
} OnConflictAction; } OnConflictAction;
/*
* LimitOption -
* LIMIT option of query
*
* This is needed in both parsenodes.h and plannodes.h, so put it here...
*/
typedef enum LimitOption
{
LIMIT_OPTION_COUNT, /* FETCH FIRST... ONLY */
LIMIT_OPTION_WITH_TIES, /* FETCH FIRST... WITH TIES */
LIMIT_OPTION_DEFAULT, /* No limit present */
} LimitOption;
#endif /* NODES_H */ #endif /* NODES_H */
...@@ -159,6 +159,7 @@ typedef struct Query ...@@ -159,6 +159,7 @@ typedef struct Query
Node *limitOffset; /* # of result tuples to skip (int8 expr) */ Node *limitOffset; /* # of result tuples to skip (int8 expr) */
Node *limitCount; /* # of result tuples to return (int8 expr) */ Node *limitCount; /* # of result tuples to return (int8 expr) */
LimitOption limitOption; /* limit type */
List *rowMarks; /* a list of RowMarkClause's */ List *rowMarks; /* a list of RowMarkClause's */
...@@ -1617,6 +1618,7 @@ typedef struct SelectStmt ...@@ -1617,6 +1618,7 @@ typedef struct SelectStmt
List *sortClause; /* sort clause (a list of SortBy's) */ List *sortClause; /* sort clause (a list of SortBy's) */
Node *limitOffset; /* # of result tuples to skip */ Node *limitOffset; /* # of result tuples to skip */
Node *limitCount; /* # of result tuples to return */ Node *limitCount; /* # of result tuples to return */
LimitOption limitOption; /* limit type */
List *lockingClause; /* FOR UPDATE (list of LockingClause's) */ List *lockingClause; /* FOR UPDATE (list of LockingClause's) */
WithClause *withClause; /* WITH clause */ WithClause *withClause; /* WITH clause */
......
...@@ -1824,6 +1824,7 @@ typedef struct LimitPath ...@@ -1824,6 +1824,7 @@ typedef struct LimitPath
Path *subpath; /* path representing input source */ Path *subpath; /* path representing input source */
Node *limitOffset; /* OFFSET parameter, or NULL if none */ Node *limitOffset; /* OFFSET parameter, or NULL if none */
Node *limitCount; /* COUNT parameter, or NULL if none */ Node *limitCount; /* COUNT parameter, or NULL if none */
LimitOption limitOption; /* FETCH FIRST with ties or exact number */
} LimitPath; } LimitPath;
......
...@@ -982,6 +982,11 @@ typedef struct Limit ...@@ -982,6 +982,11 @@ typedef struct Limit
Plan plan; Plan plan;
Node *limitOffset; /* OFFSET parameter, or NULL if none */ Node *limitOffset; /* OFFSET parameter, or NULL if none */
Node *limitCount; /* COUNT parameter, or NULL if none */ Node *limitCount; /* COUNT parameter, or NULL if none */
LimitOption limitOption; /* limit type */
int uniqNumCols; /* number of columns to check for similarity */
AttrNumber *uniqColIdx; /* their indexes in the target list */
Oid *uniqOperators; /* equality operators to compare with */
Oid *uniqCollations; /* collations for equality comparisons */
} Limit; } Limit;
......
...@@ -268,6 +268,7 @@ extern ModifyTablePath *create_modifytable_path(PlannerInfo *root, ...@@ -268,6 +268,7 @@ extern ModifyTablePath *create_modifytable_path(PlannerInfo *root,
extern LimitPath *create_limit_path(PlannerInfo *root, RelOptInfo *rel, extern LimitPath *create_limit_path(PlannerInfo *root, RelOptInfo *rel,
Path *subpath, Path *subpath,
Node *limitOffset, Node *limitCount, Node *limitOffset, Node *limitCount,
LimitOption limitOption,
int64 offset_est, int64 count_est); int64 offset_est, int64 count_est);
extern void adjust_limit_rows_costs(double *rows, extern void adjust_limit_rows_costs(double *rows,
Cost *startup_cost, Cost *total_cost, Cost *startup_cost, Cost *total_cost,
......
...@@ -56,7 +56,10 @@ extern Agg *make_agg(List *tlist, List *qual, ...@@ -56,7 +56,10 @@ extern Agg *make_agg(List *tlist, List *qual,
int numGroupCols, AttrNumber *grpColIdx, Oid *grpOperators, Oid *grpCollations, int numGroupCols, AttrNumber *grpColIdx, Oid *grpOperators, Oid *grpCollations,
List *groupingSets, List *chain, double dNumGroups, List *groupingSets, List *chain, double dNumGroups,
Size transitionSpace, Plan *lefttree); Size transitionSpace, Plan *lefttree);
extern Limit *make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount); extern Limit *make_limit(Plan *lefttree, Node *limitOffset, Node *limitCount,
LimitOption limitOption, int uniqNumCols,
AttrNumber *uniqColIdx, Oid *uniqOperators,
Oid *uniqCollations);
/* /*
* prototypes for plan/initsplan.c * prototypes for plan/initsplan.c
......
...@@ -23,7 +23,8 @@ extern int setTargetTable(ParseState *pstate, RangeVar *relation, ...@@ -23,7 +23,8 @@ extern int setTargetTable(ParseState *pstate, RangeVar *relation,
extern Node *transformWhereClause(ParseState *pstate, Node *clause, extern Node *transformWhereClause(ParseState *pstate, Node *clause,
ParseExprKind exprKind, const char *constructName); ParseExprKind exprKind, const char *constructName);
extern Node *transformLimitClause(ParseState *pstate, Node *clause, extern Node *transformLimitClause(ParseState *pstate, Node *clause,
ParseExprKind exprKind, const char *constructName); ParseExprKind exprKind, const char *constructName,
LimitOption limitOption);
extern List *transformGroupClause(ParseState *pstate, List *grouplist, extern List *transformGroupClause(ParseState *pstate, List *grouplist,
List **groupingSets, List **groupingSets,
List **targetlist, List *sortClause, List **targetlist, List *sortClause,
......
...@@ -286,6 +286,58 @@ fetch all in c4; ...@@ -286,6 +286,58 @@ fetch all in c4;
----+---- ----+----
(0 rows) (0 rows)
declare c5 cursor for select * from int8_tbl order by q1 fetch first 2 rows with ties;
fetch all in c5;
q1 | q2
-----+------------------
123 | 456
123 | 4567890123456789
(2 rows)
fetch 1 in c5;
q1 | q2
----+----
(0 rows)
fetch backward 1 in c5;
q1 | q2
-----+------------------
123 | 4567890123456789
(1 row)
fetch backward 1 in c5;
q1 | q2
-----+-----
123 | 456
(1 row)
fetch all in c5;
q1 | q2
-----+------------------
123 | 4567890123456789
(1 row)
fetch backward all in c5;
q1 | q2
-----+------------------
123 | 4567890123456789
123 | 456
(2 rows)
fetch all in c5;
q1 | q2
-----+------------------
123 | 456
123 | 4567890123456789
(2 rows)
fetch backward all in c5;
q1 | q2
-----+------------------
123 | 4567890123456789
123 | 456
(2 rows)
rollback; rollback;
-- Stress test for variable LIMIT in conjunction with bounded-heap sorting -- Stress test for variable LIMIT in conjunction with bounded-heap sorting
SELECT SELECT
...@@ -503,3 +555,118 @@ select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2 ...@@ -503,3 +555,118 @@ select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2
45020 | 45020 45020 | 45020
(3 rows) (3 rows)
--
-- FETCH FIRST
-- Check the WITH TIES clause
--
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 2 ROW WITH TIES;
thousand
----------
0
0
0
0
0
0
0
0
0
0
(10 rows)
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 1 ROW WITH TIES;
thousand
----------
0
0
0
0
0
0
0
0
0
0
(10 rows)
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 2 ROW ONLY;
thousand
----------
0
0
(2 rows)
-- should fail
SELECT ''::text AS two, unique1, unique2, stringu1
FROM onek WHERE unique1 > 50
FETCH FIRST 2 ROW WITH TIES;
ERROR: WITH TIES options can not be specified without ORDER BY clause
-- test ruleutils
CREATE VIEW limit_thousand_v_1 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST 5 ROWS WITH TIES OFFSET 10;
\d+ limit_thousand_v_1
View "public.limit_thousand_v_1"
Column | Type | Collation | Nullable | Default | Storage | Description
----------+---------+-----------+----------+---------+---------+-------------
thousand | integer | | | | plain |
View definition:
SELECT onek.thousand
FROM onek
WHERE onek.thousand < 995
ORDER BY onek.thousand
OFFSET 10
FETCH FIRST 5 ROWS WITH TIES;
CREATE VIEW limit_thousand_v_2 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand OFFSET 10 FETCH FIRST 5 ROWS ONLY;
\d+ limit_thousand_v_2
View "public.limit_thousand_v_2"
Column | Type | Collation | Nullable | Default | Storage | Description
----------+---------+-----------+----------+---------+---------+-------------
thousand | integer | | | | plain |
View definition:
SELECT onek.thousand
FROM onek
WHERE onek.thousand < 995
ORDER BY onek.thousand
OFFSET 10
LIMIT 5;
CREATE VIEW limit_thousand_v_3 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST NULL ROWS WITH TIES; -- fails
ERROR: row count cannot be NULL in FETCH FIRST ... WITH TIES clause
CREATE VIEW limit_thousand_v_3 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST (NULL+1) ROWS WITH TIES;
\d+ limit_thousand_v_3
View "public.limit_thousand_v_3"
Column | Type | Collation | Nullable | Default | Storage | Description
----------+---------+-----------+----------+---------+---------+-------------
thousand | integer | | | | plain |
View definition:
SELECT onek.thousand
FROM onek
WHERE onek.thousand < 995
ORDER BY onek.thousand
FETCH FIRST (NULL::integer + 1) ROWS WITH TIES;
CREATE VIEW limit_thousand_v_4 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST NULL ROWS ONLY;
\d+ limit_thousand_v_4
View "public.limit_thousand_v_4"
Column | Type | Collation | Nullable | Default | Storage | Description
----------+---------+-----------+----------+---------+---------+-------------
thousand | integer | | | | plain |
View definition:
SELECT onek.thousand
FROM onek
WHERE onek.thousand < 995
ORDER BY onek.thousand
LIMIT ALL;
-- leave these views
...@@ -71,6 +71,16 @@ fetch backward all in c4; ...@@ -71,6 +71,16 @@ fetch backward all in c4;
fetch backward 1 in c4; fetch backward 1 in c4;
fetch all in c4; fetch all in c4;
declare c5 cursor for select * from int8_tbl order by q1 fetch first 2 rows with ties;
fetch all in c5;
fetch 1 in c5;
fetch backward 1 in c5;
fetch backward 1 in c5;
fetch all in c5;
fetch backward all in c5;
fetch all in c5;
fetch backward all in c5;
rollback; rollback;
-- Stress test for variable LIMIT in conjunction with bounded-heap sorting -- Stress test for variable LIMIT in conjunction with bounded-heap sorting
...@@ -141,3 +151,41 @@ select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2 ...@@ -141,3 +151,41 @@ select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2
select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2 select sum(tenthous) as s1, sum(tenthous) + random()*0 as s2
from tenk1 group by thousand order by thousand limit 3; from tenk1 group by thousand order by thousand limit 3;
--
-- FETCH FIRST
-- Check the WITH TIES clause
--
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 2 ROW WITH TIES;
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 1 ROW WITH TIES;
SELECT thousand
FROM onek WHERE thousand < 5
ORDER BY thousand FETCH FIRST 2 ROW ONLY;
-- should fail
SELECT ''::text AS two, unique1, unique2, stringu1
FROM onek WHERE unique1 > 50
FETCH FIRST 2 ROW WITH TIES;
-- test ruleutils
CREATE VIEW limit_thousand_v_1 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST 5 ROWS WITH TIES OFFSET 10;
\d+ limit_thousand_v_1
CREATE VIEW limit_thousand_v_2 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand OFFSET 10 FETCH FIRST 5 ROWS ONLY;
\d+ limit_thousand_v_2
CREATE VIEW limit_thousand_v_3 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST NULL ROWS WITH TIES; -- fails
CREATE VIEW limit_thousand_v_3 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST (NULL+1) ROWS WITH TIES;
\d+ limit_thousand_v_3
CREATE VIEW limit_thousand_v_4 AS SELECT thousand FROM onek WHERE thousand < 995
ORDER BY thousand FETCH FIRST NULL ROWS ONLY;
\d+ limit_thousand_v_4
-- leave these views
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