-
Notifications
You must be signed in to change notification settings - Fork 595
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: support expression in limit clause #19834
Conversation
src/sqlparser/src/ast/query.rs
Outdated
@@ -655,7 +655,7 @@ impl fmt::Display for OrderByExpr { | |||
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] | |||
pub struct Fetch { | |||
pub with_ties: bool, | |||
pub quantity: Option<String>, | |||
pub quantity: Option<Expr>, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not as simple as it looks like. A parsing conflict can happen:
https://github.com/postgres/postgres/blob/REL_17_2/src/backend/parser/gram.y#L13233
* Allowing full expressions without parentheses causes various parsing * problems with the trailing ROW/ROWS key words. SQL spec only calls for * <simple value specification>, which is either a literal or a parameter (but * an <SQL parameter reference> could be an identifier, bringing up conflicts * with ROW/ROWS). We solve this by leveraging the presence of ONLY (see above) * to determine whether the expression is missing rather than trying to make it * optional in this rule. * c_expr covers almost all the spec-required cases (and more), but it doesn't * cover signed numeric literals, which are allowed by the spec. So we include * those here explicitly. We need FCONST as well as ICONST because values that * don't fit in the platform's "long", but do fit in bigint, should still be * accepted here. (This is possible in 64-bit Windows as well as all 32-bit * builds.)
For example,
select 1 offset 1 row fetch first row only;
^
This is valid in PostgreSQL. When we see this row
keyword, it should be part of the fetch first [n] row
syntax where n
is omitted, but it may be mistakenly treated as part of the row(field0, field1)
expression because we will attempt to parse an expression (instead of a numerical constant) after first
.
What is the actual parsing behavior when this commit sees this SQL?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
how about let me revert the changes for "fetch" but keep the changes for "limit",
the CI test required by SQLMesh integration needs limit simple expression
only.
let fetch
still stick to a literal number
I will open a new issue for the reasoning posted above
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
or we force parentheses around the expression in Fetch just like PG
SELECT id
FROM generate_series(1, 10) AS id
ORDER BY id
OFFSET 5 ROWS
FETCH NEXT (1+4) ROWS ONLY;
id
----
6
7
8
9
10
(5 rows)
SELECT id
FROM generate_series(1, 10) AS id
ORDER BY id
OFFSET 5 ROWS
FETCH NEXT 1+4 ROWS ONLY;
psql:commands.sql:11: ERROR: syntax error at or near "+"
LINE 5: FETCH NEXT 1+4 ROWS ONLY;
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
or we force parentheses around the expression in Fetch just like PG
PostgreSQL is more complex than this, but yes parenthesized expr is a subset of what PostgreSQL allows (c_expr
as defined in the linked gram.y
above, while limit
allows a_expr
. A includes C includes parenthesized expr.).
4ce5e6d
to
ee54595
Compare
c66c0db
to
9aab24b
Compare
a4b261b
to
bdbc790
Compare
3de56f6
to
705aaa7
Compare
I hereby agree to the terms of the RisingWave Labs, Inc. Contributor License Agreement.
What's changed and what's your intention?
fix #19830
required by SQLMesh's CI test
The fix is to evaluate the expression during binding stage so that we either accept the result (it is a number >=0 after casting to bigint) or reject it (all the other cases)
Checklist
Documentation
Release note