Clean up solana-stake-accounts (#9211)

* Resolve pubkey/keypair args in a separate module

* Rename CommandConfig to Args
This commit is contained in:
Greg Fitzgerald 2020-03-31 21:47:43 -06:00 committed by GitHub
parent 74774dd44f
commit 743b8cddf9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 570 additions and 534 deletions

View File

@ -0,0 +1,305 @@
use crate::args::{
Args, AuthorizeArgs, Command, CountArgs, MoveArgs, NewArgs, QueryArgs, RebaseArgs,
};
use clap::{value_t_or_exit, App, Arg, ArgMatches, SubCommand};
use solana_clap_utils::input_validators::{is_amount, is_valid_pubkey, is_valid_signer};
use solana_cli_config::CONFIG_FILE;
use solana_sdk::native_token::sol_to_lamports;
use std::ffi::OsString;
use std::process::exit;
fn fee_payer_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("fee_payer")
.long("fee-payer")
.required(true)
.takes_value(true)
.value_name("KEYPAIR")
.validator(is_valid_signer)
.help("Fee payer")
}
fn funding_keypair_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("funding_keypair")
.required(true)
.takes_value(true)
.value_name("FUNDING_KEYPAIR")
.validator(is_valid_signer)
.help("Keypair to fund accounts")
}
fn base_pubkey_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("base_pubkey")
.required(true)
.takes_value(true)
.value_name("BASE_PUBKEY")
.validator(is_valid_pubkey)
.help("Public key which stake account addresses are derived from")
}
fn new_base_keypair_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("new_base_keypair")
.required(true)
.takes_value(true)
.value_name("NEW_BASE_KEYPAIR")
.validator(is_valid_signer)
.help("New keypair which stake account addresses are derived from")
}
fn stake_authority_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("stake_authority")
.long("stake-authority")
.required(true)
.takes_value(true)
.value_name("KEYPAIR")
.validator(is_valid_signer)
.help("Stake authority")
}
fn withdraw_authority_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("withdraw_authority")
.long("withdraw-authority")
.required(true)
.takes_value(true)
.value_name("KEYPAIR")
.validator(is_valid_signer)
.help("Withdraw authority")
}
fn new_stake_authority_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("new_stake_authority")
.long("new-stake-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("New stake authority")
}
fn new_withdraw_authority_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("new_withdraw_authority")
.long("new-withdraw-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("New withdraw authority")
}
fn num_accounts_arg<'a, 'b>() -> Arg<'a, 'b> {
Arg::with_name("num_accounts")
.long("num-accounts")
.required(true)
.takes_value(true)
.value_name("NUMBER")
.help("Number of derived stake accounts")
}
pub(crate) fn get_matches<'a, I, T>(args: I) -> ArgMatches<'a>
where
I: IntoIterator<Item = T>,
T: Into<OsString> + Clone,
{
let default_config_file = CONFIG_FILE.as_ref().unwrap();
App::new("solana-stake-accounts")
.about("about")
.version("version")
.arg(
Arg::with_name("config_file")
.long("config")
.takes_value(true)
.value_name("FILEPATH")
.default_value(default_config_file)
.help("Config file"),
)
.arg(
Arg::with_name("url")
.long("url")
.global(true)
.takes_value(true)
.value_name("URL")
.help("RPC entrypoint address. i.e. http://devnet.solana.com"),
)
.subcommand(
SubCommand::with_name("new")
.about("Create derived stake accounts")
.arg(fee_payer_arg())
.arg(funding_keypair_arg().index(1))
.arg(
Arg::with_name("base_keypair")
.required(true)
.index(2)
.takes_value(true)
.value_name("BASE_KEYPAIR")
.validator(is_valid_signer)
.help("Keypair which stake account addresses are derived from"),
)
.arg(
Arg::with_name("amount")
.required(true)
.index(3)
.takes_value(true)
.value_name("AMOUNT")
.validator(is_amount)
.help("Amount to move into the new stake accounts, in SOL"),
)
.arg(
Arg::with_name("stake_authority")
.long("stake-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("Stake authority"),
)
.arg(
Arg::with_name("withdraw_authority")
.long("withdraw-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("Withdraw authority"),
)
.arg(
Arg::with_name("index")
.long("index")
.takes_value(true)
.default_value("0")
.value_name("NUMBER")
.help("Index of the derived account to create"),
),
)
.subcommand(
SubCommand::with_name("count")
.about("Count derived stake accounts")
.arg(base_pubkey_arg().index(1)),
)
.subcommand(
SubCommand::with_name("addresses")
.about("Show public keys of all derived stake accounts")
.arg(base_pubkey_arg().index(1))
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("balance")
.about("Sum balances of all derived stake accounts")
.arg(base_pubkey_arg().index(1))
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("authorize")
.about("Set new authorities in all derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(stake_authority_arg())
.arg(withdraw_authority_arg())
.arg(new_stake_authority_arg())
.arg(new_withdraw_authority_arg())
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("rebase")
.about("Relocate derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(new_base_keypair_arg().index(2))
.arg(stake_authority_arg())
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("move")
.about("Rebase and set new authorities in all derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(new_base_keypair_arg().index(2))
.arg(stake_authority_arg())
.arg(withdraw_authority_arg())
.arg(new_stake_authority_arg())
.arg(new_withdraw_authority_arg())
.arg(num_accounts_arg()),
)
.get_matches_from(args)
}
fn parse_new_args(matches: &ArgMatches<'_>) -> NewArgs<String, String> {
NewArgs {
fee_payer: value_t_or_exit!(matches, "fee_payer", String),
funding_keypair: value_t_or_exit!(matches, "funding_keypair", String),
lamports: sol_to_lamports(value_t_or_exit!(matches, "amount", f64)),
base_keypair: value_t_or_exit!(matches, "base_keypair", String),
stake_authority: value_t_or_exit!(matches, "stake_authority", String),
withdraw_authority: value_t_or_exit!(matches, "withdraw_authority", String),
index: value_t_or_exit!(matches, "index", usize),
}
}
fn parse_count_args(matches: &ArgMatches<'_>) -> CountArgs<String> {
CountArgs {
base_pubkey: value_t_or_exit!(matches, "base_pubkey", String),
}
}
fn parse_query_args(matches: &ArgMatches<'_>) -> QueryArgs<String> {
QueryArgs {
base_pubkey: value_t_or_exit!(matches, "base_pubkey", String),
num_accounts: value_t_or_exit!(matches, "num_accounts", usize),
}
}
fn parse_authorize_args(matches: &ArgMatches<'_>) -> AuthorizeArgs<String, String> {
AuthorizeArgs {
fee_payer: value_t_or_exit!(matches, "fee_payer", String),
base_pubkey: value_t_or_exit!(matches, "base_pubkey", String),
stake_authority: value_t_or_exit!(matches, "stake_authority", String),
withdraw_authority: value_t_or_exit!(matches, "withdraw_authority", String),
new_stake_authority: value_t_or_exit!(matches, "new_stake_authority", String),
new_withdraw_authority: value_t_or_exit!(matches, "new_withdraw_authority", String),
num_accounts: value_t_or_exit!(matches, "num_accounts", usize),
}
}
fn parse_rebase_args(matches: &ArgMatches<'_>) -> RebaseArgs<String, String> {
RebaseArgs {
fee_payer: value_t_or_exit!(matches, "fee_payer", String),
base_pubkey: value_t_or_exit!(matches, "base_pubkey", String),
new_base_keypair: value_t_or_exit!(matches, "new_base_keypair", String),
stake_authority: value_t_or_exit!(matches, "stake_authority", String),
num_accounts: value_t_or_exit!(matches, "num_accounts", usize),
}
}
fn parse_move_args(matches: &ArgMatches<'_>) -> MoveArgs<String, String> {
MoveArgs {
rebase_args: parse_rebase_args(matches),
authorize_args: parse_authorize_args(matches),
}
}
pub(crate) fn parse_args<I, T>(args: I) -> Args<String, String>
where
I: IntoIterator<Item = T>,
T: Into<OsString> + Clone,
{
let matches = get_matches(args);
let config_file = matches.value_of("config_file").unwrap().to_string();
let url = matches.value_of("url").map(|x| x.to_string());
let command = match matches.subcommand() {
("new", Some(matches)) => Command::New(parse_new_args(matches)),
("count", Some(matches)) => Command::Count(parse_count_args(matches)),
("addresses", Some(matches)) => Command::Addresses(parse_query_args(matches)),
("balance", Some(matches)) => Command::Balance(parse_query_args(matches)),
("authorize", Some(matches)) => Command::Authorize(parse_authorize_args(matches)),
("rebase", Some(matches)) => Command::Rebase(parse_rebase_args(matches)),
("move", Some(matches)) => Command::Move(Box::new(parse_move_args(matches))),
_ => {
eprintln!("{}", matches.usage());
exit(1);
}
};
Args {
config_file,
url,
command,
}
}

View File

@ -1,382 +1,233 @@
use clap::{value_t_or_exit, App, Arg, ArgMatches, SubCommand}; use clap::ArgMatches;
use solana_clap_utils::input_validators::{is_amount, is_valid_pubkey, is_valid_signer}; use solana_clap_utils::keypair::{pubkey_from_path, signer_from_path};
use solana_cli_config::CONFIG_FILE; use solana_remote_wallet::remote_wallet::{maybe_wallet_manager, RemoteWalletManager};
use solana_sdk::native_token::sol_to_lamports; use solana_sdk::{pubkey::Pubkey, signature::Signer};
use std::ffi::OsString; use std::error::Error;
use std::process::exit; use std::sync::Arc;
pub(crate) struct NewCommandConfig { pub(crate) struct NewArgs<P, K> {
pub fee_payer: String, pub fee_payer: K,
pub funding_keypair: String, pub funding_keypair: K,
pub base_keypair: String, pub base_keypair: K,
pub lamports: u64, pub lamports: u64,
pub stake_authority: String, pub stake_authority: P,
pub withdraw_authority: String, pub withdraw_authority: P,
pub index: usize, pub index: usize,
} }
pub(crate) struct CountCommandConfig { pub(crate) struct CountArgs<P> {
pub base_pubkey: String, pub base_pubkey: P,
} }
pub(crate) struct QueryCommandConfig { pub(crate) struct QueryArgs<P> {
pub base_pubkey: String, pub base_pubkey: P,
pub num_accounts: usize, pub num_accounts: usize,
} }
pub(crate) struct AuthorizeCommandConfig { pub(crate) struct AuthorizeArgs<P, K> {
pub fee_payer: String, pub fee_payer: K,
pub base_pubkey: String, pub base_pubkey: P,
pub stake_authority: String, pub stake_authority: K,
pub withdraw_authority: String, pub withdraw_authority: K,
pub new_stake_authority: String, pub new_stake_authority: P,
pub new_withdraw_authority: String, pub new_withdraw_authority: P,
pub num_accounts: usize, pub num_accounts: usize,
} }
pub(crate) struct RebaseCommandConfig { pub(crate) struct RebaseArgs<P, K> {
pub fee_payer: String, pub fee_payer: K,
pub base_pubkey: String, pub base_pubkey: P,
pub new_base_keypair: String, pub new_base_keypair: K,
pub stake_authority: String, pub stake_authority: K,
pub num_accounts: usize, pub num_accounts: usize,
} }
pub(crate) struct MoveCommandConfig { pub(crate) struct MoveArgs<P, K> {
pub rebase_config: RebaseCommandConfig, pub rebase_args: RebaseArgs<P, K>,
pub authorize_config: AuthorizeCommandConfig, pub authorize_args: AuthorizeArgs<P, K>,
} }
pub(crate) enum Command { pub(crate) enum Command<P, K> {
New(NewCommandConfig), New(NewArgs<P, K>),
Count(CountCommandConfig), Count(CountArgs<P>),
Addresses(QueryCommandConfig), Addresses(QueryArgs<P>),
Balance(QueryCommandConfig), Balance(QueryArgs<P>),
Authorize(AuthorizeCommandConfig), Authorize(AuthorizeArgs<P, K>),
Rebase(RebaseCommandConfig), Rebase(RebaseArgs<P, K>),
Move(Box<MoveCommandConfig>), Move(Box<MoveArgs<P, K>>),
} }
pub(crate) struct CommandConfig { pub(crate) struct Args<P, K> {
pub config_file: String, pub config_file: String,
pub url: Option<String>, pub url: Option<String>,
pub command: Command, pub command: Command<P, K>,
} }
fn fee_payer_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_stake_authority(
Arg::with_name("fee_payer") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.long("fee-payer") key_url: &str,
.required(true) ) -> Result<Box<dyn Signer>, Box<dyn Error>> {
.takes_value(true) let matches = ArgMatches::default();
.value_name("KEYPAIR") signer_from_path(&matches, key_url, "stake authority", wallet_manager)
.validator(is_valid_signer)
.help("Fee payer")
} }
fn funding_keypair_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_withdraw_authority(
Arg::with_name("funding_keypair") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.required(true) key_url: &str,
.takes_value(true) ) -> Result<Box<dyn Signer>, Box<dyn Error>> {
.value_name("FUNDING_KEYPAIR") let matches = ArgMatches::default();
.validator(is_valid_signer) signer_from_path(&matches, key_url, "withdraw authority", wallet_manager)
.help("Keypair to fund accounts")
} }
fn base_pubkey_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_new_stake_authority(
Arg::with_name("base_pubkey") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.required(true) key_url: &str,
.takes_value(true) ) -> Result<Pubkey, Box<dyn Error>> {
.value_name("BASE_PUBKEY") let matches = ArgMatches::default();
.validator(is_valid_pubkey) pubkey_from_path(&matches, key_url, "new stake authority", wallet_manager)
.help("Public key which stake account addresses are derived from")
} }
fn new_base_keypair_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_new_withdraw_authority(
Arg::with_name("new_base_keypair") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.required(true) key_url: &str,
.takes_value(true) ) -> Result<Pubkey, Box<dyn Error>> {
.value_name("NEW_BASE_KEYPAIR") let matches = ArgMatches::default();
.validator(is_valid_signer) pubkey_from_path(&matches, key_url, "new withdraw authority", wallet_manager)
.help("New keypair which stake account addresses are derived from")
} }
fn stake_authority_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_fee_payer(
Arg::with_name("stake_authority") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.long("stake-authority") key_url: &str,
.required(true) ) -> Result<Box<dyn Signer>, Box<dyn Error>> {
.takes_value(true) let matches = ArgMatches::default();
.value_name("KEYPAIR") signer_from_path(&matches, key_url, "fee-payer", wallet_manager)
.validator(is_valid_signer)
.help("Stake authority")
} }
fn withdraw_authority_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_base_pubkey(
Arg::with_name("withdraw_authority") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.long("withdraw-authority") key_url: &str,
.required(true) ) -> Result<Pubkey, Box<dyn Error>> {
.takes_value(true) let matches = ArgMatches::default();
.value_name("KEYPAIR") pubkey_from_path(&matches, key_url, "base pubkey", wallet_manager)
.validator(is_valid_signer)
.help("Withdraw authority")
} }
fn new_stake_authority_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_new_base_keypair(
Arg::with_name("new_stake_authority") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.long("new-stake-authority") key_url: &str,
.required(true) ) -> Result<Box<dyn Signer>, Box<dyn Error>> {
.takes_value(true) let matches = ArgMatches::default();
.value_name("PUBKEY") signer_from_path(&matches, key_url, "new base pubkey", wallet_manager)
.validator(is_valid_pubkey)
.help("New stake authority")
} }
fn new_withdraw_authority_arg<'a, 'b>() -> Arg<'a, 'b> { fn resolve_authorize_args(
Arg::with_name("new_withdraw_authority") wallet_manager: Option<&Arc<RemoteWalletManager>>,
.long("new-withdraw-authority") args: &AuthorizeArgs<String, String>,
.required(true) ) -> Result<AuthorizeArgs<Pubkey, Box<dyn Signer>>, Box<dyn Error>> {
.takes_value(true) let resolved_args = AuthorizeArgs {
.value_name("PUBKEY") fee_payer: resolve_fee_payer(wallet_manager, &args.fee_payer)?,
.validator(is_valid_pubkey) base_pubkey: resolve_base_pubkey(wallet_manager, &args.base_pubkey)?,
.help("New withdraw authority") stake_authority: resolve_stake_authority(wallet_manager, &args.stake_authority)?,
} withdraw_authority: resolve_withdraw_authority(wallet_manager, &args.withdraw_authority)?,
new_stake_authority: resolve_new_stake_authority(
fn num_accounts_arg<'a, 'b>() -> Arg<'a, 'b> { wallet_manager,
Arg::with_name("num_accounts") &args.new_stake_authority,
.long("num-accounts") )?,
.required(true) new_withdraw_authority: resolve_new_withdraw_authority(
.takes_value(true) wallet_manager,
.value_name("NUMBER") &args.new_withdraw_authority,
.help("Number of derived stake accounts") )?,
} num_accounts: args.num_accounts,
pub(crate) fn get_matches<'a, I, T>(args: I) -> ArgMatches<'a>
where
I: IntoIterator<Item = T>,
T: Into<OsString> + Clone,
{
let default_config_file = CONFIG_FILE.as_ref().unwrap();
App::new("solana-stake-accounts")
.about("about")
.version("version")
.arg(
Arg::with_name("config_file")
.long("config")
.takes_value(true)
.value_name("FILEPATH")
.default_value(default_config_file)
.help("Config file"),
)
.arg(
Arg::with_name("url")
.long("url")
.global(true)
.takes_value(true)
.value_name("URL")
.help("RPC entrypoint address. i.e. http://devnet.solana.com"),
)
.subcommand(
SubCommand::with_name("new")
.about("Create derived stake accounts")
.arg(fee_payer_arg())
.arg(funding_keypair_arg().index(1))
.arg(
Arg::with_name("base_keypair")
.required(true)
.index(2)
.takes_value(true)
.value_name("BASE_KEYPAIR")
.validator(is_valid_signer)
.help("Keypair which stake account addresses are derived from"),
)
.arg(
Arg::with_name("amount")
.required(true)
.index(3)
.takes_value(true)
.value_name("AMOUNT")
.validator(is_amount)
.help("Amount to move into the new stake accounts, in SOL"),
)
.arg(
Arg::with_name("stake_authority")
.long("stake-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("Stake authority"),
)
.arg(
Arg::with_name("withdraw_authority")
.long("withdraw-authority")
.required(true)
.takes_value(true)
.value_name("PUBKEY")
.validator(is_valid_pubkey)
.help("Withdraw authority"),
)
.arg(
Arg::with_name("index")
.long("index")
.takes_value(true)
.default_value("0")
.value_name("NUMBER")
.help("Index of the derived account to create"),
),
)
.subcommand(
SubCommand::with_name("count")
.about("Count derived stake accounts")
.arg(base_pubkey_arg().index(1)),
)
.subcommand(
SubCommand::with_name("addresses")
.about("Show public keys of all derived stake accounts")
.arg(base_pubkey_arg().index(1))
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("balance")
.about("Sum balances of all derived stake accounts")
.arg(base_pubkey_arg().index(1))
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("authorize")
.about("Set new authorities in all derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(stake_authority_arg())
.arg(withdraw_authority_arg())
.arg(new_stake_authority_arg())
.arg(new_withdraw_authority_arg())
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("rebase")
.about("Relocate derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(new_base_keypair_arg().index(2))
.arg(stake_authority_arg())
.arg(num_accounts_arg()),
)
.subcommand(
SubCommand::with_name("move")
.about("Rebase and set new authorities in all derived stake accounts")
.arg(fee_payer_arg())
.arg(base_pubkey_arg().index(1))
.arg(new_base_keypair_arg().index(2))
.arg(stake_authority_arg())
.arg(withdraw_authority_arg())
.arg(new_stake_authority_arg())
.arg(new_withdraw_authority_arg())
.arg(num_accounts_arg()),
)
.get_matches_from(args)
}
fn parse_new_args(matches: &ArgMatches<'_>) -> NewCommandConfig {
let fee_payer = value_t_or_exit!(matches, "fee_payer", String);
let funding_keypair = value_t_or_exit!(matches, "funding_keypair", String);
let lamports = sol_to_lamports(value_t_or_exit!(matches, "amount", f64));
let base_keypair = value_t_or_exit!(matches, "base_keypair", String);
let stake_authority = value_t_or_exit!(matches, "stake_authority", String);
let withdraw_authority = value_t_or_exit!(matches, "withdraw_authority", String);
let index = value_t_or_exit!(matches, "index", usize);
NewCommandConfig {
fee_payer,
funding_keypair,
lamports,
base_keypair,
stake_authority,
withdraw_authority,
index,
}
}
fn parse_count_args(matches: &ArgMatches<'_>) -> CountCommandConfig {
let base_pubkey = value_t_or_exit!(matches, "base_pubkey", String);
CountCommandConfig { base_pubkey }
}
fn parse_query_args(matches: &ArgMatches<'_>) -> QueryCommandConfig {
let base_pubkey = value_t_or_exit!(matches, "base_pubkey", String);
let num_accounts = value_t_or_exit!(matches, "num_accounts", usize);
QueryCommandConfig {
base_pubkey,
num_accounts,
}
}
fn parse_authorize_args(matches: &ArgMatches<'_>) -> AuthorizeCommandConfig {
let fee_payer = value_t_or_exit!(matches, "fee_payer", String);
let base_pubkey = value_t_or_exit!(matches, "base_pubkey", String);
let stake_authority = value_t_or_exit!(matches, "stake_authority", String);
let withdraw_authority = value_t_or_exit!(matches, "withdraw_authority", String);
let new_stake_authority = value_t_or_exit!(matches, "new_stake_authority", String);
let new_withdraw_authority = value_t_or_exit!(matches, "new_withdraw_authority", String);
let num_accounts = value_t_or_exit!(matches, "num_accounts", usize);
AuthorizeCommandConfig {
fee_payer,
base_pubkey,
stake_authority,
withdraw_authority,
new_stake_authority,
new_withdraw_authority,
num_accounts,
}
}
fn parse_rebase_args(matches: &ArgMatches<'_>) -> RebaseCommandConfig {
let fee_payer = value_t_or_exit!(matches, "fee_payer", String);
let base_pubkey = value_t_or_exit!(matches, "base_pubkey", String);
let new_base_keypair = value_t_or_exit!(matches, "new_base_keypair", String);
let stake_authority = value_t_or_exit!(matches, "stake_authority", String);
let num_accounts = value_t_or_exit!(matches, "num_accounts", usize);
RebaseCommandConfig {
fee_payer,
base_pubkey,
new_base_keypair,
stake_authority,
num_accounts,
}
}
fn parse_move_args(matches: &ArgMatches<'_>) -> MoveCommandConfig {
let rebase_config = parse_rebase_args(matches);
let authorize_config = parse_authorize_args(matches);
MoveCommandConfig {
rebase_config,
authorize_config,
}
}
pub(crate) fn parse_args<I, T>(args: I) -> CommandConfig
where
I: IntoIterator<Item = T>,
T: Into<OsString> + Clone,
{
let matches = get_matches(args);
let config_file = matches.value_of("config_file").unwrap().to_string();
let url = matches.value_of("url").map(|x| x.to_string());
let command = match matches.subcommand() {
("new", Some(matches)) => Command::New(parse_new_args(matches)),
("count", Some(matches)) => Command::Count(parse_count_args(matches)),
("addresses", Some(matches)) => Command::Addresses(parse_query_args(matches)),
("balance", Some(matches)) => Command::Balance(parse_query_args(matches)),
("authorize", Some(matches)) => Command::Authorize(parse_authorize_args(matches)),
("rebase", Some(matches)) => Command::Rebase(parse_rebase_args(matches)),
("move", Some(matches)) => Command::Move(Box::new(parse_move_args(matches))),
_ => {
eprintln!("{}", matches.usage());
exit(1);
}
}; };
CommandConfig { Ok(resolved_args)
config_file, }
url,
command, fn resolve_rebase_args(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
args: &RebaseArgs<String, String>,
) -> Result<RebaseArgs<Pubkey, Box<dyn Signer>>, Box<dyn Error>> {
let resolved_args = RebaseArgs {
fee_payer: resolve_fee_payer(wallet_manager, &args.fee_payer)?,
base_pubkey: resolve_base_pubkey(wallet_manager, &args.base_pubkey)?,
new_base_keypair: resolve_new_base_keypair(wallet_manager, &args.new_base_keypair)?,
stake_authority: resolve_stake_authority(wallet_manager, &args.stake_authority)?,
num_accounts: args.num_accounts,
};
Ok(resolved_args)
}
pub(crate) fn resolve_command(
command: &Command<String, String>,
) -> Result<Command<Pubkey, Box<dyn Signer>>, Box<dyn Error>> {
let wallet_manager = maybe_wallet_manager()?;
let wallet_manager = wallet_manager.as_ref();
let matches = ArgMatches::default();
match command {
Command::New(args) => {
let resolved_args = NewArgs {
fee_payer: resolve_fee_payer(wallet_manager, &args.fee_payer)?,
funding_keypair: signer_from_path(
&matches,
&args.funding_keypair,
"funding keypair",
wallet_manager,
)?,
base_keypair: signer_from_path(
&matches,
&args.base_keypair,
"base keypair",
wallet_manager,
)?,
stake_authority: pubkey_from_path(
&matches,
&args.stake_authority,
"stake authority",
wallet_manager,
)?,
withdraw_authority: pubkey_from_path(
&matches,
&args.withdraw_authority,
"withdraw authority",
wallet_manager,
)?,
lamports: args.lamports,
index: args.index,
};
Ok(Command::New(resolved_args))
}
Command::Count(args) => {
let resolved_args = CountArgs {
base_pubkey: resolve_base_pubkey(wallet_manager, &args.base_pubkey)?,
};
Ok(Command::Count(resolved_args))
}
Command::Addresses(args) => {
let resolved_args = QueryArgs {
base_pubkey: resolve_base_pubkey(wallet_manager, &args.base_pubkey)?,
num_accounts: args.num_accounts,
};
Ok(Command::Addresses(resolved_args))
}
Command::Balance(args) => {
let resolved_args = QueryArgs {
base_pubkey: resolve_base_pubkey(wallet_manager, &args.base_pubkey)?,
num_accounts: args.num_accounts,
};
Ok(Command::Balance(resolved_args))
}
Command::Authorize(args) => {
let resolved_args = resolve_authorize_args(wallet_manager, &args)?;
Ok(Command::Authorize(resolved_args))
}
Command::Rebase(args) => {
let resolved_args = resolve_rebase_args(wallet_manager, &args)?;
Ok(Command::Rebase(resolved_args))
}
Command::Move(args) => {
let resolved_args = MoveArgs {
authorize_args: resolve_authorize_args(wallet_manager, &args.authorize_args)?,
rebase_args: resolve_rebase_args(wallet_manager, &args.rebase_args)?,
};
Ok(Command::Move(Box::new(resolved_args)))
}
} }
} }

View File

@ -1,16 +1,12 @@
mod arg_parser;
mod args; mod args;
mod stake_accounts; mod stake_accounts;
use crate::args::{ use crate::arg_parser::parse_args;
parse_args, AuthorizeCommandConfig, Command, MoveCommandConfig, NewCommandConfig, use crate::args::{resolve_command, AuthorizeArgs, Command, MoveArgs, NewArgs, RebaseArgs};
RebaseCommandConfig,
};
use clap::ArgMatches;
use solana_clap_utils::keypair::{pubkey_from_path, signer_from_path};
use solana_cli_config::Config; use solana_cli_config::Config;
use solana_client::client_error::ClientError; use solana_client::client_error::ClientError;
use solana_client::rpc_client::RpcClient; use solana_client::rpc_client::RpcClient;
use solana_remote_wallet::remote_wallet::{maybe_wallet_manager, RemoteWalletManager};
use solana_sdk::{ use solana_sdk::{
message::Message, message::Message,
native_token::lamports_to_sol, native_token::lamports_to_sol,
@ -21,63 +17,6 @@ use solana_sdk::{
}; };
use std::env; use std::env;
use std::error::Error; use std::error::Error;
use std::sync::Arc;
fn resolve_stake_authority(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Box<dyn Signer>, Box<dyn Error>> {
let matches = ArgMatches::default();
signer_from_path(&matches, key_url, "stake authority", wallet_manager)
}
fn resolve_withdraw_authority(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Box<dyn Signer>, Box<dyn Error>> {
let matches = ArgMatches::default();
signer_from_path(&matches, key_url, "withdraw authority", wallet_manager)
}
fn resolve_new_stake_authority(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Pubkey, Box<dyn Error>> {
let matches = ArgMatches::default();
pubkey_from_path(&matches, key_url, "new stake authority", wallet_manager)
}
fn resolve_new_withdraw_authority(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Pubkey, Box<dyn Error>> {
let matches = ArgMatches::default();
pubkey_from_path(&matches, key_url, "new withdraw authority", wallet_manager)
}
fn resolve_fee_payer(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Box<dyn Signer>, Box<dyn Error>> {
let matches = ArgMatches::default();
signer_from_path(&matches, key_url, "fee-payer", wallet_manager)
}
fn resolve_base_pubkey(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Pubkey, Box<dyn Error>> {
let matches = ArgMatches::default();
pubkey_from_path(&matches, key_url, "base pubkey", wallet_manager)
}
fn resolve_new_base_keypair(
wallet_manager: Option<&Arc<RemoteWalletManager>>,
key_url: &str,
) -> Result<Box<dyn Signer>, Box<dyn Error>> {
let matches = ArgMatches::default();
signer_from_path(&matches, key_url, "new base pubkey", wallet_manager)
}
fn get_balance_at(client: &RpcClient, pubkey: &Pubkey, i: usize) -> Result<u64, ClientError> { fn get_balance_at(client: &RpcClient, pubkey: &Pubkey, i: usize) -> Result<u64, ClientError> {
let address = stake_accounts::derive_stake_account_address(pubkey, i); let address = stake_accounts::derive_stake_account_address(pubkey, i);
@ -105,77 +44,43 @@ fn get_balances(
fn process_new_stake_account( fn process_new_stake_account(
client: &RpcClient, client: &RpcClient,
wallet_manager: Option<&Arc<RemoteWalletManager>>, args: &NewArgs<Pubkey, Box<dyn Signer>>,
new_config: &NewCommandConfig, ) -> Result<Signature, ClientError> {
) -> Result<Signature, Box<dyn Error>> {
let matches = ArgMatches::default();
let fee_payer_keypair = resolve_fee_payer(wallet_manager, &new_config.fee_payer)?;
let funding_keypair = signer_from_path(
&matches,
&new_config.funding_keypair,
"funding keypair",
wallet_manager,
)?;
let base_keypair = signer_from_path(
&matches,
&new_config.base_keypair,
"base keypair",
wallet_manager,
)?;
let stake_authority_pubkey = pubkey_from_path(
&matches,
&new_config.stake_authority,
"stake authority",
wallet_manager,
)?;
let withdraw_authority_pubkey = pubkey_from_path(
&matches,
&new_config.withdraw_authority,
"withdraw authority",
wallet_manager,
)?;
let message = stake_accounts::new_stake_account( let message = stake_accounts::new_stake_account(
&fee_payer_keypair.pubkey(), &args.fee_payer.pubkey(),
&funding_keypair.pubkey(), &args.funding_keypair.pubkey(),
&base_keypair.pubkey(), &args.base_keypair.pubkey(),
new_config.lamports, args.lamports,
&stake_authority_pubkey, &args.stake_authority,
&withdraw_authority_pubkey, &args.withdraw_authority,
new_config.index, args.index,
); );
let signers = vec![&*fee_payer_keypair, &*funding_keypair, &*base_keypair]; let signers = vec![
&*args.fee_payer,
&*args.funding_keypair,
&*args.base_keypair,
];
let signature = send_message(client, message, &signers)?; let signature = send_message(client, message, &signers)?;
Ok(signature) Ok(signature)
} }
fn process_authorize_stake_accounts( fn process_authorize_stake_accounts(
client: &RpcClient, client: &RpcClient,
wallet_manager: Option<&Arc<RemoteWalletManager>>, args: &AuthorizeArgs<Pubkey, Box<dyn Signer>>,
authorize_config: &AuthorizeCommandConfig, ) -> Result<(), ClientError> {
) -> Result<(), Box<dyn Error>> {
let fee_payer_keypair = resolve_fee_payer(wallet_manager, &authorize_config.fee_payer)?;
let base_pubkey = resolve_base_pubkey(wallet_manager, &authorize_config.base_pubkey)?;
let stake_authority_keypair =
resolve_stake_authority(wallet_manager, &authorize_config.stake_authority)?;
let withdraw_authority_keypair =
resolve_withdraw_authority(wallet_manager, &authorize_config.withdraw_authority)?;
let new_stake_authority_pubkey =
resolve_new_stake_authority(wallet_manager, &authorize_config.new_stake_authority)?;
let new_withdraw_authority_pubkey =
resolve_new_withdraw_authority(wallet_manager, &authorize_config.new_withdraw_authority)?;
let messages = stake_accounts::authorize_stake_accounts( let messages = stake_accounts::authorize_stake_accounts(
&fee_payer_keypair.pubkey(), &args.fee_payer.pubkey(),
&base_pubkey, &args.base_pubkey,
&stake_authority_keypair.pubkey(), &args.stake_authority.pubkey(),
&withdraw_authority_keypair.pubkey(), &args.withdraw_authority.pubkey(),
&new_stake_authority_pubkey, &args.new_stake_authority,
&new_withdraw_authority_pubkey, &args.new_withdraw_authority,
authorize_config.num_accounts, args.num_accounts,
); );
let signers = vec![ let signers = vec![
&*fee_payer_keypair, &*args.fee_payer,
&*stake_authority_keypair, &*args.stake_authority,
&*withdraw_authority_keypair, &*args.withdraw_authority,
]; ];
for message in messages { for message in messages {
let signature = send_message(client, message, &signers)?; let signature = send_message(client, message, &signers)?;
@ -186,29 +91,22 @@ fn process_authorize_stake_accounts(
fn process_rebase_stake_accounts( fn process_rebase_stake_accounts(
client: &RpcClient, client: &RpcClient,
wallet_manager: Option<&Arc<RemoteWalletManager>>, args: &RebaseArgs<Pubkey, Box<dyn Signer>>,
rebase_config: &RebaseCommandConfig, ) -> Result<(), ClientError> {
) -> Result<(), Box<dyn Error>> {
let fee_payer_keypair = resolve_fee_payer(wallet_manager, &rebase_config.fee_payer)?;
let base_pubkey = resolve_base_pubkey(wallet_manager, &rebase_config.base_pubkey)?;
let new_base_keypair =
resolve_new_base_keypair(wallet_manager, &rebase_config.new_base_keypair)?;
let stake_authority_keypair =
resolve_stake_authority(wallet_manager, &rebase_config.stake_authority)?;
let addresses = let addresses =
stake_accounts::derive_stake_account_addresses(&base_pubkey, rebase_config.num_accounts); stake_accounts::derive_stake_account_addresses(&args.base_pubkey, args.num_accounts);
let balances = get_balances(&client, addresses)?; let balances = get_balances(&client, addresses)?;
let messages = stake_accounts::rebase_stake_accounts( let messages = stake_accounts::rebase_stake_accounts(
&fee_payer_keypair.pubkey(), &args.fee_payer.pubkey(),
&new_base_keypair.pubkey(), &args.new_base_keypair.pubkey(),
&stake_authority_keypair.pubkey(), &args.stake_authority.pubkey(),
&balances, &balances,
); );
let signers = vec![ let signers = vec![
&*fee_payer_keypair, &*args.fee_payer,
&*new_base_keypair, &*args.new_base_keypair,
&*stake_authority_keypair, &*args.stake_authority,
]; ];
for message in messages { for message in messages {
let signature = send_message(client, message, &signers)?; let signature = send_message(client, message, &signers)?;
@ -219,41 +117,28 @@ fn process_rebase_stake_accounts(
fn process_move_stake_accounts( fn process_move_stake_accounts(
client: &RpcClient, client: &RpcClient,
wallet_manager: Option<&Arc<RemoteWalletManager>>, move_args: &MoveArgs<Pubkey, Box<dyn Signer>>,
move_config: &MoveCommandConfig, ) -> Result<(), ClientError> {
) -> Result<(), Box<dyn Error>> { let authorize_args = &move_args.authorize_args;
let authorize_config = &move_config.authorize_config; let args = &move_args.rebase_args;
let rebase_config = &move_config.rebase_config;
let fee_payer_keypair = resolve_fee_payer(wallet_manager, &authorize_config.fee_payer)?;
let base_pubkey = resolve_base_pubkey(wallet_manager, &authorize_config.base_pubkey)?;
let new_base_keypair =
resolve_new_base_keypair(wallet_manager, &rebase_config.new_base_keypair)?;
let stake_authority_keypair =
resolve_stake_authority(wallet_manager, &authorize_config.stake_authority)?;
let withdraw_authority_keypair =
resolve_withdraw_authority(wallet_manager, &authorize_config.withdraw_authority)?;
let new_stake_authority_pubkey =
resolve_new_stake_authority(wallet_manager, &authorize_config.new_stake_authority)?;
let new_withdraw_authority_pubkey =
resolve_new_withdraw_authority(wallet_manager, &authorize_config.new_withdraw_authority)?;
let addresses = let addresses =
stake_accounts::derive_stake_account_addresses(&base_pubkey, authorize_config.num_accounts); stake_accounts::derive_stake_account_addresses(&args.base_pubkey, args.num_accounts);
let balances = get_balances(&client, addresses)?; let balances = get_balances(&client, addresses)?;
let messages = stake_accounts::move_stake_accounts( let messages = stake_accounts::move_stake_accounts(
&fee_payer_keypair.pubkey(), &args.fee_payer.pubkey(),
&new_base_keypair.pubkey(), &args.new_base_keypair.pubkey(),
&stake_authority_keypair.pubkey(), &args.stake_authority.pubkey(),
&withdraw_authority_keypair.pubkey(), &authorize_args.withdraw_authority.pubkey(),
&new_stake_authority_pubkey, &authorize_args.new_stake_authority,
&new_withdraw_authority_pubkey, &authorize_args.new_withdraw_authority,
&balances, &balances,
); );
let signers = vec![ let signers = vec![
&*fee_payer_keypair, &*args.fee_payer,
&*new_base_keypair, &*args.new_base_keypair,
&*stake_authority_keypair, &*args.stake_authority,
&*withdraw_authority_keypair, &*authorize_args.withdraw_authority,
]; ];
for message in messages { for message in messages {
let signature = send_message(client, message, &signers)?; let signature = send_message(client, message, &signers)?;
@ -273,51 +158,46 @@ fn send_message<S: Signers>(
} }
fn main() -> Result<(), Box<dyn Error>> { fn main() -> Result<(), Box<dyn Error>> {
let command_config = parse_args(env::args_os()); let command_args = parse_args(env::args_os());
let config = Config::load(&command_config.config_file)?; let config = Config::load(&command_args.config_file)?;
let json_rpc_url = command_config.url.unwrap_or(config.json_rpc_url); let json_rpc_url = command_args.url.unwrap_or(config.json_rpc_url);
let client = RpcClient::new(json_rpc_url); let client = RpcClient::new(json_rpc_url);
let wallet_manager = maybe_wallet_manager()?; match resolve_command(&command_args.command)? {
let wallet_manager = wallet_manager.as_ref(); Command::New(args) => {
match command_config.command { process_new_stake_account(&client, &args)?;
Command::New(new_config) => {
process_new_stake_account(&client, wallet_manager, &new_config)?;
} }
Command::Count(count_config) => { Command::Count(args) => {
let base_pubkey = resolve_base_pubkey(wallet_manager, &count_config.base_pubkey)?; let num_accounts = count_stake_accounts(&client, &args.base_pubkey)?;
let num_accounts = count_stake_accounts(&client, &base_pubkey)?;
println!("{}", num_accounts); println!("{}", num_accounts);
} }
Command::Addresses(query_config) => { Command::Addresses(args) => {
let base_pubkey = resolve_base_pubkey(wallet_manager, &query_config.base_pubkey)?;
let addresses = stake_accounts::derive_stake_account_addresses( let addresses = stake_accounts::derive_stake_account_addresses(
&base_pubkey, &args.base_pubkey,
query_config.num_accounts, args.num_accounts,
); );
for address in addresses { for address in addresses {
println!("{:?}", address); println!("{:?}", address);
} }
} }
Command::Balance(query_config) => { Command::Balance(args) => {
let base_pubkey = resolve_base_pubkey(wallet_manager, &query_config.base_pubkey)?;
let addresses = stake_accounts::derive_stake_account_addresses( let addresses = stake_accounts::derive_stake_account_addresses(
&base_pubkey, &args.base_pubkey,
query_config.num_accounts, args.num_accounts,
); );
let balances = get_balances(&client, addresses)?; let balances = get_balances(&client, addresses)?;
let lamports: u64 = balances.into_iter().map(|(_, bal)| bal).sum(); let lamports: u64 = balances.into_iter().map(|(_, bal)| bal).sum();
let sol = lamports_to_sol(lamports); let sol = lamports_to_sol(lamports);
println!("{} SOL", sol); println!("{} SOL", sol);
} }
Command::Authorize(authorize_config) => { Command::Authorize(args) => {
process_authorize_stake_accounts(&client, wallet_manager, &authorize_config)?; process_authorize_stake_accounts(&client, &args)?;
} }
Command::Rebase(rebase_config) => { Command::Rebase(args) => {
process_rebase_stake_accounts(&client, wallet_manager, &rebase_config)?; process_rebase_stake_accounts(&client, &args)?;
} }
Command::Move(move_config) => { Command::Move(args) => {
process_move_stake_accounts(&client, wallet_manager, &move_config)?; process_move_stake_accounts(&client, &args)?;
} }
} }
Ok(()) Ok(())