Skip to content
This repository has been archived by the owner on Aug 21, 2024. It is now read-only.

Commit

Permalink
test(concurrency): test commit tx where the sequencer address is also…
Browse files Browse the repository at this point in the history
… the sender address
  • Loading branch information
meship-starkware committed Jun 2, 2024
1 parent 02cec2c commit 467b883
Show file tree
Hide file tree
Showing 2 changed files with 81 additions and 3 deletions.
2 changes: 1 addition & 1 deletion crates/blockifier/src/concurrency/fee_utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
82 changes: 80 additions & 2 deletions crates/blockifier/src/concurrency/worker_logic_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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::versioned_state::ThreadSafeVersionedState;
use crate::concurrency::worker_logic::add_fee_to_sequencer_balance;
use crate::context::BlockContext;
Expand All @@ -31,7 +33,7 @@ use crate::transaction::test_utils::{account_invoke_tx, l1_resource_bounds};
use crate::transaction::transaction_execution::Transaction;
use crate::{invoke_tx_args, nonce, storage_key};

fn _trivial_calldata_invoke_tx(
fn trivial_calldata_invoke_tx(
account_address: ContractAddress,
nonce_manager: &mut NonceManager,
) -> AccountTransaction {
Expand Down Expand Up @@ -86,7 +88,7 @@ fn _invalid_tx(

// This function checks that the storage values of the account and sequencer balances in the
// versioned state of tx_index are equal to the expected_storage_values.
fn _validate_fee_transfer<S: StateReader>(
fn validate_fee_transfer<S: StateReader>(
executor: &WorkerExecutor<'_, S>,
account_address: ContractAddress,
tx_index: usize,
Expand Down Expand Up @@ -139,6 +141,82 @@ fn _change_account_balance_at_version(
versioned_state.apply_writes(&writes, &ContractClassMapping::default(), &HashMap::default());
}

fn validate_fee_transfer_when_sender_is_sequencer<S: StateReader>(
executor: &WorkerExecutor<'_, S>,
tx_index: usize,
account_address: ContractAddress,
) {
let execution_task_outputs = lock_mutex_in_array(&executor.execution_outputs, tx_index);
let execution_result = &execution_task_outputs.as_ref().unwrap().result;
let actual_fee = execution_result.as_ref().unwrap().actual_fee.0;
assert!(execution_result.is_ok());
let expected_sequencer_balance_low = BALANCE - actual_fee;
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),
stark_felt!(expected_sequencer_balance_high),
);
}

#[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;

// Create transactions.
let transactions_array = [Transaction::AccountTransaction(trivial_calldata_invoke_tx(
account.get_instance_address(0_u16),
&mut nonce_manager,
))];

let state_reader = test_state_reader(
&block_context.chain_info,
BALANCE,
&[(account, transactions_array.len().try_into().unwrap())],
);
let versioned_state = safe_versioned_state_for_testing(state_reader);
let executor = WorkerExecutor::new(versioned_state, &transactions_array, block_context);
let tx_index = 0;
executor.execute_tx(tx_index);
validate_fee_transfer_when_sender_is_sequencer(&executor, tx_index, account_address);
executor.commit_tx(tx_index).unwrap();
validate_fee_transfer_when_sender_is_sequencer(&executor, tx_index, account_address);
}

#[test]
fn test_worker_execute() {
// Settings.
Expand Down

0 comments on commit 467b883

Please sign in to comment.