QUANTITY
field values are big-endiansengine_newPayload
block_hash
(not equal to Keccak256(RLP(ExecutionBlockHeader))
), might be a payload with block_hash
computed with ommersHash
, difficulty
, nonce
and mixHash
field values set to meaningful PoW values
SYNCING
block_hash
extends canonical chainblock_hash
extends a side chainSYNCING
block_hash
extends canonical chainblock_hash
extends a side chain{status: INVALID_BLOCK_HASH, latestValidHash: null, validationError: errorMessage | null}
block_hash
VALID
payload extends canonical chain, the response is {status: VALID, latestValidHash: payload.blockHash, validationError: null}
INVALID
payload extends canonical chain
{status: INVALID, latestValidHash: invalidPayload.parentHash, validationError: errorMessage | null}
INVALID
payload isn't available via JSON-RPC requestINVALID
payload isn't available via Eth network protocol requestsINVALID
payloadSYNCING
P0 <- INV_P <- P1
to EL, where INV_P
is an INVALID
payload from canonical chain, wait when INV_P
is processed by EL and check the response to newPayload(P1)
:
{status: INVALID, latestValidHash: P0.blockHash, validationError: errorMessage | null}
INV_P
, and re-orgs to a valid chain, consider a test with multiple competing valid chains{status: VALID, latestValidHash: payload.blockHash, validationError: null}
newPayload(P)
is successfully processed where P
extends canonical chainengine_forkchoiceUpdated
headBlockHash
refers to a payload from canonical chain
headBlockHash
is updated{payloadStatus: {status: VALID, latestValidHash: forkchoiceState.headBlockHash, validationError: null}, payloadId: null}
headBlockHash
refers to a payload from a side chain; wait for re-org to happen (might need polling if engine_forkchoiceUpdated
has been responded with SYNCING
)
headBlockHash
is updated{payloadStatus: {status: VALID, latestValidHash: forkchoiceState.headBlockHash, validationError: null}, payloadId: null}
safeBlockHash
is updated
safeBlockHash
refers to a payload from canonical chainsafeBlockHash
refers to a payload from side chain (re-org to a side chain happens)finalizedBlockHash
is updatedpayloadAttributes.timestamp <= headBlock.timestamp
SYNCING
P0 <- INV_P <- P1
to EL, where INV_P
is an INVALID
payload from canonical chain, wait when INV_P
is processed by EL and check EL response to forkchoiceUpdated(headBlockHash=P1.blockHash)
:
{payloadStatus: {status: INVALID, latestValidHash: P0.blockHash, validationError: errorMessage | null}, payloadId: null}
INV_P
, and re-orgs to a valid chain, consider a test with multiple competing valid chainsforkchoiceUpdated
and check finalized
, safe
and head
blocks are updatedengine_getPayload
-32001: Unknown payload
to the call with random payloadId
forkchoiceUpdated
with payload attributes and check that response is {payloadStatus: {status: VALID, latestValidHash: forkchoiceState.headBlockHash, validationError: null}, payloadId: buildProcessId}
, remember buildProcessId
getPayload(payloadId=buildProcessId)
, check that reponse contains a payload with expected field values (all from payload attributes: timestamp
, prevRandao
, suggestedFeeRecipient
+ parentHash
and some other fields that make sense to check)newPayload(P)
call where P
is recently built payload, the response MUST be {status: VALID, latestValidHash: payload.blockHash, validationError: null}
getPayload
immediately followed by forkchoiceUpdated
with initiated build process, getPayload
MUST response correctlyengine_exchangeTransitionConfiguration
Less of a priority.
terminalTotalDifficulty
is correctterminalBlockHash
override scenario (all three fields have correct values)newPayload
is responsded with {status: SYNCING, latestValidHash: null, validationError: null}
if the payload extends the canonical chain and requisite data for its validation is missingforkchoiceUpdated
is responded with {payloadStatus: {status: SYNCING, latestValidHash: null, validationError: null}, payloadId: null}
if forkchoiceState.headBlockHash
references:
forkchoiceUpdated
calls in the same order as they have been made – might be better checked by fuzzerforkchoiceUpdated
calls respecting the order of fork choice updates in the fork choice store – hard to test, less of a priorityeth
namespaceLess of a priority.
Less of a priority.
engine_forkchoiceUpdated(headhBlockHash: TB.blockHash)
and run the following checks:
headBlockHash
ommersHash == 0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
difficulty == 0
mixHash == prevRandao
nonce == 0x0000000000000000
engine_forkchoiceUpdated(headhBlockHash: INV_TB.blockHash)
{status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}
INV_TB.blockHash
INV_TB.TD < TTD
INV_TB.parent.TD >= TTD
INV_TB
's PoW seal is invalidINV_TB
is invalid according to EE rulesnewPayload(P)
, where TB <- P
and TB
is a valid terminal block, and P
is a valid payload following Block structure, Block validity, Block and ommer rewards section of EIP-3675
{status: VALID, latestValidHash: payload.blockHash, validationError: null}
TB
forkchoiceUpdated(headBlockHash: P.blockHash)
{payloadStatus: {status: VALID, latestValidHash: forkchoiceState.headBlockHash, validationError: null}, payloadId: null}
P
safe
block responded with errorfinalized
block responded with errorsafe
block points to the justified payloadfinalized
block responds with errorsafe
block is as expectedfinalized
block points to the finalized payloadA
and B
where A
is the head of the PoW chain according to EL's observation and P
(a transition payload) is ancestor of B
.forkchoiceUpdated(P)
newPayload(P)
newPayload(P1)
, where P <- P1
forkchoiceUpdated(P1)
TB1 <- P0 <- P1
and TB2 <- P0' <- P1'
P
having a transaction deploying a contract that uses PREVRANDAO
and a transaction that triggers this opcode call and stores the return value into the state
P1
, i.e. P1
must become the headnewPayload(P0') + forkchoiceUpdated(P0')
P1'
– it must become canonical chainnewPayload(P)
where INV_TB <- P
and INV_TB
is an invalid terminal block
{status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}
INV_TB.TD < TTD
INV_TB.parent.TD >= TTD
INV_TB
's PoW seal is invalidINV_TB
is invalid according to EE rulesnewPayload(P)
where TB <- P
, and TB
is a valid terminal blockforkchocieUpdated(P)
newPayload(P')
where INV_TB <- P'
forkchoiceUpdated(P')
P
INV_TB.TD < TTD
INV_TB.parent.TD >= TTD
INV_TB
's PoW seal is invalidINV_TB
is invalid according to EE rulesTB <- P0 <- P1
and TB <- INV_P0' <- P1'
, where INV_P0'
is a payload that is invalid according to EE rules
newPayload
and subsequent forkchoiceUpdated(P1')
the head must still point to P1
SYNCING
SYNCING
with TB <- P0 <- P1 <- P2 <- ... <- Pn
by submitting newPayload(Pn) + forkchoiceUpdated(Pn)
; EL must not know the chain starting from TB
and must pull it from a remote peerforkchoiceUpdated
, and check:
Pn
Pn
safe
block is as expectedfinalized
block is as expectedSYNCING
SYNCING
with INV_TB <- P0 <- P1 <- P2 <- ... <- Pn
by submitting newPayload(Pn) + forkchoiceUpdated(Pn)
; EL must not know the chain starting from INV_TB
and must pull it from a remote peerforkchoiceUpdated
, and check:
{payloadStatus: {status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}, payloadId: null}
INV_TB.TD < TTD
INV_TB.parent.TD >= TTD
INV_TB
's PoW seal is invalidINV_TB
is invalid according to EE rulesTB <- INV_P0 <- P1 <- P2 <- ... <- Pn
chain, where TB
is a valid terminal block, INV_P0
is an invalid payload of a transition block, checks:
{payloadStatus: {status: INVALID, latestValidHash: ?????, validationError: errorMessage | null}, payloadId: null}
TB <- P0 <- INV_P1 <- P2 <- ... <- Pn
chain, where TB
is a valid terminal block, INV_P0
is an invalid payload of a transition block, checks:
{payloadStatus: {status: INVALID, latestValidHash: P0.blockHash, validationError: errorMessage | null}, payloadId: null}
P0
SYNCING
and re-org
A: TB <- P0 <- P1 <- P2 <- ... <- Pn
chain, check:
Pn
SYNCING
with B: INV_TB <- P0 <- P1 <- P2 <- ... <- Pn
by submitting newPayload(Pn) + forkchoiceUpdated(Pn)
; EL must not know the chain starting from INV_TB
and must pull it from a remote peerforkchoiceUpdated
, and check:
{payloadStatus: {status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}, payloadId: null}
A_Pn
INV_TB.TD < TTD
INV_TB.parent.TD >= TTD
INV_TB
's PoW seal is invalidINV_TB
is invalid according to EE rulesB: TB <- INV_P0 <- P1 <- P2 <- ... <- Pn
chain, where B_TB
is a valid terminal block, INV_P0
is an invalid payload of a transition block, checks:
{payloadStatus: {status: INVALID, latestValidHash: ?????, validationError: errorMessage | null}, payloadId: null}
A: Pn
B: TB <- P0 <- INV_P1 <- P2 <- ... <- Pn
chain, where B_TB
is a valid terminal block, INV_P0
is an invalid payload of a transition block, checks:
{payloadStatus: {status: INVALID, latestValidHash: B_P0.blockHash, validationError: errorMessage | null}, payloadId: null}
B_P0
TERMINAL_BLOCK_HASH
override
TB
with TERMINAL_BLOCK_HASH
is not the head of PoW chain according to the TD rule, i.e. there is another block that is heavier than the one with TERMINAL_BLOCK_HASH
and even having a higher block numbersafe_block_hash == justified_payload.block_hash
TERMINAL_BLOCK_HASH_ACTIVATION_EPOCH
SYNCING
Genesis <- B0 <- ... <- Bi <- TB <- TB0 <- TB1
, where TB
is a valid terminal block
TB
to a peer via NewBlock
ETH subprotocol messsages
TB
NewBlock(TB0)
to a peer
TB
TERMINAL_BLOCK_HASH = TB.blockHash
TERMINAL_BLOCK_HASH
case with longer chain: EL stops processing descendants of a terminal block received from gossip. Consider a PoW chain: Genesis <- B0 <- ... <- Bi <- TB <- TB0 <- TB1
, where TB
is a valid terminal block
TB0
TERMINAL_BLOCK_HASH = TB.blockHash
, TTD = TB.totalDifficulty
, TERMINAL_BLOCK_NUMBER
set accordinglyNewBlock(TB1)
to a peer
TB1
– the override has been applied retroactively and clients may not remove previously imported but presently invalid TB0
from the canonical chainGenesis <- B0 <- ... <- Bi <- TB <- TB0 <- TB1
, where TB
is a valid terminal block
TB
NewBlock
and NewBlockHashes
checks
P
, where TB <- P
and TB
is a valid terminal blockNewBlock(TB')
to a peer, where TB'
is also a valid terminal block
NewBlock(TB')
NewBlock(TB0')
, where TB' <- TB0'
NewBlock(TB0')
NewBlock(TB')
to a peer, where TB''
is yet another valid terminal block
NewBlock(TB'')
engine_newPayload
INVALID
with respect to
INVALID
with respect to EE rulesSYNCING
and during the sync it's been fed with INV_TB <- P1
, where INV_TB
is invalid terminal block and P1
is a transition payload built on top of this block. Then EL receives newPayload(P2)
, where P2
is a child of P1
and MUST respond with INVALID_TERMINAL_BLOCK
TERMINAL_BLOCK_HASH
override condition{status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}
SYNCING
and during the sync it's been fed with TB <- INV_P1
, where INV_P1
is INVALID
payload of a transition block. Then EL receives forkchoiceUpdated(INV_P1)
– transition MUST NOT happenINVALID_TERMINAL_BLOCK
forkchoiceUpdated
has been calledengine_forkchoiceUpdated
safeBlockHash
is zeroes
safe
block requested via JSON-RPC is responded with errorfinalizedBlockHash
is zeroes
finalized
block requested via JSON-RPC is responded with errorforkchoiceState.headBlockHash
that references a terminal block that doesn't satisfy terminal block conditions; EL must return {payloadStatus: {status: INVALID_TERMINAL_BLOCK, latestValidHash: null, validationError: errorMessage | null}, payloadId: null}
TERMINAL_BLOCK_HASH
overrideheadBlockHash
A
is the head to EL node's observation (A
has the biggest TD), B
is not; after a call with headBlockHash = A.block_hash
, the head points to A
TERMINAL_BLOCK_HASH
overrideSYNCING
and during the sync it's been fed with INV_TB <- P1
, where INV_TB
is invalid terminal block and P1
is a transition payload built on top of this block. Then EL receives forkchoiceUpdated(P1)
and MUST respond with INVALID_TERMINAL_BLOCK
INVALID_TERMINAL_BLOCK
TRANSITION_BLOCK
ommersHash == 0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
difficulty == 0
mixHash == prevRandao
nonce == 0x0000000000000000
MAX_EXTRA_DATA_BYTES
bytesPOS_FORKCHOICE_UPDATED
event, the fork choice rule MUST be altered in the following way:
FIRST_FINALIZED_BLOCK
, set the most recent finalized block to the corresponding block nominated by the eventFORK_NEXT
parameter to the FORK_NEXT_VALUE
FIRST_FINALIZED_BLOCK
, the networking stack MUST discard the following ingress messages:
TERMINAL_BLOCK_HASH
overrideTRANSITION_BLOCK
, client software MUST set the value of the mixHash
, i.e. the field with the number 13 (0-indexed) in a block header, to the latest RANDAO mix of the post beacon state of the previous blockTRANSITION_BLOCK
, the DIFFICULTY (0x44)
instruction MUST return the value of the mixHash
fieldvalidate_merge_block
TERMINAL_BLOCK_HASH
+ TERMINAL_BLOCK_HASH_ACTIVATION_EPOCH
is_valid_terminal_pow_block
block.total_difficulty >= TERMINAL_TOTAL_DIFFICULTY
parent.total_difficulty < TERMINAL_TOTAL_DIFFICULTY
ommersHash != Keccak256(RLP([]))
, difficulty != 0
, nonce != 0x0000000000000000
, len(extraData) > MAX_EXTRA_DATA_BYTES
are considered invalidbeneficiary
account starting with TRANSITION_BLOCK
POS_FORKCHOICE_UPDATED
eventPOS_FORKCHOICE_UPDATED
event is receivedTRANSITION_BLOCK
, client software applies new block validity rulesPOS_FORKCHOICE_UPDATED
client software switched its fork choice rule to PoS LMD-GHOSTTRANSITION_BLOCK
must be a child of a terminal PoW blockNewBlockHashes (0x01)
and NewBlock (0x07)
network messages are discarded after receiving the FIRST_FINALIZED_BLOCK
TRANSITION_BLOCK
deploy a contract that stores return value of DIFFICULTY (0x44)
to the stateDIFFICULTY (0x44)
in transaction executed within the parent of TRANSITION_BLOCK
equals difficulty
field valuePREVRANDAO (0x44)
in transaction executed within TRANSITION_BLOCK
equals prevRandao
field valueis_merge_transition_complete
is_merge_transition_block
is_execution_enabled
is_execution_enabled
call to process_execution_payload
process_execution_payload
notify_new_payload
is calledassert payload.parent_hash == ...
assert payload.prev_randao == ...
assert payload.timestamp == ...
assert execution_engine.notify_new_payload(payload)
state.latest_execution_payload_header = ExecutionPayloadHeader(...)
compute_timestamp_at_slot
get_inactivity_penalty_deltas
slash_validator
process_slashings
is_valid_terminal_pow_block
validate_merge_block
notify_forkchoice_updated
finalized_block_hash = Hash32()
before transition finalizedfinalized_block_hash = store.blocks[store.finalized_block_hash.root].body.execution_payload.block_hash
after transition finalizedsafe_block_hash = Hash32()
before transition justifiedsafe_block_hash = store.blocks[store.justified_checkpoint.root].body.execution_payload.block_hash
after transition justifiedon_block
validate_merge_block
if is_merge_transition_block
notify_forkchoice_updated
is calledget_pow_block_at_terminal_total_difficulty
get_terminal_pow_block
prepare_execution_payload
get_execution_payload
is_optimistic_candidate_block
SAFE_SLOTS_TO_IMPORT_OPTIMISTICALLY
ahead of the slot of the block being importedVALID
by an execution engine (either directly or indirectly), the full validate_merge_block
MUST be run against the merge block.
TERMINAL_BLOCK_HASH
override is activated assertions from validate_merge_block
MUST prevent an optimistic importINVALID
blocksnewPayload
execution_optimistic
flag is set in API in the response to queries over optimistic block