#![allow(clippy::pattern_type_mismatch)]
use crate::{
ast::{Expr, Ref},
builtins,
builtins::utils::{enforce_limit, ensure_args_count, ensure_array, ensure_object},
lexer::Span,
Rc, Value, *,
};
use alloc::collections::{BTreeMap, BTreeSet};
use core::iter::Iterator;
use anyhow::{bail, Result};
pub fn register(m: &mut builtins::BuiltinsMap<&'static str, builtins::BuiltinFcn>) {
m.insert("json.filter", (json_filter, 2));
m.insert("json.remove", (json_remove, 2));
m.insert("object.filter", (filter, 2));
m.insert("object.get", (get, 3));
m.insert("object.keys", (keys, 1));
m.insert("object.remove", (remove, 2));
m.insert("object.subset", (subset, 2));
m.insert("object.union", (object_union, 2));
m.insert("object.union_n", (object_union_n, 1));
#[cfg(feature = "jsonschema")]
{
m.insert("json.match_schema", (json_match_schema, 2));
m.insert("json.verify_schema", (json_verify_schema, 1));
}
}
fn json_filter_impl(v: &Value, filter: &Value) -> Result<Value> {
let filters = match filter {
Value::Object(fields) if fields.len() == 1 && filter[&Value::Null] == Value::Null => {
return Ok(v.clone())
}
Value::Object(fields) if !fields.is_empty() => fields,
_ => return Ok(v.clone()),
};
match v {
Value::Array(_) => {
let mut items = vec![];
for (idx, filter) in filters.iter() {
if let Value::String(idx) = idx {
if let Ok(idx) = Value::from_json_str(idx) {
let item = json_filter_impl(&v[&idx], filter)?;
if item != Value::Undefined {
items.push(item);
enforce_limit()?;
}
}
}
}
Ok(Value::from_array(items))
}
Value::Set(s) => {
let mut items = BTreeSet::new();
for (item, filter) in filters.iter() {
if s.contains(item) {
let item = json_filter_impl(item, filter)?;
if item != Value::Undefined {
items.insert(item);
enforce_limit()?;
}
}
}
Ok(Value::from_set(items))
}
Value::Object(_) => {
let mut items = BTreeMap::new();
for (key, filter) in filters.iter() {
let item = json_filter_impl(&v[key], filter)?;
if item != Value::Undefined {
items.insert(key.clone(), item);
enforce_limit()?;
}
}
Ok(Value::from_map(items))
}
_ => Ok(Value::Undefined),
}
}
fn json_remove_impl(v: &Value, filter: &Value) -> Result<Value> {
let filters = match filter {
Value::Object(fields) if !fields.is_empty() => fields,
_ => return Ok(v.clone()),
};
if filter[&Value::Null] == Value::Null {
return Ok(Value::Undefined);
}
match v {
Value::Array(a) => {
let mut items = vec![];
for (idx, item) in a.iter().enumerate() {
let idx = Value::String(format!("{idx}").into());
if let Some(f) = filters.get(&idx) {
let v = json_remove_impl(item, f)?;
if v != Value::Undefined {
items.push(v);
enforce_limit()?;
}
} else {
items.push(item.clone());
enforce_limit()?;
}
}
Ok(Value::from_array(items))
}
Value::Set(s) => {
let mut items = BTreeSet::new();
for item in s.iter() {
if let Some(f) = filters.get(item) {
let v = json_remove_impl(item, f)?;
if v != Value::Undefined {
items.insert(v);
enforce_limit()?;
}
} else {
items.insert(item.clone());
enforce_limit()?;
}
}
Ok(Value::from_set(items))
}
Value::Object(obj) => {
let mut items = BTreeMap::new();
for (key, value) in obj.iter() {
if let Some(f) = filters.get(key) {
let v = json_remove_impl(value, f)?;
if v != Value::Undefined {
items.insert(key.clone(), v);
enforce_limit()?;
}
} else {
items.insert(key.clone(), value.clone());
enforce_limit()?;
}
}
Ok(Value::from_map(items))
}
_ => Ok(Value::Undefined),
}
}
fn merge_filters(
name: &str,
param: &Expr,
itr: &mut dyn Iterator<Item = &Value>,
mut filters: Value,
) -> Result<Value> {
loop {
match itr.next() {
Some(Value::String(s)) => {
let mut fc = filters;
let mut f = &mut fc;
for p in s.split('/') {
let vref = f.make_or_get_value_mut(&[p])?;
if *vref == Value::Undefined {
*vref = Value::new_object();
}
f = vref;
enforce_limit()?;
}
if let Ok(f) = f.as_object_mut() {
f.insert(Value::Null, Value::Null);
enforce_limit()?;
};
filters = fc;
}
Some(Value::Array(a)) => {
let mut fc = filters;
let mut f = &mut fc;
for p in a.iter() {
let vref = match f {
Value::Object(obj) => {
let obj = Rc::make_mut(obj);
let entry = obj.entry(p.clone()).or_insert_with(Value::new_object);
enforce_limit()?;
entry
}
_ => break,
};
f = vref;
enforce_limit()?;
}
if let Ok(f) = f.as_object_mut() {
f.insert(Value::Null, Value::Null);
enforce_limit()?;
};
filters = fc;
}
Some(_) => {
let span = param.span();
bail!(span.error(
format!("`{name}` requires path to be '/' separated string or array of path components.").as_str()
));
}
None => break,
}
}
Ok(filters)
}
fn json_filter(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "json.filter";
ensure_args_count(span, name, params, args, 2)?;
ensure_object(name, ¶ms[0], args[0].clone())?;
let filters = match &args[1] {
Value::Array(a) => merge_filters(name, ¶ms[1], &mut a.iter(), Value::new_object())?,
Value::Set(s) => merge_filters(name, ¶ms[1], &mut s.iter(), Value::new_object())?,
_ => bail!(span.error(format!("`{name}` requires set/array argument").as_str())),
};
if let Ok(v) = filters.as_object() {
if v.is_empty() {
return Ok(Value::new_object());
}
}
json_filter_impl(&args[0], &filters)
}
fn json_remove(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "json.remove";
ensure_args_count(span, name, params, args, 2)?;
ensure_object(name, ¶ms[0], args[0].clone())?;
let filters = match &args[1] {
Value::Array(a) => merge_filters(name, ¶ms[1], &mut a.iter(), Value::new_object())?,
Value::Set(s) => merge_filters(name, ¶ms[1], &mut s.iter(), Value::new_object())?,
_ => bail!(span.error(format!("`{name}` requires set/array argument").as_str())),
};
json_remove_impl(&args[0], &filters)
}
fn filter(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.filter";
ensure_args_count(span, name, params, args, 2)?;
let mut obj = ensure_object(name, ¶ms[0], args[0].clone())?;
let obj_ref = Rc::make_mut(&mut obj);
match &args[1] {
Value::Array(a) => {
let keys: BTreeSet<&Value> = a.iter().collect();
obj_ref.retain(|k, _| keys.contains(k))
}
Value::Set(s) => obj_ref.retain(|k, _| s.contains(k)),
Value::Object(o) => obj_ref.retain(|k, _| o.contains_key(k)),
_ => bail!(span.error(format!("`{name}` requires array/object/set argument").as_str())),
};
Ok(Value::Object(obj))
}
fn get(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.get";
ensure_args_count(span, name, params, args, 3)?;
let obj = ensure_object(name, ¶ms[0], args[0].clone())?;
let default = &args[2];
Ok(match &args[1] {
Value::Array(keys) => {
let mut v = &args[0];
for a in keys.iter() {
v = &v[a];
if v == &Value::Undefined {
v = default;
break;
}
}
v.clone()
}
key => match obj.get(key) {
Some(v) => v.clone(),
_ => default.clone(),
},
})
}
fn keys(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.keys";
ensure_args_count(span, name, params, args, 1)?;
let obj = ensure_object(name, ¶ms[0], args[0].clone())?;
Ok(Value::from_set(obj.keys().cloned().collect()))
}
fn remove(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.remove";
ensure_args_count(span, name, params, args, 2)?;
let mut obj = ensure_object(name, ¶ms[0], args[0].clone())?;
let obj_ref = Rc::make_mut(&mut obj);
match &args[1] {
Value::Array(a) => {
let keys: BTreeSet<&Value> = a.iter().collect();
obj_ref.retain(|k, _| !keys.contains(k))
}
Value::Set(s) => obj_ref.retain(|k, _| !s.contains(k)),
Value::Object(o) => obj_ref.retain(|k, _| !o.contains_key(k)),
_ => bail!(span.error(format!("`{name}` requires array/object/set argument").as_str())),
};
Ok(Value::Object(obj))
}
fn is_subset(sup: &Value, sub: &Value) -> bool {
match (sup, sub) {
(Value::Object(sup), Value::Object(sub)) => {
sub.iter().all(|(k, vsub)| {
match sup.get(k) {
Some(vsup) => is_subset(vsup, vsub),
_ => false,
}
})
}
(Value::Set(sup), Value::Set(sub)) => sub.is_subset(sup),
(Value::Array(sup), Value::Array(sub)) => sup.windows(sub.len()).any(|w| w == &sub[..]),
(Value::Array(sup), Value::Set(_)) => {
let sup = Value::from_set(sup.iter().cloned().collect());
is_subset(&sup, sub)
}
(sup, sub) => sup == sub,
}
}
fn subset(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.subset";
ensure_args_count(span, name, params, args, 2)?;
Ok(Value::Bool(is_subset(&args[0], &args[1])))
}
fn union(obj1: &Value, obj2: &Value) -> Result<Value> {
match (obj1, obj2) {
(Value::Object(m1), Value::Object(m2)) => {
let mut u = obj1.clone();
let um = u.as_object_mut()?;
for (key2, value2) in m2.iter() {
let vm = match m1.get(key2) {
Some(value1) => union(value1, value2)?,
_ => value2.clone(),
};
um.insert(key2.clone(), vm);
}
Ok(u)
}
_ => Ok(obj2.clone()),
}
}
fn object_union(span: &Span, params: &[Ref<Expr>], args: &[Value], _strict: bool) -> Result<Value> {
let name = "object.union";
ensure_args_count(span, name, params, args, 2)?;
let _ = ensure_object(name, ¶ms[0], args[0].clone())?;
let _ = ensure_object(name, ¶ms[1], args[1].clone())?;
union(&args[0], &args[1])
}
fn object_union_n(
span: &Span,
params: &[Ref<Expr>],
args: &[Value],
strict: bool,
) -> Result<Value> {
let name = "object.union_n";
ensure_args_count(span, name, params, args, 1)?;
let arr = ensure_array(name, ¶ms[0], args[0].clone())?;
let mut u = Value::new_object();
for (idx, a) in arr.iter().enumerate() {
if a.as_object().is_err() {
if strict {
bail!(params[0]
.span()
.error(&format!("item at index {idx} is not an object")));
}
return Ok(Value::Undefined);
}
u = union(&u, a)?;
}
Ok(u)
}
#[cfg(feature = "jsonschema")]
fn compile_json_schema(param: &Ref<Expr>, arg: &Value) -> Result<jsonschema::Validator> {
let schema_str = match arg {
Value::String(schema_str) => schema_str.as_ref().to_string(),
_ => arg.to_json_str()?,
};
if let Ok(schema) = serde_json::from_str(&schema_str) {
match jsonschema::validator_for(&schema) {
Ok(schema) => return Ok(schema),
Err(e) => bail!(e.to_string()),
}
}
bail!(param.span().error("not a valid json schema"))
}
#[cfg(feature = "jsonschema")]
fn json_verify_schema(
span: &Span,
params: &[Ref<Expr>],
args: &[Value],
strict: bool,
) -> Result<Value> {
let name = "json.verify_schema";
ensure_args_count(span, name, params, args, 1)?;
Ok(Value::from_array(
match compile_json_schema(¶ms[0], &args[0]) {
Ok(_) => [Value::Bool(true), Value::Null],
Err(e) if strict => bail!(params[0]
.span()
.error(format!("invalid schema: {e}").as_str())),
Err(e) => [Value::Bool(false), Value::String(e.to_string().into())],
}
.to_vec(),
))
}
#[cfg(feature = "jsonschema")]
fn json_match_schema(
span: &Span,
params: &[Ref<Expr>],
args: &[Value],
strict: bool,
) -> Result<Value> {
let name = "json.match_schema";
ensure_args_count(span, name, params, args, 2)?;
let document: serde_json::Value = serde_json::from_str(&args[0].to_json_str()?)
.map_err(|err| span.error(&format!("Failed to parse JSON: {err}")))?;
Ok(Value::from_array(
match compile_json_schema(¶ms[1], &args[1]) {
Ok(schema) => match schema.validate(&document) {
Ok(_) => [Value::Bool(true), Value::Null],
Err(e) => [Value::Bool(false), Value::from(e.to_string())],
},
Err(e) if strict => bail!(params[1]
.span()
.error(format!("invalid schema: {e}").as_str())),
Err(e) => [Value::Bool(false), Value::String(e.to_string().into())],
}
.to_vec(),
))
}