2018-08-21 16:21:45 +00:00
|
|
|
# beacon_chain
|
2021-02-25 13:37:22 +00:00
|
|
|
# Copyright (c) 2018-2021 Status Research & Development GmbH
|
2018-08-21 16:21:45 +00:00
|
|
|
# Licensed and distributed under either of
|
2019-11-25 15:30:02 +00:00
|
|
|
# * 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).
|
2018-08-21 16:21:45 +00:00
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
2018-12-04 18:45:30 +00:00
|
|
|
# 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
|
2018-08-21 16:21:45 +00:00
|
|
|
#
|
2020-11-04 21:52:47 +00:00
|
|
|
# The entry point is `state_transition` which is at the bottom of the file!
|
2018-12-04 18:45:30 +00:00
|
|
|
#
|
2020-11-04 21:52:47 +00:00
|
|
|
# General notes about the code:
|
2018-12-04 18:45:30 +00:00
|
|
|
# * Weird styling - the sections taken from the spec use python styling while
|
|
|
|
# the others use NEP-1 - helps grepping identifiers in spec
|
2020-11-04 21:52:47 +00:00
|
|
|
# * 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 fixes (#2259)
* performance fixes
* don't mark tree cache as dirty on read-only List accesses
* store only blob in memory for keys and signatures, parse blob lazily
* compare public keys by blob instead of parsing / converting to raw
* compare Eth2Digest using non-constant-time comparison
* avoid some unnecessary validator copying
This branch will in particular speed up deposit processing which has
been slowing down block replay.
Pre (mainnet, 1600 blocks):
```
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3450.269, 0.000, 3450.269, 3450.269, 1, Initialize DB
0.417, 0.822, 0.036, 21.098, 1400, Load block from database
16.521, 0.000, 16.521, 16.521, 1, Load state from database
27.906, 50.846, 8.104, 1507.633, 1350, Apply block
52.617, 37.029, 20.640, 135.938, 50, Apply epoch block
```
Post:
```
3502.715, 0.000, 3502.715, 3502.715, 1, Initialize DB
0.080, 0.560, 0.035, 21.015, 1400, Load block from database
17.595, 0.000, 17.595, 17.595, 1, Load state from database
15.706, 11.028, 8.300, 107.537, 1350, Apply block
33.217, 12.622, 17.331, 60.580, 50, Apply epoch block
```
* more perf fixes
* load EpochRef cache into StateCache more aggressively
* point out security concern with public key cache
* reuse proposer index from state when processing block
* avoid genericAssign in a few more places
* don't parse key when signature is unparseable
* fix `==` overload for Eth2Digest
* preallocate validator list when getting active validators
* speed up proposer index calculation a little bit
* reuse cache when replaying blocks in ncli_db
* avoid a few more copying loops
```
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3279.158, 0.000, 3279.158, 3279.158, 1, Initialize DB
0.072, 0.357, 0.035, 13.400, 1400, Load block from database
17.295, 0.000, 17.295, 17.295, 1, Load state from database
5.918, 9.896, 0.198, 98.028, 1350, Apply block
15.888, 10.951, 7.902, 39.535, 50, Apply epoch block
0.000, 0.000, 0.000, 0.000, 0, Database block store
```
* clear full balance cache before processing rewards and penalties
```
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3947.901, 0.000, 3947.901, 3947.901, 1, Initialize DB
0.124, 0.506, 0.026, 202.370, 363345, Load block from database
97.614, 0.000, 97.614, 97.614, 1, Load state from database
0.186, 0.188, 0.012, 99.561, 357262, Advance slot, non-epoch
14.161, 5.966, 1.099, 395.511, 11524, Advance slot, epoch
1.372, 4.170, 0.017, 276.401, 363345, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database block store
```
2021-01-25 12:04:18 +00:00
|
|
|
#
|
|
|
|
# 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.
|
2018-08-21 16:21:45 +00:00
|
|
|
|
2020-04-22 05:53:02 +00:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2018-09-26 16:26:39 +00:00
|
|
|
import
|
2020-11-27 22:16:13 +00:00
|
|
|
std/tables,
|
2020-04-11 17:41:50 +00:00
|
|
|
chronicles,
|
2020-05-03 17:44:04 +00:00
|
|
|
stew/results,
|
2020-06-23 13:54:24 +00:00
|
|
|
../extras, ../ssz/merkleization, metrics,
|
2020-11-27 22:16:13 +00:00
|
|
|
./datatypes, ./crypto, ./digest, ./helpers, ./signatures, ./validator,
|
|
|
|
./state_transition_block, ./state_transition_epoch,
|
2020-06-23 13:54:24 +00:00
|
|
|
../../nbench/bench_lab
|
2018-12-13 16:00:55 +00:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
2020-04-09 09:41:02 +00:00
|
|
|
proc verify_block_signature*(
|
2020-06-25 10:23:10 +00:00
|
|
|
state: BeaconState, signed_block: SomeSignedBeaconBlock): bool {.nbench.} =
|
2020-06-16 05:45:04 +00:00
|
|
|
let
|
|
|
|
proposer_index = signed_block.message.proposer_index
|
2020-07-26 18:55:48 +00:00
|
|
|
if proposer_index >= state.validators.lenu64:
|
2020-04-09 09:41:02 +00:00
|
|
|
notice "Invalid proposer index in block",
|
2020-06-16 05:45:04 +00:00
|
|
|
blck = shortLog(signed_block.message)
|
2020-04-09 09:41:02 +00:00
|
|
|
return false
|
|
|
|
|
2020-06-16 05:45:04 +00:00
|
|
|
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):
|
2020-04-09 09:41:02 +00:00
|
|
|
notice "Block: signature verification failed",
|
2020-06-16 05:45:04 +00:00
|
|
|
blck = shortLog(signedBlock)
|
2020-04-09 09:41:02 +00:00
|
|
|
return false
|
|
|
|
|
|
|
|
true
|
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
2021-01-25 18:45:48 +00:00
|
|
|
proc verifyStateRoot(state: BeaconState, blck: BeaconBlock or SigVerifiedBeaconBlock): bool =
|
2019-09-05 19:52:34 +00:00
|
|
|
# This is inlined in state_transition(...) in spec.
|
2019-03-25 16:46:31 +00:00
|
|
|
let state_root = hash_tree_root(state)
|
2018-12-21 22:37:46 +00:00
|
|
|
if state_root != blck.state_root:
|
2019-01-25 17:35:22 +00:00
|
|
|
notice "Block: root verification failed",
|
2020-06-25 10:23:10 +00:00
|
|
|
block_state_root = shortLog(blck.state_root), state_root = shortLog(state_root)
|
2018-12-21 22:37:46 +00:00
|
|
|
false
|
|
|
|
else:
|
|
|
|
true
|
|
|
|
|
2020-06-25 10:23:10 +00:00
|
|
|
proc verifyStateRoot(state: BeaconState, blck: TrustedBeaconBlock): bool =
|
|
|
|
# This is inlined in state_transition(...) in spec.
|
|
|
|
true
|
|
|
|
|
2020-04-26 19:13:33 +00:00
|
|
|
type
|
2020-04-28 08:08:32 +00:00
|
|
|
RollbackProc* = proc(v: var BeaconState) {.gcsafe, raises: [Defect].}
|
2020-04-26 19:13:33 +00:00
|
|
|
|
|
|
|
proc noRollback*(state: var BeaconState) =
|
|
|
|
trace "Skipping rollback of broken state"
|
|
|
|
|
2020-04-30 06:44:19 +00:00
|
|
|
type
|
2021-03-24 16:20:55 +00:00
|
|
|
RollbackHashedProc* = proc(state: var HashedBeaconState) {.gcsafe, raises: [Defect].}
|
2018-12-27 20:14:37 +00:00
|
|
|
|
2019-07-15 21:10:40 +00:00
|
|
|
# Hashed-state transition functions
|
|
|
|
# ---------------------------------------------------------------
|
2019-05-04 14:10:45 +00:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
2020-04-30 06:44:19 +00:00
|
|
|
func process_slot*(state: var HashedBeaconState) {.nbench.} =
|
2019-06-14 07:50:14 +00:00
|
|
|
# Cache state root
|
2019-05-04 14:10:45 +00:00
|
|
|
let previous_slot_state_root = state.root
|
2019-07-01 09:42:37 +00:00
|
|
|
state.data.state_roots[state.data.slot mod SLOTS_PER_HISTORICAL_ROOT] =
|
2019-05-04 14:10:45 +00:00
|
|
|
previous_slot_state_root
|
|
|
|
|
2019-06-14 07:50:14 +00:00
|
|
|
# Cache latest block header state root
|
2019-05-04 14:10:45 +00:00
|
|
|
if state.data.latest_block_header.state_root == ZERO_HASH:
|
|
|
|
state.data.latest_block_header.state_root = previous_slot_state_root
|
|
|
|
|
2019-06-14 07:50:14 +00:00
|
|
|
# Cache block root
|
2019-07-01 07:53:42 +00:00
|
|
|
state.data.block_roots[state.data.slot mod SLOTS_PER_HISTORICAL_ROOT] =
|
2019-12-16 18:08:50 +00:00
|
|
|
hash_tree_root(state.data.latest_block_header)
|
2019-05-04 14:10:45 +00:00
|
|
|
|
2020-09-08 11:32:43 +00:00
|
|
|
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
|
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
2020-10-15 12:28:44 +00:00
|
|
|
proc advance_slot(
|
2020-07-10 20:47:39 +00:00
|
|
|
state: var HashedBeaconState, updateFlags: UpdateFlags,
|
2020-06-01 07:44:50 +00:00
|
|
|
epochCache: var StateCache) {.nbench.} =
|
2020-05-03 17:44:04 +00:00
|
|
|
process_slot(state)
|
2020-10-15 12:28:44 +00:00
|
|
|
|
2020-05-03 17:44:04 +00:00
|
|
|
let is_epoch_transition = (state.data.slot + 1).isEpoch
|
|
|
|
if is_epoch_transition:
|
|
|
|
# Note: Genesis epoch = 0, no need to test if before Genesis
|
2020-06-01 07:44:50 +00:00
|
|
|
process_epoch(state.data, updateFlags, epochCache)
|
2020-09-08 11:32:43 +00:00
|
|
|
clear_epoch_from_cache(
|
|
|
|
epochCache, (state.data.slot + 1).compute_epoch_at_slot)
|
2020-09-07 15:04:33 +00:00
|
|
|
|
2020-05-03 17:44:04 +00:00
|
|
|
state.data.slot += 1
|
|
|
|
|
2020-10-15 12:28:44 +00:00
|
|
|
# The root must be updated on every slot update, or the next `process_slot`
|
|
|
|
# will be incorrect
|
2020-07-10 20:47:39 +00:00
|
|
|
state.root = hash_tree_root(state.data)
|
2020-05-03 17:44:04 +00:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/beacon-chain.md#beacon-chain-state-transition-function
|
2020-05-09 12:43:15 +00:00
|
|
|
proc process_slots*(state: var HashedBeaconState, slot: Slot,
|
2020-09-07 15:04:33 +00:00
|
|
|
cache: var StateCache, updateFlags: UpdateFlags = {}): bool {.nbench.} =
|
2020-05-19 14:37:29 +00:00
|
|
|
if not (state.data.slot < slot):
|
2020-08-18 20:29:33 +00:00
|
|
|
if slotProcessed notin updateFlags 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
|
2019-07-15 21:10:40 +00:00
|
|
|
|
|
|
|
# Catch up to the target slot
|
|
|
|
while state.data.slot < slot:
|
2020-07-10 20:47:39 +00:00
|
|
|
advance_slot(state, updateFlags, cache)
|
2019-07-15 21:10:40 +00:00
|
|
|
|
2020-05-19 14:37:29 +00:00
|
|
|
true
|
|
|
|
|
2020-04-26 19:13:33 +00:00
|
|
|
proc noRollback*(state: var HashedBeaconState) =
|
|
|
|
trace "Skipping rollback of broken state"
|
2019-05-04 14:10:45 +00:00
|
|
|
|
2020-04-26 19:13:33 +00:00
|
|
|
proc state_transition*(
|
2020-07-07 23:02:14 +00:00
|
|
|
preset: RuntimePreset,
|
2020-06-25 10:23:10 +00:00
|
|
|
state: var HashedBeaconState, signedBlock: SomeSignedBeaconBlock,
|
2020-05-29 06:10:20 +00:00
|
|
|
stateCache: var StateCache,
|
2020-04-30 16:27:17 +00:00
|
|
|
flags: UpdateFlags, rollback: RollbackHashedProc): bool {.nbench.} =
|
2020-04-30 06:44:19 +00:00
|
|
|
## Time in the beacon chain moves by slots. Every time (haha.) that happens,
|
|
|
|
## we will update the beacon state. Normally, the state updates will be driven
|
|
|
|
## by the contents of a new block, but it may happen that the block goes
|
|
|
|
## missing - the state updates happen regardless.
|
|
|
|
##
|
|
|
|
## The flags are used to specify that certain validations should be skipped
|
|
|
|
## for the new block. This is done during block proposal, to create a state
|
|
|
|
## whose hash can be included in the new block.
|
|
|
|
##
|
|
|
|
## `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.
|
2020-04-26 19:13:33 +00:00
|
|
|
doAssert not rollback.isNil, "use noRollback if it's ok to mess up state"
|
2020-05-19 14:37:29 +00:00
|
|
|
|
2020-09-07 15:04:33 +00:00
|
|
|
# This only fails if it hasn't changed stateCache, so it can't create a false
|
|
|
|
# not-followed future history in stateCache.
|
|
|
|
if not process_slots(state, signedBlock.message.slot, stateCache, flags):
|
2020-05-19 14:37:29 +00:00
|
|
|
rollback(state)
|
|
|
|
return false
|
2019-07-15 21:10:40 +00:00
|
|
|
|
2020-04-30 06:44:19 +00:00
|
|
|
# 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.
|
2020-04-09 09:41:02 +00:00
|
|
|
if skipBLSValidation in flags or
|
2020-04-28 08:08:32 +00:00
|
|
|
verify_block_signature(state.data, signedBlock):
|
2020-04-09 09:41:02 +00:00
|
|
|
|
2020-07-16 13:16:51 +00:00
|
|
|
trace "state_transition: processing block, signature passed",
|
|
|
|
signature = shortLog(signedBlock.signature),
|
|
|
|
blockRoot = shortLog(signedBlock.root)
|
2020-12-02 12:23:10 +00:00
|
|
|
let res = process_block(preset, state.data, signedBlock.message, flags, stateCache)
|
|
|
|
if res.isOk:
|
2020-04-28 08:08:32 +00:00
|
|
|
if skipStateRootValidation in flags or verifyStateRoot(state.data, signedBlock.message):
|
2020-04-09 09:41:02 +00:00
|
|
|
# State root is what it should be - we're done!
|
2019-05-04 14:10:45 +00:00
|
|
|
|
2020-04-09 09:41:02 +00:00
|
|
|
# TODO when creating a new block, state_root is not yet set.. comparing
|
|
|
|
# with zero hash here is a bit fragile however, but this whole thing
|
|
|
|
# should go away with proper hash caching
|
2020-06-01 07:44:50 +00:00
|
|
|
# TODO shouldn't ever have to recalculate; verifyStateRoot() does it
|
2020-04-09 09:41:02 +00:00
|
|
|
state.root =
|
2020-04-28 08:08:32 +00:00
|
|
|
if signedBlock.message.state_root == Eth2Digest(): hash_tree_root(state.data)
|
2020-04-09 09:41:02 +00:00
|
|
|
else: signedBlock.message.state_root
|
2019-05-04 14:10:45 +00:00
|
|
|
|
2020-04-09 09:41:02 +00:00
|
|
|
return true
|
2020-12-02 12:23:10 +00:00
|
|
|
else:
|
|
|
|
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
|
2019-05-04 14:10:45 +00:00
|
|
|
|
|
|
|
# Block processing failed, roll back changes
|
2020-04-26 19:13:33 +00:00
|
|
|
rollback(state)
|
|
|
|
|
2019-05-04 14:10:45 +00:00
|
|
|
false
|
2020-05-22 14:21:22 +00:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/validator.md#preparing-for-a-beaconblock
|
2020-05-22 14:21:22 +00:00
|
|
|
proc makeBeaconBlock*(
|
2020-07-07 23:02:14 +00:00
|
|
|
preset: RuntimePreset,
|
2020-05-22 14:21:22 +00:00
|
|
|
state: var HashedBeaconState,
|
|
|
|
proposer_index: ValidatorIndex,
|
|
|
|
parent_root: Eth2Digest,
|
|
|
|
randao_reveal: ValidatorSig,
|
|
|
|
eth1_data: Eth1Data,
|
2020-06-29 17:30:19 +00:00
|
|
|
graffiti: GraffitiBytes,
|
2020-05-22 14:21:22 +00:00
|
|
|
attestations: seq[Attestation],
|
|
|
|
deposits: seq[Deposit],
|
2020-10-07 16:57:21 +00:00
|
|
|
proposerSlashings: seq[ProposerSlashing],
|
|
|
|
attesterSlashings: seq[AttesterSlashing],
|
|
|
|
voluntaryExits: seq[SignedVoluntaryExit],
|
2020-06-04 12:03:16 +00:00
|
|
|
rollback: RollbackHashedProc,
|
|
|
|
cache: var StateCache): Option[BeaconBlock] =
|
2020-05-22 14:21:22 +00:00
|
|
|
## 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 = BeaconBlock(
|
|
|
|
slot: state.data.slot,
|
|
|
|
proposer_index: proposer_index.uint64,
|
|
|
|
parent_root: parent_root,
|
|
|
|
body: BeaconBlockBody(
|
|
|
|
randao_reveal: randao_reveal,
|
|
|
|
eth1_data: eth1data,
|
|
|
|
graffiti: graffiti,
|
2020-10-07 16:57:21 +00:00
|
|
|
proposer_slashings: List[ProposerSlashing, Limit MAX_PROPOSER_SLASHINGS](
|
|
|
|
proposerSlashings),
|
|
|
|
attester_slashings: List[AttesterSlashing, Limit MAX_ATTESTER_SLASHINGS](
|
|
|
|
attesterSlashings),
|
2020-07-13 14:44:58 +00:00
|
|
|
attestations: List[Attestation, Limit MAX_ATTESTATIONS](attestations),
|
2020-10-07 16:57:21 +00:00
|
|
|
deposits: List[Deposit, Limit MAX_DEPOSITS](deposits),
|
|
|
|
voluntary_exits:
|
|
|
|
List[SignedVoluntaryExit, Limit MAX_VOLUNTARY_EXITS](voluntaryExits)))
|
2020-05-22 14:21:22 +00:00
|
|
|
|
2020-12-02 12:23:10 +00:00
|
|
|
let res = process_block(preset, state.data, blck, {skipBlsValidation}, cache)
|
2020-05-22 14:21:22 +00:00
|
|
|
|
2020-12-02 12:23:10 +00:00
|
|
|
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
|
2020-05-22 14:21:22 +00:00
|
|
|
rollback(state)
|
|
|
|
return
|
|
|
|
|
|
|
|
state.root = hash_tree_root(state.data)
|
|
|
|
blck.state_root = state.root
|
|
|
|
|
|
|
|
some(blck)
|