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

add support for nullable struct members when generating AWS SDKs #2916

Merged
merged 43 commits into from
Sep 21, 2023
Merged
Show file tree
Hide file tree
Changes from 32 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
9c17050
add support for nullable struct members when generating AWS SDKs
Velfi Aug 10, 2023
b1c7ec5
add CHANGELOG.next.toml entries
Velfi Aug 10, 2023
0f27839
fix SDK adhoc tests config generation
Velfi Aug 10, 2023
81f428b
fix XML parseStructure
Velfi Aug 10, 2023
633c257
Merge branch 'main' into zhessler-nullability-now!
Velfi Aug 14, 2023
266f54f
fix server codegen issue
Velfi Aug 14, 2023
86dc322
remove TODO
Velfi Aug 14, 2023
a5565a3
update route53 resource ID trimmer
Velfi Aug 14, 2023
2d20462
add back the prefix loop that I accidentally removed
Velfi Aug 14, 2023
acff7bd
more fixing for route53
Velfi Aug 14, 2023
976f3a2
fix query serializer generator ref
Velfi Aug 14, 2023
397c4ab
fix the QuerySerializerGenerator correctly
Velfi Aug 14, 2023
80bb768
fix idempotency token generation
Velfi Aug 15, 2023
a44ae99
no really, fix idempotency token generation
Velfi Aug 15, 2023
916265a
fix serde bugs
Velfi Aug 16, 2023
761c1e4
fix more tests
Velfi Aug 16, 2023
36b7127
fix server client and eventstream stuff
Velfi Aug 16, 2023
3cf8288
Merge branch 'main' into zhessler-nullability-now!
Velfi Aug 16, 2023
7e464cc
remove insurmountable comma
Velfi Aug 16, 2023
d2b2c55
fix another server example
Velfi Aug 16, 2023
505c778
Merge branch 'main' into zhessler-nullability-now!
Velfi Aug 17, 2023
f0e4a1f
fix gradle build config typo
Velfi Aug 17, 2023
ae6a636
Update codegen-client/src/main/kotlin/software/amazon/smithy/rust/cod…
Velfi Aug 17, 2023
491ca43
respond to PR comments
Velfi Aug 17, 2023
66fbddb
Merge branch 'main' into zhessler-nullability-now!
Velfi Aug 17, 2023
c4b9958
add missing author to changelog
Velfi Aug 17, 2023
dac4679
update CHANGELOG
Velfi Aug 17, 2023
b0c63c9
Merge branch 'main' of github.com:awslabs/smithy-rs into zhessler-nul…
rcoh Sep 11, 2023
5e6f6b9
Fix endpoints decorator test
rcoh Sep 12, 2023
94c8dbb
Create ClientBuilderInstatiator & manuall create BuildError
rcoh Sep 12, 2023
cd5e33c
Update ChangeLog
rcoh Sep 12, 2023
1374e95
Fix server
rcoh Sep 12, 2023
e661075
Merge branch 'main' of github.com:awslabs/smithy-rs into zhessler-nul…
rcoh Sep 20, 2023
09c02f7
Fix unit tests
rcoh Sep 20, 2023
b417977
Add a default instantiator that works for tests
rcoh Sep 20, 2023
0bffcc0
add protocol test for error correction & nullability
rcoh Sep 20, 2023
eee07db
Merge branch 'main' of github.com:awslabs/smithy-rs into zhessler-nul…
rcoh Sep 20, 2023
58cf184
Add nullability test and fix some bugs
rcoh Sep 20, 2023
7a4fd41
A few more fixes
rcoh Sep 21, 2023
e75ad16
wip
rcoh Sep 21, 2023
99de898
Fix request id accessor when it is marked as required
rcoh Sep 21, 2023
969e32f
Refactor && remove println
rcoh Sep 21, 2023
7de1892
Fix unit test & add an ignore for useless question mark
rcoh Sep 21, 2023
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
15 changes: 15 additions & 0 deletions CHANGELOG.next.toml
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,21 @@ references = ["smithy-rs#2911"]
meta = { "breaking" = true, "tada" = false, "bug" = false }
author = "Velfi"

[[aws-sdk-rust]]
message = "Struct members modeled as required are no longer wrapped in `Option`s [when possible](https://smithy.io/2.0/spec/aggregate-types.html#structure-member-optionality). For upgrade guidance and more info, see [here](https://github.com/awslabs/smithy-rs/discussions/2929)."
references = ["smithy-rs#2916", "aws-sdk-rust#536"]
meta = { "breaking" = true, "tada" = true, "bug" = false }
author = "Velfi"

[[smithy-rs]]
message = """
Support for Smithy IDLv2 nullability is now enabled by default. You can maintain the old behavior by setting `nullabilityCheckMode: "CLIENT_ZERO_VALUE_V1" in your codegen config.
For upgrade guidance and more info, see [here](https://github.com/awslabs/smithy-rs/discussions/2929).
"""
references = ["smithy-rs#2916", "smithy-rs#1767"]
meta = { "breaking" = false, "tada" = true, "bug" = false, "target" = "client"}
author = "Velfi"

[[aws-sdk-rust]]
message = """
All versions of SigningParams have been updated to contain an [`Identity`](https://docs.rs/aws-smithy-runtime-api/latest/aws_smithy_runtime_api/client/identity/struct.Identity.html)
Expand Down
17 changes: 4 additions & 13 deletions aws/rust-runtime/aws-config/src/sts/util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,24 +17,15 @@ pub(crate) fn into_credentials(
) -> provider::Result {
let sts_credentials = sts_credentials
.ok_or_else(|| CredentialsError::unhandled("STS credentials must be defined"))?;
let expiration = SystemTime::try_from(
sts_credentials
.expiration
.ok_or_else(|| CredentialsError::unhandled("missing expiration"))?,
)
.map_err(|_| {
let expiration = SystemTime::try_from(sts_credentials.expiration).map_err(|_| {
CredentialsError::unhandled(
"credential expiration time cannot be represented by a SystemTime",
)
})?;
Ok(AwsCredentials::new(
sts_credentials
.access_key_id
.ok_or_else(|| CredentialsError::unhandled("access key id missing from result"))?,
sts_credentials
.secret_access_key
.ok_or_else(|| CredentialsError::unhandled("secret access token missing"))?,
sts_credentials.session_token,
sts_credentials.access_key_id,
sts_credentials.secret_access_key,
Some(sts_credentials.session_token),
Some(expiration),
provider_name,
))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ use std::marker::PhantomData;
// This function is only used to strip prefixes from resource IDs at the time they're passed as
// input to a request. Resource IDs returned in responses may or may not include a prefix.
/// Strip the resource type prefix from resource ID return
fn trim_resource_id(resource_id: &mut Option<String>) {
fn trim_resource_id(resource_id: &mut String) {
const PREFIXES: &[&str] = &[
"/hostedzone/",
"hostedzone/",
Expand All @@ -27,28 +27,24 @@ fn trim_resource_id(resource_id: &mut Option<String>) {
];

for prefix in PREFIXES {
if let Some(id) = resource_id
.as_deref()
.unwrap_or_default()
.strip_prefix(prefix)
{
*resource_id = Some(id.to_string());
if let Some(trimmed_id) = resource_id.strip_prefix(prefix) {
*resource_id = trimmed_id.to_string();
return;
}
}
}

pub(crate) struct Route53ResourceIdInterceptor<G, T>
where
G: for<'a> Fn(&'a mut T) -> &'a mut Option<String>,
G: for<'a> Fn(&'a mut T) -> Option<&'a mut String>,
Copy link
Collaborator

Choose a reason for hiding this comment

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

side note, might be nice to replace these existential types with a trait for clarity

{
get_mut_resource_id: G,
_phantom: PhantomData<T>,
}

impl<G, T> fmt::Debug for Route53ResourceIdInterceptor<G, T>
where
G: for<'a> Fn(&'a mut T) -> &'a mut Option<String>,
G: for<'a> Fn(&'a mut T) -> Option<&'a mut String>,
{
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Route53ResourceIdInterceptor").finish()
Expand All @@ -57,7 +53,7 @@ where

impl<G, T> Route53ResourceIdInterceptor<G, T>
where
G: for<'a> Fn(&'a mut T) -> &'a mut Option<String>,
G: for<'a> Fn(&'a mut T) -> Option<&'a mut String>,
{
pub(crate) fn new(get_mut_resource_id: G) -> Self {
Self {
Expand All @@ -69,7 +65,7 @@ where

impl<G, T> Interceptor for Route53ResourceIdInterceptor<G, T>
where
G: for<'a> Fn(&'a mut T) -> &'a mut Option<String> + Send + Sync,
G: for<'a> Fn(&'a mut T) -> Option<&'a mut String> + Send + Sync,
T: fmt::Debug + Send + Sync + 'static,
{
fn name(&self) -> &'static str {
Expand All @@ -83,8 +79,9 @@ where
_cfg: &mut ConfigBag,
) -> Result<(), BoxError> {
let input: &mut T = context.input_mut().downcast_mut().expect("correct type");
let field = (self.get_mut_resource_id)(input);
trim_resource_id(field);
if let Some(field) = (self.get_mut_resource_id)(input) {
trim_resource_id(field)
}
Ok(())
}
}
Expand All @@ -96,48 +93,39 @@ mod test {
#[test]
fn does_not_change_regular_zones() {
struct OperationInput {
resource: Option<String>,
resource: String,
}

let mut operation = OperationInput {
resource: Some("Z0441723226OZ66S5ZCNZ".to_string()),
resource: "Z0441723226OZ66S5ZCNZ".to_string(),
};
trim_resource_id(&mut operation.resource);
assert_eq!(
&operation.resource.unwrap_or_default(),
"Z0441723226OZ66S5ZCNZ"
);
assert_eq!(&operation.resource, "Z0441723226OZ66S5ZCNZ");
}

#[test]
fn sanitizes_prefixed_zone() {
struct OperationInput {
change_id: Option<String>,
change_id: String,
}

let mut operation = OperationInput {
change_id: Some("/change/Z0441723226OZ66S5ZCNZ".to_string()),
change_id: "/change/Z0441723226OZ66S5ZCNZ".to_string(),
};
trim_resource_id(&mut operation.change_id);
assert_eq!(
&operation.change_id.unwrap_or_default(),
"Z0441723226OZ66S5ZCNZ"
);
assert_eq!(&operation.change_id, "Z0441723226OZ66S5ZCNZ");
}

#[test]
fn allow_no_leading_slash() {
struct OperationInput {
hosted_zone: Option<String>,
hosted_zone: String,
}

let mut operation = OperationInput {
hosted_zone: Some("hostedzone/Z0441723226OZ66S5ZCNZ".to_string()),
hosted_zone: "hostedzone/Z0441723226OZ66S5ZCNZ".to_string(),
};
trim_resource_id(&mut operation.hosted_zone);
assert_eq!(
&operation.hosted_zone.unwrap_or_default(),
"Z0441723226OZ66S5ZCNZ"
);
assert_eq!(&operation.hosted_zone, "Z0441723226OZ66S5ZCNZ");
}
}
8 changes: 6 additions & 2 deletions aws/sdk-adhoc-test/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,8 @@ dependencies {
implementation("software.amazon.smithy:smithy-aws-traits:$smithyVersion")
}

fun getNullabilityCheckMode(): String = properties.get("nullability.check.mode") ?: "CLIENT_CAREFUL"

val allCodegenTests = listOf(
CodegenTest(
"com.amazonaws.apigateway#BackplaneControlService",
Expand All @@ -45,7 +47,8 @@ val allCodegenTests = listOf(
extraConfig = """
,
"codegen": {
"includeFluentClient": false
"includeFluentClient": false,
"nullabilityCheckMode": ${getNullabilityCheckMode()}
},
"customizationConfig": {
"awsSdk": {
Expand All @@ -61,7 +64,8 @@ val allCodegenTests = listOf(
extraConfig = """
,
"codegen": {
"includeFluentClient": false
"includeFluentClient": false,
"nullabilityCheckMode": ${getNullabilityCheckMode()}
},
"customizationConfig": {
"awsSdk": {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -80,10 +80,11 @@ class TrimResourceIdCustomization(
RuntimeType.forInlineDependency(
InlineAwsDependency.forRustFile("route53_resource_id_preprocessor"),
).resolve("Route53ResourceIdInterceptor")

rustTemplate(
"""
#{Route53ResourceIdInterceptor}::new(|input: &mut #{Input}| {
&mut input.$fieldName
input.$fieldName.as_mut()
})
""",
"Input" to codegenContext.symbolProvider.toSymbol(inputShape),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -60,12 +60,12 @@ class TimestreamDecorator : ClientCodegenDecorator {
client.describe_endpoints().send().await.map_err(|e| {
#{ResolveEndpointError}::from_source("failed to call describe_endpoints", e)
})?;
let endpoint = describe_endpoints.endpoints().unwrap().get(0).unwrap();
let endpoint = describe_endpoints.endpoints().get(0).unwrap();
let expiry = client.config().time_source().expect("checked when ep discovery was enabled").now()
+ #{Duration}::from_secs(endpoint.cache_period_in_minutes() as u64 * 60);
Ok((
#{Endpoint}::builder()
.url(format!("https://{}", endpoint.address().unwrap()))
.url(format!("https://{}", endpoint.address()))
.build(),
expiry,
))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,22 @@
package software.amazon.smithy.rustsdk.customize.ec2

import io.kotest.matchers.shouldBe
import org.junit.jupiter.api.Test
import org.junit.jupiter.params.ParameterizedTest
import org.junit.jupiter.params.provider.CsvSource
import software.amazon.smithy.model.knowledge.NullableIndex
import software.amazon.smithy.model.shapes.StructureShape
import software.amazon.smithy.rust.codegen.core.testutil.asSmithyModel
import software.amazon.smithy.rust.codegen.core.util.lookup

internal class EC2MakePrimitivesOptionalTest {
@Test
fun `primitive shapes are boxed`() {
@ParameterizedTest
@CsvSource(
"CLIENT",
"CLIENT_CAREFUL",
"CLIENT_ZERO_VALUE_V1",
"CLIENT_ZERO_VALUE_V1_NO_INPUT",
)
fun `primitive shapes are boxed`(nullabilityCheckMode: NullableIndex.CheckMode) {
val baseModel = """
namespace test
structure Primitives {
Expand All @@ -36,7 +43,7 @@ internal class EC2MakePrimitivesOptionalTest {
val nullableIndex = NullableIndex(model)
val struct = model.lookup<StructureShape>("test#Primitives")
struct.members().forEach {
nullableIndex.isMemberNullable(it, NullableIndex.CheckMode.CLIENT_ZERO_VALUE_V1) shouldBe true
nullableIndex.isMemberNullable(it, nullabilityCheckMode) shouldBe true
}
}
}
4 changes: 3 additions & 1 deletion aws/sdk/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ val crateVersioner by lazy { aws.sdk.CrateVersioner.defaultFor(rootProject, prop

fun getRustMSRV(): String = properties.get("rust.msrv") ?: throw Exception("Rust MSRV missing")
fun getPreviousReleaseVersionManifestPath(): String? = properties.get("aws.sdk.previous.release.versions.manifest")
fun getNullabilityCheckMode(): String = properties.get("nullability.check.mode") ?: "CLIENT_CAREFUL"

fun loadServiceMembership(): Membership {
val membershipOverride = properties.get("aws.services")?.let { parseMembership(it) }
Expand Down Expand Up @@ -103,7 +104,8 @@ fun generateSmithyBuild(services: AwsServices): String {
"renameErrors": false,
"debugMode": $debugMode,
"eventStreamAllowList": [$eventStreamAllowListMembers],
"enableUserConfigurableRuntimePlugins": false
"enableUserConfigurableRuntimePlugins": false,
"nullabilityCheckMode": "${getNullabilityCheckMode()}"
},
"service": "${service.service}",
"module": "$moduleName",
Expand Down
2 changes: 1 addition & 1 deletion aws/sdk/integration-tests/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ members = [
"s3",
"s3control",
"sts",
"transcribestreaming",
"timestreamquery",
"transcribestreaming",
"webassembly",
]
12 changes: 8 additions & 4 deletions aws/sdk/integration-tests/dynamodb/tests/movies.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,25 +28,29 @@ async fn create_table(client: &Client, table_name: &str) {
KeySchemaElement::builder()
.attribute_name("year")
.key_type(KeyType::Hash)
.build(),
.build()
.unwrap(),
Comment on lines +31 to +32
Copy link
Collaborator

Choose a reason for hiding this comment

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

Should we change build to try_build and add a build that panics?

)
.key_schema(
KeySchemaElement::builder()
.attribute_name("title")
.key_type(KeyType::Range)
.build(),
.build()
.unwrap(),
)
.attribute_definitions(
AttributeDefinition::builder()
.attribute_name("year")
.attribute_type(ScalarAttributeType::N)
.build(),
.build()
.unwrap(),
)
.attribute_definitions(
AttributeDefinition::builder()
.attribute_name("title")
.attribute_type(ScalarAttributeType::S)
.build(),
.build()
.unwrap(),
Comment on lines 48 to +53
Copy link
Collaborator

Choose a reason for hiding this comment

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

side note: I wonder if we want this API to accept impl TryInto<AttributeDefinitions>

Copy link
Contributor Author

Choose a reason for hiding this comment

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

put another way, what if users could pass a builder directly?

)
.provisioned_throughput(
ProvisionedThroughput::builder()
Expand Down
14 changes: 5 additions & 9 deletions aws/sdk/integration-tests/qldbsession/tests/integration.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,19 +3,14 @@
* SPDX-License-Identifier: Apache-2.0
*/

use aws_sdk_qldbsession as qldbsession;
use aws_sdk_qldbsession::config::{Config, Credentials, Region};
use aws_sdk_qldbsession::types::StartSessionRequest;
use aws_sdk_qldbsession::Client;
use aws_smithy_client::test_connection::TestConnection;
use aws_smithy_http::body::SdkBody;
use http::Uri;
use qldbsession::config::{Config, Credentials, Region};
use qldbsession::types::StartSessionRequest;
use qldbsession::Client;
use std::time::{Duration, UNIX_EPOCH};

// TODO(DVR): having the full HTTP requests right in the code is a bit gross, consider something
// like https://github.com/davidbarsky/sigv4/blob/master/aws-sigv4/src/lib.rs#L283-L315 to store
// the requests/responses externally

#[tokio::test]
async fn signv4_use_correct_service_name() {
let conn = TestConnection::new(vec![(
Expand Down Expand Up @@ -46,7 +41,8 @@ async fn signv4_use_correct_service_name() {
.start_session(
StartSessionRequest::builder()
.ledger_name("not-real-ledger")
.build(),
.build()
.unwrap(),
)
.customize()
.await
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@ package software.amazon.smithy.rust.codegen.client.smithy

import software.amazon.smithy.build.PluginContext
import software.amazon.smithy.model.Model
import software.amazon.smithy.model.knowledge.NullableIndex
import software.amazon.smithy.model.shapes.OperationShape
import software.amazon.smithy.model.shapes.ServiceShape
import software.amazon.smithy.model.shapes.Shape
Expand Down Expand Up @@ -77,10 +76,11 @@ class ClientCodegenVisitor(
val rustSymbolProviderConfig = RustSymbolProviderConfig(
runtimeConfig = settings.runtimeConfig,
renameExceptions = settings.codegenConfig.renameExceptions,
nullabilityCheckMode = NullableIndex.CheckMode.CLIENT_ZERO_VALUE_V1,
nullabilityCheckMode = settings.codegenConfig.nullabilityCheckMode,
moduleProvider = ClientModuleProvider,
nameBuilderFor = { symbol -> "${symbol.name}Builder" },
)

val baseModel = baselineTransform(context.model)
val untransformedService = settings.getService(baseModel)
val (protocol, generator) = ClientProtocolLoader(
Expand Down
Loading