More docs

This commit is contained in:
David Pedersen 2021-06-07 16:28:40 +02:00
parent d8b0153939
commit d376e49eb9
6 changed files with 65 additions and 3 deletions

View file

@ -1,3 +1,5 @@
//! Types and traits for extracting data from requests.
use crate::{body::Body, response::IntoResponse}; use crate::{body::Body, response::IntoResponse};
use async_trait::async_trait; use async_trait::async_trait;
use bytes::Bytes; use bytes::Bytes;
@ -275,7 +277,7 @@ macro_rules! impl_parse_url {
}; };
} }
impl_parse_url!(T1, T2, T3, T4, T5, T6); impl_parse_url!(T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15, T16);
fn take_body(req: &mut Request<Body>) -> Result<Body, BodyAlreadyTaken> { fn take_body(req: &mut Request<Body>) -> Result<Body, BodyAlreadyTaken> {
struct BodyAlreadyTakenExt; struct BodyAlreadyTakenExt;
@ -287,3 +289,30 @@ fn take_body(req: &mut Request<Body>) -> Result<Body, BodyAlreadyTaken> {
Ok(body) Ok(body)
} }
} }
macro_rules! impl_from_request_tuple {
() => {};
( $head:ident, $($tail:ident),* $(,)? ) => {
#[allow(non_snake_case)]
#[async_trait]
impl<R, $head, $($tail,)*> FromRequest for ($head, $($tail,)*)
where
R: IntoResponse,
$head: FromRequest<Rejection = R> + Send,
$( $tail: FromRequest<Rejection = R> + Send, )*
{
type Rejection = R;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection> {
let $head = FromRequest::from_request(req).await?;
$( let $tail = FromRequest::from_request(req).await?; )*
Ok(($head, $($tail,)*))
}
}
impl_from_request_tuple!($($tail,)*);
};
}
impl_from_request_tuple!(T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15, T16);

View file

@ -1,3 +1,5 @@
//! Rejection response types.
use super::IntoResponse; use super::IntoResponse;
use crate::body::Body; use crate::body::Body;
@ -5,8 +7,10 @@ macro_rules! define_rejection {
( (
#[status = $status:ident] #[status = $status:ident]
#[body = $body:expr] #[body = $body:expr]
$(#[$m:meta])*
pub struct $name:ident (()); pub struct $name:ident (());
) => { ) => {
$(#[$m])*
#[derive(Debug)] #[derive(Debug)]
pub struct $name(pub(super) ()); pub struct $name(pub(super) ());
@ -22,8 +26,10 @@ macro_rules! define_rejection {
( (
#[status = $status:ident] #[status = $status:ident]
#[body = $body:expr] #[body = $body:expr]
$(#[$m:meta])*
pub struct $name:ident (BoxError); pub struct $name:ident (BoxError);
) => { ) => {
$(#[$m])*
#[derive(Debug)] #[derive(Debug)]
pub struct $name(pub(super) tower::BoxError); pub struct $name(pub(super) tower::BoxError);
@ -50,63 +56,84 @@ macro_rules! define_rejection {
define_rejection! { define_rejection! {
#[status = BAD_REQUEST] #[status = BAD_REQUEST]
#[body = "Query string was invalid or missing"] #[body = "Query string was invalid or missing"]
/// Rejection type for [`Query`](super::Query).
pub struct QueryStringMissing(()); pub struct QueryStringMissing(());
} }
define_rejection! { define_rejection! {
#[status = BAD_REQUEST] #[status = BAD_REQUEST]
#[body = "Failed to parse the response body as JSON"] #[body = "Failed to parse the response body as JSON"]
/// Rejection type for [`Json`](super::Json).
pub struct InvalidJsonBody(BoxError); pub struct InvalidJsonBody(BoxError);
} }
define_rejection! { define_rejection! {
#[status = BAD_REQUEST] #[status = BAD_REQUEST]
#[body = "Expected request with `Content-Type: application/json`"] #[body = "Expected request with `Content-Type: application/json`"]
/// Rejection type for [`Json`](super::Json) used if the `Content-Type`
/// header is missing.
pub struct MissingJsonContentType(()); pub struct MissingJsonContentType(());
} }
define_rejection! { define_rejection! {
#[status = INTERNAL_SERVER_ERROR] #[status = INTERNAL_SERVER_ERROR]
#[body = "Missing request extension"] #[body = "Missing request extension"]
/// Rejection type for [`Extension`](super::Extension) if an expected
/// request extension was not found.
pub struct MissingExtension(()); pub struct MissingExtension(());
} }
define_rejection! { define_rejection! {
#[status = BAD_REQUEST] #[status = BAD_REQUEST]
#[body = "Failed to buffer the request body"] #[body = "Failed to buffer the request body"]
/// Rejection type for extractors that buffer the request body. Used if the
/// request body cannot be buffered due to an error.
pub struct FailedToBufferBody(BoxError); pub struct FailedToBufferBody(BoxError);
} }
define_rejection! { define_rejection! {
#[status = BAD_REQUEST] #[status = BAD_REQUEST]
#[body = "Response body didn't contain valid UTF-8"] #[body = "Response body didn't contain valid UTF-8"]
/// Rejection type used when buffering the request into a [`String`] if the
/// body doesn't contain valid UTF-8.
pub struct InvalidUtf8(BoxError); pub struct InvalidUtf8(BoxError);
} }
define_rejection! { define_rejection! {
#[status = PAYLOAD_TOO_LARGE] #[status = PAYLOAD_TOO_LARGE]
#[body = "Request payload is too large"] #[body = "Request payload is too large"]
/// Rejection type for [`BytesMaxLength`](super::BytesMaxLength) if the
/// request body is too large.
pub struct PayloadTooLarge(()); pub struct PayloadTooLarge(());
} }
define_rejection! { define_rejection! {
#[status = LENGTH_REQUIRED] #[status = LENGTH_REQUIRED]
#[body = "Content length header is required"] #[body = "Content length header is required"]
/// Rejection type for [`BytesMaxLength`](super::BytesMaxLength) if the
/// request is missing the `Content-Length` header or it is invalid.
pub struct LengthRequired(()); pub struct LengthRequired(());
} }
define_rejection! { define_rejection! {
#[status = INTERNAL_SERVER_ERROR] #[status = INTERNAL_SERVER_ERROR]
#[body = "No url params found for matched route. This is a bug in tower-web. Please open an issue"] #[body = "No url params found for matched route. This is a bug in tower-web. Please open an issue"]
/// Rejection type for [`UrlParamsMap`](super::UrlParamsMap) and
/// [`UrlParams`](super::UrlParams) if you try and extract the URL params
/// more than once.
pub struct MissingRouteParams(()); pub struct MissingRouteParams(());
} }
define_rejection! { define_rejection! {
#[status = INTERNAL_SERVER_ERROR] #[status = INTERNAL_SERVER_ERROR]
#[body = "Cannot have two request body extractors for a single handler"] #[body = "Cannot have two request body extractors for a single handler"]
/// Rejection type used if you try and extract the request body more than
/// once.
pub struct BodyAlreadyTaken(()); pub struct BodyAlreadyTaken(());
} }
/// Rejection type for [`UrlParams`](super::UrlParams) if the capture route
/// param didn't have the expected type.
#[derive(Debug)] #[derive(Debug)]
pub struct InvalidUrlParam { pub struct InvalidUrlParam {
type_name: &'static str, type_name: &'static str,

View file

@ -1,3 +1,5 @@
//! Async functions that can be used to handle requests.
use crate::{ use crate::{
body::{Body, BoxBody}, body::{Body, BoxBody},
extract::FromRequest, extract::FromRequest,

View file

@ -7,7 +7,7 @@
//! handle(Request) -> Response`. //! handle(Request) -> Response`.
//! - Solid foundation. tower-web is built on top of tower and makes it easy to //! - Solid foundation. tower-web is built on top of tower and makes it easy to
//! plug in any middleware from the [tower] and [tower-http] ecosystem. //! plug in any middleware from the [tower] and [tower-http] ecosystem.
//! - Focus on routing, extracing data from requests, and generating responses. //! - Focus on routing, extracting data from requests, and generating responses.
//! tower middleware can handle the rest. //! tower middleware can handle the rest.
//! - Macro free core. Macro frameworks have their place but tower-web focuses //! - Macro free core. Macro frameworks have their place but tower-web focuses
//! on providing a core that is macro free. //! on providing a core that is macro free.

View file

@ -1,3 +1,5 @@
//! Types and traits for generating responses.
use crate::Body; use crate::Body;
use bytes::Bytes; use bytes::Bytes;
use http::{header, HeaderMap, HeaderValue, Response, StatusCode}; use http::{header, HeaderMap, HeaderValue, Response, StatusCode};

View file

@ -1,3 +1,5 @@
//! Routing between [`Service`]s.
use crate::{body::BoxBody, response::IntoResponse, ResultExt}; use crate::{body::BoxBody, response::IntoResponse, ResultExt};
use bytes::Bytes; use bytes::Bytes;
use futures_util::{future, ready}; use futures_util::{future, ready};
@ -465,7 +467,7 @@ where
} }
} }
/// Nest a group of routes (or [`Service`]) at some path. /// Nest a group of routes (or a [`Service`]) at some path.
/// ///
/// This allows you to break your application into smaller pieces and compose /// This allows you to break your application into smaller pieces and compose
/// them together. This will strip the matching prefix from the URL so the /// them together. This will strip the matching prefix from the URL so the