From 975106b20772d69f54c237bdc5a9ddd45d0a8ad9 Mon Sep 17 00:00:00 2001 From: Nikhil Benesch Date: Tue, 28 May 2019 17:30:36 -0400 Subject: [PATCH] Support EXISTS subqueries Fix #5. --- src/sqlast/mod.rs | 4 ++++ src/sqlparser.rs | 9 +++++++++ tests/sqlparser_common.rs | 40 +++++++++++++++++++++++++++++++++++++++ 3 files changed, 53 insertions(+) diff --git a/src/sqlast/mod.rs b/src/sqlast/mod.rs index 9ce0a8d43..d076fea6a 100644 --- a/src/sqlast/mod.rs +++ b/src/sqlast/mod.rs @@ -125,6 +125,9 @@ pub enum ASTNode { results: Vec, else_result: Option>, }, + /// An exists expression `EXISTS(SELECT ...)`, used in expressions like + /// `WHERE EXISTS (SELECT ...)`. + SQLExists(Box), /// A parenthesized subquery `(SELECT ...)`, used in expression like /// `SELECT (subquery) AS x` or `WHERE (subquery) = x` SQLSubquery(Box), @@ -230,6 +233,7 @@ impl ToString for ASTNode { } s + " END" } + ASTNode::SQLExists(s) => format!("EXISTS ({})", s.to_string()), ASTNode::SQLSubquery(s) => format!("({})", s.to_string()), } } diff --git a/src/sqlparser.rs b/src/sqlparser.rs index 85121eef4..6ccc63b4a 100644 --- a/src/sqlparser.rs +++ b/src/sqlparser.rs @@ -191,6 +191,7 @@ impl Parser { } "CASE" => self.parse_case_expression(), "CAST" => self.parse_cast_expression(), + "EXISTS" => self.parse_exists_expression(), "NOT" => Ok(ASTNode::SQLUnary { operator: SQLOperator::Not, 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 { + 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 pub fn parse_infix(&mut self, expr: ASTNode, precedence: u8) -> Result { debug!("parsing infix"); diff --git a/tests/sqlparser_common.rs b/tests/sqlparser_common.rs index ab5ce4d2a..f03f8109b 100644 --- a/tests/sqlparser_common.rs +++ b/tests/sqlparser_common.rs @@ -1385,6 +1385,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] fn parse_create_view() { let sql = "CREATE VIEW myschema.myview AS SELECT foo FROM bar";