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

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

/*
 * Copyright 2021 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.List;
import software.amazon.smithy.model.Model;
import software.amazon.smithy.model.shapes.MemberShape;
import software.amazon.smithy.model.shapes.Shape;
import software.amazon.smithy.model.shapes.StructureShape;
import software.amazon.smithy.model.traits.HttpQueryParamsTrait;
import software.amazon.smithy.model.traits.HttpQueryTrait;
import software.amazon.smithy.model.traits.Trait;
import software.amazon.smithy.model.validation.AbstractValidator;
import software.amazon.smithy.model.validation.ValidationEvent;
import software.amazon.smithy.model.validation.ValidationUtils;

/**
 * When the `httpQueryParams` trait is used, this validator emits a NOTE when another member of the container shape
 * applies the `httpQuery` trait which may result in a conflict within the query string.
 */
public final class HttpQueryParamsTraitValidator extends AbstractValidator {
    @Override
    public List validate(Model model) {
        if (!model.isTraitApplied(HttpQueryParamsTrait.class)) {
            return Collections.emptyList();
        } else {
            return validateQueryTraitUsage(model);
        }
    }

    private List validateQueryTraitUsage(Model model) {
        List events = new ArrayList<>();

        for (MemberShape member : model.getMemberShapesWithTrait(HttpQueryParamsTrait.class)) {
            model.getShape(member.getContainer())
                    .flatMap(Shape::asStructureShape)
                    .ifPresent(structure -> {
                        // Gather the names of member shapes, as strings, that apply HttpQuery traits
                        List queryShapes = getMembersWithTrait(structure, HttpQueryTrait.class);
                        if (queryShapes.size() > 0) {
                            events.add(createNote(structure, member.getMemberName(), queryShapes));
                        }
                    });
        }

        return events;
    }

    private List getMembersWithTrait(StructureShape structure, Class trait) {
        List members = new ArrayList<>();
        for (MemberShape member : structure.members()) {
            if (member.hasTrait(trait)) {
                members.add(member.getMemberName());
            }
        }
        return members;
    }

    private ValidationEvent createNote(Shape target, String queryParamsShape, List queryShapes) {
        return note(target, String.format("Structure member `%s` is marked with the `httpQueryParams` trait, and "
                + "`httpQuery` traits are applied to the following members: %s. The service will not be able to "
                + "disambiguate between query string parameters intended for the `%s` member and those explicitly "
                + "bound to the `httpQuery` members.", queryParamsShape, ValidationUtils.tickedList(queryShapes),
                queryParamsShape));
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy