2020-05-22 20:04:52 +03:00
|
|
|
# beacon_chain
|
2021-02-15 17:40:00 +01:00
|
|
|
# Copyright (c) 2018-2021 Status Research & Development GmbH
|
2020-05-22 20:04:52 +03: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.
|
|
|
|
|
2021-03-26 07:52:01 +01:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
import
|
|
|
|
# Standard library
|
2021-03-26 15:11:06 +01:00
|
|
|
std/[os, random, strutils],
|
2020-05-22 20:04:52 +03:00
|
|
|
|
|
|
|
# Nimble packages
|
2020-07-14 03:01:23 +02:00
|
|
|
stew/shims/[tables, macros],
|
2021-03-26 15:11:06 +01:00
|
|
|
chronos, confutils, metrics,
|
2020-05-27 20:06:28 +03:00
|
|
|
chronicles,
|
2021-03-04 09:13:23 +00:00
|
|
|
json_serialization/std/[options, net],
|
2020-05-22 20:04:52 +03:00
|
|
|
|
|
|
|
# Local modules
|
2021-03-05 14:12:00 +01:00
|
|
|
./spec/[datatypes, digest, crypto, helpers, network, signatures],
|
|
|
|
./spec/eth2_apis/beacon_rpc_client,
|
|
|
|
./sync/sync_manager,
|
|
|
|
"."/[conf, beacon_clock, version],
|
|
|
|
./networking/[eth2_network, eth2_discovery],
|
|
|
|
./beacon_node_types,
|
|
|
|
./nimbus_binary_common,
|
|
|
|
./ssz/merkleization,
|
|
|
|
./spec/eth2_apis/callsigs_types,
|
2021-03-06 08:32:55 +01:00
|
|
|
./validators/[attestation_aggregation, keystore_management, validator_pool, slashing_protection],
|
2021-03-23 06:57:10 +00:00
|
|
|
./eth/db/[kvstore, kvstore_sqlite3],
|
|
|
|
./eth/keys, ./eth/p2p/discoveryv5/random2
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-06-10 13:30:57 +03:00
|
|
|
logScope: topics = "vc"
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
type
|
|
|
|
ValidatorClient = ref object
|
|
|
|
config: ValidatorClientConf
|
2020-06-29 20:30:19 +03:00
|
|
|
graffitiBytes: GraffitiBytes
|
2020-05-22 20:04:52 +03:00
|
|
|
client: RpcHttpClient
|
|
|
|
beaconClock: BeaconClock
|
|
|
|
attachedValidators: ValidatorPool
|
2020-05-27 20:06:28 +03:00
|
|
|
fork: Fork
|
2020-06-19 12:21:17 +03:00
|
|
|
proposalsForCurrentEpoch: Table[Slot, ValidatorPubKey]
|
|
|
|
attestationsForEpoch: Table[Epoch, Table[Slot, seq[AttesterDuties]]]
|
2020-05-27 20:06:28 +03:00
|
|
|
beaconGenesis: BeaconGenesisTuple
|
|
|
|
|
2020-06-19 12:21:17 +03:00
|
|
|
template attemptUntilSuccess(vc: ValidatorClient, body: untyped) =
|
|
|
|
while true:
|
|
|
|
try:
|
|
|
|
body
|
|
|
|
break
|
|
|
|
except CatchableError as err:
|
|
|
|
warn "Caught an unexpected error", err = err.msg
|
2020-09-01 16:44:40 +03:00
|
|
|
waitFor sleepAsync(chronos.seconds(vc.config.retryDelay))
|
2020-06-19 12:21:17 +03:00
|
|
|
|
2020-05-27 20:06:28 +03:00
|
|
|
proc getValidatorDutiesForEpoch(vc: ValidatorClient, epoch: Epoch) {.gcsafe, async.} =
|
2020-07-08 13:11:22 +03:00
|
|
|
info "Getting validator duties for epoch", epoch = epoch
|
|
|
|
|
2020-05-27 20:06:28 +03:00
|
|
|
let proposals = await vc.client.get_v1_validator_duties_proposer(epoch)
|
|
|
|
# update the block proposal duties this VC should do during this epoch
|
2020-06-19 12:21:17 +03:00
|
|
|
vc.proposalsForCurrentEpoch.clear()
|
2020-05-27 20:06:28 +03:00
|
|
|
for curr in proposals:
|
|
|
|
if vc.attachedValidators.validators.contains curr.public_key:
|
2020-06-19 12:21:17 +03:00
|
|
|
vc.proposalsForCurrentEpoch.add(curr.slot, curr.public_key)
|
2020-05-27 20:06:28 +03:00
|
|
|
|
|
|
|
# couldn't use mapIt in ANY shape or form so reverting to raw loops - sorry Sean Parent :|
|
|
|
|
var validatorPubkeys: seq[ValidatorPubKey]
|
|
|
|
for key in vc.attachedValidators.validators.keys:
|
|
|
|
validatorPubkeys.add key
|
2020-06-19 12:21:17 +03:00
|
|
|
|
|
|
|
proc getAttesterDutiesForEpoch(epoch: Epoch) {.gcsafe, async.} =
|
|
|
|
# make sure there's an entry
|
|
|
|
if not vc.attestationsForEpoch.contains epoch:
|
|
|
|
vc.attestationsForEpoch.add(epoch, Table[Slot, seq[AttesterDuties]]())
|
2020-09-14 14:13:30 +03:00
|
|
|
let attestations = await vc.client.get_v1_validator_duties_attester(
|
2020-07-08 13:11:22 +03:00
|
|
|
epoch, validatorPubkeys)
|
2020-06-19 12:21:17 +03:00
|
|
|
for a in attestations:
|
|
|
|
if vc.attestationsForEpoch[epoch].hasKeyOrPut(a.slot, @[a]):
|
|
|
|
vc.attestationsForEpoch[epoch][a.slot].add(a)
|
|
|
|
|
2020-07-08 13:11:22 +03:00
|
|
|
# clear both for the current epoch and the next because a change of
|
|
|
|
# fork could invalidate the attester duties even the current epoch
|
|
|
|
vc.attestationsForEpoch.clear()
|
|
|
|
await getAttesterDutiesForEpoch(epoch)
|
2020-06-19 12:21:17 +03:00
|
|
|
# obtain the attestation duties this VC should do during the next epoch
|
|
|
|
await getAttesterDutiesForEpoch(epoch + 1)
|
2020-05-27 20:06:28 +03:00
|
|
|
|
|
|
|
# for now we will get the fork each time we update the validator duties for each epoch
|
2020-06-19 12:21:17 +03:00
|
|
|
# TODO should poll occasionally `/v1/config/fork_schedule`
|
2020-05-27 20:06:28 +03:00
|
|
|
vc.fork = await vc.client.get_v1_beacon_states_fork("head")
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-07-08 13:11:22 +03:00
|
|
|
var numAttestationsForEpoch = 0
|
|
|
|
for _, dutiesForSlot in vc.attestationsForEpoch[epoch]:
|
|
|
|
numAttestationsForEpoch += dutiesForSlot.len
|
|
|
|
|
|
|
|
info "Got validator duties for epoch",
|
|
|
|
num_proposals = vc.proposalsForCurrentEpoch.len,
|
|
|
|
num_attestations = numAttestationsForEpoch
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
proc onSlotStart(vc: ValidatorClient, lastSlot, scheduledSlot: Slot) {.gcsafe, async.} =
|
|
|
|
|
|
|
|
let
|
|
|
|
# The slot we should be at, according to the clock
|
|
|
|
beaconTime = vc.beaconClock.now()
|
|
|
|
wallSlot = beaconTime.toSlot()
|
|
|
|
|
|
|
|
let
|
|
|
|
slot = wallSlot.slot # afterGenesis == true!
|
|
|
|
nextSlot = slot + 1
|
2020-06-19 12:21:17 +03:00
|
|
|
epoch = slot.compute_epoch_at_slot
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-06-10 13:30:57 +03:00
|
|
|
info "Slot start",
|
|
|
|
lastSlot = shortLog(lastSlot),
|
|
|
|
scheduledSlot = shortLog(scheduledSlot),
|
|
|
|
beaconTime = shortLog(beaconTime),
|
2020-07-16 15:16:51 +02:00
|
|
|
portBN = vc.config.rpcPort
|
2020-06-05 12:57:40 +03:00
|
|
|
|
2020-09-01 16:38:34 +03:00
|
|
|
# Check before any re-scheduling of onSlotStart()
|
|
|
|
checkIfShouldStopAtEpoch(scheduledSlot, vc.config.stopAtEpoch)
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
try:
|
2020-06-05 12:57:40 +03:00
|
|
|
# at the start of each epoch - request all validator duties
|
|
|
|
# TODO perhaps call this not on the first slot of each Epoch but perhaps
|
|
|
|
# 1 slot earlier because there are a few back-and-forth requests which
|
|
|
|
# could take up time for attesting... Perhaps this should be called more
|
|
|
|
# than once per epoch because of forks & other events...
|
2020-12-22 10:05:36 +01:00
|
|
|
#
|
|
|
|
# calling it before epoch n starts means one can't ensure knowing about
|
|
|
|
# epoch n+1.
|
2020-06-05 12:57:40 +03:00
|
|
|
if slot.isEpoch:
|
2020-06-19 12:21:17 +03:00
|
|
|
await getValidatorDutiesForEpoch(vc, epoch)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-05-27 20:06:28 +03:00
|
|
|
# check if we have a validator which needs to propose on this slot
|
2020-06-19 12:21:17 +03:00
|
|
|
if vc.proposalsForCurrentEpoch.contains slot:
|
|
|
|
let public_key = vc.proposalsForCurrentEpoch[slot]
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2021-05-04 15:17:28 +02:00
|
|
|
notice "Proposing block", slot = slot, public_key = public_key
|
|
|
|
|
|
|
|
let validator = vc.attachedValidators.validators[public_key]
|
|
|
|
let randao_reveal = await validator.genRandaoReveal(
|
|
|
|
vc.fork, vc.beaconGenesis.genesis_validators_root, slot)
|
|
|
|
var newBlock = SignedBeaconBlock(
|
|
|
|
message: await vc.client.get_v1_validator_block(slot, vc.graffitiBytes, randao_reveal)
|
|
|
|
)
|
|
|
|
newBlock.root = hash_tree_root(newBlock.message)
|
|
|
|
|
|
|
|
# TODO: signing_root is recomputed in signBlockProposal just after
|
|
|
|
let signing_root = compute_block_root(vc.fork, vc.beaconGenesis.genesis_validators_root, slot, newBlock.root)
|
2020-09-16 13:30:03 +02:00
|
|
|
let notSlashable = vc.attachedValidators
|
2021-05-04 15:17:28 +02:00
|
|
|
.slashingProtection
|
|
|
|
.registerBlock(
|
|
|
|
newBlock.message.proposer_index.ValidatorIndex, public_key, slot,
|
|
|
|
signing_root)
|
2020-09-16 13:30:03 +02:00
|
|
|
|
2021-05-04 15:17:28 +02:00
|
|
|
if notSlashable.isOk:
|
2020-09-16 13:30:03 +02:00
|
|
|
newBlock.signature = await validator.signBlockProposal(
|
|
|
|
vc.fork, vc.beaconGenesis.genesis_validators_root, slot, newBlock.root)
|
|
|
|
|
|
|
|
discard await vc.client.post_v1_validator_block(newBlock)
|
|
|
|
else:
|
|
|
|
warn "Slashing protection activated for block proposal",
|
|
|
|
validator = public_key,
|
|
|
|
slot = slot,
|
|
|
|
existingProposal = notSlashable.error
|
2020-05-27 20:06:28 +03:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/validator.md#attesting
|
2020-06-05 12:57:40 +03:00
|
|
|
# A validator should create and broadcast the attestation to the associated
|
|
|
|
# attestation subnet when either (a) the validator has received a valid
|
|
|
|
# block from the expected block proposer for the assigned slot or
|
|
|
|
# (b) one-third of the slot has transpired (`SECONDS_PER_SLOT / 3` seconds
|
|
|
|
# after the start of slot) -- whichever comes first.
|
|
|
|
discard await vc.beaconClock.sleepToSlotOffset(
|
|
|
|
seconds(int64(SECONDS_PER_SLOT)) div 3, slot, "Waiting to send attestations")
|
2020-05-27 20:06:28 +03:00
|
|
|
|
2020-06-05 12:57:40 +03:00
|
|
|
# check if we have validators which need to attest on this slot
|
2020-07-08 13:11:22 +03:00
|
|
|
if vc.attestationsForEpoch.contains(epoch) and
|
|
|
|
vc.attestationsForEpoch[epoch].contains slot:
|
2020-09-14 14:13:30 +03:00
|
|
|
var validatorToAttestationDataRoot: Table[ValidatorPubKey, Eth2Digest]
|
2020-06-19 12:21:17 +03:00
|
|
|
for a in vc.attestationsForEpoch[epoch][slot]:
|
2020-07-08 13:11:22 +03:00
|
|
|
let validator = vc.attachedValidators.validators[a.public_key]
|
2020-12-07 14:51:14 +02:00
|
|
|
let ad = await vc.client.get_v1_validator_attestation_data(slot, a.committee_index)
|
2020-05-27 20:06:28 +03:00
|
|
|
|
2021-05-04 15:17:28 +02:00
|
|
|
# TODO signing_root is recomputed in produceAndSignAttestation/signAttestation just after
|
|
|
|
let signing_root = compute_attestation_root(
|
|
|
|
vc.fork, vc.beaconGenesis.genesis_validators_root, ad)
|
2020-09-16 13:30:03 +02:00
|
|
|
let notSlashable = vc.attachedValidators
|
2021-05-04 15:17:28 +02:00
|
|
|
.slashingProtection
|
|
|
|
.registerAttestation(
|
|
|
|
a.validator_index, a.public_key, ad.source.epoch, ad.target.epoch, signing_root)
|
2020-09-16 13:30:03 +02:00
|
|
|
if notSlashable.isOk():
|
|
|
|
# TODO I don't like these (u)int64-to-int conversions...
|
|
|
|
let attestation = await validator.produceAndSignAttestation(
|
2020-11-16 17:10:51 +01:00
|
|
|
ad, a.committee_length.int, a.validator_committee_index,
|
2020-09-16 13:30:03 +02:00
|
|
|
vc.fork, vc.beaconGenesis.genesis_validators_root)
|
|
|
|
|
2021-05-10 09:13:36 +02:00
|
|
|
notice "Sending attestation to beacon node",
|
|
|
|
public_key = a.public_key, attestation = shortLog(attestation)
|
|
|
|
let ok = await vc.client.post_v1_beacon_pool_attestations(attestation)
|
|
|
|
if not ok:
|
|
|
|
warn "Failed to send attestation to beacon node",
|
|
|
|
public_key = a.public_key, attestation = shortLog(attestation)
|
2020-09-18 17:31:45 +03:00
|
|
|
|
|
|
|
validatorToAttestationDataRoot[a.public_key] = attestation.data.hash_tree_root
|
2020-09-16 13:30:03 +02:00
|
|
|
else:
|
|
|
|
warn "Slashing protection activated for attestation",
|
|
|
|
validator = a.public_key,
|
|
|
|
badVoteDetails = $notSlashable.error
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2021-02-25 13:37:22 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v1.0.1/specs/phase0/validator.md#broadcast-aggregate
|
2020-09-14 14:13:30 +03:00
|
|
|
# If the validator is selected to aggregate (is_aggregator), then they
|
|
|
|
# broadcast their best aggregate as a SignedAggregateAndProof to the global
|
|
|
|
# aggregate channel (beacon_aggregate_and_proof) two-thirds of the way
|
|
|
|
# through the slot-that is, SECONDS_PER_SLOT * 2 / 3 seconds after the start
|
|
|
|
# of slot.
|
|
|
|
if slot > 2:
|
|
|
|
discard await vc.beaconClock.sleepToSlotOffset(
|
|
|
|
seconds(int64(SECONDS_PER_SLOT * 2)) div 3, slot,
|
|
|
|
"Waiting to aggregate attestations")
|
|
|
|
|
|
|
|
# loop again over all of our validators which need to attest on
|
|
|
|
# this slot and check if we should also aggregate attestations
|
|
|
|
for a in vc.attestationsForEpoch[epoch][slot]:
|
|
|
|
let validator = vc.attachedValidators.validators[a.public_key]
|
|
|
|
let slot_signature = await getSlotSig(validator, vc.fork,
|
|
|
|
vc.beaconGenesis.genesis_validators_root, slot)
|
|
|
|
|
2020-09-18 17:31:45 +03:00
|
|
|
if is_aggregator(a.committee_length, slot_signature) and
|
|
|
|
validatorToAttestationDataRoot.contains(a.public_key):
|
2020-10-01 20:56:42 +02:00
|
|
|
notice "Aggregating", slot = slot, public_key = a.public_key
|
2020-09-14 14:13:30 +03:00
|
|
|
|
|
|
|
let aa = await vc.client.get_v1_validator_aggregate_attestation(
|
|
|
|
slot, validatorToAttestationDataRoot[a.public_key])
|
|
|
|
let aap = AggregateAndProof(aggregator_index: a.validator_index.uint64,
|
|
|
|
aggregate: aa, selection_proof: slot_signature)
|
2020-10-01 20:56:42 +02:00
|
|
|
let sig = await signAggregateAndProof(validator,
|
2020-09-14 14:13:30 +03:00
|
|
|
aap, vc.fork, vc.beaconGenesis.genesis_validators_root)
|
|
|
|
var signedAP = SignedAggregateAndProof(message: aap, signature: sig)
|
|
|
|
discard await vc.client.post_v1_validator_aggregate_and_proofs(signedAP)
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
except CatchableError as err:
|
2020-06-19 12:21:17 +03:00
|
|
|
warn "Caught an unexpected error", err = err.msg, slot = shortLog(slot)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
|
|
|
let
|
|
|
|
nextSlotStart = saturate(vc.beaconClock.fromNow(nextSlot))
|
|
|
|
|
2020-06-10 13:30:57 +03:00
|
|
|
info "Slot end",
|
|
|
|
slot = shortLog(slot),
|
|
|
|
nextSlot = shortLog(nextSlot),
|
2020-07-16 15:16:51 +02:00
|
|
|
portBN = vc.config.rpcPort
|
2020-06-10 13:30:57 +03:00
|
|
|
|
|
|
|
when declared(GC_fullCollect):
|
|
|
|
# The slots in the validator client work as frames in a game: we want to make
|
|
|
|
# sure that we're ready for the next one and don't get stuck in lengthy
|
|
|
|
# garbage collection tasks when time is of essence in the middle of a slot -
|
|
|
|
# while this does not guarantee that we'll never collect during a slot, it
|
|
|
|
# makes sure that all the scratch space we used during slot tasks (logging,
|
|
|
|
# temporary buffers etc) gets recycled for the next slot that is likely to
|
|
|
|
# need similar amounts of memory.
|
|
|
|
GC_fullCollect()
|
2020-12-22 10:05:36 +01:00
|
|
|
|
|
|
|
if (slot - 2).isEpoch and (slot.epoch + 1) in vc.attestationsForEpoch:
|
|
|
|
for slot, attesterDuties in vc.attestationsForEpoch[slot.epoch + 1].pairs:
|
|
|
|
for ad in attesterDuties:
|
|
|
|
let
|
|
|
|
validator = vc.attachedValidators.validators[ad.public_key]
|
|
|
|
sig = await validator.getSlotSig(
|
|
|
|
vc.fork, vc.beaconGenesis.genesis_validators_root, slot)
|
|
|
|
discard await vc.client.post_v1_validator_beacon_committee_subscriptions(
|
|
|
|
ad.committee_index, ad.slot, true, ad.public_key, sig)
|
2020-06-10 13:30:57 +03:00
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
addTimer(nextSlotStart) do (p: pointer):
|
|
|
|
asyncCheck vc.onSlotStart(slot, nextSlot)
|
|
|
|
|
2021-03-26 07:52:01 +01:00
|
|
|
{.pop.} # TODO moduletests exceptions
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
programMain:
|
2020-11-09 16:23:15 +02:00
|
|
|
let config = makeBannerAndConfig("Nimbus validator client " & fullVersionStr, ValidatorClientConf)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-10-12 13:54:41 +03:00
|
|
|
setupStdoutLogging(config.logLevel)
|
|
|
|
|
2020-07-15 16:15:55 +03:00
|
|
|
setupLogging(config.logLevel, config.logFile)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2021-03-23 06:57:10 +00:00
|
|
|
# Doesn't use std/random directly, but dependencies might
|
|
|
|
let rng = keys.newRng()
|
|
|
|
if rng.isNil:
|
|
|
|
randomize()
|
|
|
|
else:
|
|
|
|
randomize(rng[].rand(high(int)))
|
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
case config.cmd
|
|
|
|
of VCNoCommand:
|
|
|
|
debug "Launching validator client",
|
|
|
|
version = fullVersionStr,
|
|
|
|
cmdParams = commandLineParams(),
|
|
|
|
config
|
|
|
|
|
|
|
|
var vc = ValidatorClient(
|
|
|
|
config: config,
|
2020-06-29 20:30:19 +03:00
|
|
|
client: newRpcHttpClient(),
|
2020-09-08 11:32:43 +00:00
|
|
|
graffitiBytes: if config.graffiti.isSome: config.graffiti.get
|
2020-06-29 20:30:19 +03:00
|
|
|
else: defaultGraffitiBytes())
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-05-27 20:06:28 +03:00
|
|
|
# load all the validators from the data dir into memory
|
2020-05-22 20:04:52 +03:00
|
|
|
for curr in vc.config.validatorKeys:
|
2020-11-28 00:34:25 +01:00
|
|
|
vc.attachedValidators.addLocalValidator(
|
performance fixes (#2259)
* performance fixes
* don't mark tree cache as dirty on read-only List accesses
* store only blob in memory for keys and signatures, parse blob lazily
* compare public keys by blob instead of parsing / converting to raw
* compare Eth2Digest using non-constant-time comparison
* avoid some unnecessary validator copying
This branch will in particular speed up deposit processing which has
been slowing down block replay.
Pre (mainnet, 1600 blocks):
```
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3450.269, 0.000, 3450.269, 3450.269, 1, Initialize DB
0.417, 0.822, 0.036, 21.098, 1400, Load block from database
16.521, 0.000, 16.521, 16.521, 1, Load state from database
27.906, 50.846, 8.104, 1507.633, 1350, Apply block
52.617, 37.029, 20.640, 135.938, 50, Apply epoch block
```
Post:
```
3502.715, 0.000, 3502.715, 3502.715, 1, Initialize DB
0.080, 0.560, 0.035, 21.015, 1400, Load block from database
17.595, 0.000, 17.595, 17.595, 1, Load state from database
15.706, 11.028, 8.300, 107.537, 1350, Apply block
33.217, 12.622, 17.331, 60.580, 50, Apply epoch block
```
* more perf fixes
* load EpochRef cache into StateCache more aggressively
* point out security concern with public key cache
* reuse proposer index from state when processing block
* avoid genericAssign in a few more places
* don't parse key when signature is unparseable
* fix `==` overload for Eth2Digest
* preallocate validator list when getting active validators
* speed up proposer index calculation a little bit
* reuse cache when replaying blocks in ncli_db
* avoid a few more copying loops
```
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3279.158, 0.000, 3279.158, 3279.158, 1, Initialize DB
0.072, 0.357, 0.035, 13.400, 1400, Load block from database
17.295, 0.000, 17.295, 17.295, 1, Load state from database
5.918, 9.896, 0.198, 98.028, 1350, Apply block
15.888, 10.951, 7.902, 39.535, 50, Apply epoch block
0.000, 0.000, 0.000, 0.000, 0, Database block store
```
* clear full balance cache before processing rewards and penalties
```
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
3947.901, 0.000, 3947.901, 3947.901, 1, Initialize DB
0.124, 0.506, 0.026, 202.370, 363345, Load block from database
97.614, 0.000, 97.614, 97.614, 1, Load state from database
0.186, 0.188, 0.012, 99.561, 357262, Advance slot, non-epoch
14.161, 5.966, 1.099, 395.511, 11524, Advance slot, epoch
1.372, 4.170, 0.017, 276.401, 363345, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database block store
```
2021-01-25 13:04:18 +01:00
|
|
|
curr.toPubKey, curr, none(ValidatorIndex))
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-09-08 11:32:43 +00:00
|
|
|
waitFor vc.client.connect($vc.config.rpcAddress, vc.config.rpcPort)
|
2020-07-08 13:11:22 +03:00
|
|
|
info "Connected to BN",
|
|
|
|
port = vc.config.rpcPort,
|
|
|
|
address = vc.config.rpcAddress
|
2020-05-27 20:06:28 +03:00
|
|
|
|
2020-06-19 12:21:17 +03:00
|
|
|
vc.attemptUntilSuccess:
|
|
|
|
# init the beacon clock
|
|
|
|
vc.beaconGenesis = waitFor vc.client.get_v1_beacon_genesis()
|
|
|
|
vc.beaconClock = BeaconClock.init(vc.beaconGenesis.genesis_time)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-10-06 10:51:33 +02:00
|
|
|
vc.attachedValidators.slashingProtection =
|
|
|
|
SlashingProtectionDB.init(
|
|
|
|
vc.beaconGenesis.genesis_validators_root,
|
2021-02-09 16:23:06 +01:00
|
|
|
config.validatorsDir(), "slashing_protection"
|
2020-10-06 10:51:33 +02:00
|
|
|
)
|
2020-09-16 13:30:03 +02:00
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
let
|
|
|
|
curSlot = vc.beaconClock.now().slotOrZero()
|
|
|
|
nextSlot = curSlot + 1 # No earlier than GENESIS_SLOT + 1
|
|
|
|
fromNow = saturate(vc.beaconClock.fromNow(nextSlot))
|
|
|
|
|
2020-06-19 12:21:17 +03:00
|
|
|
vc.attemptUntilSuccess:
|
|
|
|
waitFor vc.getValidatorDutiesForEpoch(curSlot.compute_epoch_at_slot)
|
2020-05-27 20:06:28 +03:00
|
|
|
|
2020-05-22 20:04:52 +03:00
|
|
|
info "Scheduling first slot action",
|
|
|
|
beaconTime = shortLog(vc.beaconClock.now()),
|
|
|
|
nextSlot = shortLog(nextSlot),
|
2020-07-16 15:16:51 +02:00
|
|
|
fromNow = shortLog(fromNow)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
|
|
|
addTimer(fromNow) do (p: pointer) {.gcsafe.}:
|
|
|
|
asyncCheck vc.onSlotStart(curSlot, nextSlot)
|
|
|
|
|
|
|
|
runForever()
|