1
0
mirror of https://github.com/fafhrd91/actix-net synced 2024-12-03 00:22:23 +01:00
actix-net/actix-service/src/transform.rs

241 lines
6.5 KiB
Rust
Raw Normal View History

use std::rc::Rc;
use std::sync::Arc;
2019-03-05 06:24:47 +01:00
use futures::{Async, Future, IntoFuture, Poll};
2019-03-12 20:53:08 +01:00
use crate::transform_err::{TransformFromErr, TransformMapInitErr};
use crate::{IntoNewService, NewService, Service};
/// The `Transform` trait defines the interface of a Service factory. `Transform`
/// is often implemented for middleware, defining how to manufacture a
2019-04-05 19:36:57 +02:00
/// middleware Service. A Service that is manufactured by the factory takes
/// the Service that follows it during execution as a parameter, assuming
/// ownership of the next Service. A Service can be a variety of types, such
/// as (but not limited to) another middleware Service, an extractor Service,
/// other helper Services, or the request handler endpoint Service.
///
/// A Service is created by the factory during server initialization.
///
2019-03-09 15:36:23 +01:00
/// `Config` is a service factory configuration type.
pub trait Transform<S> {
/// 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
2019-03-09 15:36:23 +01:00
type Transform: Service<
Request = Self::Request,
Response = Self::Response,
Error = Self::Error,
>;
/// Errors produced while building a service.
type InitError;
/// The future response value.
2019-03-05 06:35:47 +01:00
type Future: Future<Item = Self::Transform, Error = Self::InitError>;
2019-04-04 20:02:53 +02:00
/// Creates and returns a new Service component, asynchronously
2019-03-05 04:38:11 +01:00
fn new_transform(&self, service: S) -> Self::Future;
2019-03-12 20:53:08 +01:00
/// Map this service's factory error to a different error,
/// returning a new transform service factory.
2019-03-05 04:38:11 +01:00
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)
}
2019-03-12 20:53:08 +01:00
/// Map this service's init error to any error implementing `From` for
/// this service`s `Error`.
///
/// Note that this function consumes the receiving transform and returns a
/// wrapped version of it.
fn from_err<E>(self) -> TransformFromErr<Self, S, E>
where
Self: Sized,
E: From<Self::InitError>,
{
TransformFromErr::new(self)
}
// /// Map this service's init error to service's init error
// /// if it is implementing `Into` to this service`s `InitError`.
// ///
// /// Note that this function consumes the receiving transform and returns a
// /// wrapped version of it.
// fn into_err<E>(self) -> TransformIntoErr<Self, S>
// where
// Self: Sized,
// Self::InitError: From<Self::InitError>,
// {
// TransformFromErr::new(self)
// }
}
2019-03-09 15:36:23 +01:00
impl<T, S> Transform<S> for Rc<T>
where
2019-03-09 15:36:23 +01:00
T: Transform<S>,
{
2019-03-09 15:36:23 +01:00
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type InitError = T::InitError;
2019-03-05 04:38:11 +01:00
type Transform = T::Transform;
type Future = T::Future;
2019-03-05 04:38:11 +01:00
fn new_transform(&self, service: S) -> T::Future {
self.as_ref().new_transform(service)
}
}
2019-03-09 15:36:23 +01:00
impl<T, S> Transform<S> for Arc<T>
where
2019-03-09 15:36:23 +01:00
T: Transform<S>,
{
2019-03-09 15:36:23 +01:00
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type InitError = T::InitError;
2019-03-05 04:38:11 +01:00
type Transform = T::Transform;
type Future = T::Future;
2019-03-05 04:38:11 +01:00
fn new_transform(&self, service: S) -> T::Future {
self.as_ref().new_transform(service)
}
}
2019-03-05 04:38:11 +01:00
/// Trait for types that can be converted to a *transform service*
2019-03-09 15:36:23 +01:00
pub trait IntoTransform<T, S>
where
2019-03-09 15:36:23 +01:00
T: Transform<S>,
{
/// Convert to a `TransformService`
fn into_transform(self) -> T;
}
2019-03-09 15:36:23 +01:00
impl<T, S> IntoTransform<T, S> for T
where
2019-03-09 15:36:23 +01:00
T: Transform<S>,
{
fn into_transform(self) -> T {
self
}
}
2019-03-05 06:24:47 +01:00
2019-03-12 20:53:08 +01:00
/// Apply transform to service factory. Function returns
/// services factory that in initialization creates
/// service and applies transform to this service.
pub fn apply_transform<T, S, C, F, U>(
t: F,
service: U,
) -> impl NewService<
C,
Request = T::Request,
Response = T::Response,
Error = T::Error,
Service = T::Transform,
InitError = S::InitError,
> + Clone
where
S: NewService<C>,
T: Transform<S::Service, InitError = S::InitError>,
F: IntoTransform<T, S::Service>,
U: IntoNewService<S, C>,
{
ApplyTransform::new(t.into_transform(), service.into_new_service())
}
/// `Apply` transform to new service
pub struct ApplyTransform<T, S, C> {
s: Rc<S>,
2019-03-05 06:24:47 +01:00
t: Rc<T>,
2019-03-09 15:36:23 +01:00
_t: std::marker::PhantomData<C>,
2019-03-05 06:24:47 +01:00
}
2019-03-12 20:53:08 +01:00
impl<T, S, C> ApplyTransform<T, S, C>
2019-03-05 06:24:47 +01:00
where
2019-03-12 20:53:08 +01:00
S: NewService<C>,
T: Transform<S::Service, InitError = S::InitError>,
2019-03-05 06:24:47 +01:00
{
2019-03-12 20:53:08 +01:00
/// Create new `ApplyTransform` new service instance
pub fn new<F: IntoTransform<T, S::Service>>(t: F, service: S) -> Self {
2019-03-05 06:24:47 +01:00
Self {
2019-03-12 20:53:08 +01:00
s: Rc::new(service),
2019-03-05 06:24:47 +01:00
t: Rc::new(t.into_transform()),
_t: std::marker::PhantomData,
}
}
}
2019-03-12 20:53:08 +01:00
impl<T, S, C> Clone for ApplyTransform<T, S, C> {
fn clone(&self) -> Self {
ApplyTransform {
s: self.s.clone(),
t: self.t.clone(),
_t: std::marker::PhantomData,
}
}
}
impl<T, S, C> NewService<C> for ApplyTransform<T, S, C>
2019-03-05 06:24:47 +01:00
where
2019-03-12 20:53:08 +01:00
S: NewService<C>,
T: Transform<S::Service, InitError = S::InitError>,
2019-03-05 06:24:47 +01:00
{
2019-03-09 15:36:23 +01:00
type Request = T::Request;
2019-03-05 06:24:47 +01:00
type Response = T::Response;
type Error = T::Error;
type Service = T::Transform;
type InitError = T::InitError;
2019-03-12 20:53:08 +01:00
type Future = ApplyTransformFuture<T, S, C>;
2019-03-05 06:24:47 +01:00
2019-03-09 15:36:23 +01:00
fn new_service(&self, cfg: &C) -> Self::Future {
2019-03-05 06:24:47 +01:00
ApplyTransformFuture {
t_cell: self.t.clone(),
2019-03-12 20:53:08 +01:00
fut_a: self.s.new_service(cfg).into_future(),
2019-03-05 06:24:47 +01:00
fut_t: None,
}
}
}
2019-03-12 20:53:08 +01:00
pub struct ApplyTransformFuture<T, S, C>
2019-03-05 06:24:47 +01:00
where
2019-03-12 20:53:08 +01:00
S: NewService<C>,
T: Transform<S::Service, InitError = S::InitError>,
2019-03-05 06:24:47 +01:00
{
2019-03-12 20:53:08 +01:00
fut_a: S::Future,
2019-03-09 15:36:23 +01:00
fut_t: Option<<T::Future as IntoFuture>::Future>,
2019-03-05 06:24:47 +01:00
t_cell: Rc<T>,
}
2019-03-12 20:53:08 +01:00
impl<T, S, C> Future for ApplyTransformFuture<T, S, C>
2019-03-05 06:24:47 +01:00
where
2019-03-12 20:53:08 +01:00
S: NewService<C>,
T: Transform<S::Service, InitError = S::InitError>,
2019-03-05 06:24:47 +01:00
{
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)
}
}
}