use serde_json::Value;
use crate::{resource::InnerResourcePtr, segments::Segment, Error, Resolver, Segments};
use super::subresources::SubresourceIteratorInner;
pub(crate) fn object_iter<'a>(
(key, value): (&'a String, &'a Value),
) -> SubresourceIteratorInner<'a> {
match key.as_str() {
"additionalItems"
| "additionalProperties"
| "contains"
| "contentSchema"
| "else"
| "if"
| "not"
| "propertyNames"
| "then"
| "unevaluatedItems"
| "unevaluatedProperties" => SubresourceIteratorInner::Once(value),
"allOf" | "anyOf" | "oneOf" => {
if let Some(arr) = value.as_array() {
SubresourceIteratorInner::Array(arr.iter())
} else {
SubresourceIteratorInner::Empty
}
}
"$defs" | "definitions" | "dependentSchemas" | "patternProperties" | "properties" => {
if let Some(obj) = value.as_object() {
SubresourceIteratorInner::Object(obj.values())
} else {
SubresourceIteratorInner::Empty
}
}
"items" => match value {
Value::Array(arr) => SubresourceIteratorInner::Array(arr.iter()),
_ => SubresourceIteratorInner::Once(value),
},
_ => SubresourceIteratorInner::Empty,
}
}
pub(crate) fn maybe_in_subresource<'r>(
segments: &Segments,
resolver: &Resolver<'r>,
subresource: &InnerResourcePtr,
) -> Result<Resolver<'r>, Error> {
const IN_VALUE: &[&str] = &[
"additionalItems",
"additionalProperties",
"contains",
"contentSchema",
"else",
"if",
"not",
"propertyNames",
"then",
"unevaluatedItems",
"unevaluatedProperties",
];
const IN_CHILD: &[&str] = &[
"allOf",
"anyOf",
"oneOf",
"$defs",
"definitions",
"dependentSchemas",
"patternProperties",
"properties",
];
let mut iter = segments.iter();
while let Some(segment) = iter.next() {
if let Segment::Key(key) = segment {
if *key == "items" && subresource.contents().is_object() {
return resolver.in_subresource_inner(subresource);
}
if !IN_VALUE.contains(&key.as_ref())
&& (!IN_CHILD.contains(&key.as_ref()) || iter.next().is_none())
{
return Ok(resolver.clone());
}
}
}
resolver.in_subresource_inner(subresource)
}