dmc-schema 0.1.0

Velite-style schema builders for dmc collection records
Documentation
use crate::{Ctx, Schema, ValidationError};
use serde_json::Value;

pub struct OptionalSchema {
  pub inner: Box<dyn Schema>,
}

impl Schema for OptionalSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    if value.is_null() { Ok(Value::Null) } else { self.inner.parse(value, ctx) }
  }
}

pub struct NullableSchema {
  pub inner: Box<dyn Schema>,
}

impl Schema for NullableSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    if value.is_null() { Ok(Value::Null) } else { self.inner.parse(value, ctx) }
  }
}

pub struct DefaultSchema {
  pub inner: Box<dyn Schema>,
  pub fallback: Value,
}

impl Schema for DefaultSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    if value.is_null() { self.inner.parse(&self.fallback, ctx) } else { self.inner.parse(value, ctx) }
  }
}

pub struct TransformSchema {
  pub inner: Box<dyn Schema>,
  pub func: Box<dyn Fn(Value) -> Value + Send + Sync>,
}

impl Schema for TransformSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    let parsed = self.inner.parse(value, ctx)?;
    Ok((self.func)(parsed))
  }
}

/// `Ok(())` passes; `Err(msg)` surfaces a single root-level error.
pub type RefinePredicate = Box<dyn Fn(&Value) -> Result<(), String> + Send + Sync>;

/// Pushes zero or more errors into the caller-provided buffer.
pub type SuperRefinePredicate = Box<dyn Fn(&Value, &mut Vec<String>) + Send + Sync>;

pub struct RefineSchema {
  pub inner: Box<dyn Schema>,
  pub predicate: RefinePredicate,
}

impl Schema for RefineSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    let parsed = self.inner.parse(value, ctx)?;
    (self.predicate)(&parsed).map_err(ValidationError::root)?;
    Ok(parsed)
  }
}

pub struct SuperRefineSchema {
  pub inner: Box<dyn Schema>,
  pub predicate: SuperRefinePredicate,
}

impl Schema for SuperRefineSchema {
  fn parse(&self, value: &Value, ctx: &Ctx) -> Result<Value, ValidationError> {
    let parsed = self.inner.parse(value, ctx)?;
    let mut errors = Vec::new();
    (self.predicate)(&parsed, &mut errors);
    if !errors.is_empty() {
      return Err(ValidationError::root(errors.join("; ")));
    }
    Ok(parsed)
  }
}