Remove MaybeSharedNode from Router (#403)

In #363 I added `MaybeSharedNode` which makes `Router` faster to clone
_if_ you're using `into_make_service`. However I would like instead like
to find a solution that works even when you're not using
`into_make_service`.

Using an `Arc<RwLock<Node<_>>>` is probably the only solution but would
like to experiment. For now I'm gonna rollback `MaybeSharedNode`.
Changing it wont require user facing changes.
This commit is contained in:
David Pedersen 2021-10-24 19:49:31 +02:00 committed by GitHub
parent f10508db0b
commit 1a764bb8d7
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -20,7 +20,6 @@ use std::{
fmt, fmt,
future::ready, future::ready,
marker::PhantomData, marker::PhantomData,
sync::Arc,
task::{Context, Poll}, task::{Context, Poll},
}; };
use tower::{util::ServiceExt, ServiceBuilder}; use tower::{util::ServiceExt, ServiceBuilder};
@ -50,17 +49,7 @@ impl RouteId {
#[derive(Clone)] #[derive(Clone)]
pub struct Router<S> { pub struct Router<S> {
routes: S, routes: S,
node: MaybeSharedNode, node: Node<RouteId>,
}
// optimization that allows us to only clone the whole `Node` if we're actually
// mutating it while building the router. Once we've created a `MakeService` we
// no longer need to add routes and can `Arc` the node making it cheaper to
// clone
#[derive(Clone)]
enum MaybeSharedNode {
NotShared(Node<RouteId>),
Shared(Arc<Node<RouteId>>),
} }
impl Router<EmptyRouter> { impl Router<EmptyRouter> {
@ -71,7 +60,7 @@ impl Router<EmptyRouter> {
pub fn new() -> Self { pub fn new() -> Self {
Self { Self {
routes: EmptyRouter::not_found(), routes: EmptyRouter::not_found(),
node: MaybeSharedNode::NotShared(Node::new()), node: Node::new(),
} }
} }
} }
@ -181,7 +170,7 @@ impl<S> Router<S> {
{ {
let id = RouteId::next(); let id = RouteId::next();
if let Err(err) = self.update_node(|node| node.insert(path, id)) { if let Err(err) = self.node.insert(path, id) {
panic!("Invalid route: {}", err); panic!("Invalid route: {}", err);
} }
@ -332,7 +321,7 @@ impl<S> Router<S> {
format!("{}/*{}", path, NEST_TAIL_PARAM) format!("{}/*{}", path, NEST_TAIL_PARAM)
}; };
if let Err(err) = self.update_node(|node| node.insert(path, id)) { if let Err(err) = self.node.insert(path, id) {
panic!("Invalid route: {}", err); panic!("Invalid route: {}", err);
} }
@ -496,7 +485,7 @@ impl<S> Router<S> {
where where
S: Clone, S: Clone,
{ {
IntoMakeService::new(self.into_shared_node()) IntoMakeService::new(self)
} }
/// Convert this router into a [`MakeService`], that will store `C`'s /// Convert this router into a [`MakeService`], that will store `C`'s
@ -586,7 +575,7 @@ impl<S> Router<S> {
S: Clone, S: Clone,
C: Connected<Target>, C: Connected<Target>,
{ {
IntoMakeServiceWithConnectInfo::new(self.into_shared_node()) IntoMakeServiceWithConnectInfo::new(self)
} }
/// Merge two routers into one. /// Merge two routers into one.
@ -636,41 +625,6 @@ impl<S> Router<S> {
node: self.node, node: self.node,
} }
} }
fn update_node<F, T>(&mut self, f: F) -> T
where
F: FnOnce(&mut Node<RouteId>) -> T,
{
match &mut self.node {
MaybeSharedNode::NotShared(node) => f(node),
MaybeSharedNode::Shared(shared_node) => {
let mut node: Node<_> = Clone::clone(&*shared_node);
let result = f(&mut node);
self.node = MaybeSharedNode::NotShared(node);
result
}
}
}
fn get_node(&self) -> &Node<RouteId> {
match &self.node {
MaybeSharedNode::NotShared(node) => node,
MaybeSharedNode::Shared(shared_node) => &*shared_node,
}
}
fn into_shared_node(self) -> Self {
let node = match self.node {
MaybeSharedNode::NotShared(node) => MaybeSharedNode::Shared(Arc::new(node)),
MaybeSharedNode::Shared(shared_node) => {
MaybeSharedNode::Shared(Arc::clone(&shared_node))
}
};
Self {
routes: self.routes,
node,
}
}
} }
impl<ReqBody, ResBody, S> Service<Request<ReqBody>> for Router<S> impl<ReqBody, ResBody, S> Service<Request<ReqBody>> for Router<S>
@ -695,7 +649,7 @@ where
} }
let path = req.uri().path().to_string(); let path = req.uri().path().to_string();
if let Ok(match_) = self.get_node().at(&path) { if let Ok(match_) = self.node.at(&path) {
let id = *match_.value; let id = *match_.value;
req.extensions_mut().insert(id); req.extensions_mut().insert(id);