diff --git a/docs/CHANGELOG.md b/docs/CHANGELOG.md index 742c91923f..5f1f5e5514 100644 --- a/docs/CHANGELOG.md +++ b/docs/CHANGELOG.md @@ -14,6 +14,7 @@ Test fixtures for use by clients are available for each release on the [Github r - ✨ Add tests for [EIP-2537: Precompile for BLS12-381 curve operations](https://eips.ethereum.org/EIPS/eip-2537) ([#499](https://github.com/ethereum/execution-spec-tests/pull/499)). - ✨ [EIP-663](https://eips.ethereum.org/EIPS/eip-663): Add `test_dupn.py` and `test_swapn.py` ([#502](https://github.com/ethereum/execution-spec-tests/pull/502)). - ✨ Add tests for [EIP-6110: Supply validator deposits on chain](https://eips.ethereum.org/EIPS/eip-6110) ([#530](https://github.com/ethereum/execution-spec-tests/pull/530)). +- ✨ Add tests for [EIP-7002: Execution layer triggerable withdrawals](https://eips.ethereum.org/EIPS/eip-7002) ([#530](https://github.com/ethereum/execution-spec-tests/pull/530)). ### 🛠️ Framework diff --git a/tests/prague/eip7002_el_triggerable_withdrawals/__init__.py b/tests/prague/eip7002_el_triggerable_withdrawals/__init__.py new file mode 100644 index 0000000000..899bbcbf57 --- /dev/null +++ b/tests/prague/eip7002_el_triggerable_withdrawals/__init__.py @@ -0,0 +1,3 @@ +""" +Cross-client EIP-7002 Tests +""" diff --git a/tests/prague/eip7002_el_triggerable_withdrawals/conftest.py b/tests/prague/eip7002_el_triggerable_withdrawals/conftest.py new file mode 100644 index 0000000000..39deca64e7 --- /dev/null +++ b/tests/prague/eip7002_el_triggerable_withdrawals/conftest.py @@ -0,0 +1,89 @@ +""" +Fixtures for the EIP-7002 deposit tests. +""" +from typing import Dict, List + +import pytest + +from ethereum_test_tools import Account, Address, Block, Header + +from .helpers import WithdrawalRequest, WithdrawalRequestInteractionBase +from .spec import Spec + + +@pytest.fixture +def included_requests( + blocks_withdrawal_requests: List[List[WithdrawalRequestInteractionBase]], +) -> List[List[WithdrawalRequest]]: + """ + Return the list of withdrawal requests that should be included in each block. + """ + excess_withdrawal_requests = 0 + carry_over_requests: List[WithdrawalRequest] = [] + per_block_included_requests: List[List[WithdrawalRequest]] = [] + for block_withdrawal_requests in blocks_withdrawal_requests: + # Get fee for the current block + current_minimum_fee = Spec.get_fee(excess_withdrawal_requests) + + # With the fee, get the valid withdrawal requests for the current block + current_block_requests = [] + for w in block_withdrawal_requests: + current_block_requests += w.valid_requests(current_minimum_fee) + + # Get the withdrawal requests that should be included in the block + pending_requests = carry_over_requests + current_block_requests + per_block_included_requests.append( + pending_requests[: Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK] + ) + carry_over_requests = pending_requests[Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK :] + + # Update the excess withdrawal requests + excess_withdrawal_requests = Spec.get_excess_withdrawal_requests( + excess_withdrawal_requests, + len(current_block_requests), + ) + return per_block_included_requests + + +@pytest.fixture +def pre( + blocks_withdrawal_requests: List[List[WithdrawalRequestInteractionBase]], +) -> Dict[Address, Account]: + """ + Initial state of the accounts. Every withdrawal transaction defines their own pre-state + requirements, and this fixture aggregates them all. + """ + pre: Dict[Address, Account] = {} + for requests in blocks_withdrawal_requests: + for d in requests: + d.update_pre(pre) + return pre + + +@pytest.fixture +def blocks( + blocks_withdrawal_requests: List[List[WithdrawalRequestInteractionBase]], + included_requests: List[List[WithdrawalRequest]], +) -> List[Block]: + """ + Return the list of blocks that should be included in the test. + """ + blocks: List[Block] = [] + address_nonce: Dict[Address, int] = {} + for i in range(len(blocks_withdrawal_requests)): + txs = [] + for r in blocks_withdrawal_requests[i]: + nonce = 0 + if r.sender_account.address in address_nonce: + nonce = address_nonce[r.sender_account.address] + txs.append(r.transaction(nonce)) + address_nonce[r.sender_account.address] = nonce + 1 + blocks.append( + Block( + txs=txs, + header_verify=Header( + requests_root=included_requests[i], + ), + ) + ) + return blocks diff --git a/tests/prague/eip7002_el_triggerable_withdrawals/helpers.py b/tests/prague/eip7002_el_triggerable_withdrawals/helpers.py new file mode 100644 index 0000000000..483e3f9de8 --- /dev/null +++ b/tests/prague/eip7002_el_triggerable_withdrawals/helpers.py @@ -0,0 +1,340 @@ +""" +Helpers for the EIP-7002 deposit tests. +""" +from dataclasses import dataclass, field +from functools import cached_property +from itertools import count +from typing import Callable, ClassVar, Dict, List + +from ethereum_test_tools import Account, Address +from ethereum_test_tools import Opcodes as Op +from ethereum_test_tools import ( + TestAddress, + TestAddress2, + TestPrivateKey, + TestPrivateKey2, + Transaction, +) +from ethereum_test_tools import WithdrawalRequest as WithdrawalRequestBase + +from .spec import Spec + + +@dataclass +class SenderAccount: + """Test sender account descriptor.""" + + address: Address + key: str + + +TestAccount1 = SenderAccount(TestAddress, TestPrivateKey) +TestAccount2 = SenderAccount(TestAddress2, TestPrivateKey2) + + +class WithdrawalRequest(WithdrawalRequestBase): + """ + Class used to describe a withdrawal request in a test. + """ + + fee: int = 0 + """ + Fee to be paid for the withdrawal request. + """ + valid: bool = True + """ + Whether the withdrawal request is valid or not. + """ + gas_limit: int = 1_000_000 + """ + Gas limit for the call. + """ + calldata_modifier: Callable[[bytes], bytes] = lambda x: x + """ + Calldata modifier function. + """ + + interaction_contract_address: ClassVar[Address] = Address( + Spec.WITHDRAWAL_REQUEST_PREDEPLOY_ADDRESS + ) + + @property + def value(self) -> int: + """ + Returns the value of the withdrawal request. + """ + return self.fee + + @cached_property + def calldata(self) -> bytes: + """ + Returns the calldata needed to call the withdrawal request contract and make the + withdrawal. + """ + return self.calldata_modifier( + self.validator_public_key + self.amount.to_bytes(8, byteorder="big") + ) + + def with_source_address(self, source_address: Address) -> "WithdrawalRequest": + """ + Return a new instance of the withdrawal request with the source address set. + """ + return self.copy(source_address=source_address) + + +@dataclass(kw_only=True) +class WithdrawalRequestInteractionBase: + """ + Base class for all types of withdrawal transactions we want to test. + """ + + sender_balance: int = 32_000_000_000_000_000_000 * 100 + """ + Balance of the account that sends the transaction. + """ + sender_account: SenderAccount = field( + default_factory=lambda: SenderAccount(TestAddress, TestPrivateKey) + ) + """ + Account that will send the transaction. + """ + + def transaction(self, nonce: int) -> Transaction: + """Return a transaction for the withdrawal request.""" + raise NotImplementedError + + def update_pre(self, base_pre: Dict[Address, Account]): + """Return the pre-state of the account.""" + raise NotImplementedError + + def valid_requests(self, current_minimum_fee: int) -> List[WithdrawalRequest]: + """Return the list of withdrawal requests that should be valid in the block.""" + raise NotImplementedError + + +@dataclass(kw_only=True) +class WithdrawalRequestTransaction(WithdrawalRequestInteractionBase): + """Class used to describe a withdrawal request originated from an externally owned account.""" + + request: WithdrawalRequest + """ + Withdrawal request to be requested by the transaction. + """ + + def transaction(self, nonce: int) -> Transaction: + """Return a transaction for the withdrawal request.""" + return Transaction( + nonce=nonce, + gas_limit=self.request.gas_limit, + gas_price=0x07, + to=self.request.interaction_contract_address, + value=self.request.value, + data=self.request.calldata, + secret_key=self.sender_account.key, + ) + + def update_pre(self, base_pre: Dict[Address, Account]): + """Return the pre-state of the account.""" + base_pre.update( + { + self.sender_account.address: Account(balance=self.sender_balance), + } + ) + + def valid_requests(self, current_minimum_fee: int) -> List[WithdrawalRequest]: + """Return the list of withdrawal requests that are valid.""" + if self.request.valid and self.request.fee >= current_minimum_fee: + return [self.request.with_source_address(self.sender_account.address)] + return [] + + +@dataclass(kw_only=True) +class WithdrawalRequestContract(WithdrawalRequestInteractionBase): + """Class used to describe a deposit originated from a contract.""" + + request: List[WithdrawalRequest] | WithdrawalRequest + """ + Withdrawal request or list of withdrawal requests to be requested by the contract. + """ + + tx_gas_limit: int = 1_000_000 + """ + Gas limit for the transaction. + """ + + contract_balance: int = 32_000_000_000_000_000_000 * 100 + """ + Balance of the contract that will make the call to the pre-deploy contract. + """ + contract_address: int = 0x200 + """ + Address of the contract that will make the call to the pre-deploy contract. + """ + + call_type: Op = field(default_factory=lambda: Op.CALL) + """ + Type of call to be used to make the withdrawal request. + """ + call_depth: int = 2 + """ + Frame depth of the pre-deploy contract when it executes the call. + """ + extra_code: bytes = b"" + """ + Extra code to be added to the contract code. + """ + + @property + def requests(self) -> List[WithdrawalRequest]: + """Return the list of withdrawal requests.""" + if not isinstance(self.request, List): + return [self.request] + return self.request + + @property + def contract_code(self) -> bytes: + """Contract code used by the relay contract.""" + code = b"" + current_offset = 0 + for r in self.requests: + value_arg = [r.value] if self.call_type in (Op.CALL, Op.CALLCODE) else [] + code += Op.CALLDATACOPY(0, current_offset, len(r.calldata)) + Op.POP( + self.call_type( + Op.GAS if r.gas_limit == -1 else r.gas_limit, + r.interaction_contract_address, + *value_arg, + 0, + len(r.calldata), + 0, + 0, + ) + ) + current_offset += len(r.calldata) + return code + self.extra_code + + def transaction(self, nonce: int) -> Transaction: + """Return a transaction for the deposit request.""" + return Transaction( + nonce=nonce, + gas_limit=self.tx_gas_limit, + gas_price=0x07, + to=self.entry_address(), + value=0, + data=b"".join(r.calldata for r in self.requests), + secret_key=self.sender_account.key, + ) + + def entry_address(self) -> Address: + """Return the address of the contract entry point.""" + if self.call_depth == 2: + return Address(self.contract_address) + elif self.call_depth > 2: + return Address(self.contract_address + self.call_depth - 2) + raise ValueError("Invalid call depth") + + def extra_contracts(self) -> Dict[Address, Account]: + """Extra contracts used to simulate call depth.""" + if self.call_depth <= 2: + return {} + return { + Address(self.contract_address + i): Account( + balance=self.contract_balance, + code=Op.CALLDATACOPY(0, 0, Op.CALLDATASIZE) + + Op.POP( + Op.CALL( + Op.GAS, + self.contract_address + i - 1, + 0, + 0, + Op.CALLDATASIZE, + 0, + 0, + ) + ), + nonce=1, + ) + for i in range(1, self.call_depth - 1) + } + + def update_pre(self, base_pre: Dict[Address, Account]): + """Return the pre-state of the account.""" + while Address(self.contract_address) in base_pre: + self.contract_address += 0x100 + base_pre.update( + { + self.sender_account.address: Account(balance=self.sender_balance), + Address(self.contract_address): Account( + balance=self.contract_balance, code=self.contract_code, nonce=1 + ), + } + ) + base_pre.update(self.extra_contracts()) + + def valid_requests(self, current_minimum_fee: int) -> List[WithdrawalRequest]: + """Return the list of withdrawal requests that are valid.""" + valid_requests: List[WithdrawalRequest] = [] + for r in self.requests: + if r.valid and r.value >= current_minimum_fee: + valid_requests.append(r.with_source_address(Address(self.contract_address))) + return valid_requests + + +def get_n_fee_increments(n: int) -> List[int]: + """ + Get the first N excess withdrawal requests that increase the fee. + """ + excess_withdrawal_requests_counts = [] + last_fee = 1 + for i in count(0): + if Spec.get_fee(i) > last_fee: + excess_withdrawal_requests_counts.append(i) + last_fee = Spec.get_fee(i) + if len(excess_withdrawal_requests_counts) == n: + break + return excess_withdrawal_requests_counts + + +def get_n_fee_increment_blocks(n: int) -> List[List[WithdrawalRequestContract]]: + """ + Return N blocks that should be included in the test such that each subsequent block has an + increasing fee for the withdrawal requests. + + This is done by calculating the number of withdrawals required to reach the next fee increment + and creating a block with that number of withdrawal requests plus the number of withdrawals + required to reach the target. + """ + blocks = [] + previous_excess = 0 + nonce = count(0) + withdrawal_index = 0 + previous_fee = 0 + for required_excess_withdrawals in get_n_fee_increments(n): + withdrawals_required = ( + required_excess_withdrawals + + Spec.TARGET_WITHDRAWAL_REQUESTS_PER_BLOCK + - previous_excess + ) + contract_address = next(nonce) + fee = Spec.get_fee(previous_excess) + assert fee > previous_fee + blocks.append( + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i, + amount=0, + fee=fee, + ) + for i in range(withdrawal_index, withdrawal_index + withdrawals_required) + ], + # Increment the contract address to avoid overwriting the previous one + contract_address=0x200 + (contract_address * 0x100), + ) + ], + ) + previous_fee = fee + withdrawal_index += withdrawals_required + previous_excess = required_excess_withdrawals + + return blocks diff --git a/tests/prague/eip7002_el_triggerable_withdrawals/spec.py b/tests/prague/eip7002_el_triggerable_withdrawals/spec.py new file mode 100644 index 0000000000..f56381e437 --- /dev/null +++ b/tests/prague/eip7002_el_triggerable_withdrawals/spec.py @@ -0,0 +1,89 @@ +""" +Common procedures to test +[EIP-7002: Execution layer triggerable withdrawals](https://eips.ethereum.org/EIPS/eip-7002) +""" # noqa: E501 + +from dataclasses import dataclass + + +@dataclass(frozen=True) +class ReferenceSpec: + """ + Defines the reference spec version and git path. + """ + + git_path: str + version: str + + +ref_spec_7002 = ReferenceSpec("EIPS/eip-7002.md", "e5af719767e789c88c0e063406c6557c8f53cfba") + + +# Constants +@dataclass(frozen=True) +class Spec: + """ + Parameters from the EIP-7002 specifications as defined at + https://eips.ethereum.org/EIPS/eip-7002#configuration + + If the parameter is not currently used within the tests, it is commented + out. + """ + + WITHDRAWAL_REQUEST_PREDEPLOY_ADDRESS = 0x00A3CA265EBCB825B45F985A16CEFB49958CE017 + SYSTEM_ADDRESS = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE + + EXCESS_WITHDRAWAL_REQUESTS_STORAGE_SLOT = 0 + WITHDRAWAL_REQUEST_COUNT_STORAGE_SLOT = 1 + WITHDRAWAL_REQUEST_QUEUE_HEAD_STORAGE_SLOT = ( + 2 # Pointer to head of the withdrawal request message queue + ) + WITHDRAWAL_REQUEST_QUEUE_TAIL_STORAGE_SLOT = ( + 3 # Pointer to the tail of the withdrawal request message queue + ) + WITHDRAWAL_REQUEST_QUEUE_STORAGE_OFFSET = ( + 4 # The start memory slot of the in-state withdrawal request message queue + ) + MAX_WITHDRAWAL_REQUESTS_PER_BLOCK = ( + 16 # Maximum number of withdrawal requests that can be de-queued into a block + ) + TARGET_WITHDRAWAL_REQUESTS_PER_BLOCK = 2 + MIN_WITHDRAWAL_REQUEST_FEE = 1 + WITHDRAWAL_REQUEST_FEE_UPDATE_FRACTION = 17 + EXCESS_RETURN_GAS_STIPEND = 2300 + + MAX_AMOUNT = 2**64 - 1 + + @staticmethod + def fake_exponential(factor: int, numerator: int, denominator: int) -> int: + """ + Used to calculate the withdrawal request fee. + """ + i = 1 + output = 0 + numerator_accumulator = factor * denominator + while numerator_accumulator > 0: + output += numerator_accumulator + numerator_accumulator = (numerator_accumulator * numerator) // (denominator * i) + i += 1 + return output // denominator + + @staticmethod + def get_fee(excess_withdrawal_requests: int) -> int: + """ + Calculate the fee for the excess withdrawal requests. + """ + return Spec.fake_exponential( + Spec.MIN_WITHDRAWAL_REQUEST_FEE, + excess_withdrawal_requests, + Spec.WITHDRAWAL_REQUEST_FEE_UPDATE_FRACTION, + ) + + @staticmethod + def get_excess_withdrawal_requests(previous_excess: int, count: int) -> int: + """ + Calculate the new excess withdrawal requests. + """ + if previous_excess + count > Spec.TARGET_WITHDRAWAL_REQUESTS_PER_BLOCK: + return previous_excess + count - Spec.TARGET_WITHDRAWAL_REQUESTS_PER_BLOCK + return 0 diff --git a/tests/prague/eip7002_el_triggerable_withdrawals/test_withdrawal_requests.py b/tests/prague/eip7002_el_triggerable_withdrawals/test_withdrawal_requests.py new file mode 100644 index 0000000000..c3d8008a2e --- /dev/null +++ b/tests/prague/eip7002_el_triggerable_withdrawals/test_withdrawal_requests.py @@ -0,0 +1,714 @@ +""" +abstract: Tests [EIP-7002: Execution layer triggerable withdrawals](https://eips.ethereum.org/EIPS/eip-7002) + Test execution layer triggered exits [EIP-7002: Execution layer triggerable withdrawals](https://eips.ethereum.org/EIPS/eip-7002) + +""" # noqa: E501 + +from typing import Dict, List + +import pytest + +from ethereum_test_tools import ( + Account, + Address, + Block, + BlockchainTestFiller, + BlockException, + Environment, + Header, + Macros, +) +from ethereum_test_tools import Opcodes as Op +from ethereum_test_tools import TestAddress, TestAddress2 + +from .helpers import ( + TestAccount2, + WithdrawalRequest, + WithdrawalRequestContract, + WithdrawalRequestInteractionBase, + WithdrawalRequestTransaction, + get_n_fee_increment_blocks, +) +from .spec import Spec, ref_spec_7002 + +REFERENCE_SPEC_GIT_PATH = ref_spec_7002.git_path +REFERENCE_SPEC_VERSION = ref_spec_7002.version + +pytestmark = pytest.mark.valid_from("Prague") + + +@pytest.mark.parametrize( + "blocks_withdrawal_requests", + [ + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + ], + id="single_block_single_withdrawal_request_from_eoa", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=0, + ), + ), + ], + ], + id="single_block_single_withdrawal_request_from_eoa_insufficient_fee", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + calldata_modifier=lambda x: x[:-1], + valid=False, + ), + ), + ], + ], + id="single_block_single_withdrawal_request_from_eoa_input_too_short", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + calldata_modifier=lambda x: x + b"\x00", + valid=False, + ), + ), + ], + ], + id="single_block_single_withdrawal_request_from_eoa_input_too_long", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=Spec.MAX_AMOUNT - 1, + fee=Spec.get_fee(0), + ), + ), + ], + ], + id="single_block_multiple_withdrawal_request_from_same_eoa", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=Spec.MAX_AMOUNT - 1, + fee=Spec.get_fee(0), + ), + sender_account=TestAccount2, + ), + ], + ], + id="single_block_multiple_withdrawal_request_from_different_eoa", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=i + 1, + amount=0 if i % 2 == 0 else Spec.MAX_AMOUNT, + fee=Spec.get_fee(0), + ), + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + ], + id="single_block_max_withdrawal_requests_from_eoa", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=0, + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=Spec.MAX_AMOUNT - 1, + fee=Spec.get_fee(0), + ), + ), + ], + ], + id="single_block_multiple_withdrawal_request_first_reverts", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=Spec.MAX_AMOUNT - 1, + fee=0, + ), + ), + ], + ], + id="single_block_multiple_withdrawal_request_last_reverts", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + # Value obtained from trace minus one + gas_limit=114_247 - 1, + valid=False, + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + ], + id="single_block_multiple_withdrawal_request_first_oog", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=0, + fee=Spec.get_fee(0), + # Value obtained from trace minus one + gas_limit=80_047 - 1, + valid=False, + ), + ), + ], + ], + id="single_block_multiple_withdrawal_request_last_oog", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=i + 1, + amount=0 if i % 2 == 0 else Spec.MAX_AMOUNT, + fee=Spec.get_fee(0), + ), + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK * 2) + ], + # Block 2, no new withdrawal requests, but queued requests from previous block + [], + # Block 3, no new nor queued withdrawal requests + [], + ], + id="multiple_block_above_max_withdrawal_requests_from_eoa", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + ], + id="single_block_single_withdrawal_request_from_contract", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=1, + amount=Spec.MAX_AMOUNT, + fee=0, + ) + ] + + [ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + ) + for i in range(1, Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_first_reverts", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK - 1) + ] + + [ + WithdrawalRequest( + validator_public_key=Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK, + amount=Spec.MAX_AMOUNT - 1 + if (Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK - 1) % 2 == 0 + else 0, + fee=0, + ) + ], + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_last_reverts", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=1, + amount=Spec.MAX_AMOUNT - 1, + gas_limit=100, + fee=Spec.get_fee(0), + valid=False, + ) + ] + + [ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + gas_limit=1_000_000, + fee=Spec.get_fee(0), + valid=True, + ) + for i in range(1, Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_first_oog", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + gas_limit=1_000_000, + valid=True, + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ] + + [ + WithdrawalRequest( + validator_public_key=Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK, + amount=Spec.MAX_AMOUNT - 1, + gas_limit=100, + fee=Spec.get_fee(0), + valid=False, + ) + ], + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_last_oog", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + valid=False, + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + extra_code=Op.REVERT(0, 0), + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_caller_reverts", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=[ + WithdrawalRequest( + validator_public_key=i + 1, + amount=Spec.MAX_AMOUNT - 1 if i % 2 == 0 else 0, + fee=Spec.get_fee(0), + valid=False, + ) + for i in range(Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK) + ], + extra_code=Macros.OOG(), + ), + ], + ], + id="single_block_multiple_withdrawal_requests_from_contract_caller_oog", + ), + pytest.param( + # Test the first 50 fee increments + get_n_fee_increment_blocks(50), + id="multiple_block_fee_increments", + ), + pytest.param( + [ + # Block 1 + [ + WithdrawalRequestContract( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + valid=False, + ), + call_type=Op.DELEGATECALL, + ), + WithdrawalRequestContract( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + valid=False, + ), + call_type=Op.STATICCALL, + ), + WithdrawalRequestContract( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + valid=False, + ), + call_type=Op.CALLCODE, + ), + ], + ], + id="single_block_single_withdrawal_request_delegatecall_staticcall_callcode", + ), + ], +) +def test_withdrawal_requests( + blockchain_test: BlockchainTestFiller, + blocks: List[Block], + pre: Dict[Address, Account], +): + """ + Test making a withdrawal request to the beacon chain. + """ + blockchain_test( + genesis_environment=Environment(), + pre=pre, + post={}, + blocks=blocks, + ) + + +@pytest.mark.parametrize( + "requests,block_body_override_requests,exception", + [ + pytest.param( + [], + [ + WithdrawalRequest( + validator_public_key=0x01, + amount=0, + source_address=Address(0), + ), + ], + BlockException.INVALID_REQUESTS, + id="no_withdrawals_non_empty_requests_list", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [], + BlockException.INVALID_REQUESTS, + id="single_withdrawal_request_empty_requests_list", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [ + WithdrawalRequest( + validator_public_key=0x02, + amount=0, + source_address=TestAddress, + ) + ], + BlockException.INVALID_REQUESTS, + id="single_withdrawal_request_public_key_mismatch", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [ + WithdrawalRequest( + validator_public_key=0x01, + amount=1, + source_address=TestAddress, + ) + ], + BlockException.INVALID_REQUESTS, + id="single_withdrawal_request_amount_mismatch", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [ + WithdrawalRequest( + validator_public_key=0x01, + amount=0, + source_address=TestAddress2, + ) + ], + BlockException.INVALID_REQUESTS, + id="single_withdrawal_request_source_address_mismatch", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x02, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [ + WithdrawalRequest( + validator_public_key=0x02, + amount=0, + source_address=TestAddress, + ), + WithdrawalRequest( + validator_public_key=0x01, + amount=0, + source_address=TestAddress, + ), + ], + BlockException.INVALID_REQUESTS, + id="two_withdrawal_requests_out_of_order", + ), + pytest.param( + [ + WithdrawalRequestTransaction( + request=WithdrawalRequest( + validator_public_key=0x01, + amount=0, + fee=Spec.get_fee(0), + ), + ), + ], + [ + WithdrawalRequest( + validator_public_key=0x01, + amount=0, + source_address=TestAddress, + ), + WithdrawalRequest( + validator_public_key=0x01, + amount=0, + source_address=TestAddress, + ), + ], + BlockException.INVALID_REQUESTS, + id="single_withdrawal_requests_duplicate_in_requests_list", + ), + ], +) +def test_withdrawal_requests_negative( + blockchain_test: BlockchainTestFiller, + requests: List[WithdrawalRequestInteractionBase], + block_body_override_requests: List[WithdrawalRequest], + exception: BlockException, +): + """ + Test blocks where the requests list and the actual withdrawal requests that happened in the + block's transactions do not match. + """ + # No previous block so fee is the base + fee = 1 + current_block_requests = [] + for w in requests: + current_block_requests += w.valid_requests(fee) + included_requests = current_block_requests[: Spec.MAX_WITHDRAWAL_REQUESTS_PER_BLOCK] + + pre: Dict[Address, Account] = {} + for d in requests: + d.update_pre(pre) + + address_nonce: Dict[Address, int] = {} + txs = [] + for r in requests: + nonce = 0 + if r.sender_account.address in address_nonce: + nonce = address_nonce[r.sender_account.address] + txs.append(r.transaction(nonce)) + address_nonce[r.sender_account.address] = nonce + 1 + blockchain_test( + genesis_environment=Environment(), + pre=pre, + post={}, + blocks=[ + Block( + txs=txs, + header_verify=Header( + requests_root=included_requests, + ), + requests=block_body_override_requests, + exception=exception, + ) + ], + ) diff --git a/whitelist.txt b/whitelist.txt index af84696211..aed22e0183 100644 --- a/whitelist.txt +++ b/whitelist.txt @@ -252,6 +252,7 @@ ppas pre Pre precompile +predeploy prepend PrevRandao prestateTracer