surrealpatch/lib/src/sql/parser.rs

176 lines
4 KiB
Rust
Raw Normal View History

2020-06-29 15:36:01 +00:00
use crate::err::Error;
2022-01-16 20:31:50 +00:00
use crate::sql::error::Error::ParserError;
use crate::sql::error::Error::ScriptError;
2020-06-29 15:36:01 +00:00
use crate::sql::query::{query, Query};
use crate::sql::value::{json as value, Value};
2020-06-29 15:36:01 +00:00
use nom::Err;
use std::str;
pub fn parse(input: &str) -> Result<Query, Error> {
2022-01-14 22:35:44 +00:00
match input.trim().len() {
0 => Err(Error::EmptyError),
_ => match query(input) {
Ok((_, query)) => Ok(query),
2022-01-16 20:31:50 +00:00
Err(Err::Error(e)) => match e {
ParserError(e) => match locate(input, e) {
(s, l, c) => Err(Error::ParseError {
line: l,
char: c,
sql: s.to_string(),
}),
},
ScriptError(e) => Err(Error::LanguageError {
message: e,
2022-01-15 09:51:57 +00:00
}),
},
2022-01-16 20:31:50 +00:00
Err(Err::Failure(e)) => match e {
ParserError(e) => match locate(input, e) {
(s, l, c) => Err(Error::ParseError {
line: l,
char: c,
sql: s.to_string(),
}),
},
ScriptError(e) => Err(Error::LanguageError {
message: e,
2022-01-15 09:51:57 +00:00
}),
},
2022-01-16 20:31:50 +00:00
_ => unreachable!(),
2022-01-14 22:35:44 +00:00
},
2020-06-29 15:36:01 +00:00
}
}
pub fn json(input: &str) -> Result<Value, Error> {
match input.trim().len() {
0 => Err(Error::EmptyError),
_ => match value(input) {
Ok((_, query)) => Ok(query),
Err(Err::Error(e)) => match e {
ParserError(e) => match locate(input, e) {
(s, l, c) => Err(Error::ParseError {
line: l,
char: c,
sql: s.to_string(),
}),
},
ScriptError(e) => Err(Error::LanguageError {
message: e,
}),
},
Err(Err::Failure(e)) => match e {
ParserError(e) => match locate(input, e) {
(s, l, c) => Err(Error::ParseError {
line: l,
char: c,
sql: s.to_string(),
}),
},
ScriptError(e) => Err(Error::LanguageError {
message: e,
}),
},
_ => unreachable!(),
},
}
}
2022-01-15 09:51:57 +00:00
fn truncate(s: &str, l: usize) -> &str {
match s.char_indices().nth(l) {
None => s,
Some((i, _)) => &s[..i],
}
}
fn locate<'a>(input: &str, tried: &'a str) -> (&'a str, usize, usize) {
let index = input.len() - tried.len();
let tried = truncate(&tried, 100);
let lines = input.split('\n').collect::<Vec<&str>>();
let lines = lines.iter().map(|l| l.len()).enumerate();
let (mut total, mut chars) = (0, 0);
for (line, size) in lines {
total += size + 1;
if index < total {
let line_num = line + 1;
let char_num = index - chars;
return (tried, line_num, char_num);
}
chars += size + 1;
}
return (tried, 0, 0);
}
2020-06-29 15:36:01 +00:00
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn no_ending() {
let sql = "SELECT * FROM test";
let res = parse(sql);
assert!(res.is_ok());
}
#[test]
fn parse_query_string() {
let sql = "SELECT * FROM test;";
let res = parse(sql);
assert!(res.is_ok());
}
#[test]
fn trim_query_string() {
let sql = " SELECT * FROM test ; ";
let res = parse(sql);
assert!(res.is_ok());
}
#[test]
fn parse_complex_rubbish() {
let sql = " SELECT * FROM test ; /* shouldbespace */ ;;; ";
let res = parse(sql);
assert!(res.is_ok());
}
#[test]
fn parse_complex_failure() {
let sql = " SELECT * FROM { }} ";
let res = parse(sql);
assert!(res.is_err());
}
#[test]
fn parser_try() {
let sql = "
SELECT
*,
tags[$].value,
3s as duration,
1.345 AS number,
test AS `some thing`,
'2012-04-23T18:25:43.511Z' AS utctime,
'2012-04-23T18:25:43.511-08:00' AS pacifictime,
2021-03-29 15:43:37 +00:00
{ key: (3 + 1 + 2), other: 9 * 7, 'some thing': { otherkey: 'text', } } AS object
2020-06-29 15:36:01 +00:00
FROM $param, test, temp, test:thingy, |test:10|, |test:1..10|
WHERE IF true THEN 'YAY' ELSE 'OOPS' END
AND (0.1341, 0.5719) INSIDE { type: 'Polygon', coordinates: [[[0.1341, 0.5719], [0.1341, 0.5719]]] }
2020-06-29 15:36:01 +00:00
AND (3 + 3 * 4)=6
AND 3 + 3 * 4 = 6
AND ages CONTAINS 18
AND if IS true
SPLIT test.things
VERSION '2019-01-01'
TIMEOUT 2w;
CREATE person SET name = 'Tobie', age += 18;
";
let res = parse(sql);
assert!(res.is_ok());
let tmp = res.unwrap();
2021-03-29 15:43:37 +00:00
let enc = msgpack::to_vec(&tmp).unwrap();
let dec = msgpack::from_slice::<Query>(&enc).unwrap();
2021-03-29 15:43:37 +00:00
assert_eq!(tmp, dec);
2020-06-29 15:36:01 +00:00
}
}