jsonschema-rs/jsonschema/src/keywords/one_of.rs

94 lines
3.1 KiB
Rust

use crate::{
compilation::{compile_validators, context::CompilationContext, JSONSchema},
error::{error, no_error, CompilationError, ErrorIterator, ValidationError},
keywords::{format_vec_of_validators, CompilationResult, Validators},
validator::Validate,
};
use serde_json::{Map, Value};
pub(crate) struct OneOfValidator {
schemas: Vec<Validators>,
}
impl OneOfValidator {
#[inline]
pub(crate) fn compile(schema: &Value, context: &CompilationContext) -> CompilationResult {
if let Value::Array(items) = schema {
let mut schemas = Vec::with_capacity(items.len());
for item in items {
schemas.push(compile_validators(item, context)?)
}
Ok(Box::new(OneOfValidator { schemas }))
} else {
Err(CompilationError::SchemaError)
}
}
fn get_first_valid(&self, schema: &JSONSchema, instance: &Value) -> Option<usize> {
let mut first_valid_idx = None;
for (idx, validators) in self.schemas.iter().enumerate() {
if validators
.iter()
.all(|validator| validator.is_valid(schema, instance))
{
first_valid_idx = Some(idx);
break;
}
}
first_valid_idx
}
#[allow(clippy::integer_arithmetic)]
fn are_others_valid(&self, schema: &JSONSchema, instance: &Value, idx: usize) -> bool {
// `idx + 1` will not overflow, because the maximum possible value there is `usize::MAX - 1`
// For example we have `usize::MAX` schemas and only the last one is valid, then
// in `get_first_valid` we enumerate from `0`, and on the last index will be `usize::MAX - 1`
for validators in self.schemas.iter().skip(idx + 1) {
if validators
.iter()
.all(|validator| validator.is_valid(schema, instance))
{
return true;
}
}
false
}
}
impl Validate for OneOfValidator {
fn is_valid(&self, schema: &JSONSchema, instance: &Value) -> bool {
let first_valid_idx = self.get_first_valid(schema, instance);
if let Some(idx) = first_valid_idx {
!self.are_others_valid(schema, instance, idx)
} else {
false
}
}
fn validate<'a>(&self, schema: &'a JSONSchema, instance: &'a Value) -> ErrorIterator<'a> {
let first_valid_idx = self.get_first_valid(schema, instance);
if let Some(idx) = first_valid_idx {
if self.are_others_valid(schema, instance, idx) {
return error(ValidationError::one_of_multiple_valid(instance));
}
no_error()
} else {
error(ValidationError::one_of_not_valid(instance))
}
}
}
impl ToString for OneOfValidator {
fn to_string(&self) -> String {
format!("oneOf: [{}]", format_vec_of_validators(&self.schemas))
}
}
#[inline]
pub(crate) fn compile(
_: &Map<String, Value>,
schema: &Value,
context: &CompilationContext,
) -> Option<CompilationResult> {
Some(OneOfValidator::compile(schema, context))
}