2021-03-29 15:43:37 +00:00
|
|
|
use crate::ctx::Context;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::dbs::Options;
|
2021-03-31 12:10:13 +00:00
|
|
|
use crate::dbs::Runtime;
|
2022-02-15 01:00:30 +00:00
|
|
|
use crate::dbs::Transaction;
|
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};
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::value::{value, Value};
|
2020-06-29 15:36:01 +00:00
|
|
|
use nom::branch::alt;
|
2022-03-16 23:52:25 +00:00
|
|
|
use nom::character::complete::char;
|
2020-06-29 15:36:01 +00:00
|
|
|
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 {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value(Value),
|
2020-06-29 15:36:01 +00:00
|
|
|
Ifelse(IfelseStatement),
|
2022-05-13 20:46:56 +00:00
|
|
|
Select(SelectStatement),
|
|
|
|
Create(CreateStatement),
|
|
|
|
Update(UpdateStatement),
|
|
|
|
Delete(DeleteStatement),
|
|
|
|
Relate(RelateStatement),
|
|
|
|
Insert(InsertStatement),
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
2021-03-29 15:43:37 +00:00
|
|
|
impl PartialOrd for Subquery {
|
|
|
|
#[inline]
|
|
|
|
fn partial_cmp(&self, _: &Self) -> Option<Ordering> {
|
2022-03-23 11:56:39 +00:00
|
|
|
None
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 08:12:56 +00:00
|
|
|
impl Subquery {
|
2022-05-01 22:25:53 +00:00
|
|
|
pub(crate) async fn compute(
|
2021-03-29 15:43:37 +00:00
|
|
|
&self,
|
2021-03-31 12:10:13 +00:00
|
|
|
ctx: &Runtime,
|
2022-02-06 21:06:52 +00:00
|
|
|
opt: &Options,
|
2022-02-15 03:33:16 +00:00
|
|
|
txn: &Transaction,
|
2022-01-13 17:36:41 +00:00
|
|
|
doc: Option<&Value>,
|
|
|
|
) -> Result<Value, Error> {
|
2021-03-29 15:43:37 +00:00
|
|
|
match self {
|
2022-02-15 01:00:30 +00:00
|
|
|
Subquery::Value(ref v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Subquery::Ifelse(ref v) => v.compute(ctx, opt, txn, doc).await,
|
2021-03-29 15:43:37 +00:00
|
|
|
Subquery::Select(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
let res = v.compute(&ctx, &opt, txn, doc).await?;
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process result
|
|
|
|
match v.limit() {
|
|
|
|
1 => match v.expr.single() {
|
2022-03-25 21:14:48 +00:00
|
|
|
Some(v) => res.first().get(&ctx, &opt, txn, &v).await,
|
|
|
|
None => res.first().ok(),
|
2022-01-13 17:36:41 +00:00
|
|
|
},
|
|
|
|
_ => match v.expr.single() {
|
2022-02-15 01:00:30 +00:00
|
|
|
Some(v) => res.get(&ctx, &opt, txn, &v).await,
|
2022-01-13 17:36:41 +00:00
|
|
|
None => res.ok(),
|
|
|
|
},
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
Subquery::Create(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
match v.compute(&ctx, &opt, txn, doc).await? {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-05-04 16:14:40 +00:00
|
|
|
1 => Ok(v.remove(0)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Ok(v.into()),
|
|
|
|
},
|
|
|
|
v => Ok(v),
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
Subquery::Update(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
match v.compute(&ctx, &opt, txn, doc).await? {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-05-04 16:14:40 +00:00
|
|
|
1 => Ok(v.remove(0)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Ok(v.into()),
|
|
|
|
},
|
|
|
|
v => Ok(v),
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
Subquery::Delete(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
match v.compute(&ctx, &opt, txn, doc).await? {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-05-04 16:14:40 +00:00
|
|
|
1 => Ok(v.remove(0)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Ok(v.into()),
|
|
|
|
},
|
|
|
|
v => Ok(v),
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
Subquery::Relate(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
match v.compute(&ctx, &opt, txn, doc).await? {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-05-04 16:14:40 +00:00
|
|
|
1 => Ok(v.remove(0)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Ok(v.into()),
|
|
|
|
},
|
|
|
|
v => Ok(v),
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
Subquery::Insert(ref v) => {
|
2022-01-13 17:36:41 +00:00
|
|
|
// Duplicate options
|
|
|
|
let opt = opt.dive()?;
|
|
|
|
// Duplicate context
|
2021-03-29 15:43:37 +00:00
|
|
|
let mut ctx = Context::new(ctx);
|
2022-01-13 17:36:41 +00:00
|
|
|
// Add parent document
|
2021-03-29 15:43:37 +00:00
|
|
|
if doc.is_some() {
|
|
|
|
let doc = doc.unwrap().clone();
|
2022-04-04 22:23:31 +00:00
|
|
|
ctx.add_value("parent".into(), doc);
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
// Prepare context
|
2021-03-29 15:43:37 +00:00
|
|
|
let ctx = ctx.freeze();
|
2022-01-13 17:36:41 +00:00
|
|
|
// Process subquery
|
2022-05-13 20:46:56 +00:00
|
|
|
match v.compute(&ctx, &opt, txn, doc).await? {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-05-04 16:14:40 +00:00
|
|
|
1 => Ok(v.remove(0)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => 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> {
|
2022-03-04 16:01:32 +00:00
|
|
|
let (i, v) = map(ifelse, Subquery::Ifelse)(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
Ok((i, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn subquery_others(i: &str) -> IResult<&str, Subquery> {
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('(')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, v) = alt((
|
2022-05-13 20:46:56 +00:00
|
|
|
map(select, Subquery::Select),
|
|
|
|
map(create, Subquery::Create),
|
|
|
|
map(update, Subquery::Update),
|
|
|
|
map(delete, Subquery::Delete),
|
|
|
|
map(relate, Subquery::Relate),
|
|
|
|
map(insert, Subquery::Insert),
|
2022-03-04 16:01:32 +00:00
|
|
|
map(value, Subquery::Value),
|
2020-06-29 15:36:01 +00:00
|
|
|
))(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char(')')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
Ok((i, v))
|
|
|
|
}
|
2021-05-22 18:18:35 +00:00
|
|
|
|
|
|
|
#[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))
|
|
|
|
}
|
|
|
|
}
|