2022-05-14 12:35:08 +00:00
|
|
|
use crate::ctx::Context;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::dbs::Options;
|
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;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::idiom;
|
|
|
|
use crate::sql::idiom::Idiom;
|
2022-02-22 19:05:58 +00:00
|
|
|
use crate::sql::part::Next;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::part::Part;
|
|
|
|
use crate::sql::value::Value;
|
2022-03-16 23:52:25 +00:00
|
|
|
use nom::character::complete::char;
|
2020-06-29 15:36:01 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use std::fmt;
|
2022-05-04 21:38:19 +00:00
|
|
|
use std::ops::Deref;
|
2020-06-29 15:36:01 +00:00
|
|
|
use std::str;
|
|
|
|
|
2021-03-29 15:43:37 +00:00
|
|
|
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize)]
|
2022-05-04 21:38:19 +00:00
|
|
|
pub struct Param(pub Idiom);
|
2021-03-31 11:54:20 +00:00
|
|
|
|
|
|
|
impl From<Idiom> for Param {
|
2022-10-04 21:51:18 +00:00
|
|
|
fn from(p: Idiom) -> Self {
|
|
|
|
Self(p)
|
2022-05-04 21:38:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Deref for Param {
|
|
|
|
type Target = Idiom;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
2021-03-31 11:54:20 +00:00
|
|
|
}
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
2022-01-14 08:12:56 +00:00
|
|
|
impl Param {
|
2022-05-01 22:25:53 +00:00
|
|
|
pub(crate) async fn compute(
|
2021-03-29 15:43:37 +00:00
|
|
|
&self,
|
2022-05-14 12:35:08 +00:00
|
|
|
ctx: &Context<'_>,
|
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> {
|
|
|
|
// Find a base variable by name
|
2022-05-05 11:26:46 +00:00
|
|
|
match self.first() {
|
2022-01-13 17:36:41 +00:00
|
|
|
// The first part will be a field
|
2022-05-05 00:27:57 +00:00
|
|
|
Some(Part::Field(v)) => match v.as_str() {
|
2022-05-19 22:03:00 +00:00
|
|
|
"this" | "self" => match doc {
|
2022-04-05 23:11:33 +00:00
|
|
|
// The base document exists
|
|
|
|
Some(v) => {
|
|
|
|
// Get the path parts
|
2022-05-04 21:38:19 +00:00
|
|
|
let pth: &[Part] = self;
|
2022-08-21 12:13:38 +00:00
|
|
|
// Process the parameter value
|
2022-04-05 23:11:33 +00:00
|
|
|
let res = v.compute(ctx, opt, txn, doc).await?;
|
|
|
|
// Return the desired field
|
|
|
|
res.get(ctx, opt, txn, pth.next()).await
|
|
|
|
}
|
|
|
|
// The base document does not exist
|
|
|
|
None => Ok(Value::None),
|
|
|
|
},
|
2022-05-14 12:35:08 +00:00
|
|
|
_ => match ctx.value(v) {
|
2022-04-05 23:11:33 +00:00
|
|
|
// The base variable exists
|
|
|
|
Some(v) => {
|
|
|
|
// Get the path parts
|
2022-05-04 21:38:19 +00:00
|
|
|
let pth: &[Part] = self;
|
2022-08-21 12:13:38 +00:00
|
|
|
// Process the parameter value
|
2022-04-05 23:11:33 +00:00
|
|
|
let res = v.compute(ctx, opt, txn, doc).await?;
|
|
|
|
// Return the desired field
|
|
|
|
res.get(ctx, opt, txn, pth.next()).await
|
|
|
|
}
|
|
|
|
// The base variable does not exist
|
|
|
|
None => Ok(Value::None),
|
|
|
|
},
|
2022-01-13 17:36:41 +00:00
|
|
|
},
|
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 08:12:56 +00:00
|
|
|
impl fmt::Display for Param {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2022-05-04 21:38:19 +00:00
|
|
|
write!(f, "${}", &self.0)
|
2022-01-14 08:12:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-29 15:36:01 +00:00
|
|
|
pub fn param(i: &str) -> IResult<&str, Param> {
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('$')(i)?;
|
2022-01-13 17:36:41 +00:00
|
|
|
let (i, v) = idiom::param(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
Ok((i, Param::from(v)))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::test::Parse;
|
2020-06-29 15:36:01 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn param_normal() {
|
|
|
|
let sql = "$test";
|
|
|
|
let res = param(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("$test", format!("{}", out));
|
2022-01-13 17:36:41 +00:00
|
|
|
assert_eq!(out, Param::parse("$test"));
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn param_longer() {
|
|
|
|
let sql = "$test_and_deliver";
|
|
|
|
let res = param(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("$test_and_deliver", format!("{}", out));
|
2022-01-13 17:36:41 +00:00
|
|
|
assert_eq!(out, Param::parse("$test_and_deliver"));
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
2021-03-31 11:54:20 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn param_embedded() {
|
|
|
|
let sql = "$test.temporary[0].embedded";
|
|
|
|
let res = param(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("$test.temporary[0].embedded", format!("{}", out));
|
2022-01-13 17:36:41 +00:00
|
|
|
assert_eq!(out, Param::parse("$test.temporary[0].embedded"));
|
2021-03-31 11:54:20 +00:00
|
|
|
}
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|