mirror of
https://github.com/apache/datafusion-sqlparser-rs.git
synced 2025-07-07 17:04:59 +00:00
Refactoring
This commit is contained in:
parent
375671e208
commit
a86bd30515
6 changed files with 101 additions and 338 deletions
|
@ -1,115 +0,0 @@
|
|||
use std::sync::{Arc, Mutex};
|
||||
|
||||
extern crate datafusion_sql;
|
||||
|
||||
use datafusion_sql::ansi::tokenizer::ANSISQLTokenizer;
|
||||
use datafusion_sql::ansi::parser::ANSISQLParser;
|
||||
use datafusion_sql::tokenizer::*;
|
||||
use datafusion_sql::parser::*;
|
||||
|
||||
/// This example demonstrates building a custom ACME parser that extends the generic parser
|
||||
/// by adding support for a factorial expression `!! expr`.
|
||||
|
||||
/// Custom SQLToken
|
||||
#[derive(Debug,PartialEq)]
|
||||
enum AcmeToken {
|
||||
/// Factorial token `!!`
|
||||
Factorial
|
||||
}
|
||||
|
||||
/// Custom SQLExpr
|
||||
#[derive(Debug)]
|
||||
enum AcmeExpr {
|
||||
/// Factorial expression
|
||||
Factorial(Box<SQLExpr<AcmeExpr>>)
|
||||
}
|
||||
|
||||
struct AcmeTokenizer {
|
||||
ansi_tokenizer: Arc<Mutex<SQLTokenizer<AcmeToken>>>
|
||||
}
|
||||
|
||||
/// The ACME tokenizer looks for the factorial operator `!!` but delegates everything else
|
||||
impl SQLTokenizer<AcmeToken> for AcmeTokenizer {
|
||||
|
||||
fn precedence(&self, _token: &SQLToken<AcmeToken>) -> usize {
|
||||
unimplemented!()
|
||||
}
|
||||
|
||||
fn next_token(&mut self, chars: &mut CharSeq) -> Result<Option<SQLToken<AcmeToken>>, TokenizerError> {
|
||||
let mut ansi = self.ansi_tokenizer.lock().unwrap();
|
||||
match chars.peek() {
|
||||
Some(&ch) => match ch {
|
||||
'!' => {
|
||||
chars.mark();
|
||||
chars.next(); // consume the first `!`
|
||||
match chars.peek() {
|
||||
Some(&ch) => match ch {
|
||||
'!' => {
|
||||
chars.next(); // consume the second `!`
|
||||
Ok(Some(SQLToken::Custom(AcmeToken::Factorial)))
|
||||
},
|
||||
_ => {
|
||||
chars.reset();
|
||||
ansi.next_token(chars)
|
||||
}
|
||||
},
|
||||
None => {
|
||||
chars.reset();
|
||||
ansi.next_token(chars)
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => ansi.next_token(chars)
|
||||
}
|
||||
_ => ansi.next_token(chars)
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
struct AcmeParser {
|
||||
tokenizer: Arc<Mutex<SQLTokenizer<AcmeToken>>>
|
||||
}
|
||||
|
||||
impl AcmeParser {
|
||||
|
||||
pub fn new(tokenizer: Arc<Mutex<SQLTokenizer<AcmeToken>>>) -> Self {
|
||||
AcmeParser { tokenizer: tokenizer.clone() }
|
||||
}
|
||||
|
||||
}
|
||||
impl SQLParser<AcmeToken, AcmeExpr> for AcmeParser {
|
||||
|
||||
fn parse_prefix(&mut self, chars: &mut CharSeq) -> Result<Option<Box<SQLExpr<AcmeExpr>>>, ParserError<AcmeToken>> {
|
||||
Ok(None)
|
||||
}
|
||||
|
||||
fn parse_infix(&mut self, chars: &mut CharSeq, left: &SQLExpr<AcmeExpr>, precedence: usize) -> Result<Option<Box<SQLExpr<AcmeExpr>>>, ParserError<AcmeToken>> {
|
||||
Ok(None)
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
|
||||
let sql = "1 + !! 5 * 2";
|
||||
|
||||
// ANSI SQL tokenizer
|
||||
let ansi_tokenizer = Arc::new(Mutex::new(ANSISQLTokenizer { }));
|
||||
|
||||
// Custom ACME tokenizer
|
||||
let mut acme_tokenizer = Arc::new(Mutex::new(AcmeTokenizer {
|
||||
ansi_tokenizer: ansi_tokenizer.clone()
|
||||
}));
|
||||
|
||||
// Create parsers
|
||||
let ansi_parser = Arc::new(Mutex::new(ANSISQLParser::new(acme_tokenizer.clone())));
|
||||
let acme_parser = Arc::new(Mutex::new(AcmeParser::new(acme_tokenizer.clone())));
|
||||
|
||||
let mut pratt_parser = PrattParser {
|
||||
chars: CharSeq::new(sql),
|
||||
parsers: vec![acme_parser, ansi_parser]
|
||||
};
|
||||
|
||||
let expr = pratt_parser.parse_expr().unwrap();
|
||||
println!("{:?}", expr);
|
||||
}
|
20
examples/parse_sql.rs
Normal file
20
examples/parse_sql.rs
Normal file
|
@ -0,0 +1,20 @@
|
|||
use std::sync::{Arc, Mutex};
|
||||
|
||||
extern crate datafusion_sql;
|
||||
|
||||
use datafusion_sql::ansi::tokenizer::ANSISQLTokenizer;
|
||||
use datafusion_sql::ansi::parser::ANSISQLParser;
|
||||
use datafusion_sql::tokenizer::*;
|
||||
use datafusion_sql::parser::*;
|
||||
|
||||
|
||||
fn main() {
|
||||
|
||||
let sql = "SELECT 1 + 1";
|
||||
|
||||
// Create parsers
|
||||
match ANSISQLParser::parse(sql).unwrap() {
|
||||
Some(ast) => println!("{:?}", ast),
|
||||
_ => {}
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue