surrealpatch/lib/src/sql/statements/update.rs

170 lines
4 KiB
Rust
Raw Normal View History

use crate::ctx::Context;
2022-05-30 15:32:26 +00:00
use crate::dbs::Iterable;
2021-03-29 15:43:37 +00:00
use crate::dbs::Iterator;
use crate::dbs::Level;
use crate::dbs::Options;
use crate::dbs::Statement;
use crate::dbs::Transaction;
2021-03-29 15:43:37 +00:00
use crate::err::Error;
2020-06-29 15:36:01 +00:00
use crate::sql::comment::shouldbespace;
use crate::sql::cond::{cond, Cond};
use crate::sql::data::{data, Data};
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
2020-06-29 15:36:01 +00:00
use crate::sql::output::{output, Output};
use crate::sql::timeout::{timeout, Timeout};
use crate::sql::value::{whats, Value, Values};
use derive::Store;
2020-06-29 15:36:01 +00:00
use nom::bytes::complete::tag_no_case;
use nom::combinator::opt;
use nom::sequence::preceded;
use serde::{Deserialize, Serialize};
use std::fmt;
#[derive(Clone, Debug, Default, Eq, PartialEq, Serialize, Deserialize, Store)]
2020-06-29 15:36:01 +00:00
pub struct UpdateStatement {
pub what: Values,
2020-06-29 15:36:01 +00:00
pub data: Option<Data>,
pub cond: Option<Cond>,
pub output: Option<Output>,
pub timeout: Option<Timeout>,
pub parallel: bool,
2020-06-29 15:36:01 +00:00
}
2022-01-14 08:12:56 +00:00
impl UpdateStatement {
pub(crate) fn writeable(&self) -> bool {
true
}
pub(crate) async fn compute(
&self,
ctx: &Context<'_>,
opt: &Options,
txn: &Transaction,
doc: Option<&Value>,
) -> Result<Value, Error> {
// Allowed to run?
opt.check(Level::No)?;
2021-03-29 15:43:37 +00:00
// Create a new iterator
let mut i = Iterator::new();
// Ensure futures are stored
let opt = &opt.futures(false);
// Loop over the update targets
for w in self.what.0.iter() {
let v = w.compute(ctx, opt, txn, doc).await?;
match v {
2022-05-30 15:32:26 +00:00
Value::Table(v) => i.ingest(Iterable::Table(v)),
Value::Thing(v) => i.ingest(Iterable::Thing(v)),
Value::Edges(v) => i.ingest(Iterable::Edges(*v)),
2022-05-30 15:32:26 +00:00
Value::Model(v) => {
for v in v {
i.ingest(Iterable::Thing(v));
}
}
Value::Array(v) => {
for v in v {
match v {
Value::Table(v) => i.ingest(Iterable::Table(v)),
Value::Thing(v) => i.ingest(Iterable::Thing(v)),
Value::Edges(v) => i.ingest(Iterable::Edges(*v)),
2022-05-30 15:32:26 +00:00
Value::Model(v) => {
for v in v {
i.ingest(Iterable::Thing(v));
}
}
Value::Object(v) => match v.rid() {
Some(v) => i.ingest(Iterable::Thing(v)),
None => {
return Err(Error::UpdateStatement {
value: v.to_string(),
})
}
},
v => {
return Err(Error::UpdateStatement {
value: v.to_string(),
})
}
};
}
}
Value::Object(v) => match v.rid() {
Some(v) => i.ingest(Iterable::Thing(v)),
None => {
return Err(Error::UpdateStatement {
value: v.to_string(),
})
}
},
v => {
return Err(Error::UpdateStatement {
2022-05-06 22:09:08 +00:00
value: v.to_string(),
})
2021-03-29 15:43:37 +00:00
}
};
}
// Assign the statement
let stm = Statement::from(self);
2021-03-29 15:43:37 +00:00
// Output the results
i.output(ctx, opt, txn, &stm).await
2021-03-29 15:43:37 +00:00
}
}
2022-01-14 08:12:56 +00:00
impl fmt::Display for UpdateStatement {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "UPDATE {}", self.what)?;
if let Some(ref v) = self.data {
write!(f, " {}", v)?
}
if let Some(ref v) = self.cond {
write!(f, " {}", v)?
}
if let Some(ref v) = self.output {
write!(f, " {}", v)?
}
if let Some(ref v) = self.timeout {
write!(f, " {}", v)?
}
if self.parallel {
write!(f, " PARALLEL")?
}
2022-01-14 08:12:56 +00:00
Ok(())
}
}
2020-06-29 15:36:01 +00:00
pub fn update(i: &str) -> IResult<&str, UpdateStatement> {
let (i, _) = tag_no_case("UPDATE")(i)?;
let (i, _) = shouldbespace(i)?;
let (i, what) = whats(i)?;
let (i, data) = opt(preceded(shouldbespace, data))(i)?;
let (i, cond) = opt(preceded(shouldbespace, cond))(i)?;
let (i, output) = opt(preceded(shouldbespace, output))(i)?;
let (i, timeout) = opt(preceded(shouldbespace, timeout))(i)?;
let (i, parallel) = opt(preceded(shouldbespace, tag_no_case("PARALLEL")))(i)?;
2020-06-29 15:36:01 +00:00
Ok((
i,
UpdateStatement {
what,
data,
cond,
output,
timeout,
parallel: parallel.is_some(),
2020-06-29 15:36:01 +00:00
},
))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn update_statement() {
let sql = "UPDATE test";
let res = update(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("UPDATE test", format!("{}", out))
}
}