2021-03-11 11:10:57 +01:00
|
|
|
# beacon_chain
|
2022-01-18 13:36:52 +00:00
|
|
|
# Copyright (c) 2018-2022 Status Research & Development GmbH
|
2021-03-11 11:10:57 +01:00
|
|
|
# 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.
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
import
|
|
|
|
std/math,
|
|
|
|
stew/results,
|
|
|
|
chronicles, chronos, metrics,
|
2022-03-25 11:40:10 +00:00
|
|
|
eth/async_utils,
|
|
|
|
web3/engine_api_types,
|
|
|
|
../spec/datatypes/[phase0, altair, bellatrix],
|
2021-12-06 10:49:01 +01:00
|
|
|
../spec/[forks, signatures_batch],
|
|
|
|
../consensus_object_pools/[
|
|
|
|
attestation_pool, block_clearance, blockchain_dag, block_quarantine,
|
|
|
|
spec_cache],
|
2022-03-25 11:40:10 +00:00
|
|
|
../eth1/eth1_monitor,
|
2021-03-11 11:10:57 +01:00
|
|
|
./consensus_manager,
|
2022-03-25 11:40:10 +00:00
|
|
|
../beacon_clock,
|
2021-08-18 20:57:58 +02:00
|
|
|
../sszdump
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-12-06 10:49:01 +01:00
|
|
|
export sszdump, signatures_batch
|
2021-07-15 21:01:07 +02:00
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
# Block Processor
|
2021-03-11 11:10:57 +01:00
|
|
|
# ------------------------------------------------------------------------------
|
2021-05-28 18:34:00 +02:00
|
|
|
# The block processor moves blocks from "Incoming" to "Consensus verified"
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
declareHistogram beacon_store_block_duration_seconds,
|
|
|
|
"storeBlock() duration", buckets = [0.25, 0.5, 1, 2, 4, 8, Inf]
|
|
|
|
|
|
|
|
type
|
|
|
|
BlockEntry* = object
|
2021-07-15 21:01:07 +02:00
|
|
|
blck*: ForkedSignedBeaconBlock
|
2021-05-28 18:34:00 +02:00
|
|
|
resfut*: Future[Result[void, BlockError]]
|
|
|
|
queueTick*: Moment # Moment when block was enqueued
|
|
|
|
validationDur*: Duration # Time it took to perform gossip validation
|
2021-12-20 20:20:31 +01:00
|
|
|
src*: MsgSource
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
BlockProcessor* = object
|
|
|
|
## This manages the processing of blocks from different sources
|
2021-03-11 11:10:57 +01:00
|
|
|
## Blocks and attestations are enqueued in a gossip-validated state
|
|
|
|
##
|
|
|
|
## from:
|
|
|
|
## - Gossip (when synced)
|
|
|
|
## - SyncManager (during sync)
|
|
|
|
## - RequestManager (missing ancestor blocks)
|
|
|
|
##
|
|
|
|
## are then consensus-verified and added to:
|
|
|
|
## - the blockchain DAG
|
|
|
|
## - database
|
|
|
|
## - attestation pool
|
|
|
|
## - fork choice
|
2021-12-06 10:49:01 +01:00
|
|
|
##
|
|
|
|
## The processor will also reinsert blocks from the quarantine, should a
|
|
|
|
## parent be found.
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
# Config
|
|
|
|
# ----------------------------------------------------------------
|
|
|
|
dumpEnabled: bool
|
|
|
|
dumpDirInvalid: string
|
|
|
|
dumpDirIncoming: string
|
|
|
|
|
|
|
|
# Producers
|
|
|
|
# ----------------------------------------------------------------
|
2022-01-26 13:20:08 +01:00
|
|
|
blockQueue: AsyncQueue[BlockEntry]
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
# Consumer
|
|
|
|
# ----------------------------------------------------------------
|
|
|
|
consensusManager: ref ConsensusManager
|
|
|
|
## Blockchain DAG, AttestationPool and Quarantine
|
2022-03-25 11:40:10 +00:00
|
|
|
## Blockchain DAG, AttestationPool, Quarantine, and Eth1Manager
|
2022-01-26 13:20:08 +01:00
|
|
|
validatorMonitor: ref ValidatorMonitor
|
2021-08-19 10:45:31 +00:00
|
|
|
getBeaconTime: GetBeaconTimeFn
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-12-06 10:49:01 +01:00
|
|
|
verifier: BatchVerifier
|
|
|
|
|
2022-03-25 11:40:10 +00:00
|
|
|
proc addBlock*(
|
|
|
|
self: var BlockProcessor, src: MsgSource, blck: ForkedSignedBeaconBlock,
|
|
|
|
resfut: Future[Result[void, BlockError]] = nil,
|
|
|
|
validationDur = Duration())
|
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
# Initialization
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc new*(T: type BlockProcessor,
|
2021-04-06 13:59:11 +02:00
|
|
|
dumpEnabled: bool,
|
|
|
|
dumpDirInvalid, dumpDirIncoming: string,
|
2022-06-21 10:29:16 +02:00
|
|
|
rng: ref HmacDrbgContext, taskpool: TaskPoolPtr,
|
2021-03-11 11:10:57 +01:00
|
|
|
consensusManager: ref ConsensusManager,
|
2021-12-20 20:20:31 +01:00
|
|
|
validatorMonitor: ref ValidatorMonitor,
|
2021-08-19 10:45:31 +00:00
|
|
|
getBeaconTime: GetBeaconTimeFn): ref BlockProcessor =
|
2021-05-28 18:34:00 +02:00
|
|
|
(ref BlockProcessor)(
|
2021-04-06 13:59:11 +02:00
|
|
|
dumpEnabled: dumpEnabled,
|
|
|
|
dumpDirInvalid: dumpDirInvalid,
|
|
|
|
dumpDirIncoming: dumpDirIncoming,
|
2021-12-06 10:49:01 +01:00
|
|
|
blockQueue: newAsyncQueue[BlockEntry](),
|
2021-04-06 13:59:11 +02:00
|
|
|
consensusManager: consensusManager,
|
2021-12-20 20:20:31 +01:00
|
|
|
validatorMonitor: validatorMonitor,
|
2021-12-06 10:49:01 +01:00
|
|
|
getBeaconTime: getBeaconTime,
|
|
|
|
verifier: BatchVerifier(rng: rng, taskpool: taskpool)
|
|
|
|
)
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
# Sync callbacks
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2022-02-17 11:53:55 +00:00
|
|
|
func hasBlocks*(self: BlockProcessor): bool =
|
2021-12-06 10:49:01 +01:00
|
|
|
self.blockQueue.len() > 0
|
2021-04-26 22:39:44 +02:00
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
# Storage
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-11-05 16:39:47 +01:00
|
|
|
proc dumpInvalidBlock*(
|
2021-12-06 10:49:01 +01:00
|
|
|
self: BlockProcessor, signedBlock: ForkySignedBeaconBlock) =
|
2021-11-05 16:39:47 +01:00
|
|
|
if self.dumpEnabled:
|
|
|
|
dump(self.dumpDirInvalid, signedBlock)
|
|
|
|
|
2022-01-26 13:20:08 +01:00
|
|
|
proc dumpBlock[T](
|
2021-07-15 21:01:07 +02:00
|
|
|
self: BlockProcessor,
|
2021-11-05 08:34:34 +01:00
|
|
|
signedBlock: ForkySignedBeaconBlock,
|
2021-12-06 10:49:01 +01:00
|
|
|
res: Result[T, BlockError]) =
|
2021-03-11 11:10:57 +01:00
|
|
|
if self.dumpEnabled and res.isErr:
|
2021-12-06 10:49:01 +01:00
|
|
|
case res.error
|
|
|
|
of BlockError.Invalid:
|
2021-11-05 16:39:47 +01:00
|
|
|
self.dumpInvalidBlock(signedBlock)
|
2021-12-06 10:49:01 +01:00
|
|
|
of BlockError.MissingParent:
|
2021-11-05 16:39:47 +01:00
|
|
|
dump(self.dumpDirIncoming, signedBlock)
|
2021-03-11 11:10:57 +01:00
|
|
|
else:
|
|
|
|
discard
|
|
|
|
|
2022-01-26 13:20:08 +01:00
|
|
|
proc storeBackfillBlock(
|
|
|
|
self: var BlockProcessor,
|
|
|
|
signedBlock: ForkySignedBeaconBlock): Result[void, BlockError] =
|
|
|
|
|
|
|
|
# The block is certainly not missing any more
|
|
|
|
self.consensusManager.quarantine[].missing.del(signedBlock.root)
|
|
|
|
|
|
|
|
let res = self.consensusManager.dag.addBackfillBlock(signedBlock)
|
|
|
|
|
|
|
|
if res.isErr():
|
|
|
|
case res.error
|
|
|
|
of BlockError.MissingParent:
|
|
|
|
if signedBlock.message.parent_root in
|
|
|
|
self.consensusManager.quarantine[].unviable:
|
|
|
|
# DAG doesn't know about unviable ancestor blocks - we do! Translate
|
|
|
|
# this to the appropriate error so that sync etc doesn't retry the block
|
|
|
|
self.consensusManager.quarantine[].addUnviable(signedBlock.root)
|
|
|
|
|
|
|
|
return err(BlockError.UnviableFork)
|
|
|
|
of BlockError.UnviableFork:
|
|
|
|
# Track unviables so that descendants can be discarded properly
|
|
|
|
self.consensusManager.quarantine[].addUnviable(signedBlock.root)
|
|
|
|
else: discard
|
|
|
|
|
|
|
|
res
|
|
|
|
|
2021-10-19 17:20:55 +02:00
|
|
|
proc storeBlock*(
|
2021-07-15 21:01:07 +02:00
|
|
|
self: var BlockProcessor,
|
2021-12-20 20:20:31 +01:00
|
|
|
src: MsgSource, wallTime: BeaconTime,
|
|
|
|
signedBlock: ForkySignedBeaconBlock, queueTick: Moment = Moment.now(),
|
2021-12-02 19:34:12 +01:00
|
|
|
validationDur = Duration()): Result[BlockRef, BlockError] =
|
2021-12-20 20:20:31 +01:00
|
|
|
## storeBlock is the main entry point for unvalidated blocks - all untrusted
|
|
|
|
## blocks, regardless of origin, pass through here. When storing a block,
|
|
|
|
## we will add it to the dag and pass it to all block consumers that need
|
|
|
|
## to know about it, such as the fork choice and the monitoring
|
2021-03-11 11:10:57 +01:00
|
|
|
let
|
|
|
|
attestationPool = self.consensusManager.attestationPool
|
2021-12-02 19:34:12 +01:00
|
|
|
startTick = Moment.now()
|
2021-12-20 20:20:31 +01:00
|
|
|
wallSlot = wallTime.slotOrZero()
|
|
|
|
vm = self.validatorMonitor
|
2021-12-06 10:49:01 +01:00
|
|
|
dag = self.consensusManager.dag
|
|
|
|
|
|
|
|
# The block is certainly not missing any more
|
|
|
|
self.consensusManager.quarantine[].missing.del(signedBlock.root)
|
|
|
|
|
|
|
|
# We'll also remove the block as an orphan: it's unlikely the parent is
|
|
|
|
# missing if we get this far - should that be the case, the block will
|
|
|
|
# be re-added later
|
|
|
|
self.consensusManager.quarantine[].removeOrphan(signedBlock)
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-07-15 21:01:07 +02:00
|
|
|
type Trusted = typeof signedBlock.asTrusted()
|
Backfill support for ChainDAG (#3171)
In the ChainDAG, 3 block pointers are kept: genesis, tail and head. This
PR adds one more block pointer: the backfill block which represents the
block that has been backfilled so far.
When doing a checkpoint sync, a random block is given as starting point
- this is the tail block, and we require that the tail block has a
corresponding state.
When backfilling, we end up with blocks without corresponding states,
hence we cannot use `tail` as a backfill pointer - there is no state.
Nonetheless, we need to keep track of where we are in the backfill
process between restarts, such that we can answer GetBeaconBlocksByRange
requests.
This PR adds the basic support for backfill handling - it needs to be
integrated with backfill sync, and the REST API needs to be adjusted to
take advantage of the new backfilled blocks when responding to certain
requests.
Future work will also enable moving the tail in either direction:
* pruning means moving the tail forward in time and removing states
* backwards means recreating past states from genesis, such that
intermediate states are recreated step by step all the way to the tail -
at that point, tail, genesis and backfill will match up.
* backfilling is done when backfill != genesis - later, this will be the
WSS checkpoint instead
2021-12-13 14:36:06 +01:00
|
|
|
let blck = dag.addHeadBlock(self.verifier, signedBlock) do (
|
2021-07-15 21:01:07 +02:00
|
|
|
blckRef: BlockRef, trustedBlock: Trusted, epochRef: EpochRef):
|
2021-03-11 11:10:57 +01:00
|
|
|
# Callback add to fork choice if valid
|
|
|
|
attestationPool[].addForkChoice(
|
2021-12-21 18:56:08 +00:00
|
|
|
epochRef, blckRef, trustedBlock.message, wallTime)
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-12-20 20:20:31 +01:00
|
|
|
vm[].registerBeaconBlock(
|
|
|
|
src, wallTime, trustedBlock.message)
|
|
|
|
|
|
|
|
for attestation in trustedBlock.message.body.attestations:
|
2022-01-09 00:28:49 +01:00
|
|
|
for validator_index in get_attesting_indices(
|
|
|
|
epochRef, attestation.data.slot,
|
|
|
|
CommitteeIndex.init(attestation.data.index).expect(
|
|
|
|
"index has been checked"),
|
|
|
|
attestation.aggregation_bits):
|
|
|
|
vm[].registerAttestationInBlock(attestation.data, validator_index,
|
2022-04-06 11:23:01 +02:00
|
|
|
trustedBlock.message.slot)
|
2021-12-20 20:20:31 +01:00
|
|
|
|
2022-03-16 08:20:40 +01:00
|
|
|
withState(dag[].clearanceState):
|
2021-12-20 20:20:31 +01:00
|
|
|
when stateFork >= BeaconStateFork.Altair and
|
|
|
|
Trusted isnot phase0.TrustedSignedBeaconBlock: # altair+
|
|
|
|
for i in trustedBlock.message.body.sync_aggregate.sync_committee_bits.oneIndices():
|
|
|
|
vm[].registerSyncAggregateInBlock(
|
|
|
|
trustedBlock.message.slot, trustedBlock.root,
|
|
|
|
state.data.current_sync_committee.pubkeys.data[i])
|
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
self.dumpBlock(signedBlock, blck)
|
|
|
|
|
|
|
|
# There can be a scenario where we receive a block we already received.
|
|
|
|
# However this block was before the last finalized epoch and so its parent
|
|
|
|
# was pruned from the ForkChoice.
|
2021-12-06 10:49:01 +01:00
|
|
|
if blck.isErr():
|
2022-01-26 13:20:08 +01:00
|
|
|
case blck.error()
|
|
|
|
of BlockError.MissingParent:
|
|
|
|
if signedBlock.message.parent_root in
|
|
|
|
self.consensusManager.quarantine[].unviable:
|
|
|
|
# DAG doesn't know about unviable ancestor blocks - we do! Translate
|
|
|
|
# this to the appropriate error so that sync etc doesn't retry the block
|
|
|
|
self.consensusManager.quarantine[].addUnviable(signedBlock.root)
|
|
|
|
|
|
|
|
return err(BlockError.UnviableFork)
|
|
|
|
|
|
|
|
if not self.consensusManager.quarantine[].addOrphan(
|
|
|
|
dag.finalizedHead.slot, ForkedSignedBeaconBlock.init(signedBlock)):
|
2021-12-06 10:49:01 +01:00
|
|
|
debug "Block quarantine full",
|
|
|
|
blockRoot = shortLog(signedBlock.root),
|
|
|
|
blck = shortLog(signedBlock.message),
|
|
|
|
signature = shortLog(signedBlock.signature)
|
2022-01-26 13:20:08 +01:00
|
|
|
of BlockError.UnviableFork:
|
|
|
|
# Track unviables so that descendants can be discarded properly
|
|
|
|
self.consensusManager.quarantine[].addUnviable(signedBlock.root)
|
|
|
|
else: discard
|
2021-12-06 10:49:01 +01:00
|
|
|
|
|
|
|
return blck
|
2021-12-02 19:34:12 +01:00
|
|
|
|
|
|
|
let storeBlockTick = Moment.now()
|
|
|
|
|
|
|
|
# Eagerly update head: the incoming block "should" get selected
|
2021-12-21 18:56:08 +00:00
|
|
|
self.consensusManager[].updateHead(wallTime.slotOrZero)
|
2021-12-02 19:34:12 +01:00
|
|
|
|
|
|
|
let
|
|
|
|
updateHeadTick = Moment.now()
|
|
|
|
queueDur = startTick - queueTick
|
|
|
|
storeBlockDur = storeBlockTick - startTick
|
|
|
|
updateHeadDur = updateHeadTick - storeBlockTick
|
|
|
|
|
|
|
|
beacon_store_block_duration_seconds.observe(storeBlockDur.toFloatSeconds())
|
|
|
|
|
|
|
|
debug "Block processed",
|
|
|
|
localHeadSlot = self.consensusManager.dag.head.slot,
|
|
|
|
blockSlot = blck.get().slot,
|
|
|
|
validationDur, queueDur, storeBlockDur, updateHeadDur
|
|
|
|
|
2021-12-06 10:49:01 +01:00
|
|
|
for quarantined in self.consensusManager.quarantine[].pop(blck.get().root):
|
|
|
|
# Process the blocks that had the newly accepted block as parent
|
2021-12-20 20:20:31 +01:00
|
|
|
self.addBlock(MsgSource.gossip, quarantined)
|
2021-12-06 10:49:01 +01:00
|
|
|
|
|
|
|
blck
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2022-01-26 13:20:08 +01:00
|
|
|
# Enqueue
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
proc addBlock*(
|
|
|
|
self: var BlockProcessor, src: MsgSource, blck: ForkedSignedBeaconBlock,
|
|
|
|
resfut: Future[Result[void, BlockError]] = nil,
|
|
|
|
validationDur = Duration()) =
|
|
|
|
## Enqueue a Gossip-validated block for consensus verification
|
|
|
|
# Backpressure:
|
|
|
|
# There is no backpressure here - producers must wait for `resfut` to
|
|
|
|
# constrain their own processing
|
|
|
|
# Producers:
|
|
|
|
# - Gossip (when synced)
|
|
|
|
# - SyncManager (during sync)
|
|
|
|
# - RequestManager (missing ancestor blocks)
|
|
|
|
|
|
|
|
withBlck(blck):
|
|
|
|
if blck.message.slot <= self.consensusManager.dag.finalizedHead.slot:
|
|
|
|
# let backfill blocks skip the queue - these are always "fast" to process
|
|
|
|
# because there are no state rewinds to deal with
|
|
|
|
let res = self.storeBackfillBlock(blck)
|
|
|
|
|
2022-04-08 18:22:49 +02:00
|
|
|
if resfut != nil:
|
|
|
|
resfut.complete(res)
|
2022-01-26 13:20:08 +01:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.blockQueue.addLastNoWait(BlockEntry(
|
|
|
|
blck: blck,
|
|
|
|
resfut: resfut, queueTick: Moment.now(),
|
|
|
|
validationDur: validationDur,
|
|
|
|
src: src))
|
|
|
|
except AsyncQueueFullError:
|
|
|
|
raiseAssert "unbounded queue"
|
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
# Event Loop
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc processBlock(self: var BlockProcessor, entry: BlockEntry) =
|
2021-03-11 11:10:57 +01:00
|
|
|
logScope:
|
2021-05-28 18:34:00 +02:00
|
|
|
blockRoot = shortLog(entry.blck.root)
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
let
|
2021-08-19 10:45:31 +00:00
|
|
|
wallTime = self.getBeaconTime()
|
2021-03-11 11:10:57 +01:00
|
|
|
(afterGenesis, wallSlot) = wallTime.toSlot()
|
|
|
|
|
|
|
|
if not afterGenesis:
|
|
|
|
error "Processing block before genesis, clock turned back?"
|
|
|
|
quit 1
|
|
|
|
|
|
|
|
let
|
2021-12-02 19:34:12 +01:00
|
|
|
res = withBlck(entry.blck):
|
2021-12-20 20:20:31 +01:00
|
|
|
self.storeBlock(entry.src, wallTime, blck, entry.queueTick, entry.validationDur)
|
2021-05-28 18:34:00 +02:00
|
|
|
|
2021-12-16 15:57:16 +01:00
|
|
|
if entry.resfut != nil:
|
|
|
|
entry.resfut.complete(
|
|
|
|
if res.isOk(): Result[void, BlockError].ok()
|
|
|
|
else: Result[void, BlockError].err(res.error()))
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2022-06-17 14:16:03 +00:00
|
|
|
func `$`(h: BlockHash): string = $h.asEth2Digest
|
|
|
|
|
2022-03-25 11:40:10 +00:00
|
|
|
proc runForkchoiceUpdated(
|
2022-06-17 14:16:03 +00:00
|
|
|
self: ref BlockProcessor, headBlockRoot, finalizedBlockRoot: Eth2Digest):
|
|
|
|
Future[bool] {.async.} =
|
2022-04-05 08:40:59 +00:00
|
|
|
# Allow finalizedBlockRoot to be 0 to avoid sync deadlocks.
|
|
|
|
#
|
|
|
|
# https://github.com/ethereum/EIPs/blob/master/EIPS/eip-3675.md#pos-events
|
|
|
|
# has "Before the first finalized block occurs in the system the finalized
|
|
|
|
# block hash provided by this event is stubbed with
|
|
|
|
# `0x0000000000000000000000000000000000000000000000000000000000000000`."
|
|
|
|
# and
|
2022-06-09 14:30:13 +00:00
|
|
|
# https://github.com/ethereum/consensus-specs/blob/v1.2.0-rc.1/specs/bellatrix/validator.md#executionpayload
|
2022-04-05 08:40:59 +00:00
|
|
|
# notes "`finalized_block_hash` is the hash of the latest finalized execution
|
|
|
|
# payload (`Hash32()` if none yet finalized)"
|
2022-06-18 10:39:21 +00:00
|
|
|
doAssert not headBlockRoot.isZero
|
2022-03-25 11:40:10 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
# Minimize window for Eth1 monitor to shut down connection
|
|
|
|
await self.consensusManager.eth1Monitor.ensureDataProvider()
|
|
|
|
|
2022-06-17 14:16:03 +00:00
|
|
|
let fcuR = awaitWithTimeout(
|
2022-03-25 11:40:10 +00:00
|
|
|
forkchoiceUpdated(
|
|
|
|
self.consensusManager.eth1Monitor, headBlockRoot, finalizedBlockRoot),
|
2022-05-17 13:57:33 +00:00
|
|
|
FORKCHOICEUPDATED_TIMEOUT):
|
2022-04-05 08:40:59 +00:00
|
|
|
debug "runForkChoiceUpdated: forkchoiceUpdated timed out"
|
2022-03-25 11:40:10 +00:00
|
|
|
default(ForkchoiceUpdatedResponse)
|
2022-06-17 14:16:03 +00:00
|
|
|
|
|
|
|
debug "runForkChoiceUpdated: running forkchoiceUpdated",
|
|
|
|
headBlockRoot,
|
|
|
|
finalizedBlockRoot,
|
|
|
|
payloadStatus = $fcuR.payloadStatus.status,
|
|
|
|
latestValidHash = $fcuR.payloadStatus.latestValidHash,
|
|
|
|
validationError = $fcuR.payloadStatus.validationError
|
|
|
|
|
|
|
|
return fcuR.payloadStatus.status == PayloadExecutionStatus.valid
|
2022-03-25 11:40:10 +00:00
|
|
|
except CatchableError as err:
|
2022-04-05 08:40:59 +00:00
|
|
|
debug "runForkChoiceUpdated: forkchoiceUpdated failed",
|
2022-03-25 11:40:10 +00:00
|
|
|
err = err.msg
|
2022-06-17 14:16:03 +00:00
|
|
|
return false
|
2022-03-25 11:40:10 +00:00
|
|
|
|
2022-04-05 08:40:59 +00:00
|
|
|
proc newExecutionPayload*(
|
2022-03-25 11:40:10 +00:00
|
|
|
eth1Monitor: Eth1Monitor, executionPayload: bellatrix.ExecutionPayload):
|
|
|
|
Future[PayloadExecutionStatus] {.async.} =
|
|
|
|
debug "newPayload: inserting block into execution engine",
|
|
|
|
parentHash = executionPayload.parent_hash,
|
|
|
|
blockHash = executionPayload.block_hash,
|
|
|
|
stateRoot = shortLog(executionPayload.state_root),
|
|
|
|
receiptsRoot = shortLog(executionPayload.receipts_root),
|
|
|
|
prevRandao = shortLog(executionPayload.prev_randao),
|
|
|
|
blockNumber = executionPayload.block_number,
|
|
|
|
gasLimit = executionPayload.gas_limit,
|
|
|
|
gasUsed = executionPayload.gas_used,
|
|
|
|
timestamp = executionPayload.timestamp,
|
|
|
|
extraDataLen = executionPayload.extra_data.len,
|
|
|
|
blockHash = executionPayload.block_hash,
|
2022-06-01 12:57:28 +00:00
|
|
|
baseFeePerGas = executionPayload.base_fee_per_gas,
|
2022-03-25 11:40:10 +00:00
|
|
|
numTransactions = executionPayload.transactions.len
|
|
|
|
|
2022-04-14 20:15:34 +00:00
|
|
|
if eth1Monitor.isNil:
|
|
|
|
info "newPayload: attempting to process execution payload without an Eth1Monitor. Ensure --web3-url setting is correct."
|
|
|
|
return PayloadExecutionStatus.syncing
|
|
|
|
|
2022-03-25 11:40:10 +00:00
|
|
|
try:
|
|
|
|
let
|
|
|
|
payloadResponse =
|
|
|
|
awaitWithTimeout(
|
|
|
|
eth1Monitor.newPayload(
|
|
|
|
executionPayload.asEngineExecutionPayload),
|
2022-05-17 13:57:33 +00:00
|
|
|
NEWPAYLOAD_TIMEOUT):
|
2022-04-14 20:15:34 +00:00
|
|
|
info "newPayload: newPayload timed out"
|
2022-03-25 11:40:10 +00:00
|
|
|
PayloadStatusV1(status: PayloadExecutionStatus.syncing)
|
|
|
|
payloadStatus = payloadResponse.status
|
|
|
|
|
|
|
|
return payloadStatus
|
|
|
|
except CatchableError as err:
|
2022-04-05 08:40:59 +00:00
|
|
|
debug "newPayload failed", msg = err.msg
|
2022-03-25 11:40:10 +00:00
|
|
|
return PayloadExecutionStatus.syncing
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc runQueueProcessingLoop*(self: ref BlockProcessor) {.async.} =
|
2022-04-05 08:40:59 +00:00
|
|
|
# Don't want to vacillate between "optimistic" sync and non-optimistic
|
|
|
|
# sync heads. Relies on runQueueProcessingLoop() being the only place,
|
|
|
|
# in Nimbus, which does this.
|
|
|
|
var
|
|
|
|
optForkchoiceHeadSlot = GENESIS_SLOT # safe default
|
|
|
|
optForkchoiceHeadRoot: Eth2Digest
|
|
|
|
|
2022-06-17 14:16:03 +00:00
|
|
|
# don't keep spamming same fcU to Geth; might be restarting sync each time
|
|
|
|
lastFcHead: Eth2Digest
|
|
|
|
|
2021-03-11 11:10:57 +01:00
|
|
|
while true:
|
2021-04-26 22:39:44 +02:00
|
|
|
# Cooperative concurrency: one block per loop iteration - because
|
2021-03-11 11:10:57 +01:00
|
|
|
# we run both networking and CPU-heavy things like block processing
|
|
|
|
# on the same thread, we need to make sure that there is steady progress
|
|
|
|
# on the networking side or we get long lockups that lead to timeouts.
|
|
|
|
const
|
|
|
|
# We cap waiting for an idle slot in case there's a lot of network traffic
|
|
|
|
# taking up all CPU - we don't want to _completely_ stop processing blocks
|
2021-04-26 22:39:44 +02:00
|
|
|
# in this case - doing so also allows us to benefit from more batching /
|
|
|
|
# larger network reads when under load.
|
2021-03-11 11:10:57 +01:00
|
|
|
idleTimeout = 10.milliseconds
|
|
|
|
|
|
|
|
discard await idleAsync().withTimeout(idleTimeout)
|
|
|
|
|
2022-03-25 11:40:10 +00:00
|
|
|
let
|
|
|
|
blck = await self[].blockQueue.popFirst()
|
|
|
|
hasExecutionPayload = blck.blck.kind >= BeaconBlockFork.Bellatrix
|
2022-06-18 10:39:21 +00:00
|
|
|
isExecutionBlock =
|
|
|
|
hasExecutionPayload and
|
|
|
|
blck.blck.bellatrixData.message.body.is_execution_block
|
2022-03-25 11:40:10 +00:00
|
|
|
executionPayloadStatus =
|
2022-06-18 10:39:21 +00:00
|
|
|
if isExecutionBlock:
|
2022-06-17 14:16:03 +00:00
|
|
|
# Eth1 syncing is asynchronous from this
|
|
|
|
|
|
|
|
# TODO self.consensusManager.eth1Monitor.terminalBlockHash.isSome
|
|
|
|
# should gate this when it works more reliably
|
|
|
|
when true:
|
|
|
|
try:
|
|
|
|
# Minimize window for Eth1 monitor to shut down connection
|
|
|
|
await self.consensusManager.eth1Monitor.ensureDataProvider()
|
|
|
|
|
|
|
|
await newExecutionPayload(
|
|
|
|
self.consensusManager.eth1Monitor,
|
|
|
|
blck.blck.bellatrixData.message.body.execution_payload)
|
|
|
|
except CatchableError as err:
|
|
|
|
debug "runQueueProcessingLoop: newPayload failed",
|
|
|
|
err = err.msg
|
|
|
|
PayloadExecutionStatus.syncing
|
|
|
|
else:
|
|
|
|
debug "runQueueProcessingLoop: got execution payload before TTD"
|
2022-03-25 11:40:10 +00:00
|
|
|
PayloadExecutionStatus.syncing
|
|
|
|
else:
|
|
|
|
# Vacuously
|
|
|
|
PayloadExecutionStatus.valid
|
|
|
|
|
|
|
|
if executionPayloadStatus in [
|
|
|
|
PayloadExecutionStatus.invalid,
|
2022-05-25 10:30:37 +00:00
|
|
|
PayloadExecutionStatus.invalid_block_hash]:
|
2022-04-05 08:40:59 +00:00
|
|
|
debug "runQueueProcessingLoop: execution payload invalid",
|
2022-03-25 11:40:10 +00:00
|
|
|
executionPayloadStatus
|
2022-06-17 14:16:03 +00:00
|
|
|
# Every loop iteration ends with some version of blck.resfut.complete(),
|
|
|
|
# including processBlock(), otherwise the sync manager stalls.
|
2022-03-25 11:40:10 +00:00
|
|
|
if not blck.resfut.isNil:
|
|
|
|
blck.resfut.complete(Result[void, BlockError].err(BlockError.Invalid))
|
2022-04-05 08:40:59 +00:00
|
|
|
continue
|
|
|
|
|
2022-06-18 10:39:21 +00:00
|
|
|
if isExecutionBlock:
|
2022-04-05 08:40:59 +00:00
|
|
|
# The EL client doesn't know here whether the payload is valid, because,
|
|
|
|
# for example, in Geth's case, its parent isn't known. When Geth logs an
|
|
|
|
# "Ignoring payload with missing parent" message, this is the result. It
|
|
|
|
# is distinct from the invalid cases above, and shouldn't cause the same
|
|
|
|
# BlockError.Invalid error, because it doesn't badly on the peer sending
|
|
|
|
# it, it's just not fully verifiable yet for this node. Furthermore, the
|
|
|
|
# EL client can, e.g. via Geth, "rely on the beacon client to forcefully
|
|
|
|
# update the head with a forkchoice update request". This can occur when
|
|
|
|
# an EL client is substantially more synced than a CL client, and when a
|
|
|
|
# CL client in that position attempts to serially sync it will encounter
|
|
|
|
# potential for this message until it nearly catches up, unless using an
|
|
|
|
# approach such as forkchoiceUpdated to trigger sync.
|
|
|
|
#
|
|
|
|
# Returning the MissingParent error causes the sync manager to loop in
|
|
|
|
# place until the EL does resync/catch up, then the normal process can
|
|
|
|
# resume where there's a hybrid serial and optimistic sync model.
|
|
|
|
#
|
|
|
|
# When this occurs within a couple of epochs of the Merge, before there
|
|
|
|
# has been a chance to justify and finalize a post-merge block this can
|
|
|
|
# cause a sync deadlock unless the EL can be convinced to sync back, or
|
|
|
|
# the CL is rather more open-endedly optimistic (potentially for entire
|
|
|
|
# weak subjectivity periods) than seems optimal.
|
2022-05-08 07:09:46 +00:00
|
|
|
debug "runQueueProcessingLoop: execution payload accepted or syncing",
|
2022-04-05 08:40:59 +00:00
|
|
|
executionPayloadStatus
|
|
|
|
|
2022-06-17 14:16:03 +00:00
|
|
|
# Always do this. Geth will only initiate syncing or reorgs with this
|
|
|
|
# combination of newPayload and forkchoiceUpdated. By design this must
|
|
|
|
# be somewhat optimistic, at least by one slot, for Geth to process it
|
|
|
|
# at all. This eventually converges to the same head as the DAG by the
|
|
|
|
# time it's externally visible via validating activity.
|
|
|
|
#
|
|
|
|
# In particular, the constraints that hold here are that Geth expects a
|
|
|
|
# sequence of
|
|
|
|
# - newPayload(execution payload with block hash `h`) followed by
|
|
|
|
# - forkchoiceUpdated(head = `h`)
|
|
|
|
# This is intrinsically somewhat optimistic, because determining the
|
|
|
|
# validity of an execution payload requires the forkchoiceUpdated
|
|
|
|
# head to be set to a block hash of some execution payload with unknown
|
|
|
|
# validity; otherwise it would not be necessary to ask the EL.
|
|
|
|
#
|
|
|
|
# The main reason this isn't done more adjacently in this code flow is to
|
|
|
|
# catch outright invalid cases, where the EL can reject a payload, without
|
|
|
|
# even running forkchoiceUpdated on it.
|
|
|
|
static: doAssert high(BeaconStateFork) == BeaconStateFork.Bellatrix
|
|
|
|
let curBh =
|
|
|
|
blck.blck.bellatrixData.message.body.execution_payload.block_hash
|
|
|
|
if curBh != lastFcHead:
|
|
|
|
lastFcHead = curBh
|
|
|
|
if await self.runForkchoiceUpdated(
|
2022-06-18 10:39:21 +00:00
|
|
|
curBh,
|
2022-06-17 14:16:03 +00:00
|
|
|
self.consensusManager.dag.finalizedHead.blck.executionBlockRoot):
|
|
|
|
# Geth seldom seems to return VALID to newPayload alone, even when
|
|
|
|
# it has all the relevant information.
|
|
|
|
self[].processBlock(blck)
|
|
|
|
continue
|
|
|
|
|
|
|
|
if executionPayloadStatus != PayloadExecutionStatus.valid:
|
|
|
|
if not blck.resfut.isNil:
|
|
|
|
blck.resfut.complete(Result[void, BlockError].err(
|
|
|
|
BlockError.MissingParent))
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
# When newPayload, rather than forkchoiceUpdated, has returned valid.
|
|
|
|
doAssert executionPayloadStatus == PayloadExecutionStatus.valid
|
|
|
|
self[].processBlock(blck)
|