use std::marker; use futures::{Async, Future, Poll}; use tower_service::{NewService, Service}; /// `MapErr` service combinator pub struct MapErr { a: A, f: F, e: marker::PhantomData, } impl MapErr where A: Service, F: Fn(A::Error) -> E, { /// Create new `MapErr` combinator pub fn new(a: A, f: F) -> Self { Self { a, f, e: marker::PhantomData, } } } impl Service for MapErr where A: Service, F: Fn(A::Error) -> E, F: Clone, { type Request = A::Request; type Response = A::Response; type Error = E; type Future = MapErrFuture; fn poll_ready(&mut self) -> Poll<(), Self::Error> { self.a.poll_ready().map_err(&self.f) } fn call(&mut self, req: Self::Request) -> Self::Future { MapErrFuture::new(self.a.call(req), self.f.clone()) } } pub struct MapErrFuture where A: Service, F: Fn(A::Error) -> E, { f: F, fut: A::Future, } impl MapErrFuture where A: Service, F: Fn(A::Error) -> E, { fn new(fut: A::Future, f: F) -> Self { MapErrFuture { f, fut } } } impl Future for MapErrFuture where A: Service, F: Fn(A::Error) -> E, { type Item = A::Response; type Error = E; fn poll(&mut self) -> Poll { self.fut.poll().map_err(&self.f) } } /// `MapErrNewService` new service combinator pub struct MapErrNewService { a: A, f: F, e: marker::PhantomData, } impl MapErrNewService where A: NewService, 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 Clone for MapErrNewService where A: NewService + Clone, F: Fn(A::Error) -> E + Clone, { fn clone(&self) -> Self { Self { a: self.a.clone(), f: self.f.clone(), e: marker::PhantomData, } } } impl NewService for MapErrNewService where A: NewService, F: Fn(A::Error) -> E + Clone, { type Request = A::Request; type Response = A::Response; type Error = E; type Service = MapErr; type InitError = A::InitError; type Future = MapErrNewServiceFuture; fn new_service(&self) -> Self::Future { MapErrNewServiceFuture::new(self.a.new_service(), self.f.clone()) } } pub struct MapErrNewServiceFuture where A: NewService, F: Fn(A::Error) -> E, { fut: A::Future, f: F, } impl MapErrNewServiceFuture where A: NewService, F: Fn(A::Error) -> E, { fn new(fut: A::Future, f: F) -> Self { MapErrNewServiceFuture { f, fut } } } impl Future for MapErrNewServiceFuture where A: NewService, F: Fn(A::Error) -> E + Clone, { type Item = MapErr; type Error = A::InitError; fn poll(&mut self) -> Poll { if let Async::Ready(service) = self.fut.poll()? { Ok(Async::Ready(MapErr::new(service, self.f.clone()))) } else { Ok(Async::NotReady) } } }