mirror of
https://github.com/tokio-rs/axum.git
synced 2025-01-01 08:56:15 +01:00
Clone state a bit less (#1837)
This commit is contained in:
parent
f65fa22f34
commit
fe9c4a0b5b
4 changed files with 79 additions and 33 deletions
|
@ -8,8 +8,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||
# Unreleased
|
||||
|
||||
- **fixed:** Don't require `S: Debug` for `impl Debug for Router<S>` ([#1836])
|
||||
- **fixed:** Clone state a bit less when handling requests ([#1837])
|
||||
|
||||
[#1836]: https://github.com/tokio-rs/axum/pull/1836
|
||||
[#1837]: https://github.com/tokio-rs/axum/pull/1837
|
||||
|
||||
# 0.6.10 (03. March, 2023)
|
||||
|
||||
|
|
|
@ -107,6 +107,7 @@ futures = "0.3"
|
|||
quickcheck = "1.0"
|
||||
quickcheck_macros = "1.0"
|
||||
reqwest = { version = "0.11.14", default-features = false, features = ["json", "stream", "multipart"] }
|
||||
rustversion = "1.0.9"
|
||||
serde = { version = "1.0", features = ["derive"] }
|
||||
serde_json = "1.0"
|
||||
time = { version = "0.3", features = ["serde-human-readable"] }
|
||||
|
|
|
@ -447,7 +447,29 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
self.call_route(match_, req, state)
|
||||
let id = *match_.value;
|
||||
|
||||
#[cfg(feature = "matched-path")]
|
||||
crate::extract::matched_path::set_matched_path_for_request(
|
||||
id,
|
||||
&self.node.route_id_to_path,
|
||||
req.extensions_mut(),
|
||||
);
|
||||
|
||||
url_params::insert_url_params(req.extensions_mut(), match_.params);
|
||||
|
||||
let endpont = self
|
||||
.routes
|
||||
.get_mut(&id)
|
||||
.expect("no route for id. This is a bug in axum. Please file an issue");
|
||||
|
||||
match endpont {
|
||||
Endpoint::MethodRouter(method_router) => {
|
||||
method_router.call_with_state(req, state)
|
||||
}
|
||||
Endpoint::Route(route) => route.call(req),
|
||||
Endpoint::NestedRouter(router) => router.clone().call_with_state(req, state),
|
||||
}
|
||||
}
|
||||
Err(
|
||||
MatchError::NotFound
|
||||
|
@ -469,37 +491,6 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn call_route(
|
||||
&self,
|
||||
match_: matchit::Match<&RouteId>,
|
||||
mut req: Request<B>,
|
||||
state: S,
|
||||
) -> RouteFuture<B, Infallible> {
|
||||
let id = *match_.value;
|
||||
|
||||
#[cfg(feature = "matched-path")]
|
||||
crate::extract::matched_path::set_matched_path_for_request(
|
||||
id,
|
||||
&self.node.route_id_to_path,
|
||||
req.extensions_mut(),
|
||||
);
|
||||
|
||||
url_params::insert_url_params(req.extensions_mut(), match_.params);
|
||||
|
||||
let endpont = self
|
||||
.routes
|
||||
.get(&id)
|
||||
.expect("no route for id. This is a bug in axum. Please file an issue")
|
||||
.clone();
|
||||
|
||||
match endpont {
|
||||
Endpoint::MethodRouter(mut method_router) => method_router.call_with_state(req, state),
|
||||
Endpoint::Route(mut route) => route.call(req),
|
||||
Endpoint::NestedRouter(router) => router.call_with_state(req, state),
|
||||
}
|
||||
}
|
||||
|
||||
fn next_route_id(&mut self) -> RouteId {
|
||||
let next_id = self
|
||||
.prev_route_id
|
||||
|
|
|
@ -15,7 +15,7 @@ use serde_json::json;
|
|||
use std::{
|
||||
convert::Infallible,
|
||||
future::{ready, Ready},
|
||||
sync::atomic::{AtomicUsize, Ordering},
|
||||
sync::atomic::{AtomicBool, AtomicUsize, Ordering},
|
||||
task::{Context, Poll},
|
||||
time::Duration,
|
||||
};
|
||||
|
@ -812,3 +812,55 @@ fn method_router_fallback_with_state() {
|
|||
.fallback(get(fallback).fallback(not_found))
|
||||
.with_state(state);
|
||||
}
|
||||
|
||||
#[crate::test]
|
||||
async fn state_isnt_cloned_too_much() {
|
||||
static SETUP_DONE: AtomicBool = AtomicBool::new(false);
|
||||
static COUNT: AtomicUsize = AtomicUsize::new(0);
|
||||
|
||||
struct AppState;
|
||||
|
||||
impl Clone for AppState {
|
||||
fn clone(&self) -> Self {
|
||||
#[rustversion::since(1.65)]
|
||||
#[track_caller]
|
||||
fn count() {
|
||||
if SETUP_DONE.load(Ordering::SeqCst) {
|
||||
let bt = std::backtrace::Backtrace::force_capture();
|
||||
let bt = bt
|
||||
.to_string()
|
||||
.lines()
|
||||
.filter(|line| line.contains("axum") || line.contains("./src"))
|
||||
.collect::<Vec<_>>()
|
||||
.join("\n");
|
||||
println!("AppState::Clone:\n===============\n{}\n", bt);
|
||||
COUNT.fetch_add(1, Ordering::SeqCst);
|
||||
}
|
||||
}
|
||||
|
||||
#[rustversion::not(since(1.65))]
|
||||
fn count() {
|
||||
if SETUP_DONE.load(Ordering::SeqCst) {
|
||||
COUNT.fetch_add(1, Ordering::SeqCst);
|
||||
}
|
||||
}
|
||||
|
||||
count();
|
||||
|
||||
Self
|
||||
}
|
||||
}
|
||||
|
||||
let app = Router::new()
|
||||
.route("/", get(|_: State<AppState>| async {}))
|
||||
.with_state(AppState);
|
||||
|
||||
let client = TestClient::new(app);
|
||||
|
||||
// ignore clones made during setup
|
||||
SETUP_DONE.store(true, Ordering::SeqCst);
|
||||
|
||||
client.get("/").send().await;
|
||||
|
||||
assert_eq!(COUNT.load(Ordering::SeqCst), 4);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue