Log peer set ready and unready peers

* warn: if there are no peers at all
* info: if there are no ready peers
* trace: the number of ready and unready peers for every request

Log at most one warn or info log per minute, to avoid flooding the
terminal with log lines. Suppress warn and info logs for the first
minute, while the peer set is starting up.
This commit is contained in:
teor 2020-11-30 16:41:14 +10:00 committed by Deirdre Connolly
parent 2dd0fa0e07
commit 4d5ea4897c
1 changed files with 39 additions and 0 deletions

View File

@ -7,6 +7,7 @@ use std::{
marker::PhantomData, marker::PhantomData,
pin::Pin, pin::Pin,
task::{Context, Poll}, task::{Context, Poll},
time::Instant,
}; };
use futures::{ use futures::{
@ -100,6 +101,8 @@ where
/// the `PeerSet` propagate errors from background tasks back to the user /// the `PeerSet` propagate errors from background tasks back to the user
guards: futures::stream::FuturesUnordered<JoinHandle<Result<(), BoxError>>>, guards: futures::stream::FuturesUnordered<JoinHandle<Result<(), BoxError>>>,
inventory_registry: InventoryRegistry, inventory_registry: InventoryRegistry,
/// The last time we logged a message about the peer set size
last_peer_log: Option<Instant>,
} }
impl<D> PeerSet<D> impl<D> PeerSet<D>
@ -128,6 +131,7 @@ where
guards: futures::stream::FuturesUnordered::new(), guards: futures::stream::FuturesUnordered::new(),
handle_rx, handle_rx,
inventory_registry: InventoryRegistry::new(inv_stream), inventory_registry: InventoryRegistry::new(inv_stream),
last_peer_log: None,
} }
} }
@ -343,6 +347,40 @@ where
.boxed() .boxed()
} }
fn log_peer_set_size(&mut self) {
let ready_services_len = self.ready_services.len();
let unready_services_len = self.unready_services.len();
trace!(ready_peers = ?ready_services_len, unready_peers = ?unready_services_len);
if ready_services_len > 0 {
return;
}
// These logs are designed to be human-readable in a terminal, at the
// default Zebra log level. If you need to know the peer set size for
// every request, use the trace-level logs, or the metrics exporter.
if let Some(last_peer_log) = self.last_peer_log {
// Avoid duplicate peer set logs
if Instant::now().duration_since(last_peer_log).as_secs() < 60 {
return;
}
} else {
// Suppress initial logs until the peer set has started up.
// There can be multiple initial requests before the first peer is
// ready.
self.last_peer_log = Some(Instant::now());
return;
}
self.last_peer_log = Some(Instant::now());
if unready_services_len == 0 {
warn!("network request with no peer connections. Hint: check your network connection. Configure fast and reliable initial peers");
} else {
info!("network request with no ready peers: finding more peers, waiting for {} peers to answer requests. Hint: configure fast and reliable initial peers",
unready_services_len);
}
}
fn update_metrics(&self) { fn update_metrics(&self) {
let num_ready = self.ready_services.len(); let num_ready = self.ready_services.len();
let num_unready = self.unready_services.len(); let num_unready = self.unready_services.len();
@ -374,6 +412,7 @@ where
self.inventory_registry.poll_inventory(cx)?; self.inventory_registry.poll_inventory(cx)?;
self.poll_unready(cx); self.poll_unready(cx);
self.log_peer_set_size();
self.update_metrics(); self.update_metrics();
loop { loop {