surrealpatch/core/src/doc/field.rs

178 lines
5.1 KiB
Rust
Raw Normal View History

use crate::ctx::Context;
2024-05-28 10:43:45 +00:00
use crate::dbs::Options;
use crate::dbs::Statement;
use crate::doc::Document;
use crate::err::Error;
use crate::iam::Action;
use crate::sql::permission::Permission;
use crate::sql::value::Value;
use reblessive::tree::Stk;
impl<'a> Document<'a> {
pub async fn field(
&mut self,
stk: &mut Stk,
ctx: &Context<'_>,
opt: &Options,
_stm: &Statement<'_>,
) -> Result<(), Error> {
2024-03-18 20:59:39 +00:00
// Check import
if opt.import {
return Ok(());
}
// Get the record id
let rid = self.id.as_ref().unwrap();
// Get the user applied input
let inp = self.initial.doc.changed(self.current.doc.as_ref());
// Loop through all field statements
2024-05-28 10:43:45 +00:00
for fd in self.fd(ctx, opt).await?.iter() {
2022-04-07 10:16:24 +00:00
// Loop over each field in document
for (k, mut val) in self.current.doc.walk(&fd.name).into_iter() {
2022-04-07 10:16:24 +00:00
// Get the initial value
let old = self.initial.doc.pick(&k);
// Get the input value
let inp = inp.pick(&k);
// Check for READONLY clause
if fd.readonly && !self.is_new() && val != old {
return Err(Error::FieldReadonly {
field: fd.name.clone(),
thing: rid.to_string(),
});
}
// Get the default value
let def = match &fd.default {
Some(v) => Some(v),
_ => match &fd.value {
Some(v) if v.is_static() => Some(v),
_ => None,
},
};
// Check for a DEFAULT clause
if let Some(expr) = def {
if self.is_new() && val.is_none() {
// Configure the context
let mut ctx = Context::new(ctx);
ctx.add_value("input", &inp);
ctx.add_value("value", &val);
ctx.add_value("after", &val);
ctx.add_value("before", &old);
// Process the VALUE clause
2024-05-28 10:43:45 +00:00
val = expr.compute(stk, &ctx, opt, Some(&self.current)).await?;
}
}
// Check for a TYPE clause
if let Some(kind) = &fd.kind {
val = val.coerce_to(kind).map_err(|e| match e {
// There was a conversion error
Error::CoerceTo {
from,
..
} => Error::FieldCheck {
thing: rid.to_string(),
field: fd.name.clone(),
value: from.to_string(),
check: kind.to_string(),
},
// There was a different error
e => e,
})?;
}
2022-04-07 10:16:24 +00:00
// Check for a VALUE clause
if let Some(expr) = &fd.value {
// Only run value clause for mutable and new fields
if !fd.readonly || self.is_new() {
// Configure the context
let mut ctx = Context::new(ctx);
ctx.add_value("input", &inp);
ctx.add_value("value", &val);
ctx.add_value("after", &val);
ctx.add_value("before", &old);
// Process the VALUE clause
2024-05-28 10:43:45 +00:00
val = expr.compute(stk, &ctx, opt, Some(&self.current)).await?;
}
}
2022-04-07 10:16:24 +00:00
// Check for a TYPE clause
if let Some(kind) = &fd.kind {
val = val.coerce_to(kind).map_err(|e| match e {
// There was a conversion error
Error::CoerceTo {
from,
..
} => Error::FieldCheck {
thing: rid.to_string(),
field: fd.name.clone(),
value: from.to_string(),
check: kind.to_string(),
},
// There was a different error
e => e,
})?;
}
2022-04-07 10:16:24 +00:00
// Check for a ASSERT clause
if let Some(expr) = &fd.assert {
// Configure the context
let mut ctx = Context::new(ctx);
ctx.add_value("input", &inp);
ctx.add_value("value", &val);
ctx.add_value("after", &val);
ctx.add_value("before", &old);
2022-04-07 10:16:24 +00:00
// Process the ASSERT clause
2024-05-28 10:43:45 +00:00
if !expr.compute(stk, &ctx, opt, Some(&self.current)).await?.is_truthy() {
2022-04-07 10:16:24 +00:00
return Err(Error::FieldValue {
thing: rid.to_string(),
2022-04-07 10:16:24 +00:00
field: fd.name.clone(),
value: val.to_string(),
2022-05-06 22:09:08 +00:00
check: expr.to_string(),
2022-04-07 10:16:24 +00:00
});
}
}
// Check for a PERMISSIONS clause
if opt.check_perms(Action::Edit) {
2022-04-07 10:16:24 +00:00
// Get the permission clause
let perms = if self.is_new() {
&fd.permissions.create
2022-04-07 10:16:24 +00:00
} else {
&fd.permissions.update
};
// Match the permission clause
match perms {
2023-09-01 11:52:02 +00:00
// The field PERMISSIONS clause
// is FULL, enabling this field
// to be updated without checks.
2022-04-07 10:16:24 +00:00
Permission::Full => (),
2023-09-01 11:52:02 +00:00
// The field PERMISSIONS clause
// is NONE, meaning that this
// change will be reverted.
2022-04-07 10:16:24 +00:00
Permission::None => val = old,
2023-09-01 11:52:02 +00:00
// The field PERMISSIONS clause
// is a custom expression, so
// we check the expression and
// revert the field if denied.
2022-04-07 10:16:24 +00:00
Permission::Specific(e) => {
// Disable permissions
let opt = &opt.new_with_perms(false);
2022-04-07 10:16:24 +00:00
// Configure the context
let mut ctx = Context::new(ctx);
ctx.add_value("input", &inp);
ctx.add_value("value", &val);
ctx.add_value("after", &val);
ctx.add_value("before", &old);
2022-04-07 10:16:24 +00:00
// Process the PERMISSION clause
2024-05-28 10:43:45 +00:00
if !e.compute(stk, &ctx, opt, Some(&self.current)).await?.is_truthy() {
2022-04-07 10:16:24 +00:00
val = old
}
}
}
}
2022-04-07 10:16:24 +00:00
// Set the value of the field
match val {
2024-05-28 10:43:45 +00:00
Value::None => self.current.doc.to_mut().del(stk, ctx, opt, &k).await?,
_ => self.current.doc.to_mut().set(stk, ctx, opt, &k, val).await?,
2022-04-07 10:16:24 +00:00
};
}
}
// Carry on
Ok(())
}
}