mirror of
https://github.com/tokio-rs/axum.git
synced 2025-01-16 22:43:03 +01:00
205 lines
5.3 KiB
Rust
205 lines
5.3 KiB
Rust
use crate::{body::Body, HandleError, extract::FromRequest, response::IntoResponse};
|
|
use async_trait::async_trait;
|
|
use bytes::Bytes;
|
|
use futures_util::future;
|
|
use http::{Request, Response};
|
|
use std::{
|
|
convert::Infallible,
|
|
future::Future,
|
|
marker::PhantomData,
|
|
task::{Context, Poll},
|
|
};
|
|
use tower::{Layer, BoxError, Service, ServiceExt};
|
|
|
|
mod sealed {
|
|
pub trait HiddentTrait {}
|
|
pub struct Hidden;
|
|
impl HiddentTrait for Hidden {}
|
|
}
|
|
|
|
#[async_trait]
|
|
pub trait Handler<B, In>: Sized {
|
|
type Response: IntoResponse<B>;
|
|
|
|
// This seals the trait. We cannot use the regular "sealed super trait" approach
|
|
// due to coherence.
|
|
#[doc(hidden)]
|
|
type Sealed: sealed::HiddentTrait;
|
|
|
|
async fn call(self, req: Request<Body>) -> Response<B>;
|
|
|
|
fn layer<L>(self, layer: L) -> Layered<L::Service, In>
|
|
where
|
|
L: Layer<HandlerSvc<Self, B, In>>,
|
|
{
|
|
Layered::new(layer.layer(HandlerSvc::new(self)))
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl<F, Fut, B, Res> Handler<B, ()> for F
|
|
where
|
|
F: FnOnce(Request<Body>) -> Fut + Send + Sync,
|
|
Fut: Future<Output = Res> + Send,
|
|
Res: IntoResponse<B>,
|
|
{
|
|
type Response = Res;
|
|
|
|
type Sealed = sealed::Hidden;
|
|
|
|
async fn call(self, req: Request<Body>) -> Response<B> {
|
|
self(req).await.into_response()
|
|
}
|
|
}
|
|
|
|
macro_rules! impl_handler {
|
|
() => {};
|
|
|
|
( $head:ident, $($tail:ident),* $(,)? ) => {
|
|
#[async_trait]
|
|
#[allow(non_snake_case)]
|
|
impl<F, Fut, B, Res, $head, $($tail,)*> Handler<B, ($head, $($tail,)*)> for F
|
|
where
|
|
F: FnOnce(Request<Body>, $head, $($tail,)*) -> Fut + Send + Sync,
|
|
Fut: Future<Output = Res> + Send,
|
|
Res: IntoResponse<B>,
|
|
$head: FromRequest<B> + Send,
|
|
$( $tail: FromRequest<B> + Send, )*
|
|
{
|
|
type Response = Res;
|
|
|
|
type Sealed = sealed::Hidden;
|
|
|
|
async fn call(self, mut req: Request<Body>) -> Response<B> {
|
|
let $head = match $head::from_request(&mut req).await {
|
|
Ok(value) => value,
|
|
Err(rejection) => return rejection.into_response(),
|
|
};
|
|
|
|
$(
|
|
let $tail = match $tail::from_request(&mut req).await {
|
|
Ok(value) => value,
|
|
Err(rejection) => return rejection.into_response(),
|
|
};
|
|
)*
|
|
|
|
let res = self(req, $head, $($tail,)*).await;
|
|
|
|
res.into_response()
|
|
}
|
|
}
|
|
|
|
impl_handler!($($tail,)*);
|
|
};
|
|
}
|
|
|
|
impl_handler!(T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15, T16);
|
|
|
|
pub struct Layered<S, T> {
|
|
svc: S,
|
|
_input: PhantomData<fn() -> T>,
|
|
}
|
|
|
|
impl<S, T> Clone for Layered<S, T>
|
|
where
|
|
S: Clone,
|
|
{
|
|
fn clone(&self) -> Self {
|
|
Self::new(self.svc.clone())
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl<S, B, T> Handler<B, T> for Layered<S, T>
|
|
where
|
|
S: Service<Request<Body>, Response = Response<B>> + Send,
|
|
S::Error: IntoResponse<B>,
|
|
S::Response: IntoResponse<B>,
|
|
S::Future: Send,
|
|
{
|
|
type Response = S::Response;
|
|
|
|
type Sealed = sealed::Hidden;
|
|
|
|
async fn call(self, req: Request<Body>) -> Self::Response {
|
|
// TODO(david): add tests for nesting services
|
|
match self
|
|
.svc
|
|
.oneshot(req)
|
|
.await
|
|
.map_err(IntoResponse::into_response)
|
|
{
|
|
Ok(res) => res,
|
|
Err(res) => res,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<S, T> Layered<S, T> {
|
|
pub(crate) fn new(svc: S) -> Self {
|
|
Self {
|
|
svc,
|
|
_input: PhantomData,
|
|
}
|
|
}
|
|
|
|
pub fn handle_error<F, B, Res>(self, f: F) -> Layered<HandleError<S, F, S::Error>, T>
|
|
where
|
|
S: Service<Request<Body>, Response = Response<B>>,
|
|
F: FnOnce(S::Error) -> Res,
|
|
Res: IntoResponse<Body>,
|
|
B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
|
|
B::Error: Into<BoxError> + Send + Sync + 'static,
|
|
{
|
|
let svc = HandleError::new(self.svc, f);
|
|
Layered::new(svc)
|
|
}
|
|
}
|
|
|
|
pub struct HandlerSvc<H, B, T> {
|
|
handler: H,
|
|
_input: PhantomData<fn() -> (B, T)>,
|
|
}
|
|
|
|
impl<H, B, T> HandlerSvc<H, B, T> {
|
|
pub(crate) fn new(handler: H) -> Self {
|
|
Self {
|
|
handler,
|
|
_input: PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<H, B, T> Clone for HandlerSvc<H, B, T>
|
|
where
|
|
H: Clone,
|
|
{
|
|
fn clone(&self) -> Self {
|
|
Self {
|
|
handler: self.handler.clone(),
|
|
_input: PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<H, B, T> Service<Request<Body>> for HandlerSvc<H, B, T>
|
|
where
|
|
H: Handler<B, T> + Clone + Send + 'static,
|
|
H::Response: 'static,
|
|
{
|
|
type Response = Response<B>;
|
|
type Error = Infallible;
|
|
type Future = future::BoxFuture<'static, Result<Self::Response, Self::Error>>;
|
|
|
|
fn poll_ready(&mut self, _cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
|
// HandlerSvc can only be constructed from async functions which are always ready, or from
|
|
// `Layered` which bufferes in `<Layered as Handler>::call` and is therefore also always
|
|
// ready.
|
|
Poll::Ready(Ok(()))
|
|
}
|
|
|
|
fn call(&mut self, req: Request<Body>) -> Self::Future {
|
|
let handler = self.handler.clone();
|
|
Box::pin(async move { Ok(Handler::call(handler, req).await) })
|
|
}
|
|
}
|