diff --git a/ntex-service/CHANGES.md b/ntex-service/CHANGES.md index 973c5292..e610e0a9 100644 --- a/ntex-service/CHANGES.md +++ b/ntex-service/CHANGES.md @@ -1,5 +1,11 @@ # Changes +## [0.2.0-b.0] - 2021-08-26 + +* Simplify Transform trait + +* Add PipelineFactory::apply() combinator + ## [0.1.9] - 2021-06-03 * Add rc wrapped service, `RcService` diff --git a/ntex-service/Cargo.toml b/ntex-service/Cargo.toml index bbc5f9ab..33de04ee 100644 --- a/ntex-service/Cargo.toml +++ b/ntex-service/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "ntex-service" -version = "0.1.9" +version = "0.2.0-b.0" authors = ["ntex contributors "] description = "ntex service" keywords = ["network", "framework", "async", "futures"] diff --git a/ntex-service/src/fn_transform.rs b/ntex-service/src/fn_transform.rs index b1297165..a0c61f01 100644 --- a/ntex-service/src/fn_transform.rs +++ b/ntex-service/src/fn_transform.rs @@ -1,12 +1,9 @@ -use ntex_util::future::Ready; use std::{future::Future, marker::PhantomData}; use crate::{apply_fn, dev::Apply, Service, Transform}; /// Use function as transform service -pub fn fn_transform( - f: F, -) -> impl Transform + Clone +pub fn fn_transform(f: F) -> FnTransform where S: Service, F: Fn(Req, &S) -> R + Clone, @@ -42,15 +39,10 @@ where F: Fn(Req, &S) -> R + Clone, R: Future>, { - type Request = Req; - type Response = Res; - type Error = Err; type Transform = Apply; - type InitError = (); - type Future = Ready; - fn new_transform(&self, service: S) -> Self::Future { - Ready::Ok(apply_fn(service, self.f.clone())) + fn new_transform(&self, service: S) -> Self::Transform { + apply_fn(service, self.f.clone()) } } @@ -68,7 +60,7 @@ where #[cfg(test)] #[allow(clippy::redundant_clone)] mod tests { - use ntex_util::future::lazy; + use ntex_util::future::{lazy, Ready}; use std::task::{Context, Poll}; use super::*; @@ -94,12 +86,7 @@ mod tests { #[ntex::test] async fn transform() { - let srv = - fn_transform::(|i: usize, srv: &_| srv.call(i + 1)) - .clone() - .new_transform(Srv) - .await - .unwrap(); + let srv = fn_transform(|i: usize, srv: &Srv| srv.call(i + 1)).new_transform(Srv); let res = srv.call(10usize).await; assert!(res.is_ok()); diff --git a/ntex-service/src/lib.rs b/ntex-service/src/lib.rs index 14924c26..166c91bc 100644 --- a/ntex-service/src/lib.rs +++ b/ntex-service/src/lib.rs @@ -19,7 +19,6 @@ mod map_init_err; mod pipeline; mod then; mod transform; -mod transform_err; pub use self::apply::{apply_fn, apply_fn_factory}; pub use self::fn_service::{ @@ -342,17 +341,10 @@ pub mod dev { pub use crate::fn_service::{ FnMutService, FnService, FnServiceConfig, FnServiceFactory, FnServiceNoConfig, }; + pub use crate::fn_transform::FnTransform; pub use crate::map::{Map, MapServiceFactory}; pub use crate::map_config::{MapConfig, UnitConfig}; pub use crate::map_err::{MapErr, MapErrServiceFactory}; pub use crate::map_init_err::MapInitErr; pub use crate::transform::ApplyTransform; - pub use crate::transform_err::TransformMapInitErr; -} - -#[doc(hidden)] -pub mod util { - pub use ntex_util::future::Either; - pub use ntex_util::future::Ready; - pub use ntex_util::future::{lazy, Lazy}; } diff --git a/ntex-service/src/pipeline.rs b/ntex-service/src/pipeline.rs index d280dbf9..00a5b49a 100644 --- a/ntex-service/src/pipeline.rs +++ b/ntex-service/src/pipeline.rs @@ -7,6 +7,7 @@ use crate::map::{Map, MapServiceFactory}; use crate::map_err::{MapErr, MapErrServiceFactory}; use crate::map_init_err::MapInitErr; use crate::then::{ThenService, ThenServiceFactory}; +use crate::transform::{ApplyTransform, Transform}; use crate::{IntoService, IntoServiceFactory, Service, ServiceFactory}; /// Contruct new pipeline with one service in pipeline chain. @@ -249,6 +250,18 @@ impl PipelineFactory { } } + /// Apply transform to current service factory. + /// + /// Short version of `apply(transform, pipeline_factory(...))` + pub fn apply(self, tr: U) -> PipelineFactory> + where + U: Transform, + { + PipelineFactory { + factory: ApplyTransform::new(tr, self.factory), + } + } + /// Create `NewService` to chain on a computation for when a call to the /// service finished, passing the result of the call to the next /// service `U`. diff --git a/ntex-service/src/transform.rs b/ntex-service/src/transform.rs index b68d7c39..d8e24de8 100644 --- a/ntex-service/src/transform.rs +++ b/ntex-service/src/transform.rs @@ -1,13 +1,12 @@ use std::{future::Future, pin::Pin, rc::Rc, task::Context, task::Poll}; -use crate::transform_err::TransformMapInitErr; use crate::{IntoServiceFactory, Service, ServiceFactory}; /// Apply transform to a service. pub fn apply(t: T, factory: U) -> ApplyTransform where S: ServiceFactory, - T: Transform, + T: Transform, U: IntoServiceFactory, { ApplyTransform::new(t, factory.into_factory()) @@ -70,14 +69,9 @@ where /// where /// S: Service, /// { -/// type Request = S::Request; -/// type Response = S::Response; -/// type Error = TimeoutError; -/// type InitError = S::Error; /// type Transform = Timeout; -/// type Future = Ready>; /// -/// fn new_transform(&self, service: S) -> Self::Future { +/// fn new_transform(&self, service: S) -> Self::Transform { /// ok(TimeoutService { /// service, /// timeout: self.timeout, @@ -86,54 +80,20 @@ where /// } /// ``` pub trait Transform { - /// Requests handled by the service. - type Request; - - /// Responses given by the service. - type Response; - - /// Errors produced by the service. - type Error; - /// The `TransformService` value created by this factory - type Transform: Service< - Request = Self::Request, - Response = Self::Response, - Error = Self::Error, - >; - - /// Errors produced while building a transform service. - type InitError; - - /// The future response value. - type Future: Future>; + type Transform: Service; /// Creates and returns a new Transform component, asynchronously - fn new_transform(&self, service: S) -> Self::Future; - - /// Map this transforms's factory error to a different error, - /// returning a new transform service factory. - fn map_init_err(self, f: F) -> TransformMapInitErr - where - Self: Sized, - F: Fn(Self::InitError) -> E + Clone, - { - TransformMapInitErr::new(self, f) - } + fn new_transform(&self, service: S) -> Self::Transform; } impl Transform for Rc where T: Transform, { - type Request = T::Request; - type Response = T::Response; - type Error = T::Error; - type InitError = T::InitError; type Transform = T::Transform; - type Future = T::Future; - fn new_transform(&self, service: S) -> T::Future { + fn new_transform(&self, service: S) -> T::Transform { self.as_ref().new_transform(service) } } @@ -144,10 +104,10 @@ pub struct ApplyTransform(Rc<(T, S)>); impl ApplyTransform where S: ServiceFactory, - T: Transform, + T: Transform, { /// Create new `ApplyTransform` new service instance - fn new(t: T, service: S) -> Self { + pub(crate) fn new(t: T, service: S) -> Self { Self(Rc::new((t, service))) } } @@ -161,23 +121,21 @@ impl Clone for ApplyTransform { impl ServiceFactory for ApplyTransform where S: ServiceFactory, - T: Transform, + T: Transform, { - type Request = T::Request; - type Response = T::Response; - type Error = T::Error; + type Request = ::Request; + type Response = ::Response; + type Error = ::Error; type Config = S::Config; type Service = T::Transform; - type InitError = T::InitError; + type InitError = S::InitError; type Future = ApplyTransformFuture; fn new_service(&self, cfg: S::Config) -> Self::Future { ApplyTransformFuture { store: self.0.clone(), - state: ApplyTransformFutureState::A { - fut: self.0.as_ref().1.new_service(cfg), - }, + fut: self.0.as_ref().1.new_service(cfg), } } } @@ -186,46 +144,27 @@ pin_project_lite::pin_project! { pub struct ApplyTransformFuture where S: ServiceFactory, - T: Transform, + T: Transform, { store: Rc<(T, S)>, #[pin] - state: ApplyTransformFutureState, - } -} - -pin_project_lite::pin_project! { - #[project = ApplyTransformFutureStateProject] - pub enum ApplyTransformFutureState - where - S: ServiceFactory, - T: Transform, - { - A { #[pin] fut: S::Future }, - B { #[pin] fut: T::Future }, + fut: S::Future, } } impl Future for ApplyTransformFuture where S: ServiceFactory, - T: Transform, + T: Transform, { - type Output = Result; + type Output = Result; fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { - let mut this = self.as_mut().project(); + let this = self.as_mut().project(); - match this.state.as_mut().project() { - ApplyTransformFutureStateProject::A { fut } => match fut.poll(cx)? { - Poll::Ready(srv) => { - let fut = this.store.0.new_transform(srv); - this.state.set(ApplyTransformFutureState::B { fut }); - self.poll(cx) - } - Poll::Pending => Poll::Pending, - }, - ApplyTransformFutureStateProject::B { fut } => fut.poll(cx), + match this.fut.poll(cx)? { + Poll::Ready(srv) => Poll::Ready(Ok(this.store.0.new_transform(srv))), + Poll::Pending => Poll::Pending, } } } @@ -242,16 +181,10 @@ mod tests { struct Tr; impl Transform for Tr { - type Request = S::Request; - type Response = S::Response; - type Error = S::Error; - type Transform = Srv; - type InitError = (); - type Future = Ready; - fn new_transform(&self, service: S) -> Self::Future { - Ready::Ok(Srv(service)) + fn new_transform(&self, service: S) -> Self::Transform { + Srv(service) } } @@ -276,7 +209,7 @@ mod tests { #[ntex::test] async fn transform() { let factory = apply( - Rc::new(Tr.map_init_err(|_| ()).clone()), + Rc::new(Tr.clone()), fn_service(|i: usize| Ready::<_, ()>::Ok(i * 2)), ) .clone(); diff --git a/ntex-service/src/transform_err.rs b/ntex-service/src/transform_err.rs deleted file mode 100644 index ed32ebe8..00000000 --- a/ntex-service/src/transform_err.rs +++ /dev/null @@ -1,94 +0,0 @@ -use std::future::Future; -use std::marker::PhantomData; -use std::pin::Pin; -use std::task::{Context, Poll}; - -use super::Transform; - -/// Transform for the `map_init_err` combinator, changing the type of a new -/// transform's init error. -/// -/// This is created by the `Transform::map_init_err` method. -pub struct TransformMapInitErr { - t: T, - f: F, - e: PhantomData<(S, E)>, -} - -impl TransformMapInitErr { - pub(crate) fn new(t: T, f: F) -> Self - where - T: Transform, - F: Fn(T::InitError) -> E, - { - Self { - t, - f, - e: PhantomData, - } - } -} - -impl Clone for TransformMapInitErr -where - T: Clone, - F: Clone, -{ - fn clone(&self) -> Self { - Self { - t: self.t.clone(), - f: self.f.clone(), - e: PhantomData, - } - } -} - -impl Transform for TransformMapInitErr -where - T: Transform, - F: Fn(T::InitError) -> E + Clone, -{ - type Request = T::Request; - type Response = T::Response; - type Error = T::Error; - type Transform = T::Transform; - - type InitError = E; - type Future = TransformMapInitErrFuture; - - fn new_transform(&self, service: S) -> Self::Future { - TransformMapInitErrFuture { - fut: self.t.new_transform(service), - f: self.f.clone(), - } - } -} - -pin_project_lite::pin_project! { -pub struct TransformMapInitErrFuture -where - T: Transform, - F: Fn(T::InitError) -> E, -{ - #[pin] - fut: T::Future, - f: F, -} -} - -impl Future for TransformMapInitErrFuture -where - T: Transform, - F: Fn(T::InitError) -> E + Clone, -{ - type Output = Result; - - fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { - let this = self.project(); - if let Poll::Ready(res) = this.fut.poll(cx) { - Poll::Ready(res.map_err(this.f)) - } else { - Poll::Pending - } - } -}