Update matchit and fix nesting inconsistencies (#1086)

* Break `Router::nest` into `nest` and `nest_service` methods

* fix doc tests

* update docs

* fix

* Only accept `Router` in `Resource::{nest, nest_collection}`

* update changelog

* fix docs

* fix `MatchedPath` with `Router`s nested with `nest_service`

* Apply suggestions from code review

Co-authored-by: Jonas Platte <jplatte+git@posteo.de>

* adjust docs for fallbacks

* Always nest services as opaque

* fix old docs reference

* more tests for `MatchedPath` with nested handlers

* minor clean up

* use identifier captures in format strings

* Apply suggestions from code review

Co-authored-by: Jonas Platte <jplatte+git@posteo.de>

* fix test

Co-authored-by: Jonas Platte <jplatte+git@posteo.de>
This commit is contained in:
David Pedersen 2022-08-11 12:17:08 +02:00 committed by GitHub
parent 0090d007c0
commit 50a4be999d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
12 changed files with 233 additions and 392 deletions

View file

@ -9,12 +9,16 @@ and this project adheres to [Semantic Versioning].
- **added:** Add `RouterExt::route_with_tsr` for adding routes with an - **added:** Add `RouterExt::route_with_tsr` for adding routes with an
additional "trailing slash redirect" route ([#1119]) additional "trailing slash redirect" route ([#1119])
- **breaking:** `Resource::nest` and `Resource::nest_collection` has been
removed. You can instead convert the `Resource` into a `Router` and
add additional routes as necessary ([#1086])
- **changed:** For methods that accept some `S: Service`, the bounds have been - **changed:** For methods that accept some `S: Service`, the bounds have been
relaxed so the response type must implement `IntoResponse` rather than being a relaxed so the response type must implement `IntoResponse` rather than being a
literal `Response` literal `Response`
- **added:** Support chaining handlers with `HandlerCallWithExtractors::or` ([#1170]) - **added:** Support chaining handlers with `HandlerCallWithExtractors::or` ([#1170])
- **change:** axum-extra's MSRV is now 1.60 ([#1239]) - **change:** axum-extra's MSRV is now 1.60 ([#1239])
[#1086]: https://github.com/tokio-rs/axum/pull/1086
[#1119]: https://github.com/tokio-rs/axum/pull/1119 [#1119]: https://github.com/tokio-rs/axum/pull/1119
[#1170]: https://github.com/tokio-rs/axum/pull/1170 [#1170]: https://github.com/tokio-rs/axum/pull/1170
[#1239]: https://github.com/tokio-rs/axum/pull/1239 [#1239]: https://github.com/tokio-rs/axum/pull/1239

View file

@ -31,18 +31,7 @@ use tower_service::Service;
/// // `PUT or PATCH /users/:users_id` /// // `PUT or PATCH /users/:users_id`
/// .update(|Path(user_id): Path<u64>| async {}) /// .update(|Path(user_id): Path<u64>| async {})
/// // `DELETE /users/:users_id` /// // `DELETE /users/:users_id`
/// .destroy(|Path(user_id): Path<u64>| async {}) /// .destroy(|Path(user_id): Path<u64>| async {});
/// // Nest another router at the "member level"
/// // This defines a route for `GET /users/:users_id/tweets`
/// .nest(Router::new().route(
/// "/tweets",
/// get(|Path(user_id): Path<u64>| async {}),
/// ))
/// // Nest another router at the "collection level"
/// // This defines a route for `GET /users/featured`
/// .nest_collection(
/// Router::new().route("/featured", get(|| async {})),
/// );
/// ///
/// let app = Router::new().merge(users); /// let app = Router::new().merge(users);
/// # let _: Router<axum::body::Body> = app; /// # let _: Router<axum::body::Body> = app;
@ -136,34 +125,6 @@ where
self.route(&path, delete(handler)) self.route(&path, delete(handler))
} }
/// Nest another route at the "member level".
///
/// The routes will be nested at `/{resource_name}/:{resource_name}_id`.
pub fn nest<T>(mut self, svc: T) -> Self
where
T: Service<Request<B>, Error = Infallible> + Clone + Send + 'static,
T::Response: IntoResponse,
T::Future: Send + 'static,
{
let path = self.show_update_destroy_path();
self.router = self.router.nest(&path, svc);
self
}
/// Nest another route at the "collection level".
///
/// The routes will be nested at `/{resource_name}`.
pub fn nest_collection<T>(mut self, svc: T) -> Self
where
T: Service<Request<B>, Error = Infallible> + Clone + Send + 'static,
T::Response: IntoResponse,
T::Future: Send + 'static,
{
let path = self.index_create_path();
self.router = self.router.nest(&path, svc);
self
}
fn index_create_path(&self) -> String { fn index_create_path(&self) -> String {
format!("/{}", self.name) format!("/{}", self.name)
} }
@ -214,14 +175,7 @@ mod tests {
.show(|Path(id): Path<u64>| async move { format!("users#show id={}", id) }) .show(|Path(id): Path<u64>| async move { format!("users#show id={}", id) })
.edit(|Path(id): Path<u64>| async move { format!("users#edit id={}", id) }) .edit(|Path(id): Path<u64>| async move { format!("users#edit id={}", id) })
.update(|Path(id): Path<u64>| async move { format!("users#update id={}", id) }) .update(|Path(id): Path<u64>| async move { format!("users#update id={}", id) })
.destroy(|Path(id): Path<u64>| async move { format!("users#destroy id={}", id) }) .destroy(|Path(id): Path<u64>| async move { format!("users#destroy id={}", id) });
.nest(Router::new().route(
"/tweets",
get(|Path(id): Path<u64>| async move { format!("users#tweets id={}", id) }),
))
.nest_collection(
Router::new().route("/featured", get(|| async move { "users#featured" })),
);
let mut app = Router::new().merge(users); let mut app = Router::new().merge(users);
@ -264,16 +218,6 @@ mod tests {
call_route(&mut app, Method::DELETE, "/users/1").await, call_route(&mut app, Method::DELETE, "/users/1").await,
"users#destroy id=1" "users#destroy id=1"
); );
assert_eq!(
call_route(&mut app, Method::GET, "/users/1/tweets").await,
"users#tweets id=1"
);
assert_eq!(
call_route(&mut app, Method::GET, "/users/featured").await,
"users#featured"
);
} }
async fn call_route(app: &mut Router, method: Method, uri: &str) -> String { async fn call_route(app: &mut Router, method: Method, uri: &str) -> String {

View file

@ -12,6 +12,15 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
Use `axum::middleware::from_extractor` instead ([#1077]) Use `axum::middleware::from_extractor` instead ([#1077])
- **breaking:** Allow `Error: Into<Infallible>` for `Route::{layer, route_layer}` ([#924]) - **breaking:** Allow `Error: Into<Infallible>` for `Route::{layer, route_layer}` ([#924])
- **breaking:** `MethodRouter` now panics on overlapping routes ([#1102]) - **breaking:** `MethodRouter` now panics on overlapping routes ([#1102])
- **breaking:** Nested `Router`s will no longer delegate to the outer `Router`'s
fallback. Instead you must explicitly set a fallback on the inner `Router` ([#1086])
- **breaking:** The request `/foo/` no longer matches `/foo/*rest`. If you want
to match `/foo/` you have to add a route specifically for that ([#1086])
- **breaking:** Path params for wildcard routes no longer include the prefix
`/`. e.g. `/foo.js` will match `/*filepath` with a value of `foo.js`, _not_
`/foo.js` ([#1086])
- **fixed:** Routes like `/foo` and `/*rest` are no longer considered
overlapping. `/foo` will take priority ([#1086])
- **breaking:** Remove trailing slash redirects. Previously if you added a route - **breaking:** Remove trailing slash redirects. Previously if you added a route
for `/foo`, axum would redirect calls to `/foo/` to `/foo` (or vice versa for for `/foo`, axum would redirect calls to `/foo/` to `/foo` (or vice versa for
`/foo/`). That is no longer supported and such requests will now be sent to `/foo/`). That is no longer supported and such requests will now be sent to
@ -36,6 +45,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
[#1171]: https://github.com/tokio-rs/axum/pull/1171 [#1171]: https://github.com/tokio-rs/axum/pull/1171
[#1077]: https://github.com/tokio-rs/axum/pull/1077 [#1077]: https://github.com/tokio-rs/axum/pull/1077
[#1086]: https://github.com/tokio-rs/axum/pull/1086
[#1088]: https://github.com/tokio-rs/axum/pull/1088 [#1088]: https://github.com/tokio-rs/axum/pull/1088
[#1102]: https://github.com/tokio-rs/axum/pull/1102 [#1102]: https://github.com/tokio-rs/axum/pull/1102
[#1119]: https://github.com/tokio-rs/axum/pull/1119 [#1119]: https://github.com/tokio-rs/axum/pull/1119

View file

@ -38,7 +38,7 @@ http = "0.2.5"
http-body = "0.4.4" http-body = "0.4.4"
hyper = { version = "0.14.14", features = ["server", "tcp", "stream"] } hyper = { version = "0.14.14", features = ["server", "tcp", "stream"] }
itoa = "1.0.1" itoa = "1.0.1"
matchit = "0.5.0" matchit = "0.6"
memchr = "2.4.1" memchr = "2.4.1"
mime = "0.3.16" mime = "0.3.16"
percent-encoding = "2.1" percent-encoding = "2.1"

View file

@ -1,4 +1,4 @@
Nest a group of routes (or a [`Service`]) at some path. Nest 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. them together.
@ -64,36 +64,6 @@ let app = Router::new().nest("/:version/api", users_api);
# }; # };
``` ```
# Nesting services
`nest` also accepts any [`Service`]. This can for example be used with
[`tower_http::services::ServeDir`] to serve static files from a directory:
```rust
use axum::{
Router,
routing::get_service,
http::StatusCode,
error_handling::HandleErrorLayer,
};
use std::{io, convert::Infallible};
use tower_http::services::ServeDir;
// Serves files inside the `public` directory at `GET /public/*`
let serve_dir_service = get_service(ServeDir::new("public"))
.handle_error(|error: io::Error| async move {
(
StatusCode::INTERNAL_SERVER_ERROR,
format!("Unhandled internal error: {}", error),
)
});
let app = Router::new().nest("/public", serve_dir_service);
# async {
# axum::Server::bind(&"".parse().unwrap()).serve(app.into_make_service()).await.unwrap();
# };
```
# Differences to wildcard routes # Differences to wildcard routes
Nested routes are similar to wildcard routes. The difference is that Nested routes are similar to wildcard routes. The difference is that
@ -103,18 +73,73 @@ the prefix stripped:
```rust ```rust
use axum::{routing::get, http::Uri, Router}; use axum::{routing::get, http::Uri, Router};
let nested_router = Router::new()
.route("/", get(|uri: Uri| async {
// `uri` will _not_ contain `/bar`
}));
let app = Router::new() let app = Router::new()
.route("/foo/*rest", get(|uri: Uri| async { .route("/foo/*rest", get(|uri: Uri| async {
// `uri` will contain `/foo` // `uri` will contain `/foo`
})) }))
.nest("/bar", get(|uri: Uri| async { .nest("/bar", nested_router);
// `uri` will _not_ contain `/bar`
}));
# async { # async {
# axum::Server::bind(&"".parse().unwrap()).serve(app.into_make_service()).await.unwrap(); # axum::Server::bind(&"".parse().unwrap()).serve(app.into_make_service()).await.unwrap();
# }; # };
``` ```
# Fallbacks
When nesting a router, if a request matches the prefix but the nested router doesn't have a matching
route, the outer fallback will _not_ be called:
```rust
use axum::{routing::get, http::StatusCode, handler::Handler, Router};
async fn fallback() -> (StatusCode, &'static str) {
(StatusCode::NOT_FOUND, "Not Found")
}
let api_routes = Router::new().nest("/users", get(|| async {}));
let app = Router::new()
.nest("/api", api_routes)
.fallback(fallback.into_service());
# let _: Router = app;
```
Here requests like `GET /api/not-found` will go into `api_routes` and then to
the fallback of `api_routes` which will return an empty `404 Not Found`
response. The outer fallback declared on `app` will _not_ be called.
Think of nested services as swallowing requests that matches the prefix and
not falling back to outer router even if they don't have a matching route.
You can still add separate fallbacks to nested routers:
```rust
use axum::{routing::get, http::StatusCode, handler::Handler, Json, Router};
use serde_json::{json, Value};
async fn fallback() -> (StatusCode, &'static str) {
(StatusCode::NOT_FOUND, "Not Found")
}
async fn api_fallback() -> (StatusCode, Json<Value>) {
(StatusCode::NOT_FOUND, Json(json!({ "error": "Not Found" })))
}
let api_routes = Router::new()
.nest("/users", get(|| async {}))
// add dedicated fallback for requests starting with `/api`
.fallback(api_fallback.into_service());
let app = Router::new()
.nest("/api", api_routes)
.fallback(fallback.into_service());
# let _: Router = app;
```
# Panics # Panics
- If the route overlaps with another route. See [`Router::route`] - If the route overlaps with another route. See [`Router::route`]
@ -125,3 +150,4 @@ for more details.
`Router` only allows a single fallback. `Router` only allows a single fallback.
[`OriginalUri`]: crate::extract::OriginalUri [`OriginalUri`]: crate::extract::OriginalUri
[fallbacks]: Router::fallback

View file

@ -51,6 +51,8 @@ Examples:
- `/:id/:repo/*tree` - `/:id/:repo/*tree`
Wildcard captures can also be extracted using [`Path`](crate::extract::Path). Wildcard captures can also be extracted using [`Path`](crate::extract::Path).
Note that the leading slash is not included, i.e. for the route `/foo/*rest` and
the path `/foo/bar/baz` the value of `rest` will be `bar/baz`.
# Accepting multiple methods # Accepting multiple methods
@ -184,22 +186,6 @@ let app = Router::new()
The static route `/foo` and the dynamic route `/:key` are not considered to The static route `/foo` and the dynamic route `/:key` are not considered to
overlap and `/foo` will take precedence. overlap and `/foo` will take precedence.
Take care when using [`Router::nest`] as it behaves like a wildcard route.
Therefore this setup panics:
```rust,should_panic
use axum::{routing::get, Router};
let app = Router::new()
// this is similar to `/api/*`
.nest("/api", get(|| async {}))
// which overlaps with this route
.route("/api/users", get(|| async {}));
# async {
# axum::Server::bind(&"".parse().unwrap()).serve(app.into_make_service()).await.unwrap();
# };
```
Also panics if `path` is empty. Also panics if `path` is empty.
## Nesting ## Nesting

View file

@ -85,8 +85,9 @@ where
mod tests { mod tests {
use super::*; use super::*;
use crate::{extract::Extension, handler::Handler, routing::get, test_helpers::*, Router}; use crate::{extract::Extension, handler::Handler, routing::get, test_helpers::*, Router};
use http::Request; use http::{Request, StatusCode};
use std::task::{Context, Poll}; use std::task::{Context, Poll};
use tower::layer::layer_fn;
use tower_service::Service; use tower_service::Service;
#[derive(Clone)] #[derive(Clone)]
@ -147,25 +148,37 @@ mod tests {
.nest("/api", api) .nest("/api", api)
.nest( .nest(
"/public", "/public",
Router::new().route("/assets/*path", get(handler)), Router::new()
.route("/assets/*path", get(handler))
// have to set the middleware here since otherwise the
// matched path is just `/public/*` since we're nesting
// this router
.layer(layer_fn(SetMatchedPathExtension)),
) )
.nest("/foo", handler.into_service()) .nest("/foo", handler.into_service())
.layer(tower::layer::layer_fn(SetMatchedPathExtension)); .layer(layer_fn(SetMatchedPathExtension));
let client = TestClient::new(app); let client = TestClient::new(app);
let res = client.get("/foo").send().await;
assert_eq!(res.text().await, "/:key");
let res = client.get("/api/users/123").send().await; let res = client.get("/api/users/123").send().await;
assert_eq!(res.text().await, "/api/users/:id"); assert_eq!(res.text().await, "/api/users/:id");
// the router nested at `/public` doesn't handle `/`
let res = client.get("/public").send().await;
assert_eq!(res.status(), StatusCode::NOT_FOUND);
let res = client.get("/public/assets/css/style.css").send().await; let res = client.get("/public/assets/css/style.css").send().await;
assert_eq!( assert_eq!(
res.text().await, res.text().await,
"extractor = /public/assets/*path, middleware = /public/assets/*path" "extractor = /public/assets/*path, middleware = /public/assets/*path"
); );
let res = client.get("/foo").send().await;
assert_eq!(res.text().await, "extractor = /foo, middleware = /foo");
let res = client.get("/foo/").send().await;
assert_eq!(res.text().await, "extractor = /foo/, middleware = /foo/");
let res = client.get("/foo/bar/baz").send().await; let res = client.get("/foo/bar/baz").send().await;
assert_eq!( assert_eq!(
res.text().await, res.text().await,
@ -176,4 +189,20 @@ mod tests {
), ),
); );
} }
#[tokio::test]
async fn nested_opaque_routers_append_to_matched_path() {
let app = Router::new().nest(
"/:a",
Router::new().route(
"/:b",
get(|path: MatchedPath| async move { path.as_str().to_owned() }),
),
);
let client = TestClient::new(app);
let res = client.get("/foo/bar").send().await;
assert_eq!(res.text().await, "/:a/:b");
}
} }

View file

@ -499,10 +499,10 @@ mod tests {
let client = TestClient::new(app); let client = TestClient::new(app);
let res = client.get("/foo/bar/baz").send().await; let res = client.get("/foo/bar/baz").send().await;
assert_eq!(res.text().await, "/bar/baz"); assert_eq!(res.text().await, "bar/baz");
let res = client.get("/bar/baz/qux").send().await; let res = client.get("/bar/baz/qux").send().await;
assert_eq!(res.text().await, "/baz/qux"); assert_eq!(res.text().await, "baz/qux");
} }
#[tokio::test] #[tokio::test]

View file

@ -5,21 +5,19 @@ use crate::{
body::{Body, HttpBody}, body::{Body, HttpBody},
extract::connect_info::IntoMakeServiceWithConnectInfo, extract::connect_info::IntoMakeServiceWithConnectInfo,
response::Response, response::Response,
routing::strip_prefix::StripPrefix,
util::try_downcast, util::try_downcast,
}; };
use axum_core::response::IntoResponse; use axum_core::response::IntoResponse;
use http::Request; use http::Request;
use matchit::MatchError; use matchit::MatchError;
use std::{ use std::{
borrow::Cow,
collections::HashMap, collections::HashMap,
convert::Infallible, convert::Infallible,
fmt, fmt,
sync::Arc, sync::Arc,
task::{Context, Poll}, task::{Context, Poll},
}; };
use tower::{layer::layer_fn, util::MapResponseLayer, ServiceBuilder}; use tower::{util::MapResponseLayer, ServiceBuilder};
use tower_layer::Layer; use tower_layer::Layer;
use tower_service::Service; use tower_service::Service;
@ -65,7 +63,6 @@ pub struct Router<B = Body> {
routes: HashMap<RouteId, Endpoint<B>>, routes: HashMap<RouteId, Endpoint<B>>,
node: Arc<Node>, node: Arc<Node>,
fallback: Fallback<B>, fallback: Fallback<B>,
nested_at_root: bool,
} }
impl<B> Clone for Router<B> { impl<B> Clone for Router<B> {
@ -74,7 +71,6 @@ impl<B> Clone for Router<B> {
routes: self.routes.clone(), routes: self.routes.clone(),
node: Arc::clone(&self.node), node: Arc::clone(&self.node),
fallback: self.fallback.clone(), fallback: self.fallback.clone(),
nested_at_root: self.nested_at_root,
} }
} }
} }
@ -94,7 +90,6 @@ impl<B> fmt::Debug for Router<B> {
.field("routes", &self.routes) .field("routes", &self.routes)
.field("node", &self.node) .field("node", &self.node)
.field("fallback", &self.fallback) .field("fallback", &self.fallback)
.field("nested_at_root", &self.nested_at_root)
.finish() .finish()
} }
} }
@ -115,7 +110,6 @@ where
routes: Default::default(), routes: Default::default(),
node: Default::default(), node: Default::default(),
fallback: Fallback::Default(Route::new(NotFound)), fallback: Fallback::Default(Route::new(NotFound)),
nested_at_root: false,
} }
} }
@ -177,7 +171,7 @@ where
let mut node = let mut node =
Arc::try_unwrap(Arc::clone(&self.node)).unwrap_or_else(|node| (*node).clone()); Arc::try_unwrap(Arc::clone(&self.node)).unwrap_or_else(|node| (*node).clone());
if let Err(err) = node.insert(path, id) { if let Err(err) = node.insert(path, id) {
self.panic_on_matchit_error(err); panic!("Invalid route {path:?}: {err}");
} }
self.node = Arc::new(node); self.node = Arc::new(node);
} }
@ -200,63 +194,22 @@ where
let prefix = path; let prefix = path;
if path == "/" { let path = if path.ends_with('/') {
self.nested_at_root = true; format!("{path}*{NEST_TAIL_PARAM}")
} } else {
format!("{path}/*{NEST_TAIL_PARAM}")
};
match try_downcast::<Router<B>, _>(svc) { let svc = strip_prefix::StripPrefix::new(svc, prefix);
// if the user is nesting a `Router` we can implement nesting self = self.route(&path, svc.clone());
// by simplying copying all the routes and adding the prefix in
// front
Ok(router) => {
let Router {
mut routes,
node,
fallback,
// nesting a router that has something nested at root
// doesn't mean something is nested at root in _this_ router
// thus we don't need to propagate that
nested_at_root: _,
} = router;
if let Fallback::Custom(_) = fallback { // `/*rest` is not matched by `/` so we need to also register a router at the
panic!("Cannot nest `Router`s that has a fallback"); // prefix itself. Otherwise if you were to nest at `/foo` then `/foo` itself
} // wouldn't match, which it should
self = self.route(prefix, svc.clone());
for (id, nested_path) in &node.route_id_to_path { if !prefix.ends_with('/') {
let route = routes.remove(id).unwrap(); // same goes for `/foo/`, that should also match
let full_path: Cow<str> = if &**nested_path == "/" { self = self.route(&format!("{prefix}/"), svc);
path.into()
} else if path == "/" {
(&**nested_path).into()
} else if let Some(path) = path.strip_suffix('/') {
format!("{}{}", path, nested_path).into()
} else {
format!("{}{}", path, nested_path).into()
};
self = match route {
Endpoint::MethodRouter(method_router) => self.route(
&full_path,
method_router.layer(layer_fn(|s| StripPrefix::new(s, prefix))),
),
Endpoint::Route(route) => {
self.route(&full_path, StripPrefix::new(route, prefix))
}
};
}
debug_assert!(routes.is_empty());
}
// otherwise we add a wildcard route to the service
Err(svc) => {
let path = if path.ends_with('/') {
format!("{}*{}", path, NEST_TAIL_PARAM)
} else {
format!("{}/*{}", path, NEST_TAIL_PARAM)
};
self = self.route(&path, strip_prefix::StripPrefix::new(svc, prefix));
}
} }
self self
@ -271,7 +224,6 @@ where
routes, routes,
node, node,
fallback, fallback,
nested_at_root,
} = other.into(); } = other.into();
for (id, route) in routes { for (id, route) in routes {
@ -294,8 +246,6 @@ where
} }
}; };
self.nested_at_root = self.nested_at_root || nested_at_root;
self self
} }
@ -334,7 +284,6 @@ where
routes, routes,
node: self.node, node: self.node,
fallback, fallback,
nested_at_root: self.nested_at_root,
} }
} }
@ -371,7 +320,6 @@ where
routes, routes,
node: self.node, node: self.node,
fallback: self.fallback, fallback: self.fallback,
nested_at_root: self.nested_at_root,
} }
} }
@ -474,17 +422,6 @@ where
Endpoint::Route(inner) => inner.call(req), Endpoint::Route(inner) => inner.call(req),
} }
} }
fn panic_on_matchit_error(&self, err: matchit::InsertError) {
if self.nested_at_root {
panic!(
"Invalid route: {}. Note that `nest(\"/\", _)` conflicts with all routes. Use `Router::fallback` instead",
err,
);
} else {
panic!("Invalid route: {}", err);
}
}
} }
impl<B> Service<Request<B>> for Router<B> impl<B> Service<Request<B>> for Router<B>

View file

@ -240,7 +240,7 @@ async fn all_the_uris(
#[tokio::test] #[tokio::test]
async fn nesting_and_seeing_the_right_uri() { async fn nesting_and_seeing_the_right_uri() {
let one = Router::new().nest("/foo", Router::new().route("/bar", get(all_the_uris))); let one = Router::new().nest("/foo/", Router::new().route("/bar", get(all_the_uris)));
let two = Router::new().route("/foo", get(all_the_uris)); let two = Router::new().route("/foo", get(all_the_uris));
let client = TestClient::new(one.merge(two)); let client = TestClient::new(one.merge(two));
@ -271,7 +271,7 @@ async fn nesting_and_seeing_the_right_uri() {
#[tokio::test] #[tokio::test]
async fn nesting_and_seeing_the_right_uri_at_more_levels_of_nesting() { async fn nesting_and_seeing_the_right_uri_at_more_levels_of_nesting() {
let one = Router::new().nest( let one = Router::new().nest(
"/foo", "/foo/",
Router::new().nest("/bar", Router::new().route("/baz", get(all_the_uris))), Router::new().nest("/bar", Router::new().route("/baz", get(all_the_uris))),
); );
let two = Router::new().route("/foo", get(all_the_uris)); let two = Router::new().route("/foo", get(all_the_uris));

View file

@ -8,10 +8,9 @@ use crate::{
test_helpers::*, test_helpers::*,
BoxError, Json, Router, BoxError, Json, Router,
}; };
use http::{header::CONTENT_LENGTH, HeaderMap, Method, Request, Response, StatusCode, Uri}; use http::{header::CONTENT_LENGTH, HeaderMap, Request, Response, StatusCode, Uri};
use hyper::Body; use hyper::Body;
use serde::Deserialize; use serde_json::json;
use serde_json::{json, Value};
use std::{ use std::{
convert::Infallible, convert::Infallible,
future::{ready, Ready}, future::{ready, Ready},
@ -359,44 +358,23 @@ async fn with_and_without_trailing_slash() {
// for https://github.com/tokio-rs/axum/issues/420 // for https://github.com/tokio-rs/axum/issues/420
#[tokio::test] #[tokio::test]
async fn wildcard_with_trailing_slash() { async fn wildcard_doesnt_match_just_trailing_slash() {
#[derive(Deserialize, serde::Serialize)] let app = Router::new().route(
struct Tree { "/x/*path",
user: String, get(|Path(path): Path<String>| async move { path }),
repo: String,
path: String,
}
let app: Router = Router::new().route(
"/:user/:repo/tree/*path",
get(|Path(tree): Path<Tree>| async move { Json(tree) }),
); );
// low level check that the correct redirect happens let client = TestClient::new(app);
let res = app
.clone() let res = client.get("/x").send().await;
.oneshot(
Request::builder()
.method(Method::GET)
.uri("/user1/repo1/tree")
.body(Body::empty())
.unwrap(),
)
.await
.unwrap();
assert_eq!(res.status(), StatusCode::NOT_FOUND); assert_eq!(res.status(), StatusCode::NOT_FOUND);
// check that the params are deserialized correctly let res = client.get("/x/").send().await;
let client = TestClient::new(app); assert_eq!(res.status(), StatusCode::NOT_FOUND);
let res = client.get("/user1/repo1/tree/").send().await;
assert_eq!( let res = client.get("/x/foo/bar").send().await;
res.json::<Value>().await, assert_eq!(res.status(), StatusCode::OK);
json!({ assert_eq!(res.text().await, "foo/bar");
"user": "user1",
"repo": "repo1",
"path": "/",
})
);
} }
#[tokio::test] #[tokio::test]
@ -500,34 +478,6 @@ async fn route_layer() {
assert_eq!(res.status(), StatusCode::UNAUTHORIZED); assert_eq!(res.status(), StatusCode::UNAUTHORIZED);
} }
#[tokio::test]
#[should_panic(
expected = "Invalid route: insertion failed due to conflict with previously registered \
route: /*__private__axum_nest_tail_param. \
Note that `nest(\"/\", _)` conflicts with all routes. \
Use `Router::fallback` instead"
)]
async fn good_error_message_if_using_nest_root() {
let app = Router::new()
.nest("/", get(|| async {}))
.route("/", get(|| async {}));
TestClient::new(app);
}
#[tokio::test]
#[should_panic(
expected = "Invalid route: insertion failed due to conflict with previously registered \
route: /*__private__axum_nest_tail_param. \
Note that `nest(\"/\", _)` conflicts with all routes. \
Use `Router::fallback` instead"
)]
async fn good_error_message_if_using_nest_root_when_merging() {
let one = Router::new().nest("/", get(|| async {}));
let two = Router::new().route("/", get(|| async {}));
let app = one.merge(two);
TestClient::new(app);
}
#[tokio::test] #[tokio::test]
async fn different_methods_added_in_different_routes() { async fn different_methods_added_in_different_routes() {
let app = Router::new() let app = Router::new()
@ -545,29 +495,6 @@ async fn different_methods_added_in_different_routes() {
assert_eq!(body, "POST"); assert_eq!(body, "POST");
} }
#[tokio::test]
async fn different_methods_added_in_different_routes_deeply_nested() {
let app = Router::new()
.route("/foo/bar/baz", get(|| async { "GET" }))
.nest(
"/foo",
Router::new().nest(
"/bar",
Router::new().route("/baz", post(|| async { "POST" })),
),
);
let client = TestClient::new(app);
let res = client.get("/foo/bar/baz").send().await;
let body = res.text().await;
assert_eq!(body, "GET");
let res = client.post("/foo/bar/baz").send().await;
let body = res.text().await;
assert_eq!(body, "POST");
}
#[tokio::test] #[tokio::test]
#[should_panic(expected = "Cannot merge two `Router`s that both have a fallback")] #[should_panic(expected = "Cannot merge two `Router`s that both have a fallback")]
async fn merging_routers_with_fallbacks_panics() { async fn merging_routers_with_fallbacks_panics() {
@ -577,15 +504,6 @@ async fn merging_routers_with_fallbacks_panics() {
TestClient::new(one.merge(two)); TestClient::new(one.merge(two));
} }
#[tokio::test]
#[should_panic(expected = "Cannot nest `Router`s that has a fallback")]
async fn nesting_router_with_fallbacks_panics() {
async fn fallback() {}
let one = Router::new().fallback(fallback.into_service());
let app = Router::new().nest("/", one);
TestClient::new(app);
}
#[tokio::test] #[tokio::test]
async fn merging_routers_with_same_paths_but_different_methods() { async fn merging_routers_with_same_paths_but_different_methods() {
let one = Router::new().route("/", get(|| async { "GET" })); let one = Router::new().route("/", get(|| async { "GET" }));

View file

@ -1,8 +1,5 @@
use super::*; use super::*;
use crate::{ use crate::{body::boxed, extract::Extension};
body::boxed,
extract::{Extension, MatchedPath},
};
use std::collections::HashMap; use std::collections::HashMap;
use tower_http::services::ServeDir; use tower_http::services::ServeDir;
@ -17,7 +14,6 @@ async fn nesting_apps() {
"/users/:id", "/users/:id",
get( get(
|params: extract::Path<HashMap<String, String>>| async move { |params: extract::Path<HashMap<String, String>>| async move {
dbg!(&params);
format!( format!(
"{}: users#show ({})", "{}: users#show ({})",
params.get("version").unwrap(), params.get("version").unwrap(),
@ -239,38 +235,12 @@ async fn nested_multiple_routes() {
assert_eq!(client.get("/api/teams").send().await.text().await, "teams"); assert_eq!(client.get("/api/teams").send().await.text().await, "teams");
} }
#[tokio::test] #[test]
async fn nested_with_other_route_also_matching_with_route_first() { #[should_panic = "Invalid route \"/\": insertion failed due to conflict with previously registered route: /*__private__axum_nest_tail_param"]
let app = Router::new().route("/api", get(|| async { "api" })).nest( fn nested_at_root_with_other_routes() {
"/api", let _: Router = Router::new()
Router::new() .nest("/", Router::new().route("/users", get(|| async {})))
.route("/users", get(|| async { "users" })) .route("/", get(|| async {}));
.route("/teams", get(|| async { "teams" })),
);
let client = TestClient::new(app);
assert_eq!(client.get("/api").send().await.text().await, "api");
assert_eq!(client.get("/api/users").send().await.text().await, "users");
assert_eq!(client.get("/api/teams").send().await.text().await, "teams");
}
#[tokio::test]
async fn nested_with_other_route_also_matching_with_route_last() {
let app = Router::new()
.nest(
"/api",
Router::new()
.route("/users", get(|| async { "users" }))
.route("/teams", get(|| async { "teams" })),
)
.route("/api", get(|| async { "api" }));
let client = TestClient::new(app);
assert_eq!(client.get("/api").send().await.text().await, "api");
assert_eq!(client.get("/api/users").send().await.text().await, "users");
assert_eq!(client.get("/api/teams").send().await.text().await, "teams");
} }
#[tokio::test] #[tokio::test]
@ -366,64 +336,97 @@ async fn nest_at_capture() {
assert_eq!(res.text().await, "a=foo b=bar"); assert_eq!(res.text().await, "a=foo b=bar");
} }
#[tokio::test]
async fn nest_with_and_without_trailing() {
let app = Router::new().nest("/foo", get(|| async {}));
let client = TestClient::new(app);
let res = client.get("/foo").send().await;
assert_eq!(res.status(), StatusCode::OK);
let res = client.get("/foo/").send().await;
assert_eq!(res.status(), StatusCode::OK);
let res = client.get("/foo/bar").send().await;
assert_eq!(res.status(), StatusCode::OK);
}
#[tokio::test]
async fn doesnt_call_outer_fallback() {
let app = Router::new()
.nest("/foo", Router::new().route("/", get(|| async {})))
.fallback((|| async { (StatusCode::NOT_FOUND, "outer fallback") }).into_service());
let client = TestClient::new(app);
let res = client.get("/foo").send().await;
assert_eq!(res.status(), StatusCode::OK);
let res = client.get("/foo/not-found").send().await;
assert_eq!(res.status(), StatusCode::NOT_FOUND);
// the default fallback returns an empty body
assert_eq!(res.text().await, "");
}
#[tokio::test]
async fn nesting_with_root_inner_router() {
let app = Router::new().nest(
"/foo",
Router::new().route("/", get(|| async { "inner route" })),
);
let client = TestClient::new(app);
// `/foo/` does match the `/foo` prefix and the remaining path is technically
// empty, which is the same as `/` which matches `.route("/", _)`
let res = client.get("/foo").send().await;
assert_eq!(res.status(), StatusCode::OK);
// `/foo/` does match the `/foo` prefix and the remaining path is `/`
// which matches `.route("/", _)`
let res = client.get("/foo/").send().await;
assert_eq!(res.status(), StatusCode::OK);
}
#[tokio::test]
async fn fallback_on_inner() {
let app = Router::new()
.nest(
"/foo",
Router::new()
.route("/", get(|| async {}))
.fallback((|| async { (StatusCode::NOT_FOUND, "inner fallback") }).into_service()),
)
.fallback((|| async { (StatusCode::NOT_FOUND, "outer fallback") }).into_service());
let client = TestClient::new(app);
let res = client.get("/foo/not-found").send().await;
assert_eq!(res.status(), StatusCode::NOT_FOUND);
assert_eq!(res.text().await, "inner fallback");
}
macro_rules! nested_route_test { macro_rules! nested_route_test {
( (
$name:ident, $name:ident,
// the path we nest the inner router at
nest = $nested_path:literal, nest = $nested_path:literal,
// the route the inner router accepts
route = $route_path:literal, route = $route_path:literal,
expected = $expected_path:literal, // the route we expect to be able to call
opaque_redirect = $opaque_redirect:expr $(,)? expected = $expected_path:literal $(,)?
) => { ) => {
#[tokio::test] #[tokio::test]
async fn $name() { async fn $name() {
let inner = Router::new().route( let inner = Router::new().route($route_path, get(|| async {}));
$route_path,
get(|matched: MatchedPath| async move { matched.as_str().to_owned() }),
);
let app = Router::new().nest($nested_path, inner); let app = Router::new().nest($nested_path, inner);
let client = TestClient::new(app); let client = TestClient::new(app);
let res = client.get($expected_path).send().await; let res = client.get($expected_path).send().await;
let status = res.status(); let status = res.status();
let matched_path = res.text().await;
assert_eq!(status, StatusCode::OK, "Router"); assert_eq!(status, StatusCode::OK, "Router");
let inner = Router::new()
.route(
$route_path,
get(|matched: MatchedPath| async move { matched.as_str().to_owned() }),
)
.boxed_clone();
let app = Router::new().nest($nested_path, inner);
let client = TestClient::new(app);
let res = client.get($expected_path).send().await;
if $opaque_redirect {
assert_eq!(res.status(), StatusCode::PERMANENT_REDIRECT, "opaque");
let location = res.headers()[http::header::LOCATION].to_str().unwrap();
let res = client.get(location).send().await;
assert_eq!(res.status(), StatusCode::OK, "opaque with redirect");
assert_eq!(res.text().await, location);
} else {
assert_eq!(res.status(), StatusCode::OK, "opaque");
assert_eq!(res.text().await, matched_path);
}
} }
}; };
(
$name:ident,
nest = $nested_path:literal,
route = $route_path:literal,
expected = $expected_path:literal $(,)?
) => {
nested_route_test!(
$name,
nest = $nested_path,
route = $route_path,
expected = $expected_path,
opaque_redirect = false,
);
};
} }
// test cases taken from https://github.com/tokio-rs/axum/issues/714#issuecomment-1058144460 // test cases taken from https://github.com/tokio-rs/axum/issues/714#issuecomment-1058144460
@ -433,23 +436,7 @@ nested_route_test!(nest_3, nest = "", route = "/a/", expected = "/a/");
nested_route_test!(nest_4, nest = "/", route = "/", expected = "/"); nested_route_test!(nest_4, nest = "/", route = "/", expected = "/");
nested_route_test!(nest_5, nest = "/", route = "/a", expected = "/a"); nested_route_test!(nest_5, nest = "/", route = "/a", expected = "/a");
nested_route_test!(nest_6, nest = "/", route = "/a/", expected = "/a/"); nested_route_test!(nest_6, nest = "/", route = "/a/", expected = "/a/");
nested_route_test!(nest_7, nest = "/a", route = "/", expected = "/a");
// TODO(david): we'll revisit this in https://github.com/tokio-rs/axum/pull/1086
//// This case is different for opaque services.
////
//// The internal route becomes `/a/*__private__axum_nest_tail_param` which, according to matchit
//// doesn't match `/a`. However matchit detects that a route for `/a/` exists and so it issues a
//// redirect to `/a/`, which ends up calling the inner route as expected.
////
//// So while the behavior isn't identical, the outcome is the same
//nested_route_test!(
// nest_7,
// nest = "/a",
// route = "/",
// expected = "/a",
// opaque_redirect = true,
//);
nested_route_test!(nest_8, nest = "/a", route = "/a", expected = "/a/a"); nested_route_test!(nest_8, nest = "/a", route = "/a", expected = "/a/a");
nested_route_test!(nest_9, nest = "/a", route = "/a/", expected = "/a/a/"); nested_route_test!(nest_9, nest = "/a", route = "/a/", expected = "/a/a/");
nested_route_test!(nest_11, nest = "/a/", route = "/", expected = "/a/"); nested_route_test!(nest_11, nest = "/a/", route = "/", expected = "/a/");