2020-05-19 14:18:07 +00:00
|
|
|
# beacon_chain
|
2021-06-23 14:43:18 +00:00
|
|
|
# Copyright (c) 2018-2021 Status Research & Development GmbH
|
2020-05-19 14:18:07 +00: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.
|
|
|
|
|
2020-11-27 22:16:13 +00:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2020-05-19 14:18:07 +00:00
|
|
|
import
|
2020-11-27 22:16:13 +00:00
|
|
|
std/[tables, options],
|
|
|
|
chronicles,
|
2020-05-19 14:18:07 +00:00
|
|
|
stew/bitops2,
|
2021-01-25 18:45:48 +00:00
|
|
|
eth/keys,
|
2021-06-23 14:43:18 +00:00
|
|
|
../spec/[crypto, digest],
|
|
|
|
../spec/datatypes/[phase0, altair],
|
2020-11-27 22:16:13 +00:00
|
|
|
./block_pools_types
|
2020-05-19 14:18:07 +00:00
|
|
|
|
2020-07-30 19:18:17 +00:00
|
|
|
export options, block_pools_types
|
2020-06-18 10:03:36 +00:00
|
|
|
|
2020-07-16 13:16:51 +00:00
|
|
|
logScope:
|
|
|
|
topics = "quarant"
|
|
|
|
|
2021-01-25 18:45:48 +00:00
|
|
|
func init*(T: type QuarantineRef, rng: ref BrHmacDrbgContext): T =
|
|
|
|
result = T()
|
|
|
|
result.rng = rng
|
|
|
|
|
2021-06-01 11:13:40 +00:00
|
|
|
func checkMissing*(quarantine: QuarantineRef): seq[FetchRecord] =
|
2020-05-19 14:18:07 +00:00
|
|
|
## Return a list of blocks that we should try to resolve from other client -
|
|
|
|
## to be called periodically but not too often (once per slot?)
|
|
|
|
var done: seq[Eth2Digest]
|
|
|
|
|
|
|
|
for k, v in quarantine.missing.mpairs():
|
|
|
|
if v.tries > 8:
|
|
|
|
done.add(k)
|
|
|
|
else:
|
|
|
|
inc v.tries
|
|
|
|
|
|
|
|
for k in done:
|
|
|
|
quarantine.missing.del(k)
|
|
|
|
|
|
|
|
# simple (simplistic?) exponential backoff for retries..
|
|
|
|
for k, v in quarantine.missing.pairs():
|
|
|
|
if countOnes(v.tries.uint64) == 1:
|
2020-06-18 10:03:36 +00:00
|
|
|
result.add(FetchRecord(root: k))
|
|
|
|
|
2020-09-07 15:04:33 +00:00
|
|
|
# TODO stew/sequtils2
|
2020-09-04 06:39:46 +00:00
|
|
|
template anyIt(s, pred: untyped): bool =
|
|
|
|
# https://github.com/nim-lang/Nim/blob/version-1-2/lib/pure/collections/sequtils.nim#L682-L704
|
|
|
|
# without the items(...)
|
|
|
|
var result = false
|
|
|
|
for it {.inject.} in s:
|
|
|
|
if pred:
|
|
|
|
result = true
|
|
|
|
break
|
|
|
|
result
|
|
|
|
|
|
|
|
func containsOrphan*(
|
2021-06-23 14:43:18 +00:00
|
|
|
quarantine: QuarantineRef, signedBlock: phase0.SignedBeaconBlock): bool =
|
|
|
|
(signedBlock.root, signedBlock.signature) in quarantine.orphansPhase0
|
|
|
|
|
|
|
|
func containsOrphan*(
|
|
|
|
quarantine: QuarantineRef, signedBlock: altair.SignedBeaconBlock): bool =
|
|
|
|
(signedBlock.root, signedBlock.signature) in quarantine.orphansAltair
|
2020-09-04 06:39:46 +00:00
|
|
|
|
2021-06-01 11:13:40 +00:00
|
|
|
func addMissing*(quarantine: QuarantineRef, root: Eth2Digest) =
|
2020-08-05 14:19:43 +00:00
|
|
|
## Schedule the download a the given block
|
2020-09-04 06:39:46 +00:00
|
|
|
# Can only request by root, not by signature, so partial match suffices
|
2021-06-23 14:43:18 +00:00
|
|
|
if (not anyIt(quarantine.orphansPhase0.keys, it[0] == root)) and
|
|
|
|
(not anyIt(quarantine.orphansAltair.keys, it[0] == root)):
|
2020-08-05 14:19:43 +00:00
|
|
|
# If the block is in orphans, we no longer need it
|
|
|
|
discard quarantine.missing.hasKeyOrPut(root, MissingBlock())
|
2020-06-23 09:29:08 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
# TODO workaround for https://github.com/nim-lang/Nim/issues/18095
|
|
|
|
# copy of phase0.SomeSignedBeaconBlock from datatypes/phase0.nim
|
|
|
|
type SomeSignedPhase0Block =
|
|
|
|
phase0.SignedBeaconBlock | phase0.SigVerifiedSignedBeaconBlock |
|
|
|
|
phase0.TrustedSignedBeaconBlock
|
|
|
|
func removeOrphan*(
|
|
|
|
quarantine: QuarantineRef, signedBlock: SomeSignedPhase0Block) =
|
|
|
|
quarantine.orphansPhase0.del((signedBlock.root, signedBlock.signature))
|
|
|
|
|
|
|
|
# TODO workaround for https://github.com/nim-lang/Nim/issues/18095
|
|
|
|
# copy of altair.SomeSignedBeaconBlock from datatypes/altair.nim
|
|
|
|
type SomeSignedAltairBlock =
|
|
|
|
altair.SignedBeaconBlock | altair.SigVerifiedSignedBeaconBlock |
|
|
|
|
altair.TrustedSignedBeaconBlock
|
2020-09-04 06:39:46 +00:00
|
|
|
func removeOrphan*(
|
2021-06-23 14:43:18 +00:00
|
|
|
quarantine: QuarantineRef, signedBlock: SomeSignedAltairBlock) =
|
|
|
|
quarantine.orphansAltair.del((signedBlock.root, signedBlock.signature))
|
2020-09-04 06:39:46 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
func isViableOrphan(
|
|
|
|
dag: ChainDAGRef,
|
|
|
|
signedBlock: phase0.SignedBeaconBlock | altair.SignedBeaconBlock): bool =
|
2020-11-16 19:15:43 +00:00
|
|
|
# The orphan must be newer than the finalization point so that its parent
|
|
|
|
# either is the finalized block or more recent
|
|
|
|
signedBlock.message.slot > dag.finalizedHead.slot
|
|
|
|
|
2021-06-01 11:13:40 +00:00
|
|
|
func removeOldBlocks(quarantine: QuarantineRef, dag: ChainDAGRef) =
|
2020-09-04 06:39:46 +00:00
|
|
|
var oldBlocks: seq[(Eth2Digest, ValidatorSig)]
|
2020-08-31 09:00:38 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
template removeNonviableOrphans(orphans: untyped) =
|
|
|
|
for k, v in orphans.pairs():
|
|
|
|
if not isViableOrphan(dag, v):
|
|
|
|
oldBlocks.add k
|
2020-08-31 09:00:38 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
for k in oldBlocks:
|
|
|
|
orphans.del k
|
|
|
|
|
|
|
|
removeNonviableOrphans(quarantine.orphansPhase0)
|
|
|
|
removeNonviableOrphans(quarantine.orphansAltair)
|
2020-08-31 09:00:38 +00:00
|
|
|
|
2021-06-01 11:13:40 +00:00
|
|
|
func clearQuarantine*(quarantine: QuarantineRef) =
|
2021-06-23 14:43:18 +00:00
|
|
|
quarantine.orphansPhase0.clear()
|
|
|
|
quarantine.orphansAltair.clear()
|
2020-08-31 09:00:38 +00:00
|
|
|
quarantine.missing.clear()
|
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
# Typically, blocks will arrive in mostly topological order, with some
|
|
|
|
# out-of-order block pairs. Therefore, it is unhelpful to use either a
|
|
|
|
# FIFO or LIFO discpline, and since by definition each block gets used
|
|
|
|
# either 0 or 1 times it's not a cache either. Instead, stop accepting
|
|
|
|
# new blocks, and rely on syncing to cache up again if necessary. When
|
|
|
|
# using forward sync, blocks only arrive in an order not requiring the
|
|
|
|
# quarantine.
|
|
|
|
#
|
|
|
|
# For typical use cases, this need not be large, as they're two or three
|
|
|
|
# blocks arriving out of order due to variable network delays. As blocks
|
|
|
|
# for future slots are rejected before reaching quarantine, this usually
|
|
|
|
# will be a block for the last couple of slots for which the parent is a
|
|
|
|
# likely imminent arrival.
|
|
|
|
|
|
|
|
# Since we start forward sync when about one epoch is missing, that's as
|
|
|
|
# good a number as any.
|
|
|
|
const MAX_QUARANTINE_ORPHANS = SLOTS_PER_EPOCH
|
|
|
|
|
2021-06-01 11:13:40 +00:00
|
|
|
func add*(quarantine: QuarantineRef, dag: ChainDAGRef,
|
2021-06-23 14:43:18 +00:00
|
|
|
signedBlock: phase0.SignedBeaconBlock): bool =
|
2020-06-18 10:03:36 +00:00
|
|
|
## Adds block to quarantine's `orphans` and `missing` lists.
|
2021-06-23 14:43:18 +00:00
|
|
|
if not isViableOrphan(dag, signedBlock):
|
|
|
|
return false
|
|
|
|
|
|
|
|
quarantine.removeOldBlocks(dag)
|
|
|
|
|
|
|
|
# Even if the quarantine is full, we need to schedule its parent for
|
|
|
|
# downloading or we'll never get to the bottom of things
|
|
|
|
quarantine.addMissing(signedBlock.message.parent_root)
|
|
|
|
|
|
|
|
if quarantine.orphansPhase0.lenu64 >= MAX_QUARANTINE_ORPHANS:
|
|
|
|
return false
|
2020-08-31 09:00:38 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
quarantine.orphansPhase0[(signedBlock.root, signedBlock.signature)] =
|
|
|
|
signedBlock
|
|
|
|
quarantine.missing.del(signedBlock.root)
|
|
|
|
|
|
|
|
true
|
2020-11-16 19:15:43 +00:00
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
func add*(quarantine: QuarantineRef, dag: ChainDAGRef,
|
|
|
|
signedBlock: altair.SignedBeaconBlock): bool =
|
|
|
|
## Adds block to quarantine's `orphans` and `missing` lists.
|
2020-11-16 19:15:43 +00:00
|
|
|
if not isViableOrphan(dag, signedBlock):
|
|
|
|
return false
|
2020-08-31 09:00:38 +00:00
|
|
|
|
|
|
|
quarantine.removeOldBlocks(dag)
|
|
|
|
|
2020-11-16 19:15:43 +00:00
|
|
|
# Even if the quarantine is full, we need to schedule its parent for
|
|
|
|
# downloading or we'll never get to the bottom of things
|
|
|
|
quarantine.addMissing(signedBlock.message.parent_root)
|
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
if quarantine.orphansAltair.lenu64 >= MAX_QUARANTINE_ORPHANS:
|
2020-08-31 09:00:38 +00:00
|
|
|
return false
|
|
|
|
|
2021-06-23 14:43:18 +00:00
|
|
|
quarantine.orphansAltair[(signedBlock.root, signedBlock.signature)] =
|
|
|
|
signedBlock
|
2020-08-05 14:19:43 +00:00
|
|
|
quarantine.missing.del(signedBlock.root)
|
2020-06-18 10:03:36 +00:00
|
|
|
|
2020-08-31 09:00:38 +00:00
|
|
|
true
|