org.everit.json.schema.CombinedSchema Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of everit-json-schema Show documentation
Show all versions of everit-json-schema Show documentation
Implementation of the JSON Schema Core Draft v4 specification built with the org.json API
package org.everit.json.schema;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;
/**
* Validator for {@code allOf}, {@code oneOf}, {@code anyOf} schemas.
*/
public class CombinedSchema extends Schema {
/**
* Builder class for {@link CombinedSchema}.
*/
public static class Builder extends Schema.Builder {
private ValidationCriterion criterion;
private Collection subschemas = new ArrayList<>();
private boolean synthetic;
@Override
public CombinedSchema build() {
return new CombinedSchema(this);
}
public Builder criterion(ValidationCriterion criterion) {
this.criterion = criterion;
return this;
}
public Builder subschema(Schema subschema) {
this.subschemas.add(subschema);
return this;
}
public Builder subschemas(Collection subschemas) {
this.subschemas = subschemas;
return this;
}
public Builder isSynthetic(boolean synthetic) {
this.synthetic = synthetic;
return this;
}
}
/**
* Validation criterion.
*/
@FunctionalInterface
public interface ValidationCriterion {
/**
* Throws a {@link ValidationException} if the implemented criterion is not fulfilled by the
* {@code subschemaCount} and the {@code matchingSubschemaCount}.
*
* @param subschemaCount
* the total number of checked subschemas
* @param matchingSubschemaCount
* the number of subschemas which successfully validated the subject (did not throw
* {@link ValidationException})
*/
void validate(int subschemaCount, int matchingSubschemaCount);
}
/**
* Validation criterion for {@code allOf} schemas.
*/
public static final ValidationCriterion ALL_CRITERION = new ValidationCriterion() {
@Override
public void validate(int subschemaCount, int matchingCount) {
if (matchingCount < subschemaCount) {
throw new ValidationException(null,
format("only %d subschema matches out of %d", matchingCount, subschemaCount),
"allOf"
);
}
}
@Override
public String toString() {
return "allOf";
}
};
/**
* Validation criterion for {@code anyOf} schemas.
*/
public static final ValidationCriterion ANY_CRITERION = new ValidationCriterion() {
@Override
public void validate(int subschemaCount, int matchingCount) {
if (matchingCount == 0) {
throw new ValidationException(null, format(
"no subschema matched out of the total %d subschemas",
subschemaCount), "anyOf");
}
}
@Override
public String toString() {
return "anyOf";
}
};
/**
* Validation criterion for {@code oneOf} schemas.
*/
public static final ValidationCriterion ONE_CRITERION =
new ValidationCriterion() {
@Override
public void validate(int subschemaCount, int matchingCount) {
if (matchingCount != 1) {
throw new ValidationException(null, format("%d subschemas matched instead of one",
matchingCount), "oneOf");
}
}
@Override public String toString() {
return "oneOf";
}
};
public static Builder allOf(Collection schemas) {
return builder(schemas).criterion(ALL_CRITERION);
}
public static Builder anyOf(Collection schemas) {
return builder(schemas).criterion(ANY_CRITERION);
}
public static Builder builder() {
return new Builder();
}
public static Builder builder(Collection subschemas) {
return new Builder().subschemas(subschemas);
}
public static Builder oneOf(Collection schemas) {
return builder(schemas).criterion(ONE_CRITERION);
}
private final boolean synthetic;
private final Collection subschemas;
private final ValidationCriterion criterion;
/**
* Constructor.
*
* @param builder
* the builder containing the validation criterion and the subschemas to be checked
*/
public CombinedSchema(Builder builder) {
super(builder);
this.synthetic = builder.synthetic;
this.criterion = requireNonNull(builder.criterion, "criterion cannot be null");
this.subschemas = sortByCombinedFirst(requireNonNull(builder.subschemas, "subschemas cannot be null"));
}
private static int compareBySchemaType(Schema lschema, Schema rschema) {
return lschema instanceof CombinedSchema ?
(rschema instanceof CombinedSchema ? 0 : -1) :
(rschema instanceof CombinedSchema ? 1 : 0);
}
// ensure subschemas of type CombinedSchema are always visited first
private static Collection sortByCombinedFirst(Collection schemas) {
return schemas.stream()
.sorted(CombinedSchema::compareBySchemaType)
.collect(Collectors.toCollection(ArrayList::new));
}
public ValidationCriterion getCriterion() {
return criterion;
}
public Collection getSubschemas() {
return subschemas;
}
boolean isSynthetic() {
return synthetic;
}
@Override void accept(Visitor visitor) {
visitor.visitCombinedSchema(this);
}
@Override
public boolean definesProperty(String field) {
List matching = new ArrayList<>();
for (Schema subschema : subschemas) {
if (subschema.definesProperty(field)) {
matching.add(subschema);
}
}
try {
criterion.validate(subschemas.size(), matching.size());
} catch (ValidationException e) {
return false;
}
return true;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o instanceof CombinedSchema) {
CombinedSchema that = (CombinedSchema) o;
return that.canEqual(this) &&
Objects.equals(subschemas, that.subschemas) &&
Objects.equals(criterion, that.criterion) &&
synthetic == that.synthetic &&
super.equals(that);
} else {
return false;
}
}
@Override
public int hashCode() {
return Objects.hash(super.hashCode(), subschemas, criterion, synthetic);
}
@Override
protected boolean canEqual(Object other) {
return other instanceof CombinedSchema;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy