2018-11-24 01:58:49 +02:00
|
|
|
import
|
2019-10-25 12:59:56 +02:00
|
|
|
options, tables, sets, macros,
|
2019-09-07 13:48:05 -04:00
|
|
|
chronicles, chronos, metrics, stew/ranges/bitranges,
|
2020-01-24 09:32:52 +01:00
|
|
|
spec/[datatypes, crypto, digest, helpers],
|
2019-11-25 15:36:25 +01:00
|
|
|
beacon_node_types, eth2_network, block_pool, ssz
|
2019-03-12 16:03:14 +01:00
|
|
|
|
2020-01-24 09:32:52 +01:00
|
|
|
when networkBackend == libp2p:
|
2019-12-10 01:04:35 +02:00
|
|
|
import libp2p/switch
|
2019-08-15 18:00:12 +02:00
|
|
|
|
2019-10-03 04:51:44 +03:00
|
|
|
declarePublicGauge libp2p_peers, "Number of libp2p peers"
|
2019-09-07 13:48:05 -04:00
|
|
|
|
2019-11-11 01:28:13 +00:00
|
|
|
logScope:
|
|
|
|
topics = "sync"
|
|
|
|
|
2018-11-24 01:58:49 +02:00
|
|
|
type
|
2019-07-08 13:19:52 +00:00
|
|
|
ValidatorSetDeltaFlags {.pure.} = enum
|
|
|
|
Activation = 0
|
|
|
|
Exit = 1
|
|
|
|
|
2018-11-24 01:58:49 +02:00
|
|
|
ValidatorChangeLogEntry* = object
|
|
|
|
case kind*: ValidatorSetDeltaFlags
|
2018-12-03 17:46:22 +00:00
|
|
|
of Activation:
|
2018-11-29 03:08:34 +02:00
|
|
|
pubkey: ValidatorPubKey
|
2018-11-24 01:58:49 +02:00
|
|
|
else:
|
|
|
|
index: uint32
|
|
|
|
|
2019-12-16 19:08:50 +01:00
|
|
|
BeaconBlockCallback* = proc(blck: SignedBeaconBlock) {.gcsafe.}
|
2019-09-08 18:03:41 -04:00
|
|
|
BeaconSyncNetworkState* = ref object
|
2019-11-25 15:36:25 +01:00
|
|
|
blockPool*: BlockPool
|
|
|
|
forkVersion*: array[4, byte]
|
|
|
|
onBeaconBlock*: BeaconBlockCallback
|
2019-02-18 12:34:39 +02:00
|
|
|
|
2019-09-08 18:03:41 -04:00
|
|
|
BeaconSyncPeerState* = ref object
|
2019-09-11 12:45:22 -04:00
|
|
|
initialStatusReceived: bool
|
2019-09-08 18:03:41 -04:00
|
|
|
|
2019-07-03 10:35:05 +03:00
|
|
|
BlockRootSlot* = object
|
|
|
|
blockRoot: Eth2Digest
|
|
|
|
slot: Slot
|
|
|
|
|
2019-04-09 10:53:40 +03:00
|
|
|
const
|
2019-09-11 12:45:22 -04:00
|
|
|
MAX_REQUESTED_BLOCKS = 20'u64
|
2019-11-25 15:36:25 +01:00
|
|
|
|
|
|
|
func init*(
|
|
|
|
v: BeaconSyncNetworkState, blockPool: BlockPool,
|
|
|
|
forkVersion: array[4, byte], onBeaconBlock: BeaconBlockCallback) =
|
|
|
|
v.blockPool = blockPool
|
|
|
|
v.forkVersion = forkVersion
|
|
|
|
v.onBeaconBlock = onBeaconBlock
|
|
|
|
|
|
|
|
proc importBlocks(state: BeaconSyncNetworkState,
|
2019-12-16 19:08:50 +01:00
|
|
|
blocks: openarray[SignedBeaconBlock]) {.gcsafe.} =
|
2019-06-10 14:13:53 +03:00
|
|
|
for blk in blocks:
|
2019-11-25 15:36:25 +01:00
|
|
|
state.onBeaconBlock(blk)
|
2019-06-10 14:13:53 +03:00
|
|
|
info "Forward sync imported blocks", len = blocks.len
|
|
|
|
|
2019-09-08 18:03:41 -04:00
|
|
|
type
|
2019-09-11 12:45:22 -04:00
|
|
|
StatusMsg = object
|
2019-09-08 18:03:41 -04:00
|
|
|
forkVersion*: array[4, byte]
|
2019-09-09 19:55:01 -04:00
|
|
|
finalizedRoot*: Eth2Digest
|
|
|
|
finalizedEpoch*: Epoch
|
|
|
|
headRoot*: Eth2Digest
|
|
|
|
headSlot*: Slot
|
|
|
|
|
2019-11-25 15:36:25 +01:00
|
|
|
proc getCurrentStatus(state: BeaconSyncNetworkState): StatusMsg {.gcsafe.} =
|
2019-09-09 19:55:01 -04:00
|
|
|
let
|
2019-11-25 15:36:25 +01:00
|
|
|
blockPool = state.blockPool
|
2019-09-09 19:55:01 -04:00
|
|
|
finalizedHead = blockPool.finalizedHead
|
|
|
|
headBlock = blockPool.head.blck
|
|
|
|
headRoot = headBlock.root
|
|
|
|
headSlot = headBlock.slot
|
initial 0.9.0 spec sync (#509)
* rename compute_epoch_of_slot(...) to compute_epoch_at_slot(...)
* remove some unnecessary imports; remove some crosslink-related code and tests; complete renaming of compute_epoch_of_slot(...) to compute_epoch_at_slot(...)
* rm more transfer-related code and tests; rm more unnecessary strutils imports
* rm remaining unused imports
* remove useless get_empty_per_epoch_cache(...)/compute_start_slot_of_epoch(...) calls
* rename compute_start_slot_of_epoch(...) to compute_start_slot_at_epoch(...)
* rename ACTIVATION_EXIT_DELAY to MAX_SEED_LOOKAHEAD
* update domain types to 0.9.0
* mark AttesterSlashing, IndexedAttestation, AttestationDataAndCustodyBit, DepositData, BeaconBlockHeader, Fork, integer_squareroot(...), and process_voluntary_exit(...) as 0.9.0
* mark increase_balance(...), decrease_balance(...), get_block_root(...), CheckPoint, Deposit, PendingAttestation, HistoricalBatch, is_active_validator(...), and is_slashable_attestation_data(...) as 0.9.0
* mark compute_activation_exit_epoch(...), bls_verify(...), Validator, get_active_validator_indices(...), get_current_epoch(...), get_total_active_balance(...), and get_previous_epoch(...) as 0.9.0
* mark get_block_root_at_slot(...), ProposerSlashing, get_domain(...), VoluntaryExit, mainnet preset Gwei values, minimal preset max operations, process_block_header(...), and is_slashable_validator(...) as 0.9.0
* mark makeWithdrawalCredentials(...), get_validator_churn_limit(...), get_total_balance(...), is_valid_indexed_attestation(...), bls_aggregate_pubkeys(...), initial genesis value/constants, Attestation, get_randao_mix(...), mainnet preset max operations per block constants, minimal preset Gwei values and time parameters, process_eth1_data(...), get_shuffled_seq(...), compute_committee(...), and process_slots(...) as 0.9.0; partially update get_indexed_attestation(...) to 0.9.0 by removing crosslink refs and associated tests
* mark initiate_validator_exit(...), process_registry_updates(...), BeaconBlock, Eth1Data, compute_domain(...), process_randao(...), process_attester_slashing(...), get_base_reward(...), and process_slot(...) as 0.9.0
2019-10-30 19:41:19 +00:00
|
|
|
finalizedEpoch = finalizedHead.slot.compute_epoch_at_slot()
|
2019-09-09 19:55:01 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
StatusMsg(
|
2019-11-25 15:36:25 +01:00
|
|
|
fork_version: state.forkVersion,
|
2019-09-09 19:55:01 -04:00
|
|
|
finalizedRoot: finalizedHead.blck.root,
|
|
|
|
finalizedEpoch: finalizedEpoch,
|
|
|
|
headRoot: headRoot,
|
|
|
|
headSlot: headSlot)
|
2019-09-08 18:03:41 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
proc handleInitialStatus(peer: Peer,
|
2019-11-25 15:36:25 +01:00
|
|
|
state: BeaconSyncNetworkState,
|
2019-09-11 12:45:22 -04:00
|
|
|
ourStatus: StatusMsg,
|
|
|
|
theirStatus: StatusMsg) {.async, gcsafe.}
|
2019-09-08 16:08:44 +03:00
|
|
|
|
2018-11-29 03:08:34 +02:00
|
|
|
p2pProtocol BeaconSync(version = 1,
|
2019-08-05 03:00:49 +03:00
|
|
|
rlpxName = "bcs",
|
2019-09-08 18:03:41 -04:00
|
|
|
networkState = BeaconSyncNetworkState,
|
|
|
|
peerState = BeaconSyncPeerState):
|
2019-02-18 12:34:39 +02:00
|
|
|
|
2019-06-05 05:00:07 +03:00
|
|
|
onPeerConnected do (peer: Peer):
|
2019-09-08 18:03:41 -04:00
|
|
|
if peer.wasDialed:
|
|
|
|
let
|
2019-11-25 15:36:25 +01:00
|
|
|
ourStatus = peer.networkState.getCurrentStatus()
|
2019-11-12 22:53:19 +00:00
|
|
|
# TODO: The timeout here is so high only because we fail to
|
|
|
|
# respond in time due to high CPU load in our single thread.
|
|
|
|
theirStatus = await peer.status(ourStatus, timeout = 60.seconds)
|
2019-09-09 19:55:01 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
if theirStatus.isSome:
|
2019-11-25 15:36:25 +01:00
|
|
|
await peer.handleInitialStatus(peer.networkState, ourStatus, theirStatus.get)
|
2019-02-18 12:34:39 +02:00
|
|
|
else:
|
2019-11-06 16:56:54 +02:00
|
|
|
warn "Status response not received in time"
|
2019-02-18 12:34:39 +02:00
|
|
|
|
2019-09-07 13:48:05 -04:00
|
|
|
onPeerDisconnected do (peer: Peer):
|
|
|
|
libp2p_peers.set peer.network.peers.len.int64
|
|
|
|
|
2019-09-08 18:03:41 -04:00
|
|
|
requestResponse:
|
2019-09-11 12:45:22 -04:00
|
|
|
proc status(peer: Peer, theirStatus: StatusMsg) {.libp2pProtocol("status", 1).} =
|
2019-09-08 18:03:41 -04:00
|
|
|
let
|
2019-11-25 15:36:25 +01:00
|
|
|
ourStatus = peer.networkState.getCurrentStatus()
|
2019-09-09 19:55:01 -04:00
|
|
|
|
2019-11-15 14:09:25 +00:00
|
|
|
trace "Sending status msg", ourStatus
|
2019-09-11 12:45:22 -04:00
|
|
|
await response.send(ourStatus)
|
2019-09-08 18:03:41 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
if not peer.state.initialStatusReceived:
|
|
|
|
peer.state.initialStatusReceived = true
|
2019-11-25 15:36:25 +01:00
|
|
|
await peer.handleInitialStatus(peer.networkState, ourStatus, theirStatus)
|
2019-09-08 18:03:41 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
proc statusResp(peer: Peer, msg: StatusMsg)
|
2019-02-18 12:34:39 +02:00
|
|
|
|
2019-09-10 01:50:37 -04:00
|
|
|
proc goodbye(peer: Peer, reason: DisconnectionReason) {.libp2pProtocol("goodbye", 1).}
|
2019-05-22 10:13:15 +03:00
|
|
|
|
2019-08-05 03:00:49 +03:00
|
|
|
requestResponse:
|
2019-09-08 16:54:31 +02:00
|
|
|
proc beaconBlocksByRange(
|
2019-08-05 03:00:49 +03:00
|
|
|
peer: Peer,
|
|
|
|
headBlockRoot: Eth2Digest,
|
2019-09-10 01:50:37 -04:00
|
|
|
startSlot: Slot,
|
2019-08-05 03:00:49 +03:00
|
|
|
count: uint64,
|
|
|
|
step: uint64) {.
|
2019-09-08 18:03:41 -04:00
|
|
|
libp2pProtocol("beacon_blocks_by_range", 1).} =
|
2019-11-11 01:28:13 +00:00
|
|
|
trace "got range request", peer, count, startSlot, headBlockRoot, step
|
2019-09-10 01:50:37 -04:00
|
|
|
|
|
|
|
if count > 0'u64:
|
2019-09-11 12:45:22 -04:00
|
|
|
let count = if step != 0: min(count, MAX_REQUESTED_BLOCKS.uint64) else: 1
|
2019-11-25 15:36:25 +01:00
|
|
|
let pool = peer.networkState.blockPool
|
2019-09-11 12:45:22 -04:00
|
|
|
var results: array[MAX_REQUESTED_BLOCKS, BlockRef]
|
2019-09-10 01:50:37 -04:00
|
|
|
let
|
|
|
|
lastPos = min(count.int, results.len) - 1
|
|
|
|
firstPos = pool.getBlockRange(headBlockRoot, startSlot, step,
|
|
|
|
results.toOpenArray(0, lastPos))
|
|
|
|
for i in firstPos.int .. lastPos.int:
|
2019-11-11 01:28:13 +00:00
|
|
|
trace "wrote response block", slot = results[i].slot
|
2019-09-10 01:50:37 -04:00
|
|
|
await response.write(pool.get(results[i]).data)
|
2019-08-05 03:00:49 +03:00
|
|
|
|
2019-09-08 16:54:31 +02:00
|
|
|
proc beaconBlocksByRoot(
|
2019-08-05 03:00:49 +03:00
|
|
|
peer: Peer,
|
|
|
|
blockRoots: openarray[Eth2Digest]) {.
|
2019-09-08 18:03:41 -04:00
|
|
|
libp2pProtocol("beacon_blocks_by_root", 1).} =
|
2019-09-08 22:39:44 -04:00
|
|
|
let
|
2019-11-25 15:36:25 +01:00
|
|
|
pool = peer.networkState.blockPool
|
2019-08-05 03:00:49 +03:00
|
|
|
|
|
|
|
for root in blockRoots:
|
2019-09-08 16:08:44 +03:00
|
|
|
let blockRef = pool.getRef(root)
|
2019-09-08 22:39:44 -04:00
|
|
|
if not isNil(blockRef):
|
|
|
|
await response.write(pool.get(blockRef).data)
|
2019-08-05 03:00:49 +03:00
|
|
|
|
|
|
|
proc beaconBlocks(
|
|
|
|
peer: Peer,
|
2019-12-16 19:08:50 +01:00
|
|
|
blocks: openarray[SignedBeaconBlock])
|
2019-05-22 10:13:15 +03:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
proc handleInitialStatus(peer: Peer,
|
2019-11-25 15:36:25 +01:00
|
|
|
state: BeaconSyncNetworkState,
|
2019-09-11 12:45:22 -04:00
|
|
|
ourStatus: StatusMsg,
|
|
|
|
theirStatus: StatusMsg) {.async, gcsafe.} =
|
2019-12-10 01:04:35 +02:00
|
|
|
when networkBackend == libp2p:
|
|
|
|
# TODO: This doesn't seem like an appropraite place for this call,
|
|
|
|
# but it's hard to pick a better place at the moment.
|
|
|
|
# nim-libp2p plans to add a general `onPeerConnected` callback which
|
|
|
|
# will allow us to implement the subscription earlier.
|
|
|
|
# The root of the problem is that both sides must call `subscribeToPeer`
|
|
|
|
# before any GossipSub traffic will flow between them.
|
|
|
|
await peer.network.switch.subscribeToPeer(peer.info)
|
2019-09-09 19:55:01 -04:00
|
|
|
|
2019-11-25 15:36:25 +01:00
|
|
|
if theirStatus.forkVersion != state.forkVersion:
|
|
|
|
notice "Irrelevant peer",
|
|
|
|
peer, theirFork = theirStatus.forkVersion, ourFork = state.forkVersion
|
2019-09-09 19:55:01 -04:00
|
|
|
await peer.disconnect(IrrelevantNetwork)
|
|
|
|
return
|
|
|
|
|
|
|
|
# TODO: onPeerConnected runs unconditionally for every connected peer, but we
|
|
|
|
# don't need to sync with everybody. The beacon node should detect a situation
|
|
|
|
# where it needs to sync and it should execute the sync algorithm with a certain
|
|
|
|
# number of randomly selected peers. The algorithm itself must be extracted in a proc.
|
|
|
|
try:
|
|
|
|
libp2p_peers.set peer.network.peers.len.int64
|
|
|
|
|
|
|
|
debug "Peer connected. Initiating sync", peer,
|
2019-11-12 22:53:19 +00:00
|
|
|
localHeadSlot = ourStatus.headSlot,
|
|
|
|
remoteHeadSlot = theirStatus.headSlot,
|
|
|
|
remoteHeadRoot = theirStatus.headRoot
|
2019-09-09 19:55:01 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
let bestDiff = cmp((ourStatus.finalizedEpoch, ourStatus.headSlot),
|
|
|
|
(theirStatus.finalizedEpoch, theirStatus.headSlot))
|
2019-09-09 19:55:01 -04:00
|
|
|
if bestDiff >= 0:
|
|
|
|
# Nothing to do?
|
|
|
|
debug "Nothing to sync", peer
|
|
|
|
else:
|
|
|
|
# TODO: Check for WEAK_SUBJECTIVITY_PERIOD difference and terminate the
|
|
|
|
# connection if it's too big.
|
2019-09-11 12:45:22 -04:00
|
|
|
var s = ourStatus.headSlot + 1
|
|
|
|
var theirStatus = theirStatus
|
|
|
|
while s <= theirStatus.headSlot:
|
2019-11-26 01:39:33 +02:00
|
|
|
let numBlocksToRequest = min(uint64(theirStatus.headSlot - s) + 1,
|
2019-09-11 12:45:22 -04:00
|
|
|
MAX_REQUESTED_BLOCKS)
|
2019-09-10 01:50:37 -04:00
|
|
|
|
2019-09-11 12:45:22 -04:00
|
|
|
debug "Requesting blocks", peer, remoteHeadSlot = theirStatus.headSlot,
|
2019-09-10 01:50:37 -04:00
|
|
|
ourHeadSlot = s,
|
|
|
|
numBlocksToRequest
|
|
|
|
|
2019-11-12 22:53:19 +00:00
|
|
|
# TODO: The timeout here is so high only because we fail to
|
|
|
|
# respond in time due to high CPU load in our single thread.
|
2019-09-11 12:45:22 -04:00
|
|
|
let blocks = await peer.beaconBlocksByRange(theirStatus.headRoot, s,
|
2019-11-12 22:53:19 +00:00
|
|
|
numBlocksToRequest, 1'u64,
|
|
|
|
timeout = 60.seconds)
|
2019-09-09 19:55:01 -04:00
|
|
|
if blocks.isSome:
|
2019-09-10 01:50:37 -04:00
|
|
|
info "got blocks", total = blocks.get.len
|
2019-09-09 19:55:01 -04:00
|
|
|
if blocks.get.len == 0:
|
|
|
|
info "Got 0 blocks while syncing", peer
|
|
|
|
break
|
2019-09-10 01:50:37 -04:00
|
|
|
|
2019-11-25 15:36:25 +01:00
|
|
|
state.importBlocks(blocks.get)
|
2019-12-16 19:08:50 +01:00
|
|
|
let lastSlot = blocks.get[^1].message.slot
|
2019-09-09 19:55:01 -04:00
|
|
|
if lastSlot <= s:
|
|
|
|
info "Slot did not advance during sync", peer
|
|
|
|
break
|
|
|
|
|
|
|
|
s = lastSlot + 1
|
|
|
|
|
|
|
|
# TODO: Maybe this shouldn't happen so often.
|
|
|
|
# The alternative could be watching up a timer here.
|
2019-11-25 15:36:25 +01:00
|
|
|
|
|
|
|
let statusResp = await peer.status(state.getCurrentStatus())
|
2019-09-11 12:45:22 -04:00
|
|
|
if statusResp.isSome:
|
|
|
|
theirStatus = statusResp.get
|
2019-09-09 19:55:01 -04:00
|
|
|
else:
|
|
|
|
# We'll ignore this error and we'll try to request
|
|
|
|
# another range optimistically. If that fails, the
|
|
|
|
# syncing will be interrupted.
|
|
|
|
discard
|
|
|
|
else:
|
2019-11-05 16:02:26 +01:00
|
|
|
error "Did not get any blocks from peer. Aborting sync."
|
2019-09-09 19:55:01 -04:00
|
|
|
break
|
|
|
|
|
2019-12-02 16:38:18 +01:00
|
|
|
except CatchableError as e:
|
|
|
|
warn "Failed to sync with peer", peer, err = e.msg
|
|
|
|
|