-
Notifications
You must be signed in to change notification settings - Fork 107
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
tests/cancun: More EIP-7516 coverage
- Loading branch information
Showing
1 changed file
with
194 additions
and
0 deletions.
There are no files selected for viewing
194 changes: 194 additions & 0 deletions
194
tests/cancun/eip7516_blobgasfee/test_blobgasfee_opcode.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,194 @@ | ||
""" | ||
abstract: Tests [EIP-7516: BLOBBASEFEE opcode](https://eips.ethereum.org/EIPS/eip-7516) | ||
Test BLOBGASFEE opcode [EIP-7516: BLOBBASEFEE opcode](https://eips.ethereum.org/EIPS/eip-7516) | ||
""" # noqa: E501 | ||
from itertools import count | ||
from typing import Dict | ||
|
||
import pytest | ||
|
||
from ethereum_test_tools import Account, Block, BlockchainTestFiller, Environment | ||
from ethereum_test_tools import Opcodes as Op | ||
from ethereum_test_tools import StateTestFiller, Storage, TestAddress, Transaction | ||
|
||
REFERENCE_SPEC_GIT_PATH = "EIPS/eip-7516.md" | ||
REFERENCE_SPEC_VERSION = "2ade0452efe8124378f35284676ddfd16dd56ecd" | ||
|
||
# Code address used to call the test bytecode on every test case. | ||
code_caller_address = 0x100 | ||
code_callee_address = 0x200 | ||
|
||
BLOBBASEFEE_GAS = 2 | ||
|
||
|
||
@pytest.fixture | ||
def call_gas() -> int: | ||
""" | ||
Amount of gas to use when calling the callee code. | ||
""" | ||
return 0xFFFF | ||
|
||
|
||
@pytest.fixture | ||
def caller_code( | ||
call_gas: int, | ||
) -> bytes: | ||
""" | ||
Bytecode used to call the bytecode containing the BLOBBASEFEE opcode. | ||
""" | ||
return Op.SSTORE(Op.NUMBER, Op.CALL(call_gas, Op.PUSH20(code_callee_address), 0, 0, 0, 0, 0)) | ||
|
||
|
||
@pytest.fixture | ||
def callee_code() -> bytes: | ||
""" | ||
Bytecode under test, by default, only call the BLOBBASEFEE opcode. | ||
""" | ||
return bytes(Op.BLOBBASEFEE + Op.STOP) | ||
|
||
|
||
@pytest.fixture | ||
def pre( | ||
caller_code: bytes, | ||
callee_code: bytes, | ||
) -> Dict: | ||
""" | ||
Prepares the pre state of all test cases, by setting the balance of the | ||
source account of all test transactions, and the required code. | ||
""" | ||
return { | ||
TestAddress: Account(balance=10**40), | ||
code_caller_address: Account( | ||
balance=0, | ||
code=caller_code, | ||
), | ||
code_callee_address: Account( | ||
balance=0, | ||
code=callee_code, | ||
), | ||
} | ||
|
||
|
||
@pytest.fixture | ||
def tx() -> Transaction: | ||
""" | ||
Prepares the test transaction, by setting the destination account, the | ||
transaction value, the transaction gas limit, and the transaction data. | ||
""" | ||
return Transaction( | ||
gas_price=1_000_000_000, | ||
gas_limit=1_000_000, | ||
to=code_caller_address, | ||
value=0, | ||
data=b"", | ||
) | ||
|
||
|
||
@pytest.mark.parametrize( | ||
"callee_code,call_fails", | ||
[ | ||
pytest.param(Op.BLOBBASEFEE * 1024, False, id="no_stack_overflow"), | ||
pytest.param(Op.BLOBBASEFEE * 1025, True, id="stack_overflow"), | ||
], | ||
) | ||
@pytest.mark.valid_from("Cancun") | ||
def test_blobbasefee_stack_overflow( | ||
state_test: StateTestFiller, | ||
pre: Dict, | ||
tx: Transaction, | ||
call_fails: bool, | ||
): | ||
""" | ||
Tests that the BLOBBASEFEE opcode produces an stack overflow by using it repatedly. | ||
""" | ||
post = { | ||
code_caller_address: Account( | ||
storage={1: 0 if call_fails else 1}, | ||
), | ||
code_callee_address: Account( | ||
balance=0, | ||
), | ||
} | ||
state_test( | ||
env=Environment(), | ||
pre=pre, | ||
txs=[tx], | ||
post=post, | ||
) | ||
|
||
|
||
@pytest.mark.parametrize( | ||
"call_gas,call_fails", | ||
[ | ||
pytest.param(BLOBBASEFEE_GAS, False, id="enough_gas"), | ||
pytest.param(BLOBBASEFEE_GAS - 1, True, id="out_of_gas"), | ||
], | ||
) | ||
@pytest.mark.valid_from("Cancun") | ||
def test_blobbasefee_out_of_gas( | ||
state_test: StateTestFiller, | ||
pre: Dict, | ||
tx: Transaction, | ||
call_fails: bool, | ||
): | ||
""" | ||
Tests that the BLOBBASEFEE opcode produces an stack overflow by using it repatedly. | ||
""" | ||
post = { | ||
code_caller_address: Account( | ||
storage={1: 0 if call_fails else 1}, | ||
), | ||
code_callee_address: Account( | ||
balance=0, | ||
), | ||
} | ||
state_test( | ||
env=Environment(), | ||
pre=pre, | ||
txs=[tx], | ||
post=post, | ||
) | ||
|
||
|
||
@pytest.mark.valid_at_transition_to("Cancun") | ||
def test_blobbasefee_before_fork( | ||
blockchain_test: BlockchainTestFiller, | ||
pre: Dict, | ||
tx: Transaction, | ||
): | ||
""" | ||
Tests that the BLOBBASEFEE opcode results on exception when called before the fork. | ||
""" | ||
code_caller_storage = Storage() | ||
|
||
nonce = count(0) | ||
|
||
timestamps = [7_500, 14_999, 15_000] | ||
|
||
blocks = [] | ||
|
||
for number, timestamp in enumerate(timestamps): | ||
blocks.append( | ||
Block( | ||
txs=[tx.with_nonce(next(nonce))], | ||
timestamp=timestamp, | ||
), | ||
) | ||
code_caller_storage[number + 1] = 0 if timestamp < 15_000 else 1 | ||
|
||
post = { | ||
code_caller_address: Account( | ||
storage=code_caller_storage, | ||
), | ||
code_callee_address: Account( | ||
balance=0, | ||
), | ||
} | ||
blockchain_test( | ||
genesis_environment=Environment(), | ||
pre=pre, | ||
blocks=blocks, | ||
post=post, | ||
) |