* transaction-status: Add return data to meta * Add return data to simulation results * Use pretty-hex for printing return data * Update arg name, make TransactionRecord struct * Rename TransactionRecord -> ExecutionRecord
145 lines
4.4 KiB
Rust
145 lines
4.4 KiB
Rust
use {
|
|
assert_matches::assert_matches,
|
|
solana_banks_client::BanksClientError,
|
|
solana_program_test::{processor, ProgramTest},
|
|
solana_sdk::{
|
|
account_info::{next_account_info, AccountInfo},
|
|
commitment_config::CommitmentLevel,
|
|
entrypoint::ProgramResult,
|
|
instruction::{AccountMeta, Instruction},
|
|
msg,
|
|
program::{get_return_data, invoke, set_return_data},
|
|
program_error::ProgramError,
|
|
pubkey::Pubkey,
|
|
signature::Signer,
|
|
transaction::Transaction,
|
|
transaction_context::TransactionReturnData,
|
|
},
|
|
std::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(),
|
|
},
|
|
&[invoked_program_info.clone()],
|
|
)?;
|
|
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
|
|
#[allow(clippy::unnecessary_wraps)]
|
|
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 mut 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
|
|
#[allow(clippy::unnecessary_wraps)]
|
|
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 mut 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
|
|
);
|
|
}
|