diff --git a/ntex-service/CHANGES.md b/ntex-service/CHANGES.md index ff8d07b5..0f34f87c 100644 --- a/ntex-service/CHANGES.md +++ b/ntex-service/CHANGES.md @@ -1,5 +1,9 @@ # Changes +## [0.1.1] - 2020-04-22 + +* Add `map_config_service`, replacement for `apply_cfg` + ## [0.1.0] - 2020-03-31 * Fork to ntex namespace diff --git a/ntex-service/Cargo.toml b/ntex-service/Cargo.toml index 32aeab32..6a2170df 100644 --- a/ntex-service/Cargo.toml +++ b/ntex-service/Cargo.toml @@ -1,12 +1,12 @@ [package] name = "ntex-service" -version = "0.1.0" +version = "0.1.1" authors = ["Nikolay Kim "] -description = "Actix service" +description = "ntex service" keywords = ["network", "framework", "async", "futures"] -homepage = "https://actix.rs" -repository = "https://github.com/actix/actix-net.git" -documentation = "https://docs.rs/actix-service/" +homepage = "https://ntex.rs" +repository = "https://github.com/ntex-rs/ntex.git" +documentation = "https://docs.rs/ntex-service/" categories = ["network-programming", "asynchronous"] license = "MIT" edition = "2018" diff --git a/ntex-service/src/lib.rs b/ntex-service/src/lib.rs index 6d9cc474..3779cac3 100644 --- a/ntex-service/src/lib.rs +++ b/ntex-service/src/lib.rs @@ -21,14 +21,16 @@ mod transform; mod transform_err; pub use self::apply::{apply_fn, apply_fn_factory}; -pub use self::apply_cfg::{apply_cfg, apply_cfg_factory}; pub use self::fn_service::{ fn_factory, fn_factory_with_config, fn_mut_service, fn_service, }; -pub use self::map_config::{map_config, unit_config}; +pub use self::map_config::{map_config, map_config_service, unit_config}; pub use self::pipeline::{pipeline, pipeline_factory, Pipeline, PipelineFactory}; pub use self::transform::{apply, Transform}; +#[doc(hidden)] +pub use self::apply_cfg::{apply_cfg, apply_cfg_factory}; + /// An asynchronous function from `Request` to a `Response`. /// /// `Service` represents a service that represanting interation, taking requests and giving back diff --git a/ntex-service/src/map_config.rs b/ntex-service/src/map_config.rs index f610e957..520d370f 100644 --- a/ntex-service/src/map_config.rs +++ b/ntex-service/src/map_config.rs @@ -1,6 +1,13 @@ +use std::cell::RefCell; +use std::future::Future; use std::marker::PhantomData; +use std::pin::Pin; +use std::rc::Rc; +use std::task::{Context, Poll}; -use super::{IntoServiceFactory, ServiceFactory}; +use futures_util::ready; + +use super::{IntoServiceFactory, Service, ServiceFactory}; /// Adapt external config argument to a config for provided service factory /// @@ -15,6 +22,28 @@ where MapConfig::new(factory.into_factory(), f) } +/// Adapt external config argument to a config for provided service factory +/// +/// This function uses service for converting config. +pub fn map_config_service( + factory: U1, + mapper: U2, +) -> MapConfigService +where + T: ServiceFactory, + M: ServiceFactory< + Config = (), + Request = C, + Response = T::Config, + Error = T::InitError, + InitError = T::InitError, + >, + U1: IntoServiceFactory, + U2: IntoServiceFactory, +{ + MapConfigService::new(factory.into_factory(), mapper.into_factory()) +} + /// Replace config with unit pub fn unit_config(factory: U) -> UnitConfig where @@ -125,6 +154,138 @@ where } } +/// `map_config_service()` adapter service factory +pub struct MapConfigService(Rc>); + +struct Inner { + a: A, + m: M, + mapper: RefCell>, + e: PhantomData, +} + +impl Clone for MapConfigService { + fn clone(&self) -> Self { + Self(self.0.clone()) + } +} + +impl MapConfigService { + /// Create new `MapConfigService` combinator + pub(crate) fn new(a: A, m: M) -> Self + where + A: ServiceFactory, + M: ServiceFactory< + Config = (), + Request = C, + Response = A::Config, + Error = A::InitError, + InitError = A::InitError, + >, + { + Self(Rc::new(Inner { + a, + m, + mapper: RefCell::new(None), + e: PhantomData, + })) + } +} + +impl ServiceFactory for MapConfigService +where + A: ServiceFactory, + M: ServiceFactory< + Config = (), + Request = C, + Response = A::Config, + Error = A::InitError, + InitError = A::InitError, + >, +{ + type Request = A::Request; + type Response = A::Response; + type Error = A::Error; + + type Config = C; + type Service = A::Service; + type InitError = A::InitError; + type Future = MapConfigServiceResponse; + + fn new_service(&self, cfg: C) -> Self::Future { + let inner = self.0.clone(); + if let Some(ref mapper) = *self.0.mapper.borrow() { + MapConfigServiceResponse { + inner, + config: None, + state: ResponseState::MapConfig(mapper.call(cfg)), + } + } else { + MapConfigServiceResponse { + inner, + config: Some(cfg), + state: ResponseState::CreateMapper(self.0.m.new_service(())), + } + } + } +} + +#[pin_project::pin_project] +pub struct MapConfigServiceResponse +where + A: ServiceFactory, + M: ServiceFactory, +{ + inner: Rc>, + config: Option, + #[pin] + state: ResponseState, +} + +#[pin_project::pin_project] +enum ResponseState { + CreateMapper(#[pin] M::Future), + MapConfig(#[pin] ::Future), + CreateService(#[pin] A::Future), +} + +impl Future for MapConfigServiceResponse +where + A: ServiceFactory, + M: ServiceFactory< + Config = (), + Request = C, + Response = A::Config, + Error = A::InitError, + InitError = A::InitError, + >, +{ + type Output = Result; + + #[pin_project::project] + fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { + let mut this = self.as_mut().project(); + + #[project] + match this.state.as_mut().project() { + ResponseState::CreateMapper(fut) => { + let mapper = ready!(fut.poll(cx))?; + let fut = mapper.call(this.config.take().unwrap()); + *this.inner.mapper.borrow_mut() = Some(mapper); + this.state.set(ResponseState::MapConfig(fut)); + self.poll(cx) + } + ResponseState::MapConfig(fut) => { + let config = ready!(fut.poll(cx))?; + let fut = this.inner.a.new_service(config); + this.state.set(ResponseState::CreateService(fut)); + self.poll(cx) + } + ResponseState::CreateService(fut) => fut.poll(cx), + } + } +} + #[cfg(test)] mod tests { use futures_util::future::ok; @@ -132,7 +293,7 @@ mod tests { use std::rc::Rc; use super::*; - use crate::{fn_service, ServiceFactory}; + use crate::{fn_factory_with_config, fn_service, ServiceFactory}; #[ntex_rt::test] async fn test_map_config() { @@ -156,4 +317,28 @@ mod tests { .new_service(10) .await; } + + #[ntex_rt::test] + async fn test_map_config_service() { + let item = Rc::new(Cell::new(10usize)); + let item2 = item.clone(); + + let srv = map_config_service( + fn_factory_with_config(move |next: usize| { + let item = item2.clone(); + async move { + item.set(next); + Ok::<_, ()>(fn_service(|id: usize| ok::<_, ()>(id * 2))) + } + }), + fn_service(move |item: usize| ok::<_, ()>(item + 1)), + ) + .clone() + .new_service(10) + .await + .unwrap(); + + assert_eq!(srv.call(10usize).await.unwrap(), 20); + assert_eq!(item.get(), 11); + } }