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

re-add support for sourcing endpoint URLs from service-specific env config #3568

Merged
merged 10 commits into from
Apr 10, 2024
2 changes: 1 addition & 1 deletion aws/rust-runtime/aws-config/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "aws-config"
version = "1.1.10"
version = "1.1.11"
authors = [
"AWS Rust SDK Team <aws-sdk-rust@amazon.com>",
"Russell Cohen <rcoh@amazon.com>",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ use crate::environment::parse_url;
use crate::provider_config::ProviderConfig;
use aws_runtime::env_config::EnvConfigValue;
use aws_smithy_types::error::display::DisplayErrorContext;
use aws_types::origin::Origin;

mod env {
pub(super) const ENDPOINT_URL: &str = "AWS_ENDPOINT_URL";
Expand Down Expand Up @@ -37,6 +38,29 @@ pub async fn endpoint_url_provider(provider_config: &ProviderConfig) -> Option<S
.unwrap_or(None)
}

/// Load the value for an endpoint URL
///
/// This checks the following sources:
/// 1. The environment variable `AWS_ENDPOINT_URL=http://localhost`
/// 2. The profile key `endpoint_url=http://localhost`
///
/// If invalid values are found, the provider will return None and an error will be logged.
pub async fn endpoint_url_provider_with_origin(
provider_config: &ProviderConfig,
) -> (Option<String>, Origin) {
let env = provider_config.env();
let profiles = provider_config.profile().await;

EnvConfigValue::new()
.env(env::ENDPOINT_URL)
.profile(profile_key::ENDPOINT_URL)
.validate_and_return_origin(&env, profiles, parse_url)
.map_err(
|err| tracing::warn!(err = %DisplayErrorContext(&err), "invalid value for endpoint URL setting"),
)
.unwrap_or_default()
}

#[cfg(test)]
mod test {
use super::endpoint_url_provider;
Expand Down
87 changes: 83 additions & 4 deletions aws/rust-runtime/aws-config/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,7 @@ mod loader {
use aws_smithy_types::timeout::TimeoutConfig;
use aws_types::app_name::AppName;
use aws_types::docs_for;
use aws_types::origin::Origin;
use aws_types::os_shim_internal::{Env, Fs};
use aws_types::sdk_config::SharedHttpClient;
use aws_types::SdkConfig;
Expand Down Expand Up @@ -605,7 +606,6 @@ mod loader {
///
/// ```no_run
/// use aws_config::profile::{ProfileFileCredentialsProvider, ProfileFileRegionProvider};
/// use aws_config::profile::profile_file::{ProfileFiles, ProfileFileKind};
///
/// # async fn example() {
/// let sdk_config = aws_config::from_env()
Expand All @@ -624,8 +624,8 @@ mod loader {
/// exists to set a static endpoint for tools like `LocalStack`. When sending requests to
/// production AWS services, this method should only be used for service-specific behavior.
///
/// When this method is used, the [`Region`](aws_types::region::Region) is only used for
/// signing; it is not used to route the request.
/// When this method is used, the [`Region`](aws_types::region::Region) is only used for signing;
/// It is **not** used to route the request.
///
/// # Examples
///
Expand Down Expand Up @@ -823,6 +823,7 @@ mod loader {

// If an endpoint URL is set programmatically, then our work is done.
let endpoint_url = if self.endpoint_url.is_some() {
builder.insert_origin("endpoint_url", Origin::shared_config());
self.endpoint_url
} else {
// Otherwise, check to see if we should ignore EP URLs set in the environment.
Expand All @@ -840,7 +841,9 @@ mod loader {
None
} else {
// Otherwise, attempt to resolve one.
endpoint_url::endpoint_url_provider(&conf).await
let (v, origin) = endpoint_url::endpoint_url_provider_with_origin(&conf).await;
builder.insert_origin("endpoint_url", origin);
v
}
};
builder.set_endpoint_url(endpoint_url);
Expand Down Expand Up @@ -884,6 +887,7 @@ mod loader {
use aws_smithy_runtime::client::http::test_util::{infallible_client_fn, NeverClient};
use aws_smithy_runtime::test_util::capture_test_logs::capture_test_logs;
use aws_types::app_name::AppName;
use aws_types::origin::Origin;
use aws_types::os_shim_internal::{Env, Fs};
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc;
Expand Down Expand Up @@ -950,6 +954,81 @@ mod loader {
.http_client(no_traffic_client())
}

#[tokio::test]
async fn test_origin_programmatic() {
let _ = tracing_subscriber::fmt::try_init();
let loader = defaults(BehaviorVersion::latest())
.test_credentials()
.http_client(NeverClient::new())
.profile_name("custom")
.profile_files(
#[allow(deprecated)]
ProfileFiles::builder()
.with_contents(
#[allow(deprecated)]
ProfileFileKind::Config,
"[profile custom]\nendpoint_url = http://localhost:8989",
)
.build(),
)
.endpoint_url("http://localhost:1111")
.load()
.await;
assert_eq!(Origin::shared_config(), loader.get_origin("endpoint_url"));
}

#[tokio::test]
async fn test_origin_env() {
let _ = tracing_subscriber::fmt::try_init();
let env = Env::from_slice(&[("AWS_ENDPOINT_URL", "http://localhost:7878")]);
let loader = defaults(BehaviorVersion::latest())
.test_credentials()
.http_client(NeverClient::new())
.env(env)
.profile_name("custom")
.profile_files(
#[allow(deprecated)]
ProfileFiles::builder()
.with_contents(
#[allow(deprecated)]
ProfileFileKind::Config,
"[profile custom]\nendpoint_url = http://localhost:8989",
)
.build(),
)
.load()
.await;
assert_eq!(
Origin::shared_environment_variable(),
loader.get_origin("endpoint_url")
);
}

#[tokio::test]
async fn test_origin_fs() {
let _ = tracing_subscriber::fmt::try_init();
let loader = defaults(BehaviorVersion::latest())
.test_credentials()
.http_client(NeverClient::new())
.profile_name("custom")
.profile_files(
#[allow(deprecated)]
ProfileFiles::builder()
.with_contents(
#[allow(deprecated)]
ProfileFileKind::Config,
"[profile custom]\nendpoint_url = http://localhost:8989",
)
.build(),
)
.load()
.await;
assert_eq!(
Origin::shared_profile_file(),
loader.get_origin("endpoint_url")
);
}

Velfi marked this conversation as resolved.
Show resolved Hide resolved
#[tokio::test]
async fn load_fips() {
let conf = base_conf().use_fips(true).load().await;
Expand Down
2 changes: 1 addition & 1 deletion aws/rust-runtime/aws-runtime/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "aws-runtime"
version = "1.1.9"
version = "1.1.10"
authors = ["AWS Rust SDK Team <aws-sdk-rust@amazon.com>"]
description = "Runtime support code for the AWS SDK. This crate isn't intended to be used directly."
edition = "2021"
Expand Down
69 changes: 62 additions & 7 deletions aws/rust-runtime/aws-runtime/src/env_config.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@

use crate::env_config::property::PropertiesKey;
use crate::env_config::section::EnvConfigSections;
use aws_types::origin::Origin;
use aws_types::os_shim_internal::Env;
use aws_types::service_config::ServiceConfigKey;
use std::borrow::Cow;
Expand Down Expand Up @@ -81,6 +82,20 @@ pub struct EnvConfigSource<'a> {
scope: Scope<'a>,
}

#[allow(clippy::from_over_into)]
impl Into<Origin> for &EnvConfigSource<'_> {
fn into(self) -> Origin {
match (&self.scope, &self.location) {
(Scope::Global, Location::Environment) => Origin::shared_environment_variable(),
(Scope::Global, Location::Profile { .. }) => Origin::shared_profile_file(),
(Scope::Service { .. }, Location::Environment) => {
Origin::service_environment_variable()
}
(Scope::Service { .. }, Location::Profile { .. }) => Origin::service_profile_file(),
}
}
}

impl<'a> EnvConfigSource<'a> {
pub(crate) fn global_from_env(key: Cow<'a, str>) -> Self {
Self {
Expand Down Expand Up @@ -186,7 +201,7 @@ impl<'a> EnvConfigValue<'a> {
self
}

/// Load the value from `provider_config`, validating with `validator`
/// Load the value from the env or profile files, validating with `validator`
pub fn validate<T, E: Error + Send + Sync + 'static>(
self,
env: &Env,
Expand All @@ -204,12 +219,37 @@ impl<'a> EnvConfigValue<'a> {
.transpose()
}

/// Load the value from the env or profile files, validating with `validator`
///
/// This version of the function will also return the origin of the config.
pub fn validate_and_return_origin<T, E: Error + Send + Sync + 'static>(
self,
env: &Env,
profiles: Option<&EnvConfigSections>,
validator: impl Fn(&str) -> Result<T, E>,
) -> Result<(Option<T>, Origin), EnvConfigError<E>> {
let value = self.load(env, profiles);
match value {
Some((v, ctx)) => {
let origin: Origin = (&ctx).into();
validator(v.as_ref())
.map_err(|err| EnvConfigError {
property_source: format!("{}", ctx),
err,
})
.map(|value| (Some(value), origin))
}
None => Ok((None, Origin::unknown())),
}
}

/// Load the value from the environment
pub fn load(
&self,
env: &'a Env,
profiles: Option<&'a EnvConfigSections>,
) -> Option<(Cow<'a, str>, EnvConfigSource<'a>)> {
tracing::trace!("loading env config from {env:?}");
let env_value = self.environment_variable.as_ref().and_then(|env_var| {
// Check for a service-specific env var first
let service_config =
Expand All @@ -222,11 +262,19 @@ impl<'a> EnvConfigValue<'a> {
)
});

let value = service_config.or(global_config);
tracing::trace!("ENV value = {value:?}");
value
if let Some(v) = service_config {
tracing::trace!("(service env) {env_var} = {v:?}");
Some(v)
} else if let Some(v) = global_config {
tracing::trace!("(global env) {env_var} = {v:?}");
Some(v)
} else {
tracing::trace!("(env) no value set for {env_var}");
None
}
});

tracing::trace!("loading profile config from {profiles:?}");
Velfi marked this conversation as resolved.
Show resolved Hide resolved
let profile_value = match (profiles, self.profile_key.as_ref()) {
(Some(profiles), Some(profile_key)) => {
// Check for a service-specific profile key first
Expand All @@ -245,9 +293,16 @@ impl<'a> EnvConfigValue<'a> {
)
});

let value = service_config.or(global_config);
tracing::trace!("PROFILE value = {value:?}");
value
if let Some(v) = service_config {
tracing::trace!("(service profile) {profile_key} = {v:?}");
Some(v)
} else if let Some(v) = global_config {
tracing::trace!("(global profile) {profile_key} = {v:?}");
Some(v)
} else {
tracing::trace!("(service profile) no value set for {profile_key}");
None
}
}
_ => None,
};
Expand Down
2 changes: 1 addition & 1 deletion aws/rust-runtime/aws-types/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "aws-types"
version = "1.1.9"
version = "1.1.10"
Velfi marked this conversation as resolved.
Show resolved Hide resolved
authors = ["AWS Rust SDK Team <aws-sdk-rust@amazon.com>", "Russell Cohen <rcoh@amazon.com>"]
description = "Cross-service types for the AWS SDK."
edition = "2021"
Expand Down
1 change: 1 addition & 0 deletions aws/rust-runtime/aws-types/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
pub mod app_name;
pub mod build_metadata;
pub mod endpoint_config;
pub mod origin;
pub mod os_shim_internal;
pub mod region;
pub mod request_id;
Expand Down
Loading
Loading