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

Sourcing service config from the environment. #3493

Merged
merged 32 commits into from
Apr 1, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
2339db5
add support for service env config
Velfi Mar 12, 2024
02485ac
Merge remote-tracking branch 'origin/main' into zhessler-service-env-…
Velfi Mar 15, 2024
1c39d00
update tests
Velfi Mar 19, 2024
2d574ae
Merge branch 'main' into zhessler-service-env-config
Velfi Mar 19, 2024
6aee5d9
move test JSONs
Velfi Mar 19, 2024
d847bc8
fix clippy lints
Velfi Mar 19, 2024
4504106
fix codegen attribute
Velfi Mar 19, 2024
5746e52
fix integration test import
Velfi Mar 19, 2024
f1ba0c0
fix usage of fs_util
Velfi Mar 19, 2024
d98aa86
add support for s3 express env config
Velfi Mar 19, 2024
699e44a
Merge branch 'main' into zhessler-service-env-config
Velfi Mar 19, 2024
0e4e3be
remove comment
Velfi Mar 20, 2024
8fdcb11
Update aws/rust-runtime/aws-types/src/sdk_config.rs
Velfi Mar 25, 2024
5bc1fa1
Update aws/rust-runtime/aws-runtime/src/profile/section.rs
Velfi Mar 25, 2024
28c2453
Merge remote-tracking branch 'origin/main' into zhessler-service-env-…
Velfi Mar 25, 2024
28102c3
big rename of env config types
Velfi Mar 25, 2024
fca10de
add missing doc comment
Velfi Mar 25, 2024
25fea3c
remove unnecessary async
Velfi Mar 25, 2024
d29330b
Merge branch 'main' into zhessler-service-env-config
Velfi Mar 25, 2024
fc1c32c
appease the versioner
Velfi Mar 26, 2024
cd423f3
fix logging test broken by crate move
Velfi Mar 26, 2024
69ea83a
update external-types.toml
Velfi Mar 26, 2024
e98f476
Merge remote-tracking branch 'origin/main' into zhessler-service-env-…
Velfi Mar 28, 2024
02d7ae9
PR comments update
Velfi Mar 28, 2024
d7744fa
undo change to parse_property_line (old one was better)
Velfi Mar 28, 2024
480ab45
Merge branch 'main' into zhessler-service-env-config
Velfi Mar 28, 2024
ebfab85
remove unnecessary s3 env config test
Velfi Mar 28, 2024
590c000
add changelog entry
Velfi Mar 28, 2024
62f21dc
Merge remote-tracking branch 'origin/main' into zhessler-service-env-…
Velfi Mar 28, 2024
631a3e8
remove duplicated changelog entry
Velfi Mar 29, 2024
a8e3ac3
Update aws/rust-runtime/aws-runtime/src/env_config/file.rs
Velfi Mar 29, 2024
455dc51
Merge branch 'main' into zhessler-service-env-config
Velfi Mar 29, 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
42 changes: 42 additions & 0 deletions CHANGELOG.next.toml
Original file line number Diff line number Diff line change
Expand Up @@ -22,3 +22,45 @@ message = "Make `BehaviorVersion` be future-proof by disallowing it to be constr
references = ["aws-sdk-rust#1111", "smithy-rs#3513"]
meta = { "breaking" = true, "tada" = false, "bug" = true, "target" = "client" }
author = "Ten0"

[[smithy-rs]]
message = """
Stalled stream protection now supports request upload streams. It is currently off by default, but will be enabled by default in a future release. To enable it now, you can do the following:

```rust
let config = my_service::Config::builder()
.stalled_stream_protection(StalledStreamProtectionConfig::enabled().build())
// ...
.build();
```
"""
references = ["smithy-rs#3485"]
meta = { "breaking" = false, "tada" = true, "bug" = false }
authors = ["jdisanti"]

[[aws-sdk-rust]]
message = """
Stalled stream protection now supports request upload streams. It is currently off by default, but will be enabled by default in a future release. To enable it now, you can do the following:

```rust
let config = aws_config::defaults(BehaviorVersion::latest())
.stalled_stream_protection(StalledStreamProtectionConfig::enabled().build())
.load()
.await;
```
"""
references = ["smithy-rs#3485"]
meta = { "breaking" = false, "tada" = true, "bug" = false }
author = "jdisanti"

[[smithy-rs]]
message = "Stalled stream protection on downloads will now only trigger if the upstream source is too slow. Previously, stalled stream protection could be erroneously triggered if the user was slowly consuming the stream slower than the minimum speed limit."
references = ["smithy-rs#3485"]
meta = { "breaking" = false, "tada" = false, "bug" = true }
authors = ["jdisanti"]

[[aws-sdk-rust]]
message = "Users may now set service-specific configuration in the environment. For more information, see [this discussion topic](https://github.com/smithy-lang/smithy-rs/discussions/3537)."
references = ["smithy-rs#3493"]
meta = { "breaking" = false, "tada" = true, "bug" = false }
author = "Velfi"
9 changes: 2 additions & 7 deletions aws/rust-runtime/aws-config/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,8 @@ aws-smithy-runtime = { path = "../../sdk/build/aws-sdk/sdk/aws-smithy-runtime",
aws-smithy-runtime-api = { path = "../../sdk/build/aws-sdk/sdk/aws-smithy-runtime-api", features = ["client"] }
aws-smithy-types = { path = "../../sdk/build/aws-sdk/sdk/aws-smithy-types" }
aws-types = { path = "../../sdk/build/aws-sdk/sdk/aws-types" }
bytes = "1.1.0"
http = "0.2.4"
hyper = { version = "0.14.26", default-features = false }
time = { version = "0.3.4", features = ["parsing"] }
tokio = { version = "1.13.1", features = ["sync"] }
Expand All @@ -44,9 +46,6 @@ url = "2.3.1"
# implementation detail of IMDS credentials provider
fastrand = "2.0.0"

bytes = "1.1.0"
http = "0.2.4"

# implementation detail of SSO credential caching
aws-sdk-sso = { path = "../../sdk/build/aws-sdk/sdk/sso", default-features = false, optional = true }
ring = { version = "0.17.5", optional = true }
Expand All @@ -62,12 +61,8 @@ aws-smithy-runtime-api = { path = "../../sdk/build/aws-sdk/sdk/aws-smithy-runtim
futures-util = { version = "0.3.29", default-features = false }
tracing-test = "0.2.4"
tracing-subscriber = { version = "0.3.16", features = ["fmt", "json"] }

tokio = { version = "1.23.1", features = ["full", "test-util"] }

# used for fuzzing profile parsing
arbitrary = "1.3"

# used for test case deserialization
serde = { version = "1", features = ["derive"] }
serde_json = "1"
Expand Down
10 changes: 9 additions & 1 deletion aws/rust-runtime/aws-config/external-types.toml
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,21 @@ allowed_external_types = [
"aws_credential_types::provider::credentials::Result",
"aws_credential_types::provider::credentials::SharedCredentialsProvider",
"aws_credential_types::provider::token::ProvideToken",
"aws_runtime::env_config::error::EnvConfigFileLoadError",
"aws_runtime::env_config::file::Builder",
"aws_runtime::env_config::file::EnvConfigFileKind",
"aws_runtime::env_config::file::EnvConfigFiles",
"aws_runtime::env_config::parse::EnvConfigParseError",
"aws_runtime::env_config::property::Property",
"aws_runtime::env_config::section::EnvConfigSections",
"aws_runtime::env_config::section::Profile",
"aws_smithy_async::rt::sleep::AsyncSleep",
"aws_smithy_async::rt::sleep::SharedAsyncSleep",
"aws_smithy_async::time::SharedTimeSource",
"aws_smithy_async::time::TimeSource",
"aws_smithy_runtime_api::box_error::BoxError",
"aws_smithy_runtime::client::identity::cache::IdentityCache",
"aws_smithy_runtime::client::identity::cache::lazy::LazyCacheBuilder",
"aws_smithy_runtime_api::box_error::BoxError",
"aws_smithy_runtime_api::client::behavior_version::BehaviorVersion",
"aws_smithy_runtime_api::client::dns::ResolveDns",
"aws_smithy_runtime_api::client::dns::SharedDnsResolver",
Expand Down
30 changes: 19 additions & 11 deletions aws/rust-runtime/aws-config/src/default_provider/app_name.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
*/

use crate::provider_config::ProviderConfig;
use crate::standard_property::{PropertyResolutionError, StandardProperty};
use aws_runtime::env_config::{EnvConfigError, EnvConfigValue};
use aws_smithy_types::error::display::DisplayErrorContext;
use aws_types::app_name::{AppName, InvalidAppName};

Expand Down Expand Up @@ -56,22 +56,24 @@ impl Builder {
self
}

async fn fallback_app_name(
&self,
) -> Result<Option<AppName>, PropertyResolutionError<InvalidAppName>> {
StandardProperty::new()
async fn fallback_app_name(&self) -> Result<Option<AppName>, EnvConfigError<InvalidAppName>> {
let env = self.provider_config.env();
let profiles = self.provider_config.profile().await;

EnvConfigValue::new()
.profile("sdk-ua-app-id")
.validate(&self.provider_config, |name| AppName::new(name.to_string()))
.await
.validate(&env, profiles, |name| AppName::new(name.to_string()))
}

/// Build an [`AppName`] from the default chain
pub async fn app_name(self) -> Option<AppName> {
let standard = StandardProperty::new()
let env = self.provider_config.env();
let profiles = self.provider_config.profile().await;

let standard = EnvConfigValue::new()
.env("AWS_SDK_UA_APP_ID")
.profile("sdk_ua_app_id")
.validate(&self.provider_config, |name| AppName::new(name.to_string()))
.await;
.validate(&env, profiles, |name| AppName::new(name.to_string()));
let with_fallback = match standard {
Ok(None) => self.fallback_app_name().await,
other => other,
Expand All @@ -87,6 +89,7 @@ impl Builder {
#[cfg(test)]
mod tests {
use super::*;
#[allow(deprecated)]
use crate::profile::profile_file::{ProfileFileKind, ProfileFiles};
use crate::provider_config::ProviderConfig;
use crate::test_case::{no_traffic_client, InstantSleep};
Expand Down Expand Up @@ -123,8 +126,13 @@ mod tests {
.http_client(no_traffic_client())
.profile_name("custom")
.profile_files(
#[allow(deprecated)]
ProfileFiles::builder()
.with_file(ProfileFileKind::Config, "test_config")
.with_file(
#[allow(deprecated)]
ProfileFileKind::Config,
"test_config",
)
.build(),
)
.load()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

use crate::environment::parse_url;
use crate::provider_config::ProviderConfig;
use crate::standard_property::StandardProperty;
use aws_runtime::env_config::EnvConfigValue;
use aws_smithy_types::error::display::DisplayErrorContext;

mod env {
Expand All @@ -24,11 +24,13 @@ mod profile_key {
///
/// If invalid values are found, the provider will return None and an error will be logged.
pub async fn endpoint_url_provider(provider_config: &ProviderConfig) -> Option<String> {
StandardProperty::new()
let env = provider_config.env();
let profiles = provider_config.profile().await;

EnvConfigValue::new()
.env(env::ENDPOINT_URL)
.profile(profile_key::ENDPOINT_URL)
.validate(provider_config, parse_url)
.await
.validate(&env, profiles, parse_url)
.map_err(
|err| tracing::warn!(err = %DisplayErrorContext(&err), "invalid value for endpoint URL setting"),
)
Expand All @@ -39,6 +41,7 @@ pub async fn endpoint_url_provider(provider_config: &ProviderConfig) -> Option<S
mod test {
use super::endpoint_url_provider;
use super::env;
#[allow(deprecated)]
use crate::profile::profile_file::{ProfileFileKind, ProfileFiles};
use crate::provider_config::ProviderConfig;
use aws_types::os_shim_internal::{Env, Fs};
Expand All @@ -61,8 +64,13 @@ mod test {
.with_env(Env::from_slice(&[(env::ENDPOINT_URL, "http://localhost")]))
.with_profile_config(
Some(
#[allow(deprecated)]
ProfileFiles::builder()
.with_file(ProfileFileKind::Config, "conf")
.with_file(
#[allow(deprecated)]
ProfileFileKind::Config,
"conf",
)
.build(),
),
None,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

use crate::environment::parse_bool;
use crate::provider_config::ProviderConfig;
use crate::standard_property::StandardProperty;
use aws_runtime::env_config::EnvConfigValue;
use aws_smithy_types::error::display::DisplayErrorContext;

mod env {
Expand All @@ -26,11 +26,13 @@ mod profile_key {
pub async fn ignore_configured_endpoint_urls_provider(
provider_config: &ProviderConfig,
) -> Option<bool> {
StandardProperty::new()
let env = provider_config.env();
let profiles = provider_config.profile().await;

EnvConfigValue::new()
.env(env::IGNORE_CONFIGURED_ENDPOINT_URLS)
.profile(profile_key::IGNORE_CONFIGURED_ENDPOINT_URLS)
.validate(provider_config, parse_bool)
.await
.validate(&env, profiles, parse_bool)
.map_err(
|err| tracing::warn!(err = %DisplayErrorContext(&err), "invalid value for 'ignore configured endpoint URLs' setting"),
)
Expand All @@ -41,6 +43,7 @@ pub async fn ignore_configured_endpoint_urls_provider(
mod test {
use super::env;
use super::ignore_configured_endpoint_urls_provider;
#[allow(deprecated)]
use crate::profile::profile_file::{ProfileFileKind, ProfileFiles};
use crate::provider_config::ProviderConfig;
use aws_types::os_shim_internal::{Env, Fs};
Expand Down Expand Up @@ -70,8 +73,13 @@ mod test {
)]))
.with_profile_config(
Some(
#[allow(deprecated)]
ProfileFiles::builder()
.with_file(ProfileFileKind::Config, "conf")
.with_file(
#[allow(deprecated)]
ProfileFileKind::Config,
"conf",
)
.build(),
),
None,
Expand Down
32 changes: 17 additions & 15 deletions aws/rust-runtime/aws-config/src/default_provider/retry_config.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

use crate::provider_config::ProviderConfig;
use crate::retry::error::{RetryConfigError, RetryConfigErrorKind};
use crate::standard_property::{PropertyResolutionError, StandardProperty};
use aws_runtime::env_config::{EnvConfigError, EnvConfigValue};
use aws_smithy_types::error::display::DisplayErrorContext;
use aws_smithy_types::retry::{RetryConfig, RetryMode};
use std::str::FromStr;
Expand Down Expand Up @@ -101,29 +101,31 @@ impl Builder {

pub(crate) async fn try_retry_config(
self,
) -> Result<RetryConfig, PropertyResolutionError<RetryConfigError>> {
// Both of these can return errors due to invalid config settings and we want to surface those as early as possible
) -> Result<RetryConfig, EnvConfigError<RetryConfigError>> {
let env = self.provider_config.env();
let profiles = self.provider_config.profile().await;
// Both of these can return errors due to invalid config settings, and we want to surface those as early as possible
// hence, we'll panic if any config values are invalid (missing values are OK though)
// We match this instead of unwrapping so we can print the error with the `Display` impl instead of the `Debug` impl that unwrap uses
// We match this instead of unwrapping, so we can print the error with the `Display` impl instead of the `Debug` impl that unwrap uses
let mut retry_config = RetryConfig::standard();
let max_attempts = StandardProperty::new()
let max_attempts = EnvConfigValue::new()
.env(env::MAX_ATTEMPTS)
.profile(profile_keys::MAX_ATTEMPTS)
.validate(&self.provider_config, validate_max_attempts);
.validate(&env, profiles, validate_max_attempts);

let retry_mode = StandardProperty::new()
let retry_mode = EnvConfigValue::new()
.env(env::RETRY_MODE)
.profile(profile_keys::RETRY_MODE)
.validate(&self.provider_config, |s| {
.validate(&env, profiles, |s| {
RetryMode::from_str(s)
.map_err(|err| RetryConfigErrorKind::InvalidRetryMode { source: err }.into())
});

if let Some(max_attempts) = max_attempts.await? {
if let Some(max_attempts) = max_attempts? {
retry_config = retry_config.with_max_attempts(max_attempts);
}

if let Some(retry_mode) = retry_mode.await? {
if let Some(retry_mode) = retry_mode? {
retry_config = retry_config.with_retry_mode(retry_mode);
}

Expand All @@ -146,12 +148,12 @@ mod test {
use crate::retry::{
error::RetryConfigError, error::RetryConfigErrorKind, RetryConfig, RetryMode,
};
use crate::standard_property::PropertyResolutionError;
use aws_runtime::env_config::EnvConfigError;
use aws_types::os_shim_internal::{Env, Fs};

async fn test_provider(
vars: &[(&str, &str)],
) -> Result<RetryConfig, PropertyResolutionError<RetryConfigError>> {
) -> Result<RetryConfig, EnvConfigError<RetryConfigError>> {
super::Builder::default()
.configure(&ProviderConfig::no_configuration().with_env(Env::from_slice(vars)))
.try_retry_config()
Expand Down Expand Up @@ -296,7 +298,7 @@ max_attempts = potato
test_provider(&[(env::MAX_ATTEMPTS, "not an integer")])
.await
.unwrap_err()
.err,
.err(),
RetryConfigError {
kind: RetryConfigErrorKind::FailedToParseMaxAttempts { .. }
}
Expand Down Expand Up @@ -327,8 +329,8 @@ max_attempts = potato
async fn disallow_zero_max_attempts() {
let err = test_provider(&[(env::MAX_ATTEMPTS, "0")])
.await
.unwrap_err()
.err;
.unwrap_err();
let err = err.err();
assert!(matches!(
err,
RetryConfigError {
Expand Down
Loading
Loading