mirror of
https://github.com/fafhrd91/actix-net
synced 2024-12-03 00:22:23 +01:00
168 lines
4.3 KiB
Rust
168 lines
4.3 KiB
Rust
use std::rc::Rc;
|
|
use std::sync::Arc;
|
|
|
|
use futures::{Async, Future, IntoFuture, Poll};
|
|
|
|
use crate::transform_map_init_err::TransformMapInitErr;
|
|
use crate::{NewService, Service};
|
|
|
|
/// `Transform` service factory.
|
|
///
|
|
/// Transform factory creates service that wraps other services.
|
|
///
|
|
/// * `S` is a wrapped service.
|
|
/// * `R` requests handled by this transform service.
|
|
pub trait Transform<R, S> {
|
|
/// Responses given by the service.
|
|
type Response;
|
|
|
|
/// Errors produced by the service.
|
|
type Error;
|
|
|
|
/// The `TransformService` value created by this factory
|
|
type Transform: Service<R, Response = Self::Response, Error = Self::Error>;
|
|
|
|
/// Errors produced while building a service.
|
|
type InitError;
|
|
|
|
/// The future response value.
|
|
type Future: Future<Item = Self::Transform, Error = Self::InitError>;
|
|
|
|
/// Create and return a new service value asynchronously.
|
|
fn new_transform(&self, service: S) -> Self::Future;
|
|
|
|
/// Map this service's factory init error to a different error,
|
|
/// returning a new transform service factory.
|
|
fn map_init_err<F, E>(self, f: F) -> TransformMapInitErr<Self, S, F, E>
|
|
where
|
|
Self: Sized,
|
|
F: Fn(Self::InitError) -> E,
|
|
{
|
|
TransformMapInitErr::new(self, f)
|
|
}
|
|
}
|
|
|
|
impl<T, R, S> Transform<R, S> for Rc<T>
|
|
where
|
|
T: Transform<R, S>,
|
|
{
|
|
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 {
|
|
self.as_ref().new_transform(service)
|
|
}
|
|
}
|
|
|
|
impl<T, R, S> Transform<R, S> for Arc<T>
|
|
where
|
|
T: Transform<R, S>,
|
|
{
|
|
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 {
|
|
self.as_ref().new_transform(service)
|
|
}
|
|
}
|
|
|
|
/// Trait for types that can be converted to a *transform service*
|
|
pub trait IntoTransform<T, R, S>
|
|
where
|
|
T: Transform<R, S>,
|
|
{
|
|
/// Convert to a `TransformService`
|
|
fn into_transform(self) -> T;
|
|
}
|
|
|
|
impl<T, S, R> IntoTransform<T, S, R> for T
|
|
where
|
|
T: Transform<S, R>,
|
|
{
|
|
fn into_transform(self) -> T {
|
|
self
|
|
}
|
|
}
|
|
|
|
/// `Apply` transform new service
|
|
#[derive(Clone)]
|
|
pub struct ApplyTransform<T, R, S, Req, Cfg> {
|
|
a: S,
|
|
t: Rc<T>,
|
|
_t: std::marker::PhantomData<(R, Req, Cfg)>,
|
|
}
|
|
|
|
impl<T, R, S, Req, Cfg> ApplyTransform<T, R, S, Req, Cfg>
|
|
where
|
|
S: NewService<Req, Cfg>,
|
|
T: Transform<R, S::Service, Error = S::Error, InitError = S::InitError>,
|
|
{
|
|
/// Create new `ApplyNewService` new service instance
|
|
pub fn new<F: IntoTransform<T, R, S::Service>>(t: F, a: S) -> Self {
|
|
Self {
|
|
a,
|
|
t: Rc::new(t.into_transform()),
|
|
_t: std::marker::PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T, R, S, Req, Cfg> NewService<R, Cfg> for ApplyTransform<T, R, S, Req, Cfg>
|
|
where
|
|
S: NewService<Req, Cfg>,
|
|
T: Transform<R, S::Service, Error = S::Error, InitError = S::InitError>,
|
|
{
|
|
type Response = T::Response;
|
|
type Error = T::Error;
|
|
|
|
type Service = T::Transform;
|
|
type InitError = T::InitError;
|
|
type Future = ApplyTransformFuture<T, R, S, Req, Cfg>;
|
|
|
|
fn new_service(&self, cfg: &Cfg) -> Self::Future {
|
|
ApplyTransformFuture {
|
|
t_cell: self.t.clone(),
|
|
fut_a: self.a.new_service(cfg).into_future(),
|
|
fut_t: None,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct ApplyTransformFuture<T, R, S, Req, Cfg>
|
|
where
|
|
S: NewService<Req, Cfg>,
|
|
T: Transform<R, S::Service, Error = S::Error, InitError = S::InitError>,
|
|
{
|
|
fut_a: S::Future,
|
|
fut_t: Option<T::Future>,
|
|
t_cell: Rc<T>,
|
|
}
|
|
|
|
impl<T, R, S, Req, Cfg> Future for ApplyTransformFuture<T, R, S, Req, Cfg>
|
|
where
|
|
S: NewService<Req, Cfg>,
|
|
T: Transform<R, S::Service, Error = S::Error, InitError = S::InitError>,
|
|
{
|
|
type Item = T::Transform;
|
|
type Error = T::InitError;
|
|
|
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
|
if self.fut_t.is_none() {
|
|
if let Async::Ready(service) = self.fut_a.poll()? {
|
|
self.fut_t = Some(self.t_cell.new_transform(service).into_future());
|
|
}
|
|
}
|
|
if let Some(ref mut fut) = self.fut_t {
|
|
fut.poll()
|
|
} else {
|
|
Ok(Async::NotReady)
|
|
}
|
|
}
|
|
}
|