mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-19 02:52:58 +00:00
cbc998ed93
* initial fork-choice refactor * Add fork_choice test for "no votes" * Initial test with voting: fix handling of unknown validators and parent blocks * Fix tiebreak of votes * Cleanup debugging traces * Complexify the vote test * fakeHash use the bigEndian repr of number + fix tiebreak for good * Stash changes: found critical bug in nimcrypto `==` and var openarray * Passing fork choice tests with varying votes * Add FFG fork choice scenario + fork choice to the test suite * Not sure why lmdb / rocksdb reappeared in rebase * Add sanity checks to .nimble file + integrate fork choice tests to the test DB and test timing * Cleanup debugging echos * nimcrypto fix https://github.com/status-im/nim-beacon-chain/pull/864 as been merged, remove TODO comment * Turn fork choice exception-free * Cleanup "result" to ensure early return is properly used * Add a comment on private/public error code vs Result * result -> results following https://github.com/status-im/nim-beacon-chain/pull/866 * Address comments: - raises: [Defect] doesn't work -> TODO - process_attestation cannot fail - try/except as expression pending Nim v1.2.0 - cleanup TODOs * re-enable all sanity checks * tag no raise for process_attestation * use raises defect everywhere in fork choice and fix process_attestation test
267 lines
6.0 KiB
Nim
267 lines
6.0 KiB
Nim
# 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 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.
|
|
|
|
# import ../interpreter # included to be able to use "suiteReport"
|
|
|
|
proc setup_no_votes(): tuple[fork_choice: ForkChoice, ops: seq[Operation]] =
|
|
let balances = newSeq[Gwei](16)
|
|
let GenesisRoot = fakeHash(0)
|
|
|
|
# Initialize the fork choice context
|
|
result.fork_choice = initForkChoice(
|
|
finalized_block_slot = Slot(0), # Metadata unused in fork choice
|
|
finalized_block_state_root = default(Eth2Digest), # Metadata unused in fork choice
|
|
justified_epoch = Epoch(1),
|
|
finalized_epoch = Epoch(1),
|
|
finalized_root = GenesisRoot
|
|
).get()
|
|
|
|
# ----------------------------------
|
|
|
|
# Head should be genesis
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: GenesisRoot
|
|
)
|
|
|
|
# Add block 2
|
|
#
|
|
# 0
|
|
# /
|
|
# 2
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(2),
|
|
parent_root: GenesisRoot,
|
|
blk_justified_epoch: Epoch(1),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Head should be 2
|
|
#
|
|
# 0
|
|
# /
|
|
# 2 <- head
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(2)
|
|
)
|
|
|
|
# Add block 1 as a fork
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(1),
|
|
parent_root: GenesisRoot,
|
|
blk_justified_epoch: Epoch(1),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Head is still 2 due to tiebreaker as fakeHash(2) (0xD8...) > fakeHash(1) (0x7C...)
|
|
#
|
|
# 0
|
|
# / \
|
|
# head-> 2 1
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(2)
|
|
)
|
|
|
|
# Add block 3
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# |
|
|
# 3
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(3),
|
|
parent_root: fakeHash(1),
|
|
blk_justified_epoch: Epoch(1),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Head is still 2
|
|
#
|
|
# 0
|
|
# / \
|
|
# head-> 2 1
|
|
# |
|
|
# 3
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(2)
|
|
)
|
|
|
|
# Add block 4
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(4),
|
|
parent_root: fakeHash(2),
|
|
blk_justified_epoch: Epoch(1),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Check that head is 4
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# head-> 4 3
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(4)
|
|
)
|
|
|
|
# Add block 5 with justified epoch of 2
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
# |
|
|
# 5 <- justified epoch = 2
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(5),
|
|
parent_root: fakeHash(4),
|
|
blk_justified_epoch: Epoch(2),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Ensure the head is still 4 whilst the justified epoch is 0.
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# head-> 4 3
|
|
# |
|
|
# 5
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(1),
|
|
justified_root: GenesisRoot,
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(4)
|
|
)
|
|
|
|
# Ensure that there is an error when starting from a block with the wrong justified epoch
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
# |
|
|
# 5 <- starting from 5 with justified epoch 1 should error.
|
|
result.ops.add Operation(
|
|
kind: InvalidFindHead,
|
|
justified_epoch: Epoch(1), # <--- Wrong epoch
|
|
justified_root: fakeHash(5),
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances
|
|
)
|
|
|
|
# Set the justified epoch to 2 and the start block to 5 and ensure 5 is the head.
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
# |
|
|
# 5 <- head + justified
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(2),
|
|
justified_root: fakeHash(5),
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(5)
|
|
)
|
|
|
|
# Add block 6
|
|
#
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
# |
|
|
# 5 <- justified root
|
|
# |
|
|
# 6
|
|
result.ops.add Operation(
|
|
kind: ProcessBlock,
|
|
root: fakeHash(6),
|
|
parent_root: fakeHash(5),
|
|
blk_justified_epoch: Epoch(2),
|
|
blk_finalized_epoch: Epoch(1)
|
|
)
|
|
|
|
# Ensure 6 is the head
|
|
# 0
|
|
# / \
|
|
# 2 1
|
|
# | |
|
|
# 4 3
|
|
# |
|
|
# 5 <- justified root
|
|
# |
|
|
# 6 <- head
|
|
result.ops.add Operation(
|
|
kind: FindHead,
|
|
justified_epoch: Epoch(2),
|
|
justified_root: fakeHash(5),
|
|
finalized_epoch: Epoch(1),
|
|
justified_state_balances: balances,
|
|
expected_head: fakeHash(6)
|
|
)
|
|
|
|
proc test_no_votes() =
|
|
timedTest "fork_choice - testing no votes":
|
|
# for i in 0 ..< 6:
|
|
# echo " block (", i, ") hash: ", fakeHash(i)
|
|
# echo " ------------------------------------------------------"
|
|
|
|
var (ctx, ops) = setup_no_votes()
|
|
ctx.run(ops)
|
|
|
|
test_no_votes()
|