v0.16: Expand Config program; implement Validator Info CLI (#5045)
* Update config program to accommodate multiple signers (#4946) * Update config program to accommodate multiple signers * Update install CLI * Remove account_type u32; add handling for unsigned keys in list * ConfigKeys doc * Make config_api more robust (#4980) * Make config_api more robust * Add test and update store instruction * Improve signature checks in config_api (#5001) automerge * Add validator-info CLI (#4970) * Add validator-info CLI * Add GetProgramAccounts method to solana-client * Update validator-info args, and add get subcommand * Update ValidatorInfo lengths * Add account filter for get --all * Update testnet participation doc to reflect validator-info * Flesh out tests * Review comments
This commit is contained in:
		| @@ -1,26 +1,59 @@ | ||||
| use crate::id; | ||||
| use crate::ConfigState; | ||||
| use bincode::serialize; | ||||
| use serde_derive::{Deserialize, Serialize}; | ||||
| use solana_sdk::instruction::{AccountMeta, Instruction}; | ||||
| use solana_sdk::pubkey::Pubkey; | ||||
| use solana_sdk::short_vec; | ||||
| use solana_sdk::system_instruction; | ||||
|  | ||||
| /// A collection of keys to be stored in Config account data. | ||||
| #[derive(Debug, Default, Deserialize, Serialize)] | ||||
| pub struct ConfigKeys { | ||||
|     // Each key tuple comprises a unique `Pubkey` identifier, | ||||
|     // and `bool` whether that key is a signer of the data | ||||
|     #[serde(with = "short_vec")] | ||||
|     pub keys: Vec<(Pubkey, bool)>, | ||||
| } | ||||
|  | ||||
| impl ConfigKeys { | ||||
|     pub fn serialized_size(keys: Vec<(Pubkey, bool)>) -> usize { | ||||
|         serialize(&ConfigKeys { keys }) | ||||
|             .unwrap_or_else(|_| vec![]) | ||||
|             .len() | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Create a new, empty configuration account | ||||
| pub fn create_account<T: ConfigState>( | ||||
|     from_account_pubkey: &Pubkey, | ||||
|     config_account_pubkey: &Pubkey, | ||||
|     lamports: u64, | ||||
|     keys: Vec<(Pubkey, bool)>, | ||||
| ) -> Instruction { | ||||
|     let space = T::max_space() + ConfigKeys::serialized_size(keys) as u64; | ||||
|     system_instruction::create_account( | ||||
|         from_account_pubkey, | ||||
|         config_account_pubkey, | ||||
|         lamports, | ||||
|         T::max_space(), | ||||
|         space, | ||||
|         &id(), | ||||
|     ) | ||||
| } | ||||
|  | ||||
| /// Store new data in a configuration account | ||||
| pub fn store<T: ConfigState>(config_account_pubkey: &Pubkey, data: &T) -> Instruction { | ||||
|     let account_metas = vec![AccountMeta::new(*config_account_pubkey, true)]; | ||||
|     Instruction::new(id(), data, account_metas) | ||||
| pub fn store<T: ConfigState>( | ||||
|     config_account_pubkey: &Pubkey, | ||||
|     is_config_signer: bool, | ||||
|     keys: Vec<(Pubkey, bool)>, | ||||
|     data: &T, | ||||
| ) -> Instruction { | ||||
|     let mut account_metas = vec![AccountMeta::new(*config_account_pubkey, is_config_signer)]; | ||||
|     for (signer_pubkey, _) in keys.iter().filter(|(_, is_signer)| *is_signer) { | ||||
|         if signer_pubkey != config_account_pubkey { | ||||
|             account_metas.push(AccountMeta::new(*signer_pubkey, true)); | ||||
|         } | ||||
|     } | ||||
|     let account_data = (ConfigKeys { keys }, data); | ||||
|     Instruction::new(id(), &account_data, account_metas) | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,7 @@ | ||||
| //! Config program | ||||
|  | ||||
| use crate::config_instruction::ConfigKeys; | ||||
| use bincode::deserialize; | ||||
| use log::*; | ||||
| use solana_sdk::account::KeyedAccount; | ||||
| use solana_sdk::instruction::InstructionError; | ||||
| @@ -10,8 +12,81 @@ pub fn process_instruction( | ||||
|     keyed_accounts: &mut [KeyedAccount], | ||||
|     data: &[u8], | ||||
| ) -> Result<(), InstructionError> { | ||||
|     if keyed_accounts[0].signer_key().is_none() { | ||||
|         error!("account[0].signer_key().is_none()"); | ||||
|     let key_list: ConfigKeys = deserialize(data).map_err(|err| { | ||||
|         error!("Invalid ConfigKeys data: {:?} {:?}", data, err); | ||||
|         InstructionError::InvalidInstructionData | ||||
|     })?; | ||||
|  | ||||
|     let current_data: ConfigKeys = deserialize(&keyed_accounts[0].account.data).map_err(|err| { | ||||
|         error!("Invalid data in account[0]: {:?} {:?}", data, err); | ||||
|         InstructionError::InvalidAccountData | ||||
|     })?; | ||||
|     let current_signer_keys: Vec<Pubkey> = current_data | ||||
|         .keys | ||||
|         .iter() | ||||
|         .filter(|(_, is_signer)| *is_signer) | ||||
|         .map(|(pubkey, _)| *pubkey) | ||||
|         .collect(); | ||||
|  | ||||
|     if current_signer_keys.is_empty() { | ||||
|         // Config account keypair must be a signer on account initilization, | ||||
|         // or when no signers specified in Config data | ||||
|         if keyed_accounts[0].signer_key().is_none() { | ||||
|             error!("account[0].signer_key().is_none()"); | ||||
|             Err(InstructionError::MissingRequiredSignature)?; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let mut counter = 0; | ||||
|     for (i, (signer, _)) in key_list | ||||
|         .keys | ||||
|         .iter() | ||||
|         .filter(|(_, is_signer)| *is_signer) | ||||
|         .enumerate() | ||||
|     { | ||||
|         counter += 1; | ||||
|         if signer != keyed_accounts[0].unsigned_key() { | ||||
|             let account_index = i + 1; | ||||
|             let signer_account = keyed_accounts.get(account_index); | ||||
|             if signer_account.is_none() { | ||||
|                 error!("account {:?} is not in account list", signer); | ||||
|                 Err(InstructionError::MissingRequiredSignature)?; | ||||
|             } | ||||
|             let signer_key = signer_account.unwrap().signer_key(); | ||||
|             if signer_key.is_none() { | ||||
|                 error!("account {:?} signer_key().is_none()", signer); | ||||
|                 Err(InstructionError::MissingRequiredSignature)?; | ||||
|             } | ||||
|             if signer_key.unwrap() != signer { | ||||
|                 error!( | ||||
|                     "account[{:?}].signer_key() does not match Config data)", | ||||
|                     account_index | ||||
|                 ); | ||||
|                 Err(InstructionError::MissingRequiredSignature)?; | ||||
|             } | ||||
|             // If Config account is already initialized, update signatures must match Config data | ||||
|             if !current_data.keys.is_empty() | ||||
|                 && current_signer_keys | ||||
|                     .iter() | ||||
|                     .find(|&pubkey| pubkey == signer) | ||||
|                     .is_none() | ||||
|             { | ||||
|                 error!("account {:?} is not in stored signer list", signer); | ||||
|                 Err(InstructionError::MissingRequiredSignature)?; | ||||
|             } | ||||
|         } else if keyed_accounts[0].signer_key().is_none() { | ||||
|             error!("account[0].signer_key().is_none()"); | ||||
|             Err(InstructionError::MissingRequiredSignature)?; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Check for Config data signers not present in incoming account update | ||||
|     if current_signer_keys.len() > counter { | ||||
|         error!( | ||||
|             "too few signers: {:?}; expected: {:?}", | ||||
|             counter, | ||||
|             current_signer_keys.len() | ||||
|         ); | ||||
|         Err(InstructionError::MissingRequiredSignature)?; | ||||
|     } | ||||
|  | ||||
| @@ -20,7 +95,7 @@ pub fn process_instruction( | ||||
|         Err(InstructionError::InvalidInstructionData)?; | ||||
|     } | ||||
|  | ||||
|     keyed_accounts[0].account.data[0..data.len()].copy_from_slice(data); | ||||
|     keyed_accounts[0].account.data[0..data.len()].copy_from_slice(&data); | ||||
|     Ok(()) | ||||
| } | ||||
|  | ||||
| @@ -64,7 +139,11 @@ mod tests { | ||||
|         (bank, mint_keypair) | ||||
|     } | ||||
|  | ||||
|     fn create_config_account(bank: Bank, mint_keypair: &Keypair) -> (BankClient, Keypair) { | ||||
|     fn create_config_account( | ||||
|         bank: Bank, | ||||
|         mint_keypair: &Keypair, | ||||
|         keys: Vec<(Pubkey, bool)>, | ||||
|     ) -> (BankClient, Keypair) { | ||||
|         let config_keypair = Keypair::new(); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
| @@ -76,6 +155,7 @@ mod tests { | ||||
|                     &mint_keypair.pubkey(), | ||||
|                     &config_pubkey, | ||||
|                     1, | ||||
|                     keys, | ||||
|                 ), | ||||
|             ) | ||||
|             .expect("new_account"); | ||||
| @@ -87,7 +167,7 @@ mod tests { | ||||
|     fn test_process_create_ok() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair, vec![]); | ||||
|         let config_account_data = bank_client | ||||
|             .get_account_data(&config_keypair.pubkey()) | ||||
|             .unwrap() | ||||
| @@ -102,13 +182,16 @@ mod tests { | ||||
|     fn test_process_store_ok() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair); | ||||
|         let keys = vec![]; | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let instruction = config_instruction::store(&config_pubkey, &my_config); | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|  | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair], message) | ||||
|             .unwrap(); | ||||
| @@ -117,6 +200,8 @@ mod tests { | ||||
|             .get_account_data(&config_pubkey) | ||||
|             .unwrap() | ||||
|             .unwrap(); | ||||
|         let meta_length = ConfigKeys::serialized_size(keys); | ||||
|         let config_account_data = &config_account_data[meta_length..config_account_data.len()]; | ||||
|         assert_eq!( | ||||
|             my_config, | ||||
|             MyConfig::deserialize(&config_account_data).unwrap() | ||||
| @@ -127,12 +212,12 @@ mod tests { | ||||
|     fn test_process_store_fail_instruction_data_too_large() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair, vec![]); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let mut instruction = config_instruction::store(&config_pubkey, &my_config); | ||||
|         let mut instruction = config_instruction::store(&config_pubkey, true, vec![], &my_config); | ||||
|         instruction.data = vec![0; 123]; // <-- Replace data with a vector that's too large | ||||
|         let message = Message::new(vec![instruction]); | ||||
|         bank_client | ||||
| @@ -148,13 +233,14 @@ mod tests { | ||||
|         let system_pubkey = system_keypair.pubkey(); | ||||
|  | ||||
|         bank.transfer(42, &mint_keypair, &system_pubkey).unwrap(); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair); | ||||
|         let (bank_client, config_keypair) = create_config_account(bank, &mint_keypair, vec![]); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let transfer_instruction = | ||||
|             system_instruction::transfer(&system_pubkey, &Pubkey::new_rand(), 42); | ||||
|         let my_config = MyConfig::new(42); | ||||
|         let mut store_instruction = config_instruction::store(&config_pubkey, &my_config); | ||||
|         let mut store_instruction = | ||||
|             config_instruction::store(&config_pubkey, true, vec![], &my_config); | ||||
|         store_instruction.accounts[0].is_signer = false; // <----- not a signer | ||||
|  | ||||
|         let message = Message::new(vec![transfer_instruction, store_instruction]); | ||||
| @@ -162,4 +248,232 @@ mod tests { | ||||
|             .send_message(&[&system_keypair], message) | ||||
|             .unwrap_err(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_process_store_with_additional_signers() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let pubkey = Pubkey::new_rand(); | ||||
|         let signer0 = Keypair::new(); | ||||
|         let signer1 = Keypair::new(); | ||||
|         let keys = vec![ | ||||
|             (pubkey, false), | ||||
|             (signer0.pubkey(), true), | ||||
|             (signer1.pubkey(), true), | ||||
|         ]; | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|  | ||||
|         bank_client | ||||
|             .send_message( | ||||
|                 &[&mint_keypair, &config_keypair, &signer0, &signer1], | ||||
|                 message, | ||||
|             ) | ||||
|             .unwrap(); | ||||
|  | ||||
|         let config_account_data = bank_client | ||||
|             .get_account_data(&config_pubkey) | ||||
|             .unwrap() | ||||
|             .unwrap(); | ||||
|         let meta_length = ConfigKeys::serialized_size(keys.clone()); | ||||
|         let meta_data: ConfigKeys = deserialize(&config_account_data[0..meta_length]).unwrap(); | ||||
|         assert_eq!(meta_data.keys, keys); | ||||
|         let config_account_data = &config_account_data[meta_length..config_account_data.len()]; | ||||
|         assert_eq!( | ||||
|             my_config, | ||||
|             MyConfig::deserialize(&config_account_data).unwrap() | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_process_store_without_config_signer() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let pubkey = Pubkey::new_rand(); | ||||
|         let signer0 = Keypair::new(); | ||||
|         let keys = vec![(pubkey, false), (signer0.pubkey(), true)]; | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let instruction = | ||||
|             config_instruction::store(&config_pubkey, false, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|  | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &signer0], message) | ||||
|             .unwrap_err(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_process_store_with_bad_additional_signer() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let signer0 = Keypair::new(); | ||||
|         let signer1 = Keypair::new(); | ||||
|         let keys = vec![(signer0.pubkey(), true)]; | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         // Config-data pubkey doesn't match signer | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let mut message = | ||||
|             Message::new_with_payer(vec![instruction.clone()], Some(&mint_keypair.pubkey())); | ||||
|         message.account_keys[2] = signer1.pubkey(); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair, &signer1], message) | ||||
|             .unwrap_err(); | ||||
|  | ||||
|         // Config-data pubkey not a signer | ||||
|         let mut message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         message.header.num_required_signatures = 2; | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair], message) | ||||
|             .unwrap_err(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_config_updates() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let pubkey = Pubkey::new_rand(); | ||||
|         let signer0 = Keypair::new(); | ||||
|         let signer1 = Keypair::new(); | ||||
|         let keys = vec![ | ||||
|             (pubkey, false), | ||||
|             (signer0.pubkey(), true), | ||||
|             (signer1.pubkey(), true), | ||||
|         ]; | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|  | ||||
|         bank_client | ||||
|             .send_message( | ||||
|                 &[&mint_keypair, &config_keypair, &signer0, &signer1], | ||||
|                 message, | ||||
|             ) | ||||
|             .unwrap(); | ||||
|  | ||||
|         // Update with expected signatures | ||||
|         let new_config = MyConfig::new(84); | ||||
|         let instruction = | ||||
|             config_instruction::store(&config_pubkey, false, keys.clone(), &new_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &signer0, &signer1], message) | ||||
|             .unwrap(); | ||||
|  | ||||
|         let config_account_data = bank_client | ||||
|             .get_account_data(&config_pubkey) | ||||
|             .unwrap() | ||||
|             .unwrap(); | ||||
|         let meta_length = ConfigKeys::serialized_size(keys.clone()); | ||||
|         let meta_data: ConfigKeys = deserialize(&config_account_data[0..meta_length]).unwrap(); | ||||
|         assert_eq!(meta_data.keys, keys); | ||||
|         let config_account_data = &config_account_data[meta_length..config_account_data.len()]; | ||||
|         assert_eq!( | ||||
|             new_config, | ||||
|             MyConfig::deserialize(&config_account_data).unwrap() | ||||
|         ); | ||||
|  | ||||
|         // Attempt update with incomplete signatures | ||||
|         let keys = vec![(pubkey, false), (signer0.pubkey(), true)]; | ||||
|         let instruction = | ||||
|             config_instruction::store(&config_pubkey, false, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &signer0], message) | ||||
|             .unwrap_err(); | ||||
|  | ||||
|         // Attempt update with incorrect signatures | ||||
|         let signer2 = Keypair::new(); | ||||
|         let keys = vec![ | ||||
|             (pubkey, false), | ||||
|             (signer0.pubkey(), true), | ||||
|             (signer2.pubkey(), true), | ||||
|         ]; | ||||
|         let instruction = | ||||
|             config_instruction::store(&config_pubkey, false, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &signer0, &signer2], message) | ||||
|             .unwrap_err(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_config_updates_requiring_config() { | ||||
|         solana_logger::setup(); | ||||
|         let (bank, mint_keypair) = create_bank(10_000); | ||||
|         let pubkey = Pubkey::new_rand(); | ||||
|         let signer0 = Keypair::new(); | ||||
|         let keys = vec![ | ||||
|             (pubkey, false), | ||||
|             (signer0.pubkey(), true), | ||||
|             (signer0.pubkey(), true), | ||||
|         ]; // Dummy keys for account sizing | ||||
|         let (bank_client, config_keypair) = | ||||
|             create_config_account(bank, &mint_keypair, keys.clone()); | ||||
|         let config_pubkey = config_keypair.pubkey(); | ||||
|         let keys = vec![ | ||||
|             (pubkey, false), | ||||
|             (signer0.pubkey(), true), | ||||
|             (config_keypair.pubkey(), true), | ||||
|         ]; | ||||
|  | ||||
|         let my_config = MyConfig::new(42); | ||||
|  | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|  | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair, &signer0], message) | ||||
|             .unwrap(); | ||||
|  | ||||
|         // Update with expected signatures | ||||
|         let new_config = MyConfig::new(84); | ||||
|         let instruction = | ||||
|             config_instruction::store(&config_pubkey, true, keys.clone(), &new_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair, &signer0], message) | ||||
|             .unwrap(); | ||||
|  | ||||
|         let config_account_data = bank_client | ||||
|             .get_account_data(&config_pubkey) | ||||
|             .unwrap() | ||||
|             .unwrap(); | ||||
|         let meta_length = ConfigKeys::serialized_size(keys.clone()); | ||||
|         let meta_data: ConfigKeys = deserialize(&config_account_data[0..meta_length]).unwrap(); | ||||
|         assert_eq!(meta_data.keys, keys); | ||||
|         let config_account_data = &config_account_data[meta_length..config_account_data.len()]; | ||||
|         assert_eq!( | ||||
|             new_config, | ||||
|             MyConfig::deserialize(&config_account_data).unwrap() | ||||
|         ); | ||||
|  | ||||
|         // Attempt update with incomplete signatures | ||||
|         let keys = vec![(pubkey, false), (config_keypair.pubkey(), true)]; | ||||
|         let instruction = config_instruction::store(&config_pubkey, true, keys.clone(), &my_config); | ||||
|         let message = Message::new_with_payer(vec![instruction], Some(&mint_keypair.pubkey())); | ||||
|         bank_client | ||||
|             .send_message(&[&mint_keypair, &config_keypair], message) | ||||
|             .unwrap_err(); | ||||
|     } | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user