nimbus-eth2/beacon_chain/sync_protocol.nim

93 lines
2.8 KiB
Nim
Raw Normal View History

import
2018-11-29 03:08:34 +02:00
options,
2019-02-05 20:21:18 +01:00
chronicles, eth/[rlp, p2p], asyncdispatch2, ranges/bitranges, eth/p2p/rlpx,
2018-11-29 03:08:34 +02:00
spec/[datatypes, crypto, digest]
type
ValidatorChangeLogEntry* = object
case kind*: ValidatorSetDeltaFlags
of Activation:
2018-11-29 03:08:34 +02:00
pubkey: ValidatorPubKey
else:
index: uint32
2019-01-17 10:27:11 -08:00
ValidatorSet = seq[Validator]
2018-11-29 03:08:34 +02:00
p2pProtocol BeaconSync(version = 1,
shortName = "bcs"):
requestResponse:
2019-01-25 16:17:35 +02:00
proc getValidatorChangeLog(peer: Peer, changeLogHead: Eth2Digest) =
var bb: BeaconBlock
var bs: BeaconState
# TODO: get the changelog from the DB.
await peer.validatorChangeLog(reqId, bb, bs, [], [], @[])
proc validatorChangeLog(peer: Peer,
signedBlock: BeaconBlock,
beaconState: BeaconState,
2018-11-29 03:08:34 +02:00
added: openarray[ValidatorPubKey],
removed: openarray[uint32],
order: seq[byte])
type
# A bit shorter names for convenience
ChangeLog = BeaconSync.validatorChangeLog
ChangeLogEntry = ValidatorChangeLogEntry
2018-11-29 03:08:34 +02:00
func validate*(log: ChangeLog): bool =
# TODO:
# Assert that the number of raised bits in log.order (a.k.a population count)
# matches the number of elements in log.added
# https://en.wikichip.org/wiki/population_count
return true
iterator changes*(log: ChangeLog): ChangeLogEntry =
var
2018-11-29 03:08:34 +02:00
bits = log.added.len + log.removed.len
addedIdx = 0
removedIdx = 0
2018-11-29 03:08:34 +02:00
template nextItem(collection): auto =
let idx = `collection Idx`
inc `collection Idx`
log.collection[idx]
for i in 0 ..< bits:
2018-11-29 03:08:34 +02:00
yield if log.order.getBit(i):
ChangeLogEntry(kind: Activation, pubkey: nextItem(added))
else:
ChangeLogEntry(kind: ValidatorSetDeltaFlags.Exit, index: nextItem(removed))
2018-11-29 03:08:34 +02:00
proc getValidatorChangeLog*(node: EthereumNode, changeLogHead: Eth2Digest):
Future[(Peer, ChangeLog)] {.async.} =
while true:
2018-11-29 03:08:34 +02:00
let peer = node.randomPeerWith(BeaconSync)
if peer == nil: return
2018-11-29 03:08:34 +02:00
let res = await peer.getValidatorChangeLog(changeLogHead, timeout = 1)
if res.isSome:
return (peer, res.get)
2018-11-29 03:08:34 +02:00
proc applyValidatorChangeLog*(log: ChangeLog,
outBeaconState: var BeaconState): bool =
# TODO:
#
# 1. Validate that the signedBlock state root hash matches the
# provided beaconState
#
# 2. Validate that the applied changelog produces the correct
# new change log head
#
# 3. Check that enough signatures from the known validator set
# are present
#
# 4. Apply all changes to the validator set
#
outBeaconState.finalized_epoch =
log.signedBlock.slot div EPOCH_LENGTH
outBeaconState.validator_registry_delta_chain_tip =
log.beaconState.validator_registry_delta_chain_tip