surrealpatch/lib/src/sql/statement.rs

313 lines
10 KiB
Rust
Raw Normal View History

use crate::ctx::Context;
use crate::dbs::{Options, Transaction};
use crate::doc::CursorDoc;
2021-03-29 15:43:37 +00:00
use crate::err::Error;
2020-06-29 15:36:01 +00:00
use crate::sql::comment::{comment, mightbespace};
use crate::sql::common::colons;
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
use crate::sql::fmt::Fmt;
use crate::sql::fmt::Pretty;
use crate::sql::statements::analyze::{analyze, AnalyzeStatement};
2020-06-29 15:36:01 +00:00
use crate::sql::statements::begin::{begin, BeginStatement};
use crate::sql::statements::cancel::{cancel, CancelStatement};
use crate::sql::statements::commit::{commit, CommitStatement};
use crate::sql::statements::create::{create, CreateStatement};
use crate::sql::statements::define::{define, DefineStatement};
use crate::sql::statements::delete::{delete, DeleteStatement};
use crate::sql::statements::foreach::{foreach, ForeachStatement};
2020-06-29 15:36:01 +00:00
use crate::sql::statements::ifelse::{ifelse, IfelseStatement};
use crate::sql::statements::info::{info, InfoStatement};
use crate::sql::statements::insert::{insert, InsertStatement};
use crate::sql::statements::kill::{kill, KillStatement};
use crate::sql::statements::live::{live, LiveStatement};
use crate::sql::statements::option::{option, OptionStatement};
use crate::sql::statements::output::{output, OutputStatement};
use crate::sql::statements::r#break::{r#break, BreakStatement};
use crate::sql::statements::r#continue::{r#continue, ContinueStatement};
use crate::sql::statements::r#use::{r#use, UseStatement};
2020-06-29 15:36:01 +00:00
use crate::sql::statements::relate::{relate, RelateStatement};
use crate::sql::statements::remove::{remove, RemoveStatement};
use crate::sql::statements::select::{select, SelectStatement};
use crate::sql::statements::set::{set, SetStatement};
use crate::sql::statements::show::{show, ShowStatement};
use crate::sql::statements::sleep::{sleep, SleepStatement};
use crate::sql::statements::throw::{throw, ThrowStatement};
2020-06-29 15:36:01 +00:00
use crate::sql::statements::update::{update, UpdateStatement};
use crate::sql::value::Value;
2023-03-25 19:42:45 +00:00
use derive::Store;
2020-06-29 15:36:01 +00:00
use nom::branch::alt;
use nom::combinator::map;
use nom::multi::many0;
2021-03-29 15:43:37 +00:00
use nom::multi::separated_list1;
2020-06-29 15:36:01 +00:00
use nom::sequence::delimited;
use revision::revisioned;
2020-06-29 15:36:01 +00:00
use serde::{Deserialize, Serialize};
use std::fmt::{self, Display, Formatter, Write};
use std::ops::Deref;
use std::time::Duration;
2020-06-29 15:36:01 +00:00
2023-08-18 22:51:56 +00:00
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Store, Hash)]
#[revisioned(revision = 1)]
2020-06-29 15:36:01 +00:00
pub struct Statements(pub Vec<Statement>);
impl Deref for Statements {
type Target = Vec<Statement>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl IntoIterator for Statements {
type Item = Statement;
type IntoIter = std::vec::IntoIter<Self::Item>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
impl Display for Statements {
2020-06-29 15:36:01 +00:00
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
Display::fmt(
&Fmt::one_line_separated(self.0.iter().map(|v| Fmt::new(v, |v, f| write!(f, "{v};")))),
f,
)
2020-06-29 15:36:01 +00:00
}
}
pub fn statements(i: &str) -> IResult<&str, Statements> {
2021-03-29 15:43:37 +00:00
let (i, v) = separated_list1(colons, statement)(i)?;
2020-06-29 15:36:01 +00:00
let (i, _) = many0(alt((colons, comment)))(i)?;
Ok((i, Statements(v)))
}
2023-08-18 22:51:56 +00:00
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Store, Hash)]
#[revisioned(revision = 1)]
2020-06-29 15:36:01 +00:00
pub enum Statement {
Analyze(AnalyzeStatement),
Begin(BeginStatement),
Break(BreakStatement),
Continue(ContinueStatement),
Cancel(CancelStatement),
Commit(CommitStatement),
Create(CreateStatement),
Define(DefineStatement),
Delete(DeleteStatement),
Foreach(ForeachStatement),
Ifelse(IfelseStatement),
Info(InfoStatement),
Insert(InsertStatement),
Kill(KillStatement),
Live(LiveStatement),
Option(OptionStatement),
Output(OutputStatement),
Relate(RelateStatement),
Remove(RemoveStatement),
Select(SelectStatement),
Set(SetStatement),
Show(ShowStatement),
Sleep(SleepStatement),
Update(UpdateStatement),
Throw(ThrowStatement),
Use(UseStatement),
2020-06-29 15:36:01 +00:00
}
impl Statement {
/// Get the statement timeout duration, if any
pub fn timeout(&self) -> Option<Duration> {
match self {
Self::Create(v) => v.timeout.as_ref().map(|v| *v.0),
Self::Delete(v) => v.timeout.as_ref().map(|v| *v.0),
Self::Insert(v) => v.timeout.as_ref().map(|v| *v.0),
Self::Relate(v) => v.timeout.as_ref().map(|v| *v.0),
Self::Select(v) => v.timeout.as_ref().map(|v| *v.0),
Self::Update(v) => v.timeout.as_ref().map(|v| *v.0),
_ => None,
}
}
/// Check if we require a writeable transaction
pub(crate) fn writeable(&self) -> bool {
match self {
Self::Analyze(_) => false,
Self::Break(_) => false,
Self::Continue(_) => false,
Self::Create(v) => v.writeable(),
Self::Define(_) => true,
Self::Delete(v) => v.writeable(),
Self::Foreach(v) => v.writeable(),
Self::Ifelse(v) => v.writeable(),
Self::Info(_) => false,
Self::Insert(v) => v.writeable(),
Self::Kill(_) => true,
Self::Live(_) => true,
Self::Output(v) => v.writeable(),
Self::Option(_) => false,
Self::Relate(v) => v.writeable(),
Self::Remove(_) => true,
Self::Select(v) => v.writeable(),
Self::Set(v) => v.writeable(),
Self::Show(_) => false,
Self::Sleep(_) => false,
Self::Throw(_) => false,
Self::Update(v) => v.writeable(),
Self::Use(_) => false,
_ => unreachable!(),
}
}
/// Process this type returning a computed simple Value
pub(crate) async fn compute(
&self,
ctx: &Context<'_>,
opt: &Options,
txn: &Transaction,
doc: Option<&CursorDoc<'_>>,
) -> Result<Value, Error> {
match self {
Self::Analyze(v) => v.compute(ctx, opt, txn, doc).await,
Self::Break(v) => v.compute(ctx, opt, txn, doc).await,
Self::Continue(v) => v.compute(ctx, opt, txn, doc).await,
Self::Create(v) => v.compute(ctx, opt, txn, doc).await,
Self::Delete(v) => v.compute(ctx, opt, txn, doc).await,
Self::Define(v) => v.compute(ctx, opt, txn, doc).await,
Self::Foreach(v) => v.compute(ctx, opt, txn, doc).await,
Self::Ifelse(v) => v.compute(ctx, opt, txn, doc).await,
Self::Info(v) => v.compute(ctx, opt, txn, doc).await,
Self::Insert(v) => v.compute(ctx, opt, txn, doc).await,
Self::Kill(v) => v.compute(ctx, opt, txn, doc).await,
Self::Live(v) => v.compute(ctx, opt, txn, doc).await,
Self::Output(v) => v.compute(ctx, opt, txn, doc).await,
Self::Relate(v) => v.compute(ctx, opt, txn, doc).await,
Self::Remove(v) => v.compute(ctx, opt, txn, doc).await,
Self::Select(v) => v.compute(ctx, opt, txn, doc).await,
Self::Set(v) => v.compute(ctx, opt, txn, doc).await,
Self::Show(v) => v.compute(ctx, opt, txn, doc).await,
Self::Sleep(v) => v.compute(ctx, opt, txn, doc).await,
Self::Throw(v) => v.compute(ctx, opt, txn, doc).await,
Self::Update(v) => v.compute(ctx, opt, txn, doc).await,
2022-01-14 08:12:56 +00:00
_ => unreachable!(),
}
}
}
impl Display for Statement {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
match self {
Self::Analyze(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Begin(v) => write!(Pretty::from(f), "{v}"),
Self::Break(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Cancel(v) => write!(Pretty::from(f), "{v}"),
Self::Commit(v) => write!(Pretty::from(f), "{v}"),
Self::Continue(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Create(v) => write!(Pretty::from(f), "{v}"),
Self::Define(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Delete(v) => write!(Pretty::from(f), "{v}"),
Self::Foreach(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Insert(v) => write!(Pretty::from(f), "{v}"),
Self::Ifelse(v) => write!(Pretty::from(f), "{v}"),
Self::Info(v) => write!(Pretty::from(f), "{v}"),
Self::Kill(v) => write!(Pretty::from(f), "{v}"),
Self::Live(v) => write!(Pretty::from(f), "{v}"),
2023-02-03 11:47:07 +00:00
Self::Option(v) => write!(Pretty::from(f), "{v}"),
Self::Output(v) => write!(Pretty::from(f), "{v}"),
Self::Relate(v) => write!(Pretty::from(f), "{v}"),
Self::Remove(v) => write!(Pretty::from(f), "{v}"),
Self::Select(v) => write!(Pretty::from(f), "{v}"),
Self::Set(v) => write!(Pretty::from(f), "{v}"),
Self::Show(v) => write!(Pretty::from(f), "{v}"),
Self::Sleep(v) => write!(Pretty::from(f), "{v}"),
Self::Throw(v) => write!(Pretty::from(f), "{v}"),
Self::Update(v) => write!(Pretty::from(f), "{v}"),
Self::Use(v) => write!(Pretty::from(f), "{v}"),
2020-06-29 15:36:01 +00:00
}
}
}
pub fn statement(i: &str) -> IResult<&str, Statement> {
delimited(
mightbespace,
alt((
alt((
map(analyze, Statement::Analyze),
map(begin, Statement::Begin),
map(r#break, Statement::Break),
map(cancel, Statement::Cancel),
map(commit, Statement::Commit),
map(r#continue, Statement::Continue),
map(create, Statement::Create),
map(define, Statement::Define),
map(delete, Statement::Delete),
map(foreach, Statement::Foreach),
map(ifelse, Statement::Ifelse),
map(info, Statement::Info),
map(insert, Statement::Insert),
)),
alt((
map(kill, Statement::Kill),
map(live, Statement::Live),
map(option, Statement::Option),
map(output, Statement::Output),
map(relate, Statement::Relate),
map(remove, Statement::Remove),
map(select, Statement::Select),
map(set, Statement::Set),
map(show, Statement::Show),
map(sleep, Statement::Sleep),
map(throw, Statement::Throw),
map(update, Statement::Update),
map(r#use, Statement::Use),
)),
2020-06-29 15:36:01 +00:00
)),
mightbespace,
)(i)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn single_statement() {
let sql = "CREATE test";
let res = statement(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("CREATE test", format!("{}", out))
}
#[test]
fn multiple_statements() {
let sql = "CREATE test; CREATE temp;";
let res = statements(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
}
#[test]
fn multiple_statements_semicolons() {
let sql = "CREATE test;;;CREATE temp;;;";
let res = statements(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("CREATE test;\nCREATE temp;", format!("{}", out))
}
#[test]
fn show_table_changes() {
let sql = "SHOW CHANGES FOR TABLE test SINCE 123456";
let res = statement(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("SHOW CHANGES FOR TABLE test SINCE 123456", format!("{}", out))
}
#[test]
fn show_database_changes() {
let sql = "SHOW CHANGES FOR DATABASE SINCE 123456";
let res = statement(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("SHOW CHANGES FOR DATABASE SINCE 123456", format!("{}", out))
}
2020-06-29 15:36:01 +00:00
}