From 7fdd97b28a1d3e10e2191751353df2a0ebc03621 Mon Sep 17 00:00:00 2001 From: JESS IZEN Date: Tue, 10 Dec 2024 19:17:20 +0000 Subject: [PATCH 1/4] Add `util::BoxCloneSyncServiceLayer` --- tower/src/util/boxed/layer_clone_sync.rs | 129 +++++++++++++++++++++++ tower/src/util/boxed/mod.rs | 3 +- tower/src/util/boxed_clone_sync.rs | 2 +- tower/src/util/mod.rs | 2 +- 4 files changed, 133 insertions(+), 3 deletions(-) create mode 100644 tower/src/util/boxed/layer_clone_sync.rs diff --git a/tower/src/util/boxed/layer_clone_sync.rs b/tower/src/util/boxed/layer_clone_sync.rs new file mode 100644 index 000000000..3ee86a839 --- /dev/null +++ b/tower/src/util/boxed/layer_clone_sync.rs @@ -0,0 +1,129 @@ +use std::{fmt, sync::Arc}; +use tower_layer::{layer_fn, Layer}; +use tower_service::Service; + +use crate::util::BoxCloneSyncService; + +/// A [`Clone`] + [`Send`] + [`Sync`] boxed [`Layer`]. +/// +/// [`BoxCloneSyncServiceLayer`] turns a layer into a trait object, allowing both the [`Layer`] itself +/// and the output [`Service`] to be dynamic, while having consistent types. +/// +/// This [`Layer`] produces [`BoxCloneSyncService`] instances erasing the type of the +/// [`Service`] produced by the wrapped [`Layer`]. +/// +/// This is similar to [`BoxCloneServiceLayer`](super::BoxCloneServiceLayer) except the layer and resulting +/// service implements [`Sync`]. +/// +/// # Example +/// +/// `BoxCloneSyncServiceLayer` can, for example, be useful to create layers dynamically that otherwise wouldn't have +/// the same types, when the underlying service must be clone and sync (for example, when building a Hyper connector) +/// In this example, we include a [`Timeout`] layer only if an environment variable is set. We can use +/// `BoxCloneSyncServiceLayer` to return a consistent type regardless of runtime configuration: +/// +/// ``` +/// use std::time::Duration; +/// use tower::{Service, ServiceBuilder, BoxError}; +/// use tower::util::{BoxCloneSyncServiceLayer, BoxCloneSyncService}; +/// +/// # +/// # struct Request; +/// # struct Response; +/// # impl Response { +/// # fn new() -> Self { Self } +/// # } +/// +/// fn common_layer() -> BoxCloneSyncServiceLayer +/// where +/// S: Service + Clone + Send + Sync + 'static, +/// S::Future: Send + 'static, +/// S::Error: Into + 'static, +/// { +/// let builder = ServiceBuilder::new() +/// .concurrency_limit(100); +/// +/// if std::env::var("SET_TIMEOUT").is_ok() { +/// let layer = builder +/// .timeout(Duration::from_secs(30)) +/// .into_inner(); +/// +/// BoxCloneSyncServiceLayer::new(layer) +/// } else { +/// let layer = builder +/// .map_err(Into::into) +/// .into_inner(); +/// +/// BoxCloneSyncServiceLayer::new(layer) +/// } +/// } +/// +/// // We can clone the layer (this is true of BoxLayer as well) +/// let boxed_clone_sync_layer = common_layer(); +/// +/// let cloned_sync_layer = boxed_clone_sync_layer.clone(); +/// +/// // Using the `BoxCloneSyncServiceLayer` we can create a `BoxCloneSyncService` +/// let service: BoxCloneSyncService = ServiceBuilder::new().layer(cloned_sync_layer) +/// .service_fn(|req: Request| async { +/// Ok::<_, BoxError>(Response::new()) +/// }); +/// +/// # let service = assert_service(service); +/// +/// // And we can still clone the service +/// let cloned_service = service.clone(); +/// # +/// # fn assert_service(svc: S) -> S +/// # where S: Service { svc } +/// +/// ``` +/// +/// [`Layer`]: tower_layer::Layer +/// [`Service`]: tower_service::Service +/// [`BoxService`]: super::BoxService +/// [`Timeout`]: crate::timeout +pub struct BoxCloneSyncServiceLayer { + boxed: Arc> + Send + Sync + 'static>, +} + +impl BoxCloneSyncServiceLayer { + /// Create a new [`BoxCloneServiceLayer`]. + pub fn new(inner_layer: L) -> Self + where + L: Layer + Send + Sync + 'static, + L::Service: Service + Send + Sync + Clone + 'static, + >::Future: Send + 'static, + { + let layer = layer_fn(move |inner: In| { + let out = inner_layer.layer(inner); + BoxCloneSyncService::new(out) + }); + + Self { + boxed: Arc::new(layer), + } + } +} + +impl Layer for BoxCloneSyncServiceLayer { + type Service = BoxCloneSyncService; + + fn layer(&self, inner: In) -> Self::Service { + self.boxed.layer(inner) + } +} + +impl Clone for BoxCloneSyncServiceLayer { + fn clone(&self) -> Self { + Self { + boxed: Arc::clone(&self.boxed), + } + } +} + +impl fmt::Debug for BoxCloneSyncServiceLayer { + fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { + fmt.debug_struct("BoxCloneSyncServiceLayer").finish() + } +} \ No newline at end of file diff --git a/tower/src/util/boxed/mod.rs b/tower/src/util/boxed/mod.rs index 93a9e0dfe..4fd87d568 100644 --- a/tower/src/util/boxed/mod.rs +++ b/tower/src/util/boxed/mod.rs @@ -1,9 +1,10 @@ mod layer; mod layer_clone; +mod layer_clone_sync; mod sync; mod unsync; #[allow(unreachable_pub)] // https://github.com/rust-lang/rust/issues/57411 pub use self::{ - layer::BoxLayer, layer_clone::BoxCloneServiceLayer, sync::BoxService, unsync::UnsyncBoxService, + layer::BoxLayer, layer_clone::BoxCloneServiceLayer, layer_clone_sync::BoxCloneSyncServiceLayer, sync::BoxService, unsync::UnsyncBoxService, }; diff --git a/tower/src/util/boxed_clone_sync.rs b/tower/src/util/boxed_clone_sync.rs index e8cbe0077..d62e8ff22 100644 --- a/tower/src/util/boxed_clone_sync.rs +++ b/tower/src/util/boxed_clone_sync.rs @@ -24,7 +24,7 @@ pub struct BoxCloneSyncService( ); impl BoxCloneSyncService { - /// Create a new `BoxCloneService`. + /// Create a new `BoxCloneSyncService`. pub fn new(inner: S) -> Self where S: Service + Clone + Send + Sync + 'static, diff --git a/tower/src/util/mod.rs b/tower/src/util/mod.rs index dcc35cc42..4021f22a4 100644 --- a/tower/src/util/mod.rs +++ b/tower/src/util/mod.rs @@ -24,7 +24,7 @@ pub mod rng; pub use self::{ and_then::{AndThen, AndThenLayer}, - boxed::{BoxCloneServiceLayer, BoxLayer, BoxService, UnsyncBoxService}, + boxed::{BoxCloneServiceLayer, BoxCloneSyncServiceLayer, BoxLayer, BoxService, UnsyncBoxService}, boxed_clone::BoxCloneService, boxed_clone_sync::BoxCloneSyncService, either::Either, From 19bbd7be947b333cc3bbb97b6746ce7d79dcdbdc Mon Sep 17 00:00:00 2001 From: JESS IZEN Date: Tue, 10 Dec 2024 19:20:02 +0000 Subject: [PATCH 2/4] typo fix --- tower/src/util/boxed/layer_clone_sync.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tower/src/util/boxed/layer_clone_sync.rs b/tower/src/util/boxed/layer_clone_sync.rs index 3ee86a839..2356e337c 100644 --- a/tower/src/util/boxed/layer_clone_sync.rs +++ b/tower/src/util/boxed/layer_clone_sync.rs @@ -18,7 +18,7 @@ use crate::util::BoxCloneSyncService; /// # Example /// /// `BoxCloneSyncServiceLayer` can, for example, be useful to create layers dynamically that otherwise wouldn't have -/// the same types, when the underlying service must be clone and sync (for example, when building a Hyper connector) +/// the same types, when the underlying service must be clone and sync (for example, when building a Hyper connector). /// In this example, we include a [`Timeout`] layer only if an environment variable is set. We can use /// `BoxCloneSyncServiceLayer` to return a consistent type regardless of runtime configuration: /// @@ -126,4 +126,4 @@ impl fmt::Debug for BoxCloneSyncServiceLayer { fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { fmt.debug_struct("BoxCloneSyncServiceLayer").finish() } -} \ No newline at end of file +} From af7a83bc9f8e04216aba11bbcf063e2ee27519b7 Mon Sep 17 00:00:00 2001 From: JESS IZEN Date: Tue, 10 Dec 2024 19:20:42 +0000 Subject: [PATCH 3/4] run rustfmt --- tower/src/util/boxed/mod.rs | 3 ++- tower/src/util/mod.rs | 4 +++- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/tower/src/util/boxed/mod.rs b/tower/src/util/boxed/mod.rs index 4fd87d568..7da5d63cb 100644 --- a/tower/src/util/boxed/mod.rs +++ b/tower/src/util/boxed/mod.rs @@ -6,5 +6,6 @@ mod unsync; #[allow(unreachable_pub)] // https://github.com/rust-lang/rust/issues/57411 pub use self::{ - layer::BoxLayer, layer_clone::BoxCloneServiceLayer, layer_clone_sync::BoxCloneSyncServiceLayer, sync::BoxService, unsync::UnsyncBoxService, + layer::BoxLayer, layer_clone::BoxCloneServiceLayer, layer_clone_sync::BoxCloneSyncServiceLayer, + sync::BoxService, unsync::UnsyncBoxService, }; diff --git a/tower/src/util/mod.rs b/tower/src/util/mod.rs index 4021f22a4..4c56de813 100644 --- a/tower/src/util/mod.rs +++ b/tower/src/util/mod.rs @@ -24,7 +24,9 @@ pub mod rng; pub use self::{ and_then::{AndThen, AndThenLayer}, - boxed::{BoxCloneServiceLayer, BoxCloneSyncServiceLayer, BoxLayer, BoxService, UnsyncBoxService}, + boxed::{ + BoxCloneServiceLayer, BoxCloneSyncServiceLayer, BoxLayer, BoxService, UnsyncBoxService, + }, boxed_clone::BoxCloneService, boxed_clone_sync::BoxCloneSyncService, either::Either, From e4ccf1335e94cd978cee72e7b6b82e06076c9ff7 Mon Sep 17 00:00:00 2001 From: JESS IZEN Date: Tue, 10 Dec 2024 19:25:16 +0000 Subject: [PATCH 4/4] fix type in comment --- tower/src/util/boxed/layer_clone_sync.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tower/src/util/boxed/layer_clone_sync.rs b/tower/src/util/boxed/layer_clone_sync.rs index 2356e337c..950e66be6 100644 --- a/tower/src/util/boxed/layer_clone_sync.rs +++ b/tower/src/util/boxed/layer_clone_sync.rs @@ -88,7 +88,7 @@ pub struct BoxCloneSyncServiceLayer { } impl BoxCloneSyncServiceLayer { - /// Create a new [`BoxCloneServiceLayer`]. + /// Create a new [`BoxCloneSyncServiceLayer`]. pub fn new(inner_layer: L) -> Self where L: Layer + Send + Sync + 'static,