Skip to content

Commit

Permalink
verify blockHeader == encoded rlp in generated tests
Browse files Browse the repository at this point in the history
  • Loading branch information
winsvega committed Nov 28, 2024
1 parent 31273b7 commit 3e8a429
Show file tree
Hide file tree
Showing 4 changed files with 105 additions and 20 deletions.
79 changes: 77 additions & 2 deletions src/ethereum_test_fixtures/schemas/blockchain/genesis.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,10 @@
Define genesisHeader schema for filled .json tests
"""

from pydantic import BaseModel, model_validator
from typing import Any, Optional

from pydantic import BaseModel, Field, model_validator
from rlp import decode as rlp_decode

from ..common.types import (
DataBytes,
Expand All @@ -22,6 +25,21 @@ class BlockRecord(BaseModel):
transactions: list # noqa: N815
uncleHeaders: list # noqa: N815

blocknumber: Optional[str] = Field(
None, description="Block number for the users to read in the tests"
)

class Config:
"""Forbids any extra fields that are not declared in the model"""

extra = "forbid"


class BlockRecordShanghai(BlockRecord):
"""Block record in blockchain tests"""

withdrawals: list


class FrontierHeader(BaseModel):
"""Frontier block header in test json"""
Expand All @@ -48,6 +66,27 @@ class Config:

extra = "forbid"

def get_field_rlp_order(self) -> dict[str, Any]:
"""The order fields are encoded into rlp"""
rlp_order: dict[str, Any] = {
"parentHash": self.parentHash,
"uncleHash": self.uncleHash,
"coinbase": self.coinbase,
"stateRoot": self.stateRoot,
"transactionsTrie": self.transactionsTrie,
"receiptTrie": self.receiptTrie,
"bloom": self.bloom,
"difficulty": self.difficulty,
"number": self.number,
"gasLimit": self.gasLimit,
"gasUsed": self.gasUsed,
"timestamp": self.timestamp,
"extraData": self.extraData,
"mixHash": self.mixHash,
"nonce": self.nonce,
}
return rlp_order


class HomesteadHeader(FrontierHeader):
"""Homestead block header in test json"""
Expand All @@ -74,6 +113,12 @@ class LondonHeader(BerlinHeader):

baseFeePerGas: PrefixedEvenHex # noqa: N815

def get_field_rlp_order(self) -> dict[str, Any]:
"""The order fields are encoded into rlp"""
rlp_order: dict[str, Any] = super().get_field_rlp_order()
rlp_order["baseFeePerGas"] = self.baseFeePerGas
return rlp_order


class ParisHeader(LondonHeader):
"""Paris block header in test json"""
Expand All @@ -83,7 +128,6 @@ def check_block_header(self):
"""
Validate Paris block header rules
"""

if self.difficulty != "0x00":
raise ValueError("Starting from Paris, block difficulty must be 0x00")

Expand All @@ -93,10 +137,41 @@ class ShanghaiHeader(ParisHeader):

withdrawalsRoot: FixedHash32 # noqa: N815

def get_field_rlp_order(self) -> dict[str, Any]:
"""The order fields are encoded into rlp"""
rlp_order: dict[str, Any] = super().get_field_rlp_order()
rlp_order["withdrawalsRoot"] = self.withdrawalsRoot
return rlp_order


class CancunHeader(ShanghaiHeader):
"""Cancun block header in test json"""

blobGasUsed: PrefixedEvenHex # noqa: N815
excessBlobGas: PrefixedEvenHex # noqa: N815
parentBeaconBlockRoot: FixedHash32 # noqa: N815

def get_field_rlp_order(self) -> dict[str, Any]:
"""The order fields are encoded into rlp"""
rlp_order: dict[str, Any] = super().get_field_rlp_order()
rlp_order["blobGasUsed"] = self.blobGasUsed
rlp_order["excessBlobGas"] = self.excessBlobGas
rlp_order["parentBeaconBlockRoot"] = self.parentBeaconBlockRoot
return rlp_order


def verify_block_header_vs_rlp_string(header: FrontierHeader, rlp_string: str):
"""Check that rlp encoding of block header match header object"""
rlp = rlp_decode(bytes.fromhex(rlp_string[2:]))[0]
for rlp_index, (field_name, field) in enumerate(header.get_field_rlp_order().items()):
rlp_hex = rlp[rlp_index].hex()
"""special rlp rule"""
if rlp_hex == "" and field_name not in ["data", "to", "extraData"]:
rlp_hex = "00"
""""""
json_hex = field.root[2:]
if rlp_hex != json_hex:
raise ValueError(
f"Field `{field_name}` in json not equal to it's rlp encoding:"
f"\n {json_hex} != {rlp_hex}"
)
44 changes: 26 additions & 18 deletions src/ethereum_test_fixtures/schemas/blockchain/test.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,14 @@
Schema for filled Blockchain Test
"""

from typing import Tuple

from pydantic import BaseModel, Field, model_validator

from .genesis import (
BerlinHeader,
BlockRecord,
BlockRecordShanghai,
ByzantiumHeader,
CancunHeader,
ConstantinopleHeader,
Expand All @@ -16,6 +19,7 @@
LondonHeader,
ParisHeader,
ShanghaiHeader,
verify_block_header_vs_rlp_string,
)


Expand All @@ -31,7 +35,7 @@ class BlockchainTestFixtureModel(BaseModel):
postState: dict # noqa: N815
lastblockhash: str
genesisRLP: str # noqa: N815
blocks: list[BlockRecord]
blocks: list
sealEngine: str # noqa: N815

class Config:
Expand All @@ -45,25 +49,29 @@ def check_block_headers(self):
Validate genesis header fields based by fork
"""
# TODO str to Fork class comparison
allowed_networks = {
"Frontier": FrontierHeader,
"Homestead": HomesteadHeader,
"EIP150": HomesteadHeader,
"EIP158": HomesteadHeader,
"Byzantium": ByzantiumHeader,
"Constantinople": ConstantinopleHeader,
"ConstantinopleFix": ConstantinopleHeader,
"Istanbul": IstanbulHeader,
"Berlin": BerlinHeader,
"London": LondonHeader,
"Paris": ParisHeader,
"Shanghai": ShanghaiHeader,
"Cancun": CancunHeader,
allowed_networks: dict[str, Tuple[FrontierHeader, BlockRecord]] = {
"Frontier": (FrontierHeader, BlockRecord),
"Homestead": (HomesteadHeader, BlockRecord),
"EIP150": (HomesteadHeader, BlockRecord),
"EIP158": (HomesteadHeader, BlockRecord),
"Byzantium": (ByzantiumHeader, BlockRecord),
"Constantinople": (ConstantinopleHeader, BlockRecord),
"ConstantinopleFix": (ConstantinopleHeader, BlockRecord),
"Istanbul": (IstanbulHeader, BlockRecord),
"Berlin": (BerlinHeader, BlockRecord),
"London": (LondonHeader, BlockRecord),
"Paris": (ParisHeader, BlockRecord),
"Shanghai": (ShanghaiHeader, BlockRecordShanghai),
"Cancun": (CancunHeader, BlockRecordShanghai),
}

header_class = allowed_networks.get(self.network)
# Check that each block in test is of format of the test declared fork
header_class, record_type = allowed_networks.get(self.network)
if not header_class:
raise ValueError("Incorrect value in network field: " + self.network)
header_class(**self.genesisBlockHeader)
header = header_class(**self.genesisBlockHeader)
verify_block_header_vs_rlp_string(header, self.genesisRLP)
for block in self.blocks:
header_class(**block.blockHeader)
record: BlockRecord = record_type(**block)
header = header_class(**record.blockHeader)
verify_block_header_vs_rlp_string(header, record.rlp)
1 change: 1 addition & 0 deletions src/ethereum_test_fixtures/schemas/common/types.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ def validate_hex_hash(self):
f"The hash must be a valid hexadecimal string of "
f"{2 * self._length_in_bytes} characters after '0x'."
)
return self


class FixedHash32(FixedHash):
Expand Down
1 change: 1 addition & 0 deletions whitelist.txt
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ blockhash
blockhashes
blocknum
blocktest
blocknumber
bls
bls12
blueswen
Expand Down

0 comments on commit 3e8a429

Please sign in to comment.