From 302775d76b3cf7182eab652e5067f2aab46fdbe3 Mon Sep 17 00:00:00 2001 From: Mathieu <60658558+enitrat@users.noreply.github.com> Date: Fri, 6 Dec 2024 03:39:32 +0800 Subject: [PATCH] fix: revert tx if multicall input is incoherent (#1655) The fix made previously in #1647 did not take into account that the starknet state might have been changed. If we ever encounter a case where the multicall payload was wrongly formed and the number of executed calls doesn't match the input, we should revert the tx immediately. --- .../precompiles/kakarot_precompiles.cairo | 7 +++--- .../test_multicall_cairo_precompile.py | 25 +++++++++---------- 2 files changed, 15 insertions(+), 17 deletions(-) diff --git a/cairo_zero/kakarot/precompiles/kakarot_precompiles.cairo b/cairo_zero/kakarot/precompiles/kakarot_precompiles.cairo index 812313e02..80798cdb3 100644 --- a/cairo_zero/kakarot/precompiles/kakarot_precompiles.cairo +++ b/cairo_zero/kakarot/precompiles/kakarot_precompiles.cairo @@ -150,10 +150,9 @@ namespace KakarotPrecompiles { ) = Internals.execute_multiple_cairo_calls(caller_address, calls_len, calls_ptr, 0); if (reverted == FALSE and nb_executed_calls != number_of_calls) { - let (revert_reason_len, revert_reason) = Errors.precompileInputError(); - return ( - revert_reason_len, revert_reason, CAIRO_PRECOMPILE_GAS, Errors.EXCEPTIONAL_HALT - ); + with_attr error_message("Number of executed calls does not match precompile input") { + assert nb_executed_calls = number_of_calls; + } } return (output_len, output, gas_cost, reverted); diff --git a/tests/end_to_end/CairoPrecompiles/test_multicall_cairo_precompile.py b/tests/end_to_end/CairoPrecompiles/test_multicall_cairo_precompile.py index 2b103f0a8..29e316146 100644 --- a/tests/end_to_end/CairoPrecompiles/test_multicall_cairo_precompile.py +++ b/tests/end_to_end/CairoPrecompiles/test_multicall_cairo_precompile.py @@ -5,11 +5,10 @@ from eth_abi import encode from hypothesis import assume, given, settings from hypothesis import strategies as st -from starkware.cairo.lang.cairo_constants import DEFAULT_PRIME from kakarot_scripts.utils.kakarot import deploy, eth_send_transaction from kakarot_scripts.utils.starknet import get_contract, invoke -from tests.utils.errors import evm_error +from tests.utils.errors import cairo_error, evm_error @pytest_asyncio.fixture(scope="module") @@ -94,7 +93,7 @@ async def test_should_set_and_increase_counter_in_batch( expected_count = new_counter + 1 assert new_count == expected_count - @given(wrong_nb_calls=st.integers(min_value=0, max_value=DEFAULT_PRIME - 1)) + @given(wrong_nb_calls=st.integers(min_value=0, max_value=2**8 - 1)) async def test_should_fail_when_number_of_calls_mismatch_actual_calls( self, cairo_counter, owner, wrong_nb_calls ): @@ -107,16 +106,16 @@ async def test_should_fail_when_number_of_calls_mismatch_actual_calls( # modify the number of calls to be different than the actual calls tx_data = f"{wrong_nb_calls:064x}" + tx_data[64:] - _, response, success, _ = await eth_send_transaction( - to=f"0x{0x75003:040x}", - gas=21000 + 20000 * (len(calls)), - data=tx_data, - value=0, - caller_eoa=owner.starknet_contract, - ) - - assert not success - assert "Precompile: input error".encode() == bytes(response) + with cairo_error( + "Number of executed calls does not match precompile input" + ): + await eth_send_transaction( + to=f"0x{0x75003:040x}", + gas=21000 + 20000 * max(wrong_nb_calls, len(calls)), + data=tx_data, + value=0, + caller_eoa=owner.starknet_contract, + ) async def test_should_increase_counter_single_call_from_solidity( self, cairo_counter, multicall_cairo_counter_caller