1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
use crate::{
compilation::{compile_validators, context::CompilationContext, JSONSchema},
error::{error, no_error, ErrorIterator, ValidationError},
paths::InstancePath,
primitive_type::PrimitiveType,
schema_node::SchemaNode,
validator::{format_iter_of_validators, PartialApplication, Validate},
};
use serde_json::{Map, Value};
use super::CompilationResult;
use crate::paths::JSONPointer;
pub(crate) struct AnyOfValidator {
schemas: Vec<SchemaNode>,
schema_path: JSONPointer,
}
impl AnyOfValidator {
#[inline]
pub(crate) fn compile<'a>(
schema: &'a Value,
context: &CompilationContext,
) -> CompilationResult<'a> {
if let Value::Array(items) = schema {
let keyword_context = context.with_path("anyOf");
let mut schemas = Vec::with_capacity(items.len());
for (idx, item) in items.iter().enumerate() {
let item_context = keyword_context.with_path(idx);
let node = compile_validators(item, &item_context)?;
schemas.push(node)
}
Ok(Box::new(AnyOfValidator {
schemas,
schema_path: keyword_context.into_pointer(),
}))
} else {
Err(ValidationError::single_type_error(
JSONPointer::default(),
context.clone().into_pointer(),
schema,
PrimitiveType::Array,
))
}
}
}
impl Validate for AnyOfValidator {
fn is_valid(&self, schema: &JSONSchema, instance: &Value) -> bool {
self.schemas.iter().any(|s| s.is_valid(schema, instance))
}
fn validate<'a, 'b>(
&self,
schema: &'a JSONSchema,
instance: &'b Value,
instance_path: &InstancePath,
) -> ErrorIterator<'b> {
if self.is_valid(schema, instance) {
no_error()
} else {
error(ValidationError::any_of(
self.schema_path.clone(),
instance_path.into(),
instance,
))
}
}
fn apply<'a>(
&'a self,
schema: &JSONSchema,
instance: &Value,
instance_path: &InstancePath,
) -> PartialApplication<'a> {
let mut successes = Vec::new();
let mut failures = Vec::new();
for node in &self.schemas {
let result = node.apply_rooted(schema, instance, instance_path);
if result.is_valid() {
successes.push(result);
} else {
failures.push(result);
}
}
if successes.is_empty() {
failures.into_iter().collect()
} else {
successes.into_iter().collect()
}
}
}
impl core::fmt::Display for AnyOfValidator {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"anyOf: [{}]",
format_iter_of_validators(self.schemas.iter().map(SchemaNode::validators))
)
}
}
#[inline]
pub(crate) fn compile<'a>(
_: &'a Map<String, Value>,
schema: &'a Value,
context: &CompilationContext,
) -> Option<CompilationResult<'a>> {
Some(AnyOfValidator::compile(schema, context))
}
#[cfg(test)]
mod tests {
use crate::tests_util;
use serde_json::{json, Value};
use test_case::test_case;
#[test_case(&json!({"anyOf": [{"type": "string"}]}), &json!(1), "/anyOf")]
#[test_case(&json!({"anyOf": [{"type": "integer"}, {"type": "string"}]}), &json!({}), "/anyOf")]
fn schema_path(schema: &Value, instance: &Value, expected: &str) {
tests_util::assert_schema_path(schema, instance, expected)
}
}