Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Trait code generation #2074

Merged
merged 88 commits into from
Mar 27, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
88 commits
Select commit Hold shift + click to select a range
aa569cf
Break annotation processor out into separate class with tests
hpmellema Dec 13, 2023
e886463
Initial Setup with annotation processor
hpmellema Dec 13, 2023
293bd51
Add integrations
hpmellema Dec 14, 2023
ffb18a3
Update tests
hpmellema Dec 14, 2023
e6497b1
Refactor to simplify project structure
hpmellema Dec 14, 2023
f544dd9
Add documentation to public shapes
hpmellema Dec 14, 2023
91a1c37
Add additional javadocs on integrations
hpmellema Dec 15, 2023
62925be
Remove unnecessary use of node mapper
hpmellema Jan 18, 2024
d14280b
Simplify enum generator to a single effectively sealed class and add …
hpmellema Jan 24, 2024
1d91ad4
Add javadocs for common generators
hpmellema Jan 24, 2024
7b75f1b
Add internal annotation
hpmellema Jan 24, 2024
a7d733b
Simplify provider and fromNode generators based on feedback
hpmellema Jan 25, 2024
d2519d4
Add support for bigInteger traits and fields without truncation
hpmellema Jan 25, 2024
40de4b2
Remove annotation trait generator in favor of the collection generator
hpmellema Jan 26, 2024
a040bbe
Remove redundant check for extra properties
hpmellema Jan 26, 2024
024a70b
Clean up toNode generator member mapping
hpmellema Jan 26, 2024
734447b
Partial refactor to remove directed codegen
hpmellema Jan 26, 2024
11fb413
Flatten generators to restrict visibility and simplify generation cla…
hpmellema Jan 26, 2024
381b89e
Add a check for existing trait providers to prevent generation of con…
hpmellema Feb 5, 2024
681192b
Documentation pass
hpmellema Feb 6, 2024
1934efa
Update to use integ tests instead of annotation processor for testing…
hpmellema Feb 7, 2024
a94278a
Simplify layout
hpmellema Feb 7, 2024
916a643
Add basic plugin tests
hpmellema Feb 7, 2024
6c26f98
Use path conversion to make location strings platform agnostic
hpmellema Feb 7, 2024
03f2713
Use root symbolprovider toMemberName for member name generation
hpmellema Feb 7, 2024
84cf502
Prefer *withoutFormatting writer methods for static strings
hpmellema Feb 13, 2024
9d31996
Do not wrap html tags within docstrings
hpmellema Feb 13, 2024
d667905
Add testing for html tag handling and address minor issues with docst…
hpmellema Feb 13, 2024
d7152e6
Remove Item suffix
hpmellema Feb 13, 2024
890427e
Remove unused enum symbol property
hpmellema Feb 13, 2024
28248b7
Use consumers for node mapping
hpmellema Feb 13, 2024
eb06cdb
Update integration tests to use parameterized test inputs
hpmellema Feb 16, 2024
cbb8c48
Update to only generate traits within a single namespace
hpmellema Feb 21, 2024
1131342
Do not import duplicate symbols in import container
hpmellema Feb 21, 2024
46cfae4
Remove addImports and only allow symbol imports via formatter
hpmellema Feb 21, 2024
f16a179
Update Writer and Import container to handle name conflicts
hpmellema Feb 22, 2024
7414897
Add reserved word handling for java keywords
hpmellema Feb 22, 2024
02817f2
Use unformatted member names for serde
hpmellema Feb 22, 2024
9320793
Handle poorly formatted members and structure names
hpmellema Feb 22, 2024
b9a327d
Add context after pushing state
hpmellema Feb 27, 2024
e935a3e
UseObject nonnull instead of smithy builder state to check for requir…
hpmellema Feb 27, 2024
7fe0f08
Simplify javadoc integrations
hpmellema Feb 27, 2024
d6b8556
Add javadocs for section properties
hpmellema Feb 27, 2024
a377f96
Move ToNodeMapper property to visitor
hpmellema Feb 27, 2024
f89c2c8
Move member and fromNode mappers to visitors
hpmellema Feb 27, 2024
8ba0afc
Simplify symbol provider
hpmellema Feb 27, 2024
7b10d93
Remove mapper class
hpmellema Feb 27, 2024
ab4133c
Clean up import container
hpmellema Feb 27, 2024
1abff09
Use optional returns for non-required maps and lists
hpmellema Feb 27, 2024
2e9ac0c
Address nits
hpmellema Feb 28, 2024
f872128
Move to a single defaultName method for traits and other shapes
hpmellema Feb 28, 2024
25c1552
Fix some weird spacing in generated structures
hpmellema Feb 28, 2024
b4689dc
Update to use full names for Entryto avoid using full names for maps
hpmellema Feb 28, 2024
b4408bf
Add transformation to handle legacy enums
hpmellema Feb 28, 2024
5d1aa46
Flatten mixins
hpmellema Feb 28, 2024
35be8d4
Add tests for mixins and legacy enums
hpmellema Feb 28, 2024
2b76fc0
Update to use DataShapeVisitor class where possible
hpmellema Feb 28, 2024
6b2d932
Add support for nested document members
hpmellema Feb 28, 2024
6bad106
Add support for nested timestamps
hpmellema Feb 28, 2024
14e0596
Support timestampFormat trait
hpmellema Feb 28, 2024
159a75c
Support timestamp traits
hpmellema Feb 29, 2024
3f21790
Update enums to return a nested enum value type
hpmellema Feb 29, 2024
38dac82
Remove support for blob shapes
hpmellema Feb 29, 2024
6a1b8d6
Inline static strings
hpmellema Feb 29, 2024
225f358
Refactor tests to use a single file per trait under test
hpmellema Feb 29, 2024
59be213
Filter traits to only include sources in closure
hpmellema Mar 1, 2024
c9755df
Add support for Unstable Trait
hpmellema Mar 5, 2024
2dea8f6
Reserve smithy-model namespace
hpmellema Mar 5, 2024
c878572
Restrict to source projection
hpmellema Mar 5, 2024
861e975
Tighten restrictions on namespace and package name
hpmellema Mar 5, 2024
84c9d59
Correct some whitespace errors
hpmellema Mar 5, 2024
59ee888
Add value getter to document traits
hpmellema Mar 5, 2024
f88c10e
Add error on null enum value
hpmellema Mar 5, 2024
1136917
Add orEmpty getters for lists and maps
hpmellema Mar 5, 2024
757e061
Ensure the node cache is set for all value traits
hpmellema Mar 6, 2024
a501b20
Simplify timestamp switch case
hpmellema Mar 6, 2024
bec1754
Consolidate generators into a single class
hpmellema Mar 6, 2024
a76544d
Simplify control flow with restricted visitor for traits
hpmellema Mar 6, 2024
6db9a3d
Add support for handling multiple timestampformats if no format is sp…
hpmellema Mar 6, 2024
f325b67
Correct whitespace issue
hpmellema Mar 6, 2024
aaf6d4e
Fix StringListTrait builders
hpmellema Mar 6, 2024
cbb0daf
Use primitive types for Number traits
hpmellema Mar 26, 2024
aabfd47
Update javadocs
hpmellema Mar 26, 2024
28b3be5
Merge branch 'main' into trait-code-generation
hpmellema Mar 27, 2024
569ed18
Update smithy-trait-codegen/src/main/java/software/amazon/smithy/trai…
hpmellema Mar 27, 2024
07496a7
Apply suggestions from code review
hpmellema Mar 27, 2024
092255e
Update to address PR comments
hpmellema Mar 27, 2024
745a3de
Allow for nullable collection members
hpmellema Mar 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions settings.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -36,3 +36,4 @@ include ":smithy-aws-endpoints"
include ":smithy-aws-smoke-test-model"
include ":smithy-protocol-traits"
include ":smithy-protocol-tests"
include ":smithy-trait-codegen"
74 changes: 74 additions & 0 deletions smithy-trait-codegen/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/

description = "Plugin for Generating Trait Code from Smithy Models"

ext {
displayName = "Smithy :: Trait Code Generation"
moduleName = "software.amazon.smithy.traitcodegen"
}

dependencies {
implementation project(":smithy-codegen-core")
}

// Set up Integration testing source sets
sourceSets {
create("it") {
compileClasspath += sourceSets.main.output + configurations["testRuntimeClasspath"] + configurations["testCompileClasspath"]
runtimeClasspath += output + compileClasspath + sourceSets.test.runtimeClasspath + sourceSets.test.output

// Pull in the generated trait files
java {
srcDir("$buildDir/integ/")
}
// Add generated service provider file to resources
resources {
srcDirs += "$buildDir/generated-resources"
}
}
}

// Execute building of trait classes using an executable class
// These traits will then be passed in to the integration test (it)
// source set
tasks.register("generateTraits", JavaExec) {
classpath = sourceSets.test.runtimeClasspath + sourceSets.test.output
mainClass = "software.amazon.smithy.traitcodegen.PluginExecutor"
}

// Copy generated META-INF files to a new generated-resources directory to
// make it easy to include as resource srcDir
def generatedMetaInf = new File("$buildDir/integ/META-INF")
def destResourceDir = new File("$buildDir/generated-resources", "META-INF")
tasks.register("copyGeneratedSrcs", Copy) {
from generatedMetaInf
into destResourceDir
dependsOn("generateTraits")
}


// Add the integ test task
tasks.register("integ", Test) {
useJUnitPlatform()
testClassesDirs = sourceSets.it.output.classesDirs
classpath = sourceSets.it.runtimeClasspath
}

// Do not run checkstyle on generated trait classes
tasks["checkstyleIt"].enabled = false

// Force correct ordering so generated sources are available
tasks["compileItJava"].dependsOn("generateTraits")
tasks["compileItJava"].dependsOn("copyGeneratedSrcs")
tasks["processItResources"].dependsOn("copyGeneratedSrcs")
tasks["integ"].mustRunAfter("generateTraits")
tasks["integ"].mustRunAfter("copyGeneratedSrcs")

// Always run integ tests after base tests
tasks["test"].finalizedBy("integ")

// dont run spotbugs on integ tests
tasks["spotbugsIt"].enabled(false)
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
package software.amazon.smithy.traitcodegen.test;

import static org.junit.jupiter.api.Assertions.assertEquals;

import com.example.traits.StringTrait;
import com.example.traits.documents.DocumentTrait;
import com.example.traits.documents.StructWithNestedDocumentTrait;
import com.example.traits.enums.IntEnumTrait;
import com.example.traits.enums.StringEnumTrait;
import com.example.traits.enums.SuitTrait;
import com.example.traits.lists.ListMember;
import com.example.traits.lists.NumberListTrait;
import com.example.traits.lists.StringListTrait;
import com.example.traits.lists.StructureListTrait;
import com.example.traits.maps.MapValue;
import com.example.traits.maps.StringStringMapTrait;
import com.example.traits.maps.StringToStructMapTrait;
import com.example.traits.mixins.StructWithMixinTrait;
import com.example.traits.mixins.StructureListWithMixinMemberTrait;
import com.example.traits.names.SnakeCaseStructureTrait;
import com.example.traits.numbers.BigDecimalTrait;
import com.example.traits.numbers.BigIntegerTrait;
import com.example.traits.numbers.ByteTrait;
import com.example.traits.numbers.DoubleTrait;
import com.example.traits.numbers.FloatTrait;
import com.example.traits.numbers.IntegerTrait;
import com.example.traits.numbers.LongTrait;
import com.example.traits.numbers.ShortTrait;
import com.example.traits.structures.BasicAnnotationTrait;
import com.example.traits.structures.NestedA;
import com.example.traits.structures.NestedB;
import com.example.traits.structures.StructureTrait;
import com.example.traits.timestamps.DateTimeTimestampTrait;
import com.example.traits.timestamps.EpochSecondsTimestampTrait;
import com.example.traits.timestamps.HttpDateTimestampTrait;
import com.example.traits.timestamps.TimestampTrait;
import com.example.traits.uniqueitems.NumberSetTrait;
import com.example.traits.uniqueitems.SetMember;
import com.example.traits.uniqueitems.StringSetTrait;
import com.example.traits.uniqueitems.StructureSetTrait;
import java.util.stream.Stream;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;
import software.amazon.smithy.model.SourceLocation;
import software.amazon.smithy.model.node.ArrayNode;
import software.amazon.smithy.model.node.Node;
import software.amazon.smithy.model.node.ObjectNode;
import software.amazon.smithy.model.shapes.ShapeId;
import software.amazon.smithy.model.traits.Trait;
import software.amazon.smithy.model.traits.TraitFactory;
import software.amazon.smithy.utils.ListUtils;
import software.amazon.smithy.utils.MapUtils;

public class CreatesTraitTest {
private static final ShapeId DUMMY_ID = ShapeId.from("ns.foo#foo");
private final TraitFactory provider = TraitFactory.createServiceFactory();

static Stream<Arguments> createTraitTests() {
return Stream.of(
// Document traits
Arguments.of(DocumentTrait.ID, Node.objectNodeBuilder()
.withMember("metadata", "woo")
.withMember("more", "yay")
.build()
),
Arguments.of(StructWithNestedDocumentTrait.ID,
ObjectNode.objectNodeBuilder().withMember("doc", ObjectNode.builder()
.withMember("foo", "bar").withMember("fizz", "buzz").build()).build()),
// Enums
Arguments.of(StringEnumTrait.ID, Node.from("no")),
Arguments.of(IntEnumTrait.ID, Node.from(2)),
Arguments.of(SuitTrait.ID, Node.from("clubs")),
// Lists
Arguments.of(NumberListTrait.ID, ArrayNode.fromNodes(
Node.from(1), Node.from(2), Node.from(3))
),
Arguments.of(StringListTrait.ID, ArrayNode.fromStrings("a", "b", "c")),
Arguments.of(StructureListTrait.ID, ArrayNode.fromNodes(
ListMember.builder().a("first").b(1).c("other").build().toNode(),
ListMember.builder().a("second").b(2).c("more").build().toNode()
)),
// Maps
Arguments.of(StringStringMapTrait.ID, StringStringMapTrait.builder()
.putValues("a", "first").putValues("b", "other").build().toNode()
),
Arguments.of(StringToStructMapTrait.ID, StringToStructMapTrait.builder()
.putValues("one", MapValue.builder().a("foo").b(2).build())
.putValues("two", MapValue.builder().a("bar").b(4).build())
.build().toNode()
),
// Mixins
Arguments.of(StructureListWithMixinMemberTrait.ID,
ArrayNode.fromNodes(ObjectNode.builder().withMember("a", "a").withMember("d", "d").build())),
Arguments.of(StructWithMixinTrait.ID, StructWithMixinTrait.builder()
.d("d").build().toNode()),
// Naming Conflicts
Arguments.of(SnakeCaseStructureTrait.ID, ObjectNode.builder()
.withMember("snake_case_member", "stuff").build()),
// Numbers
Arguments.of(BigDecimalTrait.ID, Node.from(1)),
Arguments.of(BigIntegerTrait.ID, Node.from(1)),
Arguments.of(ByteTrait.ID, Node.from(1)),
Arguments.of(DoubleTrait.ID, Node.from(1.2)),
Arguments.of(FloatTrait.ID, Node.from(1.2)),
Arguments.of(IntegerTrait.ID, Node.from(1)),
Arguments.of(LongTrait.ID, Node.from(1L)),
Arguments.of(ShortTrait.ID, Node.from(1)),
// Structures
Arguments.of(BasicAnnotationTrait.ID, Node.objectNode()),
Arguments.of(StructureTrait.ID, StructureTrait.builder()
.fieldA("a")
.fieldB(true)
.fieldC(NestedA.builder()
.fieldN("nested")
.fieldQ(false)
.fieldZ(NestedB.B)
.build()
)
.fieldD(ListUtils.of("a", "b", "c"))
.fieldE(MapUtils.of("a", "one", "b", "two"))
.build().toNode()
),
// Timestamps
Arguments.of(TimestampTrait.ID, Node.from("1985-04-12T23:20:50.52Z")),
Arguments.of(DateTimeTimestampTrait.ID, Node.from("1985-04-12T23:20:50.52Z")),
Arguments.of(HttpDateTimestampTrait.ID, Node.from("Tue, 29 Apr 2014 18:30:38 GMT")),
Arguments.of(EpochSecondsTimestampTrait.ID, Node.from(1515531081.123)),
// Unique Items (sets)
Arguments.of(NumberSetTrait.ID, ArrayNode.fromNodes(
Node.from(1), Node.from(2), Node.from(3))
),
Arguments.of(StringSetTrait.ID, ArrayNode.fromStrings("a", "b", "c")),
Arguments.of(StructureSetTrait.ID, ArrayNode.fromNodes(
SetMember.builder().a("first").b(1).c("other").build().toNode(),
SetMember.builder().a("second").b(2).c("more").build().toNode()
)),
// Strings
Arguments.of(StringTrait.ID, Node.from("SPORKZ SPOONS YAY! Utensils."))
);
}

@ParameterizedTest
@MethodSource("createTraitTests")
void createsTraitFromNode(ShapeId traitId, Node fromNode) {
Trait trait = provider.createTrait(traitId, DUMMY_ID, fromNode).orElseThrow(RuntimeException::new);
assertEquals(SourceLocation.NONE, trait.getSourceLocation());
assertEquals(trait, provider.createTrait(traitId, DUMMY_ID, trait.toNode()).orElseThrow(RuntimeException::new));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
package software.amazon.smithy.traitcodegen.test;

import static org.junit.jupiter.api.Assertions.assertNotNull;

import com.example.traits.DeprecatedStringTrait;
import org.junit.jupiter.api.Test;

class DeprecatedStringTest {
@Test
void checkForDeprecatedAnnotation() {
Deprecated deprecated = DeprecatedStringTrait.class.getAnnotation(Deprecated.class);
assertNotNull(deprecated);
}
}
Loading