2024-11-23 04:33:51 +00:00
|
|
|
|
# beacon_chain
|
|
|
|
|
# Copyright (c) 2018-2024 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: [].}
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
import std/[deques, heapqueue, tables, strutils, sequtils, math]
|
2024-06-20 18:57:08 +00:00
|
|
|
|
import stew/base10, chronos, chronicles, results
|
2021-12-08 21:15:29 +00:00
|
|
|
|
import
|
2022-01-18 13:36:52 +00:00
|
|
|
|
../spec/datatypes/[base, phase0, altair],
|
2021-12-08 21:15:29 +00:00
|
|
|
|
../spec/[helpers, forks],
|
|
|
|
|
../networking/[peer_pool, eth2_network],
|
|
|
|
|
../gossip_processing/block_processor,
|
2022-01-26 12:20:08 +00:00
|
|
|
|
../consensus_object_pools/block_pools_types
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
export base, phase0, altair, merge, chronos, chronicles, results,
|
2022-01-26 12:20:08 +00:00
|
|
|
|
block_pools_types, helpers
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
type
|
2023-08-25 09:29:07 +00:00
|
|
|
|
GetSlotCallback* = proc(): Slot {.gcsafe, raises: [].}
|
2024-06-20 18:57:08 +00:00
|
|
|
|
GetBoolCallback* = proc(): bool {.gcsafe, raises: [].}
|
2023-08-25 09:29:07 +00:00
|
|
|
|
ProcessingCallback* = proc() {.gcsafe, raises: [].}
|
2023-04-18 00:12:57 +00:00
|
|
|
|
BlockVerifier* = proc(signedBlock: ForkedSignedBeaconBlock,
|
2023-05-19 16:25:11 +00:00
|
|
|
|
blobs: Opt[BlobSidecars], maybeFinalized: bool):
|
2024-02-09 08:35:41 +00:00
|
|
|
|
Future[Result[void, VerifierError]] {.async: (raises: [CancelledError]).}
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
SyncRange* = object
|
|
|
|
|
slot*: Slot
|
|
|
|
|
count*: uint64
|
|
|
|
|
|
|
|
|
|
SyncPosition* = object
|
|
|
|
|
qindex*: int
|
|
|
|
|
sindex*: int
|
|
|
|
|
|
2021-12-08 21:15:29 +00:00
|
|
|
|
SyncQueueKind* {.pure.} = enum
|
|
|
|
|
Forward, Backward
|
|
|
|
|
|
|
|
|
|
SyncRequest*[T] = object
|
2022-03-03 08:05:33 +00:00
|
|
|
|
kind*: SyncQueueKind
|
2024-11-22 19:12:45 +00:00
|
|
|
|
data*: SyncRange
|
2021-12-08 21:15:29 +00:00
|
|
|
|
item*: T
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
SyncRequestQueueItem*[T] = object
|
|
|
|
|
requests: seq[SyncRequest[T]]
|
|
|
|
|
data: SyncRange
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
SyncQueueItem[T] = object
|
|
|
|
|
requests: seq[SyncRequest[T]]
|
|
|
|
|
data: SyncRange
|
|
|
|
|
failuresCount: Natural
|
2022-09-19 09:37:42 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
SyncWaiterItem[T] = ref object
|
2024-02-09 08:35:41 +00:00
|
|
|
|
future: Future[void].Raising([CancelledError])
|
2024-11-22 19:12:45 +00:00
|
|
|
|
request: SyncRequest[T]
|
|
|
|
|
resetFlag: bool
|
|
|
|
|
|
|
|
|
|
SyncProcessError {.pure.} = enum
|
|
|
|
|
Invalid,
|
|
|
|
|
MissingParent,
|
|
|
|
|
GoodAndMissingParent,
|
|
|
|
|
UnviableFork,
|
|
|
|
|
Duplicate,
|
|
|
|
|
Empty,
|
|
|
|
|
NoError
|
|
|
|
|
|
|
|
|
|
SyncBlock = object
|
|
|
|
|
slot: Slot
|
|
|
|
|
root: Eth2Digest
|
|
|
|
|
|
|
|
|
|
SyncProcessingResult = object
|
|
|
|
|
code: SyncProcessError
|
|
|
|
|
blck: Opt[SyncBlock]
|
|
|
|
|
|
|
|
|
|
GapItem[T] = object
|
|
|
|
|
data: SyncRange
|
|
|
|
|
item: T
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
RewindPoint = object
|
|
|
|
|
failSlot: Slot
|
|
|
|
|
epochCount: uint64
|
|
|
|
|
|
|
|
|
|
SyncQueue*[T] = ref object
|
|
|
|
|
kind*: SyncQueueKind
|
|
|
|
|
inpSlot*: Slot
|
|
|
|
|
outSlot*: Slot
|
|
|
|
|
startSlot*: Slot
|
|
|
|
|
finalSlot*: Slot
|
2024-11-22 19:12:45 +00:00
|
|
|
|
rewind: Opt[RewindPoint]
|
|
|
|
|
chunkSize: uint64
|
|
|
|
|
requestsCount: Natural
|
|
|
|
|
failureResetThreshold: Natural
|
|
|
|
|
requests: Deque[SyncQueueItem[T]]
|
|
|
|
|
getSafeSlot: GetSlotCallback
|
2021-12-16 14:57:16 +00:00
|
|
|
|
blockVerifier: BlockVerifier
|
2024-11-22 19:12:45 +00:00
|
|
|
|
waiters: seq[SyncWaiterItem[T]]
|
|
|
|
|
gapList: seq[GapItem[T]]
|
|
|
|
|
lock: AsyncLock
|
|
|
|
|
ident: string
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2022-03-03 08:05:33 +00:00
|
|
|
|
chronicles.formatIt SyncQueueKind: toLowerAscii($it)
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc `$`*(srange: SyncRange): string =
|
|
|
|
|
"[" & Base10.toString(uint64(srange.slot)) & ":" &
|
|
|
|
|
Base10.toString(uint64(srange.slot + srange.count - 1)) & "]"
|
|
|
|
|
|
|
|
|
|
template shortLog[T](req: SyncRequest[T]): string =
|
|
|
|
|
$req.data & "@" & Base10.toString(req.data.count)
|
2022-03-03 08:05:33 +00:00
|
|
|
|
|
|
|
|
|
chronicles.expandIt SyncRequest:
|
|
|
|
|
`it` = shortLog(it)
|
|
|
|
|
peer = shortLog(it.item)
|
|
|
|
|
direction = toLowerAscii($it.kind)
|
2022-01-20 07:25:45 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
chronicles.formatIt Opt[SyncBlock]:
|
|
|
|
|
if it.isSome():
|
|
|
|
|
Base10.toString(uint64(it.get().slot)) & "@" & shortLog(it.get().root)
|
|
|
|
|
else:
|
|
|
|
|
"<n/a>"
|
|
|
|
|
|
|
|
|
|
func getShortMap*[T](
|
|
|
|
|
req: SyncRequest[T],
|
|
|
|
|
data: openArray[ref ForkedSignedBeaconBlock]
|
|
|
|
|
): string =
|
2021-12-08 21:15:29 +00:00
|
|
|
|
## Returns all slot numbers in ``data`` as placement map.
|
2024-11-22 19:12:45 +00:00
|
|
|
|
var
|
|
|
|
|
res = newStringOfCap(req.data.count)
|
|
|
|
|
slider = req.data.slot
|
|
|
|
|
last = 0
|
|
|
|
|
|
|
|
|
|
for i in 0 ..< req.data.count:
|
2021-12-08 21:15:29 +00:00
|
|
|
|
if last < len(data):
|
|
|
|
|
for k in last ..< len(data):
|
2022-02-07 17:20:10 +00:00
|
|
|
|
if slider == data[k][].slot:
|
2021-12-08 21:15:29 +00:00
|
|
|
|
res.add('x')
|
|
|
|
|
last = k + 1
|
|
|
|
|
break
|
2022-02-07 17:20:10 +00:00
|
|
|
|
elif slider < data[k][].slot:
|
2021-12-08 21:15:29 +00:00
|
|
|
|
res.add('.')
|
|
|
|
|
break
|
|
|
|
|
else:
|
|
|
|
|
res.add('.')
|
2022-06-06 13:56:59 +00:00
|
|
|
|
slider = slider + 1
|
2021-12-08 21:15:29 +00:00
|
|
|
|
res
|
|
|
|
|
|
2024-11-29 16:15:46 +00:00
|
|
|
|
proc getShortMap*[T](req: SyncRequest[T],
|
|
|
|
|
data: openArray[ref BlobSidecar]): string =
|
|
|
|
|
var
|
|
|
|
|
res = newStringOfCap(req.data.count)
|
|
|
|
|
slider = req.data.slot
|
|
|
|
|
last = 0
|
|
|
|
|
|
|
|
|
|
for i in 0 ..< req.data.count:
|
|
|
|
|
if last < len(data):
|
|
|
|
|
var counter = 0
|
|
|
|
|
for k in last ..< len(data):
|
|
|
|
|
if slider < data[k][].signed_block_header.message.slot:
|
|
|
|
|
break
|
|
|
|
|
elif slider == data[k][].signed_block_header.message.slot:
|
|
|
|
|
inc(counter)
|
|
|
|
|
last = last + counter
|
|
|
|
|
if counter == 0:
|
|
|
|
|
res.add('.')
|
|
|
|
|
else:
|
|
|
|
|
res.add($counter)
|
|
|
|
|
else:
|
|
|
|
|
res.add('.')
|
|
|
|
|
slider = slider + 1
|
|
|
|
|
res
|
|
|
|
|
|
|
|
|
|
proc getFullMap*[T](req: SyncRequest[T],
|
|
|
|
|
data: openArray[ref BlobSidecar]): string =
|
|
|
|
|
data.mapIt("(" & $it[].signed_block_header.message.slot & ", " & $uint64(it[].index) & ")").join(", ")
|
|
|
|
|
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc getShortMap*[T](
|
|
|
|
|
req: SyncRequest[T],
|
|
|
|
|
blobs: openArray[BlobSidecars]
|
|
|
|
|
): string =
|
|
|
|
|
static:
|
|
|
|
|
doAssert(MAX_BLOBS_PER_BLOCK < 10, "getShortMap(Blobs) should be revisited")
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-19 05:35:09 +00:00
|
|
|
|
var
|
2024-11-22 19:12:45 +00:00
|
|
|
|
res = newStringOfCap(req.data.count)
|
|
|
|
|
slider = req.data.slot
|
|
|
|
|
last = 0
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
for i in 0 ..< req.data.count:
|
|
|
|
|
if last < len(blobs):
|
|
|
|
|
if len(blobs[last]) > 0:
|
|
|
|
|
let
|
|
|
|
|
sidecar = blobs[last][0]
|
|
|
|
|
length = len(blobs[last])
|
|
|
|
|
if sidecar.signed_block_header.message.slot == slider:
|
|
|
|
|
res.add(Base10.toString(uint64(length)))
|
|
|
|
|
inc(last)
|
|
|
|
|
else:
|
|
|
|
|
res.add('.')
|
|
|
|
|
else:
|
|
|
|
|
res.add('.')
|
2024-11-19 05:35:09 +00:00
|
|
|
|
else:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
res.add('.')
|
|
|
|
|
inc(slider)
|
|
|
|
|
res
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc getShortMap*[T](
|
|
|
|
|
req: SyncRequest[T],
|
|
|
|
|
data: Opt[seq[BlobSidecars]]
|
|
|
|
|
): string =
|
|
|
|
|
if data.isNone():
|
|
|
|
|
return '.'.repeat(req.data.count)
|
|
|
|
|
getShortMap(req, data.get())
|
|
|
|
|
|
|
|
|
|
func init*(t: typedesc[SyncRange], slot: Slot, count: uint64): SyncRange =
|
|
|
|
|
SyncRange(slot: slot, count: count)
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessError],
|
|
|
|
|
kind: VerifierError): SyncProcessError =
|
|
|
|
|
case kind
|
|
|
|
|
of VerifierError.Invalid:
|
|
|
|
|
SyncProcessError.Invalid
|
|
|
|
|
of VerifierError.MissingParent:
|
|
|
|
|
SyncProcessError.MissingParent
|
|
|
|
|
of VerifierError.UnviableFork:
|
|
|
|
|
SyncProcessError.UnviableFork
|
|
|
|
|
of VerifierError.Duplicate:
|
|
|
|
|
SyncProcessError.Duplicate
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncBlock], slot: Slot, root: Eth2Digest): SyncBlock =
|
|
|
|
|
SyncBlock(slot: slot, root: root)
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessError]): SyncProcessError =
|
|
|
|
|
SyncProcessError.NoError
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessingResult], se: SyncProcessError,
|
|
|
|
|
slot: Slot, root: Eth2Digest): SyncProcessingResult =
|
|
|
|
|
SyncProcessingResult(blck: Opt.some(SyncBlock.init(slot, root)), code: se)
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessingResult],
|
|
|
|
|
se: SyncProcessError): SyncProcessingResult =
|
|
|
|
|
SyncProcessingResult(code: se)
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessingResult], se: SyncProcessError,
|
|
|
|
|
sblck: SyncBlock): SyncProcessingResult =
|
|
|
|
|
SyncProcessingResult(blck: Opt.some(sblck), code: se)
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessingResult], ve: VerifierError,
|
|
|
|
|
slot: Slot, root: Eth2Digest): SyncProcessingResult =
|
|
|
|
|
SyncProcessingResult(blck: Opt.some(SyncBlock.init(slot, root)),
|
|
|
|
|
code: SyncProcessError.init(ve))
|
|
|
|
|
|
|
|
|
|
func init(t: typedesc[SyncProcessingResult], ve: VerifierError,
|
|
|
|
|
sblck: SyncBlock): SyncProcessingResult =
|
|
|
|
|
SyncProcessingResult(blck: Opt.some(sblck), code: SyncProcessError.init(ve))
|
|
|
|
|
|
|
|
|
|
func init*[T](t: typedesc[SyncRequest], kind: SyncQueueKind,
|
|
|
|
|
item: T): SyncRequest[T] =
|
|
|
|
|
SyncRequest[T](
|
|
|
|
|
kind: kind,
|
|
|
|
|
data: SyncRange(slot: FAR_FUTURE_SLOT, count: 0'u64),
|
|
|
|
|
item: item
|
2021-12-08 21:15:29 +00:00
|
|
|
|
)
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func init*[T](t: typedesc[SyncRequest], kind: SyncQueueKind,
|
|
|
|
|
data: SyncRange, item: T): SyncRequest[T] =
|
|
|
|
|
SyncRequest[T](kind: kind, data: data, item: item)
|
|
|
|
|
|
|
|
|
|
func init[T](t: typedesc[SyncQueueItem],
|
|
|
|
|
req: SyncRequest[T]): SyncQueueItem[T] =
|
|
|
|
|
SyncQueueItem[T](data: req.data, requests: @[req])
|
|
|
|
|
|
|
|
|
|
func init[T](t: typedesc[GapItem], req: SyncRequest[T]): GapItem[T] =
|
|
|
|
|
GapItem[T](data: req.data, item: req.item)
|
|
|
|
|
|
|
|
|
|
func next(srange: SyncRange): SyncRange {.inline.} =
|
|
|
|
|
let slot = srange.slot + srange.count
|
|
|
|
|
if slot == FAR_FUTURE_SLOT:
|
|
|
|
|
# Finish range
|
|
|
|
|
srange
|
|
|
|
|
elif slot < srange.slot:
|
|
|
|
|
# Range that causes uint64 overflow, fixing.
|
|
|
|
|
SyncRange.init(slot, uint64(FAR_FUTURE_SLOT - srange.count))
|
|
|
|
|
else:
|
|
|
|
|
if slot + srange.count < slot:
|
|
|
|
|
SyncRange.init(slot, uint64(FAR_FUTURE_SLOT - srange.count))
|
|
|
|
|
else:
|
|
|
|
|
SyncRange.init(slot, srange.count)
|
2021-12-16 14:57:16 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func prev(srange: SyncRange): SyncRange {.inline.} =
|
|
|
|
|
if srange.slot == GENESIS_SLOT:
|
|
|
|
|
# Start range
|
|
|
|
|
srange
|
|
|
|
|
else:
|
|
|
|
|
let slot = srange.slot - srange.count
|
|
|
|
|
if slot > srange.slot:
|
|
|
|
|
# Range that causes uint64 underflow, fixing.
|
|
|
|
|
SyncRange.init(GENESIS_SLOT, uint64(srange.slot))
|
|
|
|
|
else:
|
|
|
|
|
SyncRange.init(slot, srange.count)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func contains(srange: SyncRange, slot: Slot): bool {.inline.} =
|
|
|
|
|
## Returns `true` if `slot` is in range of `srange`.
|
|
|
|
|
if (srange.slot + srange.count) < srange.slot:
|
|
|
|
|
(slot >= srange.slot) and (slot <= FAR_FUTURE_SLOT)
|
|
|
|
|
else:
|
|
|
|
|
(slot >= srange.slot) and (slot < (srange.slot + srange.count))
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func `>`(a, b: SyncRange): bool {.inline.} =
|
|
|
|
|
## Returns `true` if range `a` is above of range `b`.
|
|
|
|
|
(a.slot > b.slot) and (a.slot + a.count - 1 > b.slot)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func `<`(a, b: SyncRange): bool {.inline.} =
|
|
|
|
|
## Returns `true` if range `a` is below of range `b`.
|
|
|
|
|
(a.slot < b.slot) and (a.slot + a.count - 1 < b.slot)
|
2022-01-20 07:25:45 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func `==`(a, b: SyncRange): bool {.inline.} =
|
|
|
|
|
(a.slot == b.slot) and (a.count == b.count)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func `==`[T](a, b: SyncRequest[T]): bool {.inline.} =
|
|
|
|
|
(a.kind == b.kind) and (a.item == b.item) and (a.data == b.data)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc hasEndGap*[T](
|
|
|
|
|
sr: SyncRequest[T],
|
|
|
|
|
data: openArray[ref ForkedSignedBeaconBlock]
|
|
|
|
|
): bool {.inline.} =
|
2021-12-08 21:15:29 +00:00
|
|
|
|
## Returns ``true`` if response chain of blocks has gap at the end.
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if len(data) == 0:
|
2021-12-08 21:15:29 +00:00
|
|
|
|
return true
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if data[^1][].slot != (sr.data.slot + sr.data.count - 1'u64):
|
2021-12-08 21:15:29 +00:00
|
|
|
|
return true
|
2024-11-22 19:12:45 +00:00
|
|
|
|
false
|
2022-09-19 09:37:42 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc updateLastSlot*[T](sq: SyncQueue[T], last: Slot) {.inline.} =
|
|
|
|
|
## Update last slot stored in queue ``sq`` with value ``last``.
|
|
|
|
|
sq.finalSlot = last
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
proc getRewindPoint*[T](sq: SyncQueue[T], failSlot: Slot,
|
|
|
|
|
safeSlot: Slot): Slot =
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
|
|
|
|
# Calculate the latest finalized epoch.
|
2022-01-11 10:01:54 +00:00
|
|
|
|
let finalizedEpoch = epoch(safeSlot)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
# Calculate failure epoch.
|
2022-01-11 10:01:54 +00:00
|
|
|
|
let failEpoch = epoch(failSlot)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
# Calculate exponential rewind point in number of epochs.
|
|
|
|
|
let epochCount =
|
|
|
|
|
if sq.rewind.isSome():
|
|
|
|
|
let rewind = sq.rewind.get()
|
|
|
|
|
if failSlot == rewind.failSlot:
|
|
|
|
|
# `MissingParent` happened at same slot so we increase rewind point by
|
|
|
|
|
# factor of 2.
|
|
|
|
|
if failEpoch > finalizedEpoch:
|
|
|
|
|
let rewindPoint = rewind.epochCount shl 1
|
|
|
|
|
if rewindPoint < rewind.epochCount:
|
|
|
|
|
# If exponential rewind point produces `uint64` overflow we will
|
|
|
|
|
# make rewind to latest finalized epoch.
|
|
|
|
|
failEpoch - finalizedEpoch
|
|
|
|
|
else:
|
|
|
|
|
if (failEpoch < rewindPoint) or
|
|
|
|
|
(failEpoch - rewindPoint < finalizedEpoch):
|
|
|
|
|
# If exponential rewind point points to position which is far
|
|
|
|
|
# behind latest finalized epoch.
|
|
|
|
|
failEpoch - finalizedEpoch
|
|
|
|
|
else:
|
|
|
|
|
rewindPoint
|
|
|
|
|
else:
|
|
|
|
|
warn "Trying to rewind over the last finalized epoch",
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_slot = safeSlot,
|
|
|
|
|
fail_slot = failSlot,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
fail_epoch = failEpoch,
|
2021-12-08 21:15:29 +00:00
|
|
|
|
rewind_epoch_count = rewind.epochCount,
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2021-12-08 21:15:29 +00:00
|
|
|
|
0'u64
|
|
|
|
|
else:
|
|
|
|
|
# `MissingParent` happened at different slot so we going to rewind for
|
|
|
|
|
# 1 epoch only.
|
|
|
|
|
if (failEpoch < 1'u64) or (failEpoch - 1'u64 < finalizedEpoch):
|
|
|
|
|
warn "Сould not rewind further than the last finalized epoch",
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_slot = safeSlot,
|
|
|
|
|
fail_slot = failSlot,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
fail_epoch = failEpoch,
|
2021-12-08 21:15:29 +00:00
|
|
|
|
rewind_epoch_count = rewind.epochCount,
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2021-12-08 21:15:29 +00:00
|
|
|
|
0'u64
|
|
|
|
|
else:
|
|
|
|
|
1'u64
|
|
|
|
|
else:
|
|
|
|
|
# `MissingParent` happened first time.
|
|
|
|
|
if (failEpoch < 1'u64) or (failEpoch - 1'u64 < finalizedEpoch):
|
|
|
|
|
warn "Сould not rewind further than the last finalized epoch",
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_slot = safeSlot,
|
|
|
|
|
fail_slot = failSlot,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
fail_epoch = failEpoch,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2021-12-08 21:15:29 +00:00
|
|
|
|
0'u64
|
|
|
|
|
else:
|
|
|
|
|
1'u64
|
|
|
|
|
|
|
|
|
|
if epochCount == 0'u64:
|
|
|
|
|
warn "Unable to continue syncing, please restart the node",
|
2024-11-29 16:15:46 +00:00
|
|
|
|
finalized_slot = safeSlot,
|
|
|
|
|
fail_slot = failSlot,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
fail_epoch = failEpoch,
|
|
|
|
|
finalized_epoch = finalizedEpoch,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2021-12-08 21:15:29 +00:00
|
|
|
|
# Calculate the rewind epoch, which will be equal to last rewind point or
|
|
|
|
|
# finalizedEpoch
|
|
|
|
|
let rewindEpoch =
|
|
|
|
|
if sq.rewind.isNone():
|
|
|
|
|
finalizedEpoch
|
|
|
|
|
else:
|
2022-01-11 10:01:54 +00:00
|
|
|
|
epoch(sq.rewind.get().failSlot) - sq.rewind.get().epochCount
|
|
|
|
|
rewindEpoch.start_slot()
|
2021-12-08 21:15:29 +00:00
|
|
|
|
else:
|
|
|
|
|
# Calculate the rewind epoch, which should not be less than the latest
|
|
|
|
|
# finalized epoch.
|
|
|
|
|
let rewindEpoch = failEpoch - epochCount
|
|
|
|
|
# Update and save new rewind point in SyncQueue.
|
2024-11-22 19:12:45 +00:00
|
|
|
|
sq.rewind = Opt.some(
|
|
|
|
|
RewindPoint(failSlot: failSlot, epochCount: epochCount))
|
2022-01-11 10:01:54 +00:00
|
|
|
|
rewindEpoch.start_slot()
|
2021-12-08 21:15:29 +00:00
|
|
|
|
of SyncQueueKind.Backward:
|
|
|
|
|
# While we perform backward sync, the only possible slot we could rewind is
|
|
|
|
|
# latest stored block.
|
|
|
|
|
if failSlot == safeSlot:
|
|
|
|
|
warn "Unable to continue syncing, please restart the node",
|
2024-11-29 16:15:46 +00:00
|
|
|
|
safe_slot = safeSlot,
|
|
|
|
|
fail_slot = failSlot,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2021-12-08 21:15:29 +00:00
|
|
|
|
safeSlot
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func init*[T](t1: typedesc[SyncQueue], t2: typedesc[T],
|
|
|
|
|
queueKind: SyncQueueKind,
|
|
|
|
|
start, final: Slot,
|
|
|
|
|
chunkSize: uint64,
|
|
|
|
|
requestsCount: Natural,
|
|
|
|
|
failureResetThreshold: Natural,
|
|
|
|
|
getSafeSlotCb: GetSlotCallback,
|
|
|
|
|
blockVerifier: BlockVerifier,
|
|
|
|
|
ident: string = "main"): SyncQueue[T] =
|
|
|
|
|
doAssert(chunkSize > 0'u64, "Chunk size should not be zero")
|
|
|
|
|
doAssert(requestsCount > 0, "Number of requests should not be zero")
|
2023-06-05 14:42:27 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
SyncQueue[T](
|
|
|
|
|
kind: queueKind,
|
|
|
|
|
startSlot: start,
|
|
|
|
|
finalSlot: final,
|
|
|
|
|
chunkSize: chunkSize,
|
|
|
|
|
requestsCount: requestsCount,
|
|
|
|
|
failureResetThreshold: failureResetThreshold,
|
|
|
|
|
getSafeSlot: getSafeSlotCb,
|
|
|
|
|
inpSlot: start,
|
|
|
|
|
outSlot: start,
|
|
|
|
|
blockVerifier: blockVerifier,
|
|
|
|
|
requests: initDeque[SyncQueueItem[T]](),
|
|
|
|
|
lock: newAsyncLock(),
|
|
|
|
|
ident: ident
|
|
|
|
|
)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func contains[T](requests: openArray[SyncRequest[T]], source: T): bool =
|
|
|
|
|
for req in requests:
|
|
|
|
|
if req.item == source:
|
|
|
|
|
return true
|
|
|
|
|
false
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
func find[T](sq: SyncQueue[T], req: SyncRequest[T]): Opt[SyncPosition] =
|
|
|
|
|
if len(sq.requests) == 0:
|
|
|
|
|
return Opt.none(SyncPosition)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if (req.data < sq.requests[0].data) or (req.data > sq.requests[^1].data):
|
|
|
|
|
return Opt.none(SyncPosition)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
of SyncQueueKind.Backward:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if (req.data > sq.requests[0].data) or (req.data < sq.requests[^1].data) :
|
|
|
|
|
return Opt.none(SyncPosition)
|
|
|
|
|
|
|
|
|
|
for qindex, qitem in sq.requests.pairs():
|
|
|
|
|
for sindex, request in qitem.requests.pairs():
|
|
|
|
|
if request == req:
|
|
|
|
|
return Opt.some(SyncPosition(qindex: qindex, sindex: sindex))
|
|
|
|
|
|
|
|
|
|
Opt.none(SyncPosition)
|
|
|
|
|
|
|
|
|
|
proc del[T](sq: SyncQueue[T], position: SyncPosition) =
|
|
|
|
|
doAssert(len(sq.requests) > position.qindex)
|
|
|
|
|
doAssert(len(sq.requests[position.qindex].requests) > position.sindex)
|
|
|
|
|
del(sq.requests[position.qindex].requests, position.sindex)
|
|
|
|
|
|
|
|
|
|
proc del[T](sq: SyncQueue[T], request: SyncRequest[T]) =
|
|
|
|
|
let pos = sq.find(request).valueOr:
|
|
|
|
|
return
|
|
|
|
|
sq.del(pos)
|
|
|
|
|
|
|
|
|
|
proc rewardForGaps[T](sq: SyncQueue[T], score: int) =
|
|
|
|
|
mixin updateScore, getStats
|
2022-03-15 17:56:56 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
for gap in sq.gapList:
|
|
|
|
|
if score < 0:
|
|
|
|
|
# Every empty response increases penalty by 25%, but not more than 200%.
|
|
|
|
|
let
|
|
|
|
|
emptyCount = gap.item.getStats(SyncResponseKind.Empty)
|
|
|
|
|
goodCount = gap.item.getStats(SyncResponseKind.Good)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if emptyCount <= goodCount:
|
|
|
|
|
gap.item.updateScore(score)
|
|
|
|
|
else:
|
|
|
|
|
let
|
|
|
|
|
weight = int(min(emptyCount - goodCount, 8'u64))
|
|
|
|
|
newScore = score + score * weight div 4
|
|
|
|
|
gap.item.updateScore(newScore)
|
2024-11-29 16:15:46 +00:00
|
|
|
|
debug "Peer received gap penalty",
|
|
|
|
|
peer = gap.item,
|
|
|
|
|
penalty = newScore,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
else:
|
|
|
|
|
gap.item.updateScore(score)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc pop*[T](sq: SyncQueue[T], peerMaxSlot: Slot, item: T): SyncRequest[T] =
|
|
|
|
|
# Searching requests queue for an empty space.
|
|
|
|
|
var count = 0
|
|
|
|
|
for qitem in sq.requests.mitems():
|
|
|
|
|
if len(qitem.requests) < sq.requestsCount:
|
|
|
|
|
if item notin qitem.requests:
|
2021-12-16 14:57:16 +00:00
|
|
|
|
return
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if qitem.data.slot > peerMaxSlot:
|
|
|
|
|
# Peer could not satisfy our request, returning empty one.
|
|
|
|
|
SyncRequest.init(sq.kind, item)
|
|
|
|
|
else:
|
|
|
|
|
doAssert(count < sq.requestsCount,
|
|
|
|
|
"You should not pop so many requests for single peer")
|
|
|
|
|
let request = SyncRequest.init(sq.kind, qitem.data, item)
|
|
|
|
|
qitem.requests.add(request)
|
|
|
|
|
request
|
|
|
|
|
else:
|
|
|
|
|
inc(count)
|
|
|
|
|
|
|
|
|
|
doAssert(count < sq.requestsCount,
|
|
|
|
|
"You should not pop so many requests for single peer")
|
|
|
|
|
|
|
|
|
|
# No empty spaces has been found in queue, so we adding new request.
|
|
|
|
|
let newrange =
|
|
|
|
|
if len(sq.requests) > 0:
|
|
|
|
|
# All requests are filled, adding one more request.
|
|
|
|
|
let lastrange = sq.requests[^1].data
|
|
|
|
|
if sq.finalSlot in lastrange:
|
|
|
|
|
# Requests queue is already at finish position, we are not going to add
|
|
|
|
|
# one more request range.
|
|
|
|
|
return SyncRequest.init(sq.kind, item)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
lastrange.next()
|
2021-12-08 21:15:29 +00:00
|
|
|
|
of SyncQueueKind.Backward:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
lastrange.prev()
|
|
|
|
|
else:
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
|
|
|
|
SyncRange.init(sq.inpSlot, sq.chunkSize)
|
|
|
|
|
of SyncQueueKind.Backward:
|
|
|
|
|
SyncRange.init(sq.inpSlot - (sq.chunkSize - 1), sq.chunkSize)
|
2021-12-16 14:57:16 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if newrange.slot > peerMaxSlot:
|
|
|
|
|
# Peer could not satisfy our request, returning empty one.
|
|
|
|
|
SyncRequest.init(sq.kind, item)
|
|
|
|
|
else:
|
|
|
|
|
let request = SyncRequest.init(sq.kind, newrange, item)
|
|
|
|
|
sq.requests.addLast(SyncQueueItem.init(request))
|
|
|
|
|
request
|
2022-01-26 12:20:08 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc wakeupWaiters[T](sq: SyncQueue[T], resetFlag = false) =
|
|
|
|
|
## Wakeup one or all blocked waiters.
|
|
|
|
|
for item in sq.waiters:
|
|
|
|
|
item.resetFlag = resetFlag
|
|
|
|
|
if not(item.future.finished()):
|
|
|
|
|
item.future.complete()
|
2022-01-26 12:20:08 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc waitForChanges[T](
|
|
|
|
|
sq: SyncQueue[T]
|
|
|
|
|
): Future[bool] {.async: (raises: [CancelledError]).} =
|
|
|
|
|
## Create new waiter and wait for completion from `wakeupWaiters()`.
|
|
|
|
|
let
|
|
|
|
|
future =
|
|
|
|
|
Future[void].Raising([CancelledError]).init("SyncQueue.waitForChanges")
|
|
|
|
|
item = SyncWaiterItem[T](future: future, resetFlag: false)
|
2023-02-11 20:48:35 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
sq.waiters.add(item)
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
try:
|
|
|
|
|
await future
|
|
|
|
|
item.resetFlag
|
|
|
|
|
finally:
|
|
|
|
|
sq.waiters.delete(sq.waiters.find(item))
|
2022-09-19 09:37:42 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
proc wakeupAndWaitWaiters[T](
|
|
|
|
|
sq: SyncQueue[T]
|
|
|
|
|
) {.async: (raises: [CancelledError]).} =
|
|
|
|
|
## This procedure will perform wakeupWaiters(true) and blocks until last
|
|
|
|
|
## waiter will be awakened.
|
|
|
|
|
let waitChanges = sq.waitForChanges()
|
|
|
|
|
sq.wakeupWaiters(true)
|
|
|
|
|
discard await waitChanges
|
2022-01-26 12:20:08 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
template advanceImpl(kind, slot: untyped, number: uint64) =
|
|
|
|
|
case kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
|
|
|
|
if slot + number < slot:
|
|
|
|
|
slot = FAR_FUTURE_SLOT
|
2021-12-08 21:15:29 +00:00
|
|
|
|
else:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
slot = slot + number
|
|
|
|
|
of SyncQueueKind.Backward:
|
|
|
|
|
if slot - number > slot:
|
|
|
|
|
slot = GENESIS_SLOT
|
|
|
|
|
else:
|
|
|
|
|
slot = slot - number
|
|
|
|
|
|
|
|
|
|
proc advanceOutput[T](sq: SyncQueue[T], number: uint64) =
|
|
|
|
|
advanceImpl(sq.kind, sq.outSlot, number)
|
|
|
|
|
|
|
|
|
|
proc advanceInput[T](sq: SyncQueue[T], number: uint64) =
|
|
|
|
|
advanceImpl(sq.kind, sq.inpSlot, number)
|
|
|
|
|
|
|
|
|
|
proc advanceQueue[T](sq: SyncQueue[T]) =
|
|
|
|
|
if len(sq.requests) > 0:
|
|
|
|
|
let item = sq.requests.popFirst()
|
|
|
|
|
sq.advanceInput(item.data.count)
|
|
|
|
|
sq.advanceOutput(item.data.count)
|
|
|
|
|
else:
|
|
|
|
|
sq.advanceInput(sq.chunkSize)
|
|
|
|
|
sq.advanceOutput(sq.chunkSize)
|
|
|
|
|
sq.wakeupWaiters()
|
|
|
|
|
|
|
|
|
|
proc resetQueue[T](sq: SyncQueue[T]) =
|
|
|
|
|
sq.requests.reset()
|
|
|
|
|
|
|
|
|
|
proc clearAndWakeup*[T](sq: SyncQueue[T]) =
|
|
|
|
|
# Reset queue and wakeup all the waiters.
|
|
|
|
|
sq.resetQueue()
|
|
|
|
|
sq.wakeupWaiters(true)
|
|
|
|
|
|
|
|
|
|
proc isEmpty*[T](sr: SyncRequest[T]): bool =
|
|
|
|
|
# Returns `true` if request `sr` is empty.
|
|
|
|
|
sr.data.count == 0'u64
|
|
|
|
|
|
|
|
|
|
proc resetWait[T](
|
|
|
|
|
sq: SyncQueue[T],
|
|
|
|
|
toSlot: Slot
|
|
|
|
|
) {.async: (raises: [CancelledError], raw: true).} =
|
|
|
|
|
sq.inpSlot = toSlot
|
|
|
|
|
sq.outSlot = toSlot
|
|
|
|
|
# We are going to wakeup all the waiters and wait for last one.
|
|
|
|
|
sq.resetQueue()
|
|
|
|
|
sq.wakeupAndWaitWaiters()
|
|
|
|
|
|
|
|
|
|
func getOpt(blobs: Opt[seq[BlobSidecars]], i: int): Opt[BlobSidecars] =
|
|
|
|
|
if blobs.isSome:
|
|
|
|
|
Opt.some(blobs.get()[i])
|
|
|
|
|
else:
|
|
|
|
|
Opt.none(BlobSidecars)
|
|
|
|
|
|
|
|
|
|
iterator blocks(
|
|
|
|
|
kind: SyncQueueKind,
|
|
|
|
|
blcks: seq[ref ForkedSignedBeaconBlock],
|
|
|
|
|
blobs: Opt[seq[BlobSidecars]]
|
|
|
|
|
): (ref ForkedSignedBeaconBlock, Opt[BlobSidecars]) =
|
|
|
|
|
case kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
|
|
|
|
for i in countup(0, len(blcks) - 1):
|
|
|
|
|
yield (blcks[i], blobs.getOpt(i))
|
|
|
|
|
of SyncQueueKind.Backward:
|
|
|
|
|
for i in countdown(len(blcks) - 1, 0):
|
|
|
|
|
yield (blcks[i], blobs.getOpt(i))
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
proc push*[T](sq: SyncQueue[T], sr: SyncRequest[T]) =
|
|
|
|
|
## Push failed request back to queue.
|
2024-11-22 19:12:45 +00:00
|
|
|
|
let pos = sq.find(sr).valueOr:
|
|
|
|
|
debug "Request is no more relevant", request = sr
|
2021-12-08 21:15:29 +00:00
|
|
|
|
return
|
2024-11-22 19:12:45 +00:00
|
|
|
|
sq.del(pos)
|
|
|
|
|
|
|
|
|
|
proc process[T](
|
|
|
|
|
sq: SyncQueue[T],
|
|
|
|
|
sr: SyncRequest[T],
|
|
|
|
|
blcks: seq[ref ForkedSignedBeaconBlock],
|
|
|
|
|
blobs: Opt[seq[BlobSidecars]],
|
|
|
|
|
maybeFinalized: bool
|
|
|
|
|
): Future[SyncProcessingResult] {.
|
|
|
|
|
async: (raises: [CancelledError]).} =
|
|
|
|
|
var
|
|
|
|
|
slot: Opt[SyncBlock]
|
|
|
|
|
unviableBlock: Opt[SyncBlock]
|
|
|
|
|
dupBlock: Opt[SyncBlock]
|
2022-05-10 11:46:14 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
if len(blcks) == 0:
|
|
|
|
|
return SyncProcessingResult.init(SyncProcessError.Empty)
|
|
|
|
|
|
|
|
|
|
for blk, blb in blocks(sq.kind, blcks, blobs):
|
|
|
|
|
let res = await sq.blockVerifier(blk[], blb, maybeFinalized)
|
|
|
|
|
if res.isOk():
|
|
|
|
|
slot = Opt.some(SyncBlock.init(blk[].slot, blk[].root))
|
|
|
|
|
else:
|
|
|
|
|
case res.error()
|
|
|
|
|
of VerifierError.MissingParent:
|
|
|
|
|
if slot.isSome() or dupBlock.isSome():
|
|
|
|
|
return SyncProcessingResult.init(
|
|
|
|
|
SyncProcessError.GoodAndMissingParent, blk[].slot, blk[].root)
|
2022-03-15 17:56:56 +00:00
|
|
|
|
else:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
return SyncProcessingResult.init(res.error(), blk[].slot, blk[].root)
|
|
|
|
|
of VerifierError.Duplicate:
|
|
|
|
|
# Keep going, happens naturally
|
|
|
|
|
if dupBlock.isNone():
|
|
|
|
|
dupBlock = Opt.some(SyncBlock.init(blk[].slot, blk[].root))
|
|
|
|
|
of VerifierError.UnviableFork:
|
|
|
|
|
# Keep going so as to register other unviable blocks with the
|
|
|
|
|
# quarantine
|
|
|
|
|
if unviableBlock.isNone():
|
|
|
|
|
# Remember the first unviable block, so we can log it
|
|
|
|
|
unviableBlock = Opt.some(SyncBlock.init(blk[].slot, blk[].root))
|
|
|
|
|
of VerifierError.Invalid:
|
|
|
|
|
return SyncProcessingResult.init(res.error(), blk[].slot, blk[].root)
|
|
|
|
|
|
|
|
|
|
if unviableBlock.isSome():
|
|
|
|
|
return SyncProcessingResult.init(VerifierError.UnviableFork,
|
|
|
|
|
unviableBlock.get())
|
|
|
|
|
if dupBlock.isSome():
|
|
|
|
|
return SyncProcessingResult.init(VerifierError.Duplicate,
|
|
|
|
|
dupBlock.get())
|
|
|
|
|
|
|
|
|
|
SyncProcessingResult.init(SyncProcessError.NoError, slot.get())
|
|
|
|
|
|
|
|
|
|
func isError(e: SyncProcessError): bool =
|
|
|
|
|
case e
|
|
|
|
|
of SyncProcessError.Empty, SyncProcessError.NoError,
|
|
|
|
|
SyncProcessError.Duplicate, SyncProcessError.GoodAndMissingParent:
|
|
|
|
|
false
|
|
|
|
|
of SyncProcessError.Invalid, SyncProcessError.UnviableFork,
|
|
|
|
|
SyncProcessError.MissingParent:
|
|
|
|
|
true
|
|
|
|
|
|
|
|
|
|
proc push*[T](
|
|
|
|
|
sq: SyncQueue[T],
|
|
|
|
|
sr: SyncRequest[T],
|
|
|
|
|
data: seq[ref ForkedSignedBeaconBlock],
|
|
|
|
|
blobs: Opt[seq[BlobSidecars]],
|
|
|
|
|
maybeFinalized: bool = false,
|
|
|
|
|
processingCb: ProcessingCallback = nil
|
|
|
|
|
) {.async: (raises: [CancelledError]).} =
|
|
|
|
|
## Push successful result to queue ``sq``.
|
|
|
|
|
mixin updateScore, updateStats, getStats
|
|
|
|
|
|
|
|
|
|
# This is backpressure handling algorithm, this algorithm is blocking
|
|
|
|
|
# all pending `push` requests if `request` is not in range.
|
|
|
|
|
var
|
|
|
|
|
position =
|
|
|
|
|
block:
|
|
|
|
|
var pos: SyncPosition
|
|
|
|
|
while true:
|
|
|
|
|
pos = sq.find(sr).valueOr:
|
2024-11-29 16:15:46 +00:00
|
|
|
|
debug "Request is no more relevant",
|
|
|
|
|
request = sr,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
# Request is not in queue anymore, probably reset happened.
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
if pos.qindex == 0:
|
|
|
|
|
# Exiting loop when request is first in queue.
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
let res = await sq.waitForChanges()
|
|
|
|
|
if res:
|
|
|
|
|
# SyncQueue reset happen
|
2024-11-29 16:15:46 +00:00
|
|
|
|
debug "Request is no more relevant, reset happen",
|
|
|
|
|
request = sr,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
return
|
|
|
|
|
except CancelledError as exc:
|
|
|
|
|
# Removing request from queue.
|
|
|
|
|
sq.del(sr)
|
|
|
|
|
raise exc
|
|
|
|
|
pos
|
|
|
|
|
|
|
|
|
|
await sq.lock.acquire()
|
|
|
|
|
try:
|
|
|
|
|
block:
|
|
|
|
|
position = sq.find(sr).valueOr:
|
|
|
|
|
# Queue has advanced, the request is no longer relevant.
|
2024-11-29 16:15:46 +00:00
|
|
|
|
debug "Request is no more relevant",
|
|
|
|
|
request = sr,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
if not(isNil(processingCb)):
|
|
|
|
|
processingCb()
|
|
|
|
|
|
|
|
|
|
let pres = await sq.process(sr, data, blobs, maybeFinalized)
|
|
|
|
|
|
|
|
|
|
case pres.code
|
|
|
|
|
of SyncProcessError.Empty:
|
|
|
|
|
# Empty responses does not affect failures count
|
|
|
|
|
debug "Received empty response",
|
|
|
|
|
request = sr,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
|
|
|
|
|
sr.item.updateStats(SyncResponseKind.Empty, 1'u64)
|
|
|
|
|
sq.gapList.add(GapItem.init(sr))
|
|
|
|
|
sq.advanceQueue()
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.Duplicate:
|
|
|
|
|
# Duplicate responses does not affect failures count
|
|
|
|
|
debug "Received duplicate response",
|
|
|
|
|
request = sr,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
sq.gapList.reset()
|
|
|
|
|
sq.advanceQueue()
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.Invalid:
|
|
|
|
|
debug "Block pool rejected peer's response",
|
|
|
|
|
request = sr,
|
|
|
|
|
invalid_block = pres.blck,
|
|
|
|
|
failures_count = sq.requests[position.qindex].failuresCount,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
|
|
|
|
|
inc(sq.requests[position.qindex].failuresCount)
|
|
|
|
|
sq.del(position)
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.UnviableFork:
|
|
|
|
|
notice "Received blocks from an unviable fork",
|
|
|
|
|
request = sr,
|
|
|
|
|
unviable_block = pres.blck,
|
|
|
|
|
failures_count = sq.requests[position.qindex].failuresCount,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
|
|
|
|
|
sr.item.updateScore(PeerScoreUnviableFork)
|
|
|
|
|
inc(sq.requests[position.qindex].failuresCount)
|
|
|
|
|
sq.del(position)
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.MissingParent:
|
|
|
|
|
debug "Unexpected missing parent",
|
|
|
|
|
request = sr,
|
|
|
|
|
missing_parent_block = pres.blck,
|
|
|
|
|
failures_count = sq.requests[position.qindex].failuresCount,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
|
|
|
|
|
sr.item.updateScore(PeerScoreMissingValues)
|
|
|
|
|
sq.rewardForGaps(PeerScoreMissingValues)
|
|
|
|
|
sq.gapList.reset()
|
|
|
|
|
inc(sq.requests[position.qindex].failuresCount)
|
|
|
|
|
sq.del(position)
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.GoodAndMissingParent:
|
|
|
|
|
# Responses which has at least one good block and a gap does not affect
|
|
|
|
|
# failures count
|
|
|
|
|
debug "Unexpected missing parent, but no rewind needed",
|
|
|
|
|
request = sr, finalized_slot = sq.getSafeSlot(),
|
|
|
|
|
missing_parent_block = pres.blck,
|
|
|
|
|
failures_count = sq.requests[position.qindex].failuresCount,
|
|
|
|
|
blocks_count = len(data),
|
|
|
|
|
blocks_map = getShortMap(sr, data),
|
2024-11-29 16:15:46 +00:00
|
|
|
|
blobs_map = getShortMap(sr, blobs),
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
|
|
|
|
|
sr.item.updateScore(PeerScoreMissingValues)
|
|
|
|
|
sq.del(position)
|
|
|
|
|
|
|
|
|
|
of SyncProcessError.NoError:
|
|
|
|
|
sr.item.updateScore(PeerScoreGoodValues)
|
|
|
|
|
sr.item.updateStats(SyncResponseKind.Good, 1'u64)
|
|
|
|
|
sq.rewardForGaps(PeerScoreGoodValues)
|
|
|
|
|
sq.gapList.reset()
|
|
|
|
|
|
|
|
|
|
if sr.hasEndGap(data):
|
|
|
|
|
sq.gapList.add(GapItem.init(sr))
|
|
|
|
|
|
|
|
|
|
sq.advanceQueue()
|
|
|
|
|
|
|
|
|
|
if pres.code.isError():
|
|
|
|
|
if sq.requests[position.qindex].failuresCount >= sq.failureResetThreshold:
|
|
|
|
|
let point = sq.getRewindPoint(pres.blck.get().slot, sq.getSafeSlot())
|
|
|
|
|
debug "Multiple repeating errors occured, rewinding",
|
|
|
|
|
failures_count = sq.requests[position.qindex].failuresCount,
|
2024-11-29 16:15:46 +00:00
|
|
|
|
rewind_slot = point,
|
|
|
|
|
sync_ident = sq.ident,
|
|
|
|
|
direction = sq.kind,
|
|
|
|
|
topics = "syncman"
|
2024-11-22 19:12:45 +00:00
|
|
|
|
await sq.resetWait(point)
|
|
|
|
|
|
|
|
|
|
except CancelledError as exc:
|
|
|
|
|
sq.del(sr)
|
|
|
|
|
raise exc
|
|
|
|
|
finally:
|
|
|
|
|
try:
|
|
|
|
|
sq.lock.release()
|
|
|
|
|
except AsyncLockError:
|
|
|
|
|
raiseAssert "Lock is not acquired"
|
|
|
|
|
|
|
|
|
|
proc checkResponse*[T](req: SyncRequest[T],
|
|
|
|
|
data: openArray[Slot]): Result[void, cstring] =
|
|
|
|
|
if len(data) == 0:
|
|
|
|
|
# Impossible to verify empty response.
|
|
|
|
|
return ok()
|
|
|
|
|
|
|
|
|
|
if lenu64(data) > req.data.count:
|
|
|
|
|
# Number of blocks in response should be less or equal to number of
|
|
|
|
|
# requested blocks.
|
|
|
|
|
return err("Too many blocks received")
|
|
|
|
|
|
|
|
|
|
var
|
|
|
|
|
slot = req.data.slot
|
|
|
|
|
rindex = 0'u64
|
|
|
|
|
dindex = 0
|
|
|
|
|
|
|
|
|
|
while (rindex < req.data.count) and (dindex < len(data)):
|
|
|
|
|
if slot < data[dindex]:
|
2022-03-15 17:56:56 +00:00
|
|
|
|
discard
|
2024-11-22 19:12:45 +00:00
|
|
|
|
elif slot == data[dindex]:
|
|
|
|
|
inc(dindex)
|
2022-03-15 17:56:56 +00:00
|
|
|
|
else:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
return err("Incorrect order or duplicate blocks found")
|
|
|
|
|
slot += 1'u64
|
|
|
|
|
rindex += 1'u64
|
|
|
|
|
|
|
|
|
|
if dindex != len(data):
|
|
|
|
|
return err("Some of the blocks are outside the requested range")
|
|
|
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
|
|
proc checkBlobsResponse*[T](req: SyncRequest[T],
|
|
|
|
|
data: openArray[Slot]): Result[void, cstring] =
|
|
|
|
|
if len(data) == 0:
|
|
|
|
|
# Impossible to verify empty response.
|
|
|
|
|
return ok()
|
|
|
|
|
|
|
|
|
|
if lenu64(data) > (req.data.count * MAX_BLOBS_PER_BLOCK):
|
|
|
|
|
# Number of blobs in response should be less or equal to number of
|
|
|
|
|
# requested (blocks * MAX_BLOBS_PER_BLOCK).
|
|
|
|
|
return err("Too many blobs received")
|
|
|
|
|
|
|
|
|
|
var
|
|
|
|
|
pslot = data[0]
|
|
|
|
|
counter = 0'u64
|
|
|
|
|
for slot in data:
|
|
|
|
|
if slot notin req.data:
|
|
|
|
|
return err("Some of the blobs are not in requested range")
|
|
|
|
|
if slot < pslot:
|
|
|
|
|
return err("Incorrect order")
|
|
|
|
|
if slot == pslot:
|
|
|
|
|
inc(counter)
|
|
|
|
|
if counter > MAX_BLOBS_PER_BLOCK:
|
|
|
|
|
return err("Number of blobs in the block exceeds the limit")
|
2022-03-15 17:56:56 +00:00
|
|
|
|
else:
|
2024-11-22 19:12:45 +00:00
|
|
|
|
counter = 1'u64
|
|
|
|
|
pslot = slot
|
2022-03-15 17:56:56 +00:00
|
|
|
|
|
2024-11-22 19:12:45 +00:00
|
|
|
|
ok()
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
proc len*[T](sq: SyncQueue[T]): uint64 {.inline.} =
|
|
|
|
|
## Returns number of slots left in queue ``sq``.
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
2022-12-23 07:42:55 +00:00
|
|
|
|
if sq.finalSlot >= sq.outSlot:
|
|
|
|
|
sq.finalSlot + 1'u64 - sq.outSlot
|
|
|
|
|
else:
|
|
|
|
|
0'u64
|
2021-12-08 21:15:29 +00:00
|
|
|
|
of SyncQueueKind.Backward:
|
2022-12-23 07:42:55 +00:00
|
|
|
|
if sq.outSlot >= sq.finalSlot:
|
|
|
|
|
sq.outSlot + 1'u64 - sq.finalSlot
|
|
|
|
|
else:
|
|
|
|
|
0'u64
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
proc total*[T](sq: SyncQueue[T]): uint64 {.inline.} =
|
|
|
|
|
## Returns total number of slots in queue ``sq``.
|
|
|
|
|
case sq.kind
|
|
|
|
|
of SyncQueueKind.Forward:
|
2022-12-23 07:42:55 +00:00
|
|
|
|
if sq.finalSlot >= sq.startSlot:
|
|
|
|
|
sq.finalSlot + 1'u64 - sq.startSlot
|
|
|
|
|
else:
|
|
|
|
|
0'u64
|
2021-12-08 21:15:29 +00:00
|
|
|
|
of SyncQueueKind.Backward:
|
2022-12-23 07:42:55 +00:00
|
|
|
|
if sq.startSlot >= sq.finalSlot:
|
|
|
|
|
sq.startSlot + 1'u64 - sq.finalSlot
|
|
|
|
|
else:
|
|
|
|
|
0'u64
|
2021-12-08 21:15:29 +00:00
|
|
|
|
|
|
|
|
|
proc progress*[T](sq: SyncQueue[T]): uint64 =
|
2022-12-23 07:42:55 +00:00
|
|
|
|
## How many useful slots we've synced so far, adjusting for how much has
|
|
|
|
|
## become obsolete by time movements
|
2024-11-22 19:12:45 +00:00
|
|
|
|
sq.total() - len(sq)
|