2023-09-08 11:28:36 +00:00
|
|
|
use crate::sql::error::{IResult, ParseError};
|
2023-01-19 09:53:33 +00:00
|
|
|
use crate::sql::fmt::Pretty;
|
2020-06-29 15:36:01 +00:00
|
|
|
use crate::sql::statement::{statements, Statement, Statements};
|
2023-08-18 22:51:56 +00:00
|
|
|
use crate::sql::Value;
|
2022-05-21 00:35:59 +00:00
|
|
|
use derive::Store;
|
2023-09-08 11:28:36 +00:00
|
|
|
use nom::Err;
|
2023-08-17 18:03:46 +00:00
|
|
|
use revision::revisioned;
|
2020-06-29 15:36:01 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2023-01-19 09:53:33 +00:00
|
|
|
use std::fmt::Write;
|
2022-10-04 21:51:18 +00:00
|
|
|
use std::fmt::{self, Display, Formatter};
|
2022-05-04 21:34:28 +00:00
|
|
|
use std::ops::Deref;
|
2020-06-29 15:36:01 +00:00
|
|
|
use std::str;
|
|
|
|
|
2023-08-18 22:51:56 +00:00
|
|
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Query";
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Store, Hash)]
|
2023-08-17 18:03:46 +00:00
|
|
|
#[revisioned(revision = 1)]
|
2023-08-18 22:51:56 +00:00
|
|
|
#[serde(rename = "$surrealdb::private::sql::Query")]
|
2022-05-04 21:34:28 +00:00
|
|
|
pub struct Query(pub Statements);
|
2020-06-29 15:36:01 +00:00
|
|
|
|
2022-05-04 21:34:28 +00:00
|
|
|
impl Deref for Query {
|
|
|
|
type Target = Vec<Statement>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0 .0
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-20 12:30:47 +00:00
|
|
|
impl IntoIterator for Query {
|
|
|
|
type Item = Statement;
|
|
|
|
type IntoIter = std::vec::IntoIter<Self::Item>;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
|
|
self.0.into_iter()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-18 22:51:56 +00:00
|
|
|
impl From<Query> for Value {
|
|
|
|
fn from(q: Query) -> Self {
|
|
|
|
Value::Query(q)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-04 21:51:18 +00:00
|
|
|
impl Display for Query {
|
|
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
2023-01-19 09:53:33 +00:00
|
|
|
write!(Pretty::from(f), "{}", &self.0)
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn query(i: &str) -> IResult<&str, Query> {
|
2023-09-08 11:28:36 +00:00
|
|
|
let (i, v) = statements(i)?;
|
|
|
|
if !i.is_empty() {
|
|
|
|
return Err(Err::Failure(ParseError::ExplainedExpected {
|
|
|
|
tried: i,
|
|
|
|
expected: "query to end",
|
|
|
|
explained: "perhaps missing a semicolon on the previous statement?",
|
|
|
|
}));
|
|
|
|
}
|
2022-05-04 21:34:28 +00:00
|
|
|
Ok((i, Query(v)))
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn single_query() {
|
|
|
|
let sql = "CREATE test";
|
|
|
|
let res = query(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("CREATE test;", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn multiple_query() {
|
|
|
|
let sql = "CREATE test; CREATE temp;";
|
|
|
|
let res = query(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn multiple_query_semicolons() {
|
|
|
|
let sql = "CREATE test;;;CREATE temp;;;";
|
|
|
|
let res = query(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn multiple_query_semicolons_comments() {
|
|
|
|
let sql = "CREATE test;;;CREATE temp;;;/* some comment */";
|
|
|
|
let res = query(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn multiple_query_semicolons_multi_comments() {
|
|
|
|
let sql = "CREATE test;;;CREATE temp;;;/* some comment */;;;/* other comment */";
|
|
|
|
let res = query(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
|
|
|
|
}
|
|
|
|
}
|