mirror of
https://github.com/apache/datafusion-sqlparser-rs.git
synced 2025-08-25 08:24:05 +00:00
feat: Parse special keywords as functions (current_user, user, etc) (#561)
* feat: Parse special keywors as functions (current_user, user, etc) * explain special field
This commit is contained in:
parent
b6e36ad760
commit
54a29e872d
6 changed files with 111 additions and 17 deletions
|
@ -2321,20 +2321,29 @@ pub struct Function {
|
||||||
pub over: Option<WindowSpec>,
|
pub over: Option<WindowSpec>,
|
||||||
// aggregate functions may specify eg `COUNT(DISTINCT x)`
|
// aggregate functions may specify eg `COUNT(DISTINCT x)`
|
||||||
pub distinct: bool,
|
pub distinct: bool,
|
||||||
|
// Some functions must be called without trailing parentheses, for example Postgres
|
||||||
|
// do it for current_catalog, current_schema, etc. This flags is used for formatting.
|
||||||
|
pub special: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Display for Function {
|
impl fmt::Display for Function {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
write!(
|
if self.special {
|
||||||
f,
|
write!(f, "{}", self.name)?;
|
||||||
"{}({}{})",
|
} else {
|
||||||
self.name,
|
write!(
|
||||||
if self.distinct { "DISTINCT " } else { "" },
|
f,
|
||||||
display_comma_separated(&self.args),
|
"{}({}{})",
|
||||||
)?;
|
self.name,
|
||||||
if let Some(o) = &self.over {
|
if self.distinct { "DISTINCT " } else { "" },
|
||||||
write!(f, " OVER ({})", o)?;
|
display_comma_separated(&self.args),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
if let Some(o) = &self.over {
|
||||||
|
write!(f, " OVER ({})", o)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -421,6 +421,20 @@ impl<'a> Parser<'a> {
|
||||||
self.prev_token();
|
self.prev_token();
|
||||||
Ok(Expr::Value(self.parse_value()?))
|
Ok(Expr::Value(self.parse_value()?))
|
||||||
}
|
}
|
||||||
|
Keyword::CURRENT_CATALOG
|
||||||
|
| Keyword::CURRENT_USER
|
||||||
|
| Keyword::SESSION_USER
|
||||||
|
| Keyword::USER
|
||||||
|
if dialect_of!(self is PostgreSqlDialect | GenericDialect) =>
|
||||||
|
{
|
||||||
|
Ok(Expr::Function(Function {
|
||||||
|
name: ObjectName(vec![w.to_ident()]),
|
||||||
|
args: vec![],
|
||||||
|
over: None,
|
||||||
|
distinct: false,
|
||||||
|
special: true,
|
||||||
|
}))
|
||||||
|
}
|
||||||
Keyword::CURRENT_TIMESTAMP | Keyword::CURRENT_TIME | Keyword::CURRENT_DATE => {
|
Keyword::CURRENT_TIMESTAMP | Keyword::CURRENT_TIME | Keyword::CURRENT_DATE => {
|
||||||
self.parse_time_functions(ObjectName(vec![w.to_ident()]))
|
self.parse_time_functions(ObjectName(vec![w.to_ident()]))
|
||||||
}
|
}
|
||||||
|
@ -598,6 +612,7 @@ impl<'a> Parser<'a> {
|
||||||
args,
|
args,
|
||||||
over,
|
over,
|
||||||
distinct,
|
distinct,
|
||||||
|
special: false,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -612,6 +627,7 @@ impl<'a> Parser<'a> {
|
||||||
args,
|
args,
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -572,6 +572,7 @@ fn parse_select_count_wildcard() {
|
||||||
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Wildcard)],
|
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Wildcard)],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(only(&select.projection))
|
expr_from_projection(only(&select.projection))
|
||||||
);
|
);
|
||||||
|
@ -590,6 +591,7 @@ fn parse_select_count_distinct() {
|
||||||
}))],
|
}))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: true,
|
distinct: true,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(only(&select.projection))
|
expr_from_projection(only(&select.projection))
|
||||||
);
|
);
|
||||||
|
@ -1414,6 +1416,7 @@ fn parse_select_having() {
|
||||||
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Wildcard)],
|
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Wildcard)],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})),
|
})),
|
||||||
op: BinaryOperator::Gt,
|
op: BinaryOperator::Gt,
|
||||||
right: Box::new(Expr::Value(number("1")))
|
right: Box::new(Expr::Value(number("1")))
|
||||||
|
@ -1445,7 +1448,8 @@ fn parse_select_qualify() {
|
||||||
}],
|
}],
|
||||||
window_frame: None
|
window_frame: None
|
||||||
}),
|
}),
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false
|
||||||
})),
|
})),
|
||||||
op: BinaryOperator::Eq,
|
op: BinaryOperator::Eq,
|
||||||
right: Box::new(Expr::Value(number("1")))
|
right: Box::new(Expr::Value(number("1")))
|
||||||
|
@ -2532,6 +2536,7 @@ fn parse_scalar_function_in_projection() {
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(only(&select.projection))
|
expr_from_projection(only(&select.projection))
|
||||||
);
|
);
|
||||||
|
@ -2610,6 +2615,7 @@ fn parse_named_argument_function() {
|
||||||
],
|
],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(only(&select.projection))
|
expr_from_projection(only(&select.projection))
|
||||||
);
|
);
|
||||||
|
@ -2643,6 +2649,7 @@ fn parse_window_functions() {
|
||||||
window_frame: None,
|
window_frame: None,
|
||||||
}),
|
}),
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(&select.projection[0])
|
expr_from_projection(&select.projection[0])
|
||||||
);
|
);
|
||||||
|
@ -2906,7 +2913,8 @@ fn parse_at_timezone() {
|
||||||
}]),
|
}]),
|
||||||
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Expr(zero.clone()))],
|
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Expr(zero.clone()))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})),
|
})),
|
||||||
time_zone: "UTC-06:00".to_string()
|
time_zone: "UTC-06:00".to_string()
|
||||||
},
|
},
|
||||||
|
@ -2932,6 +2940,7 @@ fn parse_at_timezone() {
|
||||||
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Expr(zero,),),],
|
args: vec![FunctionArg::Unnamed(FunctionArgExpr::Expr(zero,),),],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false
|
||||||
},)),
|
},)),
|
||||||
time_zone: "UTC-06:00".to_string(),
|
time_zone: "UTC-06:00".to_string(),
|
||||||
},),),
|
},),),
|
||||||
|
@ -2941,6 +2950,7 @@ fn parse_at_timezone() {
|
||||||
],
|
],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false
|
||||||
},),
|
},),
|
||||||
alias: Ident {
|
alias: Ident {
|
||||||
value: "hour".to_string(),
|
value: "hour".to_string(),
|
||||||
|
@ -2976,6 +2986,7 @@ fn parse_table_function() {
|
||||||
)))],
|
)))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
});
|
});
|
||||||
assert_eq!(expr, expected_expr);
|
assert_eq!(expr, expected_expr);
|
||||||
assert_eq!(alias, table_alias("a"))
|
assert_eq!(alias, table_alias("a"))
|
||||||
|
@ -3148,6 +3159,7 @@ fn parse_delimited_identifiers() {
|
||||||
args: vec![],
|
args: vec![],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(&select.projection[1]),
|
expr_from_projection(&select.projection[1]),
|
||||||
);
|
);
|
||||||
|
@ -5125,6 +5137,7 @@ fn parse_time_functions() {
|
||||||
args: vec![],
|
args: vec![],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(&select.projection[0])
|
expr_from_projection(&select.projection[0])
|
||||||
);
|
);
|
||||||
|
@ -5140,6 +5153,7 @@ fn parse_time_functions() {
|
||||||
args: vec![],
|
args: vec![],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(&select.projection[0])
|
expr_from_projection(&select.projection[0])
|
||||||
);
|
);
|
||||||
|
@ -5155,6 +5169,7 @@ fn parse_time_functions() {
|
||||||
args: vec![],
|
args: vec![],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
}),
|
}),
|
||||||
expr_from_projection(&select.projection[0])
|
expr_from_projection(&select.projection[0])
|
||||||
);
|
);
|
||||||
|
|
|
@ -686,7 +686,8 @@ fn parse_insert_with_on_duplicate_update() {
|
||||||
Expr::Identifier(Ident::new("description"))
|
Expr::Identifier(Ident::new("description"))
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
Assignment {
|
Assignment {
|
||||||
|
@ -697,7 +698,8 @@ fn parse_insert_with_on_duplicate_update() {
|
||||||
Expr::Identifier(Ident::new("perm_create"))
|
Expr::Identifier(Ident::new("perm_create"))
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
Assignment {
|
Assignment {
|
||||||
|
@ -708,7 +710,8 @@ fn parse_insert_with_on_duplicate_update() {
|
||||||
Expr::Identifier(Ident::new("perm_read"))
|
Expr::Identifier(Ident::new("perm_read"))
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
Assignment {
|
Assignment {
|
||||||
|
@ -719,7 +722,8 @@ fn parse_insert_with_on_duplicate_update() {
|
||||||
Expr::Identifier(Ident::new("perm_update"))
|
Expr::Identifier(Ident::new("perm_update"))
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
Assignment {
|
Assignment {
|
||||||
|
@ -730,7 +734,8 @@ fn parse_insert_with_on_duplicate_update() {
|
||||||
Expr::Identifier(Ident::new("perm_delete"))
|
Expr::Identifier(Ident::new("perm_delete"))
|
||||||
))],
|
))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
])),
|
])),
|
||||||
|
|
|
@ -1400,6 +1400,7 @@ fn test_composite_value() {
|
||||||
)))],
|
)))],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false
|
||||||
}))))
|
}))))
|
||||||
}),
|
}),
|
||||||
select.projection[0]
|
select.projection[0]
|
||||||
|
@ -1542,6 +1543,52 @@ fn parse_declare() {
|
||||||
pg_and_generic().verified_stmt("DECLARE \"SQL_CUR0x7fa44801bc00\" BINARY INSENSITIVE SCROLL CURSOR WITH HOLD FOR SELECT * FROM table_name LIMIT 2222");
|
pg_and_generic().verified_stmt("DECLARE \"SQL_CUR0x7fa44801bc00\" BINARY INSENSITIVE SCROLL CURSOR WITH HOLD FOR SELECT * FROM table_name LIMIT 2222");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn parse_current_functions() {
|
||||||
|
let sql = "SELECT CURRENT_CATALOG, CURRENT_USER, SESSION_USER, USER";
|
||||||
|
let select = pg_and_generic().verified_only_select(sql);
|
||||||
|
assert_eq!(
|
||||||
|
&Expr::Function(Function {
|
||||||
|
name: ObjectName(vec![Ident::new("CURRENT_CATALOG")]),
|
||||||
|
args: vec![],
|
||||||
|
over: None,
|
||||||
|
distinct: false,
|
||||||
|
special: true,
|
||||||
|
}),
|
||||||
|
expr_from_projection(&select.projection[0])
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
&Expr::Function(Function {
|
||||||
|
name: ObjectName(vec![Ident::new("CURRENT_USER")]),
|
||||||
|
args: vec![],
|
||||||
|
over: None,
|
||||||
|
distinct: false,
|
||||||
|
special: true,
|
||||||
|
}),
|
||||||
|
expr_from_projection(&select.projection[1])
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
&Expr::Function(Function {
|
||||||
|
name: ObjectName(vec![Ident::new("SESSION_USER")]),
|
||||||
|
args: vec![],
|
||||||
|
over: None,
|
||||||
|
distinct: false,
|
||||||
|
special: true,
|
||||||
|
}),
|
||||||
|
expr_from_projection(&select.projection[2])
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
&Expr::Function(Function {
|
||||||
|
name: ObjectName(vec![Ident::new("USER")]),
|
||||||
|
args: vec![],
|
||||||
|
over: None,
|
||||||
|
distinct: false,
|
||||||
|
special: true,
|
||||||
|
}),
|
||||||
|
expr_from_projection(&select.projection[3])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_fetch() {
|
fn parse_fetch() {
|
||||||
pg_and_generic().verified_stmt("FETCH 2048 IN \"SQL_CUR0x7fa44801bc00\"");
|
pg_and_generic().verified_stmt("FETCH 2048 IN \"SQL_CUR0x7fa44801bc00\"");
|
||||||
|
|
|
@ -51,6 +51,7 @@ fn parse_map_access_expr() {
|
||||||
],
|
],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false,
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})],
|
})],
|
||||||
})],
|
})],
|
||||||
into: None,
|
into: None,
|
||||||
|
@ -85,7 +86,8 @@ fn parse_map_access_expr() {
|
||||||
))),
|
))),
|
||||||
],
|
],
|
||||||
over: None,
|
over: None,
|
||||||
distinct: false
|
distinct: false,
|
||||||
|
special: false,
|
||||||
})]
|
})]
|
||||||
}),
|
}),
|
||||||
op: BinaryOperator::NotEq,
|
op: BinaryOperator::NotEq,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue