diff --git a/aws/rust-runtime/aws-config/src/imds/client.rs b/aws/rust-runtime/aws-config/src/imds/client.rs index 67054a67f44..6d46692f75f 100644 --- a/aws/rust-runtime/aws-config/src/imds/client.rs +++ b/aws/rust-runtime/aws-config/src/imds/client.rs @@ -19,9 +19,11 @@ use aws_smithy_http::result::SdkError; use aws_smithy_runtime::client::orchestrator::operation::Operation; use aws_smithy_runtime::client::retries::strategy::StandardRetryStrategy; use aws_smithy_runtime_api::client::auth::AuthSchemeOptionResolverParams; -use aws_smithy_runtime_api::client::endpoint::{EndpointResolver, EndpointResolverParams}; +use aws_smithy_runtime_api::client::endpoint::{ + EndpointFuture, EndpointResolver, EndpointResolverParams, +}; use aws_smithy_runtime_api::client::interceptors::context::InterceptorContext; -use aws_smithy_runtime_api::client::orchestrator::{Future, OrchestratorError, SensitiveOutput}; +use aws_smithy_runtime_api::client::orchestrator::{OrchestratorError, SensitiveOutput}; use aws_smithy_runtime_api::client::retries::classifiers::{ ClassifyRetry, RetryAction, SharedRetryClassifier, }; @@ -522,15 +524,15 @@ struct ImdsEndpointResolver { } impl EndpointResolver for ImdsEndpointResolver { - fn resolve_endpoint(&self, _: &EndpointResolverParams) -> Future { + fn resolve_endpoint(&self, _: &EndpointResolverParams) -> EndpointFuture { let this = self.clone(); - Future::new(Box::pin(async move { + EndpointFuture::new(async move { this.endpoint_source .endpoint(this.mode_override) .await .map(|uri| Endpoint::builder().url(uri.to_string()).build()) .map_err(|err| err.into()) - })) + }) } } diff --git a/aws/rust-runtime/aws-config/src/imds/client/token.rs b/aws/rust-runtime/aws-config/src/imds/client/token.rs index ec3ffe406d6..0d6e309da83 100644 --- a/aws/rust-runtime/aws-config/src/imds/client/token.rs +++ b/aws/rust-runtime/aws-config/src/imds/client/token.rs @@ -25,11 +25,9 @@ use aws_smithy_runtime_api::client::auth::{ AuthScheme, AuthSchemeEndpointConfig, AuthSchemeId, Signer, }; use aws_smithy_runtime_api::client::identity::{ - Identity, IdentityResolver, SharedIdentityResolver, -}; -use aws_smithy_runtime_api::client::orchestrator::{ - Future, HttpRequest, HttpResponse, OrchestratorError, + Identity, IdentityFuture, IdentityResolver, SharedIdentityResolver, }; +use aws_smithy_runtime_api::client::orchestrator::{HttpRequest, HttpResponse, OrchestratorError}; use aws_smithy_runtime_api::client::runtime_components::{ GetIdentityResolver, RuntimeComponents, RuntimeComponentsBuilder, }; @@ -194,9 +192,9 @@ fn parse_token_response(response: &HttpResponse, now: SystemTime) -> Result Future { + fn resolve_identity(&self, _config_bag: &ConfigBag) -> IdentityFuture { let this = self.clone(); - Future::new(Box::pin(async move { + IdentityFuture::new(async move { let preloaded_token = this .inner .cache @@ -217,7 +215,7 @@ impl IdentityResolver for TokenResolver { let expiry = token.expiry; Ok(Identity::new(token, Some(expiry))) - })) + }) } } diff --git a/aws/rust-runtime/aws-runtime/src/identity.rs b/aws/rust-runtime/aws-runtime/src/identity.rs index 84e20d2afe6..1d42e13b3a8 100644 --- a/aws/rust-runtime/aws-runtime/src/identity.rs +++ b/aws/rust-runtime/aws-runtime/src/identity.rs @@ -6,9 +6,7 @@ /// Credentials-based identity support. pub mod credentials { use aws_credential_types::cache::SharedCredentialsCache; - use aws_smithy_runtime_api::box_error::BoxError; - use aws_smithy_runtime_api::client::identity::{Identity, IdentityResolver}; - use aws_smithy_runtime_api::client::orchestrator::Future; + use aws_smithy_runtime_api::client::identity::{Identity, IdentityFuture, IdentityResolver}; use aws_smithy_types::config_bag::ConfigBag; /// Smithy identity resolver for AWS credentials. @@ -25,13 +23,13 @@ pub mod credentials { } impl IdentityResolver for CredentialsIdentityResolver { - fn resolve_identity(&self, _config_bag: &ConfigBag) -> Future { + fn resolve_identity(&self, _config_bag: &ConfigBag) -> IdentityFuture { let cache = self.credentials_cache.clone(); - Future::new(Box::pin(async move { + IdentityFuture::new(async move { let credentials = cache.as_ref().provide_cached_credentials().await?; let expiration = credentials.expiry(); - Result::<_, BoxError>::Ok(Identity::new(credentials, expiration)) - })) + Ok(Identity::new(credentials, expiration)) + }) } } } diff --git a/rust-runtime/aws-smithy-runtime-api/src/client.rs b/rust-runtime/aws-smithy-runtime-api/src/client.rs index 2afc2546c56..08539a50ddd 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client.rs @@ -3,6 +3,68 @@ * SPDX-License-Identifier: Apache-2.0 */ +macro_rules! new_type_future { + ( + doc = $type_docs:literal, + pub struct $future_name:ident<$output:ty, $err:ty>, + ) => { + pin_project_lite::pin_project! { + #[allow(clippy::type_complexity)] + #[doc = $type_docs] + pub struct $future_name { + #[pin] + inner: aws_smithy_async::future::now_or_later::NowOrLater< + Result<$output, $err>, + aws_smithy_async::future::BoxFuture<$output, $err> + >, + } + } + + impl $future_name { + #[doc = concat!("Create a new `", stringify!($future_name), "` with the given future.")] + pub fn new(future: F) -> Self + where + F: std::future::Future> + Send + 'static, + { + Self { + inner: aws_smithy_async::future::now_or_later::NowOrLater::new(Box::pin(future)), + } + } + + #[doc = concat!(" + Create a new `", stringify!($future_name), "` with the given boxed future. + + Use this if you already have a boxed future to avoid double boxing it. + ")] + pub fn new_boxed( + future: std::pin::Pin< + Box> + Send>, + >, + ) -> Self { + Self { + inner: aws_smithy_async::future::now_or_later::NowOrLater::new(future), + } + } + + #[doc = concat!("Create a `", stringify!($future_name), "` that is immediately ready with the given result.")] + pub fn ready(result: Result<$output, $err>) -> Self { + Self { + inner: aws_smithy_async::future::now_or_later::NowOrLater::ready(result), + } + } + } + + impl std::future::Future for $future_name { + type Output = Result<$output, $err>; + + fn poll(self: std::pin::Pin<&mut Self>, cx: &mut std::task::Context<'_>) -> std::task::Poll { + let this = self.project(); + this.inner.poll(cx) + } + } + }; +} + pub mod dns; pub mod endpoint; diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/endpoint.rs b/rust-runtime/aws-smithy-runtime-api/src/client/endpoint.rs index 0037b9eb19b..3e885019e68 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/endpoint.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/endpoint.rs @@ -5,7 +5,7 @@ //! APIs needed to configure endpoint resolution for clients. -use crate::client::orchestrator::Future; +use crate::box_error::BoxError; use crate::impl_shared_conversions; use aws_smithy_types::config_bag::{Storable, StoreReplace}; use aws_smithy_types::endpoint::Endpoint; @@ -13,6 +13,11 @@ use aws_smithy_types::type_erasure::TypeErasedBox; use std::fmt; use std::sync::Arc; +new_type_future! { + doc = "Future for [`EndpointResolver::resolve_endpoint`].", + pub struct EndpointFuture, +} + /// Parameters originating from the Smithy endpoint ruleset required for endpoint resolution. /// /// The actual endpoint parameters are code generated from the Smithy model, and thus, @@ -40,7 +45,7 @@ impl Storable for EndpointResolverParams { /// Configurable endpoint resolver implementation. pub trait EndpointResolver: Send + Sync + fmt::Debug { /// Asynchronously resolves an endpoint to use from the given endpoint parameters. - fn resolve_endpoint(&self, params: &EndpointResolverParams) -> Future; + fn resolve_endpoint(&self, params: &EndpointResolverParams) -> EndpointFuture; } /// Shared endpoint resolver. @@ -57,7 +62,7 @@ impl SharedEndpointResolver { } impl EndpointResolver for SharedEndpointResolver { - fn resolve_endpoint(&self, params: &EndpointResolverParams) -> Future { + fn resolve_endpoint(&self, params: &EndpointResolverParams) -> EndpointFuture { self.0.resolve_endpoint(params) } } diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/http.rs b/rust-runtime/aws-smithy-runtime-api/src/client/http.rs index b5031a3f7d3..f3098b7204f 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/http.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/http.rs @@ -56,63 +56,14 @@ pub mod response; use crate::client::orchestrator::{HttpRequest, HttpResponse}; use crate::client::runtime_components::RuntimeComponents; use crate::impl_shared_conversions; -use aws_smithy_async::future::now_or_later::NowOrLater; use aws_smithy_http::result::ConnectorError; -use pin_project_lite::pin_project; use std::fmt; -use std::future::Future as StdFuture; -use std::pin::Pin; use std::sync::Arc; -use std::task::Poll; use std::time::Duration; -type BoxFuture = aws_smithy_async::future::BoxFuture; - -pin_project! { - /// Future for [`HttpConnector::call`]. - pub struct HttpConnectorFuture { - #[pin] - inner: NowOrLater, BoxFuture>, - } -} - -impl HttpConnectorFuture { - /// Create a new `HttpConnectorFuture` with the given future. - pub fn new(future: F) -> Self - where - F: StdFuture> + Send + 'static, - { - Self { - inner: NowOrLater::new(Box::pin(future)), - } - } - - /// Create a new `HttpConnectorFuture` with the given boxed future. - /// - /// Use this if you already have a boxed future to avoid double boxing it. - pub fn new_boxed( - future: Pin> + Send>>, - ) -> Self { - Self { - inner: NowOrLater::new(future), - } - } - - /// Create a `HttpConnectorFuture` that is immediately ready with the given result. - pub fn ready(result: Result) -> Self { - Self { - inner: NowOrLater::ready(result), - } - } -} - -impl StdFuture for HttpConnectorFuture { - type Output = Result; - - fn poll(self: Pin<&mut Self>, cx: &mut std::task::Context<'_>) -> Poll { - let this = self.project(); - this.inner.poll(cx) - } +new_type_future! { + doc = "Future for [`HttpConnector::call`].", + pub struct HttpConnectorFuture, } /// Trait with a `call` function that asynchronously converts a request into a response. diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/identity.rs b/rust-runtime/aws-smithy-runtime-api/src/client/identity.rs index ee115487c11..cd57910033a 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/identity.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/identity.rs @@ -3,8 +3,8 @@ * SPDX-License-Identifier: Apache-2.0 */ +use crate::box_error::BoxError; use crate::client::auth::AuthSchemeId; -use crate::client::orchestrator::Future; use crate::impl_shared_conversions; use aws_smithy_types::config_bag::ConfigBag; use std::any::Any; @@ -16,6 +16,11 @@ use std::time::SystemTime; #[cfg(feature = "http-auth")] pub mod http; +new_type_future! { + doc = "Future for [`IdentityResolver::resolve_identity`].", + pub struct IdentityFuture, +} + /// Resolver for identities. /// /// Every [`AuthScheme`](crate::client::auth::AuthScheme) has one or more compatible @@ -29,7 +34,7 @@ pub mod http; /// There is no fallback to other auth schemes in the absense of an identity. pub trait IdentityResolver: Send + Sync + Debug { /// Asynchronously resolves an identity for a request using the given config. - fn resolve_identity(&self, config_bag: &ConfigBag) -> Future; + fn resolve_identity(&self, config_bag: &ConfigBag) -> IdentityFuture; } /// Container for a shared identity resolver. @@ -44,7 +49,7 @@ impl SharedIdentityResolver { } impl IdentityResolver for SharedIdentityResolver { - fn resolve_identity(&self, config_bag: &ConfigBag) -> Future { + fn resolve_identity(&self, config_bag: &ConfigBag) -> IdentityFuture { self.0.resolve_identity(config_bag) } } diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/identity/http.rs b/rust-runtime/aws-smithy-runtime-api/src/client/identity/http.rs index 878943c545a..bf8204d9606 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/identity/http.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/identity/http.rs @@ -5,8 +5,7 @@ //! Identity types for HTTP auth -use crate::client::identity::{Identity, IdentityResolver}; -use crate::client::orchestrator::Future; +use crate::client::identity::{Identity, IdentityFuture, IdentityResolver}; use aws_smithy_types::config_bag::ConfigBag; use std::fmt::Debug; use std::sync::Arc; @@ -65,8 +64,8 @@ impl From for Token { } impl IdentityResolver for Token { - fn resolve_identity(&self, _config_bag: &ConfigBag) -> Future { - Future::ready(Ok(Identity::new(self.clone(), self.0.expiration))) + fn resolve_identity(&self, _config_bag: &ConfigBag) -> IdentityFuture { + IdentityFuture::ready(Ok(Identity::new(self.clone(), self.0.expiration))) } } @@ -124,7 +123,7 @@ impl Login { } impl IdentityResolver for Login { - fn resolve_identity(&self, _config_bag: &ConfigBag) -> Future { - Future::ready(Ok(Identity::new(self.clone(), self.0.expiration))) + fn resolve_identity(&self, _config_bag: &ConfigBag) -> IdentityFuture { + IdentityFuture::ready(Ok(Identity::new(self.clone(), self.0.expiration))) } } diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/orchestrator.rs b/rust-runtime/aws-smithy-runtime-api/src/client/orchestrator.rs index 328117648c9..bcafca33154 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/orchestrator.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/orchestrator.rs @@ -20,7 +20,6 @@ use crate::box_error::BoxError; use crate::client::interceptors::context::phase::Phase; use crate::client::interceptors::context::Error; use crate::client::interceptors::InterceptorError; -use aws_smithy_async::future::now_or_later::NowOrLater; use aws_smithy_http::body::SdkBody; use aws_smithy_http::result::{ConnectorError, SdkError}; use aws_smithy_types::config_bag::{Storable, StoreReplace}; @@ -34,18 +33,6 @@ pub type HttpRequest = http::Request; /// Type alias for the HTTP response type that the orchestrator uses. pub type HttpResponse = http::Response; -/// Type alias for boxed futures that are returned from several traits since async trait functions are not stable yet (as of 2023-07-21). -/// -/// See [the Rust blog](https://blog.rust-lang.org/inside-rust/2023/05/03/stabilizing-async-fn-in-trait.html) for -/// more information on async functions in traits. -pub type BoxFuture = aws_smithy_async::future::BoxFuture; - -/// Type alias for futures that are returned from several traits since async trait functions are not stable yet (as of 2023-07-21). -/// -/// See [the Rust blog](https://blog.rust-lang.org/inside-rust/2023/05/03/stabilizing-async-fn-in-trait.html) for -/// more information on async functions in traits. -pub type Future = NowOrLater, BoxFuture>; - /// Informs the orchestrator on whether or not the request body needs to be loaded into memory before transmit. /// /// This enum gets placed into the `ConfigBag` to change the orchestrator behavior. diff --git a/rust-runtime/aws-smithy-runtime-api/src/client/runtime_components.rs b/rust-runtime/aws-smithy-runtime-api/src/client/runtime_components.rs index 9d262738d19..c70162ea4c6 100644 --- a/rust-runtime/aws-smithy-runtime-api/src/client/runtime_components.rs +++ b/rust-runtime/aws-smithy-runtime-api/src/client/runtime_components.rs @@ -548,11 +548,9 @@ impl RuntimeComponentsBuilder { /// Creates a runtime components builder with all the required components filled in with fake (panicking) implementations. #[cfg(feature = "test-util")] pub fn for_tests() -> Self { - use crate::client::endpoint::EndpointResolverParams; - use crate::client::identity::Identity; - use crate::client::orchestrator::Future; + use crate::client::endpoint::{EndpointFuture, EndpointResolverParams}; + use crate::client::identity::IdentityFuture; use aws_smithy_types::config_bag::ConfigBag; - use aws_smithy_types::endpoint::Endpoint; #[derive(Debug)] struct FakeAuthSchemeOptionResolver; @@ -581,7 +579,7 @@ impl RuntimeComponentsBuilder { #[derive(Debug)] struct FakeEndpointResolver; impl EndpointResolver for FakeEndpointResolver { - fn resolve_endpoint(&self, _: &EndpointResolverParams) -> Future { + fn resolve_endpoint(&self, _: &EndpointResolverParams) -> EndpointFuture { unreachable!("fake endpoint resolver must be overridden for this test") } } @@ -608,7 +606,7 @@ impl RuntimeComponentsBuilder { #[derive(Debug)] struct FakeIdentityResolver; impl IdentityResolver for FakeIdentityResolver { - fn resolve_identity(&self, _: &ConfigBag) -> Future { + fn resolve_identity(&self, _: &ConfigBag) -> IdentityFuture { unreachable!("fake identity resolver must be overridden for this test") } } diff --git a/rust-runtime/aws-smithy-runtime/src/client/identity/no_auth.rs b/rust-runtime/aws-smithy-runtime/src/client/identity/no_auth.rs index 01121eab909..8a6ec5ab0dc 100644 --- a/rust-runtime/aws-smithy-runtime/src/client/identity/no_auth.rs +++ b/rust-runtime/aws-smithy-runtime/src/client/identity/no_auth.rs @@ -3,8 +3,7 @@ * SPDX-License-Identifier: Apache-2.0 */ -use aws_smithy_runtime_api::client::identity::{Identity, IdentityResolver}; -use aws_smithy_runtime_api::client::orchestrator::Future; +use aws_smithy_runtime_api::client::identity::{Identity, IdentityFuture, IdentityResolver}; use aws_smithy_types::config_bag::ConfigBag; /// Identity for the [`NoAuthScheme`](crate::client::auth::no_auth::NoAuthScheme) auth scheme. @@ -30,7 +29,7 @@ impl NoAuthIdentityResolver { } impl IdentityResolver for NoAuthIdentityResolver { - fn resolve_identity(&self, _: &ConfigBag) -> Future { - Future::ready(Ok(Identity::new(NoAuthIdentity::new(), None))) + fn resolve_identity(&self, _: &ConfigBag) -> IdentityFuture { + IdentityFuture::ready(Ok(Identity::new(NoAuthIdentity::new(), None))) } } diff --git a/rust-runtime/aws-smithy-runtime/src/client/orchestrator/auth.rs b/rust-runtime/aws-smithy-runtime/src/client/orchestrator/auth.rs index bbc6001aafd..56f6dab0dc4 100644 --- a/rust-runtime/aws-smithy-runtime/src/client/orchestrator/auth.rs +++ b/rust-runtime/aws-smithy-runtime/src/client/orchestrator/auth.rs @@ -145,10 +145,10 @@ mod tests { SharedAuthSchemeOptionResolver, Signer, }; use aws_smithy_runtime_api::client::identity::{ - Identity, IdentityResolver, SharedIdentityResolver, + Identity, IdentityFuture, IdentityResolver, SharedIdentityResolver, }; use aws_smithy_runtime_api::client::interceptors::context::{Input, InterceptorContext}; - use aws_smithy_runtime_api::client::orchestrator::{Future, HttpRequest}; + use aws_smithy_runtime_api::client::orchestrator::HttpRequest; use aws_smithy_runtime_api::client::runtime_components::{ GetIdentityResolver, RuntimeComponentsBuilder, }; @@ -160,8 +160,8 @@ mod tests { #[derive(Debug)] struct TestIdentityResolver; impl IdentityResolver for TestIdentityResolver { - fn resolve_identity(&self, _config_bag: &ConfigBag) -> Future { - Future::ready(Ok(Identity::new("doesntmatter", None))) + fn resolve_identity(&self, _config_bag: &ConfigBag) -> IdentityFuture { + IdentityFuture::ready(Ok(Identity::new("doesntmatter", None))) } } diff --git a/rust-runtime/aws-smithy-runtime/src/client/orchestrator/endpoints.rs b/rust-runtime/aws-smithy-runtime/src/client/orchestrator/endpoints.rs index ffb3ac92123..7afd3c41660 100644 --- a/rust-runtime/aws-smithy-runtime/src/client/orchestrator/endpoints.rs +++ b/rust-runtime/aws-smithy-runtime/src/client/orchestrator/endpoints.rs @@ -9,9 +9,11 @@ use aws_smithy_http::endpoint::{ SharedEndpointResolver, }; use aws_smithy_runtime_api::box_error::BoxError; -use aws_smithy_runtime_api::client::endpoint::{EndpointResolver, EndpointResolverParams}; +use aws_smithy_runtime_api::client::endpoint::{ + EndpointFuture, EndpointResolver, EndpointResolverParams, +}; use aws_smithy_runtime_api::client::interceptors::context::InterceptorContext; -use aws_smithy_runtime_api::client::orchestrator::{Future, HttpRequest}; +use aws_smithy_runtime_api::client::orchestrator::HttpRequest; use aws_smithy_runtime_api::client::runtime_components::RuntimeComponents; use aws_smithy_types::config_bag::{ConfigBag, Storable, StoreReplace}; use aws_smithy_types::endpoint::Endpoint; @@ -44,8 +46,8 @@ impl StaticUriEndpointResolver { } impl EndpointResolver for StaticUriEndpointResolver { - fn resolve_endpoint(&self, _params: &EndpointResolverParams) -> Future { - Future::ready(Ok(Endpoint::builder() + fn resolve_endpoint(&self, _params: &EndpointResolverParams) -> EndpointFuture { + EndpointFuture::ready(Ok(Endpoint::builder() .url(self.endpoint.to_string()) .build())) } @@ -99,7 +101,7 @@ impl EndpointResolver for DefaultEndpointResolver where Params: Debug + Send + Sync + 'static, { - fn resolve_endpoint(&self, params: &EndpointResolverParams) -> Future { + fn resolve_endpoint(&self, params: &EndpointResolverParams) -> EndpointFuture { let ep = match params.get::() { Some(params) => self.inner.resolve_endpoint(params).map_err(Box::new), None => Err(Box::new(ResolveEndpointError::message( @@ -107,7 +109,7 @@ where ))), } .map_err(|e| e as _); - Future::ready(ep) + EndpointFuture::ready(ep) } }