diff --git a/ntex/src/connect/openssl.rs b/ntex/src/connect/openssl.rs index a706c964..ef2b7752 100644 --- a/ntex/src/connect/openssl.rs +++ b/ntex/src/connect/openssl.rs @@ -5,7 +5,7 @@ pub use tls_openssl::ssl::{Error as SslError, HandshakeError, SslConnector, SslM use ntex_tls::openssl::SslConnector as IoSslConnector; -use crate::io::{Base, Io}; +use crate::io::{Base, Io, SealedService}; use crate::service::{Service, ServiceFactory}; use crate::util::{PoolId, Ready}; @@ -76,6 +76,11 @@ impl Connector { } } } + + /// Produce sealed io stream (IoBoxed) + pub fn seal(self) -> SealedService> { + SealedService::new(self) + } } impl Clone for Connector { diff --git a/ntex/src/connect/resolve.rs b/ntex/src/connect/resolve.rs index 2093ced1..8d3893a7 100644 --- a/ntex/src/connect/resolve.rs +++ b/ntex/src/connect/resolve.rs @@ -102,8 +102,7 @@ impl Clone for Resolver { } } -impl ServiceFactory for Resolver { - type Request = Connect; +impl ServiceFactory> for Resolver { type Response = Connect; type Error = ConnectError; type Config = (); @@ -116,8 +115,7 @@ impl ServiceFactory for Resolver { } } -impl Service for Resolver { - type Request = Connect; +impl Service> for Resolver { type Response = Connect; type Error = ConnectError; type Future = Pin, Self::Error>>>>; diff --git a/ntex/src/connect/rustls.rs b/ntex/src/connect/rustls.rs index 3ff764bf..f83d6f10 100644 --- a/ntex/src/connect/rustls.rs +++ b/ntex/src/connect/rustls.rs @@ -5,7 +5,7 @@ pub use tls_rustls::{ClientConfig, ServerName}; use ntex_tls::rustls::TlsConnector; -use crate::io::{Base, Io}; +use crate::io::{Base, Io, SealedService}; use crate::service::{Service, ServiceFactory}; use crate::util::{PoolId, Ready}; @@ -80,6 +80,11 @@ impl Connector { } } } + + /// Produce sealed io stream (IoBoxed) + pub fn seal(self) -> SealedService> { + SealedService::new(self) + } } impl Clone for Connector { diff --git a/ntex/src/connect/service.rs b/ntex/src/connect/service.rs index 9b47e393..178f5624 100644 --- a/ntex/src/connect/service.rs +++ b/ntex/src/connect/service.rs @@ -1,7 +1,7 @@ use std::task::{Context, Poll}; use std::{collections::VecDeque, future::Future, io, net::SocketAddr, pin::Pin}; -use crate::io::{types, Io}; +use crate::io::{types, Io, SealedService}; use crate::rt::tcp_connect_in; use crate::service::{Service, ServiceFactory}; use crate::util::{Either, PoolId, PoolRef, Ready}; @@ -46,6 +46,11 @@ impl Connector { pool: self.pool, } } + + /// Produce sealed io stream (IoBoxed) + pub fn seal(self) -> SealedService, Io> { + SealedService::new(self) + } } impl Default for Connector { @@ -63,8 +68,7 @@ impl Clone for Connector { } } -impl ServiceFactory for Connector { - type Request = Connect; +impl ServiceFactory> for Connector { type Response = Io; type Error = ConnectError; type Config = (); @@ -78,8 +82,7 @@ impl ServiceFactory for Connector { } } -impl Service for Connector { - type Request = Connect; +impl Service> for Connector { type Response = Io; type Error = ConnectError; type Future = ConnectServiceResponse; @@ -96,7 +99,7 @@ impl Service for Connector { } enum ConnectState { - Resolve( as Service>::Future), + Resolve( as Service>>::Future), Connect(TcpConnectorResponse), } @@ -107,7 +110,7 @@ pub struct ConnectServiceResponse { } impl ConnectServiceResponse { - pub(super) fn new(fut: as Service>::Future) -> Self { + pub(super) fn new(fut: as Service>>::Future) -> Self { Self { state: ConnectState::Resolve(fut), pool: PoolId::P0.pool_ref(), diff --git a/ntex/src/server/builder.rs b/ntex/src/server/builder.rs index 857054b2..116d3b83 100644 --- a/ntex/src/server/builder.rs +++ b/ntex/src/server/builder.rs @@ -198,7 +198,7 @@ impl ServerBuilder { where U: net::ToSocketAddrs, F: Fn(Config) -> R + Send + Clone + 'static, - R: ServiceFactory, + R: ServiceFactory, { let sockets = bind_addr(addr, self.backlog)?; @@ -226,7 +226,7 @@ impl ServerBuilder { N: AsRef, U: AsRef, F: Fn(Config) -> R + Send + Clone + 'static, - R: ServiceFactory, + R: ServiceFactory, { use std::os::unix::net::UnixListener; @@ -255,7 +255,7 @@ impl ServerBuilder { ) -> io::Result where F: Fn(Config) -> R + Send + Clone + 'static, - R: ServiceFactory, + R: ServiceFactory, { use std::net::{IpAddr, Ipv4Addr, SocketAddr}; let token = self.token.next(); @@ -280,7 +280,7 @@ impl ServerBuilder { ) -> io::Result where F: Fn(Config) -> R + Send + Clone + 'static, - R: ServiceFactory, + R: ServiceFactory, { let token = self.token.next(); self.services.push(Factory::create( diff --git a/ntex/src/server/config.rs b/ntex/src/server/config.rs index 2c5e88f1..926fd2b8 100644 --- a/ntex/src/server/config.rs +++ b/ntex/src/server/config.rs @@ -282,8 +282,8 @@ impl ServiceRuntime { /// *ServiceConfig::bind()* or *ServiceConfig::listen()* methods. pub fn service(&self, name: &str, service: F) where - F: service::IntoServiceFactory, - T: service::ServiceFactory + 'static, + F: service::IntoServiceFactory, + T: service::ServiceFactory + 'static, T::Future: 'static, T::Service: 'static, T::InitError: fmt::Debug, @@ -297,8 +297,8 @@ impl ServiceRuntime { /// *ServiceConfig::bind()* or *ServiceConfig::listen()* methods. pub fn service_in(&self, name: &str, pool: PoolId, service: F) where - F: service::IntoServiceFactory, - T: service::ServiceFactory + 'static, + F: service::IntoServiceFactory, + T: service::ServiceFactory + 'static, T::Future: 'static, T::Service: 'static, T::InitError: fmt::Debug, @@ -329,7 +329,7 @@ impl ServiceRuntime { type BoxedNewService = Box< dyn service::ServiceFactory< - Request = (Option, ServerMessage), + (Option, ServerMessage), Response = (), Error = (), InitError = (), @@ -344,15 +344,14 @@ struct ServiceFactory { pool: PoolId, } -impl service::ServiceFactory for ServiceFactory +impl service::ServiceFactory<(Option, ServerMessage)> for ServiceFactory where - T: service::ServiceFactory, + T: service::ServiceFactory, T::Future: 'static, T::Service: 'static, T::Error: 'static, T::InitError: fmt::Debug + 'static, { - type Request = (Option, ServerMessage); type Response = (); type Error = (); type InitError = (); diff --git a/ntex/src/server/service.rs b/ntex/src/server/service.rs index bd4cc3a4..71cea3c8 100644 --- a/ntex/src/server/service.rs +++ b/ntex/src/server/service.rs @@ -21,7 +21,7 @@ pub(super) enum ServerMessage { } pub(super) trait StreamServiceFactory: Send + Clone + 'static { - type Factory: ServiceFactory; + type Factory: ServiceFactory; fn create(&self, _: Config) -> Self::Factory; } @@ -38,7 +38,7 @@ pub(super) trait InternalServiceFactory: Send { pub(super) type BoxedServerService = Box< dyn Service< - Request = (Option, ServerMessage), + (Option, ServerMessage), Response = (), Error = (), Future = Ready<(), ()>, @@ -59,13 +59,12 @@ impl StreamService { } } -impl Service for StreamService +impl Service<(Option, ServerMessage)> for StreamService where - T: Service, + T: Service, T::Future: 'static, T::Error: 'static, { - type Request = (Option, ServerMessage); type Response = (); type Error = (); type Future = Ready<(), ()>; @@ -195,7 +194,7 @@ impl InternalServiceFactory for Box { impl StreamServiceFactory for F where F: Fn(Config) -> T + Send + Clone + 'static, - T: ServiceFactory, + T: ServiceFactory, { type Factory = T; diff --git a/ntex/src/server/test.rs b/ntex/src/server/test.rs index 2f36e2c0..b9ff5211 100644 --- a/ntex/src/server/test.rs +++ b/ntex/src/server/test.rs @@ -41,7 +41,7 @@ use crate::{io::Io, service::ServiceFactory}; pub fn test_server(factory: F) -> TestServer where F: Fn() -> R + Send + Clone + 'static, - R: ServiceFactory, + R: ServiceFactory, { let (tx, rx) = mpsc::channel(); diff --git a/ntex/src/util/buffer.rs b/ntex/src/util/buffer.rs index e2fc3a28..b437acc0 100644 --- a/ntex/src/util/buffer.rs +++ b/ntex/src/util/buffer.rs @@ -1,7 +1,9 @@ //! Service that buffers incomming requests. use std::cell::{Cell, RefCell}; use std::task::{Context, Poll}; -use std::{collections::VecDeque, future::Future, pin::Pin, rc::Rc}; +use std::{ + collections::VecDeque, future::Future, marker::PhantomData, pin::Pin, rc::Rc, +}; use crate::channel::oneshot; use crate::service::{IntoService, Service, Transform}; @@ -11,12 +13,13 @@ use crate::util::Either; /// Buffer - service factory for service that can buffer incoming request. /// /// Default number of buffered requests is 16 -pub struct Buffer { +pub struct Buffer { buf_size: usize, err: Rc E>, + _t: PhantomData, } -impl Buffer { +impl Buffer { pub fn new(f: F) -> Self where F: Fn() -> E + 'static, @@ -24,6 +27,7 @@ impl Buffer { Self { buf_size: 16, err: Rc::new(f), + _t: PhantomData, } } @@ -33,20 +37,21 @@ impl Buffer { } } -impl Clone for Buffer { +impl Clone for Buffer { fn clone(&self) -> Self { Self { buf_size: self.buf_size, err: self.err.clone(), + _t: PhantomData, } } } -impl Transform for Buffer +impl Transform for Buffer where - S: Service, + S: Service, { - type Service = BufferService; + type Service = BufferService; fn new_transform(&self, service: S) -> Self::Service { BufferService { @@ -65,26 +70,26 @@ where /// Buffer service - service that can buffer incoming requests. /// /// Default number of buffered requests is 16 -pub struct BufferService, E> { +pub struct BufferService, E> { size: usize, - inner: Rc>, + inner: Rc>, } -struct Inner, E> { +struct Inner, E> { ready: Cell, service: S, waker: LocalWaker, err: Rc E>, - buf: RefCell, S::Request)>>, + buf: RefCell, R)>>, } -impl BufferService +impl BufferService where - S: Service, + S: Service, { pub fn new(size: usize, err: F, service: U) -> Self where - U: IntoService, + U: IntoService, F: Fn() -> E + 'static, { Self { @@ -100,9 +105,9 @@ where } } -impl Clone for BufferService +impl Clone for BufferService where - S: Service + Clone, + S: Service + Clone, { fn clone(&self) -> Self { Self { @@ -118,14 +123,13 @@ where } } -impl Service for BufferService +impl Service for BufferService where - S: Service, + S: Service, { - type Request = S::Request; type Response = S::Response; type Error = S::Error; - type Future = Either>; + type Future = Either>; #[inline] fn poll_ready(&self, cx: &mut Context<'_>) -> Poll> { @@ -158,7 +162,7 @@ where } #[inline] - fn call(&self, req: S::Request) -> Self::Future { + fn call(&self, req: R) -> Self::Future { if self.inner.ready.get() { self.inner.ready.set(false); Either::Left(self.inner.service.call(req)) @@ -178,21 +182,21 @@ where pin_project_lite::pin_project! { #[doc(hidden)] - pub struct BufferServiceResponse, E> { + pub struct BufferServiceResponse, E> { #[pin] - state: State, + state: State, } } pin_project_lite::pin_project! { #[project = StateProject] - enum State, E> { - Tx { rx: oneshot::Receiver, inner: Rc> }, - Srv { #[pin] fut: S::Future, inner: Rc> }, + enum State, E> { + Tx { rx: oneshot::Receiver, inner: Rc> }, + Srv { #[pin] fut: S::Future, inner: Rc> }, } } -impl, E> Future for BufferServiceResponse { +impl, E> Future for BufferServiceResponse { type Output = Result; fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { diff --git a/ntex/src/util/inflight.rs b/ntex/src/util/inflight.rs index 2b0c10ee..b2c3bfc5 100644 --- a/ntex/src/util/inflight.rs +++ b/ntex/src/util/inflight.rs @@ -1,5 +1,5 @@ //! Service that limits number of in-flight async requests. -use std::{future::Future, pin::Pin, task::Context, task::Poll}; +use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll}; use super::counter::{Counter, CounterGuard}; use crate::{IntoService, Service, Transform}; @@ -24,14 +24,14 @@ impl Default for InFlight { } } -impl Transform for InFlight -where - S: Service, -{ +impl Transform for InFlight { type Service = InFlightService; fn new_transform(&self, service: S) -> Self::Service { - InFlightService::new(self.max_inflight, service) + InFlightService { + service, + count: Counter::new(self.max_inflight), + } } } @@ -40,13 +40,11 @@ pub struct InFlightService { service: S, } -impl InFlightService -where - S: Service, -{ - pub fn new(max: usize, service: U) -> Self +impl InFlightService { + pub fn new(max: usize, service: U) -> Self where - U: IntoService, + S: Service, + U: IntoService, { Self { count: Counter::new(max), @@ -55,14 +53,13 @@ where } } -impl Service for InFlightService +impl Service for InFlightService where - T: Service, + T: Service, { - type Request = T::Request; type Response = T::Response; type Error = T::Error; - type Future = InFlightServiceResponse; + type Future = InFlightServiceResponse; #[inline] fn poll_ready(&self, cx: &mut Context<'_>) -> Poll> { @@ -82,24 +79,26 @@ where } #[inline] - fn call(&self, req: T::Request) -> Self::Future { + fn call(&self, req: R) -> Self::Future { InFlightServiceResponse { fut: self.service.call(req), _guard: self.count.get(), + _t: PhantomData, } } } pin_project_lite::pin_project! { #[doc(hidden)] - pub struct InFlightServiceResponse { + pub struct InFlightServiceResponse, R> { #[pin] fut: T::Future, _guard: CounterGuard, + _t: PhantomData } } -impl Future for InFlightServiceResponse { +impl, R> Future for InFlightServiceResponse { type Output = Result; fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { diff --git a/ntex/src/util/keepalive.rs b/ntex/src/util/keepalive.rs index d6b20da5..3bb21811 100644 --- a/ntex/src/util/keepalive.rs +++ b/ntex/src/util/keepalive.rs @@ -7,13 +7,13 @@ use crate::{util::Ready, Service, ServiceFactory}; /// KeepAlive service factory /// /// Controls min time between requests. -pub struct KeepAlive { +pub struct KeepAlive { f: F, ka: Millis, - _t: marker::PhantomData<(R, E)>, + _t: marker::PhantomData, } -impl KeepAlive +impl KeepAlive where F: Fn() -> E + Clone, { @@ -30,7 +30,7 @@ where } } -impl Clone for KeepAlive +impl Clone for KeepAlive where F: Clone, { @@ -43,16 +43,15 @@ where } } -impl ServiceFactory for KeepAlive +impl ServiceFactory for KeepAlive where F: Fn() -> E + Clone, { - type Request = R; type Response = R; type Error = E; type InitError = Infallible; type Config = (); - type Service = KeepAliveService; + type Service = KeepAliveService; type Future = Ready; fn new_service(&self, _: ()) -> Self::Future { @@ -60,15 +59,15 @@ where } } -pub struct KeepAliveService { +pub struct KeepAliveService { f: F, dur: Millis, sleep: Sleep, expire: Cell, - _t: marker::PhantomData<(R, E)>, + _t: marker::PhantomData, } -impl KeepAliveService +impl KeepAliveService where F: Fn() -> E, { @@ -85,11 +84,10 @@ where } } -impl Service for KeepAliveService +impl Service for KeepAliveService where F: Fn() -> E, { - type Request = R; type Response = R; type Error = E; type Future = Ready; diff --git a/ntex/src/util/sink.rs b/ntex/src/util/sink.rs index 06cc7454..a61452ee 100644 --- a/ntex/src/util/sink.rs +++ b/ntex/src/util/sink.rs @@ -40,11 +40,10 @@ where } } -impl Service for SinkService +impl Service for SinkService where S: Sink + Unpin, { - type Request = I; type Response = (); type Error = S::Error; type Future = Ready<(), S::Error>; diff --git a/ntex/src/util/stream.rs b/ntex/src/util/stream.rs index 89d5584d..bd43c686 100644 --- a/ntex/src/util/stream.rs +++ b/ntex/src/util/stream.rs @@ -5,12 +5,12 @@ use crate::service::{IntoService, Service}; use crate::{util::poll_fn, Sink, Stream}; pin_project_lite::pin_project! { - pub struct Dispatcher + pub struct Dispatcher where R: 'static, - S: Service>, + S: Service>, S: 'static, - T: Stream>, + T: Stream>, T: Unpin, U: Sink>, U: Unpin, @@ -24,18 +24,18 @@ pin_project_lite::pin_project! { } } -impl Dispatcher +impl Dispatcher where R: 'static, - S: Service> + 'static, + S: Service> + 'static, S::Error: fmt::Debug, - T: Stream> + Unpin, + T: Stream> + Unpin, U: Sink> + Unpin + 'static, U::Error: fmt::Debug, { pub fn new(stream: T, sink: U, service: F) -> Self where - F: IntoService, + F: IntoService, { Dispatcher { stream, @@ -47,12 +47,13 @@ where } } -impl Future for Dispatcher +impl Future for Dispatcher where R: 'static, - S: Service> + 'static, - S::Error: fmt::Debug, - T: Stream> + Unpin, + S: Service> + 'static, + S::Future: 'static, + S::Error: fmt::Debug + 'static, + T: Stream> + Unpin, U: Sink> + Unpin + 'static, U::Error: fmt::Debug, { diff --git a/ntex/src/util/timeout.rs b/ntex/src/util/timeout.rs index 51696c0d..00eb150b 100644 --- a/ntex/src/util/timeout.rs +++ b/ntex/src/util/timeout.rs @@ -2,7 +2,10 @@ //! //! If the response does not complete within the specified timeout, the response //! will be aborted. -use std::{fmt, future::Future, marker, pin::Pin, task::Context, task::Poll}; +use std::{ + fmt, future::Future, marker, marker::PhantomData, pin::Pin, task::Context, + task::Poll, +}; use crate::service::{IntoService, Service, Transform}; use crate::time::{sleep, Millis, Sleep}; @@ -82,10 +85,7 @@ impl Clone for Timeout { } } -impl Transform for Timeout -where - S: Service, -{ +impl Transform for Timeout { type Service = TimeoutService; fn new_transform(&self, service: S) -> Self::Service { @@ -103,14 +103,12 @@ pub struct TimeoutService { timeout: Millis, } -impl TimeoutService -where - S: Service, -{ - pub fn new(timeout: T, service: U) -> Self +impl TimeoutService { + pub fn new(timeout: T, service: U) -> Self where T: Into, - U: IntoService, + S: Service, + U: IntoService, { TimeoutService { timeout: timeout.into(), @@ -119,14 +117,13 @@ where } } -impl Service for TimeoutService +impl Service for TimeoutService where - S: Service, + S: Service, { - type Request = S::Request; type Response = S::Response; type Error = TimeoutError; - type Future = Either, TimeoutServiceResponse2>; + type Future = Either, TimeoutServiceResponse2>; #[inline] fn poll_ready(&self, cx: &mut Context<'_>) -> Poll> { @@ -138,15 +135,17 @@ where self.service.poll_shutdown(cx, is_error) } - fn call(&self, request: S::Request) -> Self::Future { + fn call(&self, request: R) -> Self::Future { if self.timeout.is_zero() { Either::Right(TimeoutServiceResponse2 { fut: self.service.call(request), + _t: PhantomData, }) } else { Either::Left(TimeoutServiceResponse { fut: self.service.call(request), sleep: sleep(self.timeout), + _t: PhantomData, }) } } @@ -156,16 +155,17 @@ pin_project_lite::pin_project! { /// `TimeoutService` response future #[doc(hidden)] #[derive(Debug)] - pub struct TimeoutServiceResponse { + pub struct TimeoutServiceResponse, R> { #[pin] fut: T::Future, sleep: Sleep, + _t: PhantomData } } -impl Future for TimeoutServiceResponse +impl Future for TimeoutServiceResponse where - T: Service, + T: Service, { type Output = Result>; @@ -191,15 +191,16 @@ pin_project_lite::pin_project! { /// `TimeoutService` response future #[doc(hidden)] #[derive(Debug)] - pub struct TimeoutServiceResponse2 { + pub struct TimeoutServiceResponse2, R> { #[pin] fut: T::Future, + _t: PhantomData, } } -impl Future for TimeoutServiceResponse2 +impl Future for TimeoutServiceResponse2 where - T: Service, + T: Service, { type Output = Result>; diff --git a/ntex/src/util/variant.rs b/ntex/src/util/variant.rs index deb30ec9..d0f76e4d 100644 --- a/ntex/src/util/variant.rs +++ b/ntex/src/util/variant.rs @@ -1,121 +1,112 @@ //! Contains `Variant` service and related types and functions. -use std::{future::Future, pin::Pin, task::Context, task::Poll}; +use std::{future::Future, marker::PhantomData, pin::Pin, task::Context, task::Poll}; use crate::service::{IntoServiceFactory, Service, ServiceFactory}; /// Construct `Variant` service factory. /// /// Variant service allow to combine multiple different services into a single service. -pub fn variant(factory: A) -> Variant { - Variant { factory } +pub fn variant, V1R>(factory: V1) -> Variant { + Variant { + factory, + _t: PhantomData, + } } /// Combine multiple different service types into a single service. -pub struct Variant { +pub struct Variant { factory: A, + _t: PhantomData, } -impl Variant +impl Variant where - A: ServiceFactory, + A: ServiceFactory, A::Config: Clone, { /// Convert to a Variant with two request types - pub fn and(self, factory: F) -> VariantFactory2 + pub fn v2(self, factory: F) -> VariantFactory2 where B: ServiceFactory< + BR, Config = A::Config, Response = A::Response, Error = A::Error, InitError = A::InitError, >, - F: IntoServiceFactory, + F: IntoServiceFactory, { VariantFactory2 { - A: self.factory, - V2: factory.into_factory(), - } - } - - /// Convert to a Variant with two request types - pub fn v2(self, factory: F) -> VariantFactory2 - where - B: ServiceFactory< - Config = A::Config, - Response = A::Response, - Error = A::Error, - InitError = A::InitError, - >, - F: IntoServiceFactory, - { - VariantFactory2 { - A: self.factory, + V1: self.factory, V2: factory.into_factory(), + _t: PhantomData, } } } -macro_rules! variant_impl_and ({$fac1_type:ident, $fac2_type:ident, $name:ident, $m_name:ident, ($($T:ident),+)} => { +macro_rules! variant_impl_and ({$fac1_type:ident, $fac2_type:ident, $name:ident, $r_name:ident, $m_name:ident, ($($T:ident),+), ($($R:ident),+)} => { - impl $fac1_type + #[allow(non_snake_case)] + impl $fac1_type where - V1: ServiceFactory, + V1: ServiceFactory, V1::Config: Clone, { /// Convert to a Variant with more request types - pub fn $m_name<$name, F>(self, factory: F) -> $fac2_type - where $name: ServiceFactory< + pub fn $m_name<$name, $r_name, F>(self, factory: F) -> $fac2_type + where $name: ServiceFactory<$r_name, Config = V1::Config, Response = V1::Response, Error = V1::Error, InitError = V1::InitError>, - F: IntoServiceFactory<$name>, + F: IntoServiceFactory<$name, $r_name>, { $fac2_type { - A: self.A, + V1: self.V1, $($T: self.$T,)+ $name: factory.into_factory(), + _t: PhantomData } } } }); -macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, $fac_type:ident, $(($n:tt, $T:ident)),+} => { +macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, $fac_type:ident, $(($n:tt, $T:ident, $R:ident)),+} => { #[allow(non_snake_case)] - pub enum $enum_type { - V1(V1), - $($T($T),)+ + pub enum $enum_type { + V1(V1R), + $($T($R),)+ } #[allow(non_snake_case)] - pub struct $srv_type { - a: V1, + pub struct $srv_type { + V1: V1, $($T: $T,)+ + _t: PhantomData<(V1R, $($R),+)>, } - impl Clone for $srv_type { + impl Clone for $srv_type { fn clone(&self) -> Self { Self { - a: self.a.clone(), + _t: PhantomData, + V1: self.V1.clone(), $($T: self.$T.clone(),)+ } } } - impl Service for $srv_type + impl Service<$enum_type> for $srv_type where - V1: Service, - $($T: Service),+ + V1: Service, + $($T: Service<$R, Response = V1::Response, Error = V1::Error>),+ { - type Request = $enum_type; type Response = V1::Response; type Error = V1::Error; type Future = $mod_name::ServiceResponse; - #[inline] fn poll_ready(&self, cx: &mut Context<'_>) -> Poll> { - let mut ready = self.a.poll_ready(cx)?.is_ready(); + let mut ready = self.V1.poll_ready(cx)?.is_ready(); $(ready = self.$T.poll_ready(cx)?.is_ready() && ready;)+ if ready { @@ -125,62 +116,61 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, } } - #[inline] fn poll_shutdown(&self, cx: &mut Context<'_>, is_error: bool) -> Poll<()> { - let mut ready = self.a.poll_shutdown(cx, is_error).is_ready(); - $(ready = self.$T.poll_shutdown(cx, is_error).is_ready() && ready;)+ + let mut ready = self.V1.poll_shutdown(cx, is_error).is_ready(); + $(ready = self.$T.poll_shutdown(cx, is_error).is_ready() && ready;)+ - if ready { - Poll::Ready(()) - } else { - Poll::Pending + if ready { + Poll::Ready(()) + } else { + Poll::Pending + } } - } - #[inline] - fn call(&self, req: Self::Request) -> Self::Future { + fn call(&self, req: $enum_type) -> Self::Future { match req { - $enum_type::V1(req) => $mod_name::ServiceResponse::V1 { fut: self.a.call(req) }, + $enum_type::V1(req) => $mod_name::ServiceResponse::V1 { fut: self.V1.call(req) }, $($enum_type::$T(req) => $mod_name::ServiceResponse::$T { fut: self.$T.call(req) },)+ } } } #[allow(non_snake_case)] - pub struct $fac_type { - A: V1, + pub struct $fac_type { + V1: V1, $($T: $T,)+ + _t: PhantomData<(V1R, $($R,)+)>, } - impl Clone for $fac_type { + impl Clone for $fac_type { fn clone(&self) -> Self { Self { - A: self.A.clone(), + _t: PhantomData, + V1: self.V1.clone(), $($T: self.$T.clone(),)+ } } } - impl ServiceFactory for $fac_type + impl ServiceFactory<$enum_type> for $fac_type where - V1: ServiceFactory, + V1: ServiceFactory, V1::Config: Clone, - $($T: ServiceFactory),+ + $($T: ServiceFactory<$R, Config = V1::Config, Response = V1::Response, Error = V1::Error, InitError = V1::InitError>),+ { - type Request = $enum_type; type Response = V1::Response; type Error = V1::Error; type Config = V1::Config; type InitError = V1::InitError; - type Service = $srv_type; - type Future = $mod_name::ServiceFactoryResponse; + type Service = $srv_type; + type Future = $mod_name::ServiceFactoryResponse; fn new_service(&self, cfg: Self::Config) -> Self::Future { $mod_name::ServiceFactoryResponse { - a: None, + V1: None, items: Default::default(), $($T: self.$T.new_service(cfg.clone()),)+ - a_fut: self.A.new_service(cfg), + V1_fut: self.V1.new_service(cfg), } } } @@ -192,18 +182,18 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, pin_project_lite::pin_project! { #[project = ServiceResponseProject] - pub enum ServiceResponse { - V1{ #[pin] fut: A }, + pub enum ServiceResponse { + V1{ #[pin] fut: V1 }, $($T{ #[pin] fut: $T },)+ } } - impl Future for ServiceResponse + impl Future for ServiceResponse where - A: Future, - $($T: Future),+ + V1: Future, + $($T: Future),+ { - type Output = A::Output; + type Output = V1::Output; fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { match self.project() { @@ -216,29 +206,29 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, pin_project_lite::pin_project! { #[doc(hidden)] - pub struct ServiceFactoryResponse { - pub(super) a: Option, + pub struct ServiceFactoryResponse, $($T: ServiceFactory<$R>,)+ V1R, $($R,)+> { + pub(super) V1: Option, pub(super) items: ($(Option<$T::Service>,)+), - #[pin] pub(super) a_fut: A::Future, + #[pin] pub(super) V1_fut: V1::Future, $(#[pin] pub(super) $T: $T::Future),+ } } - impl Future for ServiceFactoryResponse + impl Future for ServiceFactoryResponse where - A: ServiceFactory, - $($T: ServiceFactory),+ + V1: ServiceFactory, + $($T: ServiceFactory<$R, Response = V1::Response, Error = V1::Error, InitError = V1::InitError,>),+ { - type Output = Result<$srv_type, A::InitError>; + type Output = Result<$srv_type, V1::InitError>; fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { let this = self.project(); let mut ready = true; - if this.a.is_none() { - match this.a_fut.poll(cx) { + if this.V1.is_none() { + match this.V1_fut.poll(cx) { Poll::Ready(Ok(item)) => { - *this.a = Some(item); + *this.V1 = Some(item); } Poll::Pending => ready = false, Poll::Ready(Err(e)) => return Poll::Ready(Err(e.into())), @@ -259,8 +249,9 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, if ready { Poll::Ready(Ok($srv_type { - a: this.a.take().unwrap(), + V1: this.V1.take().unwrap(), $($T: this.items.$n.take().unwrap(),)+ + _t: PhantomData })) } else { Poll::Pending @@ -272,33 +263,32 @@ macro_rules! variant_impl ({$mod_name:ident, $enum_type:ident, $srv_type:ident, }); #[rustfmt::skip] -variant_impl!(v2, Variant2, VariantService2, VariantFactory2, (0, V2)); +variant_impl!(v2, Variant2, VariantService2, VariantFactory2, (0, V2, V2R)); #[rustfmt::skip] -variant_impl!(v3, Variant3, VariantService3, VariantFactory3, (0, V2), (1, V3)); +variant_impl!(v3, Variant3, VariantService3, VariantFactory3, (0, V2, V2R), (1, V3, V3R)); #[rustfmt::skip] -variant_impl!(v4, Variant4, VariantService4, VariantFactory4, (0, V2), (1, V3), (2, V4)); +variant_impl!(v4, Variant4, VariantService4, VariantFactory4, (0, V2, V2R), (1, V3, V3R), (2, V4, V4R)); #[rustfmt::skip] -variant_impl!(v5, Variant5, VariantService5, VariantFactory5, (0, V2), (1, V3), (2, V4), (3, V5)); +variant_impl!(v5, Variant5, VariantService5, VariantFactory5, (0, V2, V2R), (1, V3, V3R), (2, V4, V4R), (3, V5, V5R)); #[rustfmt::skip] -variant_impl!(v6, Variant6, VariantService6, VariantFactory6, (0, V2), (1, V3), (2, V4), (3, V5), (4, V6)); +variant_impl!(v6, Variant6, VariantService6, VariantFactory6, (0, V2, V2R), (1, V3, V3R), (2, V4, V4R), (3, V5, V5R), (4, V6, V6R)); #[rustfmt::skip] -variant_impl!(v7, Variant7, VariantService7, VariantFactory7, (0, V2), (1, V3), (2, V4), (3, V5), (4, V6), (5, V7)); +variant_impl!(v7, Variant7, VariantService7, VariantFactory7, (0, V2, V2R), (1, V3, V3R), (2, V4, V4R), (3, V5, V5R), (4, V6, V6R), (5, V7, V7R)); #[rustfmt::skip] -variant_impl!(v8, Variant8, VariantService8, VariantFactory8, (0, V2), (1, V3), (2, V4), (3, V5), (4, V6), (5, V7), (6, V8)); +variant_impl!(v8, Variant8, VariantService8, VariantFactory8, (0, V2, V2R), (1, V3, V3R), (2, V4, V4R), (3, V5, V5R), (4, V6, V6R), (5, V7, V7R), (6, V8, V8R)); -variant_impl_and!(VariantFactory2, VariantFactory3, V3, v3, (V2)); -variant_impl_and!(VariantFactory3, VariantFactory4, V4, v4, (V2, V3)); -variant_impl_and!(VariantFactory4, VariantFactory5, V5, v5, (V2, V3, V4)); -variant_impl_and!(VariantFactory5, VariantFactory6, V6, v6, (V2, V3, V4, V5)); -variant_impl_and!( - VariantFactory6, - VariantFactory7, - V7, - v7, - (V2, V3, V4, V5, V6) -); #[rustfmt::skip] -variant_impl_and!(VariantFactory7, VariantFactory8, V8, v8, (V2, V3, V4, V5, V6, V7)); +variant_impl_and!(VariantFactory2, VariantFactory3, V3, V3R, v3, (V2), (V2R)); +#[rustfmt::skip] +variant_impl_and!(VariantFactory3, VariantFactory4, V4, V4R, v4, (V2, V3), (V2R, V3R)); +#[rustfmt::skip] +variant_impl_and!(VariantFactory4, VariantFactory5, V5, V5R, v5, (V2, V3, V4), (V2R, V3R, V4R)); +#[rustfmt::skip] +variant_impl_and!(VariantFactory5, VariantFactory6, V6, V6R, v6, (V2, V3, V4, V5), (V2R, V3R, V4R, V5R)); +#[rustfmt::skip] +variant_impl_and!(VariantFactory6, VariantFactory7, V7, V7R, v7, (V2, V3, V4, V5, V6), (V2R, V3R, V4R, V5R, V6R)); +#[rustfmt::skip] +variant_impl_and!(VariantFactory7, VariantFactory8, V8, V8R, v8, (V2, V3, V4, V5, V6, V7), (V2R, V3R, V4R, V5R, V6R, V7R)); #[cfg(test)] mod tests {