Fix routing for opaque nested services (#842)

* Fix routing for opaque nested services

* Also test `MatchedPath`

* clean up and more comments
This commit is contained in:
David Pedersen 2022-03-09 11:25:06 +01:00 committed by GitHub
parent 70e3024068
commit 401c7a324a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 141 additions and 57 deletions

View file

@ -70,6 +70,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
requests ([#734]) requests ([#734])
- **fixed:** Fix wrong `content-length` for `HEAD` requests to endpoints that returns chunked - **fixed:** Fix wrong `content-length` for `HEAD` requests to endpoints that returns chunked
responses ([#755]) responses ([#755])
- **fixed:** Fixed several routing bugs related to nested "opaque" tower services (i.e.
non-`Router` services) ([#841] and [#842])
- **changed:** Update to tokio-tungstenite 0.17 ([#791]) - **changed:** Update to tokio-tungstenite 0.17 ([#791])
[#644]: https://github.com/tokio-rs/axum/pull/644 [#644]: https://github.com/tokio-rs/axum/pull/644
@ -89,6 +91,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
[#819]: https://github.com/tokio-rs/axum/pull/819 [#819]: https://github.com/tokio-rs/axum/pull/819
[#823]: https://github.com/tokio-rs/axum/pull/823 [#823]: https://github.com/tokio-rs/axum/pull/823
[#824]: https://github.com/tokio-rs/axum/pull/824 [#824]: https://github.com/tokio-rs/axum/pull/824
[#841]: https://github.com/tokio-rs/axum/pull/841
[#842]: https://github.com/tokio-rs/axum/pull/842
# 0.4.4 (13. January, 2022) # 0.4.4 (13. January, 2022)

View file

@ -237,8 +237,8 @@ where
} }
// otherwise we add a wildcard route to the service // otherwise we add a wildcard route to the service
Err(svc) => { Err(svc) => {
let path = if path == "/" { let path = if path.ends_with('/') {
format!("/*{}", NEST_TAIL_PARAM) format!("{}*{}", path, NEST_TAIL_PARAM)
} else { } else {
format!("{}/*{}", path, NEST_TAIL_PARAM) format!("{}/*{}", path, NEST_TAIL_PARAM)
}; };

View file

@ -42,62 +42,78 @@ where
} }
fn strip_prefix(uri: &Uri, prefix: &str) -> Option<Uri> { fn strip_prefix(uri: &Uri, prefix: &str) -> Option<Uri> {
let path_and_query: http::uri::PathAndQuery = if let Some(path_and_query) = uri.path_and_query() let path_and_query = uri.path_and_query()?;
{
// Check whether the prefix matches the path and if so how long the matching prefix is.
//
// # Examples
//
// prefix = /api
// uri = /api/users
// ^^^^ this much is matched and the length is 4. Thus if we chop off the first 4
// characters we get the remainder
//
// prefix = /api/:version
// uri = /api/v0/users
// ^^^^^^^ this much is matched and the length is 7.
let mut matching_prefix_length = Some(0);
for item in zip_longest(segments(path_and_query.path()), segments(prefix)) {
// count the `/`
*matching_prefix_length.as_mut().unwrap() += 1;
match item { // Check whether the prefix matches the path and if so how long the matching prefix is.
Item::Both(path_segment, prefix_segment) => { //
if prefix_segment.starts_with(':') || path_segment == prefix_segment { // For example:
*matching_prefix_length.as_mut().unwrap() += path_segment.len(); //
} else { // prefix = /api
matching_prefix_length = None; // path = /api/users
break; // ^^^^ this much is matched and the length is 4. Thus if we chop off the first 4
} // characters we get the remainder
} //
Item::First(_) => { // prefix = /api/:version
// path = /api/v0/users
// ^^^^^^^ this much is matched and the length is 7.
let mut matching_prefix_length = Some(0);
for item in zip_longest(segments(path_and_query.path()), segments(prefix)) {
// count the `/`
*matching_prefix_length.as_mut().unwrap() += 1;
match item {
Item::Both(path_segment, prefix_segment) => {
if prefix_segment.starts_with(':') || path_segment == prefix_segment {
// the prefix segment is either a param, which matches anything, or
// it actually matches the path segment
*matching_prefix_length.as_mut().unwrap() += path_segment.len();
} else if prefix_segment.is_empty() {
// the prefix ended in a `/` so we got a match.
//
// For example:
//
// prefix = /foo/
// path = /foo/bar
//
// The prefix matches and the new path should be `/bar`
break; break;
} } else {
Item::Second(_) => { // the prefix segment didn't match so there is no match
matching_prefix_length = None; matching_prefix_length = None;
break; break;
} }
} }
// the path had more segments than the prefix but we got a match.
//
// For example:
//
// prefix = /foo
// path = /foo/bar
Item::First(_) => {
break;
}
// the prefix had more segments than the path so there is no match
Item::Second(_) => {
matching_prefix_length = None;
break;
}
} }
}
let after_prefix = if let Some(idx) = matching_prefix_length { // if the prefix matches it will always do so up until a `/`, it cannot match only
uri.path().split_at(idx).1 // part of a segment. Therefore this will always be at a char boundary and `split_at` wont
} else { // panic
return None; let after_prefix = uri.path().split_at(matching_prefix_length?).1;
};
match (after_prefix.starts_with('/'), path_and_query.query()) { let new_path_and_query = match (after_prefix.starts_with('/'), path_and_query.query()) {
(true, None) => after_prefix.parse().unwrap(), (true, None) => after_prefix.parse().unwrap(),
(true, Some(query)) => format!("{}?{}", after_prefix, query).parse().unwrap(), (true, Some(query)) => format!("{}?{}", after_prefix, query).parse().unwrap(),
(false, None) => format!("/{}", after_prefix).parse().unwrap(), (false, None) => format!("/{}", after_prefix).parse().unwrap(),
(false, Some(query)) => format!("/{}?{}", after_prefix, query).parse().unwrap(), (false, Some(query)) => format!("/{}?{}", after_prefix, query).parse().unwrap(),
}
} else {
return None;
}; };
let mut parts = uri.clone().into_parts(); let mut parts = uri.clone().into_parts();
parts.path_and_query = Some(path_and_query); parts.path_and_query = Some(new_path_and_query);
Some(Uri::from_parts(parts).unwrap()) Some(Uri::from_parts(parts).unwrap())
} }
@ -183,7 +199,7 @@ mod tests {
single_segment_root_prefix, single_segment_root_prefix,
uri = "/a", uri = "/a",
prefix = "/", prefix = "/",
expected = None, expected = Some("/a"),
); );
test!( test!(
@ -334,6 +350,13 @@ mod tests {
expected = Some("/"), expected = Some("/"),
); );
test!(
param_13,
uri = "/a/a",
prefix = "/a/",
expected = Some("/a"),
);
#[quickcheck] #[quickcheck]
fn does_not_panic(uri_and_prefix: UriAndPrefix) -> bool { fn does_not_panic(uri_and_prefix: UriAndPrefix) -> bool {
let UriAndPrefix { uri, prefix } = uri_and_prefix; let UriAndPrefix { uri, prefix } = uri_and_prefix;

View file

@ -1,8 +1,10 @@
use tower_http::services::ServeDir;
use super::*; use super::*;
use crate::{body::boxed, extract::Extension}; use crate::{
body::boxed,
extract::{Extension, MatchedPath},
};
use std::collections::HashMap; use std::collections::HashMap;
use tower_http::services::ServeDir;
#[tokio::test] #[tokio::test]
async fn nesting_apps() { async fn nesting_apps() {
@ -369,19 +371,59 @@ macro_rules! nested_route_test {
$name:ident, $name:ident,
nest = $nested_path:literal, nest = $nested_path:literal,
route = $route_path:literal, route = $route_path:literal,
expected = $expected_path:literal $(,)? expected = $expected_path:literal,
opaque_redirect = $opaque_redirect:expr $(,)?
) => { ) => {
#[tokio::test] #[tokio::test]
async fn $name() { async fn $name() {
let inner = Router::new().route($route_path, get(|| async {})); let inner = Router::new().route(
$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);
assert_eq!( let res = client.get($expected_path).send().await;
client.get($expected_path).send().await.status(), let status = res.status();
StatusCode::OK let matched_path = res.text().await;
); 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
@ -391,7 +433,22 @@ 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");
// 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/");