forest/cli/subcommands/
net_cmd.rs1use crate::libp2p::{Multiaddr, Protocol};
5use crate::rpc::{self, net::AddrInfo, prelude::*};
6use ahash::{HashMap, HashSet};
7use cid::multibase;
8use clap::Subcommand;
9use itertools::Itertools;
10
11use crate::cli::subcommands::cli_error_and_die;
12
13#[derive(Debug, Subcommand)]
14pub enum NetCommands {
15 Listen,
17 Info,
19 Peers {
21 #[arg(short, long)]
23 agent: bool,
24 },
25 Connect {
27 address: String,
29 },
30 Disconnect {
32 id: String,
34 },
35 Reachability,
37}
38
39impl NetCommands {
40 pub async fn run(self, client: rpc::Client) -> anyhow::Result<()> {
41 match self {
42 Self::Listen => {
43 let info = NetAddrsListen::call(&client, ()).await?;
44 let addresses: Vec<String> = info
45 .addrs
46 .iter()
47 .map(|addr| format!("{}/p2p/{}", addr, info.id))
48 .collect();
49 println!("{}", addresses.join("\n"));
50 Ok(())
51 }
52 Self::Info => {
53 let info = NetInfo::call(&client, ()).await?;
54 println!("forest libp2p swarm info:");
55 println!("num peers: {}", info.num_peers);
56 println!("num connections: {}", info.num_connections);
57 println!("num pending: {}", info.num_pending);
58 println!("num pending incoming: {}", info.num_pending_incoming);
59 println!("num pending outgoing: {}", info.num_pending_outgoing);
60 println!("num established: {}", info.num_established);
61 Ok(())
62 }
63 Self::Peers { agent } => {
64 let addrs = NetPeers::call(&client, ()).await?;
65 let peer_to_agents: HashMap<String, String> = if agent {
66 let agents = futures::future::join_all(
67 addrs
68 .iter()
69 .map(|info| NetAgentVersion::call(&client, (info.id.clone(),))),
70 )
71 .await
72 .into_iter()
73 .map(|res| res.unwrap_or_else(|_| "<agent unknown>".to_owned()));
74
75 HashMap::from_iter(addrs.iter().map(|info| info.id.to_owned()).zip(agents))
76 } else {
77 HashMap::default()
78 };
79
80 let output: Vec<String> = addrs
81 .into_iter()
82 .filter_map(|info| {
83 let addresses: Vec<String> = info
84 .addrs
85 .into_iter()
86 .filter(|addr| match addr.iter().next().unwrap() {
87 Protocol::Ip4(ip_addr) => !ip_addr.is_loopback(),
88 Protocol::Ip6(ip_addr) => !ip_addr.is_loopback(),
89 _ => true,
90 })
91 .map(|addr| addr.to_string())
92 .unique()
93 .collect();
94 if addresses.is_empty() {
95 return None;
96 }
97
98 let result = format!("{}, [{}]", info.id, addresses.join(", "));
99
100 if agent {
101 Some(
102 [
103 result,
104 peer_to_agents
105 .get(&info.id)
106 .cloned()
107 .unwrap_or_else(|| "<agent unknown>".to_owned()),
108 ]
109 .join(", "),
110 )
111 } else {
112 Some(result)
113 }
114 })
115 .collect();
116 println!("{}", output.join("\n"));
117 Ok(())
118 }
119 Self::Connect { address } => {
120 let addr: Multiaddr = address
121 .parse()
122 .map_err(|e| {
123 cli_error_and_die(format!("Error parsing multiaddr. Error was: {e}"), 1);
124 })
125 .expect("Parse provided multiaddr from string");
126
127 let mut id = "".to_owned();
128
129 for protocol in addr.iter() {
130 if let Protocol::P2p(p2p) = protocol {
131 id = multibase::encode(multibase::Base::Base58Btc, p2p.to_bytes());
132 id = id.split_off(1);
133 }
134 }
135
136 if id.is_empty() {
137 cli_error_and_die("Needs a /p2p/ protocol present in multiaddr", 1)
138 }
139
140 let addrs = HashSet::from_iter([addr]);
141 let addr_info = AddrInfo {
142 id: id.clone(),
143 addrs,
144 };
145
146 NetConnect::call(&client, (addr_info,)).await?;
147 println!("connect {id}: success");
148 Ok(())
149 }
150 Self::Disconnect { id } => {
151 NetDisconnect::call(&client, (id.clone(),)).await?;
152 println!("disconnect {id}: success");
153 Ok(())
154 }
155 Self::Reachability => {
156 let nat_status = NetAutoNatStatus::call(&client, ()).await?;
157 println!("AutoNAT status: {}", nat_status.reachability_as_str());
158 if let Some(public_addrs) = nat_status.public_addrs
159 && !public_addrs.is_empty()
160 {
161 println!("Public address: [{}]", public_addrs.join(" "));
164 }
165 Ok(())
166 }
167 }
168 }
169}