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

software.amazon.smithy.model.validation.validators.HttpQueryTraitValidator Maven / Gradle / Ivy

/*
 * Copyright 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 *  http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file 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 software.amazon.smithy.model.validation.validators;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import software.amazon.smithy.model.Model;
import software.amazon.smithy.model.knowledge.OperationIndex;
import software.amazon.smithy.model.pattern.UriPattern;
import software.amazon.smithy.model.shapes.MemberShape;
import software.amazon.smithy.model.shapes.OperationShape;
import software.amazon.smithy.model.shapes.Shape;
import software.amazon.smithy.model.shapes.StructureShape;
import software.amazon.smithy.model.traits.HttpQueryTrait;
import software.amazon.smithy.model.traits.HttpTrait;
import software.amazon.smithy.model.validation.AbstractValidator;
import software.amazon.smithy.model.validation.ValidationEvent;
import software.amazon.smithy.model.validation.ValidationUtils;

/**
 * Validates that httpQuery trait bindings are case-sensitively unique.
 */
public final class HttpQueryTraitValidator extends AbstractValidator {

    @Override
    public List validate(Model model) {
        if (!model.isTraitApplied(HttpQueryTrait.class)) {
            return Collections.emptyList();
        } else {
            return validateBindings(getQueryBindings(model), getStructureToOperations(model));
        }
    }

    private Map>> getQueryBindings(Model model) {
        Map>> queryBindings = new HashMap<>();

        // Find all members in the model that have the HttpQuery trait.
        for (MemberShape member : model.getMemberShapesWithTrait(HttpQueryTrait.class)) {
            // Get the structure of the member. Validation events are going to be
            // applied to the structure and not to members.
            model.getShape(member.getContainer()).flatMap(Shape::asStructureShape).ifPresent(structure -> {
                HttpQueryTrait trait = member.expectTrait(HttpQueryTrait.class);
                queryBindings
                        .computeIfAbsent(structure, s -> new HashMap<>())
                        .computeIfAbsent(trait.getValue(), v -> new HashSet<>())
                        .add(member.getMemberName());
            });
        }

        return queryBindings;
    }

    private List validateBindings(
        Map>> queryBindings,
        Map> structureToOperations
    ) {
        List events = new ArrayList<>();

        for (Map.Entry>> entry : queryBindings.entrySet()) {
            for (Map.Entry> paramsToMembers : entry.getValue().entrySet()) {
                // Emit if there are bindings on this shape for the same query string parameter.
                if (paramsToMembers.getValue().size() > 1) {
                    events.add(error(entry.getKey(), String.format(
                            "`httpQuery` parameter name binding conflicts found for the `%s` parameter in the "
                            + "following structure members: %s",
                            paramsToMembers.getKey(), ValidationUtils.tickedList(paramsToMembers.getValue()))));
                }
            }

            List operations = structureToOperations.getOrDefault(entry.getKey(),
                                                                                 Collections.emptyList());
            for (OperationShape operation : operations) {
                UriPattern pattern = operation.expectTrait(HttpTrait.class).getUri();
                for (Map.Entry literalEntry : pattern.getQueryLiterals().entrySet()) {
                    String literalKey = literalEntry.getKey();
                    if (entry.getValue().containsKey(literalKey)) {
                        events.add(error(entry.getKey(), String.format(
                            "`httpQuery` name `%s` conflicts with the `http` trait of the `%s` operation: `%s`",
                            literalKey, operation.getId(), pattern)));
                    }
                }
            }
        }

        return events;
    }

    private Map> getStructureToOperations(Model model) {
        OperationIndex index = OperationIndex.of(model);
        Map> structureToOperations = new HashMap<>();
        for (OperationShape operation : model.getOperationShapesWithTrait(HttpTrait.class)) {
            index.getInput(operation)
                 .ifPresent(structure -> structureToOperations
                     .computeIfAbsent(structure, key -> new ArrayList<>())
                     .add(operation));
        }
        return structureToOperations;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy