use std::fmt;
use tower_async_layer::Layer;
use tower_async_service::Service;
#[derive(Clone)]
pub struct AndThen<S, F> {
inner: S,
f: F,
}
impl<S, F> fmt::Debug for AndThen<S, F>
where
S: fmt::Debug,
{
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("AndThen")
.field("inner", &self.inner)
.field("f", &format_args!("{}", std::any::type_name::<F>()))
.finish()
}
}
#[derive(Clone, Debug)]
pub struct AndThenLayer<F> {
f: F,
}
impl<S, F> AndThen<S, F> {
pub fn new(inner: S, f: F) -> Self {
AndThen { f, inner }
}
pub fn layer(f: F) -> AndThenLayer<F> {
AndThenLayer { f }
}
}
impl<S, F, Request, Fut, Output, Error> Service<Request> for AndThen<S, F>
where
S: Service<Request>,
S::Error: Into<Error>,
F: Fn(S::Response) -> Fut,
Fut: std::future::Future<Output = Result<Output, Error>>,
{
type Response = Output;
type Error = Error;
async fn call(&self, request: Request) -> Result<Self::Response, Self::Error> {
let result = self.inner.call(request).await;
match result {
Ok(response) => (self.f)(response).await,
Err(error) => Err(error.into()),
}
}
}
impl<F> AndThenLayer<F> {
pub fn new(f: F) -> Self {
AndThenLayer { f }
}
}
impl<S, F> Layer<S> for AndThenLayer<F>
where
F: Clone,
{
type Service = AndThen<S, F>;
fn layer(&self, inner: S) -> Self::Service {
AndThen {
f: self.f.clone(),
inner,
}
}
}