surrealdb_core/sql/
fetch.rsuse crate::ctx::Context;
use crate::dbs::Options;
use crate::err::Error;
use crate::sql::fmt::Fmt;
use crate::sql::statements::info::InfoStructure;
use crate::sql::{Idiom, Value};
use crate::syn;
use reblessive::tree::Stk;
use revision::revisioned;
use serde::{Deserialize, Serialize};
use std::fmt::{self, Display, Formatter};
use std::ops::Deref;
#[revisioned(revision = 1)]
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
#[non_exhaustive]
pub struct Fetchs(pub Vec<Fetch>);
impl Deref for Fetchs {
type Target = Vec<Fetch>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl IntoIterator for Fetchs {
type Item = Fetch;
type IntoIter = std::vec::IntoIter<Self::Item>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
impl fmt::Display for Fetchs {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "FETCH {}", Fmt::comma_separated(&self.0))
}
}
impl InfoStructure for Fetchs {
fn structure(self) -> Value {
self.into_iter().map(Fetch::structure).collect::<Vec<_>>().into()
}
}
#[revisioned(revision = 2)]
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
#[non_exhaustive]
pub struct Fetch(
#[revision(end = 2, convert_fn = "convert_fetch_idiom")] pub Idiom,
#[revision(start = 2)] pub Value,
);
impl Fetch {
fn convert_fetch_idiom(&mut self, _revision: u16, old: Idiom) -> Result<(), revision::Error> {
self.0 = if old.is_empty() {
Value::None
} else {
Value::Idiom(old)
};
Ok(())
}
pub(crate) async fn compute(
&self,
stk: &mut Stk,
ctx: &Context,
opt: &Options,
idioms: &mut Vec<Idiom>,
) -> Result<(), Error> {
let strand_or_idiom = |v: Value| match v {
Value::Strand(s) => Ok(Idiom::from(s.0)),
Value::Idiom(i) => Ok(i.to_owned()),
v => Err(Error::InvalidFetch {
value: v,
}),
};
match &self.0 {
Value::Idiom(idiom) => {
idioms.push(idiom.to_owned());
Ok(())
}
Value::Param(param) => {
let v = param.compute(stk, ctx, opt, None).await?;
idioms.push(strand_or_idiom(v)?);
Ok(())
}
Value::Function(f) => {
if f.name() == Some("type::field") {
let v = match f.args().first().unwrap() {
Value::Param(v) => v.compute(stk, ctx, opt, None).await?,
v => v.to_owned(),
};
idioms.push(strand_or_idiom(v)?);
Ok(())
} else if f.name() == Some("type::fields") {
let args = match f.args().first().unwrap() {
Value::Param(v) => v.compute(stk, ctx, opt, None).await?,
v => v.to_owned(),
};
let args: Vec<Value> = args.try_into()?;
for v in args.into_iter() {
let i = match v {
Value::Param(v) => {
strand_or_idiom(v.compute(stk, ctx, opt, None).await?)?
}
Value::Strand(s) => syn::idiom(s.as_str())?,
Value::Idiom(i) => i,
v => {
return Err(Error::InvalidFetch {
value: v,
})
}
};
idioms.push(i);
}
Ok(())
} else {
Err(Error::InvalidFetch {
value: Value::Function(f.clone()),
})
}
}
v => Err(Error::InvalidFetch {
value: v.clone(),
}),
}
}
}
impl From<Value> for Fetch {
fn from(value: Value) -> Self {
Self(value)
}
}
impl Deref for Fetch {
type Target = Value;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl Display for Fetch {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(&self.0, f)
}
}
impl InfoStructure for Fetch {
fn structure(self) -> Value {
self.to_string().into()
}
}