Skip to content

Commit 86ff69b

Browse files
committed
Add transform that can remove invalid defaults
If a default trait is incompatible with the range trait of the member or the member target, then the shape is essentially in a state where the member is invalid: omit a value for the member at it is automatically incompatible with the range trait of the member. This transformer finds such cases and removes invalid default values. It is likely that services with such a model did not intend for the members to actually have a default value, and this typically only happens with members that have a default value of `0`.
1 parent 6e7cc38 commit 86ff69b

File tree

6 files changed

+214
-18
lines changed

6 files changed

+214
-18
lines changed

smithy-model/src/main/java/software/amazon/smithy/model/transform/ModelTransformer.java

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -656,4 +656,14 @@ public Model addClientOptional(Model model, boolean applyWhenNoDefaultValue) {
656656
public Model downgradeToV1(Model model) {
657657
return new DowngradeToV1().transform(this, model);
658658
}
659+
660+
/**
661+
* Remove default traits if the default conflicts with the range trait of the shape.
662+
*
663+
* @param model Model to transform.
664+
* @return Returns the transformed model.
665+
*/
666+
public Model removeInvalidDefaults(Model model) {
667+
return new RemoveInvalidDefaults().transform(this, model);
668+
}
659669
}
Lines changed: 75 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
package software.amazon.smithy.model.transform;
7+
8+
import java.util.HashSet;
9+
import java.util.Set;
10+
import java.util.logging.Logger;
11+
import software.amazon.smithy.model.Model;
12+
import software.amazon.smithy.model.node.Node;
13+
import software.amazon.smithy.model.shapes.MemberShape;
14+
import software.amazon.smithy.model.shapes.Shape;
15+
import software.amazon.smithy.model.traits.DefaultTrait;
16+
import software.amazon.smithy.model.traits.RangeTrait;
17+
18+
/**
19+
* Removes default values from shapes where the default value is incompatible with
20+
* the constraint traits of the shape.
21+
*/
22+
final class RemoveInvalidDefaults {
23+
24+
private static final Logger LOGGER = Logger.getLogger(RemoveInvalidDefaults.class.getName());
25+
26+
Model transform(ModelTransformer transformer, Model model) {
27+
Set<Shape> invalidDefaults = new HashSet<>();
28+
Set<Shape> updates = new HashSet<>();
29+
30+
// First collect invalid shapes. Members with invalid defaults either need to remove the default or
31+
// set the default to null if their target shape's default remains intact but the member is invalid.
32+
for (Shape shape : model.getShapesWithTrait(DefaultTrait.class)) {
33+
shape.getMemberTrait(model, RangeTrait.class).ifPresent(rangeTrait -> {
34+
DefaultTrait defaultTrait = shape.expectTrait(DefaultTrait.class);
35+
if (defaultTrait.toNode().isNumberNode()) {
36+
defaultTrait.toNode().expectNumberNode().asBigDecimal().ifPresent(value -> {
37+
if (rangeTrait.getMin().filter(min -> value.compareTo(min) < 0).isPresent()
38+
|| rangeTrait.getMin().filter(max -> value.compareTo(max) > 0).isPresent()) {
39+
invalidDefaults.add(shape);
40+
}
41+
});
42+
}
43+
});
44+
}
45+
46+
for (Shape shape : invalidDefaults) {
47+
updates.add(modify(shape, model, invalidDefaults));
48+
}
49+
50+
return transformer.replaceShapes(model, updates);
51+
}
52+
53+
private Shape modify(Shape shape, Model model, Set<Shape> otherShapes) {
54+
// To show up here, the shape has to have a range trait, or the target has to have one.
55+
RangeTrait rangeTrait = shape.getMemberTrait(model, RangeTrait.class).get();
56+
LOGGER.info(() -> "Removing default trait from " + shape.getId()
57+
+ " because of an incompatible range trait: "
58+
+ Node.printJson(rangeTrait.toNode()));
59+
60+
// Members that target a shape with a default value need to set their default to null to override it.
61+
// Other members and other shapes can simply remove the default trait.
62+
if (shape.isMemberShape()) {
63+
MemberShape member = shape.asMemberShape().get();
64+
boolean targetHasDefault = model.getShape(member.getTarget())
65+
// Treat target shapes that will have their default removed as if it doesn't have a default.
66+
.filter(target -> !otherShapes.contains(target) && target.hasTrait(DefaultTrait.class))
67+
.isPresent();
68+
if (targetHasDefault) {
69+
return member.toBuilder().addTrait(new DefaultTrait(Node.nullNode())).build();
70+
}
71+
}
72+
73+
return Shape.shapeToBuilder(shape).removeTrait(DefaultTrait.ID).build();
74+
}
75+
}

smithy-model/src/main/java/software/amazon/smithy/model/validation/node/RangeTraitPlugin.java

Lines changed: 16 additions & 18 deletions
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,6 @@
1515

1616
package software.amazon.smithy.model.validation.node;
1717

18-
import java.math.BigDecimal;
1918
import software.amazon.smithy.model.node.Node;
2019
import software.amazon.smithy.model.node.Node.NonNumericFloat;
2120
import software.amazon.smithy.model.node.NumberNode;
@@ -70,27 +69,26 @@ private void checkNonNumeric(Shape shape, RangeTrait trait, StringNode node, Emi
7069
}
7170

7271
protected void check(Shape shape, Context context, RangeTrait trait, NumberNode node, Emitter emitter) {
73-
Number number = node.getValue();
74-
BigDecimal decimal = number instanceof BigDecimal
75-
? (BigDecimal) number
76-
: new BigDecimal(number.toString());
77-
7872
trait.getMin().ifPresent(min -> {
79-
if (decimal.compareTo(new BigDecimal(min.toString())) < 0) {
80-
emitter.accept(node, getSeverity(node, context), String.format(
81-
"Value provided for `%s` must be greater than or equal to %s, but found %s",
82-
shape.getId(), min, number),
83-
shape.isMemberShape() ? MEMBER : TARGET);
84-
}
73+
node.asBigDecimal().ifPresent(decimal -> {
74+
if (decimal.compareTo(min) < 0) {
75+
emitter.accept(node, getSeverity(node, context), String.format(
76+
"Value provided for `%s` must be greater than or equal to %s, but found %s",
77+
shape.getId(), min, decimal),
78+
shape.isMemberShape() ? MEMBER : TARGET);
79+
}
80+
});
8581
});
8682

8783
trait.getMax().ifPresent(max -> {
88-
if (decimal.compareTo(new BigDecimal(max.toString())) > 0) {
89-
emitter.accept(node, getSeverity(node, context), String.format(
90-
"Value provided for `%s` must be less than or equal to %s, but found %s",
91-
shape.getId(), max, number),
92-
shape.isMemberShape() ? MEMBER : TARGET);
93-
}
84+
node.asBigDecimal().ifPresent(decimal -> {
85+
if (decimal.compareTo(max) > 0) {
86+
emitter.accept(node, getSeverity(node, context), String.format(
87+
"Value provided for `%s` must be less than or equal to %s, but found %s",
88+
shape.getId(), max, decimal),
89+
shape.isMemberShape() ? MEMBER : TARGET);
90+
}
91+
});
9492
});
9593
}
9694

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
/*
2+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
package software.amazon.smithy.model.transform;
7+
8+
import org.junit.jupiter.api.Test;
9+
import software.amazon.smithy.model.Model;
10+
import software.amazon.smithy.model.node.Node;
11+
import software.amazon.smithy.model.shapes.ModelSerializer;
12+
13+
public class RemoveInvalidDefaultsTest {
14+
@Test
15+
public void removeInvalidDefaultsBasedOnRangeTrait() {
16+
Model input = Model.assembler()
17+
.addImport(getClass().getResource("bad-defaults-range-trait.smithy"))
18+
.assemble()
19+
.unwrap();
20+
Model output = Model.assembler()
21+
.addImport(getClass().getResource("bad-defaults-range-trait.fixed.smithy"))
22+
.assemble()
23+
.unwrap();
24+
25+
ModelTransformer transformer = ModelTransformer.create();
26+
27+
Model result = transformer.removeInvalidDefaults(input);
28+
29+
Node actual = ModelSerializer.builder().build().serialize(result);
30+
Node expected = ModelSerializer.builder().build().serialize(output);
31+
Node.assertEquals(actual, expected);
32+
}
33+
}
Lines changed: 39 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
$version: "2.0"
2+
3+
namespace smithy.example
4+
5+
structure Foo {
6+
// Default was removed.
7+
@range(min: 1)
8+
invalid1: Integer
9+
10+
// Default was removed.
11+
invalid2: ValueGreaterThanZero
12+
13+
// The default of the target shape was removed, so it can be removed here too.
14+
invalid3: ValueGreaterThanZeroWithDefault
15+
16+
// Cancel out the root level default.
17+
@range(min: 1)
18+
invalid4: PrimitiveInteger = null
19+
20+
valid1: ValueGreaterThanZero = 1
21+
22+
valid2: ValueGreaterThanZero
23+
24+
valid3: Integer
25+
26+
valid4: Integer = 0
27+
28+
@range(min: 1)
29+
valid5: Integer
30+
31+
@range(min: 1)
32+
valid6: Integer = 1
33+
}
34+
35+
@range(min: 1)
36+
integer ValueGreaterThanZero
37+
38+
@range(min: 1)
39+
integer ValueGreaterThanZeroWithDefault // default was removed
Lines changed: 41 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
$version: "2.0"
2+
3+
namespace smithy.example
4+
5+
structure Foo {
6+
// The member itself creates an invalid combination of the range trait and default value.
7+
@range(min: 1)
8+
invalid1: Integer = 0
9+
10+
// The member adds a default value that is incompatible with the target shape.
11+
invalid2: ValueGreaterThanZero = 0
12+
13+
// The member targets a shape where the range trait is incompatible with the default of the member.
14+
invalid3: ValueGreaterThanZeroWithDefault = 0
15+
16+
// The range trait here makes the default value invalid. This member targets a root shape with a default, so the
17+
// default has to be set to null to cancel out the root level default.
18+
@range(min: 1)
19+
invalid4: PrimitiveInteger = 0
20+
21+
valid1: ValueGreaterThanZero = 1
22+
23+
valid2: ValueGreaterThanZero
24+
25+
valid3: Integer
26+
27+
valid4: Integer = 0
28+
29+
@range(min: 1)
30+
valid5: Integer
31+
32+
@range(min: 1)
33+
valid6: Integer = 1
34+
}
35+
36+
@range(min: 1)
37+
integer ValueGreaterThanZero
38+
39+
@range(min: 1)
40+
@default(0) // bad default and range trait combination.
41+
integer ValueGreaterThanZeroWithDefault

0 commit comments

Comments
 (0)