route grooming fix

This commit is contained in:
John Smith 2022-12-15 18:41:44 -05:00
parent f0674e46d1
commit 8d80fbb228
4 changed files with 139 additions and 55 deletions

@ -1 +1 @@
Subproject commit c153eb3015d6d118e5d467865510d053ddd84533 Subproject commit b127b2d3c653fea163a776dd58b3798f28aeeee3

View File

@ -353,7 +353,6 @@ impl UI {
format!(" Error: {}", e), format!(" Error: {}", e),
color, color,
)); ));
return;
} }
} }
// save to history unless it's a duplicate // save to history unless it's a duplicate

View File

@ -188,6 +188,12 @@ impl RouteSpecDetail {
pub fn get_stats_mut(&mut self) -> &mut RouteStats { pub fn get_stats_mut(&mut self) -> &mut RouteStats {
&mut self.stats &mut self.stats
} }
pub fn is_published(&self) -> bool {
self.published
}
pub fn hop_count(&self) -> usize {
self.hops.len()
}
} }
/// The core representation of the RouteSpecStore that can be serialized /// The core representation of the RouteSpecStore that can be serialized
@ -1082,6 +1088,11 @@ impl RouteSpecStore {
avoid_node_ids: &[DHTKey], avoid_node_ids: &[DHTKey],
) -> Option<DHTKey> { ) -> Option<DHTKey> {
let cur_ts = get_timestamp(); let cur_ts = get_timestamp();
let mut routes = Vec::new();
// Get all valid routes, allow routes that need testing
// but definitely prefer routes that have been recently tested
for detail in &inner.content.details { for detail in &inner.content.details {
if detail.1.stability >= stability if detail.1.stability >= stability
&& detail.1.sequencing >= sequencing && detail.1.sequencing >= sequencing
@ -1089,7 +1100,6 @@ impl RouteSpecStore {
&& detail.1.hops.len() <= max_hop_count && detail.1.hops.len() <= max_hop_count
&& detail.1.directions.is_superset(directions) && detail.1.directions.is_superset(directions)
&& !detail.1.published && !detail.1.published
&& !detail.1.stats.needs_testing(cur_ts)
{ {
let mut avoid = false; let mut avoid = false;
for h in &detail.1.hops { for h in &detail.1.hops {
@ -1099,11 +1109,29 @@ impl RouteSpecStore {
} }
} }
if !avoid { if !avoid {
return Some(*detail.0); routes.push(detail);
} }
} }
} }
None
// Sort the routes by preference
routes.sort_by(|a, b| {
let a_needs_testing = a.1.stats.needs_testing(cur_ts);
let b_needs_testing = b.1.stats.needs_testing(cur_ts);
if !a_needs_testing && b_needs_testing {
return cmp::Ordering::Less;
}
if !b_needs_testing && a_needs_testing {
return cmp::Ordering::Greater;
}
let a_latency = a.1.stats.latency_stats().average;
let b_latency = b.1.stats.latency_stats().average;
a_latency.cmp(&b_latency)
});
// Return the best one if we got one
routes.first().map(|r| *r.0)
} }
/// List all allocated routes /// List all allocated routes

View File

@ -4,53 +4,30 @@ use futures_util::stream::{FuturesUnordered, StreamExt};
use futures_util::FutureExt; use futures_util::FutureExt;
use stop_token::future::FutureExt as StopFutureExt; use stop_token::future::FutureExt as StopFutureExt;
const BACKGROUND_SAFETY_ROUTE_COUNT: usize = 2;
impl RoutingTable { impl RoutingTable {
/// Keep private routes assigned and accessible /// Test set of routes and remove the ones that don't test clean
#[instrument(level = "trace", skip(self, stop_token), err)] #[instrument(level = "trace", skip(self, stop_token), err)]
pub(crate) async fn private_route_management_task_routine( async fn test_route_set(
self, &self,
stop_token: StopToken, stop_token: StopToken,
_last_ts: u64, routes_needing_testing: Vec<DHTKey>,
cur_ts: u64,
) -> EyreResult<()> { ) -> EyreResult<()> {
// Get our node's current node info and network class and do the right thing
let network_class = self
.get_network_class(RoutingDomain::PublicInternet)
.unwrap_or(NetworkClass::Invalid);
// If we don't know our network class then don't do this yet
if network_class == NetworkClass::Invalid {
return Ok(());
}
// Collect any routes that need that need testing
let rss = self.route_spec_store(); let rss = self.route_spec_store();
let mut routes_needing_testing = rss.list_allocated_routes(|k, v| { log_rtab!(debug "Testing routes: {:?}", routes_needing_testing);
let stats = v.get_stats();
if stats.needs_testing(cur_ts) {
return Some(*k);
} else {
return None;
}
});
let mut remote_routes_needing_testing = rss.list_remote_routes(|k, v| {
let stats = v.get_stats();
if stats.needs_testing(cur_ts) {
return Some(*k);
} else {
return None;
}
});
routes_needing_testing.append(&mut remote_routes_needing_testing);
// Test all the routes that need testing at the same time
#[derive(Default, Debug)] #[derive(Default, Debug)]
struct TestRouteContext { struct TestRouteContext {
failed: bool, failed: bool,
dead_routes: Vec<DHTKey>, dead_routes: Vec<DHTKey>,
} }
if !routes_needing_testing.is_empty() { if routes_needing_testing.is_empty() {
return Ok(());
}
// Test all the routes that need testing at the same time
let mut unord = FuturesUnordered::new(); let mut unord = FuturesUnordered::new();
let ctx = Arc::new(Mutex::new(TestRouteContext::default())); let ctx = Arc::new(Mutex::new(TestRouteContext::default()));
for r in routes_needing_testing { for r in routes_needing_testing {
@ -61,7 +38,7 @@ impl RoutingTable {
let success = match rss.test_route(&r).await { let success = match rss.test_route(&r).await {
Ok(v) => v, Ok(v) => v,
Err(e) => { Err(e) => {
log_rtab!(error "test route failed: {}", e); log_rtab!(error "Test route failed: {}", e);
ctx.lock().failed = true; ctx.lock().failed = true;
return; return;
} }
@ -87,8 +64,88 @@ impl RoutingTable {
log_rtab!(debug "Dead route: {}", &r); log_rtab!(debug "Dead route: {}", &r);
rss.release_route(r); rss.release_route(r);
} }
Ok(())
} }
/// Keep private routes assigned and accessible
#[instrument(level = "trace", skip(self, stop_token), err)]
pub(crate) async fn private_route_management_task_routine(
self,
stop_token: StopToken,
_last_ts: u64,
cur_ts: u64,
) -> EyreResult<()> {
// Get our node's current node info and network class and do the right thing
let network_class = self
.get_network_class(RoutingDomain::PublicInternet)
.unwrap_or(NetworkClass::Invalid);
// If we don't know our network class then don't do this yet
if network_class == NetworkClass::Invalid {
return Ok(());
}
// Test locally allocated routes first
// This may remove dead routes
let rss = self.route_spec_store();
let routes_needing_testing = rss.list_allocated_routes(|k, v| {
let stats = v.get_stats();
if stats.needs_testing(cur_ts) {
return Some(*k);
} else {
return None;
}
});
self.test_route_set(stop_token.clone(), routes_needing_testing)
.await?;
// Ensure we have a minimum of N allocated local, unpublished routes with the default number of hops
let default_route_hop_count =
self.with_config(|c| c.network.rpc.default_route_hop_count as usize);
let mut local_unpublished_route_count = 0usize;
rss.list_allocated_routes(|_k, v| {
if !v.is_published() && v.hop_count() == default_route_hop_count {
local_unpublished_route_count += 1;
}
Option::<()>::None
});
if local_unpublished_route_count < BACKGROUND_SAFETY_ROUTE_COUNT {
let routes_to_allocate = BACKGROUND_SAFETY_ROUTE_COUNT - local_unpublished_route_count;
// Newly allocated routes
let mut newly_allocated_routes = Vec::new();
for _n in 0..routes_to_allocate {
// Parameters here must be the default safety route spec
// These will be used by test_remote_route as well
if let Some(k) = rss.allocate_route(
Stability::default(),
Sequencing::default(),
default_route_hop_count,
DirectionSet::all(),
&[],
)? {
newly_allocated_routes.push(k);
}
}
// Immediately test them
self.test_route_set(stop_token.clone(), newly_allocated_routes)
.await?;
}
// Test remote routes next
let remote_routes_needing_testing = rss.list_remote_routes(|k, v| {
let stats = v.get_stats();
if stats.needs_testing(cur_ts) {
return Some(*k);
} else {
return None;
}
});
self.test_route_set(stop_token.clone(), remote_routes_needing_testing)
.await?;
// Send update (also may send updates for released routes done by other parts of the program) // Send update (also may send updates for released routes done by other parts of the program)
rss.send_route_update(); rss.send_route_update();