Remove some needless type params

This commit is contained in:
David Pedersen 2021-06-06 22:41:52 +02:00
parent 696fce7b76
commit 7620f17edc
8 changed files with 132 additions and 139 deletions

View file

@ -3,8 +3,8 @@ use async_trait::async_trait;
use bytes::Bytes; use bytes::Bytes;
use http::{header, Request, Response}; use http::{header, Request, Response};
use rejection::{ use rejection::{
BodyAlreadyTaken, FailedToBufferBody, InvalidJsonBody, InvalidUtf8, LengthRequired, BodyAlreadyTaken, FailedToBufferBody, InvalidJsonBody, InvalidUrlParam, InvalidUtf8,
MissingExtension, MissingJsonContentType, MissingRouteParams, PayloadTooLarge, LengthRequired, MissingExtension, MissingJsonContentType, MissingRouteParams, PayloadTooLarge,
QueryStringMissing, QueryStringMissing,
}; };
use serde::de::DeserializeOwned; use serde::de::DeserializeOwned;
@ -13,16 +13,16 @@ use std::{collections::HashMap, convert::Infallible, str::FromStr};
pub mod rejection; pub mod rejection;
#[async_trait] #[async_trait]
pub trait FromRequest<B>: Sized { pub trait FromRequest: Sized {
type Rejection: IntoResponse<B>; type Rejection: IntoResponse;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection>; async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection>;
} }
#[async_trait] #[async_trait]
impl<T, B> FromRequest<B> for Option<T> impl<T> FromRequest for Option<T>
where where
T: FromRequest<B>, T: FromRequest,
{ {
type Rejection = Infallible; type Rejection = Infallible;
@ -35,7 +35,7 @@ where
pub struct Query<T>(pub T); pub struct Query<T>(pub T);
#[async_trait] #[async_trait]
impl<T> FromRequest<Body> for Query<T> impl<T> FromRequest for Query<T>
where where
T: DeserializeOwned, T: DeserializeOwned,
{ {
@ -52,7 +52,7 @@ where
pub struct Json<T>(pub T); pub struct Json<T>(pub T);
#[async_trait] #[async_trait]
impl<T> FromRequest<Body> for Json<T> impl<T> FromRequest for Json<T>
where where
T: DeserializeOwned, T: DeserializeOwned,
{ {
@ -98,7 +98,7 @@ fn has_content_type<B>(req: &Request<B>, expected_content_type: &str) -> bool {
pub struct Extension<T>(pub T); pub struct Extension<T>(pub T);
#[async_trait] #[async_trait]
impl<T> FromRequest<Body> for Extension<T> impl<T> FromRequest for Extension<T>
where where
T: Clone + Send + Sync + 'static, T: Clone + Send + Sync + 'static,
{ {
@ -116,7 +116,7 @@ where
} }
#[async_trait] #[async_trait]
impl FromRequest<Body> for Bytes { impl FromRequest for Bytes {
type Rejection = Response<Body>; type Rejection = Response<Body>;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> { async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
@ -132,7 +132,7 @@ impl FromRequest<Body> for Bytes {
} }
#[async_trait] #[async_trait]
impl FromRequest<Body> for String { impl FromRequest for String {
type Rejection = Response<Body>; type Rejection = Response<Body>;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> { async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
@ -153,7 +153,7 @@ impl FromRequest<Body> for String {
} }
#[async_trait] #[async_trait]
impl FromRequest<Body> for Body { impl FromRequest for Body {
type Rejection = BodyAlreadyTaken; type Rejection = BodyAlreadyTaken;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> { async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
@ -165,7 +165,7 @@ impl FromRequest<Body> for Body {
pub struct BytesMaxLength<const N: u64>(pub Bytes); pub struct BytesMaxLength<const N: u64>(pub Bytes);
#[async_trait] #[async_trait]
impl<const N: u64> FromRequest<Body> for BytesMaxLength<N> { impl<const N: u64> FromRequest for BytesMaxLength<N> {
type Rejection = Response<Body>; type Rejection = Response<Body>;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> { async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
@ -208,7 +208,7 @@ impl UrlParamsMap {
} }
#[async_trait] #[async_trait]
impl FromRequest<Body> for UrlParamsMap { impl FromRequest for UrlParamsMap {
type Rejection = MissingRouteParams; type Rejection = MissingRouteParams;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> { async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
@ -224,30 +224,6 @@ impl FromRequest<Body> for UrlParamsMap {
} }
} }
#[derive(Debug)]
pub struct InvalidUrlParam {
type_name: &'static str,
}
impl InvalidUrlParam {
fn new<T>() -> Self {
InvalidUrlParam {
type_name: std::any::type_name::<T>(),
}
}
}
impl IntoResponse<Body> for InvalidUrlParam {
fn into_response(self) -> http::Response<Body> {
let mut res = http::Response::new(Body::from(format!(
"Invalid URL param. Expected something of type `{}`",
self.type_name
)));
*res.status_mut() = http::StatusCode::BAD_REQUEST;
res
}
}
pub struct UrlParams<T>(pub T); pub struct UrlParams<T>(pub T);
macro_rules! impl_parse_url { macro_rules! impl_parse_url {
@ -255,7 +231,7 @@ macro_rules! impl_parse_url {
( $head:ident, $($tail:ident),* $(,)? ) => { ( $head:ident, $($tail:ident),* $(,)? ) => {
#[async_trait] #[async_trait]
impl<$head, $($tail,)*> FromRequest<Body> for UrlParams<($head, $($tail,)*)> impl<$head, $($tail,)*> FromRequest for UrlParams<($head, $($tail,)*)>
where where
$head: FromStr + Send, $head: FromStr + Send,
$( $tail: FromStr + Send, )* $( $tail: FromStr + Send, )*

View file

@ -10,7 +10,7 @@ macro_rules! define_rejection {
#[derive(Debug)] #[derive(Debug)]
pub struct $name(pub(super) ()); pub struct $name(pub(super) ());
impl IntoResponse<Body> for $name { impl IntoResponse for $name {
fn into_response(self) -> http::Response<Body> { fn into_response(self) -> http::Response<Body> {
let mut res = http::Response::new(Body::from($body)); let mut res = http::Response::new(Body::from($body));
*res.status_mut() = http::StatusCode::$status; *res.status_mut() = http::StatusCode::$status;
@ -36,7 +36,7 @@ macro_rules! define_rejection {
} }
} }
impl IntoResponse<Body> for $name { impl IntoResponse for $name {
fn into_response(self) -> http::Response<Body> { fn into_response(self) -> http::Response<Body> {
let mut res = let mut res =
http::Response::new(Body::from(format!(concat!($body, ": {}"), self.0))); http::Response::new(Body::from(format!(concat!($body, ": {}"), self.0)));
@ -106,3 +106,27 @@ define_rejection! {
#[body = "Cannot have two request body extractors for a single handler"] #[body = "Cannot have two request body extractors for a single handler"]
pub struct BodyAlreadyTaken(()); pub struct BodyAlreadyTaken(());
} }
#[derive(Debug)]
pub struct InvalidUrlParam {
type_name: &'static str,
}
impl InvalidUrlParam {
pub(super) fn new<T>() -> Self {
InvalidUrlParam {
type_name: std::any::type_name::<T>(),
}
}
}
impl IntoResponse for InvalidUrlParam {
fn into_response(self) -> http::Response<Body> {
let mut res = http::Response::new(Body::from(format!(
"Invalid URL param. Expected something of type `{}`",
self.type_name
)));
*res.status_mut() = http::StatusCode::BAD_REQUEST;
res
}
}

View file

@ -17,23 +17,23 @@ use std::{
}; };
use tower::{util::Oneshot, BoxError, Layer, Service, ServiceExt}; use tower::{util::Oneshot, BoxError, Layer, Service, ServiceExt};
pub fn get<H, B, T>(handler: H) -> OnMethod<IntoService<H, B, T>, EmptyRouter> pub fn get<H, T>(handler: H) -> OnMethod<IntoService<H, T>, EmptyRouter>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
on(MethodFilter::Get, handler) on(MethodFilter::Get, handler)
} }
pub fn post<H, B, T>(handler: H) -> OnMethod<IntoService<H, B, T>, EmptyRouter> pub fn post<H, T>(handler: H) -> OnMethod<IntoService<H, T>, EmptyRouter>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
on(MethodFilter::Post, handler) on(MethodFilter::Post, handler)
} }
pub fn on<H, B, T>(method: MethodFilter, handler: H) -> OnMethod<IntoService<H, B, T>, EmptyRouter> pub fn on<H, T>(method: MethodFilter, handler: H) -> OnMethod<IntoService<H, T>, EmptyRouter>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
OnMethod { OnMethod {
method, method,
@ -51,41 +51,37 @@ mod sealed {
} }
#[async_trait] #[async_trait]
pub trait Handler<B, In>: Sized { pub trait Handler<In>: Sized {
type Response: IntoResponse<B>;
// This seals the trait. We cannot use the regular "sealed super trait" approach // This seals the trait. We cannot use the regular "sealed super trait" approach
// due to coherence. // due to coherence.
#[doc(hidden)] #[doc(hidden)]
type Sealed: sealed::HiddentTrait; type Sealed: sealed::HiddentTrait;
async fn call(self, req: Request<Body>) -> Response<B>; async fn call(self, req: Request<Body>) -> Response<BoxBody>;
fn layer<L>(self, layer: L) -> Layered<L::Service, In> fn layer<L>(self, layer: L) -> Layered<L::Service, In>
where where
L: Layer<IntoService<Self, B, In>>, L: Layer<IntoService<Self, In>>,
{ {
Layered::new(layer.layer(IntoService::new(self))) Layered::new(layer.layer(IntoService::new(self)))
} }
fn into_service(self) -> IntoService<Self, B, In> { fn into_service(self) -> IntoService<Self, In> {
IntoService::new(self) IntoService::new(self)
} }
} }
#[async_trait] #[async_trait]
impl<F, Fut, B, Res> Handler<B, ()> for F impl<F, Fut, Res> Handler<()> for F
where where
F: FnOnce(Request<Body>) -> Fut + Send + Sync, F: FnOnce(Request<Body>) -> Fut + Send + Sync,
Fut: Future<Output = Res> + Send, Fut: Future<Output = Res> + Send,
Res: IntoResponse<B>, Res: IntoResponse,
{ {
type Response = Res;
type Sealed = sealed::Hidden; type Sealed = sealed::Hidden;
async fn call(self, req: Request<Body>) -> Response<B> { async fn call(self, req: Request<Body>) -> Response<BoxBody> {
self(req).await.into_response() self(req).await.into_response().map(BoxBody::new)
} }
} }
@ -95,34 +91,32 @@ macro_rules! impl_handler {
( $head:ident, $($tail:ident),* $(,)? ) => { ( $head:ident, $($tail:ident),* $(,)? ) => {
#[async_trait] #[async_trait]
#[allow(non_snake_case)] #[allow(non_snake_case)]
impl<F, Fut, B, Res, $head, $($tail,)*> Handler<B, ($head, $($tail,)*)> for F impl<F, Fut, Res, $head, $($tail,)*> Handler<($head, $($tail,)*)> for F
where where
F: FnOnce(Request<Body>, $head, $($tail,)*) -> Fut + Send + Sync, F: FnOnce(Request<Body>, $head, $($tail,)*) -> Fut + Send + Sync,
Fut: Future<Output = Res> + Send, Fut: Future<Output = Res> + Send,
Res: IntoResponse<B>, Res: IntoResponse,
$head: FromRequest<B> + Send, $head: FromRequest + Send,
$( $tail: FromRequest<B> + Send, )* $( $tail: FromRequest + Send, )*
{ {
type Response = Res;
type Sealed = sealed::Hidden; type Sealed = sealed::Hidden;
async fn call(self, mut req: Request<Body>) -> Response<B> { async fn call(self, mut req: Request<Body>) -> Response<BoxBody> {
let $head = match $head::from_request(&mut req).await { let $head = match $head::from_request(&mut req).await {
Ok(value) => value, Ok(value) => value,
Err(rejection) => return rejection.into_response(), Err(rejection) => return rejection.into_response().map(BoxBody::new),
}; };
$( $(
let $tail = match $tail::from_request(&mut req).await { let $tail = match $tail::from_request(&mut req).await {
Ok(value) => value, Ok(value) => value,
Err(rejection) => return rejection.into_response(), Err(rejection) => return rejection.into_response().map(BoxBody::new),
}; };
)* )*
let res = self(req, $head, $($tail,)*).await; let res = self(req, $head, $($tail,)*).await;
res.into_response() res.into_response().map(BoxBody::new)
} }
} }
@ -147,27 +141,26 @@ where
} }
#[async_trait] #[async_trait]
impl<S, B, T> Handler<B, T> for Layered<S, T> impl<S, T, B> Handler<T> for Layered<S, T>
where where
S: Service<Request<Body>, Response = Response<B>> + Send, S: Service<Request<Body>, Response = Response<B>> + Send,
S::Error: IntoResponse<B>, // S::Response: IntoResponse,
S::Response: IntoResponse<B>, S::Error: IntoResponse,
S::Future: Send, S::Future: Send,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static,
{ {
type Response = S::Response;
type Sealed = sealed::Hidden; type Sealed = sealed::Hidden;
async fn call(self, req: Request<Body>) -> Self::Response { async fn call(self, req: Request<Body>) -> Response<BoxBody> {
// TODO(david): add tests for nesting services
match self match self
.svc .svc
.oneshot(req) .oneshot(req)
.await .await
.map_err(IntoResponse::into_response) .map_err(IntoResponse::into_response)
{ {
Ok(res) => res, Ok(res) => res.map(BoxBody::new),
Err(res) => res, Err(res) => res.map(BoxBody::new),
} }
} }
} }
@ -184,21 +177,19 @@ impl<S, T> Layered<S, T> {
where where
S: Service<Request<Body>, Response = Response<B>>, S: Service<Request<Body>, Response = Response<B>>,
F: FnOnce(S::Error) -> Res, F: FnOnce(S::Error) -> Res,
Res: IntoResponse<B>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static,
{ {
let svc = HandleError::new(self.svc, f); let svc = HandleError::new(self.svc, f);
Layered::new(svc) Layered::new(svc)
} }
} }
pub struct IntoService<H, B, T> { pub struct IntoService<H, T> {
handler: H, handler: H,
_marker: PhantomData<fn() -> (B, T)>, _marker: PhantomData<fn() -> T>,
} }
impl<H, B, T> IntoService<H, B, T> { impl<H, T> IntoService<H, T> {
fn new(handler: H) -> Self { fn new(handler: H) -> Self {
Self { Self {
handler, handler,
@ -207,7 +198,7 @@ impl<H, B, T> IntoService<H, B, T> {
} }
} }
impl<H, B, T> Clone for IntoService<H, B, T> impl<H, T> Clone for IntoService<H, T>
where where
H: Clone, H: Clone,
{ {
@ -219,12 +210,11 @@ where
} }
} }
impl<H, B, T> Service<Request<Body>> for IntoService<H, B, T> impl<H, T> Service<Request<Body>> for IntoService<H, T>
where where
H: Handler<B, T> + Clone + Send + 'static, H: Handler<T> + Clone + Send + 'static,
H::Response: 'static,
{ {
type Response = Response<B>; type Response = Response<BoxBody>;
type Error = Infallible; type Error = Infallible;
type Future = future::BoxFuture<'static, Result<Self::Response, Self::Error>>; type Future = future::BoxFuture<'static, Result<Self::Response, Self::Error>>;
@ -237,7 +227,10 @@ where
fn call(&mut self, req: Request<Body>) -> Self::Future { fn call(&mut self, req: Request<Body>) -> Self::Future {
let handler = self.handler.clone(); let handler = self.handler.clone();
Box::pin(async move { Ok(Handler::call(handler, req).await) }) Box::pin(async move {
let res = Handler::call(handler, req).await;
Ok(res)
})
} }
} }
@ -249,27 +242,23 @@ pub struct OnMethod<S, F> {
} }
impl<S, F> OnMethod<S, F> { impl<S, F> OnMethod<S, F> {
pub fn get<H, B, T>(self, handler: H) -> OnMethod<IntoService<H, B, T>, Self> pub fn get<H, T>(self, handler: H) -> OnMethod<IntoService<H, T>, Self>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
self.on(MethodFilter::Get, handler) self.on(MethodFilter::Get, handler)
} }
pub fn post<H, B, T>(self, handler: H) -> OnMethod<IntoService<H, B, T>, Self> pub fn post<H, T>(self, handler: H) -> OnMethod<IntoService<H, T>, Self>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
self.on(MethodFilter::Post, handler) self.on(MethodFilter::Post, handler)
} }
pub fn on<H, B, T>( pub fn on<H, T>(self, method: MethodFilter, handler: H) -> OnMethod<IntoService<H, T>, Self>
self,
method: MethodFilter,
handler: H,
) -> OnMethod<IntoService<H, B, T>, Self>
where where
H: Handler<B, T>, H: Handler<T>,
{ {
OnMethod { OnMethod {
method, method,

View file

@ -637,7 +637,7 @@ pub trait ServiceExt<B>: Service<Request<Body>, Response = Response<B>> {
where where
Self: Sized, Self: Sized,
F: FnOnce(Self::Error) -> Res, F: FnOnce(Self::Error) -> Res,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {

View file

@ -5,29 +5,28 @@ use serde::Serialize;
use std::{borrow::Cow, convert::Infallible}; use std::{borrow::Cow, convert::Infallible};
use tower::util::Either; use tower::util::Either;
// TODO(david): can we change this to not be generic over the body and just use hyper::Body? pub trait IntoResponse {
pub trait IntoResponse<B> { fn into_response(self) -> Response<Body>;
fn into_response(self) -> Response<B>;
} }
impl IntoResponse<Body> for () { impl IntoResponse for () {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
Response::new(Body::empty()) Response::new(Body::empty())
} }
} }
impl<B> IntoResponse<B> for Infallible { impl IntoResponse for Infallible {
fn into_response(self) -> Response<B> { fn into_response(self) -> Response<Body> {
match self {} match self {}
} }
} }
impl<T, K, B> IntoResponse<B> for Either<T, K> impl<T, K> IntoResponse for Either<T, K>
where where
T: IntoResponse<B>, T: IntoResponse,
K: IntoResponse<B>, K: IntoResponse,
{ {
fn into_response(self) -> Response<B> { fn into_response(self) -> Response<Body> {
match self { match self {
Either::A(inner) => inner.into_response(), Either::A(inner) => inner.into_response(),
Either::B(inner) => inner.into_response(), Either::B(inner) => inner.into_response(),
@ -35,12 +34,12 @@ where
} }
} }
impl<B, T, E> IntoResponse<B> for Result<T, E> impl<T, E> IntoResponse for Result<T, E>
where where
T: IntoResponse<B>, T: IntoResponse,
E: IntoResponse<B>, E: IntoResponse,
{ {
fn into_response(self) -> Response<B> { fn into_response(self) -> Response<Body> {
match self { match self {
Ok(value) => value.into_response(), Ok(value) => value.into_response(),
Err(err) => err.into_response(), Err(err) => err.into_response(),
@ -48,27 +47,27 @@ where
} }
} }
impl<B> IntoResponse<B> for Response<B> { impl IntoResponse for Response<Body> {
fn into_response(self) -> Self { fn into_response(self) -> Self {
self self
} }
} }
impl IntoResponse<Body> for &'static str { impl IntoResponse for &'static str {
#[inline] #[inline]
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
Cow::Borrowed(self).into_response() Cow::Borrowed(self).into_response()
} }
} }
impl IntoResponse<Body> for String { impl IntoResponse for String {
#[inline] #[inline]
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
Cow::<'static, str>::Owned(self).into_response() Cow::<'static, str>::Owned(self).into_response()
} }
} }
impl IntoResponse<Body> for std::borrow::Cow<'static, str> { impl IntoResponse for std::borrow::Cow<'static, str> {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
let mut res = Response::new(Body::from(self)); let mut res = Response::new(Body::from(self));
res.headers_mut() res.headers_mut()
@ -77,7 +76,7 @@ impl IntoResponse<Body> for std::borrow::Cow<'static, str> {
} }
} }
impl IntoResponse<Body> for Bytes { impl IntoResponse for Bytes {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
let mut res = Response::new(Body::from(self)); let mut res = Response::new(Body::from(self));
res.headers_mut().insert( res.headers_mut().insert(
@ -88,7 +87,7 @@ impl IntoResponse<Body> for Bytes {
} }
} }
impl IntoResponse<Body> for &'static [u8] { impl IntoResponse for &'static [u8] {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
let mut res = Response::new(Body::from(self)); let mut res = Response::new(Body::from(self));
res.headers_mut().insert( res.headers_mut().insert(
@ -99,7 +98,7 @@ impl IntoResponse<Body> for &'static [u8] {
} }
} }
impl IntoResponse<Body> for Vec<u8> { impl IntoResponse for Vec<u8> {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
let mut res = Response::new(Body::from(self)); let mut res = Response::new(Body::from(self));
res.headers_mut().insert( res.headers_mut().insert(
@ -110,7 +109,7 @@ impl IntoResponse<Body> for Vec<u8> {
} }
} }
impl IntoResponse<Body> for std::borrow::Cow<'static, [u8]> { impl IntoResponse for std::borrow::Cow<'static, [u8]> {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
let mut res = Response::new(Body::from(self)); let mut res = Response::new(Body::from(self));
res.headers_mut().insert( res.headers_mut().insert(
@ -121,7 +120,7 @@ impl IntoResponse<Body> for std::borrow::Cow<'static, [u8]> {
} }
} }
impl IntoResponse<Body> for StatusCode { impl IntoResponse for StatusCode {
fn into_response(self) -> Response<Body> { fn into_response(self) -> Response<Body> {
Response::builder() Response::builder()
.status(self) .status(self)
@ -130,7 +129,7 @@ impl IntoResponse<Body> for StatusCode {
} }
} }
impl<T> IntoResponse<Body> for (StatusCode, T) impl<T> IntoResponse for (StatusCode, T)
where where
T: Into<Body>, T: Into<Body>,
{ {
@ -142,7 +141,7 @@ where
} }
} }
impl<T> IntoResponse<Body> for (StatusCode, HeaderMap, T) impl<T> IntoResponse for (StatusCode, HeaderMap, T)
where where
T: Into<Body>, T: Into<Body>,
{ {
@ -156,7 +155,7 @@ where
pub struct Html<T>(pub T); pub struct Html<T>(pub T);
impl<T> IntoResponse<Body> for Html<T> impl<T> IntoResponse for Html<T>
where where
T: Into<Body>, T: Into<Body>,
{ {
@ -170,7 +169,7 @@ where
pub struct Json<T>(pub T); pub struct Json<T>(pub T);
impl<T> IntoResponse<Body> for Json<T> impl<T> IntoResponse for Json<T>
where where
T: Serialize, T: Serialize,
{ {

View file

@ -389,7 +389,7 @@ impl<S> Layered<S> {
where where
S: Service<Request<Body>, Response = Response<B>> + Clone, S: Service<Request<Body>, Response = Response<B>> + Clone,
F: FnOnce(S::Error) -> Res, F: FnOnce(S::Error) -> Res,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {
@ -428,7 +428,7 @@ impl<S, F, B, Res> Service<Request<Body>> for HandleError<S, F>
where where
S: Service<Request<Body>, Response = Response<B>> + Clone, S: Service<Request<Body>, Response = Response<B>> + Clone,
F: FnOnce(S::Error) -> Res + Clone, F: FnOnce(S::Error) -> Res + Clone,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {
@ -459,7 +459,7 @@ impl<Fut, F, B, E, Res> Future for HandleErrorFuture<Fut, F>
where where
Fut: Future<Output = Result<Response<B>, E>>, Fut: Future<Output = Result<Response<B>, E>>,
F: FnOnce(E) -> Res, F: FnOnce(E) -> Res,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {

View file

@ -105,10 +105,12 @@ where
#[derive(Clone)] #[derive(Clone)]
pub struct HandleError<S, F> { pub struct HandleError<S, F> {
inner: S, pub(crate) inner: S,
f: F, pub(crate) f: F,
} }
impl<S, F> crate::routing::RoutingDsl for HandleError<S, F> {}
impl<S, F> HandleError<S, F> { impl<S, F> HandleError<S, F> {
pub(crate) fn new(inner: S, f: F) -> Self { pub(crate) fn new(inner: S, f: F) -> Self {
Self { inner, f } Self { inner, f }
@ -131,7 +133,7 @@ impl<S, F, B, Res> Service<Request<Body>> for HandleError<S, F>
where where
S: Service<Request<Body>, Response = Response<B>> + Clone, S: Service<Request<Body>, Response = Response<B>> + Clone,
F: FnOnce(S::Error) -> Res + Clone, F: FnOnce(S::Error) -> Res + Clone,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {
@ -162,7 +164,7 @@ impl<Fut, F, E, B, Res> Future for HandleErrorFuture<Fut, F>
where where
Fut: Future<Output = Result<Response<B>, E>>, Fut: Future<Output = Result<Response<B>, E>>,
F: FnOnce(E) -> Res, F: FnOnce(E) -> Res,
Res: IntoResponse<Body>, Res: IntoResponse,
B: http_body::Body<Data = Bytes> + Send + Sync + 'static, B: http_body::Body<Data = Bytes> + Send + Sync + 'static,
B::Error: Into<BoxError> + Send + Sync + 'static, B::Error: Into<BoxError> + Send + Sync + 'static,
{ {

View file

@ -8,7 +8,7 @@ use std::{
time::Duration, time::Duration,
}; };
use tower::{make::Shared, BoxError, Service, ServiceBuilder}; use tower::{make::Shared, BoxError, Service, ServiceBuilder};
use tower_http::compression::CompressionLayer; use tower_http::{compression::CompressionLayer, trace::TraceLayer};
#[tokio::test] #[tokio::test]
async fn hello_world() { async fn hello_world() {
@ -444,8 +444,6 @@ async fn middleware_on_single_route() {
#[tokio::test] #[tokio::test]
async fn handling_errors_from_layered_single_routes() { async fn handling_errors_from_layered_single_routes() {
use tower::timeout::TimeoutLayer;
async fn handle(_req: Request<Body>) -> &'static str { async fn handle(_req: Request<Body>) -> &'static str {
tokio::time::sleep(Duration::from_secs(10)).await; tokio::time::sleep(Duration::from_secs(10)).await;
"" ""
@ -454,7 +452,12 @@ async fn handling_errors_from_layered_single_routes() {
let app = route( let app = route(
"/", "/",
get(handle get(handle
.layer(TimeoutLayer::new(Duration::from_millis(100))) .layer(
ServiceBuilder::new()
.timeout(Duration::from_secs(30))
.layer(TraceLayer::new_for_http())
.into_inner(),
)
.handle_error(|_error: BoxError| StatusCode::INTERNAL_SERVER_ERROR)), .handle_error(|_error: BoxError| StatusCode::INTERNAL_SERVER_ERROR)),
); );