-
Notifications
You must be signed in to change notification settings - Fork 218
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add warnings for ignored HTTP member bindings
This commit adds a new validator that emits a warning when a member has an HTTP member binding trait applied and its container is referenced through a relationship where it would be ignored. Spec clarification of this behavior has been added.
- Loading branch information
Showing
8 changed files
with
312 additions
and
34 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
170 changes: 170 additions & 0 deletions
170
.../software/amazon/smithy/model/validation/validators/HttpBindingTraitIgnoredValidator.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,170 @@ | ||
/* | ||
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. | ||
* SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
package software.amazon.smithy.model.validation.validators; | ||
|
||
import static java.lang.String.format; | ||
|
||
import java.util.ArrayList; | ||
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.NeighborProviderIndex; | ||
import software.amazon.smithy.model.neighbor.NeighborProvider; | ||
import software.amazon.smithy.model.neighbor.Relationship; | ||
import software.amazon.smithy.model.neighbor.RelationshipType; | ||
import software.amazon.smithy.model.shapes.MemberShape; | ||
import software.amazon.smithy.model.shapes.ShapeId; | ||
import software.amazon.smithy.model.shapes.StructureShape; | ||
import software.amazon.smithy.model.traits.HttpHeaderTrait; | ||
import software.amazon.smithy.model.traits.HttpLabelTrait; | ||
import software.amazon.smithy.model.traits.HttpPayloadTrait; | ||
import software.amazon.smithy.model.traits.HttpPrefixHeadersTrait; | ||
import software.amazon.smithy.model.traits.HttpQueryParamsTrait; | ||
import software.amazon.smithy.model.traits.HttpQueryTrait; | ||
import software.amazon.smithy.model.traits.HttpResponseCodeTrait; | ||
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.utils.ListUtils; | ||
import software.amazon.smithy.utils.StringUtils; | ||
|
||
/** | ||
* Emits warnings when a structure member has an HTTP binding trait that will be ignored | ||
* in some contexts to which it is bound. | ||
* * When httpLabel, httpQueryParams, or httpQuery is applied to a member of a shape that | ||
* is not used as operation inputs. | ||
* * When httpResponseCode is applied to a member of a shape that is not used as an | ||
* operation output. | ||
* * When any other HTTP member binding trait is applied to a member of a shape that is | ||
* not used as a top-level operation input, output, or error. | ||
*/ | ||
public class HttpBindingTraitIgnoredValidator extends AbstractValidator { | ||
private static final List<ShapeId> IGNORED_OUTSIDE_INPUT = ListUtils.of( | ||
HttpLabelTrait.ID, | ||
HttpQueryParamsTrait.ID, | ||
HttpQueryTrait.ID); | ||
private static final List<ShapeId> IGNORED_OUTSIDE_OUTPUT = ListUtils.of( | ||
HttpResponseCodeTrait.ID); | ||
private static final List<ShapeId> HTTP_MEMBER_BINDING_TRAITS = ListUtils.of( | ||
HttpHeaderTrait.ID, | ||
HttpLabelTrait.ID, | ||
HttpPayloadTrait.ID, | ||
HttpPrefixHeadersTrait.ID, | ||
HttpQueryParamsTrait.ID, | ||
HttpQueryTrait.ID, | ||
HttpResponseCodeTrait.ID); | ||
|
||
@Override | ||
public List<ValidationEvent> validate(Model model) { | ||
List<ValidationEvent> events = new ArrayList<>(); | ||
for (MemberShape memberShape : model.getMemberShapes()) { | ||
// Retain all traits that are HTTP member binding. | ||
// Keep the trait instance around so that it can be used it later for source location. | ||
Map<ShapeId, Trait> traits = new HashMap<>(memberShape.getAllTraits()); | ||
for (ShapeId traitId : memberShape.getAllTraits().keySet()) { | ||
if (!HTTP_MEMBER_BINDING_TRAITS.contains(traitId)) { | ||
traits.remove(traitId); | ||
} | ||
} | ||
|
||
// The traits set is now the HTTP binding traits that are ignored outside | ||
// the top level of an operation's components. | ||
if (!traits.isEmpty()) { | ||
StructureShape containerShape = model.expectShape(memberShape.getContainer(), StructureShape.class); | ||
NeighborProvider reverse = NeighborProviderIndex.of(model).getReverseProvider(); | ||
List<Relationship> relationships = reverse.getNeighbors(containerShape); | ||
|
||
// All relationships and trait possibilities are checked at once to de-duplicate | ||
// several parts of the iteration logic. | ||
events.addAll(checkRelationships(memberShape, traits, relationships)); | ||
} | ||
} | ||
return events; | ||
} | ||
|
||
private List<ValidationEvent> checkRelationships( | ||
MemberShape memberShape, | ||
Map<ShapeId, Trait> traits, | ||
List<Relationship> relationships | ||
) { | ||
// Prepare which traits need relationship tracking for. | ||
Set<ShapeId> ignoredOutsideInputTraits = new HashSet<>(traits.keySet()); | ||
ignoredOutsideInputTraits.retainAll(IGNORED_OUTSIDE_INPUT); | ||
Set<ShapeId> ignoredOutsideOutputTraits = new HashSet<>(traits.keySet()); | ||
ignoredOutsideOutputTraits.retainAll(IGNORED_OUTSIDE_OUTPUT); | ||
|
||
// Store relationships so we can emit one event per ignored binding. | ||
Map<RelationshipType, ShapeId> relationshipToContainer = new HashMap<>(); | ||
List<ValidationEvent> events = new ArrayList<>(); | ||
for (Relationship relationship : relationships) { | ||
// Skip members of the container. | ||
if (relationship.getRelationshipType() == RelationshipType.MEMBER_CONTAINER) { | ||
continue; | ||
} | ||
|
||
// Track if we've got a non-input relationship and a trait that's ignored outside input. | ||
// Continue so we don't emit a duplicate for non-top-level. | ||
if (relationship.getRelationshipType() != RelationshipType.INPUT | ||
&& !ignoredOutsideInputTraits.isEmpty() | ||
) { | ||
relationshipToContainer.put(relationship.getRelationshipType(), relationship.getNeighborShapeId()); | ||
continue; | ||
} | ||
|
||
// Track if we've got a non-output relationship and a trait that's ignored outside output. | ||
// Continue so we don't emit a duplicate for non-top-level. | ||
if (relationship.getRelationshipType() != RelationshipType.OUTPUT | ||
&& !ignoredOutsideOutputTraits.isEmpty() | ||
) { | ||
relationshipToContainer.put(relationship.getRelationshipType(), relationship.getNeighborShapeId()); | ||
continue; | ||
} | ||
|
||
// Track if there are non-top-level relationship and any HTTP member binding trait. | ||
if (relationship.getRelationshipType() != RelationshipType.INPUT | ||
&& relationship.getRelationshipType() != RelationshipType.OUTPUT | ||
&& relationship.getRelationshipType() != RelationshipType.ERROR | ||
) { | ||
relationshipToContainer.put(relationship.getRelationshipType(), relationship.getNeighborShapeId()); | ||
} | ||
} | ||
|
||
// If we detected invalid relationships, build the right event message based | ||
// on the ignored traits. | ||
if (!relationshipToContainer.isEmpty()) { | ||
if (!ignoredOutsideInputTraits.isEmpty()) { | ||
ShapeId traitId = ignoredOutsideInputTraits.iterator().next(); | ||
events.add(emit("input", memberShape, traits, traitId, relationshipToContainer)); | ||
|
||
} else if (!ignoredOutsideOutputTraits.isEmpty()) { | ||
ShapeId traitId = ignoredOutsideOutputTraits.iterator().next(); | ||
events.add(emit("output", memberShape, traits, traitId, relationshipToContainer)); | ||
} else { | ||
// The traits list is always non-empty here, so just grab the first. | ||
ShapeId traitId = traits.keySet().iterator().next(); | ||
events.add(emit("top-level", memberShape, traits, traitId, relationshipToContainer)); | ||
} | ||
} | ||
|
||
return events; | ||
} | ||
|
||
private ValidationEvent emit( | ||
String type, | ||
MemberShape memberShape, | ||
Map<ShapeId, Trait> traits, | ||
ShapeId traitId, | ||
Map<RelationshipType, ShapeId> relationshipToContainer | ||
) { | ||
return warning(memberShape, traits.get(traitId), | ||
format("This member has the `%s` trait applied but its container has the following " | ||
+ "non-%s relationships: [%s]", traitId, type, relationshipToContainer), | ||
StringUtils.capitalize(type.replace("-", ""))); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.