use clap::{crate_description, crate_name, AppSettings, Arg, ArgGroup, ArgMatches, SubCommand}; use console::style; use solana_clap_utils::{ input_validators::is_url, keypair::{ self, keypair_input, KeypairWithSource, ASK_SEED_PHRASE_ARG, SKIP_SEED_PHRASE_VALIDATION_ARG, }, }; use solana_cli::{ cli::{app, parse_command, process_command, CliCommandInfo, CliConfig, CliError}, config::{self, Config}, display::{println_name_value, println_name_value_or}, }; use solana_sdk::signature::read_keypair_file; use std::error; fn parse_settings(matches: &ArgMatches<'_>) -> Result> { let parse_args = match matches.subcommand() { ("config", Some(matches)) => match matches.subcommand() { ("get", Some(subcommand_matches)) => { if let Some(config_file) = matches.value_of("config_file") { let config = Config::load(config_file).unwrap_or_default(); if let Some(field) = subcommand_matches.value_of("specific_setting") { let (value, default_value) = match field { "url" => (config.url, CliConfig::default_json_rpc_url()), "keypair" => (config.keypair_path, CliConfig::default_keypair_path()), _ => unreachable!(), }; println_name_value_or(&format!("* {}:", field), &value, &default_value); } else { println_name_value("Wallet Config:", config_file); println_name_value_or( "* url:", &config.url, &CliConfig::default_json_rpc_url(), ); println_name_value_or( "* keypair:", &config.keypair_path, &CliConfig::default_keypair_path(), ); } } else { println!( "{} Either provide the `--config` arg or ensure home directory exists to use the default config location", style("No config file found.").bold() ); } false } ("set", Some(subcommand_matches)) => { if let Some(config_file) = matches.value_of("config_file") { let mut config = Config::load(config_file).unwrap_or_default(); if let Some(url) = subcommand_matches.value_of("json_rpc_url") { config.url = url.to_string(); } if let Some(keypair) = subcommand_matches.value_of("keypair") { config.keypair_path = keypair.to_string(); } config.save(config_file)?; println_name_value("Wallet Config Updated:", config_file); println_name_value("* url:", &config.url); println_name_value("* keypair:", &config.keypair_path); } else { println!( "{} Either provide the `--config` arg or ensure home directory exists to use the default config location", style("No config file found.").bold() ); } false } _ => unreachable!(), }, _ => true, }; Ok(parse_args) } pub fn parse_args(matches: &ArgMatches<'_>) -> Result> { let config = if let Some(config_file) = matches.value_of("config_file") { Config::load(config_file).unwrap_or_default() } else { Config::default() }; let json_rpc_url = if let Some(url) = matches.value_of("json_rpc_url") { url.to_string() } else if config.url != "" { config.url } else { let default = CliConfig::default(); default.json_rpc_url }; let CliCommandInfo { command, require_keypair, } = parse_command(&matches)?; let (keypair, keypair_path) = if require_keypair { let KeypairWithSource { keypair, source } = keypair_input(&matches, "keypair")?; match source { keypair::Source::File => ( keypair, Some(matches.value_of("keypair").unwrap().to_string()), ), keypair::Source::SeedPhrase => (keypair, None), keypair::Source::Generated => { let keypair_path = if config.keypair_path != "" { config.keypair_path } else { let default_keypair_path = CliConfig::default_keypair_path(); if !std::path::Path::new(&default_keypair_path).exists() { return Err(CliError::KeypairFileNotFound(format!( "Generate a new keypair at {} with `solana-keygen new`", default_keypair_path )) .into()); } default_keypair_path }; let keypair = read_keypair_file(&keypair_path).or_else(|err| { Err(CliError::BadParameter(format!( "{}: Unable to open keypair file: {}", err, keypair_path ))) })?; (keypair, Some(keypair_path)) } } } else { let default = CliConfig::default(); (default.keypair, None) }; Ok(CliConfig { command, json_rpc_url, keypair, keypair_path, rpc_client: None, verbose: matches.is_present("verbose"), }) } fn main() -> Result<(), Box> { solana_logger::setup(); let matches = app( crate_name!(), crate_description!(), solana_clap_utils::version!(), ) .arg({ let arg = Arg::with_name("config_file") .short("C") .long("config") .value_name("PATH") .takes_value(true) .global(true) .help("Configuration file to use"); if let Some(ref config_file) = *config::CONFIG_FILE { arg.default_value(&config_file) } else { arg } }) .arg( Arg::with_name("json_rpc_url") .short("u") .long("url") .value_name("URL") .takes_value(true) .global(true) .validator(is_url) .help("JSON RPC URL for the solana cluster"), ) .arg( Arg::with_name("keypair") .short("k") .long("keypair") .value_name("PATH") .global(true) .takes_value(true) .help("/path/to/id.json"), ) .arg( Arg::with_name("verbose") .long("verbose") .short("v") .global(true) .help("Show extra information header"), ) .arg( Arg::with_name(ASK_SEED_PHRASE_ARG.name) .long(ASK_SEED_PHRASE_ARG.long) .value_name("KEYPAIR NAME") .global(true) .takes_value(true) .possible_values(&["keypair"]) .help(ASK_SEED_PHRASE_ARG.help), ) .arg( Arg::with_name(SKIP_SEED_PHRASE_VALIDATION_ARG.name) .long(SKIP_SEED_PHRASE_VALIDATION_ARG.long) .global(true) .help(SKIP_SEED_PHRASE_VALIDATION_ARG.help), ) .subcommand( SubCommand::with_name("config") .about("Solana command-line tool configuration settings") .aliases(&["get", "set"]) .setting(AppSettings::SubcommandRequiredElseHelp) .subcommand( SubCommand::with_name("get") .about("Get current config settings") .arg( Arg::with_name("specific_setting") .index(1) .value_name("CONFIG_FIELD") .takes_value(true) .possible_values(&["url", "keypair"]) .help("Return a specific config setting"), ), ) .subcommand( SubCommand::with_name("set") .about("Set a config setting") .group( ArgGroup::with_name("config_settings") .args(&["json_rpc_url", "keypair"]) .multiple(true) .required(true), ), ), ) .get_matches(); if parse_settings(&matches)? { let config = parse_args(&matches)?; let result = process_command(&config)?; println!("{}", result); } Ok(()) }