2021-03-26 07:52:01 +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.
|
|
|
|
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2020-06-18 13:03:36 +03:00
|
|
|
import options, sequtils, strutils
|
|
|
|
import chronos, chronicles
|
2021-08-12 15:08:20 +02:00
|
|
|
import
|
|
|
|
../spec/datatypes/[phase0, altair],
|
|
|
|
../spec/forks,
|
|
|
|
../networking/eth2_network,
|
|
|
|
"."/sync_protocol, "."/sync_manager
|
2020-08-12 12:29:11 +03:00
|
|
|
export sync_manager
|
2020-06-18 13:03:36 +03:00
|
|
|
|
|
|
|
logScope:
|
|
|
|
topics = "requman"
|
|
|
|
|
|
|
|
const
|
2020-08-12 12:29:11 +03:00
|
|
|
SYNC_MAX_REQUESTED_BLOCKS* = 32 # Spec allows up to MAX_REQUEST_BLOCKS.
|
|
|
|
## Maximum number of blocks which will be requested in each
|
|
|
|
## `beaconBlocksByRoot` invocation.
|
2020-06-18 13:03:36 +03:00
|
|
|
PARALLEL_REQUESTS* = 2
|
|
|
|
## Number of peers we using to resolve our request.
|
2019-03-28 16:03:19 +02:00
|
|
|
|
2019-11-25 15:36:25 +01:00
|
|
|
type
|
2021-12-16 15:57:16 +01:00
|
|
|
BlockVerifier* =
|
|
|
|
proc(signedBlock: ForkedSignedBeaconBlock):
|
|
|
|
Future[Result[void, BlockError]] {.gcsafe, raises: [Defect].}
|
|
|
|
|
2019-11-25 15:36:25 +01:00
|
|
|
RequestManager* = object
|
|
|
|
network*: Eth2Node
|
2020-08-12 12:29:11 +03:00
|
|
|
inpQueue*: AsyncQueue[FetchRecord]
|
2021-12-16 15:57:16 +01:00
|
|
|
blockVerifier: BlockVerifier
|
2020-06-18 13:03:36 +03:00
|
|
|
loopFuture: Future[void]
|
2019-11-25 15:36:25 +01:00
|
|
|
|
2020-06-18 13:03:36 +03:00
|
|
|
func shortLog*(x: seq[Eth2Digest]): string =
|
|
|
|
"[" & x.mapIt(shortLog(it)).join(", ") & "]"
|
|
|
|
|
|
|
|
func shortLog*(x: seq[FetchRecord]): string =
|
|
|
|
"[" & x.mapIt(shortLog(it.root)).join(", ") & "]"
|
|
|
|
|
|
|
|
proc init*(T: type RequestManager, network: Eth2Node,
|
2021-12-16 15:57:16 +01:00
|
|
|
blockVerifier: BlockVerifier): RequestManager =
|
2020-08-12 12:29:11 +03:00
|
|
|
RequestManager(
|
|
|
|
network: network,
|
|
|
|
inpQueue: newAsyncQueue[FetchRecord](),
|
2021-12-16 15:57:16 +01:00
|
|
|
blockVerifier: blockVerifier
|
2020-06-18 13:03:36 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
proc checkResponse(roots: openArray[Eth2Digest],
|
2021-07-15 21:01:07 +02:00
|
|
|
blocks: openArray[ForkedSignedBeaconBlock]): bool =
|
2020-06-18 13:03:36 +03:00
|
|
|
## This procedure checks peer's response.
|
|
|
|
var checks = @roots
|
|
|
|
if len(blocks) > len(roots):
|
|
|
|
return false
|
|
|
|
for blk in blocks:
|
2020-07-16 15:16:51 +02:00
|
|
|
let res = checks.find(blk.root)
|
2020-06-18 13:03:36 +03:00
|
|
|
if res == -1:
|
|
|
|
return false
|
|
|
|
else:
|
|
|
|
checks.del(res)
|
|
|
|
return true
|
|
|
|
|
|
|
|
proc fetchAncestorBlocksFromNetwork(rman: RequestManager,
|
|
|
|
items: seq[Eth2Digest]) {.async.} =
|
2020-03-22 22:54:47 +02:00
|
|
|
var peer: Peer
|
|
|
|
try:
|
2020-06-18 13:03:36 +03:00
|
|
|
peer = await rman.network.peerPool.acquire()
|
|
|
|
debug "Requesting blocks by root", peer = peer, blocks = shortLog(items),
|
|
|
|
peer_score = peer.getScore()
|
|
|
|
|
2021-07-15 21:01:07 +02:00
|
|
|
let blocks = if peer.useSyncV2():
|
|
|
|
await peer.beaconBlocksByRoot_v2(BlockRootsList items)
|
|
|
|
else:
|
|
|
|
(await peer.beaconBlocksByRoot(BlockRootsList items)).map() do (blcks: seq[phase0.SignedBeaconBlock]) -> auto:
|
|
|
|
blcks.mapIt(ForkedSignedBeaconBlock.init(it))
|
|
|
|
|
2020-05-13 01:37:07 +03:00
|
|
|
if blocks.isOk:
|
2020-06-18 13:03:36 +03:00
|
|
|
let ublocks = blocks.get()
|
|
|
|
if checkResponse(items, ublocks):
|
2020-08-12 12:29:11 +03:00
|
|
|
var res: Result[void, BlockError]
|
|
|
|
if len(ublocks) > 0:
|
|
|
|
for b in ublocks:
|
2021-12-16 15:57:16 +01:00
|
|
|
res = await rman.blockVerifier(b)
|
2021-12-06 10:49:01 +01:00
|
|
|
if res.isErr():
|
|
|
|
case res.error()
|
|
|
|
of BlockError.MissingParent:
|
|
|
|
# Ignoring because the order of the blocks that
|
|
|
|
# we requested may be different from the order in which we need
|
|
|
|
# these blocks to apply.
|
|
|
|
discard
|
|
|
|
of BlockError.Duplicate, BlockError.UnviableFork:
|
|
|
|
# Ignoring because these errors could occur due to the
|
|
|
|
# concurrent/parallel requests we made.
|
|
|
|
discard
|
|
|
|
of BlockError.Invalid:
|
|
|
|
# We stop processing blocks further to avoid DoS attack with big
|
|
|
|
# chunk of incorrect blocks.
|
|
|
|
break
|
2020-08-12 12:29:11 +03:00
|
|
|
else:
|
|
|
|
res = Result[void, BlockError].ok()
|
|
|
|
|
|
|
|
if res.isOk():
|
2020-10-06 15:10:02 +03:00
|
|
|
if len(ublocks) > 0:
|
|
|
|
# We reward peer only if it returns something.
|
|
|
|
peer.updateScore(PeerScoreGoodBlocks)
|
2020-08-12 12:29:11 +03:00
|
|
|
else:
|
2020-08-26 18:24:59 +03:00
|
|
|
# We are not penalizing other errors because of the reasons described
|
|
|
|
# above.
|
|
|
|
if res.error == BlockError.Invalid:
|
|
|
|
peer.updateScore(PeerScoreBadBlocks)
|
2020-06-18 13:03:36 +03:00
|
|
|
else:
|
|
|
|
peer.updateScore(PeerScoreBadResponse)
|
|
|
|
else:
|
|
|
|
peer.updateScore(PeerScoreNoBlocks)
|
|
|
|
|
|
|
|
except CancelledError as exc:
|
|
|
|
raise exc
|
|
|
|
except CatchableError as exc:
|
2020-08-04 18:05:29 +03:00
|
|
|
peer.updateScore(PeerScoreNoBlocks)
|
2020-06-18 13:03:36 +03:00
|
|
|
debug "Error while fetching ancestor blocks", exc = exc.msg,
|
|
|
|
items = shortLog(items), peer = peer, peer_score = peer.getScore()
|
|
|
|
raise exc
|
2020-03-22 22:54:47 +02:00
|
|
|
finally:
|
|
|
|
if not(isNil(peer)):
|
2020-06-18 13:03:36 +03:00
|
|
|
rman.network.peerPool.release(peer)
|
|
|
|
|
|
|
|
proc requestManagerLoop(rman: RequestManager) {.async.} =
|
|
|
|
var rootList = newSeq[Eth2Digest]()
|
|
|
|
var workers = newSeq[Future[void]](PARALLEL_REQUESTS)
|
|
|
|
while true:
|
|
|
|
try:
|
|
|
|
rootList.setLen(0)
|
2020-08-12 12:29:11 +03:00
|
|
|
let req = await rman.inpQueue.popFirst()
|
2020-06-18 13:03:36 +03:00
|
|
|
rootList.add(req.root)
|
|
|
|
|
2020-08-12 12:29:11 +03:00
|
|
|
var count = min(SYNC_MAX_REQUESTED_BLOCKS - 1, len(rman.inpQueue))
|
2020-06-18 13:03:36 +03:00
|
|
|
while count > 0:
|
2020-08-12 12:29:11 +03:00
|
|
|
rootList.add(rman.inpQueue.popFirstNoWait().root)
|
2020-06-18 13:03:36 +03:00
|
|
|
dec(count)
|
|
|
|
|
2021-12-16 15:57:16 +01:00
|
|
|
let start = SyncMoment.now(0)
|
2020-06-18 13:03:36 +03:00
|
|
|
|
|
|
|
for i in 0 ..< PARALLEL_REQUESTS:
|
|
|
|
workers[i] = rman.fetchAncestorBlocksFromNetwork(rootList)
|
|
|
|
|
|
|
|
# We do not care about
|
|
|
|
await allFutures(workers)
|
|
|
|
|
2021-12-16 15:57:16 +01:00
|
|
|
let finish = SyncMoment.now(uint64(len(rootList)))
|
2020-06-18 13:03:36 +03:00
|
|
|
|
|
|
|
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),
|
2020-08-12 12:29:11 +03:00
|
|
|
queue_size = len(rman.inpQueue),
|
2020-06-18 13:03:36 +03:00
|
|
|
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.
|
2020-08-12 12:29:11 +03:00
|
|
|
rman.loopFuture = rman.requestManagerLoop()
|
2020-06-18 13:03:36 +03:00
|
|
|
|
|
|
|
proc stop*(rman: RequestManager) =
|
|
|
|
## Stop Request Manager's loop.
|
|
|
|
if not(isNil(rman.loopFuture)):
|
|
|
|
rman.loopFuture.cancel()
|
2020-02-26 15:31:24 +02:00
|
|
|
|
2020-06-18 13:03:36 +03:00
|
|
|
proc fetchAncestorBlocks*(rman: RequestManager, roots: seq[FetchRecord]) =
|
|
|
|
## Enqueue list missing blocks roots ``roots`` for download by
|
|
|
|
## Request Manager ``rman``.
|
|
|
|
for item in roots:
|
2021-03-26 07:52:01 +01:00
|
|
|
try:
|
|
|
|
rman.inpQueue.addLastNoWait(item)
|
|
|
|
except AsyncQueueFullError: raiseAssert "unbounded queue"
|