#[allow(deprecated)]
use solana_gossip::{contact_info::ContactInfo, gossip_service::discover};
use {
clap::{
crate_description, crate_name, value_t, value_t_or_exit, App, AppSettings, Arg, ArgMatches,
SubCommand,
},
log::{error, info, warn},
solana_clap_utils::{
hidden_unless_forced,
input_parsers::{keypair_of, pubkeys_of},
input_validators::{is_keypair_or_ask_keyword, is_port, is_pubkey},
},
solana_pubkey::Pubkey,
solana_streamer::socket::SocketAddrSpace,
std::{
error,
net::{IpAddr, Ipv4Addr, SocketAddr},
process::exit,
time::Duration,
},
};
fn parse_matches() -> ArgMatches<'static> {
let shred_version_arg = Arg::with_name("shred_version")
.long("shred-version")
.value_name("VERSION")
.takes_value(true)
.default_value("0")
.help("Filter gossip nodes by this shred version");
let gossip_port_arg = clap::Arg::with_name("gossip_port")
.long("gossip-port")
.value_name("PORT")
.takes_value(true)
.validator(is_port)
.help("Gossip port number for the node");
let gossip_host_arg = clap::Arg::with_name("gossip_host")
.long("gossip-host")
.value_name("HOST")
.takes_value(true)
.validator(solana_net_utils::is_host)
.help("DEPRECATED: --gossip-host is no longer supported. Use --bind-address instead.");
let bind_address_arg = clap::Arg::with_name("bind_address")
.long("bind-address")
.value_name("HOST")
.takes_value(true)
.validator(solana_net_utils::is_host)
.help("IP address to bind the node to for gossip (replaces --gossip-host)");
App::new(crate_name!())
.about(crate_description!())
.version(solana_version::version!())
.setting(AppSettings::SubcommandRequiredElseHelp)
.arg(
Arg::with_name("allow_private_addr")
.long("allow-private-addr")
.takes_value(false)
.help("Allow contacting private ip addresses")
.hidden(hidden_unless_forced()),
)
.subcommand(
SubCommand::with_name("rpc-url")
.about("Get an RPC URL for the cluster")
.arg(
Arg::with_name("entrypoint")
.short("n")
.long("entrypoint")
.value_name("HOST:PORT")
.takes_value(true)
.required(true)
.validator(solana_net_utils::is_host_port)
.help("Rendezvous with the cluster at this entry point"),
)
.arg(
Arg::with_name("all")
.long("all")
.takes_value(false)
.help("Return all RPC URLs"),
)
.arg(
Arg::with_name("any")
.long("any")
.takes_value(false)
.conflicts_with("all")
.help("Return any RPC URL"),
)
.arg(
Arg::with_name("timeout")
.long("timeout")
.value_name("SECONDS")
.takes_value(true)
.default_value("15")
.help("Timeout in seconds"),
)
.arg(&shred_version_arg)
.arg(&gossip_port_arg)
.arg(&gossip_host_arg)
.arg(&bind_address_arg)
.setting(AppSettings::DisableVersion),
)
.subcommand(
SubCommand::with_name("spy")
.about("Monitor the gossip entrypoint")
.setting(AppSettings::DisableVersion)
.arg(
Arg::with_name("entrypoint")
.short("n")
.long("entrypoint")
.value_name("HOST:PORT")
.takes_value(true)
.validator(solana_net_utils::is_host_port)
.help("Rendezvous with the cluster at this entrypoint"),
)
.arg(
Arg::with_name("identity")
.short("i")
.long("identity")
.value_name("PATH")
.takes_value(true)
.validator(is_keypair_or_ask_keyword)
.help("Identity keypair [default: ephemeral keypair]"),
)
.arg(
Arg::with_name("num_nodes")
.short("N")
.long("num-nodes")
.value_name("NUM")
.takes_value(true)
.conflicts_with("num_nodes_exactly")
.help("Wait for at least NUM nodes to be visible"),
)
.arg(
Arg::with_name("num_nodes_exactly")
.short("E")
.long("num-nodes-exactly")
.value_name("NUM")
.takes_value(true)
.help("Wait for exactly NUM nodes to be visible"),
)
.arg(
Arg::with_name("node_pubkey")
.short("p")
.long("pubkey")
.value_name("PUBKEY")
.takes_value(true)
.validator(is_pubkey)
.multiple(true)
.help("Public key of a specific node to wait for"),
)
.arg(&shred_version_arg)
.arg(&gossip_port_arg)
.arg(&gossip_host_arg)
.arg(&bind_address_arg)
.arg(
Arg::with_name("timeout")
.long("timeout")
.value_name("SECONDS")
.takes_value(true)
.help("Maximum time to wait in seconds [default: wait forever]"),
),
)
.get_matches()
}
fn parse_bind_address(matches: &ArgMatches, entrypoint_addr: Option<SocketAddr>) -> IpAddr {
if let Some(bind_address) = matches.value_of("bind_address") {
solana_net_utils::parse_host(bind_address).unwrap_or_else(|e| {
eprintln!("failed to parse bind-address: {e}");
exit(1);
})
} else if let Some(gossip_host) = matches.value_of("gossip_host") {
warn!("--gossip-host is deprecated. Use --bind-address instead.");
solana_net_utils::parse_host(gossip_host).unwrap_or_else(|e| {
eprintln!("failed to parse gossip-host: {e}");
exit(1);
})
} else if let Some(entrypoint_addr) = entrypoint_addr {
solana_net_utils::get_public_ip_addr_with_binding(
&entrypoint_addr,
IpAddr::V4(Ipv4Addr::UNSPECIFIED),
)
.unwrap_or_else(|err| {
eprintln!("Failed to contact cluster entrypoint {entrypoint_addr}: {err}");
exit(1);
})
} else {
IpAddr::V4(Ipv4Addr::LOCALHOST)
}
}
#[allow(deprecated)]
fn process_spy_results(
timeout: Option<u64>,
validators: Vec<ContactInfo>,
num_nodes: Option<usize>,
num_nodes_exactly: Option<usize>,
pubkeys: Option<&[Pubkey]>,
) {
if timeout.is_some() {
if let Some(num) = num_nodes {
if validators.len() < num {
let add = if num_nodes_exactly.is_some() {
""
} else {
" or more"
};
eprintln!("Error: Insufficient validators discovered. Expecting {num}{add}",);
exit(1);
}
}
if let Some(nodes) = pubkeys {
for node in nodes {
if !validators.iter().any(|x| {
#[allow(deprecated)]
let pubkey = x.pubkey();
pubkey == node
}) {
eprintln!("Error: Could not find node {node:?}");
exit(1);
}
}
}
}
if let Some(num_nodes_exactly) = num_nodes_exactly {
if validators.len() > num_nodes_exactly {
eprintln!("Error: Extra nodes discovered. Expecting exactly {num_nodes_exactly}");
exit(1);
}
}
}
fn get_entrypoint_shred_version(entrypoint: &Option<SocketAddr>) -> Option<u16> {
let Some(entrypoint) = entrypoint else {
error!("cannot obtain shred-version without an entrypoint");
return None;
};
match solana_net_utils::get_cluster_shred_version(entrypoint) {
Err(err) => {
error!("get_cluster_shred_version failed: {entrypoint}, {err}");
None
}
Ok(0) => {
error!("entrypoint {entrypoint} returned shred-version zero");
None
}
Ok(shred_version) => {
info!("obtained shred-version {shred_version} from entrypoint: {entrypoint}");
Some(shred_version)
}
}
}
fn process_spy(matches: &ArgMatches, socket_addr_space: SocketAddrSpace) -> std::io::Result<()> {
let num_nodes_exactly = matches
.value_of("num_nodes_exactly")
.map(|num| num.to_string().parse().unwrap());
let num_nodes = matches
.value_of("num_nodes")
.map(|num| num.to_string().parse().unwrap())
.or(num_nodes_exactly);
let timeout = matches
.value_of("timeout")
.map(|secs| secs.to_string().parse().unwrap());
let pubkeys = pubkeys_of(matches, "node_pubkey");
let identity_keypair = keypair_of(matches, "identity");
let entrypoint_addr = parse_entrypoint(matches);
let gossip_addr = get_gossip_address(matches, entrypoint_addr);
let mut shred_version = value_t_or_exit!(matches, "shred_version", u16);
if shred_version == 0 {
shred_version = get_entrypoint_shred_version(&entrypoint_addr)
.expect("need non-zero shred-version to join the cluster");
}
let discover_timeout = Duration::from_secs(timeout.unwrap_or(u64::MAX));
#[allow(deprecated)]
let (_all_peers, validators) = discover(
identity_keypair,
entrypoint_addr.as_ref(),
num_nodes,
discover_timeout,
pubkeys.as_deref(), None, Some(&gossip_addr), shred_version,
socket_addr_space,
)?;
process_spy_results(
timeout,
validators,
num_nodes,
num_nodes_exactly,
pubkeys.as_deref(),
);
Ok(())
}
fn parse_entrypoint(matches: &ArgMatches) -> Option<SocketAddr> {
matches.value_of("entrypoint").map(|entrypoint| {
solana_net_utils::parse_host_port(entrypoint).unwrap_or_else(|e| {
eprintln!("failed to parse entrypoint address: {e}");
exit(1);
})
})
}
fn process_rpc_url(
matches: &ArgMatches,
socket_addr_space: SocketAddrSpace,
) -> std::io::Result<()> {
let any = matches.is_present("any");
let all = matches.is_present("all");
let timeout = value_t_or_exit!(matches, "timeout", u64);
let entrypoint_addr = parse_entrypoint(matches);
let gossip_addr = get_gossip_address(matches, entrypoint_addr);
let mut shred_version = value_t_or_exit!(matches, "shred_version", u16);
if shred_version == 0 {
shred_version = get_entrypoint_shred_version(&entrypoint_addr)
.expect("need non-zero shred-version to join the cluster");
}
#[allow(deprecated)]
let (_all_peers, validators) = discover(
None, entrypoint_addr.as_ref(),
Some(1), Duration::from_secs(timeout),
None, entrypoint_addr.as_ref(), Some(&gossip_addr), shred_version,
socket_addr_space,
)?;
let rpc_addrs: Vec<_> = validators
.iter()
.filter(|node| {
any || all || {
#[allow(deprecated)]
let addrs = node.gossip();
addrs
.map(|addr| Some(addr) == entrypoint_addr)
.unwrap_or_default()
}
})
.filter_map(
#[allow(deprecated)]
ContactInfo::rpc,
)
.filter(|addr| socket_addr_space.check(addr))
.collect();
if rpc_addrs.is_empty() {
eprintln!("No RPC URL found");
exit(1);
}
for rpc_addr in rpc_addrs {
println!("http://{rpc_addr}");
if any {
break;
}
}
Ok(())
}
fn get_gossip_address(matches: &ArgMatches, entrypoint_addr: Option<SocketAddr>) -> SocketAddr {
let bind_address = parse_bind_address(matches, entrypoint_addr);
SocketAddr::new(
bind_address,
value_t!(matches, "gossip_port", u16).unwrap_or_else(|_| {
solana_net_utils::find_available_port_in_range(
IpAddr::V4(Ipv4Addr::UNSPECIFIED),
(0, 1),
)
.expect("unable to find an available gossip port")
}),
)
}
fn main() -> Result<(), Box<dyn error::Error>> {
agave_logger::setup_with_default_filter();
let matches = parse_matches();
let socket_addr_space = SocketAddrSpace::new(matches.is_present("allow_private_addr"));
match matches.subcommand() {
("spy", Some(matches)) => {
process_spy(matches, socket_addr_space)?;
}
("rpc-url", Some(matches)) => {
process_rpc_url(matches, socket_addr_space)?;
}
_ => unreachable!(),
}
Ok(())
}