jsonschema-rs/jsonschema/src/keywords/legacy/type_draft_4.rs

146 lines
4.7 KiB
Rust

use crate::{
compilation::{context::CompilationContext, JSONSchema},
error::{error, no_error, CompilationError, ErrorIterator, ValidationError},
keywords::{type_, CompilationResult},
primitive_type::{PrimitiveType, PrimitiveTypesBitMap},
validator::Validate,
};
use serde_json::{Map, Number, Value};
use std::convert::TryFrom;
pub(crate) struct MultipleTypesValidator {
types: PrimitiveTypesBitMap,
}
impl MultipleTypesValidator {
#[inline]
pub(crate) fn compile(items: &[Value]) -> CompilationResult {
let mut types = PrimitiveTypesBitMap::new();
for item in items {
match item {
Value::String(string) => {
if let Ok(primitive_type) = PrimitiveType::try_from(string.as_str()) {
types |= primitive_type;
} else {
return Err(CompilationError::SchemaError);
}
}
_ => return Err(CompilationError::SchemaError),
}
}
Ok(Box::new(MultipleTypesValidator { types }))
}
}
impl Validate for MultipleTypesValidator {
fn is_valid(&self, _: &JSONSchema, instance: &Value) -> bool {
match instance {
Value::Array(_) => self.types.contains_type(PrimitiveType::Array),
Value::Bool(_) => self.types.contains_type(PrimitiveType::Boolean),
Value::Null => self.types.contains_type(PrimitiveType::Null),
Value::Number(num) => {
self.types.contains_type(PrimitiveType::Number)
|| (self.types.contains_type(PrimitiveType::Integer) && is_integer(num))
}
Value::Object(_) => self.types.contains_type(PrimitiveType::Object),
Value::String(_) => self.types.contains_type(PrimitiveType::String),
}
}
fn validate<'a>(&self, schema: &'a JSONSchema, instance: &'a Value) -> ErrorIterator<'a> {
if self.is_valid(schema, instance) {
no_error()
} else {
error(ValidationError::multiple_type_error(instance, self.types))
}
}
}
impl ToString for MultipleTypesValidator {
fn to_string(&self) -> String {
format!(
"type: [{}]",
self.types
.into_iter()
.map(|type_| format!("{}", type_))
.collect::<Vec<String>>()
.join(", ")
)
}
}
pub(crate) struct IntegerTypeValidator {}
impl IntegerTypeValidator {
#[inline]
pub(crate) fn compile() -> CompilationResult {
Ok(Box::new(IntegerTypeValidator {}))
}
}
impl Validate for IntegerTypeValidator {
fn is_valid(&self, _: &JSONSchema, instance: &Value) -> bool {
if let Value::Number(num) = instance {
is_integer(num)
} else {
false
}
}
fn validate<'a>(&self, schema: &'a JSONSchema, instance: &'a Value) -> ErrorIterator<'a> {
if self.is_valid(schema, instance) {
no_error()
} else {
error(ValidationError::single_type_error(
instance,
PrimitiveType::Integer,
))
}
}
}
impl ToString for IntegerTypeValidator {
fn to_string(&self) -> String {
"type: integer".to_string()
}
}
fn is_integer(num: &Number) -> bool {
num.is_u64() || num.is_i64()
}
#[inline]
pub(crate) fn compile(
_: &Map<String, Value>,
schema: &Value,
_: &CompilationContext,
) -> Option<CompilationResult> {
match schema {
Value::String(item) => compile_single_type(item.as_str()),
Value::Array(items) => {
if items.len() == 1 {
if let Some(Value::String(item)) = items.iter().next() {
compile_single_type(item.as_str())
} else {
Some(Err(CompilationError::SchemaError))
}
} else {
Some(MultipleTypesValidator::compile(items))
}
}
_ => Some(Err(CompilationError::SchemaError)),
}
}
fn compile_single_type(item: &str) -> Option<CompilationResult> {
match PrimitiveType::try_from(item) {
Ok(PrimitiveType::Array) => Some(type_::ArrayTypeValidator::compile()),
Ok(PrimitiveType::Boolean) => Some(type_::BooleanTypeValidator::compile()),
Ok(PrimitiveType::Integer) => Some(IntegerTypeValidator::compile()),
Ok(PrimitiveType::Null) => Some(type_::NullTypeValidator::compile()),
Ok(PrimitiveType::Number) => Some(type_::NumberTypeValidator::compile()),
Ok(PrimitiveType::Object) => Some(type_::ObjectTypeValidator::compile()),
Ok(PrimitiveType::String) => Some(type_::StringTypeValidator::compile()),
Err(()) => Some(Err(CompilationError::SchemaError)),
}
}