surrealpatch/src/sql/statements/update.rs

123 lines
2.8 KiB
Rust
Raw Normal View History

2021-03-29 15:43:37 +00:00
use crate::dbs;
use crate::dbs::Executor;
use crate::dbs::Iterator;
use crate::dbs::Runtime;
2021-03-29 15:43:37 +00:00
use crate::doc::Document;
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};
2021-03-29 15:43:37 +00:00
use crate::sql::literal::{whats, Literal, Literals};
2020-06-29 15:36:01 +00:00
use crate::sql::output::{output, Output};
use crate::sql::timeout::{timeout, Timeout};
use nom::bytes::complete::tag_no_case;
use nom::combinator::opt;
use nom::sequence::preceded;
use nom::IResult;
use serde::{Deserialize, Serialize};
use std::fmt;
2021-03-29 15:43:37 +00:00
#[derive(Clone, Debug, Default, Eq, PartialEq, Serialize, Deserialize)]
2020-06-29 15:36:01 +00:00
pub struct UpdateStatement {
2021-03-29 15:43:37 +00:00
pub what: Literals,
2020-06-29 15:36:01 +00:00
#[serde(skip_serializing_if = "Option::is_none")]
pub data: Option<Data>,
#[serde(skip_serializing_if = "Option::is_none")]
pub cond: Option<Cond>,
#[serde(skip_serializing_if = "Option::is_none")]
pub output: Option<Output>,
#[serde(skip_serializing_if = "Option::is_none")]
pub timeout: Option<Timeout>,
}
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)?
}
Ok(())
}
}
2021-03-29 15:43:37 +00:00
impl dbs::Process for UpdateStatement {
fn process(
&self,
ctx: &Runtime,
2021-03-29 15:43:37 +00:00
exe: &Executor,
doc: Option<&Document>,
) -> Result<Literal, Error> {
// Create a new iterator
let i = Iterator::new();
// Loop over the select targets
for w in self.what.to_owned() {
match w.process(ctx, exe, doc)? {
Literal::Table(_) => {
i.process_table(ctx, exe);
}
Literal::Thing(_) => {
i.process_thing(ctx, exe);
}
Literal::Model(_) => {
i.process_model(ctx, exe);
}
Literal::Array(_) => {
i.process_array(ctx, exe);
}
Literal::Object(_) => {
i.process_object(ctx, exe);
}
_ => {
todo!() // Return error
}
};
}
// Output the results
i.output(ctx, exe)
}
}
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)?;
Ok((
i,
UpdateStatement {
what,
data,
cond,
output,
timeout,
},
))
}
#[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))
}
}