use crate::cnf::PROTECTED_PARAM_NAMES; use crate::ctx::Context; use crate::dbs::Options; use crate::dbs::Transaction; use crate::err::Error; use crate::sql::comment::{comment, mightbespace}; use crate::sql::common::colons; use crate::sql::error::IResult; use crate::sql::fmt::{is_pretty, pretty_indent, Fmt, Pretty}; 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::output::{output, OutputStatement}; use crate::sql::statements::relate::{relate, RelateStatement}; use crate::sql::statements::select::{select, SelectStatement}; use crate::sql::statements::set::{set, SetStatement}; use crate::sql::statements::update::{update, UpdateStatement}; use crate::sql::value::{value, Value}; use nom::branch::alt; use nom::character::complete::char; use nom::combinator::map; use nom::multi::many0; use nom::multi::separated_list1; use nom::sequence::delimited; use serde::{Deserialize, Serialize}; use std::cmp::Ordering; use std::fmt::{self, Display, Formatter, Write}; use std::ops::Deref; pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Block"; #[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)] #[serde(rename = "$surrealdb::private::sql::Block")] pub struct Block(pub Vec); impl Deref for Block { type Target = Vec; fn deref(&self) -> &Self::Target { &self.0 } } impl From for Block { fn from(v: Value) -> Self { Block(vec![Entry::Value(v)]) } } impl Block { pub(crate) fn writeable(&self) -> bool { self.iter().any(Entry::writeable) } pub(crate) async fn compute( &self, ctx: &Context<'_>, opt: &Options, txn: &Transaction, doc: Option<&Value>, ) -> Result { // Duplicate context let mut ctx = Context::new(ctx); // Loop over the statements for v in self.iter() { match v { Entry::Set(v) => { // Check if the variable is a protected variable let val = match PROTECTED_PARAM_NAMES.contains(&v.name.as_str()) { // The variable isn't protected and can be stored false => v.compute(&ctx, opt, txn, doc).await, // The user tried to set a protected variable true => { return Err(Error::InvalidParam { name: v.name.to_owned(), }) } }?; // Set the parameter ctx.add_value(v.name.to_owned(), val); } Entry::Ifelse(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Select(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Create(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Update(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Delete(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Relate(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Insert(v) => { v.compute(&ctx, opt, txn, doc).await?; } Entry::Output(v) => { return v.compute(&ctx, opt, txn, doc).await; } Entry::Value(v) => { return v.compute(&ctx, opt, txn, doc).await; } } } // Return nothing Ok(Value::None) } } impl Display for Block { fn fmt(&self, f: &mut Formatter) -> fmt::Result { let mut f = Pretty::from(f); match (self.len(), self.first()) { (1, Some(Entry::Value(v))) => { write!(f, "{{ {v} }}") } (l, _) => { f.write_char('{')?; if l > 1 { f.write_char('\n')?; } else if !is_pretty() { f.write_char(' ')?; } let indent = pretty_indent(); if is_pretty() { write!( f, "{}", &Fmt::two_line_separated( self.0.iter().map(|args| Fmt::new(args, |v, f| write!(f, "{};", v))), ) )?; } else { write!( f, "{}", &Fmt::one_line_separated( self.0.iter().map(|args| Fmt::new(args, |v, f| write!(f, "{};", v))), ) )?; } drop(indent); if l > 1 { f.write_char('\n')?; } else if !is_pretty() { f.write_char(' ')?; } f.write_char('}') } } } } pub fn block(i: &str) -> IResult<&str, Block> { let (i, _) = char('{')(i)?; let (i, _) = mightbespace(i)?; let (i, v) = separated_list1(colons, entry)(i)?; let (i, _) = many0(alt((colons, comment)))(i)?; let (i, _) = mightbespace(i)?; let (i, _) = char('}')(i)?; Ok((i, Block(v))) } #[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize, Hash)] pub enum Entry { Value(Value), Set(SetStatement), Ifelse(IfelseStatement), Select(SelectStatement), Create(CreateStatement), Update(UpdateStatement), Delete(DeleteStatement), Relate(RelateStatement), Insert(InsertStatement), Output(OutputStatement), } impl PartialOrd for Entry { #[inline] fn partial_cmp(&self, _: &Self) -> Option { None } } impl Entry { pub(crate) fn writeable(&self) -> bool { match self { Self::Set(v) => v.writeable(), Self::Value(v) => v.writeable(), Self::Ifelse(v) => v.writeable(), Self::Select(v) => v.writeable(), Self::Create(v) => v.writeable(), Self::Update(v) => v.writeable(), Self::Delete(v) => v.writeable(), Self::Relate(v) => v.writeable(), Self::Insert(v) => v.writeable(), Self::Output(v) => v.writeable(), } } } impl Display for Entry { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { Self::Set(v) => write!(f, "{v}"), Self::Value(v) => Display::fmt(v, f), Self::Ifelse(v) => write!(f, "{v}"), Self::Select(v) => write!(f, "{v}"), Self::Create(v) => write!(f, "{v}"), Self::Update(v) => write!(f, "{v}"), Self::Delete(v) => write!(f, "{v}"), Self::Relate(v) => write!(f, "{v}"), Self::Insert(v) => write!(f, "{v}"), Self::Output(v) => write!(f, "{v}"), } } } pub fn entry(i: &str) -> IResult<&str, Entry> { delimited( mightbespace, alt(( map(set, Entry::Set), map(output, Entry::Output), map(ifelse, Entry::Ifelse), map(select, Entry::Select), map(create, Entry::Create), map(update, Entry::Update), map(relate, Entry::Relate), map(delete, Entry::Delete), map(insert, Entry::Insert), map(value, Entry::Value), )), mightbespace, )(i) } #[cfg(test)] mod tests { use super::*; #[test] fn block_value() { let sql = "{ 80 }"; let res = block(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!(sql, format!("{}", out)) } #[test] fn block_ifelse() { let sql = "{ RETURN IF true THEN 50 ELSE 40 END; }"; let res = block(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!(sql, format!("{}", out)) } #[test] fn block_multiple() { let sql = r#"{ LET $person = (SELECT * FROM person WHERE first = $first AND last = $last AND birthday = $birthday); RETURN IF $person[0].id THEN $person[0] ELSE (CREATE person SET first = $first, last = $last, birthday = $birthday) END; }"#; let res = block(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!(sql, format!("{:#}", out)) } }