Official state tests - parsing and loading beacon state (#227)
* initial commit of official state tests * sanity check fixture * Parsing official state test is mostly working (Except BLS signature) * Successfully load state test * Use json-serialization instead of json and display deserialized and from scratch beacon state hashes * Add official state test as a smoke parsing test
This commit is contained in:
parent
ff7adcefbd
commit
ad133a0222
|
@ -0,0 +1,3 @@
|
||||||
|
[submodule "tests/official/eth2.0-tests"]
|
||||||
|
path = tests/official/eth2.0-tests
|
||||||
|
url = https://github.com/ethereum/eth2.0-tests
|
|
@ -16,4 +16,5 @@ import
|
||||||
./test_ssz,
|
./test_ssz,
|
||||||
./test_state_transition,
|
./test_state_transition,
|
||||||
./test_sync_protocol,
|
./test_sync_protocol,
|
||||||
./test_validator
|
./test_validator,
|
||||||
|
./official/test_fixture_state
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
Note on serialization hacks:
|
||||||
|
|
||||||
|
### FAR_FUTURE_SLOT (18446744073709551615)
|
||||||
|
|
||||||
|
The FAR_FUTURE_SLOT (18446744073709551615) has been rewritten as a string **in the YAML file**
|
||||||
|
as it's 2^64-1 and Nim by default try to parse it into a int64 (which can represents up to 2^63-1).
|
||||||
|
|
||||||
|
The YAML file is then converted to JSON for easy input to the json serialization/deserialization
|
||||||
|
with beacon chain type support.
|
||||||
|
|
||||||
|
"18446744073709551615" is then replaced again by uint64 18446744073709551615.
|
||||||
|
|
||||||
|
### Compressed signature
|
||||||
|
|
||||||
|
In `latest_block_header` field, the signatures and randao_reveals are
|
||||||
|
`"0x000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"`
|
||||||
|
but that is not a valid compressed BLS signature, the zero signature should be:
|
||||||
|
`"0xc00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"`
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 3ec28295b0c8365f0ec7ad79cfe933755021ee1b
|
|
@ -0,0 +1,153 @@
|
||||||
|
# beacon_chain
|
||||||
|
# Copyright (c) 2018 Status Research & Development GmbH
|
||||||
|
# Licensed and distributed under either of
|
||||||
|
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
|
||||||
|
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
|
||||||
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
|
|
||||||
|
import
|
||||||
|
# Standard lib
|
||||||
|
json, streams, strutils,
|
||||||
|
# Dependencies
|
||||||
|
yaml.tojson,
|
||||||
|
# Status libs
|
||||||
|
blscurve, nimcrypto, byteutils,
|
||||||
|
eth/common, serialization, json_serialization,
|
||||||
|
# Beacon chain internals
|
||||||
|
../../beacon_chain/spec/[datatypes, crypto, digest]
|
||||||
|
|
||||||
|
export nimcrypto.toHex
|
||||||
|
|
||||||
|
type
|
||||||
|
# TODO: use ref object to avoid allocating
|
||||||
|
# so much on the stack - pending https://github.com/status-im/nim-json-serialization/issues/3
|
||||||
|
StateTest* = object
|
||||||
|
title*: string
|
||||||
|
summary*: string
|
||||||
|
test_suite*: string
|
||||||
|
fork*: string
|
||||||
|
test_cases*: seq[TestCase]
|
||||||
|
|
||||||
|
TestConstants* = object
|
||||||
|
SHARD_COUNT*: int
|
||||||
|
TARGET_COMMITTEE_SIZE*: int
|
||||||
|
MAX_BALANCE_CHURN_QUOTIENT*: int
|
||||||
|
MAX_INDICES_PER_SLASHABLE_VOTE*: int
|
||||||
|
MAX_EXIT_DEQUEUES_PER_EPOCH*: int
|
||||||
|
SHUFFLE_ROUND_COUNT*: int
|
||||||
|
DEPOSIT_CONTRACT_TREE_DEPTH*: int
|
||||||
|
MIN_DEPOSIT_AMOUNT*: uint64
|
||||||
|
MAX_DEPOSIT_AMOUNT*: uint64
|
||||||
|
FORK_CHOICE_BALANCE_INCREMENT*: uint64
|
||||||
|
EJECTION_BALANCE*: uint64
|
||||||
|
GENESIS_FORK_VERSION*: uint32
|
||||||
|
GENESIS_SLOT*: Slot
|
||||||
|
GENESIS_EPOCH*: Epoch
|
||||||
|
GENESIS_START_SHARD*: uint64
|
||||||
|
BLS_WITHDRAWAL_PREFIX_BYTE*: array[1, byte]
|
||||||
|
SECONDS_PER_SLOT*: uint64
|
||||||
|
MIN_ATTESTATION_INCLUSION_DELAY*: uint64
|
||||||
|
SLOTS_PER_EPOCH*: int
|
||||||
|
MIN_SEED_LOOKAHEAD*: int
|
||||||
|
ACTIVATION_EXIT_DELAY*: int
|
||||||
|
EPOCHS_PER_ETH1_VOTING_PERIOD*: uint64
|
||||||
|
SLOTS_PER_HISTORICAL_ROOT*: int
|
||||||
|
MIN_VALIDATOR_WITHDRAWABILITY_DELAY*: uint64
|
||||||
|
PERSISTENT_COMMITTEE_PERIOD*: uint64
|
||||||
|
LATEST_RANDAO_MIXES_LENGTH*: int
|
||||||
|
LATEST_ACTIVE_INDEX_ROOTS_LENGTH*: int
|
||||||
|
LATEST_SLASHED_EXIT_LENGTH*: int
|
||||||
|
BASE_REWARD_QUOTIENT*: uint64
|
||||||
|
WHISTLEBLOWER_REWARD_QUOTIENT*: uint64
|
||||||
|
ATTESTATION_INCLUSION_REWARD_QUOTIENT*: uint64
|
||||||
|
INACTIVITY_PENALTY_QUOTIENT*: uint64
|
||||||
|
MIN_PENALTY_QUOTIENT*: int
|
||||||
|
MAX_PROPOSER_SLASHINGS*: int
|
||||||
|
MAX_ATTESTER_SLASHINGS*: int
|
||||||
|
MAX_ATTESTATIONS*: int
|
||||||
|
MAX_DEPOSITS*: int
|
||||||
|
MAX_VOLUNTARY_EXITS*: int
|
||||||
|
MAX_TRANSFERS*: int
|
||||||
|
DOMAIN_BEACON_BLOCK*: SignatureDomain
|
||||||
|
DOMAIN_RANDAO*: SignatureDomain
|
||||||
|
DOMAIN_ATTESTATION*: SignatureDomain
|
||||||
|
DOMAIN_DEPOSIT*: SignatureDomain
|
||||||
|
DOMAIN_VOLUNTARY_EXIT*: SignatureDomain
|
||||||
|
DOMAIN_TRANSFER*: SignatureDomain
|
||||||
|
|
||||||
|
TestCase* = object
|
||||||
|
name*: string
|
||||||
|
config*: TestConstants
|
||||||
|
verify_signatures*: bool
|
||||||
|
initial_state*: BeaconState
|
||||||
|
blocks*: seq[BeaconBlock]
|
||||||
|
expected_state*: ExpectedState
|
||||||
|
|
||||||
|
ExpectedState* = object
|
||||||
|
## TODO what is this?
|
||||||
|
slot*: Slot
|
||||||
|
|
||||||
|
# #######################
|
||||||
|
# Default init
|
||||||
|
proc default*(T: typedesc): T = discard
|
||||||
|
|
||||||
|
# #######################
|
||||||
|
# JSON deserialization
|
||||||
|
|
||||||
|
proc readValue*[N: static int](r: var JsonReader, a: var array[N, byte]) {.inline.} =
|
||||||
|
# Needed for;
|
||||||
|
# - BLS_WITHDRAWAL_PREFIX_BYTE
|
||||||
|
# - FOrk datatypes
|
||||||
|
# TODO: are all bytes and bytearray serialized as hex?
|
||||||
|
# if so export that to nim-eth
|
||||||
|
hexToByteArray(r.readValue(string), a)
|
||||||
|
|
||||||
|
proc parseStateTests*(jsonPath: string): StateTest =
|
||||||
|
try:
|
||||||
|
result = Json.loadFile(jsonPath, StateTest)
|
||||||
|
except SerializationError as err:
|
||||||
|
writeStackTrace()
|
||||||
|
stderr.write "Json load issue for file \"", jsonPath, "\"\n"
|
||||||
|
stderr.write err.formatMsg(jsonPath), "\n"
|
||||||
|
quit 1
|
||||||
|
|
||||||
|
# #######################
|
||||||
|
# Yaml to JSON conversion
|
||||||
|
|
||||||
|
proc yamlToJson*(file: string): seq[JsonNode] =
|
||||||
|
try:
|
||||||
|
let fs = openFileStream(file)
|
||||||
|
defer: fs.close()
|
||||||
|
result = fs.loadToJson()
|
||||||
|
except IOError:
|
||||||
|
echo "Exception when reading file: " & file
|
||||||
|
raise
|
||||||
|
except OverflowError:
|
||||||
|
echo "Overflow exception when parsing. Did you stringify 18446744073709551615 (-1)?"
|
||||||
|
raise
|
||||||
|
|
||||||
|
when isMainModule:
|
||||||
|
# Do not forget to stringify FAR_EPOCH_SLOT = 18446744073709551615 (-1) in the YAML file
|
||||||
|
# And unstringify it in the produced JSON file
|
||||||
|
|
||||||
|
import os, typetraits
|
||||||
|
|
||||||
|
const
|
||||||
|
DefaultYML = "tests/official/sanity-check_default-config_100-vals-first_test.yaml"
|
||||||
|
DefaultOutputPath = "tests/official/sanity-check_default-config_100-vals-first_test.json"
|
||||||
|
|
||||||
|
var fileName, outputPath: string
|
||||||
|
if paramCount() == 0:
|
||||||
|
fileName = DefaultYML
|
||||||
|
outputPath = DefaultOutputPath
|
||||||
|
elif paramCount() == 1:
|
||||||
|
fileName = paramStr(1)
|
||||||
|
outputPath = DefaultOutputPath
|
||||||
|
elif paramCount() >= 2:
|
||||||
|
fileName = paramStr(1)
|
||||||
|
outputPath = paramStr(2)
|
||||||
|
|
||||||
|
let jsonString = $DefaultYML.yamlToJson[0]
|
||||||
|
DefaultOutputPath.writeFile jsonString
|
||||||
|
|
||||||
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,53 @@
|
||||||
|
# beacon_chain
|
||||||
|
# Copyright (c) 2018 Status Research & Development GmbH
|
||||||
|
# Licensed and distributed under either of
|
||||||
|
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
|
||||||
|
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
|
||||||
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
|
|
||||||
|
import
|
||||||
|
# Standard lib
|
||||||
|
ospaths, strutils, json, unittest,
|
||||||
|
# Beacon chain internals
|
||||||
|
../../beacon_chain/spec/[datatypes, crypto, digest, beaconstate],
|
||||||
|
../../beacon_chain/ssz,
|
||||||
|
# Test utilities
|
||||||
|
./fixtures_utils
|
||||||
|
|
||||||
|
const TestFolder = currentSourcePath.rsplit(DirSep, 1)[0]
|
||||||
|
const TestsPath = "/sanity-check_default-config_100-vals-first_test.json"
|
||||||
|
|
||||||
|
suite "Official - State tests": # Initializing a beacon state from the deposits
|
||||||
|
var stateTests: StateTest
|
||||||
|
test "Parsing the official state tests into Nimbus beacon types":
|
||||||
|
stateTests = parseStateTests(TestFolder & TestsPath)
|
||||||
|
doAssert $stateTests.test_cases[0].name == "test_empty_block_transition"
|
||||||
|
var initialState: BeaconState
|
||||||
|
test "Initializing from scratch a new beacon chain with the same constants and deposit configuration as official state":
|
||||||
|
var deposits: seq[Deposit]
|
||||||
|
var index = 0'u64
|
||||||
|
for v in stateTests.test_cases[0].initial_state.validator_registry:
|
||||||
|
deposits.add Deposit(
|
||||||
|
proof: default(array[DEPOSIT_CONTRACT_TREE_DEPTH, Eth2Digest]),
|
||||||
|
index: index,
|
||||||
|
deposit_data: DepositData(
|
||||||
|
amount: 32000000000'u64, # TODO: read that from validator_balances
|
||||||
|
timestamp: 0'u64, # TODO: not initialized in test
|
||||||
|
deposit_input: DepositInput(
|
||||||
|
pubkey: v.pubkey,
|
||||||
|
withdrawal_credentials: v.withdrawal_credentials,
|
||||||
|
proof_of_possession: default(ValidatorSig) # TODO: not initialized in test
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
initialState = get_genesis_beacon_state(
|
||||||
|
genesis_validator_deposits = deposits,
|
||||||
|
genesis_time = 0,
|
||||||
|
genesis_eth1_data = Eth1Data()
|
||||||
|
)
|
||||||
|
test "[For information] Comparing state hashes":
|
||||||
|
# TODO - Add official hashes when available
|
||||||
|
# TODO - Make that a blocking test requirement
|
||||||
|
echo "Deserialized state hash: 0x" & $stateTests.test_cases[0].initial_state.hash_tree_root()
|
||||||
|
echo "From-scratch state hash: 0x" & $initialState.hash_tree_root()
|
Loading…
Reference in New Issue