diff --git a/token/cli/src/clap_app.rs b/token/cli/src/clap_app.rs index 4a040de9af0..2a513812367 100644 --- a/token/cli/src/clap_app.rs +++ b/token/cli/src/clap_app.rs @@ -188,38 +188,37 @@ impl fmt::Display for AccountMetaRole { write!(f, "{:?}", self) } } -pub fn parse_transfer_hook_account(string: T) -> Result -where - T: AsRef + fmt::Display, -{ - match string.as_ref().split(':').collect::>().as_slice() { - [address, role] => { - let address = Pubkey::from_str(address).map_err(|e| format!("{e}"))?; - let meta = match AccountMetaRole::from_str(role).map_err(|e| format!("{e}"))? { - AccountMetaRole::Readonly => AccountMeta::new_readonly(address, false), - AccountMetaRole::Writable => AccountMeta::new(address, false), - AccountMetaRole::ReadonlySigner => AccountMeta::new_readonly(address, true), - AccountMetaRole::WritableSigner => AccountMeta::new(address, true), - }; - Ok(meta) + +#[derive(Clone, Copy)] +pub(crate) struct TransferHookAccount { + address: Pubkey, + role: AccountMetaRole, +} +impl FromStr for TransferHookAccount { + type Err = String; + + fn from_str(s: &str) -> Result { + match s.split(':').collect::>().as_slice() { + [address, role] => { + let address = Pubkey::from_str(address).map_err(|e| format!("{e}"))?; + let role = AccountMetaRole::from_str(role).map_err(|e| format!("{e}"))?; + Ok(Self { address, role }) + } + _ => Err("Transfer hook account must be present as
:".to_string()), } - _ => Err("Transfer hook account must be present as
:".to_string()), } } -fn validate_transfer_hook_account(string: T) -> Result<(), String> -where - T: AsRef + fmt::Display, -{ - match string.as_ref().split(':').collect::>().as_slice() { - [address, role] => { - is_valid_pubkey(address)?; - AccountMetaRole::from_str(role) - .map(|_| ()) - .map_err(|e| format!("{e}")) +impl TransferHookAccount { + pub(crate) fn create_account_meta(&self) -> AccountMeta { + match self.role { + AccountMetaRole::Readonly => AccountMeta::new_readonly(self.address, false), + AccountMetaRole::Writable => AccountMeta::new(self.address, false), + AccountMetaRole::ReadonlySigner => AccountMeta::new_readonly(self.address, true), + AccountMetaRole::WritableSigner => AccountMeta::new(self.address, true), } - _ => Err("Transfer hook account must be present as
:".to_string()), } } + #[derive(Debug, Clone, PartialEq, EnumIter, EnumString, IntoStaticStr)] #[strum(serialize_all = "kebab-case")] pub enum CliAuthorityType { @@ -1429,7 +1428,7 @@ pub fn app<'a>( .arg( Arg::with_name("transfer_hook_account") .long("transfer-hook-account") - .validator(|s| validate_transfer_hook_account(s)) + .value_parser(clap::value_parser!(TransferHookAccount)) .value_name("PUBKEY:ROLE") .takes_value(true) .multiple(true) diff --git a/token/cli/src/command.rs b/token/cli/src/command.rs index 27f7d41983c..ab141b81fa5 100644 --- a/token/cli/src/command.rs +++ b/token/cli/src/command.rs @@ -3937,11 +3937,13 @@ pub async fn process_command<'a>( let use_unchecked_instruction = arg_matches.is_present("use_unchecked_instruction"); let expected_fee = arg_matches.get_one::("expected_fee").copied(); let memo = value_t!(arg_matches, "memo", String).ok(); - let transfer_hook_accounts = arg_matches.values_of("transfer_hook_account").map(|v| { - v.into_iter() - .map(|s| parse_transfer_hook_account(s).unwrap()) - .collect::>() - }); + let transfer_hook_accounts = arg_matches + .get_many::("transfer_hook_account") + .map(|v| { + v.into_iter() + .map(|account| account.create_account_meta()) + .collect::>() + }); command_transfer( config,