pretty print and ability to save file added
Signed-off-by: Uncle Stretch <uncle.stretch@ghostchain.io>
This commit is contained in:
		
							parent
							
								
									b3cd0d6386
								
							
						
					
					
						commit
						00579b38b2
					
				
							
								
								
									
										10
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										10
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -991,6 +991,7 @@ dependencies = [ | |||||||
|  "maxminddb", |  "maxminddb", | ||||||
|  "parity-scale-codec", |  "parity-scale-codec", | ||||||
|  "pin-project", |  "pin-project", | ||||||
|  |  "pretty-table", | ||||||
|  "primitive-types", |  "primitive-types", | ||||||
|  "thiserror 2.0.9", |  "thiserror 2.0.9", | ||||||
|  "tokio", |  "tokio", | ||||||
| @ -2432,6 +2433,15 @@ dependencies = [ | |||||||
|  "zerocopy", |  "zerocopy", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "pretty-table" | ||||||
|  | version = "0.1.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0ebabdeeb9dde45278a6fd4caf2ec417c8c8d81b75d4450dc8d8305c25a4ac1d" | ||||||
|  | dependencies = [ | ||||||
|  |  "thiserror 1.0.69", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "primitive-types" | name = "primitive-types" | ||||||
| version = "0.13.1" | version = "0.13.1" | ||||||
|  | |||||||
| @ -15,6 +15,7 @@ ip_network = "0.4.1" | |||||||
| libp2p = { version = "0.52.0", features = ["dns", "identify", "kad", "macros", "mdns", "noise", "ping", "tcp", "tokio", "yamux", "websocket", "request-response"] } | libp2p = { version = "0.52.0", features = ["dns", "identify", "kad", "macros", "mdns", "noise", "ping", "tcp", "tokio", "yamux", "websocket", "request-response"] } | ||||||
| maxminddb = "0.24.0" | maxminddb = "0.24.0" | ||||||
| pin-project = "1.1.7" | pin-project = "1.1.7" | ||||||
|  | pretty-table = "0.1.3" | ||||||
| primitive-types = { version = "0.13.1", default-features = false, features = ["codec", "scale-info", "serde"] } | primitive-types = { version = "0.13.1", default-features = false, features = ["codec", "scale-info", "serde"] } | ||||||
| thiserror = "2.0.9" | thiserror = "2.0.9" | ||||||
| tokio = { version = "1.42.0", features = ["macros", "time", "rt-multi-thread"] } | tokio = { version = "1.42.0", features = ["macros", "time", "rt-multi-thread"] } | ||||||
|  | |||||||
| @ -1,16 +1,18 @@ | |||||||
| use std::{collections::HashMap, net::IpAddr}; | use std::{collections::HashMap, net::IpAddr}; | ||||||
| 
 | 
 | ||||||
|  | use pretty_table::prelude::*; | ||||||
|  | use ip_network::IpNetwork; | ||||||
| use trust_dns_resolver::{ | use trust_dns_resolver::{ | ||||||
|     config::{ResolverConfig, ResolverOpts}, |     config::{ResolverConfig, ResolverOpts}, | ||||||
|     TokioAsyncResolver, |     TokioAsyncResolver, | ||||||
| }; | }; | ||||||
| use maxminddb::{geoip2::City, Reader as GeoIpReader}; | use maxminddb::{geoip2::City, Reader as GeoIpReader}; | ||||||
| use libp2p::{ | use libp2p::{ | ||||||
|     identify::Info, multiaddr::Protocol, 
 |     identify::Info, multiaddr::Protocol, Multiaddr, PeerId | ||||||
|     PeerId |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| pub struct Locator<'a> { | pub struct Locator<'a> { | ||||||
|  |     maybe_filename: &'a Option<String>, | ||||||
|     peers: &'a HashMap<&'a PeerId, &'a Info>, |     peers: &'a HashMap<&'a PeerId, &'a Info>, | ||||||
|     db: maxminddb::Reader<&'static [u8]>, |     db: maxminddb::Reader<&'static [u8]>, | ||||||
| } | } | ||||||
| @ -18,24 +20,39 @@ pub struct Locator<'a> { | |||||||
| impl<'a> Locator<'a> { | impl<'a> Locator<'a> { | ||||||
|     const CITY_DATA: &'static  [u8] = include_bytes!("../artifacts/GeoLite2-City.mmdb"); |     const CITY_DATA: &'static  [u8] = include_bytes!("../artifacts/GeoLite2-City.mmdb"); | ||||||
| 
 | 
 | ||||||
|     pub fn new(peers: &'a HashMap<&'a PeerId, &'a Info>) -> Self { |     pub fn new( | ||||||
|  |         peers: &'a HashMap<&'a PeerId, &'a Info>, 
 | ||||||
|  |         maybe_filename: &'a Option<String>, | ||||||
|  |     ) -> Self { | ||||||
|         Self { 
 |         Self { 
 | ||||||
|             peers, |             peers, | ||||||
|  |             maybe_filename, | ||||||
|             db: GeoIpReader::from_source(Self::CITY_DATA) |             db: GeoIpReader::from_source(Self::CITY_DATA) | ||||||
|                 .expect("City data is always valid; qed"), |                 .expect("City data is always valid; qed"), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     fn is_public_adress(&self, addr: &Multiaddr) -> bool { | ||||||
|  |         let ip = match addr.iter().next() { | ||||||
|  |             Some(Protocol::Ip4(ip)) => IpNetwork::from(ip), | ||||||
|  |             Some(Protocol::Ip6(ip)) => IpNetwork::from(ip), | ||||||
|  |             Some(Protocol::Dns(_)) | Some(Protocol::Dns4(_)) | Some(Protocol::Dns6(_)) => return true, | ||||||
|  |             _ => return false, | ||||||
|  |         }; | ||||||
|  |         ip.is_global() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     pub async fn locate_peers(&self) { |     pub async fn locate_peers(&self) { | ||||||
|         let mut geo_peers: HashMap<PeerId, String> = HashMap::new(); |         let mut geo_peers = Vec::new(); | ||||||
| 
 | 
 | ||||||
|         let resolver = TokioAsyncResolver::tokio( |         let resolver = TokioAsyncResolver::tokio( | ||||||
|             ResolverConfig::default(), |             ResolverConfig::default(), | ||||||
|             ResolverOpts::default(), |             ResolverOpts::default(), | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|         for (peer, info) in self.peers { |         for (_, info) in self.peers { | ||||||
|             for addr in &info.listen_addrs { |             for addr in &info.listen_addrs { | ||||||
|  |                 if !self.is_public_adress(addr) { continue; } | ||||||
|                 let located = match addr.iter().next() { |                 let located = match addr.iter().next() { | ||||||
|                     Some(Protocol::Ip4(ip)) => self.locate(IpAddr::V4(ip)), |                     Some(Protocol::Ip4(ip)) => self.locate(IpAddr::V4(ip)), | ||||||
|                     Some(Protocol::Ip6(ip)) => self.locate(IpAddr::V6(ip)), |                     Some(Protocol::Ip6(ip)) => self.locate(IpAddr::V6(ip)), | ||||||
| @ -50,14 +67,22 @@ impl<'a> Locator<'a> { | |||||||
|                     _ => continue, |                     _ => continue, | ||||||
|                 }; |                 }; | ||||||
| 
 | 
 | ||||||
|                 let Some(located) = located else { continue; }; |                 geo_peers.push(vec![ | ||||||
|                 geo_peers.insert(**peer, located); |                     addr.to_string(), 
 | ||||||
|  |                     located.unwrap_or_default(), | ||||||
|  |                 ]); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         for (peer, location) in geo_peers { |         match &self.maybe_filename { | ||||||
|             println!("\tPeer={peer} location={location}"); |             Some(filename) => { | ||||||
|  |                 if let Err(err) = write_table_to_file(filename.clone(), geo_peers) { | ||||||
|  |                     tracing::error!("Could not write to file {} because of: {err}", &filename); | ||||||
|                 } |                 } | ||||||
|  |                 println!("Results saved to {}", &filename); | ||||||
|  |             }, | ||||||
|  |             None => print_table!(geo_peers), | ||||||
|  |         }; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn locate(&self, ip: IpAddr) -> Option<String> { |     fn locate(&self, ip: IpAddr) -> Option<String> { | ||||||
|  | |||||||
| @ -29,6 +29,9 @@ struct Opts { | |||||||
|         value_parser = parse_duration, |         value_parser = parse_duration, | ||||||
|     )] |     )] | ||||||
|     timeout: std::time::Duration, |     timeout: std::time::Duration, | ||||||
|  | 
 | ||||||
|  |     #[clap(short, long, default_value = None, value_parser)] | ||||||
|  |     filename: Option<String>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn parse_duration(arg: &str) -> Result<std::time::Duration, std::num::ParseIntError> { | fn parse_duration(arg: &str) -> Result<std::time::Duration, std::num::ParseIntError> { | ||||||
| @ -48,6 +51,7 @@ async fn main() -> Result<(), Box<dyn std::error::Error>> { | |||||||
|         .with_genesis(args.genesis) |         .with_genesis(args.genesis) | ||||||
|         .with_bootnodes(args.bootnodes) |         .with_bootnodes(args.bootnodes) | ||||||
|         .with_timeout(args.timeout) |         .with_timeout(args.timeout) | ||||||
|  |         .with_filename(args.filename) | ||||||
|         .build()? |         .build()? | ||||||
|         .walk_around() |         .walk_around() | ||||||
|         .await |         .await | ||||||
|  | |||||||
| @ -6,14 +6,12 @@ use std::{ | |||||||
| 
 | 
 | ||||||
| use futures::StreamExt; | use futures::StreamExt; | ||||||
| use codec::Decode; | use codec::Decode; | ||||||
| use ip_network::IpNetwork; |  | ||||||
| use primitive_types::H256; | use primitive_types::H256; | ||||||
| use libp2p::{ | use libp2p::{ | ||||||
|     identify::Info, identity, kad::{ |     identify::Info, identity, kad::{ | ||||||
|         Event as KademliaEvent, GetClosestPeersError, GetClosestPeersOk, 
 |         Event as KademliaEvent, GetClosestPeersError, GetClosestPeersOk, 
 | ||||||
|         QueryId, QueryResult, |         QueryId, QueryResult, | ||||||
|     }, 
 |     }, 
 | ||||||
|     multiaddr::Protocol, 
 |  | ||||||
|     swarm::{Config as SwarmConfig, SwarmEvent}, 
 |     swarm::{Config as SwarmConfig, SwarmEvent}, 
 | ||||||
|     Multiaddr, PeerId, Swarm 
 |     Multiaddr, PeerId, Swarm 
 | ||||||
| }; | }; | ||||||
| @ -37,6 +35,7 @@ use super::p2p::{ | |||||||
| pub struct Walker { | pub struct Walker { | ||||||
|     genesis: String, |     genesis: String, | ||||||
|     timeout: Duration, |     timeout: Duration, | ||||||
|  |     maybe_filename: Option<String>, | ||||||
|     swarm: Swarm<Behaviour>, |     swarm: Swarm<Behaviour>, | ||||||
|     queries: HashSet<QueryId>, |     queries: HashSet<QueryId>, | ||||||
|     discovered_with_address: HashMap<PeerId, HashSet<Multiaddr>>, |     discovered_with_address: HashMap<PeerId, HashSet<Multiaddr>>, | ||||||
| @ -159,22 +158,12 @@ impl Walker { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn is_public_adress(&self, addr: &Multiaddr) -> bool { |  | ||||||
|         let ip = match addr.iter().next() { |  | ||||||
|             Some(Protocol::Ip4(ip)) => IpNetwork::from(ip), |  | ||||||
|             Some(Protocol::Ip6(ip)) => IpNetwork::from(ip), |  | ||||||
|             Some(Protocol::Dns(_)) | Some(Protocol::Dns4(_)) | Some(Protocol::Dns6(_)) => return true, |  | ||||||
|             _ => return false, |  | ||||||
|         }; |  | ||||||
|         ip.is_global() |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub async fn walk_around(&mut self) -> Result<(), Box<dyn Error>> { |     pub async fn walk_around(&mut self) -> Result<(), Box<dyn Error>> { | ||||||
|         let _ = tokio::time::timeout(self.timeout, self.inner_walk()).await; |         let _ = tokio::time::timeout(self.timeout, self.inner_walk()).await; | ||||||
| 
 | 
 | ||||||
|         println!("Number of dialed peers: {}", self.dialed_peers.len()); |         println!("Number of dialed peers    : {}", self.dialed_peers.len()); | ||||||
|         println!("Total peers discovered: {}", self.discovered_with_address.len()); |         println!("Total peers discovered    : {}", self.discovered_with_address.len()); | ||||||
|         println!("Peers with local iden.: {}", self.peer_details.len()); |         println!("Peers with local identity : {}", self.peer_details.len()); | ||||||
| 
 | 
 | ||||||
|         let infos: HashMap<_, _> = self |         let infos: HashMap<_, _> = self | ||||||
|             .peer_details |             .peer_details | ||||||
| @ -185,19 +174,10 @@ impl Walker { | |||||||
|                     .any(|stream_proto| stream_proto.as_ref().contains(&self.genesis)) |                     .any(|stream_proto| stream_proto.as_ref().contains(&self.genesis)) | ||||||
|             }) |             }) | ||||||
|             .collect(); |             .collect(); | ||||||
|         println!("Support correct genesis {}: {}", &self.genesis, infos.len()); |         println!("Support correct genesis   : {}", infos.len()); | ||||||
| 
 |  | ||||||
|         let peers_with_public: HashMap<_, _> = infos |  | ||||||
|             .iter() |  | ||||||
|             .filter(|(_, info)| info.listen_addrs |  | ||||||
|                 .iter() |  | ||||||
|                 .any(|a| self.is_public_adress(&a))) |  | ||||||
|             .collect(); |  | ||||||
|         println!("Peers with public address: {}", peers_with_public.len()); |  | ||||||
|         println!("Peers with private address: {}", infos.len() - peers_with_public.len()); |  | ||||||
|         println!("Peers with associated role: {}", self.peer_role.len()); |         println!("Peers with associated role: {}", self.peer_role.len()); | ||||||
| 
 | 
 | ||||||
|         Locator::new(&infos).locate_peers().await; |         Locator::new(&infos, &self.maybe_filename).locate_peers().await; | ||||||
| 
 | 
 | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| @ -207,6 +187,7 @@ pub struct WalkerBuilder { | |||||||
|     genesis: String, |     genesis: String, | ||||||
|     bootnodes: Vec<String>, |     bootnodes: Vec<String>, | ||||||
|     timeout: Duration, |     timeout: Duration, | ||||||
|  |     maybe_filename: Option<String>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for WalkerBuilder { | impl Default for WalkerBuilder { | ||||||
| @ -215,6 +196,7 @@ impl Default for WalkerBuilder { | |||||||
|             genesis: Default::default(), 
 |             genesis: Default::default(), 
 | ||||||
|             bootnodes: Default::default(), |             bootnodes: Default::default(), | ||||||
|             timeout: Default::default(), |             timeout: Default::default(), | ||||||
|  |             maybe_filename: None, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -255,7 +237,8 @@ impl WalkerBuilder { | |||||||
|             let discovery = DiscoveryBuilder::new() |             let discovery = DiscoveryBuilder::new() | ||||||
|                 .record_ttl(Some(Duration::from_secs(0))) |                 .record_ttl(Some(Duration::from_secs(0))) | ||||||
|                 .provider_ttl(Some(Duration::from_secs(0))) |                 .provider_ttl(Some(Duration::from_secs(0))) | ||||||
|                 .query_timeout(Duration::from_secs(5 * 60)) |                 //.query_timeout(Duration::from_secs(5 * 60))
 | ||||||
|  |                 .query_timeout(self.timeout) | ||||||
|                 .build(local_peer_id, &self.genesis); |                 .build(local_peer_id, &self.genesis); | ||||||
| 
 | 
 | ||||||
|             let peer_info = PeerBehaviour::new(local_key.public()); |             let peer_info = PeerBehaviour::new(local_key.public()); | ||||||
| @ -296,10 +279,16 @@ impl WalkerBuilder { | |||||||
|         self |         self | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     pub fn with_filename(mut self, maybe_filename: Option<String>) -> Self { | ||||||
|  |         self.maybe_filename = maybe_filename; | ||||||
|  |         self | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     pub fn build(&self) -> Result<Walker, Box<dyn Error>> { |     pub fn build(&self) -> Result<Walker, Box<dyn Error>> { | ||||||
|         let swarm = self.create_swarm()?; |         let swarm = self.create_swarm()?; | ||||||
|         Ok(Walker { |         Ok(Walker { | ||||||
|             swarm, |             swarm, | ||||||
|  |             maybe_filename: self.maybe_filename.clone(), | ||||||
|             genesis: self.genesis.clone(), |             genesis: self.genesis.clone(), | ||||||
|             timeout: self.timeout, |             timeout: self.timeout, | ||||||
|             queries: HashSet::with_capacity(1024), |             queries: HashSet::with_capacity(1024), | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user