All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.dimajix.shaded.everit.schema.ValidationException Maven / Gradle / Ivy

There is a newer version: 1.2.0-synapse3.3-spark3.3-hadoop3.3
Show newest version
package com.dimajix.shaded.everit.schema;

import static java.util.Collections.singletonList;
import static java.util.Objects.requireNonNull;
import static java.util.stream.Collectors.toCollection;
import static com.dimajix.shaded.everit.schema.JSONPointer.escape;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

import com.dimajix.shaded.json.JSONArray;
import com.dimajix.shaded.json.JSONObject;

/**
 * Thrown by {@link Schema} subclasses on validation failure.
 */
public class ValidationException extends RuntimeException {
    private static final long serialVersionUID = 6192047123024651924L;



    private static int getViolationCount(List causes) {
        int causeCount = causes.stream().mapToInt(ValidationException::getViolationCount).sum();
        return Math.max(1, causeCount);
    }

    private static List getAllMessages(List causes) {
        List messages = causes.stream()
                .filter(cause -> cause.causingExceptions.isEmpty())
                .map(ValidationException::getMessage)
                .collect(toCollection(ArrayList::new));
        messages.addAll(causes.stream()
                .filter(cause -> !cause.causingExceptions.isEmpty())
                .flatMap(cause -> getAllMessages(cause.getCausingExceptions()).stream())
                .collect(Collectors.toList()));
        return messages;
    }

    /**
     * Sort of static factory method. It is used by {@link ObjectSchema} and {@link ArraySchema} to
     * create {@code ValidationException}s, handling the case of multiple violations occuring during
     * validation.
     * 
    *
  • If {@code failures} is empty, then it doesn't do anything
  • *
  • If {@code failures} contains 1 exception instance, then that will be thrown
  • *
  • Otherwise a new exception instance will be created, its {@link #getViolatedSchema() * violated schema} will be {@code rootFailingSchema}, and its {@link #getCausingExceptions() * causing exceptions} will be the {@code failures} list
  • *
* * @param rootFailingSchema * the schema which detected the {@code failures} * @param failures * list containing validation failures to be thrown by this method */ public static void throwFor(Schema rootFailingSchema, List failures) { int failureCount = failures.size(); if (failureCount == 0) { return; } else if (failureCount == 1) { throw failures.get(0); } else { throw createWrappingException(rootFailingSchema, failures); } } static ValidationException createWrappingException(Schema rootFailingSchema, List failures) { return new InternalValidationException(rootFailingSchema, new StringBuilder("#"), getViolationCount(failures) + " schema violations found", new ArrayList<>(failures), null, rootFailingSchema.getSchemaLocation()); } private final StringBuilder pointerToViolation; private final String schemaLocation; private final transient Schema violatedSchema; private final List causingExceptions; private final String keyword; /** * Deprecated, use {@code ValidationException(Schema, Class, Object)} instead. * * @param expectedType * the expected type * @param actualValue * the violating value */ @Deprecated public ValidationException(Class expectedType, Object actualValue) { this(null, expectedType, actualValue); } /** * Constructor, creates an instance with {@code keyword="type"}. * * @param violatedSchema * the schema instance which detected the schema violation * @param expectedType * the expected type * @param actualValue * the violating value */ public ValidationException(Schema violatedSchema, Class expectedType, Object actualValue) { this(violatedSchema, expectedType, actualValue, "type"); } /** * Constructor for type-mismatch failures. It is usually more convenient to use * {@link #ValidationException(Schema, Class, Object)} instead. * * @param violatedSchema * the schema instance which detected the schema violation * @param expectedType * the expected type * @param actualValue * the violating value * @param keyword * the violating keyword */ @Deprecated public ValidationException(Schema violatedSchema, Class expectedType, Object actualValue, String keyword) { this(violatedSchema, new StringBuilder("#"), "expected type: " + expectedType.getSimpleName() + ", found: " + (actualValue == null ? "null" : actualValue.getClass().getSimpleName()), Collections.emptyList(), keyword, null); } /** * Constructor for type-mismatch failures. It is usually more convenient to use * {@link #ValidationException(Schema, Class, Object)} instead. * * @param violatedSchema * the schema instance which detected the schema violation * @param expectedType * the expected type * @param actualValue * the violating value * @param keyword * the violating keyword * @param schemaLocation * a path denoting the location of the violated keyword in the schema JSON */ public ValidationException(Schema violatedSchema, Class expectedType, Object actualValue, String keyword, String schemaLocation) { this(violatedSchema, new StringBuilder("#"), "expected type: " + expectedType.getSimpleName() + ", found: " + (actualValue == null ? "null" : actualValue.getClass().getSimpleName()), Collections.emptyList(), keyword, schemaLocation); } /** * Constructor. * * @param violatedSchema * the schema instance which detected the schema violation * @param message * the readable exception message * @deprecated use one of the constructors which explicitly specify the violated keyword instead */ @Deprecated public ValidationException(Schema violatedSchema, String message) { this(violatedSchema, new StringBuilder("#"), message, Collections.emptyList()); } /** * Constructor. * * @param violatedSchema * the schama instance which detected the schema violation * @param message * the readable exception message * @param keyword * the violated keyword */ @Deprecated public ValidationException(Schema violatedSchema, String message, String keyword) { this(violatedSchema, new StringBuilder("#"), message, Collections.emptyList(), keyword, null); } /** * Constructor. * * @param violatedSchema * the schama instance which detected the schema violation * @param message * the readable exception message * @param keyword * the violated keyword * @param schemaLocation * the path to the violated schema fragment (from the schema root) */ public ValidationException(Schema violatedSchema, String message, String keyword, String schemaLocation) { this(violatedSchema, new StringBuilder("#"), message, Collections.emptyList(), keyword, schemaLocation); } /*** * Constructor. * * @param violatedSchema * the schema instance which detected the schema violation * @param pointerToViolation * a JSON pointer denoting the part of the document which violates the schema * @param message * the readable exception message * @param causingExceptions * a (possibly empty) list of validation failures. It is used if multiple schema * violations are found by violatedSchema * @deprecated please explicitly specify the violated keyword using one of these constructors: *
    *
  • {@link #ValidationException(Schema, StringBuilder, String, List, String, String)} *
  • {@link #ValidationException(Schema, String, String)} *
  • {@link #ValidationException(Schema, Class, Object, String)} *
*/ @Deprecated ValidationException(Schema violatedSchema, StringBuilder pointerToViolation, String message, List causingExceptions) { this(violatedSchema, pointerToViolation, message, causingExceptions, null, null); } /*** * Constructor. * * @param violatedSchema * the schema instance which detected the schema violation * @param pointerToViolation * a JSON pointer denoting the part of the document which violates the schema * @param message * the readable exception message * @param causingExceptions * a (possibly empty) list of validation failures. It is used if multiple schema * violations are found by violatedSchema * @param keyword * the violated keyword */ ValidationException(Schema violatedSchema, StringBuilder pointerToViolation, String message, List causingExceptions, String keyword, String schemaLocation) { super(message); this.violatedSchema = violatedSchema; this.pointerToViolation = pointerToViolation; this.causingExceptions = Collections.unmodifiableList(causingExceptions); this.keyword = keyword; this.schemaLocation = schemaLocation; } /** * Deprecated, use {@code ValidationException(Schema, String)} instead. * * @param message * readable exception message */ @Deprecated public ValidationException(String message) { this((Schema) null, new StringBuilder("#"), message, Collections.emptyList()); } private ValidationException(StringBuilder pointerToViolation, Schema violatedSchema, String message, List causingExceptions, String keyword, String schemaLocation) { this(violatedSchema, pointerToViolation, message, causingExceptions, keyword, schemaLocation); } /** * Constructor. * * @param violatedSchema * the schema instance which detected the schema violation * @param message * the readable exception message * @param causingExceptions * a (possibly empty) list of validation failures. It is used if multiple schema * violations are found by violatedSchema * @deprecated use one of the constructors which explicitly specify the keyword instead */ @Deprecated public ValidationException(Schema violatedSchema, String message, List causingExceptions) { this(violatedSchema, new StringBuilder("#"), message, causingExceptions); } public List getCausingExceptions() { return causingExceptions; } /** * Returns all messages collected from all violations, including nested causing exceptions. * * @return all messages */ public List getAllMessages() { if (causingExceptions.isEmpty()) { return singletonList(getMessage()); } else { return new ArrayList<>(getAllMessages(causingExceptions)); } } /** * Returns a programmer-readable error description prepended by {@link #getPointerToViolation() * the pointer to the violating fragment} of the JSON document. * * @return the error description */ @Override public String getMessage() { return getPointerToViolation() + ": " + super.getMessage(); } /** * Returns a programmer-readable error description. Unlike {@link #getMessage()} this doesn't * contain the JSON pointer denoting the violating document fragment. * * @return the error description */ public String getErrorMessage() { return super.getMessage(); } /** * A JSON pointer denoting the part of the document which violates the schema. It always points * from the root of the document to the violating data fragment, therefore it always starts with * #. * * @return the JSON pointer */ public String getPointerToViolation() { if (pointerToViolation == null) { return null; } return pointerToViolation.toString(); } public Schema getViolatedSchema() { return violatedSchema; } /** * Creates a new {@code ViolationException} instance based on this one, but with changed * {@link #getPointerToViolation() JSON pointer}. * * @param fragment * the fragment of the JSON pointer to be prepended to existing pointers * @return the new instance */ public ValidationException prepend(String fragment) { return prepend(fragment, this.violatedSchema); } /** * Creates a new {@code ViolationException} instance based on this one, but with changed * {@link #getPointerToViolation() JSON pointer} and {link {@link #getViolatedSchema() violated * schema}. * * @param fragment * the fragment of the JSON pointer to be prepended to existing pointers * @param violatedSchema * the violated schema, which may not be the same as {@link #getViolatedSchema()} * @return the new {@code ViolationException} instance */ public ValidationException prepend(String fragment, Schema violatedSchema) { String escapedFragment = escape(requireNonNull(fragment, "fragment cannot be null")); StringBuilder newPointer = this.pointerToViolation.insert(1, '/').insert(2, escapedFragment); List prependedCausingExceptions = causingExceptions.stream() .map(exc -> exc.prepend(escapedFragment)) .collect(Collectors.toList()); return new InternalValidationException(violatedSchema, newPointer, super.getMessage(), prependedCausingExceptions, this.keyword, this.schemaLocation); } public int getViolationCount() { return getViolationCount(causingExceptions); } public String getKeyword() { return keyword; } /** * Creates a JSON representation of the failure. *

* The returned {@code JSONObject} contains the following keys: *

    *
  • {@code "message"}: a programmer-friendly exception message. This value is a non-nullable * string.
  • *
  • {@code "keyword"}: a JSON Schema keyword which was used in the schema and violated by the * input JSON. This value is a nullable string.
  • *
  • {@code "pointerToViolation"}: a JSON Pointer denoting the path from the root of the * document to the invalid fragment of it. This value is a non-nullable string. See * {@link #getPointerToViolation()}
  • *
  • {@code "causingExceptions"}: is a (possibly empty) array of violations which caused this * exception. See {@link #getCausingExceptions()}
  • *
  • {@code "schemaLocation"}: a string denoting the path to the violated schema keyword in the schema * JSON (since version 1.6.0)
  • *
* * @return a JSON description of the validation error */ public JSONObject toJSON() { JSONObject rval = new JSONObject(); rval.put("keyword", keyword); if (pointerToViolation == null) { rval.put("pointerToViolation", JSONObject.NULL); } else { rval.put("pointerToViolation", getPointerToViolation()); } rval.put("message", super.getMessage()); List causeJsons = causingExceptions.stream() .map(ValidationException::toJSON) .collect(Collectors.toList()); rval.put("causingExceptions", new JSONArray(causeJsons)); if (schemaLocation != null) { rval.put("schemaLocation", schemaLocation); } return rval; } /** * @return a path denoting the location of the violated keyword in the schema * @since 1.6.0 */ public String getSchemaLocation() { return schemaLocation; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; ValidationException that = (ValidationException) o; if (!pointerToViolation.toString().equals(that.pointerToViolation.toString())) return false; if (schemaLocation != null ? !schemaLocation.equals(that.schemaLocation) : that.schemaLocation != null) return false; if (!violatedSchema.equals(that.violatedSchema)) return false; if (!causingExceptions.equals(that.causingExceptions)) return false; return Objects.equals(keyword, that.keyword) && Objects.equals(getMessage(), that.getMessage()); } @Override public int hashCode() { int result = pointerToViolation == null ? 0 : pointerToViolation.hashCode(); result = 31 * result + (schemaLocation != null ? schemaLocation.hashCode() : 0); result = 31 * result + (violatedSchema == null ? 0 : violatedSchema.hashCode()); result = 31 * result + (causingExceptions == null ? 0 : causingExceptions.hashCode()); result = 31 * result + (keyword == null ? 0 : keyword.hashCode()); return result; } ValidationException copy() { return new ValidationException(pointerToViolation, violatedSchema, super.getMessage(), causingExceptions, keyword, schemaLocation); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy