Include account index in rent paying account tx error (#25189)
This commit is contained in:
parent
389d78e424
commit
f81c5df1f0
|
@ -73,6 +73,7 @@ pub(crate) fn check_rent_state(
|
||||||
transaction_context: &TransactionContext,
|
transaction_context: &TransactionContext,
|
||||||
index: usize,
|
index: usize,
|
||||||
do_support_realloc: bool,
|
do_support_realloc: bool,
|
||||||
|
include_account_index_in_err: bool,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
if let Some((pre_rent_state, post_rent_state)) = pre_rent_state.zip(post_rent_state) {
|
if let Some((pre_rent_state, post_rent_state)) = pre_rent_state.zip(post_rent_state) {
|
||||||
let expect_msg = "account must exist at TransactionContext index if rent-states are Some";
|
let expect_msg = "account must exist at TransactionContext index if rent-states are Some";
|
||||||
|
@ -87,6 +88,7 @@ pub(crate) fn check_rent_state(
|
||||||
.expect(expect_msg)
|
.expect(expect_msg)
|
||||||
.borrow(),
|
.borrow(),
|
||||||
do_support_realloc,
|
do_support_realloc,
|
||||||
|
include_account_index_in_err.then(|| index),
|
||||||
)?;
|
)?;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -98,6 +100,7 @@ pub(crate) fn check_rent_state_with_account(
|
||||||
address: &Pubkey,
|
address: &Pubkey,
|
||||||
account_state: &AccountSharedData,
|
account_state: &AccountSharedData,
|
||||||
do_support_realloc: bool,
|
do_support_realloc: bool,
|
||||||
|
account_index: Option<usize>,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
submit_rent_state_metrics(pre_rent_state, post_rent_state);
|
submit_rent_state_metrics(pre_rent_state, post_rent_state);
|
||||||
if !solana_sdk::incinerator::check_id(address)
|
if !solana_sdk::incinerator::check_id(address)
|
||||||
|
@ -107,9 +110,15 @@ pub(crate) fn check_rent_state_with_account(
|
||||||
"Account {} not rent exempt, state {:?}",
|
"Account {} not rent exempt, state {:?}",
|
||||||
address, account_state,
|
address, account_state,
|
||||||
);
|
);
|
||||||
return Err(TransactionError::InvalidRentPayingAccount);
|
if let Some(account_index) = account_index {
|
||||||
|
let account_index = account_index as u8;
|
||||||
|
Err(TransactionError::InsufficientFundsForRent { account_index })
|
||||||
|
} else {
|
||||||
|
Err(TransactionError::InvalidRentPayingAccount)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|
|
@ -385,6 +385,9 @@ impl Accounts {
|
||||||
payer_address,
|
payer_address,
|
||||||
payer_account,
|
payer_account,
|
||||||
feature_set.is_active(&feature_set::do_support_realloc::id()),
|
feature_set.is_active(&feature_set::do_support_realloc::id()),
|
||||||
|
feature_set
|
||||||
|
.is_active(&feature_set::include_account_index_in_rent_error::ID)
|
||||||
|
.then(|| payer_index),
|
||||||
);
|
);
|
||||||
// Feature gate only wraps the actual error return so that the metrics and debug
|
// Feature gate only wraps the actual error return so that the metrics and debug
|
||||||
// logging generated by `check_rent_state_with_account()` can be examined before
|
// logging generated by `check_rent_state_with_account()` can be examined before
|
||||||
|
|
|
@ -240,7 +240,7 @@ impl RentDebits {
|
||||||
}
|
}
|
||||||
|
|
||||||
type BankStatusCache = StatusCache<Result<()>>;
|
type BankStatusCache = StatusCache<Result<()>>;
|
||||||
#[frozen_abi(digest = "HSBFEjhoubTjeGKeBJvRDAiCBTVeFfcWNPZSbDW1w4H4")]
|
#[frozen_abi(digest = "2YZk2K45HmmAafmxPJnYVXyQ7uA7WuBrRkpwrCawdK31")]
|
||||||
pub type BankSlotDelta = SlotDelta<Result<()>>;
|
pub type BankSlotDelta = SlotDelta<Result<()>>;
|
||||||
|
|
||||||
// Eager rent collection repeats in cyclic manner.
|
// Eager rent collection repeats in cyclic manner.
|
||||||
|
@ -4245,7 +4245,8 @@ impl Bank {
|
||||||
})
|
})
|
||||||
.map_err(|err| {
|
.map_err(|err| {
|
||||||
match err {
|
match err {
|
||||||
TransactionError::InvalidRentPayingAccount => {
|
TransactionError::InvalidRentPayingAccount
|
||||||
|
| TransactionError::InsufficientFundsForRent { .. } => {
|
||||||
error_counters.invalid_rent_paying_account += 1;
|
error_counters.invalid_rent_paying_account += 1;
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
|
@ -17421,9 +17422,9 @@ pub(crate) mod tests {
|
||||||
recent_blockhash,
|
recent_blockhash,
|
||||||
);
|
);
|
||||||
let result = bank.process_transaction(&tx);
|
let result = bank.process_transaction(&tx);
|
||||||
assert_ne!(
|
assert_eq!(
|
||||||
result.unwrap_err(),
|
result.unwrap_err(),
|
||||||
TransactionError::InvalidRentPayingAccount
|
TransactionError::InstructionError(0, InstructionError::Custom(1))
|
||||||
);
|
);
|
||||||
assert_ne!(
|
assert_ne!(
|
||||||
fee_payer_balance,
|
fee_payer_balance,
|
||||||
|
@ -17466,7 +17467,7 @@ pub(crate) mod tests {
|
||||||
let result = bank.process_transaction(&tx);
|
let result = bank.process_transaction(&tx);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
result.unwrap_err(),
|
result.unwrap_err(),
|
||||||
TransactionError::InvalidRentPayingAccount
|
TransactionError::InsufficientFundsForRent { account_index: 0 }
|
||||||
);
|
);
|
||||||
assert!(check_account_is_rent_exempt(
|
assert!(check_account_is_rent_exempt(
|
||||||
&rent_exempt_fee_payer.pubkey()
|
&rent_exempt_fee_payer.pubkey()
|
||||||
|
@ -17488,7 +17489,7 @@ pub(crate) mod tests {
|
||||||
let result = bank.process_transaction(&tx);
|
let result = bank.process_transaction(&tx);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
result.unwrap_err(),
|
result.unwrap_err(),
|
||||||
TransactionError::InvalidRentPayingAccount
|
TransactionError::InsufficientFundsForRent { account_index: 0 }
|
||||||
);
|
);
|
||||||
assert!(check_account_is_rent_exempt(
|
assert!(check_account_is_rent_exempt(
|
||||||
&rent_exempt_fee_payer.pubkey()
|
&rent_exempt_fee_payer.pubkey()
|
||||||
|
@ -17515,7 +17516,7 @@ pub(crate) mod tests {
|
||||||
let result = bank.process_transaction(&tx);
|
let result = bank.process_transaction(&tx);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
result.unwrap_err(),
|
result.unwrap_err(),
|
||||||
TransactionError::InvalidRentPayingAccount
|
TransactionError::InsufficientFundsForRent { account_index: 0 }
|
||||||
);
|
);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
fee_payer_balance - fee,
|
fee_payer_balance - fee,
|
||||||
|
@ -17568,7 +17569,7 @@ pub(crate) mod tests {
|
||||||
let result = bank.process_transaction(&tx);
|
let result = bank.process_transaction(&tx);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
result.unwrap_err(),
|
result.unwrap_err(),
|
||||||
TransactionError::InvalidRentPayingAccount
|
TransactionError::InsufficientFundsForRent { account_index: 0 }
|
||||||
);
|
);
|
||||||
assert!(check_account_is_rent_exempt(
|
assert!(check_account_is_rent_exempt(
|
||||||
&rent_exempt_fee_payer.pubkey()
|
&rent_exempt_fee_payer.pubkey()
|
||||||
|
@ -17935,10 +17936,16 @@ pub(crate) mod tests {
|
||||||
mock_program_id,
|
mock_program_id,
|
||||||
recent_blockhash,
|
recent_blockhash,
|
||||||
);
|
);
|
||||||
assert_eq!(
|
let expected_err = {
|
||||||
bank.process_transaction(&tx).unwrap_err(),
|
let account_index = tx
|
||||||
TransactionError::InvalidRentPayingAccount,
|
.message
|
||||||
);
|
.account_keys
|
||||||
|
.iter()
|
||||||
|
.position(|key| key == &rent_paying_pubkey)
|
||||||
|
.unwrap() as u8;
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index }
|
||||||
|
};
|
||||||
|
assert_eq!(bank.process_transaction(&tx).unwrap_err(), expected_err);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
rent_exempt_minimum_small - 1,
|
rent_exempt_minimum_small - 1,
|
||||||
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
||||||
|
@ -17971,10 +17978,16 @@ pub(crate) mod tests {
|
||||||
mock_program_id,
|
mock_program_id,
|
||||||
recent_blockhash,
|
recent_blockhash,
|
||||||
);
|
);
|
||||||
assert_eq!(
|
let expected_err = {
|
||||||
bank.process_transaction(&tx).unwrap_err(),
|
let account_index = tx
|
||||||
TransactionError::InvalidRentPayingAccount,
|
.message
|
||||||
);
|
.account_keys
|
||||||
|
.iter()
|
||||||
|
.position(|key| key == &rent_paying_pubkey)
|
||||||
|
.unwrap() as u8;
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index }
|
||||||
|
};
|
||||||
|
assert_eq!(bank.process_transaction(&tx).unwrap_err(), expected_err);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
rent_exempt_minimum_large,
|
rent_exempt_minimum_large,
|
||||||
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
||||||
|
@ -18007,10 +18020,16 @@ pub(crate) mod tests {
|
||||||
mock_program_id,
|
mock_program_id,
|
||||||
recent_blockhash,
|
recent_blockhash,
|
||||||
);
|
);
|
||||||
assert_eq!(
|
let expected_err = {
|
||||||
bank.process_transaction(&tx).unwrap_err(),
|
let account_index = tx
|
||||||
TransactionError::InvalidRentPayingAccount,
|
.message
|
||||||
);
|
.account_keys
|
||||||
|
.iter()
|
||||||
|
.position(|key| key == &rent_paying_pubkey)
|
||||||
|
.unwrap() as u8;
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index }
|
||||||
|
};
|
||||||
|
assert_eq!(bank.process_transaction(&tx).unwrap_err(), expected_err);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
rent_exempt_minimum_small,
|
rent_exempt_minimum_small,
|
||||||
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
bank.get_account(&rent_paying_pubkey).unwrap().lamports()
|
||||||
|
@ -18044,10 +18063,16 @@ pub(crate) mod tests {
|
||||||
account_data_size_small as u64,
|
account_data_size_small as u64,
|
||||||
&system_program::id(),
|
&system_program::id(),
|
||||||
);
|
);
|
||||||
assert_eq!(
|
let expected_err = {
|
||||||
bank.process_transaction(&tx).unwrap_err(),
|
let account_index = tx
|
||||||
TransactionError::InvalidRentPayingAccount,
|
.message
|
||||||
);
|
.account_keys
|
||||||
|
.iter()
|
||||||
|
.position(|key| key == &created_keypair.pubkey())
|
||||||
|
.unwrap() as u8;
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index }
|
||||||
|
};
|
||||||
|
assert_eq!(bank.process_transaction(&tx).unwrap_err(), expected_err);
|
||||||
|
|
||||||
// create account, rent exempt
|
// create account, rent exempt
|
||||||
let tx = system_transaction::create_account(
|
let tx = system_transaction::create_account(
|
||||||
|
@ -18093,10 +18118,16 @@ pub(crate) mod tests {
|
||||||
recent_blockhash,
|
recent_blockhash,
|
||||||
(account_data_size_small + 1) as u64,
|
(account_data_size_small + 1) as u64,
|
||||||
);
|
);
|
||||||
assert_eq!(
|
let expected_err = {
|
||||||
bank.process_transaction(&tx).unwrap_err(),
|
let account_index = tx
|
||||||
TransactionError::InvalidRentPayingAccount,
|
.message
|
||||||
);
|
.account_keys
|
||||||
|
.iter()
|
||||||
|
.position(|key| key == &created_keypair.pubkey())
|
||||||
|
.unwrap() as u8;
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index }
|
||||||
|
};
|
||||||
|
assert_eq!(bank.process_transaction(&tx).unwrap_err(), expected_err);
|
||||||
|
|
||||||
// bring balance of account up to rent exemption
|
// bring balance of account up to rent exemption
|
||||||
let tx = system_transaction::transfer(
|
let tx = system_transaction::transfer(
|
||||||
|
|
|
@ -61,6 +61,9 @@ impl Bank {
|
||||||
let do_support_realloc = self
|
let do_support_realloc = self
|
||||||
.feature_set
|
.feature_set
|
||||||
.is_active(&feature_set::do_support_realloc::id());
|
.is_active(&feature_set::do_support_realloc::id());
|
||||||
|
let include_account_index_in_err = self
|
||||||
|
.feature_set
|
||||||
|
.is_active(&feature_set::include_account_index_in_rent_error::id());
|
||||||
for (i, (pre_state_info, post_state_info)) in
|
for (i, (pre_state_info, post_state_info)) in
|
||||||
pre_state_infos.iter().zip(post_state_infos).enumerate()
|
pre_state_infos.iter().zip(post_state_infos).enumerate()
|
||||||
{
|
{
|
||||||
|
@ -70,6 +73,7 @@ impl Bank {
|
||||||
transaction_context,
|
transaction_context,
|
||||||
i,
|
i,
|
||||||
do_support_realloc,
|
do_support_realloc,
|
||||||
|
include_account_index_in_err,
|
||||||
) {
|
) {
|
||||||
// Feature gate only wraps the actual error return so that the metrics and debug
|
// Feature gate only wraps the actual error return so that the metrics and debug
|
||||||
// logging generated by `check_rent_state()` can be examined before feature
|
// logging generated by `check_rent_state()` can be examined before feature
|
||||||
|
|
|
@ -404,6 +404,10 @@ pub mod disable_deploy_of_alloc_free_syscall {
|
||||||
solana_sdk::declare_id!("79HWsX9rpnnJBPcdNURVqygpMAfxdrAirzAGAVmf92im");
|
solana_sdk::declare_id!("79HWsX9rpnnJBPcdNURVqygpMAfxdrAirzAGAVmf92im");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub mod include_account_index_in_rent_error {
|
||||||
|
solana_sdk::declare_id!("2R72wpcQ7qV7aTJWUumdn8u5wmmTyXbK7qzEy7YSAgyY");
|
||||||
|
}
|
||||||
|
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
/// Map of feature identifiers to user-visible description
|
/// Map of feature identifiers to user-visible description
|
||||||
pub static ref FEATURE_NAMES: HashMap<Pubkey, &'static str> = [
|
pub static ref FEATURE_NAMES: HashMap<Pubkey, &'static str> = [
|
||||||
|
@ -498,6 +502,7 @@ lazy_static! {
|
||||||
(stake_raise_minimum_delegation_to_1_sol::id(), "Raise minimum stake delegation to 1.0 SOL #24357"),
|
(stake_raise_minimum_delegation_to_1_sol::id(), "Raise minimum stake delegation to 1.0 SOL #24357"),
|
||||||
(add_set_compute_unit_price_ix::id(), "add compute budget ix for setting a compute unit price"),
|
(add_set_compute_unit_price_ix::id(), "add compute budget ix for setting a compute unit price"),
|
||||||
(disable_deploy_of_alloc_free_syscall::id(), "disable new deployments of deprecated sol_alloc_free_ syscall"),
|
(disable_deploy_of_alloc_free_syscall::id(), "disable new deployments of deprecated sol_alloc_free_ syscall"),
|
||||||
|
(include_account_index_in_rent_error::id(), "include account index in rent tx error #25190"),
|
||||||
/*************** ADD NEW FEATURES HERE ***************/
|
/*************** ADD NEW FEATURES HERE ***************/
|
||||||
]
|
]
|
||||||
.iter()
|
.iter()
|
||||||
|
|
|
@ -141,6 +141,12 @@ pub enum TransactionError {
|
||||||
/// Transaction contains a duplicate instruction that is not allowed
|
/// Transaction contains a duplicate instruction that is not allowed
|
||||||
#[error("Transaction contains a duplicate instruction ({0}) that is not allowed")]
|
#[error("Transaction contains a duplicate instruction ({0}) that is not allowed")]
|
||||||
DuplicateInstruction(u8),
|
DuplicateInstruction(u8),
|
||||||
|
|
||||||
|
/// Transaction results in an account without insufficient funds for rent
|
||||||
|
#[error(
|
||||||
|
"Transaction results in an account ({account_index}) without insufficient funds for rent"
|
||||||
|
)]
|
||||||
|
InsufficientFundsForRent { account_index: u8 },
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<SanitizeError> for TransactionError {
|
impl From<SanitizeError> for TransactionError {
|
||||||
|
|
|
@ -21,7 +21,7 @@ message Memo {
|
||||||
message TransactionError {
|
message TransactionError {
|
||||||
TransactionErrorType transaction_error = 1;
|
TransactionErrorType transaction_error = 1;
|
||||||
InstructionError instruction_error = 2;
|
InstructionError instruction_error = 2;
|
||||||
DuplicateInstructionError duplicate_instruction_error = 3;
|
TransactionDetails transaction_details = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum TransactionErrorType {
|
enum TransactionErrorType {
|
||||||
|
@ -56,6 +56,7 @@ enum TransactionErrorType {
|
||||||
WOULD_EXCEED_MAX_VOTE_COST_LIMIT = 28;
|
WOULD_EXCEED_MAX_VOTE_COST_LIMIT = 28;
|
||||||
WOULD_EXCEED_ACCOUNT_DATA_TOTAL_LIMIT = 29;
|
WOULD_EXCEED_ACCOUNT_DATA_TOTAL_LIMIT = 29;
|
||||||
DUPLICATE_INSTRUCTION = 30;
|
DUPLICATE_INSTRUCTION = 30;
|
||||||
|
INSUFFICIENT_FUNDS_FOR_RENT = 31;
|
||||||
}
|
}
|
||||||
|
|
||||||
message InstructionError {
|
message InstructionError {
|
||||||
|
@ -64,7 +65,7 @@ message InstructionError {
|
||||||
CustomError custom = 3;
|
CustomError custom = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
message DuplicateInstructionError {
|
message TransactionDetails {
|
||||||
uint32 index = 1;
|
uint32 index = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -716,10 +716,20 @@ impl TryFrom<tx_by_addr::TransactionError> for TransactionError {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(duplicate_instruction) = transaction_error.duplicate_instruction_error {
|
if let Some(transaction_details) = transaction_error.transaction_details {
|
||||||
return Ok(TransactionError::DuplicateInstruction(
|
match transaction_error.transaction_error {
|
||||||
duplicate_instruction.index as u8,
|
30 => {
|
||||||
));
|
return Ok(TransactionError::DuplicateInstruction(
|
||||||
|
transaction_details.index as u8,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
31 => {
|
||||||
|
return Ok(TransactionError::InsufficientFundsForRent {
|
||||||
|
account_index: transaction_details.index as u8,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(match transaction_error.transaction_error {
|
Ok(match transaction_error.transaction_error {
|
||||||
|
@ -852,6 +862,9 @@ impl From<TransactionError> for tx_by_addr::TransactionError {
|
||||||
TransactionError::DuplicateInstruction(_) => {
|
TransactionError::DuplicateInstruction(_) => {
|
||||||
tx_by_addr::TransactionErrorType::DuplicateInstruction
|
tx_by_addr::TransactionErrorType::DuplicateInstruction
|
||||||
}
|
}
|
||||||
|
TransactionError::InsufficientFundsForRent { .. } => {
|
||||||
|
tx_by_addr::TransactionErrorType::InsufficientFundsForRent
|
||||||
|
}
|
||||||
} as i32,
|
} as i32,
|
||||||
instruction_error: match transaction_error {
|
instruction_error: match transaction_error {
|
||||||
TransactionError::InstructionError(index, ref instruction_error) => {
|
TransactionError::InstructionError(index, ref instruction_error) => {
|
||||||
|
@ -1023,12 +1036,17 @@ impl From<TransactionError> for tx_by_addr::TransactionError {
|
||||||
}
|
}
|
||||||
_ => None,
|
_ => None,
|
||||||
},
|
},
|
||||||
duplicate_instruction_error: match transaction_error {
|
transaction_details: match transaction_error {
|
||||||
TransactionError::DuplicateInstruction(index) => {
|
TransactionError::DuplicateInstruction(index) => {
|
||||||
Some(tx_by_addr::DuplicateInstructionError {
|
Some(tx_by_addr::TransactionDetails {
|
||||||
index: index as u32,
|
index: index as u32,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
TransactionError::InsufficientFundsForRent { account_index } => {
|
||||||
|
Some(tx_by_addr::TransactionDetails {
|
||||||
|
index: account_index as u32,
|
||||||
|
})
|
||||||
|
}
|
||||||
_ => None,
|
_ => None,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -1706,6 +1724,14 @@ mod test {
|
||||||
transaction_error,
|
transaction_error,
|
||||||
tx_by_addr_transaction_error.try_into().unwrap()
|
tx_by_addr_transaction_error.try_into().unwrap()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let transaction_error = TransactionError::InsufficientFundsForRent { account_index: 10 };
|
||||||
|
let tx_by_addr_transaction_error: tx_by_addr::TransactionError =
|
||||||
|
transaction_error.clone().into();
|
||||||
|
assert_eq!(
|
||||||
|
transaction_error,
|
||||||
|
tx_by_addr_transaction_error.try_into().unwrap()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1713,12 +1739,13 @@ mod test {
|
||||||
let ix_index = 1;
|
let ix_index = 1;
|
||||||
let custom_error = 42;
|
let custom_error = 42;
|
||||||
for error in tx_by_addr::TransactionErrorType::into_enum_iter() {
|
for error in tx_by_addr::TransactionErrorType::into_enum_iter() {
|
||||||
if error != tx_by_addr::TransactionErrorType::InstructionError {
|
match error {
|
||||||
if error == tx_by_addr::TransactionErrorType::DuplicateInstruction {
|
tx_by_addr::TransactionErrorType::DuplicateInstruction
|
||||||
|
| tx_by_addr::TransactionErrorType::InsufficientFundsForRent => {
|
||||||
let tx_by_addr_error = tx_by_addr::TransactionError {
|
let tx_by_addr_error = tx_by_addr::TransactionError {
|
||||||
transaction_error: error as i32,
|
transaction_error: error as i32,
|
||||||
instruction_error: None,
|
instruction_error: None,
|
||||||
duplicate_instruction_error: Some(tx_by_addr::DuplicateInstructionError {
|
transaction_details: Some(tx_by_addr::TransactionDetails {
|
||||||
index: ix_index,
|
index: ix_index,
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
@ -1727,11 +1754,47 @@ mod test {
|
||||||
.try_into()
|
.try_into()
|
||||||
.unwrap_or_else(|_| panic!("{:?} conversion implemented?", error));
|
.unwrap_or_else(|_| panic!("{:?} conversion implemented?", error));
|
||||||
assert_eq!(tx_by_addr_error, transaction_error.into());
|
assert_eq!(tx_by_addr_error, transaction_error.into());
|
||||||
} else {
|
}
|
||||||
|
tx_by_addr::TransactionErrorType::InstructionError => {
|
||||||
|
for ix_error in tx_by_addr::InstructionErrorType::into_enum_iter() {
|
||||||
|
if ix_error != tx_by_addr::InstructionErrorType::Custom {
|
||||||
|
let tx_by_addr_error = tx_by_addr::TransactionError {
|
||||||
|
transaction_error: error as i32,
|
||||||
|
instruction_error: Some(tx_by_addr::InstructionError {
|
||||||
|
index: ix_index,
|
||||||
|
error: ix_error as i32,
|
||||||
|
custom: None,
|
||||||
|
}),
|
||||||
|
transaction_details: None,
|
||||||
|
};
|
||||||
|
let transaction_error: TransactionError =
|
||||||
|
tx_by_addr_error.clone().try_into().unwrap_or_else(|_| {
|
||||||
|
panic!("{:?} conversion implemented?", ix_error)
|
||||||
|
});
|
||||||
|
assert_eq!(tx_by_addr_error, transaction_error.into());
|
||||||
|
} else {
|
||||||
|
let tx_by_addr_error = tx_by_addr::TransactionError {
|
||||||
|
transaction_error: error as i32,
|
||||||
|
instruction_error: Some(tx_by_addr::InstructionError {
|
||||||
|
index: ix_index,
|
||||||
|
error: ix_error as i32,
|
||||||
|
custom: Some(tx_by_addr::CustomError {
|
||||||
|
custom: custom_error,
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
transaction_details: None,
|
||||||
|
};
|
||||||
|
let transaction_error: TransactionError =
|
||||||
|
tx_by_addr_error.clone().try_into().unwrap();
|
||||||
|
assert_eq!(tx_by_addr_error, transaction_error.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
let tx_by_addr_error = tx_by_addr::TransactionError {
|
let tx_by_addr_error = tx_by_addr::TransactionError {
|
||||||
transaction_error: error as i32,
|
transaction_error: error as i32,
|
||||||
instruction_error: None,
|
instruction_error: None,
|
||||||
duplicate_instruction_error: None,
|
transaction_details: None,
|
||||||
};
|
};
|
||||||
let transaction_error: TransactionError = tx_by_addr_error
|
let transaction_error: TransactionError = tx_by_addr_error
|
||||||
.clone()
|
.clone()
|
||||||
|
@ -1739,40 +1802,6 @@ mod test {
|
||||||
.unwrap_or_else(|_| panic!("{:?} conversion implemented?", error));
|
.unwrap_or_else(|_| panic!("{:?} conversion implemented?", error));
|
||||||
assert_eq!(tx_by_addr_error, transaction_error.into());
|
assert_eq!(tx_by_addr_error, transaction_error.into());
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
for ix_error in tx_by_addr::InstructionErrorType::into_enum_iter() {
|
|
||||||
if ix_error != tx_by_addr::InstructionErrorType::Custom {
|
|
||||||
let tx_by_addr_error = tx_by_addr::TransactionError {
|
|
||||||
transaction_error: error as i32,
|
|
||||||
instruction_error: Some(tx_by_addr::InstructionError {
|
|
||||||
index: ix_index,
|
|
||||||
error: ix_error as i32,
|
|
||||||
custom: None,
|
|
||||||
}),
|
|
||||||
duplicate_instruction_error: None,
|
|
||||||
};
|
|
||||||
let transaction_error: TransactionError = tx_by_addr_error
|
|
||||||
.clone()
|
|
||||||
.try_into()
|
|
||||||
.unwrap_or_else(|_| panic!("{:?} conversion implemented?", ix_error));
|
|
||||||
assert_eq!(tx_by_addr_error, transaction_error.into());
|
|
||||||
} else {
|
|
||||||
let tx_by_addr_error = tx_by_addr::TransactionError {
|
|
||||||
transaction_error: error as i32,
|
|
||||||
instruction_error: Some(tx_by_addr::InstructionError {
|
|
||||||
index: ix_index,
|
|
||||||
error: ix_error as i32,
|
|
||||||
custom: Some(tx_by_addr::CustomError {
|
|
||||||
custom: custom_error,
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
duplicate_instruction_error: None,
|
|
||||||
};
|
|
||||||
let transaction_error: TransactionError =
|
|
||||||
tx_by_addr_error.clone().try_into().unwrap();
|
|
||||||
assert_eq!(tx_by_addr_error, transaction_error.into());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue