mirror of
https://github.com/tokio-rs/axum.git
synced 2025-09-29 22:10:59 +00:00
100 lines
2.8 KiB
Rust
100 lines
2.8 KiB
Rust
//! [`Service`](tower::Service) future types.
|
|
|
|
use crate::{
|
|
body::{box_body, BoxBody},
|
|
response::IntoResponse,
|
|
util::{Either, EitherProj},
|
|
BoxError,
|
|
};
|
|
use bytes::Bytes;
|
|
use futures_util::ready;
|
|
use http::{Method, Request, Response};
|
|
use http_body::Empty;
|
|
use pin_project_lite::pin_project;
|
|
use std::{
|
|
future::Future,
|
|
pin::Pin,
|
|
task::{Context, Poll},
|
|
};
|
|
use tower::util::Oneshot;
|
|
use tower_service::Service;
|
|
|
|
pin_project! {
|
|
/// Response future for [`HandleError`](super::HandleError).
|
|
#[derive(Debug)]
|
|
pub struct HandleErrorFuture<Fut, F> {
|
|
#[pin]
|
|
pub(super) inner: Fut,
|
|
pub(super) f: Option<F>,
|
|
}
|
|
}
|
|
|
|
impl<Fut, F, E, E2, B, Res> Future for HandleErrorFuture<Fut, F>
|
|
where
|
|
Fut: Future<Output = Result<Response<B>, E>>,
|
|
F: FnOnce(E) -> Result<Res, E2>,
|
|
Res: IntoResponse,
|
|
B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
|
|
B::Error: Into<BoxError> + Send + Sync + 'static,
|
|
{
|
|
type Output = Result<Response<BoxBody>, E2>;
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
let this = self.project();
|
|
|
|
match ready!(this.inner.poll(cx)) {
|
|
Ok(res) => Ok(res.map(box_body)).into(),
|
|
Err(err) => {
|
|
let f = this.f.take().unwrap();
|
|
match f(err) {
|
|
Ok(res) => Ok(res.into_response().map(box_body)).into(),
|
|
Err(err) => Err(err).into(),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pin_project! {
|
|
/// The response future for [`OnMethod`](super::OnMethod).
|
|
pub struct OnMethodFuture<S, F, B>
|
|
where
|
|
S: Service<Request<B>>,
|
|
F: Service<Request<B>>
|
|
{
|
|
#[pin]
|
|
pub(super) inner: Either<
|
|
Oneshot<S, Request<B>>,
|
|
Oneshot<F, Request<B>>,
|
|
>,
|
|
// pub(super) inner: crate::routing::future::RouteFuture<S, F, B>,
|
|
pub(super) req_method: Method,
|
|
}
|
|
}
|
|
|
|
impl<S, F, B, ResBody> Future for OnMethodFuture<S, F, B>
|
|
where
|
|
S: Service<Request<B>, Response = Response<ResBody>> + Clone,
|
|
ResBody: http_body::Body<Data = Bytes> + Send + Sync + 'static,
|
|
ResBody::Error: Into<BoxError>,
|
|
F: Service<Request<B>, Response = Response<BoxBody>, Error = S::Error>,
|
|
{
|
|
type Output = Result<Response<BoxBody>, S::Error>;
|
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
let this = self.project();
|
|
|
|
let response = match this.inner.project() {
|
|
EitherProj::A { inner } => ready!(inner.poll(cx))?.map(box_body),
|
|
EitherProj::B { inner } => ready!(inner.poll(cx))?,
|
|
};
|
|
|
|
if this.req_method == &Method::HEAD {
|
|
let response = response.map(|_| box_body(Empty::new()));
|
|
Poll::Ready(Ok(response))
|
|
} else {
|
|
Poll::Ready(Ok(response))
|
|
}
|
|
}
|
|
}
|