surrealpatch/src/sql/subquery.rs

258 lines
6.7 KiB
Rust
Raw Normal View History

2021-03-29 15:43:37 +00:00
use crate::ctx::Context;
use crate::dbs::Executor;
use crate::dbs::Options;
use crate::dbs::Runtime;
2021-03-29 15:43:37 +00:00
use crate::err::Error;
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
2020-06-29 15:36:01 +00:00
use crate::sql::statements::create::{create, CreateStatement};
use crate::sql::statements::delete::{delete, DeleteStatement};
use crate::sql::statements::ifelse::{ifelse, IfelseStatement};
use crate::sql::statements::insert::{insert, InsertStatement};
use crate::sql::statements::relate::{relate, RelateStatement};
use crate::sql::statements::select::{select, SelectStatement};
use crate::sql::statements::update::{update, UpdateStatement};
use crate::sql::value::{value, Value};
2020-06-29 15:36:01 +00:00
use nom::branch::alt;
use nom::bytes::complete::tag;
use nom::combinator::map;
use serde::{Deserialize, Serialize};
2021-03-29 15:43:37 +00:00
use std::cmp::Ordering;
2020-06-29 15:36:01 +00:00
use std::fmt;
2021-03-29 15:43:37 +00:00
#[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
2020-06-29 15:36:01 +00:00
pub enum Subquery {
Value(Value),
2020-06-29 15:36:01 +00:00
Select(SelectStatement),
Create(CreateStatement),
Update(UpdateStatement),
Delete(DeleteStatement),
Relate(RelateStatement),
Insert(InsertStatement),
Ifelse(IfelseStatement),
}
2021-03-29 15:43:37 +00:00
impl PartialOrd for Subquery {
#[inline]
fn partial_cmp(&self, _: &Self) -> Option<Ordering> {
unreachable!()
}
}
2022-01-14 08:12:56 +00:00
impl Subquery {
pub async fn compute(
2021-03-29 15:43:37 +00:00
&self,
ctx: &Runtime,
opt: &Options,
exe: &Executor<'_>,
doc: Option<&Value>,
) -> Result<Value, Error> {
2021-03-29 15:43:37 +00:00
match self {
2022-01-14 08:12:56 +00:00
Subquery::Value(ref v) => v.compute(ctx, opt, exe, doc).await,
Subquery::Ifelse(ref v) => v.compute(ctx, opt, exe, doc).await,
2021-03-29 15:43:37 +00:00
Subquery::Select(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
let res = v.compute(&ctx, &opt, exe, doc).await?;
// Process result
match v.limit() {
1 => match v.expr.single() {
Some(v) => res.first(&ctx, &opt, exe).await?.get(&ctx, &opt, exe, &v).await,
None => res.first(&ctx, &opt, exe).await,
},
_ => match v.expr.single() {
Some(v) => res.get(&ctx, &opt, exe, &v).await,
None => res.ok(),
},
}
2021-03-29 15:43:37 +00:00
}
Subquery::Create(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
match v.compute(&ctx, &opt, exe, doc).await? {
Value::Array(mut v) => match v.len() {
1 => Ok(v.value.remove(0)),
_ => Ok(v.into()),
},
v => Ok(v),
}
2021-03-29 15:43:37 +00:00
}
Subquery::Update(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
match v.compute(&ctx, &opt, exe, doc).await? {
Value::Array(mut v) => match v.len() {
1 => Ok(v.value.remove(0)),
_ => Ok(v.into()),
},
v => Ok(v),
}
2021-03-29 15:43:37 +00:00
}
Subquery::Delete(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
match v.compute(&ctx, &opt, exe, doc).await? {
Value::Array(mut v) => match v.len() {
1 => Ok(v.value.remove(0)),
_ => Ok(v.into()),
},
v => Ok(v),
}
2021-03-29 15:43:37 +00:00
}
Subquery::Relate(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
match v.compute(&ctx, &opt, exe, doc).await? {
Value::Array(mut v) => match v.len() {
1 => Ok(v.value.remove(0)),
_ => Ok(v.into()),
},
v => Ok(v),
}
2021-03-29 15:43:37 +00:00
}
Subquery::Insert(ref v) => {
// Duplicate options
let opt = opt.dive()?;
// Duplicate context
2021-03-29 15:43:37 +00:00
let mut ctx = Context::new(ctx);
// Add parent document
2021-03-29 15:43:37 +00:00
if doc.is_some() {
let doc = doc.unwrap().clone();
ctx.add_value(String::from("parent"), doc);
2021-03-29 15:43:37 +00:00
}
// Prepare context
2021-03-29 15:43:37 +00:00
let ctx = ctx.freeze();
// Process subquery
2022-01-14 08:12:56 +00:00
match v.compute(&ctx, &opt, exe, doc).await? {
Value::Array(mut v) => match v.len() {
1 => Ok(v.value.remove(0)),
_ => Ok(v.into()),
},
v => Ok(v),
2021-03-29 15:43:37 +00:00
}
}
}
}
}
2022-01-14 08:12:56 +00:00
impl fmt::Display for Subquery {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Subquery::Value(v) => write!(f, "({})", v),
Subquery::Select(v) => write!(f, "({})", v),
Subquery::Create(v) => write!(f, "({})", v),
Subquery::Update(v) => write!(f, "({})", v),
Subquery::Delete(v) => write!(f, "({})", v),
Subquery::Relate(v) => write!(f, "({})", v),
Subquery::Insert(v) => write!(f, "({})", v),
Subquery::Ifelse(v) => write!(f, "{}", v),
}
}
}
2020-06-29 15:36:01 +00:00
pub fn subquery(i: &str) -> IResult<&str, Subquery> {
alt((subquery_ifelse, subquery_others))(i)
}
fn subquery_ifelse(i: &str) -> IResult<&str, Subquery> {
let (i, v) = map(ifelse, |v| Subquery::Ifelse(v))(i)?;
Ok((i, v))
}
fn subquery_others(i: &str) -> IResult<&str, Subquery> {
let (i, _) = tag("(")(i)?;
let (i, v) = alt((
map(select, |v| Subquery::Select(v)),
map(create, |v| Subquery::Create(v)),
map(update, |v| Subquery::Update(v)),
map(delete, |v| Subquery::Delete(v)),
map(relate, |v| Subquery::Relate(v)),
map(insert, |v| Subquery::Insert(v)),
map(value, |v| Subquery::Value(v)),
2020-06-29 15:36:01 +00:00
))(i)?;
let (i, _) = tag(")")(i)?;
Ok((i, v))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn subquery_expression_statement() {
let sql = "(1 + 2 + 3)";
let res = subquery(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("(1 + 2 + 3)", format!("{}", out))
}
#[test]
fn subquery_ifelse_statement() {
let sql = "IF true THEN false END";
let res = subquery(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("IF true THEN false END", format!("{}", out))
}
#[test]
fn subquery_select_statement() {
let sql = "(SELECT * FROM test)";
let res = subquery(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("(SELECT * FROM test)", format!("{}", out))
}
}