2019-02-19 23:35:02 +00:00
|
|
|
# beacon_chain
|
2020-03-11 09:26:18 +00:00
|
|
|
# Copyright (c) 2018-2020 Status Research & Development GmbH
|
2019-02-19 23:35:02 +00:00
|
|
|
# Licensed and distributed under either of
|
2019-11-25 15:30:02 +00:00
|
|
|
# * 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).
|
2019-02-19 23:35:02 +00:00
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
2019-11-14 10:47:55 +00:00
|
|
|
{.used.}
|
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
import
|
|
|
|
../beacon_chain/spec/datatypes,
|
|
|
|
../beacon_chain/ssz
|
|
|
|
|
2019-02-19 23:35:02 +00:00
|
|
|
import
|
2020-04-29 11:44:07 +00:00
|
|
|
unittest,
|
|
|
|
chronicles,
|
|
|
|
stew/byteutils,
|
|
|
|
./testutil, ./testblockutil,
|
2020-07-09 23:02:40 +00:00
|
|
|
../beacon_chain/spec/[digest, validator, state_transition,
|
|
|
|
helpers, beaconstate, presets],
|
2020-07-09 09:29:32 +00:00
|
|
|
../beacon_chain/[beacon_node_types, attestation_pool, block_pool, extras],
|
|
|
|
../beacon_chain/fork_choice/[fork_choice_types, fork_choice]
|
|
|
|
|
|
|
|
template wrappedTimedTest(name: string, body: untyped) =
|
|
|
|
# `check` macro takes a copy of whatever it's checking, on the stack!
|
|
|
|
block: # Symbol namespacing
|
|
|
|
proc wrappedTest() =
|
|
|
|
timedTest name:
|
|
|
|
body
|
|
|
|
wrappedTest()
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
suiteReport "Attestation pool processing" & preset():
|
|
|
|
## For now just test that we can compile and execute block processing with
|
|
|
|
## mock data.
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
setup:
|
|
|
|
# Genesis state that results in 3 members per committee
|
|
|
|
var
|
2020-07-07 23:02:14 +00:00
|
|
|
blockPool = newClone(BlockPool.init(defaultRuntimePreset, makeTestDB(SLOTS_PER_EPOCH * 3)))
|
2020-06-10 06:58:12 +00:00
|
|
|
pool = newClone(AttestationPool.init(blockPool[]))
|
|
|
|
state = newClone(loadTailState(blockPool[]))
|
2020-04-29 11:44:07 +00:00
|
|
|
# Slot 0 is a finalized slot - won't be making attestations for it..
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, state.data.data.slot + 1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
timedTest "Can add and retrieve simple attestation" & preset():
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
|
|
|
# Create an attestation for slot 1!
|
|
|
|
beacon_committee = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, beacon_committee[0], cache)
|
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation)
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, MIN_ATTESTATION_INCLUSION_DELAY.Slot + 1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let attestations = pool[].getAttestationsForBlock(state.data.data)
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
|
|
|
attestations.len == 1
|
|
|
|
|
|
|
|
timedTest "Attestations may arrive in any order" & preset():
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
|
|
|
# Create an attestation for slot 1!
|
|
|
|
bc0 = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation0 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[0], cache)
|
|
|
|
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, state.data.data.slot + 1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
let
|
|
|
|
bc1 = get_beacon_committee(state.data.data,
|
|
|
|
state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation1 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc1[0], cache)
|
|
|
|
|
|
|
|
# test reverse order
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation1)
|
|
|
|
pool[].addAttestation(attestation0)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-05-19 14:37:29 +00:00
|
|
|
discard process_slots(state.data, MIN_ATTESTATION_INCLUSION_DELAY.Slot + 1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let attestations = pool[].getAttestationsForBlock(state.data.data)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
check:
|
|
|
|
attestations.len == 1
|
|
|
|
|
|
|
|
timedTest "Attestations should be combined" & preset():
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
|
|
|
# Create an attestation for slot 1!
|
|
|
|
bc0 = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation0 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[0], cache)
|
|
|
|
attestation1 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[1], cache)
|
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation0)
|
|
|
|
pool[].addAttestation(attestation1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, MIN_ATTESTATION_INCLUSION_DELAY.Slot + 1)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let attestations = pool[].getAttestationsForBlock(state.data.data)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
check:
|
|
|
|
attestations.len == 1
|
|
|
|
|
|
|
|
timedTest "Attestations may overlap, bigger first" & preset():
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
var
|
|
|
|
# Create an attestation for slot 1!
|
|
|
|
bc0 = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation0 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[0], cache)
|
|
|
|
attestation1 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[1], cache)
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
attestation0.combine(attestation1, {})
|
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation0)
|
|
|
|
pool[].addAttestation(attestation1)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, MIN_ATTESTATION_INCLUSION_DELAY.Slot + 1)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let attestations = pool[].getAttestationsForBlock(state.data.data)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
|
|
|
attestations.len == 1
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
timedTest "Attestations may overlap, smaller first" & preset():
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
var
|
|
|
|
# Create an attestation for slot 1!
|
|
|
|
bc0 = get_beacon_committee(state.data.data,
|
|
|
|
state.data.data.slot, 0.CommitteeIndex, cache)
|
|
|
|
attestation0 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[0], cache)
|
|
|
|
attestation1 = makeAttestation(
|
|
|
|
state.data.data, state.blck.root, bc0[1], cache)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
attestation0.combine(attestation1, {})
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation1)
|
|
|
|
pool[].addAttestation(attestation0)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-05-19 15:46:29 +00:00
|
|
|
check:
|
|
|
|
process_slots(state.data, MIN_ATTESTATION_INCLUSION_DELAY.Slot + 1)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let attestations = pool[].getAttestationsForBlock(state.data.data)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
|
|
|
attestations.len == 1
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
timedTest "Fork choice returns latest block with no attestations":
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
2020-06-10 06:58:12 +00:00
|
|
|
b1 = addTestBlock(state.data, blockPool[].tail.root, cache)
|
2020-07-16 13:16:51 +00:00
|
|
|
b1Add = blockpool[].addRawBlock(b1) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
2020-06-10 06:58:12 +00:00
|
|
|
|
|
|
|
let head = pool[].selectHead()
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
2020-07-09 09:29:32 +00:00
|
|
|
head == b1Add[]
|
2019-12-19 14:13:35 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
2020-07-16 13:16:51 +00:00
|
|
|
b2 = addTestBlock(state.data, b1.root, cache)
|
|
|
|
b2Add = blockpool[].addRawBlock(b2) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
2020-06-10 06:58:12 +00:00
|
|
|
|
|
|
|
let head2 = pool[].selectHead()
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
2020-07-09 09:29:32 +00:00
|
|
|
head2 == b2Add[]
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
timedTest "Fork choice returns block with attestation":
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
2020-06-10 06:58:12 +00:00
|
|
|
b10 = makeTestBlock(state.data, blockPool[].tail.root, cache)
|
2020-07-16 13:16:51 +00:00
|
|
|
b10Add = blockpool[].addRawBlock(b10) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
2020-06-10 06:58:12 +00:00
|
|
|
|
|
|
|
let head = pool[].selectHead()
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
2020-07-09 09:29:32 +00:00
|
|
|
head == b10Add[]
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
2020-06-10 06:58:12 +00:00
|
|
|
b11 = makeTestBlock(state.data, blockPool[].tail.root, cache,
|
2020-06-29 17:30:19 +00:00
|
|
|
graffiti = GraffitiBytes [1'u8, 0, 0, 0 ,0 ,0 ,0 ,0 ,0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
|
2020-04-29 11:44:07 +00:00
|
|
|
)
|
2020-07-16 13:16:51 +00:00
|
|
|
b11Add = blockpool[].addRawBlock(b11) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
2019-12-19 13:02:28 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
bc1 = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, 1.CommitteeIndex, cache)
|
2020-07-16 13:16:51 +00:00
|
|
|
attestation0 = makeAttestation(state.data.data, b10.root, bc1[0], cache)
|
2019-06-03 08:26:38 +00:00
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation0)
|
2019-06-03 08:26:38 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let head2 = pool[].selectHead()
|
2019-06-03 08:26:38 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
check:
|
|
|
|
# Single vote for b10 and no votes for b11
|
2020-07-09 09:29:32 +00:00
|
|
|
head2 == b10Add[]
|
2019-06-03 08:26:38 +00:00
|
|
|
|
2020-04-29 11:44:07 +00:00
|
|
|
let
|
2020-07-16 13:16:51 +00:00
|
|
|
attestation1 = makeAttestation(state.data.data, b11.root, bc1[1], cache)
|
|
|
|
attestation2 = makeAttestation(state.data.data, b11.root, bc1[2], cache)
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation1)
|
2019-06-03 08:26:38 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let head3 = pool[].selectHead()
|
|
|
|
# Warning - the tiebreak are incorrect and guaranteed consensus fork, it should be bigger
|
2020-07-16 13:16:51 +00:00
|
|
|
let smaller = if b10.root.data < b11.root.data: b10Add else: b11Add
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
check:
|
2020-06-10 06:58:12 +00:00
|
|
|
# Ties broken lexicographically in spec -> ?
|
|
|
|
# all implementations favor the biggest root
|
|
|
|
# TODO
|
|
|
|
# currently using smaller as we have used for over a year
|
2020-07-09 09:29:32 +00:00
|
|
|
head3 == smaller[]
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-07-09 09:29:32 +00:00
|
|
|
pool[].addAttestation(attestation2)
|
2020-04-29 11:44:07 +00:00
|
|
|
|
2020-06-10 06:58:12 +00:00
|
|
|
let head4 = pool[].selectHead()
|
2020-04-29 11:44:07 +00:00
|
|
|
|
|
|
|
check:
|
|
|
|
# Two votes for b11
|
2020-07-09 09:29:32 +00:00
|
|
|
head4 == b11Add[]
|
|
|
|
|
|
|
|
timedTest "Trying to add a block twice tags the second as an error":
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-07-09 09:29:32 +00:00
|
|
|
let
|
|
|
|
b10 = makeTestBlock(state.data, blockPool[].tail.root, cache)
|
2020-07-16 13:16:51 +00:00
|
|
|
b10Add = blockpool[].addRawBlock(b10) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
|
|
|
|
|
|
|
let head = pool[].selectHead()
|
|
|
|
|
|
|
|
check:
|
|
|
|
head == b10Add[]
|
|
|
|
|
|
|
|
# -------------------------------------------------------------
|
|
|
|
# Add back the old block to ensure we have a duplicate error
|
|
|
|
let b10_clone = b10 # Assumes deep copy
|
2020-07-16 13:16:51 +00:00
|
|
|
let b10Add_clone = blockpool[].addRawBlock(b10_clone) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
|
|
|
doAssert: b10Add_clone.error == Duplicate
|
|
|
|
|
|
|
|
wrappedTimedTest "Trying to add a duplicate block from an old pruned epoch is tagged as an error":
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-07-09 09:29:32 +00:00
|
|
|
|
|
|
|
blockpool[].addFlags {skipBLSValidation}
|
|
|
|
pool.forkChoice_v2.proto_array.prune_threshold = 1
|
|
|
|
|
|
|
|
let
|
|
|
|
b10 = makeTestBlock(state.data, blockPool[].tail.root, cache)
|
2020-07-16 13:16:51 +00:00
|
|
|
b10Add = blockpool[].addRawBlock(b10) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
|
|
|
|
|
|
|
let head = pool[].selectHead()
|
|
|
|
|
|
|
|
doAssert: head == b10Add[]
|
|
|
|
|
2020-07-09 23:02:40 +00:00
|
|
|
let block_ok = state_transition(defaultRuntimePreset, state.data, b10, {}, noRollback)
|
2020-07-09 09:29:32 +00:00
|
|
|
doAssert: block_ok
|
|
|
|
|
|
|
|
# -------------------------------------------------------------
|
|
|
|
let b10_clone = b10 # Assumes deep copy
|
|
|
|
|
|
|
|
# -------------------------------------------------------------
|
|
|
|
# Pass an epoch
|
2020-07-16 13:16:51 +00:00
|
|
|
var block_root = b10.root
|
2020-07-09 09:29:32 +00:00
|
|
|
|
|
|
|
var attestations: seq[Attestation]
|
|
|
|
|
|
|
|
for epoch in 0 ..< 5:
|
|
|
|
let start_slot = compute_start_slot_at_epoch(Epoch epoch)
|
|
|
|
for slot in start_slot ..< start_slot + SLOTS_PER_EPOCH:
|
|
|
|
|
|
|
|
let new_block = makeTestBlock(state.data, block_root, cache, attestations = attestations)
|
2020-07-09 23:02:40 +00:00
|
|
|
let block_ok = state_transition(defaultRuntimePreset, state.data, new_block, {skipBLSValidation}, noRollback)
|
2020-07-09 09:29:32 +00:00
|
|
|
doAssert: block_ok
|
|
|
|
|
2020-07-16 13:16:51 +00:00
|
|
|
block_root = new_block.root
|
|
|
|
let blockRef = blockpool[].addRawBlock(new_block) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
|
|
|
|
|
|
|
let head = pool[].selectHead()
|
|
|
|
doassert: head == blockRef[]
|
|
|
|
blockPool[].updateHead(head)
|
|
|
|
|
|
|
|
attestations.setlen(0)
|
|
|
|
for index in 0 ..< get_committee_count_at_slot(state.data.data, slot.Slot):
|
|
|
|
let committee = get_beacon_committee(
|
|
|
|
state.data.data, state.data.data.slot, index.CommitteeIndex, cache)
|
|
|
|
|
|
|
|
# Create a bitfield filled with the given count per attestation,
|
|
|
|
# exactly on the right-most part of the committee field.
|
|
|
|
var aggregation_bits = init(CommitteeValidatorsBits, committee.len)
|
|
|
|
for v in 0 ..< committee.len * 2 div 3 + 1:
|
|
|
|
aggregation_bits[v] = true
|
|
|
|
|
|
|
|
attestations.add Attestation(
|
|
|
|
aggregation_bits: aggregation_bits,
|
|
|
|
data: makeAttestationData(
|
|
|
|
state.data.data, state.data.data.slot,
|
|
|
|
index, blockroot
|
|
|
|
)
|
|
|
|
# signature: ValidatorSig()
|
|
|
|
)
|
|
|
|
|
2020-07-15 10:44:18 +00:00
|
|
|
cache = StateCache()
|
2020-07-09 09:29:32 +00:00
|
|
|
|
|
|
|
# -------------------------------------------------------------
|
|
|
|
# Prune
|
|
|
|
|
|
|
|
echo "\nPruning all blocks before: ", shortlog(blockPool[].finalizedHead), '\n'
|
|
|
|
doAssert: blockPool[].finalizedHead.slot != 0
|
|
|
|
|
|
|
|
pool[].pruneBefore(blockPool[].finalizedHead)
|
2020-07-16 13:16:51 +00:00
|
|
|
doAssert: b10.root notin pool.forkChoice_v2
|
2020-07-09 09:29:32 +00:00
|
|
|
|
|
|
|
# Add back the old block to ensure we have a duplicate error
|
2020-07-16 13:16:51 +00:00
|
|
|
let b10Add_clone = blockpool[].addRawBlock(b10_clone) do (validBlock: BlockRef):
|
2020-07-09 09:29:32 +00:00
|
|
|
# Callback Add to fork choice
|
|
|
|
pool[].addForkChoice_v2(validBlock)
|
|
|
|
doAssert: b10Add_clone.error == Duplicate
|