Skip to content

Commit df3c565

Browse files
davispalamb
andauthored
Improve parsing performance by reducing token cloning (apache#1587)
Co-authored-by: Andrew Lamb <[email protected]>
1 parent 024a878 commit df3c565

File tree

6 files changed

+344
-244
lines changed

6 files changed

+344
-244
lines changed

README.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -240,11 +240,14 @@ You can run them with:
240240
```
241241
git checkout main
242242
cd sqlparser_bench
243-
cargo bench
243+
cargo bench -- --save-baseline main
244244
git checkout <your branch>
245-
cargo bench
245+
cargo bench -- --baseline main
246246
```
247247

248+
By adding the `--save-baseline main` and `--baseline main` you can track the
249+
progress of your improvements as you continue working on the feature branch.
250+
248251
## Licensing
249252

250253
All code in this repository is licensed under the [Apache Software License 2.0](LICENSE.txt).

src/dialect/mod.rs

+9
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,15 @@ macro_rules! dialect_of {
7575
};
7676
}
7777

78+
// Similar to above, but for applying directly against an instance of dialect
79+
// instead of a struct member named dialect. This avoids lifetime issues when
80+
// mixing match guards and token references.
81+
macro_rules! dialect_is {
82+
($dialect:ident is $($dialect_type:ty)|+) => {
83+
($($dialect.is::<$dialect_type>())||+)
84+
}
85+
}
86+
7887
/// Encapsulates the differences between SQL implementations.
7988
///
8089
/// # SQL Dialects

src/dialect/postgresql.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -245,11 +245,11 @@ impl Dialect for PostgreSqlDialect {
245245

246246
pub fn parse_create(parser: &mut Parser) -> Option<Result<Statement, ParserError>> {
247247
let name = parser.maybe_parse(|parser| -> Result<ObjectName, ParserError> {
248-
parser.expect_keyword(Keyword::CREATE)?;
249-
parser.expect_keyword(Keyword::TYPE)?;
248+
parser.expect_keyword_is(Keyword::CREATE)?;
249+
parser.expect_keyword_is(Keyword::TYPE)?;
250250
let name = parser.parse_object_name(false)?;
251-
parser.expect_keyword(Keyword::AS)?;
252-
parser.expect_keyword(Keyword::ENUM)?;
251+
parser.expect_keyword_is(Keyword::AS)?;
252+
parser.expect_keyword_is(Keyword::ENUM)?;
253253
Ok(name)
254254
});
255255

src/dialect/snowflake.rs

+8-8
Original file line numberDiff line numberDiff line change
@@ -273,7 +273,7 @@ pub fn parse_create_table(
273273
match &next_token.token {
274274
Token::Word(word) => match word.keyword {
275275
Keyword::COPY => {
276-
parser.expect_keyword(Keyword::GRANTS)?;
276+
parser.expect_keyword_is(Keyword::GRANTS)?;
277277
builder = builder.copy_grants(true);
278278
}
279279
Keyword::COMMENT => {
@@ -297,7 +297,7 @@ pub fn parse_create_table(
297297
break;
298298
}
299299
Keyword::CLUSTER => {
300-
parser.expect_keyword(Keyword::BY)?;
300+
parser.expect_keyword_is(Keyword::BY)?;
301301
parser.expect_token(&Token::LParen)?;
302302
let cluster_by = Some(WrappedCollection::Parentheses(
303303
parser.parse_comma_separated(|p| p.parse_identifier(false))?,
@@ -360,14 +360,14 @@ pub fn parse_create_table(
360360
parser.prev_token();
361361
}
362362
Keyword::AGGREGATION => {
363-
parser.expect_keyword(Keyword::POLICY)?;
363+
parser.expect_keyword_is(Keyword::POLICY)?;
364364
let aggregation_policy = parser.parse_object_name(false)?;
365365
builder = builder.with_aggregation_policy(Some(aggregation_policy));
366366
}
367367
Keyword::ROW => {
368368
parser.expect_keywords(&[Keyword::ACCESS, Keyword::POLICY])?;
369369
let policy = parser.parse_object_name(false)?;
370-
parser.expect_keyword(Keyword::ON)?;
370+
parser.expect_keyword_is(Keyword::ON)?;
371371
parser.expect_token(&Token::LParen)?;
372372
let columns = parser.parse_comma_separated(|p| p.parse_identifier(false))?;
373373
parser.expect_token(&Token::RParen)?;
@@ -536,15 +536,15 @@ pub fn parse_copy_into(parser: &mut Parser) -> Result<Statement, ParserError> {
536536
let from_stage: ObjectName;
537537
let stage_params: StageParamsObject;
538538

539-
parser.expect_keyword(Keyword::FROM)?;
539+
parser.expect_keyword_is(Keyword::FROM)?;
540540
// check if data load transformations are present
541541
match parser.next_token().token {
542542
Token::LParen => {
543543
// data load with transformations
544-
parser.expect_keyword(Keyword::SELECT)?;
544+
parser.expect_keyword_is(Keyword::SELECT)?;
545545
from_transformations = parse_select_items_for_data_load(parser)?;
546546

547-
parser.expect_keyword(Keyword::FROM)?;
547+
parser.expect_keyword_is(Keyword::FROM)?;
548548
from_stage = parse_snowflake_stage_name(parser)?;
549549
stage_params = parse_stage_params(parser)?;
550550

@@ -860,7 +860,7 @@ fn parse_identity_property(parser: &mut Parser) -> Result<IdentityProperty, Pars
860860
))
861861
} else if parser.parse_keyword(Keyword::START) {
862862
let seed = parser.parse_number()?;
863-
parser.expect_keyword(Keyword::INCREMENT)?;
863+
parser.expect_keyword_is(Keyword::INCREMENT)?;
864864
let increment = parser.parse_number()?;
865865

866866
Some(IdentityPropertyFormatKind::StartAndIncrement(

src/parser/alter.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -52,11 +52,11 @@ impl Parser<'_> {
5252
/// [PostgreSQL](https://www.postgresql.org/docs/current/sql-alterpolicy.html)
5353
pub fn parse_alter_policy(&mut self) -> Result<Statement, ParserError> {
5454
let name = self.parse_identifier(false)?;
55-
self.expect_keyword(Keyword::ON)?;
55+
self.expect_keyword_is(Keyword::ON)?;
5656
let table_name = self.parse_object_name(false)?;
5757

5858
if self.parse_keyword(Keyword::RENAME) {
59-
self.expect_keyword(Keyword::TO)?;
59+
self.expect_keyword_is(Keyword::TO)?;
6060
let new_name = self.parse_identifier(false)?;
6161
Ok(Statement::AlterPolicy {
6262
name,
@@ -232,7 +232,7 @@ impl Parser<'_> {
232232
Some(Keyword::BYPASSRLS) => RoleOption::BypassRLS(true),
233233
Some(Keyword::NOBYPASSRLS) => RoleOption::BypassRLS(false),
234234
Some(Keyword::CONNECTION) => {
235-
self.expect_keyword(Keyword::LIMIT)?;
235+
self.expect_keyword_is(Keyword::LIMIT)?;
236236
RoleOption::ConnectionLimit(Expr::Value(self.parse_number_value()?))
237237
}
238238
Some(Keyword::CREATEDB) => RoleOption::CreateDB(true),
@@ -256,7 +256,7 @@ impl Parser<'_> {
256256
Some(Keyword::SUPERUSER) => RoleOption::SuperUser(true),
257257
Some(Keyword::NOSUPERUSER) => RoleOption::SuperUser(false),
258258
Some(Keyword::VALID) => {
259-
self.expect_keyword(Keyword::UNTIL)?;
259+
self.expect_keyword_is(Keyword::UNTIL)?;
260260
RoleOption::ValidUntil(Expr::Value(self.parse_value()?))
261261
}
262262
_ => self.expected("option", self.peek_token())?,

0 commit comments

Comments
 (0)