2022-04-07 10:16:52 +00:00
|
|
|
use crate::ctx::Context;
|
2022-02-06 01:14:56 +00:00
|
|
|
use crate::dbs::Options;
|
|
|
|
use crate::dbs::Statement;
|
2022-02-15 01:00:30 +00:00
|
|
|
use crate::dbs::Transaction;
|
2022-02-06 01:14:56 +00:00
|
|
|
use crate::doc::Document;
|
|
|
|
use crate::err::Error;
|
|
|
|
use crate::sql::field::Field;
|
|
|
|
use crate::sql::idiom::Idiom;
|
|
|
|
use crate::sql::output::Output;
|
2022-06-04 08:55:05 +00:00
|
|
|
use crate::sql::part::Part;
|
2022-04-07 10:16:52 +00:00
|
|
|
use crate::sql::permission::Permission;
|
2022-02-06 01:14:56 +00:00
|
|
|
use crate::sql::value::Value;
|
|
|
|
|
2022-02-13 19:03:00 +00:00
|
|
|
impl<'a> Document<'a> {
|
2022-02-06 01:14:56 +00:00
|
|
|
pub async fn pluck(
|
|
|
|
&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-05-13 20:46:56 +00:00
|
|
|
stm: &Statement<'_>,
|
2022-02-06 01:14:56 +00:00
|
|
|
) -> Result<Value, Error> {
|
|
|
|
// Ensure futures are run
|
|
|
|
let opt = &opt.futures(true);
|
2022-04-07 10:16:52 +00:00
|
|
|
// Process the desired output
|
|
|
|
let mut out = match stm.output() {
|
2022-02-06 01:14:56 +00:00
|
|
|
Some(v) => match v {
|
2022-03-06 10:58:59 +00:00
|
|
|
Output::None => Err(Error::Ignore),
|
2022-02-06 01:14:56 +00:00
|
|
|
Output::Null => Ok(Value::Null),
|
|
|
|
Output::Diff => Ok(self.initial.diff(&self.current, Idiom::default()).into()),
|
2022-02-15 01:00:30 +00:00
|
|
|
Output::After => self.current.compute(ctx, opt, txn, Some(&self.current)).await,
|
|
|
|
Output::Before => self.initial.compute(ctx, opt, txn, Some(&self.initial)).await,
|
2022-02-06 01:14:56 +00:00
|
|
|
Output::Fields(v) => {
|
|
|
|
let mut out = match v.all() {
|
2022-02-15 01:00:30 +00:00
|
|
|
true => self.current.compute(ctx, opt, txn, Some(&self.current)).await?,
|
2022-02-06 01:14:56 +00:00
|
|
|
false => Value::base(),
|
|
|
|
};
|
2022-03-04 16:01:32 +00:00
|
|
|
for v in v.other() {
|
2022-02-06 01:14:56 +00:00
|
|
|
match v {
|
|
|
|
Field::All => (),
|
2022-06-04 08:55:05 +00:00
|
|
|
Field::Alone(v) => match v {
|
|
|
|
// This expression is a multi-output graph traversal
|
|
|
|
Value::Idiom(v) if v.is_multi_yield() => {
|
|
|
|
// Store the different output yields here
|
|
|
|
let mut res: Vec<(&[Part], Value)> = Vec::new();
|
|
|
|
// Split the expression by each output alias
|
|
|
|
for v in v.split_inclusive(Idiom::split_multi_yield) {
|
|
|
|
// Use the last fetched value for each fetch
|
|
|
|
let x = match res.last() {
|
|
|
|
Some((_, r)) => r,
|
|
|
|
None => self.current.as_ref(),
|
|
|
|
};
|
|
|
|
// Continue fetching the next idiom part
|
|
|
|
let x = x
|
|
|
|
.get(ctx, opt, txn, v)
|
|
|
|
.await?
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?;
|
|
|
|
// Add the result to the temporary store
|
|
|
|
res.push((v, x));
|
|
|
|
}
|
|
|
|
// Assign each fetched yield to the output
|
|
|
|
for (p, x) in res {
|
|
|
|
match p.last().unwrap().alias() {
|
|
|
|
// This is an alias expression part
|
|
|
|
Some(i) => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
// This is the end of the expression
|
|
|
|
None => out.set(ctx, opt, txn, v, x).await?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This expression is a normal field expression
|
|
|
|
_ => {
|
|
|
|
let x = v.compute(ctx, opt, txn, Some(&self.current)).await?;
|
|
|
|
out.set(ctx, opt, txn, v.to_idiom().as_ref(), x).await?
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Field::Alias(v, i) => match v {
|
|
|
|
// This expression is a multi-output graph traversal
|
|
|
|
Value::Idiom(v) if v.is_multi_yield() => {
|
|
|
|
// Store the different output yields here
|
|
|
|
let mut res: Vec<(&[Part], Value)> = Vec::new();
|
|
|
|
// Split the expression by each output alias
|
|
|
|
for v in v.split_inclusive(Idiom::split_multi_yield) {
|
|
|
|
// Use the last fetched value for each fetch
|
|
|
|
let x = match res.last() {
|
|
|
|
Some((_, r)) => r,
|
|
|
|
None => self.current.as_ref(),
|
|
|
|
};
|
|
|
|
// Continue fetching the next idiom part
|
|
|
|
let x = x
|
|
|
|
.get(ctx, opt, txn, v)
|
|
|
|
.await?
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?;
|
|
|
|
// Add the result to the temporary store
|
|
|
|
res.push((v, x));
|
|
|
|
}
|
|
|
|
// Assign each fetched yield to the output
|
|
|
|
for (p, x) in res {
|
|
|
|
match p.last().unwrap().alias() {
|
|
|
|
// This is an alias expression part
|
|
|
|
Some(i) => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
// This is the end of the expression
|
|
|
|
None => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
let x = v.compute(ctx, opt, txn, Some(&self.current)).await?;
|
|
|
|
out.set(ctx, opt, txn, i, x).await?
|
|
|
|
}
|
|
|
|
},
|
2022-02-06 01:14:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(out)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => match stm {
|
2022-04-01 09:15:52 +00:00
|
|
|
Statement::Select(s) => {
|
|
|
|
let mut out = match s.expr.all() {
|
2022-02-15 01:00:30 +00:00
|
|
|
true => self.current.compute(ctx, opt, txn, Some(&self.current)).await?,
|
2022-02-06 01:14:56 +00:00
|
|
|
false => Value::base(),
|
|
|
|
};
|
2022-04-01 09:15:52 +00:00
|
|
|
for v in s.expr.other() {
|
2022-02-06 01:14:56 +00:00
|
|
|
match v {
|
|
|
|
Field::All => (),
|
2022-03-25 18:43:22 +00:00
|
|
|
Field::Alone(v) => match v {
|
2022-06-04 08:55:05 +00:00
|
|
|
// This expression is a grouped aggregate function
|
2022-04-01 09:15:52 +00:00
|
|
|
Value::Function(f) if s.group.is_some() && f.is_aggregate() => {
|
2022-03-25 18:43:22 +00:00
|
|
|
let x = match f.args().len() {
|
2022-06-04 08:55:05 +00:00
|
|
|
// If no function arguments, then compute the result
|
2022-03-25 18:43:22 +00:00
|
|
|
0 => f.compute(ctx, opt, txn, Some(&self.current)).await?,
|
2022-06-04 08:55:05 +00:00
|
|
|
// If arguments, then pass the first value through
|
2022-03-25 18:43:22 +00:00
|
|
|
_ => {
|
|
|
|
f.args()[0]
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?
|
|
|
|
}
|
|
|
|
};
|
|
|
|
out.set(ctx, opt, txn, v.to_idiom().as_ref(), x).await?;
|
|
|
|
}
|
2022-06-04 08:55:05 +00:00
|
|
|
// This expression is a multi-output graph traversal
|
|
|
|
Value::Idiom(v) if v.is_multi_yield() => {
|
|
|
|
// Store the different output yields here
|
|
|
|
let mut res: Vec<(&[Part], Value)> = Vec::new();
|
|
|
|
// Split the expression by each output alias
|
|
|
|
for v in v.split_inclusive(Idiom::split_multi_yield) {
|
|
|
|
// Use the last fetched value for each fetch
|
|
|
|
let x = match res.last() {
|
|
|
|
Some((_, r)) => r,
|
|
|
|
None => self.current.as_ref(),
|
|
|
|
};
|
|
|
|
// Continue fetching the next idiom part
|
|
|
|
let x = x
|
|
|
|
.get(ctx, opt, txn, v)
|
|
|
|
.await?
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?;
|
|
|
|
// Add the result to the temporary store
|
|
|
|
res.push((v, x));
|
|
|
|
}
|
|
|
|
// Assign each fetched yield to the output
|
|
|
|
for (p, x) in res {
|
|
|
|
match p.last().unwrap().alias() {
|
|
|
|
// This is an alias expression part
|
|
|
|
Some(i) => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
// This is the end of the expression
|
|
|
|
None => out.set(ctx, opt, txn, v, x).await?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This expression is a normal field expression
|
2022-03-25 18:43:22 +00:00
|
|
|
_ => {
|
|
|
|
let x = v.compute(ctx, opt, txn, Some(&self.current)).await?;
|
|
|
|
out.set(ctx, opt, txn, v.to_idiom().as_ref(), x).await?;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Field::Alias(v, i) => match v {
|
2022-06-04 08:55:05 +00:00
|
|
|
// This expression is a grouped aggregate function
|
2022-04-01 09:15:52 +00:00
|
|
|
Value::Function(f) if s.group.is_some() && f.is_aggregate() => {
|
2022-03-25 18:43:22 +00:00
|
|
|
let x = match f.args().len() {
|
2022-06-04 08:55:05 +00:00
|
|
|
// If no function arguments, then compute the result
|
2022-03-25 18:43:22 +00:00
|
|
|
0 => f.compute(ctx, opt, txn, Some(&self.current)).await?,
|
2022-06-04 08:55:05 +00:00
|
|
|
// If arguments, then pass the first value through
|
2022-03-25 18:43:22 +00:00
|
|
|
_ => {
|
|
|
|
f.args()[0]
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?
|
|
|
|
}
|
|
|
|
};
|
|
|
|
out.set(ctx, opt, txn, i, x).await?;
|
|
|
|
}
|
2022-06-04 08:55:05 +00:00
|
|
|
// This expression is a multi-output graph traversal
|
|
|
|
Value::Idiom(v) if v.is_multi_yield() => {
|
|
|
|
// Store the different output yields here
|
|
|
|
let mut res: Vec<(&[Part], Value)> = Vec::new();
|
|
|
|
// Split the expression by each output alias
|
|
|
|
for v in v.split_inclusive(Idiom::split_multi_yield) {
|
|
|
|
// Use the last fetched value for each fetch
|
|
|
|
let x = match res.last() {
|
|
|
|
Some((_, r)) => r,
|
|
|
|
None => self.current.as_ref(),
|
|
|
|
};
|
|
|
|
// Continue fetching the next idiom part
|
|
|
|
let x = x
|
|
|
|
.get(ctx, opt, txn, v)
|
|
|
|
.await?
|
|
|
|
.compute(ctx, opt, txn, Some(&self.current))
|
|
|
|
.await?;
|
|
|
|
// Add the result to the temporary store
|
|
|
|
res.push((v, x));
|
|
|
|
}
|
|
|
|
// Assign each fetched yield to the output
|
|
|
|
for (p, x) in res {
|
|
|
|
match p.last().unwrap().alias() {
|
|
|
|
// This is an alias expression part
|
|
|
|
Some(i) => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
// This is the end of the expression
|
|
|
|
None => out.set(ctx, opt, txn, i, x).await?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This expression is a normal field expression
|
2022-03-25 18:43:22 +00:00
|
|
|
_ => {
|
|
|
|
let x = v.compute(ctx, opt, txn, Some(&self.current)).await?;
|
|
|
|
out.set(ctx, opt, txn, i, x).await?;
|
|
|
|
}
|
|
|
|
},
|
2022-02-06 01:14:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(out)
|
|
|
|
}
|
2022-04-01 09:15:52 +00:00
|
|
|
Statement::Create(_) => {
|
|
|
|
self.current.compute(ctx, opt, txn, Some(&self.current)).await
|
|
|
|
}
|
|
|
|
Statement::Update(_) => {
|
|
|
|
self.current.compute(ctx, opt, txn, Some(&self.current)).await
|
|
|
|
}
|
|
|
|
Statement::Relate(_) => {
|
|
|
|
self.current.compute(ctx, opt, txn, Some(&self.current)).await
|
|
|
|
}
|
|
|
|
Statement::Insert(_) => {
|
|
|
|
self.current.compute(ctx, opt, txn, Some(&self.current)).await
|
|
|
|
}
|
|
|
|
_ => Err(Error::Ignore),
|
2022-02-06 01:14:56 +00:00
|
|
|
},
|
2022-04-07 10:16:52 +00:00
|
|
|
}?;
|
2022-04-13 17:31:28 +00:00
|
|
|
// Check if this record exists
|
|
|
|
if self.id.is_some() {
|
|
|
|
// Loop through all field statements
|
|
|
|
for fd in self.fd(opt, txn).await?.iter() {
|
|
|
|
// Loop over each field in document
|
|
|
|
for k in out.each(&fd.name).iter() {
|
|
|
|
// Process field permissions
|
|
|
|
match &fd.permissions.select {
|
|
|
|
Permission::Full => (),
|
|
|
|
Permission::None => out.del(ctx, opt, txn, k).await?,
|
|
|
|
Permission::Specific(e) => {
|
|
|
|
// Get the current value
|
|
|
|
let val = self.current.pick(k);
|
|
|
|
// Configure the context
|
|
|
|
let mut ctx = Context::new(ctx);
|
2022-05-14 12:35:08 +00:00
|
|
|
ctx.add_value("value".into(), &val);
|
2022-04-13 17:31:28 +00:00
|
|
|
// Process the PERMISSION clause
|
|
|
|
if !e.compute(&ctx, opt, txn, Some(&self.current)).await?.is_truthy() {
|
|
|
|
out.del(&ctx, opt, txn, k).await?
|
|
|
|
}
|
2022-04-07 10:16:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-06 01:14:56 +00:00
|
|
|
}
|
2022-04-07 10:16:52 +00:00
|
|
|
// Output result
|
|
|
|
Ok(out)
|
2022-02-06 01:14:56 +00:00
|
|
|
}
|
|
|
|
}
|