1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2025-01-02 13:28:44 +00:00

add NewConfigurableService trait

This commit is contained in:
Nikolay Kim 2018-08-23 22:12:10 -07:00
parent aba9b09450
commit f2ef824011
3 changed files with 466 additions and 5 deletions

459
src/configurable.rs Normal file
View file

@ -0,0 +1,459 @@
use std::marker;
use futures::{future, future::FutureResult, Async, Future, IntoFuture, Poll};
use tower_service::Service;
use service::{AndThen, FnService, MapErr};
/// Creates new `Service` values.
///
/// Acts as a service factory. This is useful for cases where new `Service`
/// values must be produced. One case is a TCP servier listener. The listner
/// accepts new TCP streams, obtains a new `Service` value using the
/// `NewConfigurableService` trait, and uses that new `Service` value to
/// process inbound requests on that new TCP stream.
pub trait NewConfigurableService {
/// Requests handled by the service
type Request;
/// Responses given by the service
type Response;
/// Errors produced by the service
type Error;
/// The `Service` value created by this factory
type Service: Service<
Request = Self::Request,
Response = Self::Response,
Error = Self::Error,
>;
/// Pipeline configuration
type Config;
/// Errors produced while building a service.
type InitError;
/// The future of the `Service` instance.
type Future: Future<Item = Self::Service, Error = Self::InitError>;
/// Create and return a new service value asynchronously.
fn new_service(&self, Self::Config) -> Self::Future;
fn and_then<F, B>(self, new_service: F) -> AndThenNewConfigurableService<Self, B>
where
Self: Sized,
F: IntoNewConfigurableService<B>,
B: NewConfigurableService<
Request = Self::Response,
Error = Self::Error,
Config = Self::Config,
InitError = Self::InitError,
>,
{
AndThenNewConfigurableService::new(self, new_service)
}
fn map_err<F, E>(self, f: F) -> MapErrNewConfigurableService<Self, F, E>
where
Self: Sized,
F: Fn(Self::Error) -> E,
{
MapErrNewConfigurableService::new(self, f)
}
fn map_init_err<F, E>(self, f: F) -> MapInitErr<Self, F, E>
where
Self: Sized,
F: Fn(Self::InitError) -> E,
{
MapInitErr::new(self, f)
}
}
/// Trait for types that can be converted to a Service
pub trait IntoNewConfigurableService<T>
where
T: NewConfigurableService,
{
/// Create service
fn into_new_service(self) -> T;
}
impl<T> IntoNewConfigurableService<T> for T
where
T: NewConfigurableService,
{
fn into_new_service(self) -> T {
self
}
}
pub struct FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg>
where
F: Fn(Req) -> Fut,
Fut: IntoFuture<Item = Resp, Error = Err>,
{
f: F,
req: marker::PhantomData<Req>,
resp: marker::PhantomData<Resp>,
err: marker::PhantomData<Err>,
ierr: marker::PhantomData<IErr>,
cfg: marker::PhantomData<Cfg>,
}
impl<F, Req, Resp, Err, IErr, Fut, Cfg>
FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg>
where
F: Fn(Req) -> Fut + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>,
{
fn new(f: F) -> Self {
FnNewConfigurableService {
f,
req: marker::PhantomData,
resp: marker::PhantomData,
err: marker::PhantomData,
ierr: marker::PhantomData,
cfg: marker::PhantomData,
}
}
}
impl<F, Req, Resp, Err, IErr, Fut, Cfg> NewConfigurableService
for FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg>
where
F: Fn(Req) -> Fut + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>,
{
type Request = Req;
type Response = Resp;
type Error = Err;
type Service = FnService<F, Req, Resp, Err, Fut>;
type Config = Cfg;
type InitError = IErr;
type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self, _: Cfg) -> Self::Future {
future::ok(FnService::new(self.f.clone()))
}
}
impl<F, Req, Resp, Err, IErr, Fut, Cfg>
IntoNewConfigurableService<FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg>>
for F
where
F: Fn(Req) -> Fut + Clone + 'static,
Fut: IntoFuture<Item = Resp, Error = Err>,
{
fn into_new_service(self) -> FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg> {
FnNewConfigurableService::new(self)
}
}
impl<F, Req, Resp, Err, IErr, Fut, Cfg> Clone
for FnNewConfigurableService<F, Req, Resp, Err, IErr, Fut, Cfg>
where
F: Fn(Req) -> Fut + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>,
{
fn clone(&self) -> Self {
Self::new(self.f.clone())
}
}
/// `AndThenNewConfigurableService` new service combinator
pub struct AndThenNewConfigurableService<A, B> {
a: A,
b: B,
}
impl<A, B> AndThenNewConfigurableService<A, B>
where
A: NewConfigurableService,
B: NewConfigurableService,
{
/// Create new `AndThen` combinator
pub fn new<F: IntoNewConfigurableService<B>>(a: A, f: F) -> Self {
Self {
a,
b: f.into_new_service(),
}
}
}
impl<A, B> NewConfigurableService for AndThenNewConfigurableService<A, B>
where
A: NewConfigurableService<
Response = B::Request,
Config = B::Config,
InitError = B::InitError,
>,
A::Config: Clone,
A::Error: Into<B::Error>,
B: NewConfigurableService,
{
type Request = A::Request;
type Response = B::Response;
type Error = B::Error;
type Service = AndThen<A::Service, B::Service>;
type Config = A::Config;
type InitError = A::InitError;
type Future = AndThenNewConfigurableServiceFuture<A, B>;
fn new_service(&self, cfg: A::Config) -> Self::Future {
AndThenNewConfigurableServiceFuture::new(
self.a.new_service(cfg.clone()),
self.b.new_service(cfg),
)
}
}
impl<A, B> Clone for AndThenNewConfigurableService<A, B>
where
A: NewConfigurableService<Response = B::Request, InitError = B::InitError> + Clone,
A::Error: Into<B::Error>,
B: NewConfigurableService + Clone,
{
fn clone(&self) -> Self {
Self {
a: self.a.clone(),
b: self.b.clone(),
}
}
}
pub struct AndThenNewConfigurableServiceFuture<A, B>
where
A: NewConfigurableService,
B: NewConfigurableService,
{
fut_b: B::Future,
fut_a: A::Future,
a: Option<A::Service>,
b: Option<B::Service>,
}
impl<A, B> AndThenNewConfigurableServiceFuture<A, B>
where
A: NewConfigurableService,
B: NewConfigurableService,
{
fn new(fut_a: A::Future, fut_b: B::Future) -> Self {
AndThenNewConfigurableServiceFuture {
fut_a,
fut_b,
a: None,
b: None,
}
}
}
impl<A, B> Future for AndThenNewConfigurableServiceFuture<A, B>
where
A: NewConfigurableService,
A::Error: Into<B::Error>,
B: NewConfigurableService<Request = A::Response, InitError = A::InitError>,
{
type Item = AndThen<A::Service, B::Service>;
type Error = B::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Async::Ready(service) = self.fut_a.poll()? {
self.a = Some(service);
}
if let Async::Ready(service) = self.fut_b.poll()? {
self.b = Some(service);
}
if self.a.is_some() && self.b.is_some() {
Ok(Async::Ready(AndThen::new(
self.a.take().unwrap(),
self.b.take().unwrap(),
)))
} else {
Ok(Async::NotReady)
}
}
}
/// `MapErrNewService` new service combinator
pub struct MapErrNewConfigurableService<A, F, E> {
a: A,
f: F,
e: marker::PhantomData<E>,
}
impl<A, F, E> MapErrNewConfigurableService<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::Error) -> E,
{
/// Create new `MapErr` new service instance
pub fn new(a: A, f: F) -> Self {
Self {
a,
f,
e: marker::PhantomData,
}
}
}
impl<A, F, E> Clone for MapErrNewConfigurableService<A, F, E>
where
A: NewConfigurableService + Clone,
F: Fn(A::Error) -> E + Clone,
{
fn clone(&self) -> Self {
Self {
a: self.a.clone(),
f: self.f.clone(),
e: marker::PhantomData,
}
}
}
impl<A, F, E> NewConfigurableService for MapErrNewConfigurableService<A, F, E>
where
A: NewConfigurableService + Clone,
F: Fn(A::Error) -> E + Clone,
{
type Request = A::Request;
type Response = A::Response;
type Error = E;
type Service = MapErr<A::Service, F, E>;
type Config = A::Config;
type InitError = A::InitError;
type Future = MapErrNewConfigurableServiceFuture<A, F, E>;
fn new_service(&self, cfg: Self::Config) -> Self::Future {
MapErrNewConfigurableServiceFuture::new(self.a.new_service(cfg), self.f.clone())
}
}
pub struct MapErrNewConfigurableServiceFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::Error) -> E,
{
fut: A::Future,
f: F,
}
impl<A, F, E> MapErrNewConfigurableServiceFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::Error) -> E,
{
fn new(fut: A::Future, f: F) -> Self {
MapErrNewConfigurableServiceFuture { f, fut }
}
}
impl<A, F, E> Future for MapErrNewConfigurableServiceFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::Error) -> E + Clone,
{
type Item = MapErr<A::Service, F, E>;
type Error = A::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Async::Ready(service) = self.fut.poll()? {
Ok(Async::Ready(MapErr::new(service, self.f.clone())))
} else {
Ok(Async::NotReady)
}
}
}
/// `MapInitErr` service combinator
pub struct MapInitErr<A, F, E> {
a: A,
f: F,
e: marker::PhantomData<E>,
}
impl<A, F, E> MapInitErr<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::InitError) -> E,
{
/// Create new `MapInitErr` combinator
pub fn new(a: A, f: F) -> Self {
Self {
a,
f,
e: marker::PhantomData,
}
}
}
impl<A, F, E> Clone for MapInitErr<A, F, E>
where
A: NewConfigurableService + Clone,
F: Fn(A::InitError) -> E + Clone,
{
fn clone(&self) -> Self {
Self {
a: self.a.clone(),
f: self.f.clone(),
e: marker::PhantomData,
}
}
}
impl<A, F, E> NewConfigurableService for MapInitErr<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::InitError) -> E + Clone,
{
type Request = A::Request;
type Response = A::Response;
type Error = A::Error;
type Service = A::Service;
type Config = A::Config;
type InitError = E;
type Future = MapInitErrFuture<A, F, E>;
fn new_service(&self, cfg: Self::Config) -> Self::Future {
MapInitErrFuture::new(self.a.new_service(cfg), self.f.clone())
}
}
pub struct MapInitErrFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::InitError) -> E,
{
f: F,
fut: A::Future,
}
impl<A, F, E> MapInitErrFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::InitError) -> E,
{
fn new(fut: A::Future, f: F) -> Self {
MapInitErrFuture { f, fut }
}
}
impl<A, F, E> Future for MapInitErrFuture<A, F, E>
where
A: NewConfigurableService,
F: Fn(A::InitError) -> E,
{
type Item = A::Service;
type Error = E;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
self.fut.poll().map_err(&self.f)
}
}

View file

@ -62,7 +62,7 @@ impl Service for Connector {
fn call(&mut self, addr: String) -> Self::Future { fn call(&mut self, addr: String) -> Self::Future {
ConnectorFuture { ConnectorFuture {
fut: ResolveFut::new(addr, 0, &self.resolver), fut: ResolveFut::new(addr, 0, &self.resolver),
fut2: None fut2: None,
} }
} }
} }
@ -78,14 +78,14 @@ impl Future for ConnectorFuture {
fn poll(&mut self) -> Poll<Self::Item, Self::Error> { fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Some(ref mut fut) = self.fut2 { if let Some(ref mut fut) = self.fut2 {
return fut.poll() return fut.poll();
} }
match self.fut.poll()? { match self.fut.poll()? {
Async::Ready(addrs) => { Async::Ready(addrs) => {
self.fut2 = Some(TcpConnector::new(addrs)); self.fut2 = Some(TcpConnector::new(addrs));
self.poll() self.poll()
}, }
Async::NotReady => Ok(Async::NotReady) Async::NotReady => Ok(Async::NotReady),
} }
} }
} }

View file

@ -52,13 +52,15 @@ use actix::Message;
pub use tower_service::{NewService, Service}; pub use tower_service::{NewService, Service};
pub(crate) mod accept; pub(crate) mod accept;
mod configurable;
mod connector;
mod server; mod server;
mod server_service; mod server_service;
pub mod service; pub mod service;
pub mod ssl; pub mod ssl;
mod worker; mod worker;
mod connector;
pub use configurable::NewConfigurableService;
pub use connector::{Connector, ConnectorError}; pub use connector::{Connector, ConnectorError};
pub use server::Server; pub use server::Server;
pub use service::{IntoNewService, IntoService, NewServiceExt}; pub use service::{IntoNewService, IntoService, NewServiceExt};