Struct axum::routing::method_routing::MethodRouter
source · pub struct MethodRouter<S = (), B = Body, E = Infallible> { /* private fields */ }
Expand description
A Service
that accepts requests based on a MethodFilter
and
allows chaining additional handlers and services.
§When does MethodRouter
implement Service
?
Whether or not MethodRouter
implements Service
depends on the state type it requires.
use tower::Service;
use axum::{routing::get, extract::State, body::Body, http::Request};
// this `MethodRouter` doesn't require any state, i.e. the state is `()`,
let method_router = get(|| async {});
// and thus it implements `Service`
assert_service(method_router);
// this requires a `String` and doesn't implement `Service`
let method_router = get(|_: State<String>| async {});
// until you provide the `String` with `.with_state(...)`
let method_router_with_state = method_router.with_state(String::new());
// and then it implements `Service`
assert_service(method_router_with_state);
// helper to check that a value implements `Service`
fn assert_service<S>(service: S)
where
S: Service<Request<Body>>,
{}
Implementations§
source§impl<S, B> MethodRouter<S, B, Infallible>
impl<S, B> MethodRouter<S, B, Infallible>
sourcepub fn on<H, T>(self, filter: MethodFilter, handler: H) -> Self
pub fn on<H, T>(self, filter: MethodFilter, handler: H) -> Self
Chain an additional handler that will accept requests matching the given
MethodFilter
.
§Example
use axum::{
routing::get,
Router,
routing::MethodFilter
};
async fn handler() {}
async fn other_handler() {}
// Requests to `GET /` will go to `handler` and `DELETE /` will go to
// `other_handler`
let app = Router::new().route("/", get(handler).on(MethodFilter::DELETE, other_handler));
sourcepub fn delete<H, T>(self, handler: H) -> Self
pub fn delete<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept DELETE
requests.
See MethodRouter::get
for an example.
sourcepub fn get<H, T>(self, handler: H) -> Self
pub fn get<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept GET
requests.
§Example
use axum::{routing::post, Router};
async fn handler() {}
async fn other_handler() {}
// Requests to `POST /` will go to `handler` and `GET /` will go to
// `other_handler`.
let app = Router::new().route("/", post(handler).get(other_handler));
Note that get
routes will also be called for HEAD
requests but will have
the response body removed. Make sure to add explicit HEAD
routes
afterwards.
sourcepub fn head<H, T>(self, handler: H) -> Self
pub fn head<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept HEAD
requests.
See MethodRouter::get
for an example.
sourcepub fn options<H, T>(self, handler: H) -> Self
pub fn options<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept OPTIONS
requests.
See MethodRouter::get
for an example.
sourcepub fn patch<H, T>(self, handler: H) -> Self
pub fn patch<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept PATCH
requests.
See MethodRouter::get
for an example.
sourcepub fn post<H, T>(self, handler: H) -> Self
pub fn post<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept POST
requests.
See MethodRouter::get
for an example.
sourcepub fn put<H, T>(self, handler: H) -> Self
pub fn put<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept PUT
requests.
See MethodRouter::get
for an example.
sourcepub fn trace<H, T>(self, handler: H) -> Self
pub fn trace<H, T>(self, handler: H) -> Self
Chain an additional handler that will only accept TRACE
requests.
See MethodRouter::get
for an example.
source§impl<B> MethodRouter<(), B, Infallible>
impl<B> MethodRouter<(), B, Infallible>
sourcepub fn into_make_service(self) -> IntoMakeService<Self>
pub fn into_make_service(self) -> IntoMakeService<Self>
Convert the handler into a MakeService
.
This allows you to serve a single handler if you don’t need any routing:
use axum::{
Server,
handler::Handler,
http::{Uri, Method},
response::IntoResponse,
routing::get,
};
use std::net::SocketAddr;
async fn handler(method: Method, uri: Uri, body: String) -> String {
format!("received `{} {}` with body `{:?}`", method, uri, body)
}
let router = get(handler).post(handler);
Server::bind(&SocketAddr::from(([127, 0, 0, 1], 3000)))
.serve(router.into_make_service())
.await?;
source§impl<S, B, E> MethodRouter<S, B, E>
impl<S, B, E> MethodRouter<S, B, E>
sourcepub fn new() -> Self
pub fn new() -> Self
Create a default MethodRouter
that will respond with 405 Method Not Allowed
to all
requests.
sourcepub fn with_state<S2>(self, state: S) -> MethodRouter<S2, B, E>
pub fn with_state<S2>(self, state: S) -> MethodRouter<S2, B, E>
Provide the state for the router.
sourcepub fn on_service<T>(self, filter: MethodFilter, svc: T) -> Self
pub fn on_service<T>(self, filter: MethodFilter, svc: T) -> Self
Chain an additional service that will accept requests matching the given
MethodFilter
.
§Example
use axum::{
http::Request,
Router,
routing::{MethodFilter, on_service},
};
use http::Response;
use std::convert::Infallible;
use hyper::Body;
let service = tower::service_fn(|request: Request<Body>| async {
Ok::<_, Infallible>(Response::new(Body::empty()))
});
// Requests to `DELETE /` will go to `service`
let app = Router::new().route("/", on_service(MethodFilter::DELETE, service));
sourcepub fn delete_service<T>(self, svc: T) -> Self
pub fn delete_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept DELETE
requests.
See MethodRouter::get_service
for an example.
sourcepub fn get_service<T>(self, svc: T) -> Self
pub fn get_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept GET
requests.
§Example
use axum::{
http::Request,
Router,
routing::post_service,
};
use http::Response;
use std::convert::Infallible;
use hyper::Body;
let service = tower::service_fn(|request: Request<Body>| async {
Ok::<_, Infallible>(Response::new(Body::empty()))
});
let other_service = tower::service_fn(|request: Request<Body>| async {
Ok::<_, Infallible>(Response::new(Body::empty()))
});
// Requests to `POST /` will go to `service` and `GET /` will go to
// `other_service`.
let app = Router::new().route("/", post_service(service).get_service(other_service));
Note that get
routes will also be called for HEAD
requests but will have
the response body removed. Make sure to add explicit HEAD
routes
afterwards.
sourcepub fn head_service<T>(self, svc: T) -> Self
pub fn head_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept HEAD
requests.
See MethodRouter::get_service
for an example.
sourcepub fn options_service<T>(self, svc: T) -> Self
pub fn options_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept OPTIONS
requests.
See MethodRouter::get_service
for an example.
sourcepub fn patch_service<T>(self, svc: T) -> Self
pub fn patch_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept PATCH
requests.
See MethodRouter::get_service
for an example.
sourcepub fn post_service<T>(self, svc: T) -> Self
pub fn post_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept POST
requests.
See MethodRouter::get_service
for an example.
sourcepub fn put_service<T>(self, svc: T) -> Self
pub fn put_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept PUT
requests.
See MethodRouter::get_service
for an example.
sourcepub fn trace_service<T>(self, svc: T) -> Self
pub fn trace_service<T>(self, svc: T) -> Self
Chain an additional service that will only accept TRACE
requests.
See MethodRouter::get_service
for an example.
sourcepub fn fallback_service<T>(self, svc: T) -> Self
pub fn fallback_service<T>(self, svc: T) -> Self
Add a fallback service to the router.
This service will be called if no routes matches the incoming request.
use axum::{
Router,
routing::get,
handler::Handler,
response::IntoResponse,
http::{StatusCode, Method, Uri},
};
let handler = get(|| async {}).fallback(fallback);
let app = Router::new().route("/", handler);
async fn fallback(method: Method, uri: Uri) -> (StatusCode, String) {
(StatusCode::NOT_FOUND, format!("`{}` not allowed for {}", method, uri))
}
§When used with MethodRouter::merge
Two routers that both have a fallback cannot be merged. Doing so results in a panic:
use axum::{
routing::{get, post},
handler::Handler,
response::IntoResponse,
http::{StatusCode, Uri},
};
let one = get(|| async {}).fallback(fallback_one);
let two = post(|| async {}).fallback(fallback_two);
let method_route = one.merge(two);
async fn fallback_one() -> impl IntoResponse { /* ... */ }
async fn fallback_two() -> impl IntoResponse { /* ... */ }
§Setting the Allow
header
By default MethodRouter
will set the Allow
header when returning 405 Method Not Allowed
. This is also done when the fallback is used unless the response
generated by the fallback already sets the Allow
header.
This means if you use fallback
to accept additional methods, you should make
sure you set the Allow
header correctly.
sourcepub fn layer<L, NewReqBody, NewError>(
self,
layer: L,
) -> MethodRouter<S, NewReqBody, NewError>where
L: Layer<Route<B, E>> + Clone + Send + 'static,
L::Service: Service<Request<NewReqBody>> + Clone + Send + 'static,
<L::Service as Service<Request<NewReqBody>>>::Response: IntoResponse + 'static,
<L::Service as Service<Request<NewReqBody>>>::Error: Into<NewError> + 'static,
<L::Service as Service<Request<NewReqBody>>>::Future: Send + 'static,
E: 'static,
S: 'static,
NewReqBody: HttpBody + 'static,
NewError: 'static,
pub fn layer<L, NewReqBody, NewError>(
self,
layer: L,
) -> MethodRouter<S, NewReqBody, NewError>where
L: Layer<Route<B, E>> + Clone + Send + 'static,
L::Service: Service<Request<NewReqBody>> + Clone + Send + 'static,
<L::Service as Service<Request<NewReqBody>>>::Response: IntoResponse + 'static,
<L::Service as Service<Request<NewReqBody>>>::Error: Into<NewError> + 'static,
<L::Service as Service<Request<NewReqBody>>>::Future: Send + 'static,
E: 'static,
S: 'static,
NewReqBody: HttpBody + 'static,
NewError: 'static,
Apply a tower::Layer
to all routes in the router.
This can be used to add additional processing to a request for a group of routes.
Note that the middleware is only applied to existing routes. So you have to
first add your routes (and / or fallback) and then call layer
afterwards. Additional
routes added after layer
is called will not have the middleware added.
Works similarly to Router::layer
. See that method for
more details.
§Example
use axum::{routing::get, Router};
use tower::limit::ConcurrencyLimitLayer;
async fn hander() {}
let app = Router::new().route(
"/",
// All requests to `GET /` will be sent through `ConcurrencyLimitLayer`
get(hander).layer(ConcurrencyLimitLayer::new(64)),
);
sourcepub fn route_layer<L>(self, layer: L) -> MethodRouter<S, B, E>
pub fn route_layer<L>(self, layer: L) -> MethodRouter<S, B, E>
Apply a tower::Layer
to the router that will only run if the request matches
a route.
Note that the middleware is only applied to existing routes. So you have to
first add your routes (and / or fallback) and then call layer
afterwards. Additional
routes added after layer
is called will not have the middleware added.
This works similarly to MethodRouter::layer
except the middleware will only run if
the request matches a route. This is useful for middleware that return early
(such as authorization) which might otherwise convert a 405 Method Not Allowed
into a
401 Unauthorized
.
§Example
use axum::{
routing::get,
Router,
};
use tower_http::validate_request::ValidateRequestHeaderLayer;
let app = Router::new().route(
"/foo",
get(|| async {})
.route_layer(ValidateRequestHeaderLayer::bearer("password"))
);
// `GET /foo` with a valid token will receive `200 OK`
// `GET /foo` with a invalid token will receive `401 Unauthorized`
// `POST /FOO` with a invalid token will receive `405 Method Not Allowed`
sourcepub fn merge(self, other: MethodRouter<S, B, E>) -> Self
pub fn merge(self, other: MethodRouter<S, B, E>) -> Self
Merge two routers into one.
This is useful for breaking routers into smaller pieces and combining them into one.
use axum::{
routing::{get, post},
Router,
};
let get = get(|| async {});
let post = post(|| async {});
let merged = get.merge(post);
let app = Router::new().route("/", merged);
// Our app now accepts
// - GET /
// - POST /
sourcepub fn handle_error<F, T>(self, f: F) -> MethodRouter<S, B, Infallible>where
F: Clone + Send + Sync + 'static,
HandleError<Route<B, E>, F, T>: Service<Request<B>, Error = Infallible>,
<HandleError<Route<B, E>, F, T> as Service<Request<B>>>::Future: Send,
<HandleError<Route<B, E>, F, T> as Service<Request<B>>>::Response: IntoResponse + Send,
T: 'static,
E: 'static,
B: 'static,
S: 'static,
pub fn handle_error<F, T>(self, f: F) -> MethodRouter<S, B, Infallible>where
F: Clone + Send + Sync + 'static,
HandleError<Route<B, E>, F, T>: Service<Request<B>, Error = Infallible>,
<HandleError<Route<B, E>, F, T> as Service<Request<B>>>::Future: Send,
<HandleError<Route<B, E>, F, T> as Service<Request<B>>>::Response: IntoResponse + Send,
T: 'static,
E: 'static,
B: 'static,
S: 'static,
Apply a HandleErrorLayer
.
This is a convenience method for doing self.layer(HandleErrorLayer::new(f))
.
Trait Implementations§
source§impl<S, B, E> Clone for MethodRouter<S, B, E>
impl<S, B, E> Clone for MethodRouter<S, B, E>
source§impl<S, B, E> Debug for MethodRouter<S, B, E>
impl<S, B, E> Debug for MethodRouter<S, B, E>
source§impl<S, B, E> Default for MethodRouter<S, B, E>
impl<S, B, E> Default for MethodRouter<S, B, E>
source§impl<S, B> Handler<(), S, B> for MethodRouter<S, B>
impl<S, B> Handler<(), S, B> for MethodRouter<S, B>
source§type Future = InfallibleRouteFuture<B>
type Future = InfallibleRouteFuture<B>
source§fn call(self, req: Request<B>, state: S) -> Self::Future
fn call(self, req: Request<B>, state: S) -> Self::Future
source§fn layer<L, NewReqBody>(self, layer: L) -> Layered<L, Self, T, S, B, NewReqBody>
fn layer<L, NewReqBody>(self, layer: L) -> Layered<L, Self, T, S, B, NewReqBody>
tower::Layer
to the handler. Read moresource§fn with_state(self, state: S) -> HandlerService<Self, T, S, B>
fn with_state(self, state: S) -> HandlerService<Self, T, S, B>
Service
by providing the stateAuto Trait Implementations§
impl<S, B, E> Freeze for MethodRouter<S, B, E>
impl<S = (), B = Body, E = Infallible> !RefUnwindSafe for MethodRouter<S, B, E>
impl<S, B, E> Send for MethodRouter<S, B, E>
impl<S = (), B = Body, E = Infallible> !Sync for MethodRouter<S, B, E>
impl<S, B, E> Unpin for MethodRouter<S, B, E>
impl<S = (), B = Body, E = Infallible> !UnwindSafe for MethodRouter<S, B, E>
Blanket Implementations§
source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
source§impl<T> CloneToUninit for Twhere
T: Clone,
impl<T> CloneToUninit for Twhere
T: Clone,
source§unsafe fn clone_to_uninit(&self, dst: *mut T)
unsafe fn clone_to_uninit(&self, dst: *mut T)
clone_to_uninit
)source§impl<H, T, B> HandlerWithoutStateExt<T, B> for H
impl<H, T, B> HandlerWithoutStateExt<T, B> for H
source§fn into_service(self) -> HandlerService<H, T, (), B>
fn into_service(self) -> HandlerService<H, T, (), B>
Service
and no state.source§fn into_make_service(self) -> IntoMakeService<HandlerService<H, T, (), B>>
fn into_make_service(self) -> IntoMakeService<HandlerService<H, T, (), B>>
MakeService
and no state. Read moresource§impl<T> Instrument for T
impl<T> Instrument for T
source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
source§impl<S, R> ServiceExt<R> for Swhere
S: Service<R>,
impl<S, R> ServiceExt<R> for Swhere
S: Service<R>,
source§fn into_make_service(self) -> IntoMakeService<S>
fn into_make_service(self) -> IntoMakeService<S>
MakeService
, that is a Service
whose
response is another service. Read moresource§impl<T, Request> ServiceExt<Request> for T
impl<T, Request> ServiceExt<Request> for T
source§fn ready(&mut self) -> Ready<'_, Self, Request>where
Self: Sized,
fn ready(&mut self) -> Ready<'_, Self, Request>where
Self: Sized,
source§fn ready_and(&mut self) -> Ready<'_, Self, Request>where
Self: Sized,
fn ready_and(&mut self) -> Ready<'_, Self, Request>where
Self: Sized,
ServiceExt::ready
method insteadsource§fn ready_oneshot(self) -> ReadyOneshot<Self, Request>where
Self: Sized,
fn ready_oneshot(self) -> ReadyOneshot<Self, Request>where
Self: Sized,
source§fn oneshot(self, req: Request) -> Oneshot<Self, Request>where
Self: Sized,
fn oneshot(self, req: Request) -> Oneshot<Self, Request>where
Self: Sized,
Service
, calling with the providing request once it is ready.source§fn and_then<F>(self, f: F) -> AndThen<Self, F>
fn and_then<F>(self, f: F) -> AndThen<Self, F>
poll_ready
method. Read moresource§fn map_response<F, Response>(self, f: F) -> MapResponse<Self, F>
fn map_response<F, Response>(self, f: F) -> MapResponse<Self, F>
poll_ready
method. Read moresource§fn map_err<F, Error>(self, f: F) -> MapErr<Self, F>
fn map_err<F, Error>(self, f: F) -> MapErr<Self, F>
poll_ready
method. Read moresource§fn map_result<F, Response, Error>(self, f: F) -> MapResult<Self, F>
fn map_result<F, Response, Error>(self, f: F) -> MapResult<Self, F>
Result<Self::Response, Self::Error>
)
to a different value, regardless of whether the future succeeds or
fails. Read more