2021-03-11 11:10:57 +01:00
|
|
|
# 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.
|
|
|
|
|
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,
|
2021-09-27 14:22:58 +00:00
|
|
|
../spec/datatypes/[phase0, altair, merge],
|
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],
|
2021-03-11 11:10:57 +01:00
|
|
|
./consensus_manager,
|
2021-10-19 16:09:26 +02: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-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
|
|
|
|
# ----------------------------------------------------------------
|
2021-12-06 10:49:01 +01:00
|
|
|
blockQueue*: AsyncQueue[BlockEntry] # Exported for "test_sync_manager"
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
# Consumer
|
|
|
|
# ----------------------------------------------------------------
|
|
|
|
consensusManager: ref ConsensusManager
|
|
|
|
## Blockchain DAG, AttestationPool and Quarantine
|
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
|
|
|
|
|
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,
|
2021-12-06 10:49:01 +01:00
|
|
|
rng: ref BrHmacDrbgContext, taskpool: TaskPoolPtr,
|
2021-03-11 11:10:57 +01:00
|
|
|
consensusManager: ref ConsensusManager,
|
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-06 10:49:01 +01:00
|
|
|
getBeaconTime: getBeaconTime,
|
|
|
|
verifier: BatchVerifier(rng: rng, taskpool: taskpool)
|
|
|
|
)
|
2021-03-11 11:10:57 +01:00
|
|
|
|
|
|
|
# Sync callbacks
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc done*(entry: BlockEntry) =
|
|
|
|
## Send signal to [Sync/Request]Manager that the block ``entry`` has passed
|
2021-03-11 11:10:57 +01:00
|
|
|
## verification successfully.
|
2021-05-28 18:34:00 +02:00
|
|
|
if entry.resfut != nil:
|
|
|
|
entry.resfut.complete(Result[void, BlockError].ok())
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc fail*(entry: BlockEntry, error: BlockError) =
|
2021-03-11 11:10:57 +01:00
|
|
|
## Send signal to [Sync/Request]Manager that the block ``blk`` has NOT passed
|
|
|
|
## verification with specific ``error``.
|
2021-05-28 18:34:00 +02:00
|
|
|
if entry.resfut != nil:
|
|
|
|
entry.resfut.complete(Result[void, BlockError].err(error))
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc 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
|
|
|
# Enqueue
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc addBlock*(
|
2021-07-15 21:01:07 +02:00
|
|
|
self: var BlockProcessor, blck: ForkedSignedBeaconBlock,
|
2021-05-28 18:34:00 +02:00
|
|
|
resfut: Future[Result[void, BlockError]] = nil,
|
|
|
|
validationDur = Duration()) =
|
2021-03-11 11:10:57 +01:00
|
|
|
## Enqueue a Gossip-validated block for consensus verification
|
|
|
|
# Backpressure:
|
2021-10-19 17:20:55 +02:00
|
|
|
# There is no backpressure here - producers must wait for `resfut` to
|
|
|
|
# constrain their own processing
|
2021-03-11 11:10:57 +01:00
|
|
|
# Producers:
|
|
|
|
# - Gossip (when synced)
|
|
|
|
# - SyncManager (during sync)
|
|
|
|
# - RequestManager (missing ancestor blocks)
|
|
|
|
|
2021-04-26 22:39:44 +02:00
|
|
|
# addLast doesn't fail with unbounded queues, but we'll add asyncSpawn as a
|
|
|
|
# sanity check
|
2021-05-28 18:34:00 +02:00
|
|
|
try:
|
2021-12-06 10:49:01 +01:00
|
|
|
self.blockQueue.addLastNoWait(BlockEntry(
|
2021-07-19 11:58:30 +00:00
|
|
|
blck: blck,
|
|
|
|
resfut: resfut, queueTick: Moment.now(),
|
2021-05-28 18:34:00 +02:00
|
|
|
validationDur: validationDur))
|
|
|
|
except AsyncQueueFullError:
|
|
|
|
raiseAssert "unbounded queue"
|
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)
|
|
|
|
|
2021-03-11 11:10:57 +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
|
|
|
|
|
2021-10-19 17:20:55 +02:00
|
|
|
proc storeBlock*(
|
2021-07-15 21:01:07 +02:00
|
|
|
self: var BlockProcessor,
|
2021-11-05 08:34:34 +01:00
|
|
|
signedBlock: ForkySignedBeaconBlock,
|
2021-12-02 19:34:12 +01:00
|
|
|
wallSlot: Slot, queueTick: Moment = Moment.now(),
|
|
|
|
validationDur = Duration()): Result[BlockRef, BlockError] =
|
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-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()
|
2021-12-06 10:49:01 +01:00
|
|
|
let blck = dag.addRawBlock(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(
|
|
|
|
epochRef, blckRef, trustedBlock.message, wallSlot)
|
|
|
|
|
|
|
|
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():
|
|
|
|
if blck.error() == BlockError.MissingParent:
|
|
|
|
if not self.consensusManager.quarantine[].add(
|
|
|
|
dag, ForkedSignedBeaconBlock.init(signedBlock)):
|
|
|
|
debug "Block quarantine full",
|
|
|
|
blockRoot = shortLog(signedBlock.root),
|
|
|
|
blck = shortLog(signedBlock.message),
|
|
|
|
signature = shortLog(signedBlock.signature)
|
|
|
|
|
|
|
|
return blck
|
2021-12-02 19:34:12 +01:00
|
|
|
|
|
|
|
let storeBlockTick = Moment.now()
|
|
|
|
|
|
|
|
# Eagerly update head: the incoming block "should" get selected
|
|
|
|
self.consensusManager[].updateHead(wallSlot)
|
|
|
|
|
|
|
|
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
|
|
|
|
self.addBlock(quarantined)
|
|
|
|
|
|
|
|
blck
|
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):
|
|
|
|
self.storeBlock(blck, wallSlot, entry.queueTick, entry.validationDur)
|
2021-05-28 18:34:00 +02:00
|
|
|
|
2021-12-06 10:49:01 +01:00
|
|
|
if res.isOk() or res.error() == BlockError.Duplicate:
|
|
|
|
# Duplicate blocks are ok from a sync point of view, so we mark
|
2021-05-28 18:34:00 +02:00
|
|
|
# them as successful
|
|
|
|
entry.done()
|
2021-03-11 11:10:57 +01:00
|
|
|
else:
|
2021-05-28 18:34:00 +02:00
|
|
|
entry.fail(res.error())
|
2021-03-11 11:10:57 +01:00
|
|
|
|
2021-05-28 18:34:00 +02:00
|
|
|
proc runQueueProcessingLoop*(self: ref BlockProcessor) {.async.} =
|
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)
|
|
|
|
|
2021-12-06 10:49:01 +01:00
|
|
|
self[].processBlock(await self[].blockQueue.popFirst())
|