Skip to content

Commit 975106b

Browse files
committed
Support EXISTS subqueries
Fix #5.
1 parent 2308c1c commit 975106b

File tree

3 files changed

+53
-0
lines changed

3 files changed

+53
-0
lines changed

src/sqlast/mod.rs

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -125,6 +125,9 @@ pub enum ASTNode {
125125
results: Vec<ASTNode>,
126126
else_result: Option<Box<ASTNode>>,
127127
},
128+
/// An exists expression `EXISTS(SELECT ...)`, used in expressions like
129+
/// `WHERE EXISTS (SELECT ...)`.
130+
SQLExists(Box<SQLQuery>),
128131
/// A parenthesized subquery `(SELECT ...)`, used in expression like
129132
/// `SELECT (subquery) AS x` or `WHERE (subquery) = x`
130133
SQLSubquery(Box<SQLQuery>),
@@ -230,6 +233,7 @@ impl ToString for ASTNode {
230233
}
231234
s + " END"
232235
}
236+
ASTNode::SQLExists(s) => format!("EXISTS ({})", s.to_string()),
233237
ASTNode::SQLSubquery(s) => format!("({})", s.to_string()),
234238
}
235239
}

src/sqlparser.rs

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,7 @@ impl Parser {
191191
}
192192
"CASE" => self.parse_case_expression(),
193193
"CAST" => self.parse_cast_expression(),
194+
"EXISTS" => self.parse_exists_expression(),
194195
"NOT" => Ok(ASTNode::SQLUnary {
195196
operator: SQLOperator::Not,
196197
expr: Box::new(self.parse_subexpr(Self::UNARY_NOT_PREC)?),
@@ -404,6 +405,14 @@ impl Parser {
404405
})
405406
}
406407

408+
/// Parse a SQL EXISTS expression e.g. `WHERE EXISTS(SELECT ...)`.
409+
pub fn parse_exists_expression(&mut self) -> Result<ASTNode, ParserError> {
410+
self.expect_token(&Token::LParen)?;
411+
let exists_node = ASTNode::SQLExists(Box::new(self.parse_query()?));
412+
self.expect_token(&Token::RParen)?;
413+
Ok(exists_node)
414+
}
415+
407416
/// Parse an operator following an expression
408417
pub fn parse_infix(&mut self, expr: ASTNode, precedence: u8) -> Result<ASTNode, ParserError> {
409418
debug!("parsing infix");

tests/sqlparser_common.rs

Lines changed: 40 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1385,6 +1385,46 @@ fn parse_scalar_subqueries() {
13851385
});
13861386
}
13871387

1388+
#[test]
1389+
fn parse_exists_subquery() {
1390+
let expected_inner = verified_query("SELECT 1");
1391+
let sql = "SELECT * FROM t WHERE EXISTS (SELECT 1)";
1392+
let select = verified_only_select(sql);
1393+
assert_eq!(
1394+
ASTNode::SQLExists(Box::new(expected_inner.clone())),
1395+
select.selection.unwrap(),
1396+
);
1397+
1398+
let sql = "SELECT * FROM t WHERE NOT EXISTS (SELECT 1)";
1399+
let select = verified_only_select(sql);
1400+
assert_eq!(
1401+
ASTNode::SQLUnary {
1402+
operator: SQLOperator::Not,
1403+
expr: Box::new(ASTNode::SQLExists(Box::new(expected_inner))),
1404+
},
1405+
select.selection.unwrap(),
1406+
);
1407+
1408+
verified_stmt("SELECT * FROM t WHERE EXISTS (WITH u AS (SELECT 1) SELECT * FROM u)");
1409+
verified_stmt("SELECT EXISTS (SELECT 1)");
1410+
1411+
let res = parse_sql_statements("SELECT EXISTS (");
1412+
assert_eq!(
1413+
ParserError::ParserError(
1414+
"Expected SELECT or a subquery in the query body, found: EOF".to_string()
1415+
),
1416+
res.unwrap_err(),
1417+
);
1418+
1419+
let res = parse_sql_statements("SELECT EXISTS (NULL)");
1420+
assert_eq!(
1421+
ParserError::ParserError(
1422+
"Expected SELECT or a subquery in the query body, found: NULL".to_string()
1423+
),
1424+
res.unwrap_err(),
1425+
);
1426+
}
1427+
13881428
#[test]
13891429
fn parse_create_view() {
13901430
let sql = "CREATE VIEW myschema.myview AS SELECT foo FROM bar";

0 commit comments

Comments
 (0)