mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-10 14:26:26 +00:00
8ebd496fbe
* Working Altair transition tests * with fixed upstream test vectors, remove state root workaround * switch upgrade_to_altair() to returning a reference * remove test_state_transition * fix invalid fork state/block combinations error messages * avoid memory copies by reintroducing state_transition_slots(var SomeHashedBeaconState)
513 lines
22 KiB
Nim
513 lines
22 KiB
Nim
# beacon_chain
|
|
# Copyright (c) 2018-2021 Status Research & Development GmbH
|
|
# Licensed and distributed under either of
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
# State transition, as described in
|
|
# https://github.com/ethereum/eth2.0-specs/blob/master/specs/core/0_beacon-chain.md#beacon-chain-state-transition-function
|
|
#
|
|
# The entry point is `state_transition` which is at the bottom of the file!
|
|
#
|
|
# General notes about the code:
|
|
# * Weird styling - the sections taken from the spec use python styling while
|
|
# the others use NEP-1 - helps grepping identifiers in spec
|
|
# * When updating the code, add TODO sections to mark where there are clear
|
|
# improvements to be made - other than that, keep things similar to spec unless
|
|
# motivated by security or performance considerations
|
|
#
|
|
# Performance notes:
|
|
# * The state transition is used in two contexts: to verify that incoming blocks
|
|
# are correct and to replay existing blocks from database. Incoming blocks
|
|
# are processed one-by-one while replay happens multiple blocks at a time.
|
|
# * Although signature verification is the slowest operation in the state
|
|
# state transition, we skip it during replay - this is also when we repeatedly
|
|
# call the state transition, making the non-signature part of the code
|
|
# important from a performance point of view.
|
|
# * It's important to start with a prefilled cache - generating the shuffled
|
|
# list of active validators is generally very slow.
|
|
# * Throughout, the code is affected by inefficient for loop codegen, meaning
|
|
# that we have to iterate over indices and pick out the value manually:
|
|
# https://github.com/nim-lang/Nim/issues/14421
|
|
# * Throughout, we're affected by inefficient `let` borrowing, meaning we
|
|
# often have to take the address of a sequence item due to the above - look
|
|
# for `let ... = unsafeAddr sequence[idx]`
|
|
# * Throughout, we're affected by the overloading rules that prefer a `var`
|
|
# overload to a non-var overload - look for `asSeq()` - when the `var`
|
|
# overload is used, the hash tree cache is cleared, which, aside from being
|
|
# slow itself, causes additional processing to recalculate the merkle tree.
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
import
|
|
std/tables,
|
|
chronicles,
|
|
stew/results,
|
|
../extras, ../ssz/merkleization, metrics,
|
|
./datatypes/[phase0, altair], ./crypto, ./digest, ./helpers, ./signatures, ./validator, ./beaconstate,
|
|
./state_transition_block, ./state_transition_epoch,
|
|
../../nbench/bench_lab
|
|
|
|
# TODO why need anything except the first two?
|
|
type Foo = phase0.SomeSignedBeaconBlock | altair.SomeSignedBeaconBlock | phase0.SignedBeaconBlock | altair.SignedBeaconBlock | phase0.TrustedSignedBeaconBlock | altair.TrustedSignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock | altair.SigVerifiedSignedBeaconBlock
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
|
proc verify_block_signature(
|
|
#state: SomeBeaconState, signed_block: SomeSomeSignedBeaconBlock): bool {.nbench.} =
|
|
state: SomeBeaconState, signed_block: Foo): bool {.nbench.} =
|
|
#state: SomeBeaconState, signed_block: phase0.SomeSignedBeaconBlock | altair.SomeSignedBeaconBlock): bool {.nbench.} =
|
|
let
|
|
proposer_index = signed_block.message.proposer_index
|
|
if proposer_index >= state.validators.lenu64:
|
|
notice "Invalid proposer index in block",
|
|
blck = shortLog(signed_block.message)
|
|
return false
|
|
|
|
if not verify_block_signature(
|
|
state.fork, state.genesis_validators_root, signed_block.message.slot,
|
|
signed_block.message, state.validators[proposer_index].pubkey,
|
|
signed_block.signature):
|
|
notice "Block: signature verification failed",
|
|
blck = shortLog(signedBlock)
|
|
return false
|
|
|
|
true
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
|
proc verifyStateRoot(state: SomeBeaconState, blck: phase0.BeaconBlock or phase0.SigVerifiedBeaconBlock or altair.BeaconBlock or altair.SigVerifiedBeaconBlock): bool =
|
|
# This is inlined in state_transition(...) in spec.
|
|
let state_root = hash_tree_root(state)
|
|
if state_root != blck.state_root:
|
|
notice "Block: root verification failed",
|
|
block_state_root = shortLog(blck.state_root), state_root = shortLog(state_root)
|
|
false
|
|
else:
|
|
true
|
|
|
|
func verifyStateRoot(state: phase0.BeaconState, blck: phase0.TrustedBeaconBlock): bool =
|
|
# This is inlined in state_transition(...) in spec.
|
|
true
|
|
|
|
func verifyStateRoot(state: altair.BeaconState, blck: altair.TrustedBeaconBlock): bool =
|
|
# This is inlined in state_transition(...) in spec.
|
|
true
|
|
|
|
# one of these can happen on the fork block itself (it's a phase 0 block which
|
|
# creates an Altair state)
|
|
func verifyStateRoot(state: altair.BeaconState, blck: phase0.TrustedBeaconBlock): bool =
|
|
# This is inlined in state_transition(...) in spec.
|
|
true
|
|
|
|
func verifyStateRoot(state: phase0.BeaconState, blck: altair.TrustedBeaconBlock): bool =
|
|
# This is inlined in state_transition(...) in spec.
|
|
true
|
|
|
|
type
|
|
RollbackProc* = proc(v: var phase0.BeaconState) {.gcsafe, raises: [Defect].}
|
|
|
|
func noRollback*(state: var phase0.BeaconState) =
|
|
trace "Skipping rollback of broken state"
|
|
|
|
type
|
|
RollbackHashedProc* = proc(state: var phase0.HashedBeaconState) {.gcsafe, raises: [Defect].}
|
|
|
|
# Hashed-state transition functions
|
|
# ---------------------------------------------------------------
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
|
func process_slot*(
|
|
state: var SomeBeaconState, pre_state_root: Eth2Digest) {.nbench.} =
|
|
# `process_slot` is the first stage of per-slot processing - it is run for
|
|
# every slot, including epoch slots - it does not however update the slot
|
|
# number! `pre_state_root` refers to the state root of the incoming
|
|
# state before any slot processing has been done.
|
|
|
|
# Cache state root
|
|
state.state_roots[state.slot mod SLOTS_PER_HISTORICAL_ROOT] = pre_state_root
|
|
|
|
# Cache latest block header state root
|
|
if state.latest_block_header.state_root == ZERO_HASH:
|
|
state.latest_block_header.state_root = pre_state_root
|
|
|
|
# Cache block root
|
|
state.block_roots[state.slot mod SLOTS_PER_HISTORICAL_ROOT] =
|
|
hash_tree_root(state.latest_block_header)
|
|
|
|
func clear_epoch_from_cache(cache: var StateCache, epoch: Epoch) =
|
|
cache.shuffled_active_validator_indices.del epoch
|
|
let
|
|
start_slot = epoch.compute_start_slot_at_epoch
|
|
end_slot = (epoch + 1).compute_start_slot_at_epoch
|
|
|
|
for i in start_slot ..< end_slot:
|
|
cache.beacon_proposer_indices.del i
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
|
proc advance_slot(
|
|
state: var SomeBeaconState, previous_slot_state_root: Eth2Digest,
|
|
flags: UpdateFlags, cache: var StateCache, rewards: var RewardInfo) {.nbench.} =
|
|
# Do the per-slot and potentially the per-epoch processing, then bump the
|
|
# slot number - we've now arrived at the slot state on top of which a block
|
|
# optionally can be applied.
|
|
process_slot(state, previous_slot_state_root)
|
|
|
|
rewards.statuses.setLen(0)
|
|
rewards.total_balances = TotalBalances()
|
|
|
|
let is_epoch_transition = (state.slot + 1).isEpoch
|
|
if is_epoch_transition:
|
|
# Note: Genesis epoch = 0, no need to test if before Genesis
|
|
process_epoch(state, flags, cache, rewards)
|
|
clear_epoch_from_cache(cache, (state.slot + 1).compute_epoch_at_slot)
|
|
|
|
state.slot += 1
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
|
proc process_slots*(state: var SomeHashedBeaconState, slot: Slot,
|
|
cache: var StateCache, rewards: var RewardInfo,
|
|
flags: UpdateFlags = {}): bool {.nbench.} =
|
|
## Process one or more slot transitions without blocks - if the slot transtion
|
|
## passes an epoch boundary, epoch processing will run and `rewards` will be
|
|
## updated, else it will be cleared
|
|
if not (state.data.slot < slot):
|
|
if slotProcessed notin flags or state.data.slot != slot:
|
|
notice(
|
|
"Unusual request for a slot in the past",
|
|
state_root = shortLog(state.root),
|
|
current_slot = state.data.slot,
|
|
target_slot = slot
|
|
)
|
|
return false
|
|
|
|
# Catch up to the target slot
|
|
while state.data.slot < slot:
|
|
advance_slot(state.data, state.root, flags, cache, rewards)
|
|
|
|
# The root must be updated on every slot update, or the next `process_slot`
|
|
# will be incorrect
|
|
state.root = hash_tree_root(state.data)
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.1.0-alpha.6/specs/altair/fork.md#upgrading-the-state
|
|
# says to put upgrading here too, TODO. It may not work in state reply
|
|
# otherwise, since updateStateData() uses this function.
|
|
|
|
true
|
|
|
|
func noRollback*(state: var phase0.HashedBeaconState) =
|
|
trace "Skipping rollback of broken state"
|
|
|
|
type
|
|
BeaconStateFork* = enum
|
|
forkPhase0,
|
|
forkAltair
|
|
|
|
ForkedHashedBeaconState* = object
|
|
case beaconStateFork*: BeaconStateFork
|
|
of forkPhase0: hbsPhase0*: phase0.HashedBeaconState
|
|
of forkAltair: hbsAltair*: altair.HashedBeaconState
|
|
|
|
# Dispatch functions
|
|
template getStateField*(x: ForkedHashedBeaconState, y: untyped): untyped =
|
|
case x.beaconStateFork:
|
|
of forkPhase0: x.hbsPhase0.data.y
|
|
of forkAltair: x.hbsAltair.data.y
|
|
|
|
template getStateField*(x: var ForkedHashedBeaconState, y: untyped): untyped =
|
|
case x.beaconStateFork:
|
|
of forkPhase0: x.hbsPhase0.data.y
|
|
of forkAltair: x.hbsAltair.data.y
|
|
|
|
template getStateRoot*(x: ForkedHashedBeaconState): Eth2Digest =
|
|
case x.beaconStateFork:
|
|
of forkPhase0: x.hbsPhase0.root
|
|
of forkAltair: x.hbsAltair.root
|
|
|
|
template hash_tree_root*(x: ForkedHashedBeaconState): Eth2Digest =
|
|
case x.beaconStateFork:
|
|
of forkPhase0: hash_tree_root(x.hbsPhase0.data)
|
|
of forkAltair: hash_tree_root(x.hbsAltair.data)
|
|
|
|
template callWithBS*(op: untyped, y: ForkedHashedBeaconState): untyped =
|
|
let bs {.inject.} =
|
|
case y.beaconStateFork:
|
|
of forkPhase0: y.hbsPhase0.data
|
|
of forkAltair: y.hbsAltair.data
|
|
op
|
|
|
|
proc maybeUpgradeStateToAltair(
|
|
state: var ForkedHashedBeaconState, altairForkSlot: Slot) =
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.1.0-alpha.6/specs/altair/fork.md#upgrading-the-state
|
|
|
|
# Both state_transition_slots() and state_transition_block() call this, so
|
|
# only run it once by checking for existing fork.
|
|
if getStateField(state, slot) == altairForkSlot and
|
|
state.beaconStateFork == forkPhase0:
|
|
var newState = upgrade_to_altair(state.hbsPhase0.data)
|
|
state = ForkedHashedBeaconState(
|
|
beaconStateFork: forkAltair,
|
|
hbsAltair: altair.HashedBeaconState(
|
|
root: hash_tree_root(newState[]), data: newState[]))
|
|
|
|
proc state_transition_slots(
|
|
preset: RuntimePreset,
|
|
state: var ForkedHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock | phase0.TrustedSignedBeaconBlock | altair.SignedBeaconBlock,
|
|
cache: var StateCache, rewards: var RewardInfo, flags: UpdateFlags,
|
|
altairForkSlot: Slot): bool {.nbench.} =
|
|
let slot = signedBlock.message.slot
|
|
|
|
# Update the state so its slot matches that of the block
|
|
while getStateField(state, slot) < slot:
|
|
case state.beaconStateFork:
|
|
of forkPhase0:
|
|
advance_slot(
|
|
state.hbsPhase0.data, state.hbsPhase0.root, flags, cache, rewards)
|
|
|
|
if state.hbsPhase0.data.slot < slot:
|
|
# Don't update state root for the slot of the block
|
|
state.hbsPhase0.root = hash_tree_root(state.hbsPhase0.data)
|
|
of forkAltair:
|
|
advance_slot(
|
|
state.hbsAltair.data, state.hbsAltair.root, flags, cache, rewards)
|
|
|
|
if getStateField(state, slot) < slot:
|
|
# Don't update state root for the slot of the block
|
|
state.hbsAltair.root = hash_tree_root(state.hbsAltair.data)
|
|
|
|
maybeUpgradeStateToAltair(state, altairForkSlot)
|
|
|
|
true
|
|
|
|
proc state_transition_slots(
|
|
preset: RuntimePreset,
|
|
state: var SomeHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
phase0.TrustedSignedBeaconBlock | altair.SignedBeaconBlock,
|
|
cache: var StateCache, rewards: var RewardInfo, flags: UpdateFlags):
|
|
bool {.nbench.} =
|
|
# TODO remove when the HashedBeaconState state_transition is removed; it's
|
|
# to avoid requiring a wrapped/memory-copied version
|
|
let slot = signedBlock.message.slot
|
|
if not (state.data.slot < slot):
|
|
if slotProcessed notin flags or state.data.slot != slot:
|
|
notice "State must precede block",
|
|
state_root = shortLog(state.root),
|
|
current_slot = state.data.slot,
|
|
blck = shortLog(signedBlock)
|
|
return false
|
|
|
|
# Update the state so its slot matches that of the block
|
|
while state.data.slot < slot:
|
|
advance_slot(state.data, state.root, flags, cache, rewards)
|
|
|
|
if state.data.slot < slot:
|
|
# Don't update state root for the slot of the block
|
|
state.root = hash_tree_root(state.data)
|
|
|
|
true
|
|
|
|
proc state_transition_block*(
|
|
preset: RuntimePreset,
|
|
state: var SomeHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
phase0.TrustedSignedBeaconBlock | altair.SignedBeaconBlock,
|
|
cache: var StateCache, flags: UpdateFlags, rollback: RollbackHashedProc):
|
|
bool {.nbench.} =
|
|
## `rollback` is called if the transition fails and the given state has been
|
|
## partially changed. If a temporary state was given to `state_transition`,
|
|
## it is safe to use `noRollback` and leave it broken, else the state
|
|
## object should be rolled back to a consistent state. If the transition fails
|
|
## before the state has been updated, `rollback` will not be called.
|
|
|
|
# Block updates - these happen when there's a new block being suggested
|
|
# by the block proposer. Every actor in the network will update its state
|
|
# according to the contents of this block - but first they will validate
|
|
# that the block is sane.
|
|
doAssert not rollback.isNil, "use noRollback if it's ok to mess up state"
|
|
|
|
if not (skipBLSValidation in flags or
|
|
verify_block_signature(state.data, signedBlock)):
|
|
when not (state is altair.HashedBeaconState):
|
|
# TODO re-enable in Altair
|
|
rollback(state)
|
|
return false
|
|
|
|
trace "state_transition: processing block, signature passed",
|
|
signature = shortLog(signedBlock.signature),
|
|
blockRoot = shortLog(signedBlock.root)
|
|
|
|
let res = process_block(preset, state.data, signedBlock.message, flags, cache)
|
|
|
|
if not res.isOk():
|
|
debug "state_transition: process_block failed",
|
|
blck = shortLog(signedBlock.message),
|
|
slot = state.data.slot,
|
|
eth1_deposit_index = state.data.eth1_deposit_index,
|
|
deposit_root = shortLog(state.data.eth1_data.deposit_root),
|
|
error = res.error
|
|
when not (state is altair.HashedBeaconState):
|
|
# TODO re-enable in Altair
|
|
rollback(state)
|
|
return false
|
|
|
|
if not (skipStateRootValidation in flags or
|
|
verifyStateRoot(state.data, signedBlock.message)):
|
|
when not (state is altair.HashedBeaconState):
|
|
# TODO re-enable in Altair
|
|
rollback(state)
|
|
return false
|
|
|
|
# only blocks currently being produced have an empty state root - we use a
|
|
# separate function for those
|
|
doAssert signedBlock.message.state_root != Eth2Digest(),
|
|
"see makeBeaconBlock for block production"
|
|
state.root = signedBlock.message.state_root
|
|
|
|
true
|
|
|
|
proc state_transition_block*(
|
|
preset: RuntimePreset,
|
|
state: var ForkedHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
phase0.TrustedSignedBeaconBlock |
|
|
altair.SignedBeaconBlock | altair.SigVerifiedSignedBeaconBlock,
|
|
cache: var StateCache, flags: UpdateFlags,
|
|
rollback: RollbackHashedProc, altairForkSlot: Slot): bool {.nbench.} =
|
|
## `rollback` is called if the transition fails and the given state has been
|
|
## partially changed. If a temporary state was given to `state_transition`,
|
|
## it is safe to use `noRollback` and leave it broken, else the state
|
|
## object should be rolled back to a consistent state. If the transition fails
|
|
## before the state has been updated, `rollback` will not be called.
|
|
|
|
# Ensure state_transition_block()-only callers trigger this
|
|
maybeUpgradeStateToAltair(state, altairForkSlot)
|
|
|
|
case state.beaconStateFork:
|
|
of forkPhase0: state_transition_block(
|
|
preset, state.hbsPhase0, signedBlock, cache, flags, rollback)
|
|
of forkAltair: state_transition_block(
|
|
preset, state.hbsAltair, signedBlock, cache, flags, rollback)
|
|
|
|
proc state_transition*(
|
|
preset: RuntimePreset,
|
|
state: var ForkedHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
phase0.TrustedSignedBeaconBlock | altair.SignedBeaconBlock,
|
|
cache: var StateCache, rewards: var RewardInfo, flags: UpdateFlags,
|
|
rollback: RollbackHashedProc,
|
|
altairForkEpoch: Epoch = FAR_FUTURE_EPOCH): bool {.nbench.} =
|
|
## Apply a block to the state, advancing the slot counter as necessary. The
|
|
## given state must be of a lower slot, or, in case the `slotProcessed` flag
|
|
## is set, can be the slot state of the same slot as the block (where the
|
|
## slot state is the state without any block applied). To create a slot state,
|
|
## advance the state corresponding to the the parent block using
|
|
## `process_slots`.
|
|
##
|
|
## To run the state transition function in preparation for block production,
|
|
## use `makeBeaconBlock` instead.
|
|
##
|
|
## `rollback` is called if the transition fails and the given state has been
|
|
## partially changed. If a temporary state was given to `state_transition`,
|
|
## it is safe to use `noRollback` and leave it broken, else the state
|
|
## object should be rolled back to a consistent state. If the transition fails
|
|
## before the state has been updated, `rollback` will not be called.
|
|
let slot = signedBlock.message.slot
|
|
if not (getStateField(state, slot) < slot):
|
|
if slotProcessed notin flags or getStateField(state, slot) != slot:
|
|
notice "State must precede block",
|
|
state_root = shortLog(getStateRoot(state)),
|
|
current_slot = getStateField(state, slot),
|
|
blck = shortLog(signedBlock)
|
|
return false
|
|
|
|
if not state_transition_slots(
|
|
preset, state, signedBlock, cache, rewards, flags,
|
|
altairForkEpoch.compute_start_slot_at_epoch):
|
|
return false
|
|
state_transition_block(
|
|
preset, state, signedBlock, cache, flags, rollback,
|
|
altairForkEpoch.compute_start_slot_at_epoch)
|
|
|
|
proc state_transition*(
|
|
preset: RuntimePreset,
|
|
state: var SomeHashedBeaconState,
|
|
signedBlock: phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
phase0.TrustedSignedBeaconBlock | altair.SignedBeaconBlock,
|
|
cache: var StateCache, rewards: var RewardInfo, flags: UpdateFlags,
|
|
rollback: RollbackHashedProc): bool =
|
|
# Does not follow hard forks; suitable only where that's irrelevant.
|
|
# TODO remove when callers gone
|
|
let slot = signedBlock.message.slot
|
|
if not (state.data.slot < slot):
|
|
if slotProcessed notin flags or state.data.slot != slot:
|
|
notice "State must precede block",
|
|
state_root = shortLog(state.root),
|
|
current_slot = state.data.slot,
|
|
blck = shortLog(signedBlock)
|
|
return false
|
|
|
|
if not state_transition_slots(
|
|
preset, state, signedBlock, cache, rewards, flags):
|
|
return false
|
|
state_transition_block(
|
|
preset, state, signedBlock, cache, flags, rollback)
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/validator.md#preparing-for-a-beaconblock
|
|
proc makeBeaconBlock*(
|
|
preset: RuntimePreset,
|
|
state: var phase0.HashedBeaconState,
|
|
proposer_index: ValidatorIndex,
|
|
parent_root: Eth2Digest,
|
|
randao_reveal: ValidatorSig,
|
|
eth1_data: Eth1Data,
|
|
graffiti: GraffitiBytes,
|
|
attestations: seq[Attestation],
|
|
deposits: seq[Deposit],
|
|
proposerSlashings: seq[ProposerSlashing],
|
|
attesterSlashings: seq[AttesterSlashing],
|
|
voluntaryExits: seq[SignedVoluntaryExit],
|
|
executionPayload: ExecutionPayload,
|
|
rollback: RollbackHashedProc,
|
|
cache: var StateCache): Option[phase0.BeaconBlock] =
|
|
## Create a block for the given state. The last block applied to it must be
|
|
## the one identified by parent_root and process_slots must be called up to
|
|
## the slot for which a block is to be created.
|
|
|
|
# To create a block, we'll first apply a partial block to the state, skipping
|
|
# some validations.
|
|
|
|
var blck = phase0.BeaconBlock(
|
|
slot: state.data.slot,
|
|
proposer_index: proposer_index.uint64,
|
|
parent_root: parent_root,
|
|
body: phase0.BeaconBlockBody(
|
|
randao_reveal: randao_reveal,
|
|
eth1_data: eth1data,
|
|
graffiti: graffiti,
|
|
proposer_slashings: List[ProposerSlashing, Limit MAX_PROPOSER_SLASHINGS](
|
|
proposerSlashings),
|
|
attester_slashings: List[AttesterSlashing, Limit MAX_ATTESTER_SLASHINGS](
|
|
attesterSlashings),
|
|
attestations: List[Attestation, Limit MAX_ATTESTATIONS](attestations),
|
|
deposits: List[Deposit, Limit MAX_DEPOSITS](deposits),
|
|
voluntary_exits:
|
|
List[SignedVoluntaryExit, Limit MAX_VOLUNTARY_EXITS](voluntaryExits)))
|
|
|
|
let res = process_block(preset, state.data, blck, {skipBlsValidation}, cache)
|
|
|
|
if res.isErr:
|
|
warn "Unable to apply new block to state",
|
|
blck = shortLog(blck),
|
|
slot = state.data.slot,
|
|
eth1_deposit_index = state.data.eth1_deposit_index,
|
|
deposit_root = shortLog(state.data.eth1_data.deposit_root),
|
|
error = res.error
|
|
rollback(state)
|
|
return
|
|
|
|
state.root = hash_tree_root(state.data)
|
|
blck.state_root = state.root
|
|
|
|
return some(blck)
|