Forward sync refactoring. (#1191)

* Forward sync refactoring.
Rename Quarantine.pending to Quarantine.orphans.
Removing "old" fields.

* Fix test's FetchRecord.

* Fix `checkResponse` to not allow duplicates in response.
This commit is contained in:
Eugene Kabanov 2020-06-18 13:03:36 +03:00 committed by GitHub
parent db870fead4
commit 4436c85ff7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 160 additions and 105 deletions

View File

@ -213,7 +213,6 @@ proc init*(T: type BeaconNode, conf: BeaconNodeConf): Future[BeaconNode] {.async
nickname: nickname, nickname: nickname,
network: network, network: network,
netKeys: netKeys, netKeys: netKeys,
requestManager: RequestManager.init(network),
db: db, db: db,
config: conf, config: conf,
attachedValidators: ValidatorPool.init(), attachedValidators: ValidatorPool.init(),
@ -227,6 +226,11 @@ proc init*(T: type BeaconNode, conf: BeaconNodeConf): Future[BeaconNode] {.async
topicAggregateAndProofs: topicAggregateAndProofs, topicAggregateAndProofs: topicAggregateAndProofs,
) )
res.requestManager = RequestManager.init(network,
proc(signedBlock: SignedBeaconBlock) =
onBeaconBlock(res, signedBlock)
)
traceAsyncErrors res.addLocalValidators() traceAsyncErrors res.addLocalValidators()
# This merely configures the BeaconSync # This merely configures the BeaconSync
@ -501,21 +505,8 @@ proc handleMissingBlocks(node: BeaconNode) =
let missingBlocks = node.blockPool.checkMissing() let missingBlocks = node.blockPool.checkMissing()
if missingBlocks.len > 0: if missingBlocks.len > 0:
var left = missingBlocks.len var left = missingBlocks.len
info "Requesting detected missing blocks", blocks = shortLog(missingBlocks)
info "Requesting detected missing blocks", missingBlocks node.requestManager.fetchAncestorBlocks(missingBlocks)
node.requestManager.fetchAncestorBlocks(missingBlocks) do (b: SignedBeaconBlock):
onBeaconBlock(node, b)
# TODO instead of waiting for a full second to try the next missing block
# fetching, we'll do it here again in case we get all blocks we asked
# for (there might be new parents to fetch). of course, this is not
# good because the onSecond fetching also kicks in regardless but
# whatever - this is just a quick fix for making the testnet easier
# work with while the sync problem is dealt with more systematically
# dec left
# if left == 0:
# discard setTimer(Moment.now()) do (p: pointer):
# handleMissingBlocks(node)
proc onSecond(node: BeaconNode) {.async.} = proc onSecond(node: BeaconNode) {.async.} =
## This procedure will be called once per second. ## This procedure will be called once per second.
@ -815,6 +806,8 @@ proc run*(node: BeaconNode) =
node.onSecondLoop = runOnSecondLoop(node) node.onSecondLoop = runOnSecondLoop(node)
node.forwardSyncLoop = runForwardSyncLoop(node) node.forwardSyncLoop = runForwardSyncLoop(node)
node.requestManager.start()
# main event loop # main event loop
while status == BeaconNodeStatus.Running: while status == BeaconNodeStatus.Running:
try: try:
@ -1163,4 +1156,3 @@ programMain:
config.depositContractAddress, config.depositContractAddress,
config.depositPrivateKey, config.depositPrivateKey,
delayGenerator) delayGenerator)

View File

@ -37,7 +37,7 @@ type
## ##
## Invalid blocks are dropped immediately. ## Invalid blocks are dropped immediately.
pending*: Table[Eth2Digest, SignedBeaconBlock] ##\ orphans*: Table[Eth2Digest, SignedBeaconBlock] ##\
## Blocks that have passed validation but that we lack a link back to tail ## Blocks that have passed validation but that we lack a link back to tail
## for - when we receive a "missing link", we can use this data to build ## for - when we receive a "missing link", we can use this data to build
## an entire branch ## an entire branch
@ -49,12 +49,10 @@ type
inAdd*: bool inAdd*: bool
MissingBlock* = object MissingBlock* = object
slots*: uint64 # number of slots that are suspected missing
tries*: int tries*: int
FetchRecord* = object FetchRecord* = object
root*: Eth2Digest root*: Eth2Digest
historySlots*: uint64
CandidateChains* = ref object CandidateChains* = ref object
## Pool of blocks responsible for keeping a DAG of resolved blocks. ## Pool of blocks responsible for keeping a DAG of resolved blocks.

View File

@ -12,7 +12,7 @@ import
metrics, stew/results, metrics, stew/results,
../ssz/merkleization, ../state_transition, ../extras, ../ssz/merkleization, ../state_transition, ../extras,
../spec/[crypto, datatypes, digest, helpers, signatures], ../spec/[crypto, datatypes, digest, helpers, signatures],
block_pools_types, candidate_chains block_pools_types, candidate_chains, quarantine
export results export results
@ -32,7 +32,7 @@ func getOrResolve*(dag: CandidateChains, quarantine: var Quarantine, root: Eth2D
result = dag.getRef(root) result = dag.getRef(root)
if result.isNil: if result.isNil:
quarantine.missing[root] = MissingBlock(slots: 1) quarantine.missing[root] = MissingBlock()
proc add*( proc add*(
dag: var CandidateChains, quarantine: var Quarantine, dag: var CandidateChains, quarantine: var Quarantine,
@ -99,12 +99,12 @@ proc addResolvedBlock(
defer: quarantine.inAdd = false defer: quarantine.inAdd = false
var keepGoing = true var keepGoing = true
while keepGoing: while keepGoing:
let retries = quarantine.pending let retries = quarantine.orphans
for k, v in retries: for k, v in retries:
discard add(dag, quarantine, k, v) discard add(dag, quarantine, k, v)
# Keep going for as long as the pending dag is shrinking # Keep going for as long as the pending dag is shrinking
# TODO inefficient! so what? # TODO inefficient! so what?
keepGoing = quarantine.pending.len < retries.len keepGoing = quarantine.orphans.len < retries.len
blockRef blockRef
proc add*( proc add*(
@ -165,9 +165,9 @@ proc add*(
return err Invalid return err Invalid
# The block might have been in either of pending or missing - we don't want # The block might have been in either of `orphans` or `missing` - we don't
# any more work done on its behalf # want any more work done on its behalf
quarantine.pending.del(blockRoot) quarantine.orphans.del(blockRoot)
# The block is resolved, now it's time to validate it to ensure that the # The block is resolved, now it's time to validate it to ensure that the
# blocks we add to the database are clean for the given state # blocks we add to the database are clean for the given state
@ -209,7 +209,7 @@ proc add*(
# the pending dag calls this function back later in a loop, so as long # the pending dag calls this function back later in a loop, so as long
# as dag.add(...) requires a SignedBeaconBlock, easier to keep them in # as dag.add(...) requires a SignedBeaconBlock, easier to keep them in
# pending too. # pending too.
quarantine.pending[blockRoot] = signedBlock quarantine.add(dag, signedBlock, some(blockRoot))
# TODO possibly, it makes sense to check the database - that would allow sync # TODO possibly, it makes sense to check the database - that would allow sync
# to simply fill up the database with random blocks the other clients # to simply fill up the database with random blocks the other clients
@ -217,7 +217,7 @@ proc add*(
# junk that's not part of the block graph # junk that's not part of the block graph
if blck.parent_root in quarantine.missing or if blck.parent_root in quarantine.missing or
blck.parent_root in quarantine.pending: blck.parent_root in quarantine.orphans:
return err MissingParent return err MissingParent
# This is an unresolved block - put its parent on the missing list for now... # This is an unresolved block - put its parent on the missing list for now...
@ -232,24 +232,11 @@ proc add*(
# filter. # filter.
# TODO when we receive the block, we don't know how many others we're missing # TODO when we receive the block, we don't know how many others we're missing
# from that branch, so right now, we'll just do a blind guess # from that branch, so right now, we'll just do a blind guess
let parentSlot = blck.slot - 1
quarantine.missing[blck.parent_root] = MissingBlock(
slots:
# The block is at least two slots ahead - try to grab whole history
if parentSlot > dag.head.blck.slot:
parentSlot - dag.head.blck.slot
else:
# It's a sibling block from a branch that we're missing - fetch one
# epoch at a time
max(1.uint64, SLOTS_PER_EPOCH.uint64 -
(parentSlot.uint64 mod SLOTS_PER_EPOCH.uint64))
)
debug "Unresolved block (parent missing)", debug "Unresolved block (parent missing)",
blck = shortLog(blck), blck = shortLog(blck),
blockRoot = shortLog(blockRoot), blockRoot = shortLog(blockRoot),
pending = quarantine.pending.len, orphans = quarantine.orphans.len,
missing = quarantine.missing.len, missing = quarantine.missing.len,
cat = "filtering" cat = "filtering"
@ -345,8 +332,7 @@ proc isValidBeaconBlock*(
# not specific to this, but by the pending dag keying on the htr of the # not specific to this, but by the pending dag keying on the htr of the
# BeaconBlock, not SignedBeaconBlock, opens up certain spoofing attacks. # BeaconBlock, not SignedBeaconBlock, opens up certain spoofing attacks.
debug "parent unknown, putting block in quarantine" debug "parent unknown, putting block in quarantine"
quarantine.pending[hash_tree_root(signed_beacon_block.message)] = quarantine.add(dag, signed_beacon_block)
signed_beacon_block
return err(MissingParent) return err(MissingParent)
# The proposer signature, signed_beacon_block.signature, is valid with # The proposer signature, signed_beacon_block.signature, is valid with

View File

@ -6,13 +6,15 @@
# at your option. This file may not be copied, modified, or distributed except according to those terms. # at your option. This file may not be copied, modified, or distributed except according to those terms.
import import
chronicles, tables, chronicles, tables, options,
stew/bitops2, stew/bitops2,
metrics, metrics,
../spec/digest, ../spec/[datatypes, digest],
../ssz/merkleization,
block_pools_types block_pools_types
export options
logScope: topics = "quarant" logScope: topics = "quarant"
{.push raises: [Defect].} {.push raises: [Defect].}
@ -35,4 +37,19 @@ func checkMissing*(quarantine: var Quarantine): seq[FetchRecord] =
# simple (simplistic?) exponential backoff for retries.. # simple (simplistic?) exponential backoff for retries..
for k, v in quarantine.missing.pairs(): for k, v in quarantine.missing.pairs():
if countOnes(v.tries.uint64) == 1: if countOnes(v.tries.uint64) == 1:
result.add(FetchRecord(root: k, historySlots: v.slots)) result.add(FetchRecord(root: k))
func add*(quarantine: var Quarantine, dag: CandidateChains,
sblck: SignedBeaconBlock,
broot: Option[Eth2Digest] = none[Eth2Digest]()) =
## Adds block to quarantine's `orphans` and `missing` lists.
let blockRoot = if broot.isSome():
broot.get()
else:
hash_tree_root(sblck.message)
quarantine.orphans[blockRoot] = sblck
let parentRoot = sblck.message.parent_root
if parentRoot notin quarantine.missing:
quarantine.missing[parentRoot] = MissingBlock()

View File

@ -1,71 +1,133 @@
import import options, sequtils, strutils
options, random, import chronos, chronicles
chronos, chronicles, import spec/[datatypes, digest], eth2_network, beacon_node_types, sync_protocol,
spec/datatypes, sync_manager, ssz/merkleization
eth2_network, beacon_node_types, sync_protocol,
eth/async_utils logScope:
topics = "requman"
const
MAX_REQUEST_BLOCKS* = 4 # Specification's value is 1024.
## Maximum number of blocks, which can be requested by beaconBlocksByRoot.
PARALLEL_REQUESTS* = 2
## Number of peers we using to resolve our request.
type type
RequestManager* = object RequestManager* = object
network*: Eth2Node network*: Eth2Node
queue*: AsyncQueue[FetchRecord]
responseHandler*: FetchAncestorsResponseHandler
loopFuture: Future[void]
proc init*(T: type RequestManager, network: Eth2Node): T =
T(network: network)
type
FetchAncestorsResponseHandler = proc (b: SignedBeaconBlock) {.gcsafe.} FetchAncestorsResponseHandler = proc (b: SignedBeaconBlock) {.gcsafe.}
proc fetchAncestorBlocksFromPeer( func shortLog*(x: seq[Eth2Digest]): string =
peer: Peer, "[" & x.mapIt(shortLog(it)).join(", ") & "]"
rec: FetchRecord,
responseHandler: FetchAncestorsResponseHandler) {.async.} =
# TODO: It's not clear if this function follows the intention of the
# FetchRecord data type. Perhaps it is supposed to get a range of blocks
# instead. In order to do this, we'll need the slot number of the known
# block to be stored in the FetchRecord, so we can ask for a range of
# blocks starting N positions before this slot number.
try:
let blocks = await peer.beaconBlocksByRoot(BlockRootsList @[rec.root])
if blocks.isOk:
for b in blocks.get:
responseHandler(b)
except CatchableError as err:
debug "Error while fetching ancestor blocks",
err = err.msg, root = rec.root, peer = peer
proc fetchAncestorBlocksFromNetwork( func shortLog*(x: seq[FetchRecord]): string =
network: Eth2Node, "[" & x.mapIt(shortLog(it.root)).join(", ") & "]"
rec: FetchRecord,
responseHandler: FetchAncestorsResponseHandler) {.async.} = proc init*(T: type RequestManager, network: Eth2Node,
responseCb: FetchAncestorsResponseHandler): T =
T(
network: network, queue: newAsyncQueue[FetchRecord](),
responseHandler: responseCb
)
proc checkResponse(roots: openArray[Eth2Digest],
blocks: openArray[SignedBeaconBlock]): bool =
## This procedure checks peer's response.
var checks = @roots
if len(blocks) > len(roots):
return false
for blk in blocks:
let blockRoot = hash_tree_root(blk.message)
let res = checks.find(blockRoot)
if res == -1:
return false
else:
checks.del(res)
return true
proc fetchAncestorBlocksFromNetwork(rman: RequestManager,
items: seq[Eth2Digest]) {.async.} =
var peer: Peer var peer: Peer
try: try:
peer = await network.peerPool.acquire() peer = await rman.network.peerPool.acquire()
let blocks = await peer.beaconBlocksByRoot(BlockRootsList @[rec.root]) debug "Requesting blocks by root", peer = peer, blocks = shortLog(items),
peer_score = peer.getScore()
let blocks = await peer.beaconBlocksByRoot(BlockRootsList items)
if blocks.isOk: if blocks.isOk:
for b in blocks.get: let ublocks = blocks.get()
responseHandler(b) if checkResponse(items, ublocks):
except CatchableError as err: for b in ublocks:
debug "Error while fetching ancestor blocks", rman.responseHandler(b)
err = err.msg, root = rec.root, peer = peer peer.updateScore(PeerScoreGoodBlocks)
else:
peer.updateScore(PeerScoreBadResponse)
else:
peer.updateScore(PeerScoreNoBlocks)
except CancelledError as exc:
raise exc
except CatchableError as exc:
debug "Error while fetching ancestor blocks", exc = exc.msg,
items = shortLog(items), peer = peer, peer_score = peer.getScore()
raise exc
finally: finally:
if not(isNil(peer)): if not(isNil(peer)):
network.peerPool.release(peer) rman.network.peerPool.release(peer)
proc fetchAncestorBlocks*(requestManager: RequestManager, proc requestManagerLoop(rman: RequestManager) {.async.} =
roots: seq[FetchRecord], var rootList = newSeq[Eth2Digest]()
responseHandler: FetchAncestorsResponseHandler) = var workers = newSeq[Future[void]](PARALLEL_REQUESTS)
# TODO: we could have some fancier logic here: while true:
# try:
# * Keeps track of what was requested rootList.setLen(0)
# (this would give a little bit of time for the asked peer to respond) let req = await rman.queue.popFirst()
# rootList.add(req.root)
# * Keep track of the average latency of each peer
# (we can give priority to peers with better latency)
#
const ParallelRequests = 2
for i in 0 ..< ParallelRequests: var count = min(MAX_REQUEST_BLOCKS - 1, len(rman.queue))
traceAsyncErrors fetchAncestorBlocksFromNetwork(requestManager.network, while count > 0:
roots.sample(), rootList.add(rman.queue.popFirstNoWait().root)
responseHandler) dec(count)
let start = SyncMoment.now(Slot(0))
for i in 0 ..< PARALLEL_REQUESTS:
workers[i] = rman.fetchAncestorBlocksFromNetwork(rootList)
# We do not care about
await allFutures(workers)
let finish = SyncMoment.now(Slot(0) + uint64(len(rootList)))
var succeed = 0
for worker in workers:
if worker.finished() and not(worker.failed()):
inc(succeed)
debug "Request manager tick", blocks_count = len(rootList),
succeed = succeed,
failed = (len(workers) - succeed),
queue_size = len(rman.queue),
sync_speed = speed(start, finish)
except CatchableError as exc:
debug "Got a problem in request manager", exc = exc.msg
proc start*(rman: var RequestManager) =
## Start Request Manager's loop.
rman.loopFuture = requestManagerLoop(rman)
proc stop*(rman: RequestManager) =
## Stop Request Manager's loop.
if not(isNil(rman.loopFuture)):
rman.loopFuture.cancel()
proc fetchAncestorBlocks*(rman: RequestManager, roots: seq[FetchRecord]) =
## Enqueue list missing blocks roots ``roots`` for download by
## Request Manager ``rman``.
for item in roots:
rman.queue.addLastNoWait(item)

View File

@ -178,7 +178,7 @@ suiteReport "Block pool processing" & preset():
check: check:
pool.get(b2Root).isNone() # Unresolved, shouldn't show up pool.get(b2Root).isNone() # Unresolved, shouldn't show up
FetchRecord(root: b1Root, historySlots: 1) in pool.checkMissing() FetchRecord(root: b1Root) in pool.checkMissing()
check: pool.add(b1Root, b1).isOk check: pool.add(b1Root, b1).isOk