From e22c71a927e864849a3158da06fb0565bd4591e8 Mon Sep 17 00:00:00 2001 From: meship-starkware Date: Tue, 28 May 2024 14:01:06 +0300 Subject: [PATCH] test(concurrency): test commit tx where the sequencer address is also the sender address --- .../blockifier/src/concurrency/fee_utils.rs | 2 +- .../src/concurrency/worker_logic_test.rs | 71 ++++++++++++++++++- 2 files changed, 70 insertions(+), 3 deletions(-) diff --git a/crates/blockifier/src/concurrency/fee_utils.rs b/crates/blockifier/src/concurrency/fee_utils.rs index 85d01871ca..1d202684f5 100644 --- a/crates/blockifier/src/concurrency/fee_utils.rs +++ b/crates/blockifier/src/concurrency/fee_utils.rs @@ -8,7 +8,7 @@ mod test; // We read account balance (sender), and sequencer balance (recipient). The balance is of type // `Uint256`, consist of two felts (lsb, msb). Hence, storage read values = // [account_balance, 0, sequencer_balance, 0] -const STORAGE_READ_SEQUENCER_BALANCE_INDICES: (usize, usize) = (2, 3); +pub const STORAGE_READ_SEQUENCER_BALANCE_INDICES: (usize, usize) = (2, 3); // Completes the fee transfer execution by fixing the call info to have the correct sequencer // balance. In concurrency mode, the fee transfer is executed with a false (constant) sequencer diff --git a/crates/blockifier/src/concurrency/worker_logic_test.rs b/crates/blockifier/src/concurrency/worker_logic_test.rs index 1b91f8bd78..e416fb987f 100644 --- a/crates/blockifier/src/concurrency/worker_logic_test.rs +++ b/crates/blockifier/src/concurrency/worker_logic_test.rs @@ -9,8 +9,10 @@ use starknet_api::{contract_address, patricia_key, stark_felt}; use super::WorkerExecutor; use crate::abi::abi_utils::get_fee_token_var_address; use crate::abi::sierra_types::next_storage_key; +use crate::concurrency::fee_utils::STORAGE_READ_SEQUENCER_BALANCE_INDICES; use crate::concurrency::scheduler::{Task, TransactionStatus}; use crate::concurrency::test_utils::safe_versioned_state_for_testing; +use crate::concurrency::utils::lock_mutex_in_array; use crate::concurrency::worker_logic::add_fee_to_sequencer_balance; use crate::context::BlockContext; use crate::execution::execution_utils::{felt_to_stark_felt, stark_felt_to_felt}; @@ -35,7 +37,7 @@ use crate::transaction::test_utils::{ use crate::transaction::transaction_execution::Transaction; use crate::{declare_tx_args, invoke_tx_args, nonce, storage_key}; -fn _trivial_calldata_invoke_tx( +fn trivial_calldata_invoke_tx( account_address: ContractAddress, nonce: Nonce, ) -> AccountTransaction { @@ -77,7 +79,7 @@ fn _transfer_to_sequencer_tx( /// Checks that the storage values of the account and sequencer balances in the /// versioned state of tx_index equals the expected values. -fn _validate_fee_transfer( +fn validate_fee_transfer( executor: &WorkerExecutor<'_, S>, account_address: ContractAddress, tx_index: usize, @@ -107,6 +109,71 @@ fn _validate_fee_transfer( } } +#[test] +// When the sequencer is the sender, we use the sequential (full) fee transfer. +// Thus, we skip the last step of commit tx, meaning the execution result before and after +// commit tx should be the same (except for re-execution changes). +fn test_commit_tx_when_sender_is_sequencer() { + let mut block_context = BlockContext::create_for_account_testing_with_concurrency_mode(true); + let account = FeatureContract::AccountWithoutValidations(CairoVersion::Cairo1); + let mut nonce_manager = NonceManager::default(); + let account_address = account.get_instance_address(0_u16); + let mut block_info = block_context.block_info.clone(); + block_info.sequencer_address = account_address; + block_context.block_info = block_info; + + let txs = [Transaction::AccountTransaction(trivial_calldata_invoke_tx( + account_address, + nonce_manager.next(account_address), + ))]; + + let state_reader = test_state_reader( + &block_context.chain_info, + BALANCE, + &[(account, txs.len().try_into().unwrap())], + ); + let versioned_state = safe_versioned_state_for_testing(state_reader); + let executor = WorkerExecutor::new(versioned_state, &txs, block_context); + let tx_index = 0; + executor.execute_tx(tx_index); + executor.commit_tx(tx_index).unwrap(); + let execution_task_outputs = lock_mutex_in_array(&executor.execution_outputs, tx_index); + let execution_result = &execution_task_outputs.as_ref().unwrap().result; + assert!(execution_result.is_ok()); + let actual_fee = execution_result.as_ref().unwrap().actual_fee.0; + let expected_sequencer_balance_low = BALANCE; + let expected_sequencer_balance_high = 0_u128; + for (index, expected_balance) in [ + // Intuitively, the expected sequencer balance should be the initial sequencer balance, but + // in transfer, we read the sender balance and subtruct the transfer amount, then + // reading the receiver balance and adding the transfer amount. As the sequencer is the + // sender and the receiver when reading the sender balance as the receiver, we get + // the sequencer balance - actual fee. + (STORAGE_READ_SEQUENCER_BALANCE_INDICES.0, expected_sequencer_balance_low - actual_fee), + (STORAGE_READ_SEQUENCER_BALANCE_INDICES.1, expected_sequencer_balance_high), + ] { + assert_eq!( + execution_result + .as_ref() + .unwrap() + .fee_transfer_call_info + .as_ref() + .unwrap() + .storage_read_values[index], + stark_felt!(expected_balance) + ); + } + drop(execution_task_outputs); + + validate_fee_transfer( + &executor, + account_address, + tx_index, + stark_felt!(BALANCE), + stark_felt!(expected_sequencer_balance_low), + ); +} + #[test] fn test_worker_execute() { // Settings.