nimbus-eth1/fluffy/tests/test_accumulator.nim

140 lines
4.7 KiB
Nim

# Nimbus
# Copyright (c) 2022-2024 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.
{.used.}
{.push raises: [].}
import
unittest2,
stint,
eth/common/eth_types_rlp,
../eth_data/history_data_json_store,
../network/history/[history_content, accumulator],
./test_helpers
suite "Header Accumulator":
test "Header Accumulator Canonical Verification":
const
# Amount of headers to be created and added to the accumulator
amount = mergeBlockNumber
# Headers to test verification for.
# Note: This test assumes at least 5 epochs
headersToTest = [
0,
epochSize - 1,
epochSize,
epochSize * 2 - 1,
epochSize * 2,
epochSize * 3 - 1,
epochSize * 3,
epochSize * 3 + 1,
int(amount) - 1,
]
var headers: seq[BlockHeader]
for i in 0 ..< amount:
# Note: These test headers will not be a blockchain, as the parent hashes
# are not properly filled in. That's fine however for this test, as that
# is not the way the headers are verified with the accumulator.
headers.add(BlockHeader(number: i, difficulty: 1.stuint(256)))
let accumulatorRes = buildAccumulatorData(headers)
check accumulatorRes.isOk()
let (accumulator, epochAccumulators) = accumulatorRes.get()
block: # Test valid headers
for i in headersToTest:
let header = headers[i]
let proof = buildProof(header, epochAccumulators)
check:
proof.isOk()
verifyAccumulatorProof(accumulator, header, proof.get()).isOk()
block: # Test invalid headers
# Post merge block number must fail (> than latest header in accumulator)
var proof: AccumulatorProof
let header = BlockHeader(number: mergeBlockNumber)
check verifyAccumulatorProof(accumulator, header, proof).isErr()
# Test altered block headers by altering the difficulty
for i in headersToTest:
let proof = buildProof(headers[i], epochAccumulators)
check:
proof.isOk()
# Alter the block header so the proof no longer matches
let header = BlockHeader(number: i.uint64, difficulty: 2.stuint(256))
check verifyAccumulatorProof(accumulator, header, proof.get()).isErr()
block: # Test invalid proofs
var proof: AccumulatorProof
for i in headersToTest:
check verifyAccumulatorProof(accumulator, headers[i], proof).isErr()
test "Header Accumulator - Not Finished":
# Less headers than needed to finish the accumulator
const amount = mergeBlockNumber - 1
var headers: seq[BlockHeader]
for i in 0 ..< amount:
headers.add(BlockHeader(number: i, difficulty: 1.stuint(256)))
let accumulatorRes = buildAccumulator(headers)
check accumulatorRes.isErr()
test "Header BlockNumber to EpochAccumulator Root":
# Note: This test assumes at least 3 epochs
const amount = mergeBlockNumber
var
headerHashes: seq[Hash256] = @[]
headers: seq[BlockHeader]
for i in 0 ..< amount:
let header = BlockHeader(number: i, difficulty: u256(1))
headers.add(header)
headerHashes.add(header.blockHash())
let accumulatorRes = buildAccumulator(headers)
check accumulatorRes.isOk()
let accumulator = accumulatorRes.get()
# Valid response for block numbers in epoch 0
block:
for i in 0 ..< epochSize:
let res = accumulator.getBlockEpochDataForBlockNumber(u256(i))
check:
res.isOk()
res.get().epochHash == accumulator.historicalEpochs[0]
# Valid response for block numbers in epoch 1
block:
for i in epochSize ..< (2 * epochSize):
let res = accumulator.getBlockEpochDataForBlockNumber(u256(i))
check:
res.isOk()
res.get().epochHash == accumulator.historicalEpochs[1]
# Valid response for block numbers in the incomplete (= last) epoch
block:
const startIndex = mergeBlockNumber - (mergeBlockNumber mod epochSize)
for i in startIndex ..< mergeBlockNumber:
let res = accumulator.getBlockEpochDataForBlockNumber(u256(i))
check:
res.isOk()
res.get().epochHash == accumulator.historicalEpochs[preMergeEpochs - 1]
# Error for block number at and past merge
block:
check:
accumulator.getBlockEpochDataForBlockNumber(u256(mergeBlockNumber)).isErr()
accumulator.getBlockEpochDataForBlockNumber(u256(mergeBlockNumber + 1)).isErr()