* Add failing test for unsane tx in RPC preflight (cherry picked from commite25846e1ad) * Add From for SanitizeError > TransactionError (cherry picked from commit3f73affb2e) * Sanitize transactions during RPC preflight test (cherry picked from commit29b3265dc7) * Harden RPC preflight test inputs (cherry picked from commit14339dec0a) Co-authored-by: Trent Nelson <trent@solana.com>
This commit is contained in:
		| @@ -3475,7 +3475,7 @@ pub mod tests { | |||||||
|         let block_commitment_cache = Arc::new(RwLock::new( |         let block_commitment_cache = Arc::new(RwLock::new( | ||||||
|             BlockCommitmentCache::default_with_blockstore(blockstore.clone()), |             BlockCommitmentCache::default_with_blockstore(blockstore.clone()), | ||||||
|         )); |         )); | ||||||
|         let bank_forks = new_bank_forks().0; |         let (bank_forks, mint_keypair, ..) = new_bank_forks(); | ||||||
|         let health = RpcHealth::stub(); |         let health = RpcHealth::stub(); | ||||||
|  |  | ||||||
|         // Freeze bank 0 to prevent a panic in `run_transaction_simulation()` |         // Freeze bank 0 to prevent a panic in `run_transaction_simulation()` | ||||||
| @@ -3503,8 +3503,8 @@ pub mod tests { | |||||||
|             )), |             )), | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         let bad_transaction = |         let mut bad_transaction = | ||||||
|             system_transaction::transfer(&Keypair::new(), &Pubkey::default(), 42, Hash::default()); |             system_transaction::transfer(&mint_keypair, &Pubkey::new_rand(), 42, Hash::default()); | ||||||
|  |  | ||||||
|         // sendTransaction will fail because the blockhash is invalid |         // sendTransaction will fail because the blockhash is invalid | ||||||
|         let req = format!( |         let req = format!( | ||||||
| @@ -3519,9 +3519,23 @@ pub mod tests { | |||||||
|             ) |             ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|  |         // sendTransaction will fail due to insanity | ||||||
|  |         bad_transaction.message.instructions[0].program_id_index = 255u8; | ||||||
|         let recent_blockhash = bank_forks.read().unwrap().root_bank().last_blockhash(); |         let recent_blockhash = bank_forks.read().unwrap().root_bank().last_blockhash(); | ||||||
|  |         bad_transaction.sign(&[&mint_keypair], recent_blockhash); | ||||||
|  |         let req = format!( | ||||||
|  |             r#"{{"jsonrpc":"2.0","id":1,"method":"sendTransaction","params":["{}"]}}"#, | ||||||
|  |             bs58::encode(serialize(&bad_transaction).unwrap()).into_string() | ||||||
|  |         ); | ||||||
|  |         let res = io.handle_request_sync(&req, meta.clone()); | ||||||
|  |         assert_eq!( | ||||||
|  |             res, | ||||||
|  |             Some( | ||||||
|  |                 r#"{"jsonrpc":"2.0","error":{"code":-32002,"message":"Transaction simulation failed: TransactionError::SanitizeFailure"},"id":1}"#.to_string(), | ||||||
|  |             ) | ||||||
|  |         ); | ||||||
|         let mut bad_transaction = |         let mut bad_transaction = | ||||||
|             system_transaction::transfer(&Keypair::new(), &Pubkey::default(), 42, recent_blockhash); |             system_transaction::transfer(&mint_keypair, &Pubkey::new_rand(), 42, recent_blockhash); | ||||||
|  |  | ||||||
|         // sendTransaction will fail due to poor node health |         // sendTransaction will fail due to poor node health | ||||||
|         health.stub_set_health_status(Some(RpcHealthStatus::Behind)); |         health.stub_set_health_status(Some(RpcHealthStatus::Behind)); | ||||||
|   | |||||||
| @@ -629,8 +629,7 @@ impl Accounts { | |||||||
|         use solana_sdk::sanitize::Sanitize; |         use solana_sdk::sanitize::Sanitize; | ||||||
|         let keys: Vec<Result<_>> = OrderedIterator::new(txs, txs_iteration_order) |         let keys: Vec<Result<_>> = OrderedIterator::new(txs, txs_iteration_order) | ||||||
|             .map(|tx| { |             .map(|tx| { | ||||||
|                 tx.sanitize() |                 tx.sanitize().map_err(TransactionError::from)?; | ||||||
|                     .map_err(|_| TransactionError::SanitizeFailure)?; |  | ||||||
|  |  | ||||||
|                 if Self::has_duplicates(&tx.message.account_keys) { |                 if Self::has_duplicates(&tx.message.account_keys) { | ||||||
|                     return Err(TransactionError::AccountLoadedTwice); |                     return Err(TransactionError::AccountLoadedTwice); | ||||||
|   | |||||||
| @@ -48,6 +48,7 @@ use solana_sdk::{ | |||||||
|     native_loader, nonce, |     native_loader, nonce, | ||||||
|     program_utils::limited_deserialize, |     program_utils::limited_deserialize, | ||||||
|     pubkey::Pubkey, |     pubkey::Pubkey, | ||||||
|  |     sanitize::Sanitize, | ||||||
|     signature::{Keypair, Signature}, |     signature::{Keypair, Signature}, | ||||||
|     slot_hashes::SlotHashes, |     slot_hashes::SlotHashes, | ||||||
|     slot_history::SlotHistory, |     slot_history::SlotHistory, | ||||||
| @@ -1135,7 +1136,11 @@ impl Bank { | |||||||
|         &'a self, |         &'a self, | ||||||
|         txs: &'b [Transaction], |         txs: &'b [Transaction], | ||||||
|     ) -> TransactionBatch<'a, 'b> { |     ) -> TransactionBatch<'a, 'b> { | ||||||
|         let mut batch = TransactionBatch::new(vec![Ok(()); txs.len()], &self, txs, None); |         let lock_results: Vec<_> = txs | ||||||
|  |             .iter() | ||||||
|  |             .map(|tx| tx.sanitize().map_err(|e| e.into())) | ||||||
|  |             .collect(); | ||||||
|  |         let mut batch = TransactionBatch::new(lock_results, &self, txs, None); | ||||||
|         batch.needs_unlock = false; |         batch.needs_unlock = false; | ||||||
|         batch |         batch | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -81,6 +81,12 @@ impl std::fmt::Display for TransactionError { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | impl From<SanitizeError> for TransactionError { | ||||||
|  |     fn from(_: SanitizeError) -> Self { | ||||||
|  |         Self::SanitizeFailure | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| /// An atomic transaction | /// An atomic transaction | ||||||
| #[derive(Debug, PartialEq, Default, Eq, Clone, Serialize, Deserialize)] | #[derive(Debug, PartialEq, Default, Eq, Clone, Serialize, Deserialize)] | ||||||
| pub struct Transaction { | pub struct Transaction { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user