2019-11-13 20:10:38 -08:00
|
|
|
use clap::{crate_description, crate_name, value_t, App, Arg, ArgMatches};
|
2019-08-21 10:23:33 -07:00
|
|
|
use solana_core::gen_keys::GenKeys;
|
2019-12-16 13:05:17 -08:00
|
|
|
use solana_faucet::faucet::FAUCET_PORT;
|
2020-02-12 13:15:12 -08:00
|
|
|
use solana_sdk::signature::{read_keypair_file, Keypair};
|
2019-04-27 10:06:58 -07:00
|
|
|
use std::net::SocketAddr;
|
|
|
|
use std::process::exit;
|
|
|
|
use std::time::Duration;
|
2019-04-17 11:28:26 -07:00
|
|
|
|
|
|
|
pub struct Config {
|
2019-05-03 15:00:19 -07:00
|
|
|
pub entrypoint_addr: SocketAddr,
|
2019-12-16 13:05:17 -08:00
|
|
|
pub faucet_addr: SocketAddr,
|
2019-04-17 11:28:26 -07:00
|
|
|
pub identity: Keypair,
|
|
|
|
pub threads: usize,
|
|
|
|
pub num_nodes: usize,
|
|
|
|
pub duration: Duration,
|
2019-04-23 16:48:17 -07:00
|
|
|
pub transfer_delay: u64,
|
2019-04-17 11:28:26 -07:00
|
|
|
pub fund_amount: u64,
|
|
|
|
pub batch_size: usize,
|
2019-04-23 16:48:17 -07:00
|
|
|
pub chunk_size: usize,
|
2019-04-17 11:28:26 -07:00
|
|
|
pub account_groups: usize,
|
2019-06-13 11:51:35 -07:00
|
|
|
pub client_ids_and_stake_file: String,
|
|
|
|
pub write_to_client_file: bool,
|
|
|
|
pub read_from_client_file: bool,
|
2019-04-17 11:28:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Config {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
2019-05-03 15:00:19 -07:00
|
|
|
entrypoint_addr: SocketAddr::from(([127, 0, 0, 1], 8001)),
|
2019-12-16 13:05:17 -08:00
|
|
|
faucet_addr: SocketAddr::from(([127, 0, 0, 1], FAUCET_PORT)),
|
2019-04-17 11:28:26 -07:00
|
|
|
identity: Keypair::new(),
|
|
|
|
num_nodes: 1,
|
|
|
|
threads: 4,
|
|
|
|
duration: Duration::new(u64::max_value(), 0),
|
2019-04-23 16:48:17 -07:00
|
|
|
transfer_delay: 0,
|
2019-04-17 11:28:26 -07:00
|
|
|
fund_amount: 100_000,
|
|
|
|
batch_size: 100,
|
2019-04-23 16:48:17 -07:00
|
|
|
chunk_size: 100,
|
2019-04-17 11:28:26 -07:00
|
|
|
account_groups: 100,
|
2019-06-13 11:51:35 -07:00
|
|
|
client_ids_and_stake_file: String::new(),
|
|
|
|
write_to_client_file: false,
|
|
|
|
read_from_client_file: false,
|
2019-04-17 11:28:26 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-13 20:10:38 -08:00
|
|
|
pub fn build_args<'a, 'b>(version: &'b str) -> App<'a, 'b> {
|
2019-04-17 11:28:26 -07:00
|
|
|
App::new(crate_name!())
|
|
|
|
.about(crate_description!())
|
2019-11-13 20:10:38 -08:00
|
|
|
.version(version)
|
2019-04-17 11:28:26 -07:00
|
|
|
.arg(
|
2019-05-03 15:00:19 -07:00
|
|
|
Arg::with_name("entrypoint")
|
2019-04-17 11:28:26 -07:00
|
|
|
.short("n")
|
2019-05-03 15:00:19 -07:00
|
|
|
.long("entrypoint")
|
2019-04-17 11:28:26 -07:00
|
|
|
.value_name("HOST:PORT")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("127.0.0.1:8001")
|
2019-05-03 15:00:19 -07:00
|
|
|
.help("Cluster entry point; defaults to 127.0.0.1:8001"),
|
2019-04-17 11:28:26 -07:00
|
|
|
)
|
|
|
|
.arg(
|
2019-12-16 13:05:17 -08:00
|
|
|
Arg::with_name("faucet")
|
2019-04-17 11:28:26 -07:00
|
|
|
.short("d")
|
2019-12-16 13:05:17 -08:00
|
|
|
.long("faucet")
|
2019-04-17 11:28:26 -07:00
|
|
|
.value_name("HOST:PORT")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("127.0.0.1:9900")
|
2019-12-16 13:05:17 -08:00
|
|
|
.help("Location of the faucet; defaults to 127.0.0.1:9900"),
|
2019-04-17 11:28:26 -07:00
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("identity")
|
|
|
|
.short("i")
|
|
|
|
.long("identity")
|
|
|
|
.value_name("PATH")
|
|
|
|
.takes_value(true)
|
|
|
|
.help("File containing a client identity (keypair)"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("threads")
|
|
|
|
.long("threads")
|
|
|
|
.value_name("<threads>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
2019-04-23 16:48:17 -07:00
|
|
|
.default_value("1")
|
2019-04-17 11:28:26 -07:00
|
|
|
.help("Number of threads submitting transactions"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("num-nodes")
|
|
|
|
.long("num-nodes")
|
|
|
|
.value_name("NUM")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("1")
|
|
|
|
.help("Wait for NUM nodes to converge"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("duration")
|
|
|
|
.long("duration")
|
|
|
|
.value_name("SECS")
|
|
|
|
.takes_value(true)
|
|
|
|
.default_value("60")
|
|
|
|
.help("Seconds to run benchmark, then exit; default is forever"),
|
|
|
|
)
|
|
|
|
.arg(
|
2019-04-23 16:48:17 -07:00
|
|
|
Arg::with_name("transfer-delay")
|
|
|
|
.long("transfer-delay")
|
2019-04-17 11:28:26 -07:00
|
|
|
.value_name("<delay>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("0")
|
2019-04-23 16:48:17 -07:00
|
|
|
.help("Delay between each chunk"),
|
2019-04-17 11:28:26 -07:00
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("fund-amount")
|
|
|
|
.long("fund-amount")
|
|
|
|
.value_name("<fund>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("100000")
|
|
|
|
.help("Number of lamports to fund to each signer"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("batch-size")
|
|
|
|
.long("batch-size")
|
|
|
|
.value_name("<batch>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("1000")
|
2019-04-23 16:48:17 -07:00
|
|
|
.help("Number of transactions before the signer rolls over"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("chunk-size")
|
|
|
|
.long("chunk-size")
|
|
|
|
.value_name("<cunk>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.default_value("500")
|
|
|
|
.help("Number of transactions to generate and send at a time"),
|
2019-04-17 11:28:26 -07:00
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("account-groups")
|
|
|
|
.long("account-groups")
|
|
|
|
.value_name("<groups>")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
2019-04-23 16:48:17 -07:00
|
|
|
.default_value("10")
|
2019-04-17 11:28:26 -07:00
|
|
|
.help("Number of account groups to cycle for each batch"),
|
|
|
|
)
|
2019-06-13 11:51:35 -07:00
|
|
|
.arg(
|
|
|
|
Arg::with_name("write-client-keys")
|
|
|
|
.long("write-client-keys")
|
|
|
|
.value_name("FILENAME")
|
|
|
|
.takes_value(true)
|
|
|
|
.help("Generate client keys and stakes and write the list to YAML file"),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("read-client-keys")
|
|
|
|
.long("read-client-keys")
|
|
|
|
.value_name("FILENAME")
|
|
|
|
.takes_value(true)
|
|
|
|
.help("Read client keys and stakes from the YAML file"),
|
|
|
|
)
|
2019-04-17 11:28:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn extract_args<'a>(matches: &ArgMatches<'a>) -> Config {
|
|
|
|
let mut args = Config::default();
|
|
|
|
|
2019-11-12 12:37:13 -08:00
|
|
|
args.entrypoint_addr = solana_net_utils::parse_host_port(
|
|
|
|
matches.value_of("entrypoint").unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap_or_else(|e| {
|
|
|
|
eprintln!("failed to parse entrypoint address: {}", e);
|
|
|
|
exit(1)
|
|
|
|
});
|
2019-04-27 10:06:58 -07:00
|
|
|
|
2019-12-16 13:05:17 -08:00
|
|
|
args.faucet_addr = solana_net_utils::parse_host_port(matches.value_of("faucet").unwrap())
|
2019-04-27 10:06:58 -07:00
|
|
|
.unwrap_or_else(|e| {
|
2019-12-16 13:05:17 -08:00
|
|
|
eprintln!("failed to parse faucet address: {}", e);
|
2019-04-27 10:06:58 -07:00
|
|
|
exit(1)
|
|
|
|
});
|
2019-04-17 11:28:26 -07:00
|
|
|
|
|
|
|
if matches.is_present("identity") {
|
2019-10-10 16:01:03 -07:00
|
|
|
args.identity = read_keypair_file(matches.value_of("identity").unwrap())
|
2019-04-17 11:28:26 -07:00
|
|
|
.expect("can't read client identity");
|
|
|
|
} else {
|
|
|
|
args.identity = {
|
|
|
|
let seed = [42_u8; 32];
|
2019-04-18 13:37:20 -07:00
|
|
|
let mut rnd = GenKeys::new(seed);
|
|
|
|
rnd.gen_keypair()
|
2019-04-17 11:28:26 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
args.threads = value_t!(matches.value_of("threads"), usize).expect("Failed to parse threads");
|
|
|
|
args.num_nodes =
|
|
|
|
value_t!(matches.value_of("num-nodes"), usize).expect("Failed to parse num-nodes");
|
|
|
|
let duration = value_t!(matches.value_of("duration"), u64).expect("Failed to parse duration");
|
|
|
|
args.duration = Duration::from_secs(duration);
|
2019-04-23 16:48:17 -07:00
|
|
|
args.transfer_delay =
|
|
|
|
value_t!(matches.value_of("transfer-delay"), u64).expect("Failed to parse transfer-delay");
|
2019-04-17 11:28:26 -07:00
|
|
|
args.fund_amount =
|
|
|
|
value_t!(matches.value_of("fund-amount"), u64).expect("Failed to parse fund-amount");
|
|
|
|
args.batch_size =
|
|
|
|
value_t!(matches.value_of("batch-size"), usize).expect("Failed to parse batch-size");
|
2019-04-23 16:48:17 -07:00
|
|
|
args.chunk_size =
|
|
|
|
value_t!(matches.value_of("chunk-size"), usize).expect("Failed to parse chunk-size");
|
2019-04-17 11:28:26 -07:00
|
|
|
args.account_groups = value_t!(matches.value_of("account-groups"), usize)
|
|
|
|
.expect("Failed to parse account-groups");
|
|
|
|
|
2019-06-13 11:51:35 -07:00
|
|
|
if let Some(s) = matches.value_of("write-client-keys") {
|
|
|
|
args.write_to_client_file = true;
|
|
|
|
args.client_ids_and_stake_file = s.to_string();
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(s) = matches.value_of("read-client-keys") {
|
|
|
|
assert!(!args.write_to_client_file);
|
|
|
|
args.read_from_client_file = true;
|
|
|
|
args.client_ids_and_stake_file = s.to_string();
|
|
|
|
}
|
2019-04-17 11:28:26 -07:00
|
|
|
args
|
|
|
|
}
|