205 lines
4.9 KiB
Rust
205 lines
4.9 KiB
Rust
use crate::dbs::Iterator;
|
|
use crate::dbs::Level;
|
|
use crate::dbs::Options;
|
|
use crate::dbs::Runtime;
|
|
use crate::dbs::Transaction;
|
|
use crate::err::Error;
|
|
use crate::sql::comment::shouldbespace;
|
|
use crate::sql::cond::{cond, Cond};
|
|
use crate::sql::error::IResult;
|
|
use crate::sql::fetch::{fetch, Fetchs};
|
|
use crate::sql::field::{fields, Fields};
|
|
use crate::sql::group::{group, Groups};
|
|
use crate::sql::limit::{limit, Limit};
|
|
use crate::sql::order::{order, Orders};
|
|
use crate::sql::split::{split, Splits};
|
|
use crate::sql::start::{start, Start};
|
|
use crate::sql::timeout::{timeout, Timeout};
|
|
use crate::sql::value::{selects, Value, Values};
|
|
use crate::sql::version::{version, Version};
|
|
use derive::Store;
|
|
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)]
|
|
pub struct SelectStatement {
|
|
pub expr: Fields,
|
|
pub what: Values,
|
|
pub cond: Option<Cond>,
|
|
pub split: Option<Splits>,
|
|
pub group: Option<Groups>,
|
|
pub order: Option<Orders>,
|
|
pub limit: Option<Limit>,
|
|
pub start: Option<Start>,
|
|
pub fetch: Option<Fetchs>,
|
|
pub version: Option<Version>,
|
|
pub timeout: Option<Timeout>,
|
|
}
|
|
|
|
impl SelectStatement {
|
|
pub fn limit(&self) -> usize {
|
|
match self.limit {
|
|
Some(Limit(v)) => v,
|
|
None => 0,
|
|
}
|
|
}
|
|
pub fn start(&self) -> usize {
|
|
match self.start {
|
|
Some(Start(v)) => v,
|
|
None => 0,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl SelectStatement {
|
|
pub async fn compute(
|
|
&self,
|
|
ctx: &Runtime,
|
|
opt: &Options,
|
|
txn: &Transaction,
|
|
doc: Option<&Value>,
|
|
) -> Result<Value, Error> {
|
|
// Allowed to run?
|
|
opt.check(Level::No)?;
|
|
// Create a new iterator
|
|
let mut i = Iterator::from(self);
|
|
// Ensure futures are processed
|
|
let opt = &opt.futures(true);
|
|
// Loop over the select targets
|
|
for w in self.what.0.iter() {
|
|
let v = w.compute(ctx, opt, txn, doc).await?;
|
|
match v {
|
|
Value::Table(_) => i.prepare(v),
|
|
Value::Thing(_) => i.prepare(v),
|
|
Value::Model(_) => i.prepare(v),
|
|
Value::Array(_) => i.prepare(v),
|
|
v => i.prepare(v),
|
|
};
|
|
}
|
|
// Output the results
|
|
i.output(ctx, opt, txn).await
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for SelectStatement {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "SELECT {} FROM {}", self.expr, self.what)?;
|
|
if let Some(ref v) = self.cond {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.split {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.group {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.order {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.limit {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.start {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.fetch {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.version {
|
|
write!(f, " {}", v)?
|
|
}
|
|
if let Some(ref v) = self.timeout {
|
|
write!(f, " {}", v)?
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub fn select(i: &str) -> IResult<&str, SelectStatement> {
|
|
let (i, _) = tag_no_case("SELECT")(i)?;
|
|
let (i, _) = shouldbespace(i)?;
|
|
let (i, expr) = fields(i)?;
|
|
let (i, _) = shouldbespace(i)?;
|
|
let (i, _) = tag_no_case("FROM")(i)?;
|
|
let (i, _) = shouldbespace(i)?;
|
|
let (i, what) = selects(i)?;
|
|
let (i, cond) = opt(preceded(shouldbespace, cond))(i)?;
|
|
let (i, split) = opt(preceded(shouldbespace, split))(i)?;
|
|
let (i, group) = opt(preceded(shouldbespace, group))(i)?;
|
|
let (i, order) = opt(preceded(shouldbespace, order))(i)?;
|
|
let (i, limit) = opt(preceded(shouldbespace, limit))(i)?;
|
|
let (i, start) = opt(preceded(shouldbespace, start))(i)?;
|
|
let (i, fetch) = opt(preceded(shouldbespace, fetch))(i)?;
|
|
let (i, version) = opt(preceded(shouldbespace, version))(i)?;
|
|
let (i, timeout) = opt(preceded(shouldbespace, timeout))(i)?;
|
|
Ok((
|
|
i,
|
|
SelectStatement {
|
|
expr,
|
|
what,
|
|
cond,
|
|
split,
|
|
group,
|
|
order,
|
|
limit,
|
|
start,
|
|
fetch,
|
|
version,
|
|
timeout,
|
|
},
|
|
))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn select_statement_param() {
|
|
let sql = "SELECT * FROM $test";
|
|
let res = select(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!(sql, format!("{}", out))
|
|
}
|
|
|
|
#[test]
|
|
fn select_statement_table() {
|
|
let sql = "SELECT * FROM test";
|
|
let res = select(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!(sql, format!("{}", out));
|
|
}
|
|
|
|
#[test]
|
|
fn select_statement_thing() {
|
|
let sql = "SELECT * FROM test:thingy ORDER BY name";
|
|
let res = select(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!(sql, format!("{}", out))
|
|
}
|
|
|
|
#[test]
|
|
fn select_statement_clash() {
|
|
let sql = "SELECT * FROM order ORDER BY order";
|
|
let res = select(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!(sql, format!("{}", out))
|
|
}
|
|
|
|
#[test]
|
|
fn select_statement_table_thing() {
|
|
let sql = "SELECT *, ((1 + 3) / 4), 1.3999 AS tester FROM test, test:thingy";
|
|
let res = select(sql);
|
|
assert!(res.is_ok());
|
|
let out = res.unwrap().1;
|
|
assert_eq!(sql, format!("{}", out))
|
|
}
|
|
}
|