Improve error message for nest("/", _) (#500)

This commit is contained in:
David Pedersen 2021-11-11 19:22:25 +01:00 committed by GitHub
parent fb8156f17c
commit 34942f3d70
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 62 additions and 2 deletions

View file

@ -66,6 +66,7 @@ pub struct Router<B = Body> {
routes: HashMap<RouteId, Route<B>>, routes: HashMap<RouteId, Route<B>>,
node: Node, node: Node,
fallback: Fallback<B>, fallback: Fallback<B>,
nested_at_root: bool,
} }
impl<B> Clone for Router<B> { impl<B> Clone for Router<B> {
@ -74,6 +75,7 @@ impl<B> Clone for Router<B> {
routes: self.routes.clone(), routes: self.routes.clone(),
node: self.node.clone(), node: self.node.clone(),
fallback: self.fallback.clone(), fallback: self.fallback.clone(),
nested_at_root: self.nested_at_root,
} }
} }
} }
@ -103,6 +105,7 @@ 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,
} }
} }
@ -129,7 +132,7 @@ where
let id = RouteId::next(); let id = RouteId::next();
if let Err(err) = self.node.insert(path, id) { if let Err(err) = self.node.insert(path, id) {
panic!("Invalid route: {}", err); self.panic_on_matchit_error(err);
} }
self.routes.insert(id, Route::new(service)); self.routes.insert(id, Route::new(service));
@ -156,6 +159,10 @@ where
let prefix = path; let prefix = path;
if path == "/" {
self.nested_at_root = true;
}
match try_downcast::<Router<B>, _>(svc) { match try_downcast::<Router<B>, _>(svc) {
// if the user is nesting a `Router` we can implement nesting // if the user is nesting a `Router` we can implement nesting
// by simplying copying all the routes and adding the prefix in // by simplying copying all the routes and adding the prefix in
@ -166,6 +173,10 @@ where
node, node,
// discard the fallback of the nested router // discard the fallback of the nested router
fallback: _, 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; } = router;
for (id, nested_path) in node.paths { for (id, nested_path) in node.paths {
@ -201,10 +212,11 @@ where
routes, routes,
node, node,
fallback, fallback,
nested_at_root,
} = other; } = other;
if let Err(err) = self.node.merge(node) { if let Err(err) = self.node.merge(node) {
panic!("Invalid route: {}", err); self.panic_on_matchit_error(err);
} }
for (id, route) in routes { for (id, route) in routes {
@ -218,6 +230,8 @@ where
(Fallback::Custom(_), pick @ Fallback::Custom(_)) => pick, (Fallback::Custom(_), pick @ Fallback::Custom(_)) => pick,
}; };
self.nested_at_root = self.nested_at_root || nested_at_root;
self self
} }
@ -253,6 +267,7 @@ where
routes, routes,
node: self.node, node: self.node,
fallback, fallback,
nested_at_root: self.nested_at_root,
} }
} }
@ -286,6 +301,7 @@ where
routes, routes,
node: self.node, node: self.node,
fallback: self.fallback, fallback: self.fallback,
nested_at_root: self.nested_at_root,
} }
} }
@ -380,6 +396,17 @@ where
RouterFuture::from_oneshot(route.oneshot(req)) RouterFuture::from_oneshot(route.oneshot(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

@ -504,3 +504,36 @@ async fn route_layer() {
let res = client.post("/foo").send().await; let res = client.post("/foo").send().await;
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: /foo"
)]
async fn conflicting_route() {
let app = Router::new()
.route("/foo", get(|| async {}))
.route("/foo", get(|| async {}));
TestClient::new(app);
}
#[tokio::test]
#[should_panic(
expected = "Invalid route: insertion failed due to conflict with previously registered route: /*axum_nest. 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: /*axum_nest. 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);
}