mirror of
https://github.com/apache/datafusion-sqlparser-rs.git
synced 2025-09-09 23:46:20 +00:00
commit
7aff70772b
3 changed files with 53 additions and 0 deletions
|
@ -119,6 +119,9 @@ pub enum ASTNode {
|
||||||
results: Vec<ASTNode>,
|
results: Vec<ASTNode>,
|
||||||
else_result: Option<Box<ASTNode>>,
|
else_result: Option<Box<ASTNode>>,
|
||||||
},
|
},
|
||||||
|
/// An exists expression `EXISTS(SELECT ...)`, used in expressions like
|
||||||
|
/// `WHERE EXISTS (SELECT ...)`.
|
||||||
|
SQLExists(Box<SQLQuery>),
|
||||||
/// A parenthesized subquery `(SELECT ...)`, used in expression like
|
/// A parenthesized subquery `(SELECT ...)`, used in expression like
|
||||||
/// `SELECT (subquery) AS x` or `WHERE (subquery) = x`
|
/// `SELECT (subquery) AS x` or `WHERE (subquery) = x`
|
||||||
SQLSubquery(Box<SQLQuery>),
|
SQLSubquery(Box<SQLQuery>),
|
||||||
|
@ -208,6 +211,7 @@ impl ToString for ASTNode {
|
||||||
}
|
}
|
||||||
s + " END"
|
s + " END"
|
||||||
}
|
}
|
||||||
|
ASTNode::SQLExists(s) => format!("EXISTS ({})", s.to_string()),
|
||||||
ASTNode::SQLSubquery(s) => format!("({})", s.to_string()),
|
ASTNode::SQLSubquery(s) => format!("({})", s.to_string()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -191,6 +191,7 @@ impl Parser {
|
||||||
}
|
}
|
||||||
"CASE" => self.parse_case_expression(),
|
"CASE" => self.parse_case_expression(),
|
||||||
"CAST" => self.parse_cast_expression(),
|
"CAST" => self.parse_cast_expression(),
|
||||||
|
"EXISTS" => self.parse_exists_expression(),
|
||||||
"NOT" => Ok(ASTNode::SQLUnary {
|
"NOT" => Ok(ASTNode::SQLUnary {
|
||||||
operator: SQLOperator::Not,
|
operator: SQLOperator::Not,
|
||||||
expr: Box::new(self.parse_subexpr(Self::UNARY_NOT_PREC)?),
|
expr: Box::new(self.parse_subexpr(Self::UNARY_NOT_PREC)?),
|
||||||
|
@ -404,6 +405,14 @@ impl Parser {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Parse a SQL EXISTS expression e.g. `WHERE EXISTS(SELECT ...)`.
|
||||||
|
pub fn parse_exists_expression(&mut self) -> Result<ASTNode, ParserError> {
|
||||||
|
self.expect_token(&Token::LParen)?;
|
||||||
|
let exists_node = ASTNode::SQLExists(Box::new(self.parse_query()?));
|
||||||
|
self.expect_token(&Token::RParen)?;
|
||||||
|
Ok(exists_node)
|
||||||
|
}
|
||||||
|
|
||||||
/// Parse an operator following an expression
|
/// Parse an operator following an expression
|
||||||
pub fn parse_infix(&mut self, expr: ASTNode, precedence: u8) -> Result<ASTNode, ParserError> {
|
pub fn parse_infix(&mut self, expr: ASTNode, precedence: u8) -> Result<ASTNode, ParserError> {
|
||||||
debug!("parsing infix");
|
debug!("parsing infix");
|
||||||
|
|
|
@ -1434,6 +1434,46 @@ fn parse_scalar_subqueries() {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn parse_exists_subquery() {
|
||||||
|
let expected_inner = verified_query("SELECT 1");
|
||||||
|
let sql = "SELECT * FROM t WHERE EXISTS (SELECT 1)";
|
||||||
|
let select = verified_only_select(sql);
|
||||||
|
assert_eq!(
|
||||||
|
ASTNode::SQLExists(Box::new(expected_inner.clone())),
|
||||||
|
select.selection.unwrap(),
|
||||||
|
);
|
||||||
|
|
||||||
|
let sql = "SELECT * FROM t WHERE NOT EXISTS (SELECT 1)";
|
||||||
|
let select = verified_only_select(sql);
|
||||||
|
assert_eq!(
|
||||||
|
ASTNode::SQLUnary {
|
||||||
|
operator: SQLOperator::Not,
|
||||||
|
expr: Box::new(ASTNode::SQLExists(Box::new(expected_inner))),
|
||||||
|
},
|
||||||
|
select.selection.unwrap(),
|
||||||
|
);
|
||||||
|
|
||||||
|
verified_stmt("SELECT * FROM t WHERE EXISTS (WITH u AS (SELECT 1) SELECT * FROM u)");
|
||||||
|
verified_stmt("SELECT EXISTS (SELECT 1)");
|
||||||
|
|
||||||
|
let res = parse_sql_statements("SELECT EXISTS (");
|
||||||
|
assert_eq!(
|
||||||
|
ParserError::ParserError(
|
||||||
|
"Expected SELECT or a subquery in the query body, found: EOF".to_string()
|
||||||
|
),
|
||||||
|
res.unwrap_err(),
|
||||||
|
);
|
||||||
|
|
||||||
|
let res = parse_sql_statements("SELECT EXISTS (NULL)");
|
||||||
|
assert_eq!(
|
||||||
|
ParserError::ParserError(
|
||||||
|
"Expected SELECT or a subquery in the query body, found: NULL".to_string()
|
||||||
|
),
|
||||||
|
res.unwrap_err(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_create_view() {
|
fn parse_create_view() {
|
||||||
let sql = "CREATE VIEW myschema.myview AS SELECT foo FROM bar";
|
let sql = "CREATE VIEW myschema.myview AS SELECT foo FROM bar";
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue