| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- use {
- assert_matches::assert_matches,
- solana_account_info::{next_account_info, AccountInfo},
- solana_banks_client::BanksClientError,
- solana_commitment_config::CommitmentLevel,
- solana_instruction::{AccountMeta, Instruction},
- solana_msg::msg,
- solana_program::program::{get_return_data, invoke, set_return_data},
- solana_program_error::{ProgramError, ProgramResult},
- solana_program_test::{processor, ProgramTest},
- solana_pubkey::Pubkey,
- solana_signer::Signer,
- solana_transaction::Transaction,
- solana_transaction_context::TransactionReturnData,
- std::{slice, str::from_utf8},
- };
- // Process instruction to get return data from another program
- fn get_return_data_process_instruction(
- _program_id: &Pubkey,
- accounts: &[AccountInfo],
- input: &[u8],
- ) -> ProgramResult {
- msg!("Processing get_return_data instruction before CPI");
- let account_info_iter = &mut accounts.iter();
- let invoked_program_info = next_account_info(account_info_iter)?;
- invoke(
- &Instruction {
- program_id: *invoked_program_info.key,
- accounts: vec![],
- data: input.to_vec(),
- },
- slice::from_ref(invoked_program_info),
- )?;
- let return_data = get_return_data().unwrap();
- msg!("Processing get_return_data instruction after CPI");
- msg!("{}", from_utf8(&return_data.1).unwrap());
- assert_eq!(return_data.1, input.to_vec());
- Ok(())
- }
- // Process instruction to echo input back to another program
- fn set_return_data_process_instruction(
- _program_id: &Pubkey,
- _accounts: &[AccountInfo],
- input: &[u8],
- ) -> ProgramResult {
- msg!("Processing invoked instruction before set_return_data");
- set_return_data(input);
- msg!("Processing invoked instruction after set_return_data");
- Ok(())
- }
- #[tokio::test]
- async fn return_data() {
- let get_return_data_program_id = Pubkey::new_unique();
- let mut program_test = ProgramTest::new(
- "get_return_data",
- get_return_data_program_id,
- processor!(get_return_data_process_instruction),
- );
- let set_return_data_program_id = Pubkey::new_unique();
- program_test.add_program(
- "set_return_data",
- set_return_data_program_id,
- processor!(set_return_data_process_instruction),
- );
- let context = program_test.start_with_context().await;
- let instructions = vec![Instruction {
- program_id: get_return_data_program_id,
- accounts: vec![AccountMeta::new_readonly(set_return_data_program_id, false)],
- data: vec![240, 159, 166, 150],
- }];
- let transaction = Transaction::new_signed_with_payer(
- &instructions,
- Some(&context.payer.pubkey()),
- &[&context.payer],
- context.last_blockhash,
- );
- context
- .banks_client
- .process_transaction(transaction)
- .await
- .unwrap();
- }
- // Process instruction to echo input back to another program
- fn error_set_return_data_process_instruction(
- _program_id: &Pubkey,
- _accounts: &[AccountInfo],
- input: &[u8],
- ) -> ProgramResult {
- set_return_data(input);
- Err(ProgramError::InvalidInstructionData)
- }
- #[tokio::test]
- async fn simulation_return_data() {
- let error_set_return_data_program_id = Pubkey::new_unique();
- let program_test = ProgramTest::new(
- "error_set_return_data",
- error_set_return_data_program_id,
- processor!(error_set_return_data_process_instruction),
- );
- let context = program_test.start_with_context().await;
- let expected_data = vec![240, 159, 166, 150];
- let instructions = vec![Instruction {
- program_id: error_set_return_data_program_id,
- accounts: vec![],
- data: expected_data.clone(),
- }];
- let transaction = Transaction::new_signed_with_payer(
- &instructions,
- Some(&context.payer.pubkey()),
- &[&context.payer],
- context.last_blockhash,
- );
- let error = context
- .banks_client
- .process_transaction_with_preflight_and_commitment(transaction, CommitmentLevel::Confirmed)
- .await
- .unwrap_err();
- assert_matches!(
- error,
- BanksClientError::SimulationError {
- return_data: Some(TransactionReturnData {
- program_id,
- data,
- }),
- ..
- } if program_id == error_set_return_data_program_id && data == expected_data
- );
- }
|