diff --git a/misc/metrics/CHANGELOG.md b/misc/metrics/CHANGELOG.md index c8eecc36e6f..58460094ae8 100644 --- a/misc/metrics/CHANGELOG.md +++ b/misc/metrics/CHANGELOG.md @@ -16,6 +16,10 @@ - Update to `libp2p-gossipsub` `v0.43.0`. +- Add `protocol_stack` metrics. See [PR 2982]. + +[PR 2982]: https://github.com/libp2p/rust-libp2p/pull/2982/ + # 0.10.0 - Update to `libp2p-swarm` `v0.40.0`. diff --git a/misc/metrics/examples/metrics/main.rs b/misc/metrics/examples/metrics/main.rs index ab1e041bcc4..687dba3f332 100644 --- a/misc/metrics/examples/metrics/main.rs +++ b/misc/metrics/examples/metrics/main.rs @@ -54,7 +54,7 @@ use futures::stream::StreamExt; use libp2p::core::Multiaddr; use libp2p::metrics::{Metrics, Recorder}; use libp2p::swarm::{NetworkBehaviour, SwarmEvent}; -use libp2p::{identity, ping, PeerId, Swarm}; +use libp2p::{identify, identity, ping, PeerId, Swarm}; use libp2p_swarm::keep_alive; use log::info; use prometheus_client::registry::Registry; @@ -68,11 +68,12 @@ fn main() -> Result<(), Box> { let local_key = identity::Keypair::generate_ed25519(); let local_peer_id = PeerId::from(local_key.public()); + let local_pub_key = local_key.public(); info!("Local peer id: {:?}", local_peer_id); let mut swarm = Swarm::without_executor( block_on(libp2p::development_transport(local_key))?, - Behaviour::default(), + Behaviour::new(local_pub_key), local_peer_id, ); @@ -95,6 +96,10 @@ fn main() -> Result<(), Box> { info!("{:?}", ping_event); metrics.record(&ping_event); } + SwarmEvent::Behaviour(BehaviourEvent::Identify(identify_event)) => { + info!("{:?}", identify_event); + metrics.record(&identify_event); + } swarm_event => { info!("{:?}", swarm_event); metrics.record(&swarm_event); @@ -109,8 +114,22 @@ fn main() -> Result<(), Box> { /// /// For illustrative purposes, this includes the [`keep_alive::Behaviour`]) behaviour so the ping actually happen /// and can be observed via the metrics. -#[derive(NetworkBehaviour, Default)] +#[derive(NetworkBehaviour)] struct Behaviour { + identify: identify::Behaviour, keep_alive: keep_alive::Behaviour, ping: ping::Behaviour, } + +impl Behaviour { + fn new(local_pub_key: libp2p::identity::PublicKey) -> Self { + Self { + ping: ping::Behaviour::default(), + identify: identify::Behaviour::new(identify::Config::new( + "/ipfs/0.1.0".into(), + local_pub_key, + )), + keep_alive: keep_alive::Behaviour::default(), + } + } +} diff --git a/misc/metrics/src/identify.rs b/misc/metrics/src/identify.rs index 8f91521713f..688c67a6190 100644 --- a/misc/metrics/src/identify.rs +++ b/misc/metrics/src/identify.rs @@ -18,9 +18,11 @@ // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. +use crate::protocol_stack; use libp2p_core::PeerId; -use prometheus_client::encoding::text::{EncodeMetric, Encoder}; +use prometheus_client::encoding::text::{Encode, EncodeMetric, Encoder}; use prometheus_client::metrics::counter::Counter; +use prometheus_client::metrics::family::Family; use prometheus_client::metrics::histogram::{exponential_buckets, Histogram}; use prometheus_client::metrics::MetricType; use prometheus_client::registry::Registry; @@ -36,6 +38,7 @@ pub struct Metrics { received_info_listen_addrs: Histogram, received_info_protocols: Histogram, sent: Counter, + listen_addresses: Family, } impl Metrics { @@ -100,6 +103,13 @@ impl Metrics { Box::new(sent.clone()), ); + let listen_addresses = Family::default(); + sub_registry.register( + "listen_addresses", + "Number of listen addresses for remote peer per protocol stack", + Box::new(listen_addresses.clone()), + ); + Self { protocols, error, @@ -108,6 +118,7 @@ impl Metrics { received_info_listen_addrs, received_info_protocols, sent, + listen_addresses, } } } @@ -167,6 +178,13 @@ impl super::Recorder for Metrics { .observe(info.protocols.len() as f64); self.received_info_listen_addrs .observe(info.listen_addrs.len() as f64); + for listen_addr in &info.listen_addrs { + self.listen_addresses + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(listen_addr), + }) + .inc(); + } } libp2p_identify::Event::Sent { .. } => { self.sent.inc(); @@ -190,6 +208,11 @@ impl super::Recorder>>>, diff --git a/misc/metrics/src/lib.rs b/misc/metrics/src/lib.rs index d42f6b3ffe0..351887260df 100644 --- a/misc/metrics/src/lib.rs +++ b/misc/metrics/src/lib.rs @@ -38,6 +38,7 @@ mod identify; mod kad; #[cfg(feature = "ping")] mod ping; +mod protocol_stack; #[cfg(feature = "relay")] mod relay; mod swarm; diff --git a/misc/metrics/src/protocol_stack.rs b/misc/metrics/src/protocol_stack.rs new file mode 100644 index 00000000000..1715b51f034 --- /dev/null +++ b/misc/metrics/src/protocol_stack.rs @@ -0,0 +1,27 @@ +use libp2p_core::multiaddr::Multiaddr; + +pub fn as_string(ma: &Multiaddr) -> String { + let len = ma + .protocol_stack() + .fold(0, |acc, proto| acc + proto.len() + 1); + let mut protocols = String::with_capacity(len); + for proto_tag in ma.protocol_stack() { + protocols.push('/'); + protocols.push_str(proto_tag); + } + protocols +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn ip6_tcp_wss_p2p() { + let ma = Multiaddr::try_from("/ip6/2001:8a0:7ac5:4201:3ac9:86ff:fe31:7095/tcp/8000/wss/p2p/QmcgpsyWgH8Y8ajJz1Cu72KnS5uo2Aa2LpzU7kinSupNKC").expect("testbad"); + + let protocol_stack = as_string(&ma); + + assert_eq!(protocol_stack, "/ip6/tcp/wss/p2p"); + } +} diff --git a/misc/metrics/src/swarm.rs b/misc/metrics/src/swarm.rs index e9c5a0493ce..c4fa8712d14 100644 --- a/misc/metrics/src/swarm.rs +++ b/misc/metrics/src/swarm.rs @@ -18,37 +18,38 @@ // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. +use crate::protocol_stack; use prometheus_client::encoding::text::Encode; use prometheus_client::metrics::counter::Counter; use prometheus_client::metrics::family::Family; use prometheus_client::registry::Registry; pub struct Metrics { - connections_incoming: Counter, + connections_incoming: Family, connections_incoming_error: Family, connections_established: Family, connections_closed: Family, - new_listen_addr: Counter, - expired_listen_addr: Counter, + new_listen_addr: Family, + expired_listen_addr: Family, - listener_closed: Counter, + listener_closed: Family, listener_error: Counter, dial_attempt: Counter, outgoing_connection_error: Family, - connected_to_banned_peer: Counter, + connected_to_banned_peer: Family, } impl Metrics { pub fn new(registry: &mut Registry) -> Self { let sub_registry = registry.sub_registry_with_prefix("swarm"); - let connections_incoming = Counter::default(); + let connections_incoming = Family::default(); sub_registry.register( "connections_incoming", - "Number of incoming connections", + "Number of incoming connections per address stack", Box::new(connections_incoming.clone()), ); @@ -59,21 +60,21 @@ impl Metrics { Box::new(connections_incoming_error.clone()), ); - let new_listen_addr = Counter::default(); + let new_listen_addr = Family::default(); sub_registry.register( "new_listen_addr", "Number of new listen addresses", Box::new(new_listen_addr.clone()), ); - let expired_listen_addr = Counter::default(); + let expired_listen_addr = Family::default(); sub_registry.register( "expired_listen_addr", "Number of expired listen addresses", Box::new(expired_listen_addr.clone()), ); - let listener_closed = Counter::default(); + let listener_closed = Family::default(); sub_registry.register( "listener_closed", "Number of listeners closed", @@ -101,7 +102,7 @@ impl Metrics { Box::new(outgoing_connection_error.clone()), ); - let connected_to_banned_peer = Counter::default(); + let connected_to_banned_peer = Family::default(); sub_registry.register( "connected_to_banned_peer", "Number of connection attempts to banned peer", @@ -146,6 +147,7 @@ impl super::Recorder super::Recorder { - self.connections_incoming.inc(); + libp2p_swarm::SwarmEvent::IncomingConnection { send_back_addr, .. } => { + self.connections_incoming + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(send_back_addr), + }) + .inc(); } - libp2p_swarm::SwarmEvent::IncomingConnectionError { error, .. } => { + libp2p_swarm::SwarmEvent::IncomingConnectionError { + error, + send_back_addr, + .. + } => { self.connections_incoming_error .get_or_create(&IncomingConnectionErrorLabels { error: error.into(), + protocols: protocol_stack::as_string(send_back_addr), }) .inc(); } @@ -221,17 +233,35 @@ impl super::Recorder { - self.connected_to_banned_peer.inc(); + libp2p_swarm::SwarmEvent::BannedPeer { endpoint, .. } => { + self.connected_to_banned_peer + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(endpoint.get_remote_address()), + }) + .inc(); } - libp2p_swarm::SwarmEvent::NewListenAddr { .. } => { - self.new_listen_addr.inc(); + libp2p_swarm::SwarmEvent::NewListenAddr { address, .. } => { + self.new_listen_addr + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(address), + }) + .inc(); } - libp2p_swarm::SwarmEvent::ExpiredListenAddr { .. } => { - self.expired_listen_addr.inc(); + libp2p_swarm::SwarmEvent::ExpiredListenAddr { address, .. } => { + self.expired_listen_addr + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(address), + }) + .inc(); } - libp2p_swarm::SwarmEvent::ListenerClosed { .. } => { - self.listener_closed.inc(); + libp2p_swarm::SwarmEvent::ListenerClosed { addresses, .. } => { + for address in addresses { + self.listener_closed + .get_or_create(&AddressLabels { + protocols: protocol_stack::as_string(address), + }) + .inc(); + } } libp2p_swarm::SwarmEvent::ListenerError { .. } => { self.listener_error.inc(); @@ -246,11 +276,18 @@ impl super::Recorder