com.networknt.schema.AnyOfValidator Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of json-schema-validator Show documentation
Show all versions of json-schema-validator Show documentation
A json schema validator that supports draft v4, v6, v7, v2019-09 and v2020-12
/*
* Copyright (c) 2016 Network New Technologies Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.networknt.schema;
import com.fasterxml.jackson.databind.JsonNode;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.*;
import java.util.stream.Collectors;
/**
* {@link JsonValidator} for anyOf.
*/
public class AnyOfValidator extends BaseJsonValidator {
private static final Logger logger = LoggerFactory.getLogger(AnyOfValidator.class);
private static final String DISCRIMINATOR_REMARK = "and the discriminator-selected candidate schema didn't pass validation";
private final List schemas = new ArrayList<>();
private final DiscriminatorContext discriminatorContext;
private Boolean canShortCircuit = null;
public AnyOfValidator(SchemaLocation schemaLocation, JsonNodePath evaluationPath, JsonNode schemaNode, JsonSchema parentSchema, ValidationContext validationContext) {
super(schemaLocation, evaluationPath, schemaNode, parentSchema, ValidatorTypeCode.ANY_OF, validationContext);
int size = schemaNode.size();
for (int i = 0; i < size; i++) {
this.schemas.add(validationContext.newSchema(schemaLocation.append(i), evaluationPath.append(i),
schemaNode.get(i), parentSchema));
}
if (this.validationContext.getConfig().isOpenAPI3StyleDiscriminators()) {
this.discriminatorContext = new DiscriminatorContext();
} else {
this.discriminatorContext = null;
}
}
@Override
public Set validate(ExecutionContext executionContext, JsonNode node, JsonNode rootNode, JsonNodePath instanceLocation) {
debug(logger, node, rootNode, instanceLocation);
// get the Validator state object storing validation data
ValidatorState state = executionContext.getValidatorState();
if (this.validationContext.getConfig().isOpenAPI3StyleDiscriminators()) {
executionContext.enterDiscriminatorContext(this.discriminatorContext, instanceLocation);
}
boolean initialHasMatchedNode = state.hasMatchedNode();
Set allErrors = new LinkedHashSet<>();
int numberOfValidSubSchemas = 0;
try {
// Save flag as nested schema evaluation shouldn't trigger fail fast
boolean failFast = executionContext.isFailFast();
try {
executionContext.setFailFast(false);
for (JsonSchema schema : this.schemas) {
Set errors = Collections.emptySet();
state.setMatchedNode(initialHasMatchedNode);
TypeValidator typeValidator = schema.getTypeValidator();
if (typeValidator != null) {
// If schema has type validator and node type doesn't match with schemaType then
// ignore it
// For union type, it is a must to call TypeValidator
if (typeValidator.getSchemaType() != JsonType.UNION && !typeValidator.equalsToSchemaType(node)) {
allErrors.addAll(typeValidator.validate(executionContext, node, rootNode, instanceLocation));
continue;
}
}
if (!state.isWalkEnabled()) {
errors = schema.validate(executionContext, node, rootNode, instanceLocation);
} else {
errors = schema.walk(executionContext, node, rootNode, instanceLocation, true);
}
// check if any validation errors have occurred
if (errors.isEmpty()) {
// check whether there are no errors HOWEVER we have validated the exact
// validator
if (!state.hasMatchedNode()) {
continue;
}
// we found a valid subschema, so increase counter
numberOfValidSubSchemas++;
}
if (errors.isEmpty() && (!this.validationContext.getConfig().isOpenAPI3StyleDiscriminators())
&& canShortCircuit() && canShortCircuit(executionContext)) {
// Clear all errors.
allErrors.clear();
// return empty errors.
return errors;
} else if (this.validationContext.getConfig().isOpenAPI3StyleDiscriminators()) {
if (this.discriminatorContext.isDiscriminatorMatchFound()) {
if (!errors.isEmpty()) {
allErrors.addAll(errors);
allErrors.add(message().instanceNode(node).instanceLocation(instanceLocation)
.locale(executionContext.getExecutionConfig().getLocale())
.failFast(executionContext.isFailFast())
.arguments(DISCRIMINATOR_REMARK).build());
} else {
// Clear all errors.
allErrors.clear();
}
return errors;
}
}
allErrors.addAll(errors);
}
} finally {
// Restore flag
executionContext.setFailFast(failFast);
}
// determine only those errors which are NOT of type "required" property missing
Set childNotRequiredErrors = allErrors.stream()
.filter(error -> !ValidatorTypeCode.REQUIRED.getValue().equals(error.getType()))
.collect(Collectors.toCollection(LinkedHashSet::new));
// in case we had at least one (anyOf, i.e. any number >= 1 of) valid subschemas, we can remove all other errors about "required" properties
if (numberOfValidSubSchemas >= 1 && childNotRequiredErrors.isEmpty()) {
allErrors = childNotRequiredErrors;
}
if (this.validationContext.getConfig().isOpenAPI3StyleDiscriminators() && this.discriminatorContext.isActive()) {
final Set errors = new LinkedHashSet<>();
errors.add(message().instanceNode(node).instanceLocation(instanceLocation)
.locale(executionContext.getExecutionConfig().getLocale())
.arguments(
"based on the provided discriminator. No alternative could be chosen based on the discriminator property")
.build());
return Collections.unmodifiableSet(errors);
}
} finally {
if (this.validationContext.getConfig().isOpenAPI3StyleDiscriminators()) {
executionContext.leaveDiscriminatorContextImmediately(instanceLocation);
}
if (allErrors.isEmpty()) {
state.setMatchedNode(true);
}
}
if (numberOfValidSubSchemas >= 1) {
return Collections.emptySet();
}
return Collections.unmodifiableSet(allErrors);
}
@Override
public Set walk(ExecutionContext executionContext, JsonNode node, JsonNode rootNode, JsonNodePath instanceLocation, boolean shouldValidateSchema) {
if (shouldValidateSchema) {
return validate(executionContext, node, rootNode, instanceLocation);
}
for (JsonSchema schema : this.schemas) {
schema.walk(executionContext, node, rootNode, instanceLocation, false);
}
return new LinkedHashSet<>();
}
/**
* If annotation collection is enabled cannot short circuit.
*
* @see anyOf
* @param executionContext the execution context
* @return true if can short circuit
*/
protected boolean canShortCircuit(ExecutionContext executionContext) {
return !executionContext.getExecutionConfig().isAnnotationCollectionEnabled();
}
/**
* If annotations are require for evaluation cannot short circuit.
*
* @return true if can short circuit
*/
protected boolean canShortCircuit() {
if (this.canShortCircuit == null) {
boolean canShortCircuit = true;
for (JsonValidator validator : getEvaluationParentSchema().getValidators()) {
if ("unevaluatedProperties".equals(validator.getKeyword())
|| "unevaluatedItems".equals(validator.getKeyword())) {
canShortCircuit = false;
}
}
this.canShortCircuit = canShortCircuit;
}
return this.canShortCircuit;
}
@Override
public void preloadJsonSchema() {
preloadJsonSchemas(this.schemas);
canShortCircuit(); // cache flag
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy