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

io.apicurio.registry.rules.validity.ApicurioDataModelContentValidator Maven / Gradle / Ivy

package io.apicurio.registry.rules.validity;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import io.apicurio.datamodels.Library;
import io.apicurio.datamodels.TraverserDirection;
import io.apicurio.datamodels.models.Document;
import io.apicurio.datamodels.models.Node;
import io.apicurio.datamodels.models.Referenceable;
import io.apicurio.datamodels.models.visitors.AllNodeVisitor;
import io.apicurio.datamodels.validation.ValidationProblem;
import io.apicurio.registry.content.TypedContent;
import io.apicurio.registry.content.util.ContentTypeUtil;
import io.apicurio.registry.rest.v3.beans.ArtifactReference;
import io.apicurio.registry.rules.RuleViolation;
import io.apicurio.registry.rules.RuleViolationException;
import io.apicurio.registry.rules.integrity.IntegrityLevel;
import io.apicurio.registry.types.RuleType;

import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;

/**
 * A content validator implementation for the OpenAPI and AsyncAPI content types.
 */
public abstract class ApicurioDataModelContentValidator implements ContentValidator {

    /**
     * @see io.apicurio.registry.rules.validity.ContentValidator#validate(ValidityLevel, TypedContent, Map)
     */
    @Override
    public void validate(ValidityLevel level, TypedContent content,
            Map resolvedReferences) throws RuleViolationException {
        Document document = null;
        if (level == ValidityLevel.SYNTAX_ONLY || level == ValidityLevel.FULL) {
            try {
                JsonNode node = ContentTypeUtil.parseJsonOrYaml(content);
                document = Library.readDocument((ObjectNode) node);
            } catch (Exception e) {
                throw new RuleViolationException("Syntax violation for " + getDataModelType() + " artifact.",
                        RuleType.VALIDITY, level.name(), e);
            }
        }

        if (level == ValidityLevel.FULL) {
            List problems = Library.validate(document, null);
            if (!problems.isEmpty()) {
                Set causes = problems.stream()
                        .map(problem -> new RuleViolation(problem.message, problem.nodePath.toString()))
                        .collect(Collectors.toSet());
                throw new RuleViolationException(
                        "The " + getDataModelType() + " artifact is not semantically valid. "
                                + problems.size() + " problems found.",
                        RuleType.VALIDITY, level.name(), causes);
            }
        }
    }

    /**
     * @see io.apicurio.registry.rules.validity.ContentValidator#validateReferences(TypedContent, List)
     */
    @Override
    public void validateReferences(TypedContent content, List references)
            throws RuleViolationException {
        Set mappedRefs = references.stream().map(ref -> ref.getName()).collect(Collectors.toSet());
        Set all$refs = getAll$refs(content);
        Set violations = all$refs.stream().filter(ref -> !mappedRefs.contains(ref))
                .map(missingRef -> {
                    return new RuleViolation("Unmapped reference detected.", missingRef);
                }).collect(Collectors.toSet());
        if (!violations.isEmpty()) {
            throw new RuleViolationException("Unmapped reference(s) detected.", RuleType.INTEGRITY,
                    IntegrityLevel.ALL_REFS_MAPPED.name(), violations);
        }
    }

    private Set getAll$refs(TypedContent content) {
        try {
            RefFinder refFinder = new RefFinder();
            JsonNode node = ContentTypeUtil.parseJsonOrYaml(content);
            Document document = Library.readDocument((ObjectNode) node);
            Library.visitTree(document, refFinder, TraverserDirection.down);
            return refFinder.references;
        } catch (Exception e) {
            return Collections.emptySet();
        }
    }

    /**
     * Returns the type of data model being validated. Subclasses must implement.
     */
    protected abstract String getDataModelType();

    private static class RefFinder extends AllNodeVisitor {

        Set references = new HashSet<>();

        /**
         * @see io.apicurio.datamodels.models.visitors.AllNodeVisitor#visitNode(io.apicurio.datamodels.models.Node)
         */
        @Override
        protected void visitNode(Node node) {
            if (node instanceof Referenceable) {
                String theRef = ((Referenceable) node).get$ref();
                if (theRef != null && !theRef.startsWith("#/")) {
                    references.add(theRef);
                }
            }
        }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy