2018-09-28 00:05:48 +00:00
|
|
|
//! Contains `Either` service and related types and functions.
|
|
|
|
use futures::{future, Async, Future, Poll};
|
|
|
|
|
|
|
|
use super::service::{NewService, Service};
|
|
|
|
|
|
|
|
/// Combine two different service types into a single type.
|
|
|
|
///
|
|
|
|
/// Both services must be of the same request, response, and error types.
|
|
|
|
/// `EitherService` is useful for handling conditional branching in service
|
|
|
|
/// middleware to different inner service types.
|
|
|
|
pub enum EitherService<A, B> {
|
|
|
|
A(A),
|
|
|
|
B(B),
|
|
|
|
}
|
|
|
|
|
2018-11-30 02:56:15 +00:00
|
|
|
impl<A, B, Request> Service<Request> for EitherService<A, B>
|
2018-09-28 00:05:48 +00:00
|
|
|
where
|
2018-11-30 02:56:15 +00:00
|
|
|
A: Service<Request>,
|
|
|
|
B: Service<Request, Response = A::Response, Error = A::Error>,
|
2018-09-28 00:05:48 +00:00
|
|
|
{
|
|
|
|
type Response = A::Response;
|
|
|
|
type Error = A::Error;
|
|
|
|
type Future = future::Either<A::Future, B::Future>;
|
|
|
|
|
|
|
|
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
|
|
|
match self {
|
|
|
|
EitherService::A(ref mut inner) => inner.poll_ready(),
|
|
|
|
EitherService::B(ref mut inner) => inner.poll_ready(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-30 02:56:15 +00:00
|
|
|
fn call(&mut self, req: Request) -> Self::Future {
|
2018-09-28 00:05:48 +00:00
|
|
|
match self {
|
|
|
|
EitherService::A(ref mut inner) => future::Either::A(inner.call(req)),
|
|
|
|
EitherService::B(ref mut inner) => future::Either::B(inner.call(req)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Combine two different new service types into a single type.
|
|
|
|
pub enum Either<A, B> {
|
|
|
|
A(A),
|
|
|
|
B(B),
|
|
|
|
}
|
|
|
|
|
2018-11-30 02:56:15 +00:00
|
|
|
impl<A, B, Request> NewService<Request> for Either<A, B>
|
2018-09-28 00:05:48 +00:00
|
|
|
where
|
2018-11-30 02:56:15 +00:00
|
|
|
A: NewService<Request>,
|
|
|
|
B: NewService<Request, Response = A::Response, Error = A::Error, InitError = A::InitError>,
|
2018-09-28 00:05:48 +00:00
|
|
|
{
|
|
|
|
type Response = A::Response;
|
|
|
|
type Error = A::Error;
|
|
|
|
type InitError = A::InitError;
|
|
|
|
type Service = EitherService<A::Service, B::Service>;
|
2018-11-30 02:56:15 +00:00
|
|
|
type Future = EitherNewService<A, B, Request>;
|
2018-09-28 00:05:48 +00:00
|
|
|
|
|
|
|
fn new_service(&self) -> Self::Future {
|
|
|
|
match self {
|
|
|
|
Either::A(ref inner) => EitherNewService::A(inner.new_service()),
|
|
|
|
Either::B(ref inner) => EitherNewService::B(inner.new_service()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
2018-11-30 02:56:15 +00:00
|
|
|
pub enum EitherNewService<A: NewService<R>, B: NewService<R>, R> {
|
2018-09-28 00:05:48 +00:00
|
|
|
A(A::Future),
|
|
|
|
B(B::Future),
|
|
|
|
}
|
|
|
|
|
2018-11-30 02:56:15 +00:00
|
|
|
impl<A, B, Request> Future for EitherNewService<A, B, Request>
|
2018-09-28 00:05:48 +00:00
|
|
|
where
|
2018-11-30 02:56:15 +00:00
|
|
|
A: NewService<Request>,
|
|
|
|
B: NewService<Request, Response = A::Response, Error = A::Error, InitError = A::InitError>,
|
2018-09-28 00:05:48 +00:00
|
|
|
{
|
|
|
|
type Item = EitherService<A::Service, B::Service>;
|
|
|
|
type Error = A::InitError;
|
|
|
|
|
|
|
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
|
|
|
match self {
|
|
|
|
EitherNewService::A(ref mut fut) => {
|
|
|
|
let service = try_ready!(fut.poll());
|
|
|
|
Ok(Async::Ready(EitherService::A(service)))
|
|
|
|
}
|
|
|
|
EitherNewService::B(ref mut fut) => {
|
|
|
|
let service = try_ready!(fut.poll());
|
|
|
|
Ok(Async::Ready(EitherService::B(service)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|