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

chore: add scaffolding task #1202

Merged
merged 9 commits into from
Feb 6, 2024
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
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 .github/workflows/continuous-integration.yml
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ jobs:
shell: bash
run: |
pwd
./gradlew :build-support:test
./gradlew publishToMavenLocal
./gradlew apiCheck
./gradlew test jvmTest
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,20 +49,18 @@ data class AwsService(
*/
val version: String,

/**
* Get the artifact name to use for the service derived from the sdkId. This will be the `A` in the GAV coordinates
* and the directory name under `services/`.
*/
val artifactName: String,

/**
* A description of the service (taken from the title trait)
*/
val description: String? = null,

)

/**
* Get the artifact name to use for the service derived from the sdkId. This will be the `A` in the GAV coordinates
* and the directory name under `services/`.
*/
val AwsService.artifactName: String
get() = sdkIdToArtifactName(sdkId)

/**
* Returns a lambda for a service model file that respects the given bootstrap config
*
Expand All @@ -72,6 +70,7 @@ val AwsService.artifactName: String
fun fileToService(
project: Project,
bootstrap: BootstrapConfig,
pkgManifest: PackageManifest,
): (File) -> AwsService? = { file: File ->
val sdkVersion = project.findProperty("sdkVersion") as? String ?: error("expected sdkVersion to be set on project ${project.name}")
val filename = file.nameWithoutExtension
Expand Down Expand Up @@ -111,14 +110,16 @@ fun fileToService(

else -> {
project.logger.info("discovered service: ${serviceTrait.sdkId}")
val pkgMetadata = pkgManifest.bySdkId[sdkId] ?: error("unable to find package metadata for sdkId: $sdkId")
AwsService(
serviceShapeId = service.id.toString(),
packageName = packageNamespaceForService(sdkId),
packageName = pkgMetadata.namespace,
packageVersion = sdkVersion,
modelFile = file,
projectionName = filename,
sdkId = sdkId,
version = service.version,
artifactName = pkgMetadata.artifactName,
description = packageDescription,
)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,3 +37,16 @@ internal fun sdkIdToArtifactName(sdkId: String): String = sdkId.replace(" ", "")
* catapult! See AwsSdkCatapultWorkspaceTools:lib/source/merge/smithy-model-handler.ts
*/
fun sdkIdToModelFilename(sdkId: String): String = sdkId.trim().replace("""[\s]+""".toRegex(), "-").lowercase()

// FIX - replace with case utils from smithy-kotlin once we verify we can change the implementation
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: FIXFIXME

private fun String.lowercaseAndCapitalize() = lowercase().replaceFirstChar(Char::uppercaseChar)
private val wordBoundary = "[^a-zA-Z0-9]+".toRegex()
private fun String.pascalCase(): String = split(wordBoundary).pascalCase()
fun List<String>.pascalCase() = joinToString(separator = "") { it.lowercaseAndCapitalize() }

private const val BRAZIL_GROUP_NAME = "AwsSdkKotlin"

/**
* Maps an sdkId from a model to the brazil package name to use
*/
fun sdkIdToBrazilName(sdkId: String): String = "${BRAZIL_GROUP_NAME}${sdkId.pascalCase()}"
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/
package aws.sdk.kotlin.gradle.sdk

import kotlinx.serialization.ExperimentalSerializationApi
import kotlinx.serialization.Serializable
import kotlinx.serialization.json.Json
import kotlinx.serialization.json.decodeFromStream
import java.io.File

/**
* Manifest containing additional metadata about services.
*/
@OptIn(ExperimentalSerializationApi::class)
@Serializable
data class PackageManifest(
val packages: List<PackageMetadata>,
) {

val bySdkId: Map<String, PackageMetadata> = packages.associateBy(PackageMetadata::sdkId)
companion object {
fun fromFile(file: File): PackageManifest =
file.inputStream().use {
Json.decodeFromStream<PackageManifest>(it)
}
}
}

/**
* Validate the package manifest for errors throwing an exception if any exist.
*/
fun PackageManifest.validate() {
val distinct = mutableMapOf<String, PackageMetadata>()
val errors = mutableListOf<String>()
packages.forEach {
val existing = distinct[it.sdkId]
if (existing != null) {
errors.add("multiple packages with same sdkId `${it.sdkId}`: first: $existing; second: $it")
}
distinct[it.sdkId] = it
}

check(errors.isEmpty()) { errors.joinToString(separator = "\n") }
}

@Serializable
data class PackageMetadata(
public val sdkId: String,
public val namespace: String,
public val artifactName: String,
public val brazilName: String,
) {
Comment on lines +56 to +62
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggestion: Add KDocs. In particular, it's not clear that artifactName means "Maven artifact name (i.e., the 'A' in 'GAV')" solely by looking at the code.

companion object {

/**
* Create a new [PackageMetadata] from inferring values using the given sdkId
*/
fun from(sdkId: String): PackageMetadata =
PackageMetadata(
sdkId,
packageNamespaceForService(sdkId),
sdkIdToArtifactName(sdkId),
sdkIdToBrazilName(sdkId),
)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/
package aws.sdk.kotlin.gradle.sdk.tasks

import aws.sdk.kotlin.gradle.sdk.PackageManifest
import aws.sdk.kotlin.gradle.sdk.PackageMetadata
import aws.sdk.kotlin.gradle.sdk.orNull
import aws.sdk.kotlin.gradle.sdk.validate
import kotlinx.serialization.ExperimentalSerializationApi
import kotlinx.serialization.encodeToString
import kotlinx.serialization.json.Json
import org.gradle.api.DefaultTask
import org.gradle.api.file.DirectoryProperty
import org.gradle.api.file.RegularFileProperty
import org.gradle.api.provider.Property
import org.gradle.api.tasks.*
import org.gradle.api.tasks.options.Option
import software.amazon.smithy.aws.traits.ServiceTrait
import software.amazon.smithy.model.Model
import software.amazon.smithy.model.shapes.ServiceShape
import kotlin.streams.toList

/**
* Task to update the package manifest which is used by the bootstrap process to generate service clients.
* New services are required to be scaffolded
*/
abstract class Scaffold : DefaultTask() {

@get:Option(option = "model", description = "the path to a single model file to scaffold")
@get:Optional
@get:InputFile
public abstract val modelFile: RegularFileProperty

@get:Optional
@get:Option(option = "model-dir", description = "the path to a directory of model files to scaffold")
@get:InputDirectory
public abstract val modelDir: DirectoryProperty

@get:Optional
@get:Option(
option = "discover",
description = "Flag to discover and process only new packages not currently in the manifest. Only applicable when used in conjunction with `model-dir`",
)
@get:Input
public abstract val discover: Property<Boolean>

@OptIn(ExperimentalSerializationApi::class)
@TaskAction
fun updatePackageManifest() {
check(modelFile.isPresent || modelDir.isPresent) { "one of `model` or `model-dir` is required" }
check(!(modelFile.isPresent && modelDir.isPresent)) { "only one of `model` or `model-dir` can be set" }
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Style: Could combine:

check(modelFile.isPresent != modelDir.isPresent) { "Exactly one of `model` or `model-dir` must be set" }


val manifestFile = project.file("packages.json")

val manifest = if (manifestFile.exists()) {
val manifest = PackageManifest.fromFile(manifestFile)
manifest.validate()
manifest
} else {
PackageManifest(emptyList())
}

val model = Model.assembler()
.discoverModels()
.apply {
val import = if (modelFile.isPresent) modelFile else modelDir
addImport(import.get().asFile.absolutePath)
}
.assemble()
.result
.get()

val discoveredPackages = model
.shapes(ServiceShape::class.java)
.toList()
.mapNotNull { it.getTrait(ServiceTrait::class.java).orNull()?.sdkId }
.map { PackageMetadata.from(it) }

val newPackages = validatedPackages(manifest, discoveredPackages)

if (newPackages.isEmpty()) {
logger.lifecycle("no new packages to scaffold")
return
}

logger.lifecycle("scaffolding ${newPackages.size} new service packages")

val updatedPackages = manifest.packages + newPackages
val updatedManifest = manifest.copy(packages = updatedPackages.sortedBy { it.sdkId })

val json = Json { prettyPrint = true }
val contents = json.encodeToString(updatedManifest)
manifestFile.writeText(contents)
}

private fun validatedPackages(manifest: PackageManifest, discovered: List<PackageMetadata>): List<PackageMetadata> =
if (modelDir.isPresent && discover.orNull == true) {
val bySdkId = manifest.packages.associateBy(PackageMetadata::sdkId)
discovered.filter { it.sdkId !in bySdkId }
} else {
discovered.forEach { pkg ->
val existing = manifest.packages.find { it.sdkId == pkg.sdkId }
check(existing == null) { "found existing package in manifest for sdkId `${pkg.sdkId}`: $existing" }
}
discovered
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -8,22 +8,20 @@ import org.gradle.kotlin.dsl.extra
import org.gradle.testfixtures.ProjectBuilder
import org.junit.jupiter.api.io.TempDir
import java.io.File
import kotlin.test.Test
import kotlin.test.assertEquals
import kotlin.test.assertNull
import kotlin.test.*

class AwsServiceTest {

val modelContents = """
${"$"}version: "2.0"
${"$"}version: "2"
namespace gradle.test
Comment on lines 15 to 17
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Question: Why did this change?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've slept and hacked since then and I don't recall. I want to say something about the model discovery and validation triggering something here and so I just fixed it to make smithy happy again.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

services still model this as 2.0, so if it breaks in this test, would it also break once we try to use it for a real service?

probably not, because the CI which bootstraps services passes, still weird.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In Smithy IDL, both of these statements are valid but seem to mean the exact same thing—tooling must support an IDL version of anywhere from 2.0 (inclusive) to 3.0 (exclusive). I'd be curious what broke and if there's maybe a bug somewhere else.


use aws.api#service
use aws.protocols#awsJson1_0

@service(sdkId: "Test Gradle")
@awsJson1_0
service TestService{
service TestService {
operations: [],
version: "1-alpha"
}
Expand All @@ -34,9 +32,23 @@ class AwsServiceTest {
val actual: AwsService?,
)

private val defaultPackageManifest = PackageManifest(
listOf(
PackageMetadata(
"Test Gradle",
// namespace and artifact name intentionally don't match the sdkId derivations to verify we pull from
// the metadata rather than inferring again
"aws.sdk.kotlin.services.testgradle2",
"test-gradle",
"AwsSdkKotlinTestGradle",
),
),
)

private fun testWith(
tempDir: File,
bootstrap: BootstrapConfig,
manifest: PackageManifest = defaultPackageManifest,
): TestResult {
val project = ProjectBuilder.builder()
.build()
Expand All @@ -46,7 +58,7 @@ class AwsServiceTest {
val model = tempDir.resolve("test-gradle.smithy")
model.writeText(modelContents)

val lambda = fileToService(project, bootstrap)
val lambda = fileToService(project, bootstrap, manifest)
val actual = lambda(model)
return TestResult(model, actual)
}
Expand All @@ -69,12 +81,13 @@ class AwsServiceTest {
val result = testWith(tempDir, bootstrap)
val expected = AwsService(
"gradle.test#TestService",
"aws.sdk.kotlin.services.testgradle",
"aws.sdk.kotlin.services.testgradle2",
"1.2.3",
Comment on lines 82 to 85
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Question: Is this rename significant?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's proving that the namespace is taken from the package manifest given vs the inferred namespace from sdkId.

result.model,
"test-gradle",
"Test Gradle",
"1-alpha",
"test-gradle",
"The AWS SDK for Kotlin client for Test Gradle",
)
assertEquals(expected, result.actual)
Expand All @@ -98,4 +111,12 @@ class AwsServiceTest {
assertNull(result.actual, "expected null for bootstrap with $bootstrap")
}
}

@Test
fun testFileToServiceMissingPackageMetadata(@TempDir tempDir: File) {
val ex = assertFailsWith<IllegalStateException> {
testWith(tempDir, BootstrapConfig.ALL, PackageManifest(emptyList()))
}
assertContains(ex.message!!, "unable to find package metadata for sdkId: Test Gradle")
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0
*/
package aws.sdk.kotlin.gradle.sdk

import kotlin.test.Test
import kotlin.test.assertContains
import kotlin.test.assertFailsWith

class PackageManifestTest {
@Test
fun testValidate() {
val manifest = PackageManifest(
listOf(
PackageMetadata("Package 1", "aws.sdk.kotlin.services.package1", "package1", "AwsSdkKotlinPackage1"),
PackageMetadata("Package 2", "aws.sdk.kotlin.services.package2", "package2", "AwsSdkKotlinPackage2"),
),
)

manifest.validate()

val badManifest = manifest.copy(
manifest.packages + listOf(
PackageMetadata("Package 2", "aws.sdk.kotlin.services.package2", "package2", "AwsSdkKotlinPackage2"),
),
)

val ex = assertFailsWith<IllegalStateException> { badManifest.validate() }

assertContains(ex.message!!, "multiple packages with same sdkId `Package 2`")
}
}
Loading
Loading