use super::adapt::{AdaptLayer, Adapted};
use crate::{Exc, ExchangeError};
use tower::{util::MapErr, Layer, Service};
pub trait Request: Sized {
type Response;
}
pub trait Adaptor<R: Request>: Request {
fn from_request(req: R) -> Result<Self, ExchangeError>;
fn into_response(resp: Self::Response) -> Result<R::Response, ExchangeError>;
}
impl<T, R, E> Adaptor<R> for T
where
T: Request,
R: Request,
T: TryFrom<R, Error = E>,
T::Response: TryInto<R::Response, Error = E>,
ExchangeError: From<E>,
{
fn from_request(req: R) -> Result<Self, ExchangeError>
where
Self: Sized,
{
Ok(Self::try_from(req)?)
}
fn into_response(resp: Self::Response) -> Result<<R as Request>::Response, ExchangeError> {
Ok(resp.try_into()?)
}
}
pub trait ExcService<R>: Service<R, Response = R::Response, Error = ExchangeError>
where
R: Request,
{
fn apply<L, R2>(self, layer: &L) -> L::Service
where
Self: Sized,
R2: Request,
L: Layer<Self>,
L::Service: ExcService<R2>,
{
layer.layer(self)
}
fn adapt<R2>(self) -> Adapted<Self, R, R2>
where
Self: Sized,
R2: Request,
R2::Response: Send + 'static,
R: Adaptor<R2>,
Self::Future: Send + 'static,
{
self.apply(&AdaptLayer::default())
}
}
impl<S, R> ExcService<R> for S
where
S: Service<R, Response = R::Response, Error = ExchangeError>,
R: Request,
{
}
type MapErrFn<E> = fn(E) -> ExchangeError;
pub trait IntoExc<R>: Service<R, Response = R::Response>
where
Self::Error: Into<ExchangeError>,
R: Request,
{
fn into_exc(self) -> Exc<MapErr<Self, MapErrFn<Self::Error>>, R>
where
Self: Sized,
{
Exc::new(MapErr::new(self, Self::Error::into))
}
}
impl<S, R> IntoExc<R> for S
where
S: Service<R, Response = R::Response>,
S::Error: Into<ExchangeError>,
R: Request,
{
}