From 1b5a65af8b655354b5723ad3f8d6fb7e508702a1 Mon Sep 17 00:00:00 2001 From: Jordon Phillips Date: Fri, 8 Dec 2023 14:13:04 +0100 Subject: [PATCH] Add ability to find input/output/error bindings This updates the operation index to allow you to easily find all the operations for which a given shape is used as an input or output and all the operations or services for which a given shape is used as an error. --- .../model/knowledge/OperationIndex.java | 47 ++++++++++- .../model/knowledge/OperationIndexTest.java | 46 ++++++++++- .../model/knowledge/operation-index-test.json | 81 ------------------- .../knowledge/operation-index-test.smithy | 58 +++++++++++++ 4 files changed, 148 insertions(+), 84 deletions(-) delete mode 100644 smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.json create mode 100644 smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.smithy diff --git a/smithy-model/src/main/java/software/amazon/smithy/model/knowledge/OperationIndex.java b/smithy-model/src/main/java/software/amazon/smithy/model/knowledge/OperationIndex.java index 531c20765f1..0b0fc2accd9 100644 --- a/smithy-model/src/main/java/software/amazon/smithy/model/knowledge/OperationIndex.java +++ b/smithy-model/src/main/java/software/amazon/smithy/model/knowledge/OperationIndex.java @@ -18,6 +18,7 @@ import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; +import java.util.HashSet; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; @@ -37,6 +38,7 @@ import software.amazon.smithy.model.traits.OutputTrait; import software.amazon.smithy.model.traits.UnitTypeTrait; import software.amazon.smithy.utils.ListUtils; +import software.amazon.smithy.utils.SetUtils; /** * Index of operation IDs to their resolved input, output, and error @@ -50,11 +52,20 @@ public final class OperationIndex implements KnowledgeIndex { private final Map inputs = new HashMap<>(); private final Map outputs = new HashMap<>(); private final Map> errors = new HashMap<>(); + private final Map> boundInputOperations = new HashMap<>(); + private final Map> boundOutputOperations = new HashMap<>(); + private final Map> boundErrorShapes = new HashMap<>(); public OperationIndex(Model model) { for (OperationShape operation : model.getOperationShapes()) { - getStructure(model, operation.getInputShape()).ifPresent(shape -> inputs.put(operation.getId(), shape)); - getStructure(model, operation.getOutputShape()).ifPresent(shape -> outputs.put(operation.getId(), shape)); + getStructure(model, operation.getInputShape()).ifPresent(shape -> { + inputs.put(operation.getId(), shape); + boundInputOperations.computeIfAbsent(shape.getId(), id -> new HashSet<>()).add(operation); + }); + getStructure(model, operation.getOutputShape()).ifPresent(shape -> { + outputs.put(operation.getId(), shape); + boundOutputOperations.computeIfAbsent(shape.getId(), id -> new HashSet<>()).add(operation); + }); addErrorsFromShape(model, operation.getId(), operation.getErrors()); } @@ -65,8 +76,10 @@ public OperationIndex(Model model) { private void addErrorsFromShape(Model model, ShapeId source, List errorShapeIds) { List errorShapes = new ArrayList<>(errorShapeIds.size()); + Shape sourceShape = model.expectShape(source); for (ShapeId target : errorShapeIds) { model.getShape(target).flatMap(Shape::asStructureShape).ifPresent(errorShapes::add); + boundErrorShapes.computeIfAbsent(target, id -> new HashSet<>()).add(sourceShape); } errors.put(source, errorShapes); } @@ -158,6 +171,16 @@ public boolean isInputStructure(ToShapeId structureId) { return false; } + /** + * Gets all the operations that bind the given shape as input. + * + * @param input The structure that may be used as input. + * @return Returns a set of operations that bind the given input shape. + */ + public Set getInputBindings(ToShapeId input) { + return SetUtils.copyOf(boundInputOperations.getOrDefault(input.toShapeId(), Collections.emptySet())); + } + /** * Gets the optional output structure of an operation, and returns an * empty optional if the output targets {@code smithy.api#Unit}. @@ -241,6 +264,16 @@ public boolean isOutputStructure(ToShapeId structureId) { return false; } + /** + * Gets all the operations that bind the given shape as output. + * + * @param output The structure that may be used as output. + * @return Returns a set of operations that bind the given output shape. + */ + public Set getOutputBindings(ToShapeId output) { + return SetUtils.copyOf(boundOutputOperations.getOrDefault(output.toShapeId(), Collections.emptySet())); + } + /** * Gets the list of error structures defined on an operation. * @@ -275,4 +308,14 @@ public List getErrors(ToShapeId service, ToShapeId operation) { private Optional getStructure(Model model, ToShapeId id) { return model.getShape(id.toShapeId()).flatMap(Shape::asStructureShape); } + + /** + * Gets all the operations and services that bind the given shape as an error. + * + * @param error The structure that may be used as an error. + * @return Returns a set of operations and services that bind the given error shape. + */ + public Set getErrorBindings(ToShapeId error) { + return SetUtils.copyOf(boundErrorShapes.getOrDefault(error.toShapeId(), Collections.emptySet())); + } } diff --git a/smithy-model/src/test/java/software/amazon/smithy/model/knowledge/OperationIndexTest.java b/smithy-model/src/test/java/software/amazon/smithy/model/knowledge/OperationIndexTest.java index a7770907000..435abfa3250 100644 --- a/smithy-model/src/test/java/software/amazon/smithy/model/knowledge/OperationIndexTest.java +++ b/smithy-model/src/test/java/software/amazon/smithy/model/knowledge/OperationIndexTest.java @@ -23,15 +23,18 @@ import java.util.Collections; import java.util.Optional; +import java.util.Set; import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.Test; import software.amazon.smithy.model.Model; +import software.amazon.smithy.model.shapes.OperationShape; import software.amazon.smithy.model.shapes.ServiceShape; import software.amazon.smithy.model.shapes.Shape; import software.amazon.smithy.model.shapes.ShapeId; import software.amazon.smithy.model.shapes.StructureShape; import software.amazon.smithy.model.traits.UnitTypeTrait; +import software.amazon.smithy.utils.SetUtils; public class OperationIndexTest { @@ -40,7 +43,7 @@ public class OperationIndexTest { @BeforeAll public static void before() { model = Model.assembler() - .addImport(OperationIndexTest.class.getResource("operation-index-test.json")) + .addImport(OperationIndexTest.class.getResource("operation-index-test.smithy")) .assemble() .unwrap(); } @@ -103,6 +106,18 @@ public void determinesIfShapeIsUsedAsInput() { assertThat(opIndex.isInputStructure(output), is(false)); } + @Test + public void getsInputBindings() { + OperationIndex index = OperationIndex.of(model); + Set actual = index.getInputBindings(ShapeId.from("ns.foo#Input")); + Set expected = SetUtils.of( + model.expectShape(ShapeId.from("ns.foo#B"), OperationShape.class), + model.expectShape(ShapeId.from("ns.foo#C"), OperationShape.class) + ); + assertThat(actual, equalTo(expected)); + assertThat(index.getInputBindings(ShapeId.from("ns.foo#Output")), empty()); + } + @Test public void determinesIfShapeIsUsedAsOutput() { OperationIndex opIndex = OperationIndex.of(model); @@ -113,6 +128,18 @@ public void determinesIfShapeIsUsedAsOutput() { assertThat(opIndex.isOutputStructure(input), is(false)); } + @Test + public void getsOutputBindings() { + OperationIndex index = OperationIndex.of(model); + Set actual = index.getOutputBindings(ShapeId.from("ns.foo#Output")); + Set expected = SetUtils.of( + model.expectShape(ShapeId.from("ns.foo#B"), OperationShape.class), + model.expectShape(ShapeId.from("ns.foo#C"), OperationShape.class) + ); + assertThat(actual, equalTo(expected)); + assertThat(index.getOutputBindings(ShapeId.from("ns.foo#Input")), empty()); + } + @Test public void getsOperationErrorsAndInheritedErrors() { OperationIndex opIndex = OperationIndex.of(model); @@ -130,4 +157,21 @@ public void getsOperationErrorsAndInheritedErrors() { assertThat(opIndex.getErrors(b), containsInAnyOrder(error1, error2)); assertThat(opIndex.getErrors(service, b), containsInAnyOrder(error1, error2, common1, common2)); } + + @Test + public void getsErrorBindings() { + OperationIndex index = OperationIndex.of(model); + Set actual = index.getErrorBindings(ShapeId.from("ns.foo#CommonError1")); + Set expected = SetUtils.of( + model.expectShape(ShapeId.from("ns.foo#MyService"), ServiceShape.class), + model.expectShape(ShapeId.from("ns.foo#C"), OperationShape.class) + ); + assertThat(actual, equalTo(expected)); + + actual = index.getErrorBindings(ShapeId.from("ns.foo#Error1")); + expected = SetUtils.of(model.expectShape(ShapeId.from("ns.foo#B"), OperationShape.class)); + assertThat(actual, equalTo(expected)); + + assertThat(index.getOutputBindings(ShapeId.from("ns.foo#UnusedError")), empty()); + } } diff --git a/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.json b/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.json deleted file mode 100644 index 77f42787713..00000000000 --- a/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.json +++ /dev/null @@ -1,81 +0,0 @@ -{ - "smithy": "2.0", - "shapes": { - "ns.foo#MyService": { - "type": "service", - "version": "2017-01-17", - "operations": [ - { - "target": "ns.foo#A" - }, - { - "target": "ns.foo#B" - } - ], - "errors": [ - { - "target": "ns.foo#CommonError1" - }, - { - "target": "ns.foo#CommonError2" - } - ] - }, - "ns.foo#A": { - "type": "operation", - "traits": { - "smithy.api#readonly": {} - } - }, - "ns.foo#B": { - "type": "operation", - "input": { - "target": "ns.foo#Input" - }, - "output": { - "target": "ns.foo#Output" - }, - "errors": [ - { - "target": "ns.foo#Error1" - }, - { - "target": "ns.foo#Error2" - } - ], - "traits": { - "smithy.api#readonly": {} - } - }, - "ns.foo#Input": { - "type": "structure" - }, - "ns.foo#Output": { - "type": "structure" - }, - "ns.foo#Error1": { - "type": "structure", - "traits": { - "smithy.api#error": "client" - } - }, - "ns.foo#Error2": { - "type": "structure", - "traits": { - "smithy.api#error": "server" - } - }, - "ns.foo#CommonError1": { - "type": "structure", - "traits": { - "smithy.api#error": "server" - } - }, - "ns.foo#CommonError2": { - "type": "structure", - "traits": { - "smithy.api#error": "server" - } - } - } -} diff --git a/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.smithy b/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.smithy new file mode 100644 index 00000000000..d633de21b63 --- /dev/null +++ b/smithy-model/src/test/resources/software/amazon/smithy/model/knowledge/operation-index-test.smithy @@ -0,0 +1,58 @@ +$version: "2.0" + +namespace ns.foo + +service MyService { + version: "2017-01-17" + operations: [ + A + B + ] + errors: [ + CommonError1 + CommonError2 + ] +} + +@readonly +operation A { + input: Unit + output: Unit +} + +@readonly +operation B { + input: Input + output: Output + errors: [ + Error1 + Error2 + ] +} + +operation C { + input: Input + output: Output, + errors: [ + CommonError1 + ] +} + +@error("server") +structure CommonError1 {} + +@error("server") +structure CommonError2 {} + +@error("client") +structure Error1 {} + +@error("server") +structure Error2 {} + +@error("client") +structure UnusedError {} + +structure Input {} + +structure Output {}