mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-09 05:52:45 +00:00
23eea197f6
* Implement split preset/config support This is the initial bulk refactor to introduce runtime config values in a number of places, somewhat replacing the existing mechanism of loading network metadata. It still needs more work, this is the initial refactor that introduces runtime configuration in some of the places that need it. The PR changes the way presets and constants work, to match the spec. In particular, a "preset" now refers to the compile-time configuration while a "cfg" or "RuntimeConfig" is the dynamic part. A single binary can support either mainnet or minimal, but not both. Support for other presets has been removed completely (can be readded, in case there's need). There's a number of outstanding tasks: * `SECONDS_PER_SLOT` still needs fixing * loading custom runtime configs needs redoing * checking constants against YAML file * yeerongpilly support `build/nimbus_beacon_node --network=yeerongpilly --discv5:no --log-level=DEBUG` * load fork epoch from config * fix fork digest sent in status * nicer error string for request failures * fix tools * one more * fixup * fixup * fixup * use "standard" network definition folder in local testnet Files are loaded from their standard locations, including genesis etc, to conform to the format used in the `eth2-networks` repo. * fix launch scripts, allow unknown config values * fix base config of rest test * cleanups * bundle mainnet config using common loader * fix spec links and names * only include supported preset in binary * drop yeerongpilly, add altair-devnet-0, support boot_enr.yaml
153 lines
5.2 KiB
Nim
153 lines
5.2 KiB
Nim
import
|
|
stats, os, strformat, times,
|
|
../tests/testblockutil,
|
|
../beacon_chain/[extras, beacon_chain_db],
|
|
../beacon_chain/ssz/[merkleization, ssz_serialization],
|
|
../beacon_chain/spec/[
|
|
beaconstate, crypto, datatypes, digest, forkedbeaconstate_helpers,
|
|
helpers, presets],
|
|
../beacon_chain/consensus_object_pools/[blockchain_dag, block_pools_types],
|
|
../beacon_chain/eth1/eth1_monitor
|
|
|
|
template withTimer*(stats: var RunningStat, body: untyped) =
|
|
# TODO unify timing somehow
|
|
let start = cpuTime()
|
|
|
|
block:
|
|
body
|
|
|
|
let stop = cpuTime()
|
|
stats.push stop - start
|
|
|
|
template withTimerRet*(stats: var RunningStat, body: untyped): untyped =
|
|
let start = cpuTime()
|
|
let tmp = block:
|
|
body
|
|
let stop = cpuTime()
|
|
stats.push stop - start
|
|
|
|
tmp
|
|
|
|
func verifyConsensus*(state: BeaconState, attesterRatio: auto) =
|
|
if attesterRatio < 0.63:
|
|
doAssert state.current_justified_checkpoint.epoch == 0
|
|
doAssert state.finalized_checkpoint.epoch == 0
|
|
|
|
# Quorum is 2/3 of validators, and at low numbers, quantization effects
|
|
# can dominate, so allow for play above/below attesterRatio of 2/3.
|
|
if attesterRatio < 0.72:
|
|
return
|
|
|
|
let current_epoch = get_current_epoch(state)
|
|
if current_epoch >= 3:
|
|
doAssert state.current_justified_checkpoint.epoch + 1 >= current_epoch
|
|
if current_epoch >= 4:
|
|
doAssert state.finalized_checkpoint.epoch + 2 >= current_epoch
|
|
|
|
func verifyConsensus*(state: ForkedHashedBeaconState, attesterRatio: auto) =
|
|
if attesterRatio < 0.63:
|
|
doAssert getStateField(state, current_justified_checkpoint).epoch == 0
|
|
doAssert getStateField(state, finalized_checkpoint).epoch == 0
|
|
|
|
# Quorum is 2/3 of validators, and at low numbers, quantization effects
|
|
# can dominate, so allow for play above/below attesterRatio of 2/3.
|
|
if attesterRatio < 0.72:
|
|
return
|
|
|
|
let current_epoch = get_current_epoch(state)
|
|
if current_epoch >= 3:
|
|
doAssert getStateField(
|
|
state, current_justified_checkpoint).epoch + 1 >= current_epoch
|
|
if current_epoch >= 4:
|
|
doAssert getStateField(
|
|
state, finalized_checkpoint).epoch + 2 >= current_epoch
|
|
|
|
proc loadGenesis*(validators: Natural, validate: bool):
|
|
(ref HashedBeaconState, DepositContractSnapshot) =
|
|
let
|
|
genesisFn =
|
|
&"genesis_{const_preset}_{validators}_{SPEC_VERSION}.ssz"
|
|
contractSnapshotFn =
|
|
&"deposit_contract_snapshot_{const_preset}_{validators}_{SPEC_VERSION}.ssz"
|
|
res = (ref HashedBeaconState)()
|
|
|
|
if fileExists(genesisFn) and fileExists(contractSnapshotFn):
|
|
res.data = SSZ.loadFile(genesisFn, BeaconState)
|
|
res.root = hash_tree_root(res.data)
|
|
if res.data.slot != GENESIS_SLOT:
|
|
echo "Can only start from genesis state"
|
|
quit 1
|
|
|
|
if res.data.validators.len != validators:
|
|
echo &"Supplied genesis file has {res.data.validators.len} validators, while {validators} where requested, running anyway"
|
|
|
|
echo &"Loaded {genesisFn}..."
|
|
|
|
# TODO check that the private keys are interop keys
|
|
|
|
let contractSnapshot = SSZ.loadFile(contractSnapshotFn,
|
|
DepositContractSnapshot)
|
|
(res, contractSnapshot)
|
|
else:
|
|
echo "Genesis file not found, making one up (use nimbus_beacon_node createTestnet to make one)"
|
|
|
|
echo "Preparing validators..."
|
|
let
|
|
flags = if validate: {} else: {skipBlsValidation}
|
|
deposits = makeInitialDeposits(validators.uint64, flags)
|
|
|
|
echo "Generating Genesis..."
|
|
var merkleizer = init DepositsMerkleizer
|
|
for d in deposits:
|
|
merkleizer.addChunk hash_tree_root(d).data
|
|
let contractSnapshot = DepositContractSnapshot(
|
|
depositContractState: merkleizer.toDepositContractState)
|
|
|
|
res.data = initialize_beacon_state_from_eth1(
|
|
defaultRuntimeConfig,
|
|
Eth2Digest(),
|
|
0,
|
|
deposits,
|
|
flags)[]
|
|
|
|
res.root = hash_tree_root(res.data)
|
|
|
|
echo &"Saving to {genesisFn}..."
|
|
SSZ.saveFile(genesisFn, res.data)
|
|
echo &"Saving to {contractSnapshotFn}..."
|
|
SSZ.saveFile(contractSnapshotFn, contractSnapshot)
|
|
|
|
(res, contractSnapshot)
|
|
|
|
proc printTimers*[Timers: enum](
|
|
validate: bool,
|
|
timers: array[Timers, RunningStat]
|
|
) =
|
|
proc fmtTime(t: float): string = &"{t * 1000 :>12.3f}, "
|
|
|
|
echo "All time are ms"
|
|
echo &"{\"Average\" :>12}, {\"StdDev\" :>12}, {\"Min\" :>12}, " &
|
|
&"{\"Max\" :>12}, {\"Samples\" :>12}, {\"Test\" :>12}"
|
|
|
|
if not validate:
|
|
echo "Validation is turned off meaning that no BLS operations are performed"
|
|
|
|
for t in Timers:
|
|
echo fmtTime(timers[t].mean), fmtTime(timers[t].standardDeviationS),
|
|
fmtTime(timers[t].min), fmtTime(timers[t].max), &"{timers[t].n :>12}, ",
|
|
$t
|
|
|
|
proc printTimers*[Timers: enum](
|
|
state: BeaconState, attesters: RunningStat, validate: bool,
|
|
timers: array[Timers, RunningStat]) =
|
|
echo "Validators: ", state.validators.len, ", epoch length: ", SLOTS_PER_EPOCH
|
|
echo "Validators per attestation (mean): ", attesters.mean
|
|
printTimers(validate, timers)
|
|
|
|
proc printTimers*[Timers: enum](
|
|
state: ForkedHashedBeaconState, attesters: RunningStat, validate: bool,
|
|
timers: array[Timers, RunningStat]) =
|
|
echo "Validators: ", getStateField(state, validators).len, ", epoch length: ", SLOTS_PER_EPOCH
|
|
echo "Validators per attestation (mean): ", attesters.mean
|
|
printTimers(validate, timers)
|