com.networknt.schema.RecursiveRefValidator 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
The newest version!
/*
* 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.Collections;
import java.util.Set;
import java.util.function.Supplier;
/**
* {@link JsonValidator} that resolves $recursiveRef.
*/
public class RecursiveRefValidator extends BaseJsonValidator {
private static final Logger logger = LoggerFactory.getLogger(RecursiveRefValidator.class);
protected final JsonSchemaRef schema;
public RecursiveRefValidator(SchemaLocation schemaLocation, JsonNodePath evaluationPath, JsonNode schemaNode, JsonSchema parentSchema, ValidationContext validationContext) {
super(schemaLocation, evaluationPath, schemaNode, parentSchema, ValidatorTypeCode.RECURSIVE_REF, validationContext);
String refValue = schemaNode.asText();
if (!"#".equals(refValue)) {
ValidationMessage validationMessage = message()
.type(ValidatorTypeCode.RECURSIVE_REF.getValue()).code("internal.invalidRecursiveRef")
.message("{0}: The value of a $recursiveRef must be '#' but is '{1}'").instanceLocation(schemaLocation.getFragment())
.instanceNode(this.schemaNode)
.evaluationPath(evaluationPath).arguments(refValue).build();
throw new JsonSchemaException(validationMessage);
}
this.schema = getRefSchema(parentSchema, validationContext, refValue, evaluationPath);
}
static JsonSchemaRef getRefSchema(JsonSchema parentSchema, ValidationContext validationContext, String refValue,
JsonNodePath evaluationPath) {
return new JsonSchemaRef(getSupplier(() -> getSchema(parentSchema, validationContext, refValue, evaluationPath), validationContext.getConfig().isCacheRefs()));
}
static Supplier getSupplier(Supplier supplier, boolean cache) {
return cache ? new CachedSupplier<>(supplier) : supplier;
}
static JsonSchema getSchema(JsonSchema parentSchema, ValidationContext validationContext, String refValue,
JsonNodePath evaluationPath) {
JsonSchema refSchema = parentSchema.findSchemaResourceRoot(); // Get the document
JsonSchema current = refSchema;
JsonSchema check = null;
String base = null;
String baseCheck = null;
if (refSchema != null)
base = current.getSchemaLocation().getAbsoluteIri() != null ? current.getSchemaLocation().getAbsoluteIri().toString() : "";
if (current.isRecursiveAnchor()) {
// Check dynamic scope
while (current.getEvaluationParentSchema() != null) {
current = current.getEvaluationParentSchema();
baseCheck = current.getSchemaLocation().getAbsoluteIri() != null ? current.getSchemaLocation().getAbsoluteIri().toString() : "";
if (!base.equals(baseCheck)) {
base = baseCheck;
// Check if it has a dynamic anchor
check = current.findSchemaResourceRoot();
if (check.isRecursiveAnchor()) {
refSchema = check;
}
}
}
}
if (refSchema != null) {
refSchema = refSchema.fromRef(parentSchema, evaluationPath);
}
return refSchema;
}
@Override
public Set validate(ExecutionContext executionContext, JsonNode node, JsonNode rootNode, JsonNodePath instanceLocation) {
debug(logger, executionContext, node, rootNode, instanceLocation);
JsonSchema refSchema = this.schema.getSchema();
if (refSchema == null) {
ValidationMessage validationMessage = message().type(ValidatorTypeCode.RECURSIVE_REF.getValue())
.code("internal.unresolvedRef").message("{0}: Reference {1} cannot be resolved")
.instanceLocation(instanceLocation).evaluationPath(getEvaluationPath())
.arguments(schemaNode.asText()).build();
throw new InvalidSchemaRefException(validationMessage);
}
return refSchema.validate(executionContext, node, rootNode, instanceLocation);
}
@Override
public Set walk(ExecutionContext executionContext, JsonNode node, JsonNode rootNode, JsonNodePath instanceLocation, boolean shouldValidateSchema) {
debug(logger, executionContext, node, rootNode, instanceLocation);
// This is important because if we use same JsonSchemaFactory for creating multiple JSONSchema instances,
// these schemas will be cached along with config. We have to replace the config for cached $ref references
// with the latest config. Reset the config.
JsonSchema refSchema = this.schema.getSchema();
if (refSchema == null) {
ValidationMessage validationMessage = message().type(ValidatorTypeCode.RECURSIVE_REF.getValue())
.code("internal.unresolvedRef").message("{0}: Reference {1} cannot be resolved")
.instanceLocation(instanceLocation).evaluationPath(getEvaluationPath())
.arguments(schemaNode.asText()).build();
throw new InvalidSchemaRefException(validationMessage);
}
if (node == null) {
// Check for circular dependency
SchemaLocation schemaLocation = refSchema.getSchemaLocation();
JsonSchema check = refSchema;
boolean circularDependency = false;
while (check.getEvaluationParentSchema() != null) {
check = check.getEvaluationParentSchema();
if (check.getSchemaLocation().equals(schemaLocation)) {
circularDependency = true;
break;
}
}
if (circularDependency) {
return Collections.emptySet();
}
}
return refSchema.walk(executionContext, node, rootNode, instanceLocation, shouldValidateSchema);
}
public JsonSchemaRef getSchemaRef() {
return this.schema;
}
@Override
public void preloadJsonSchema() {
JsonSchema jsonSchema = null;
try {
jsonSchema = this.schema.getSchema();
} catch (JsonSchemaException e) {
throw e;
} catch (RuntimeException e) {
throw new JsonSchemaException(e);
}
// Check for circular dependency
// Only one cycle is pre-loaded
// The rest of the cycles will load at execution time depending on the input
// data
SchemaLocation schemaLocation = jsonSchema.getSchemaLocation();
JsonSchema check = jsonSchema;
boolean circularDependency = false;
int depth = 0;
while (check.getEvaluationParentSchema() != null) {
depth++;
check = check.getEvaluationParentSchema();
if (check.getSchemaLocation().equals(schemaLocation)) {
circularDependency = true;
break;
}
}
if (this.validationContext.getConfig().isCacheRefs() && !circularDependency
&& depth < this.validationContext.getConfig().getPreloadJsonSchemaRefMaxNestingDepth()) {
jsonSchema.initializeValidators();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy