2022-06-16 08:50:29 +02:00
|
|
|
# Nimbus
|
|
|
|
# Copyright (c) 2022 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: [Defect].}
|
|
|
|
|
|
|
|
import
|
|
|
|
unittest2, stint, stew/byteutils,
|
2022-09-03 20:15:35 +02:00
|
|
|
eth/common/eth_types_rlp,
|
2022-06-16 08:50:29 +02:00
|
|
|
../populate_db,
|
2022-08-24 16:41:04 +02:00
|
|
|
../network/history/[history_content, accumulator]
|
|
|
|
|
2022-08-24 22:12:56 +02:00
|
|
|
func buildProof(
|
2022-08-24 16:41:04 +02:00
|
|
|
accumulator: Accumulator,
|
|
|
|
epochAccumulators: seq[(ContentKey, EpochAccumulator)],
|
|
|
|
header: BlockHeader):
|
|
|
|
Result[seq[Digest], string] =
|
|
|
|
let
|
|
|
|
epochIndex = getEpochIndex(header)
|
|
|
|
epochAccumulator = epochAccumulators[epochIndex][1]
|
|
|
|
|
|
|
|
headerRecordIndex = getHeaderRecordIndex(header, epochIndex)
|
|
|
|
gIndex = GeneralizedIndex(epochSize*2*2 + (headerRecordIndex*2))
|
|
|
|
|
|
|
|
return epochAccumulator.build_proof(gIndex)
|
2022-06-16 08:50:29 +02:00
|
|
|
|
|
|
|
suite "Header Accumulator":
|
|
|
|
test "Header Accumulator Update":
|
|
|
|
const
|
|
|
|
hashTreeRoots = [
|
|
|
|
"b629833240bb2f5eabfb5245be63d730ca4ed30d6a418340ca476e7c1f1d98c0",
|
|
|
|
"00cbebed829e1babb93f2300bebe7905a98cb86993c7fc09bb5b04626fd91ae5",
|
|
|
|
"88cce8439ebc0c1d007177ffb6831c15c07b4361984cc52235b6fd728434f0c7"]
|
|
|
|
|
|
|
|
dataFile = "./fluffy/tests/blocks/mainnet_blocks_1-2.json"
|
|
|
|
|
2022-08-01 21:00:21 +02:00
|
|
|
let blockDataRes = readJsonType(dataFile, BlockDataTable)
|
2022-06-16 08:50:29 +02:00
|
|
|
|
|
|
|
check blockDataRes.isOk()
|
|
|
|
let blockData = blockDataRes.get()
|
|
|
|
|
|
|
|
var headers: seq[BlockHeader]
|
|
|
|
# Len of headers from blockdata + genesis header
|
|
|
|
headers.setLen(blockData.len() + 1)
|
|
|
|
|
|
|
|
headers[0] = getGenesisHeader()
|
|
|
|
|
|
|
|
for k, v in blockData.pairs:
|
|
|
|
let res = v.readBlockHeader()
|
|
|
|
check res.isOk()
|
|
|
|
let header = res.get()
|
|
|
|
headers[header.blockNumber.truncate(int)] = header
|
|
|
|
|
|
|
|
var accumulator: Accumulator
|
|
|
|
|
|
|
|
for i, hash in hashTreeRoots:
|
|
|
|
updateAccumulator(accumulator, headers[i])
|
|
|
|
|
|
|
|
check accumulator.hash_tree_root().data.toHex() == hashTreeRoots[i]
|
2022-06-23 21:00:59 +02:00
|
|
|
|
2022-09-09 13:12:09 +02:00
|
|
|
test "Header Accumulator Canonical Verification":
|
2022-06-23 21:00:59 +02:00
|
|
|
const
|
|
|
|
# Amount of headers to be created and added to the accumulator
|
|
|
|
amount = 25000
|
|
|
|
# Headers to test verification for
|
|
|
|
headersToTest = [
|
|
|
|
0,
|
|
|
|
epochSize - 1,
|
|
|
|
epochSize,
|
|
|
|
epochSize*2 - 1,
|
|
|
|
epochSize*2,
|
|
|
|
epochSize*3 - 1,
|
|
|
|
epochSize*3,
|
|
|
|
epochSize*3 + 1,
|
|
|
|
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(
|
|
|
|
blockNumber: i.stuint(256), difficulty: 1.stuint(256)))
|
|
|
|
|
2022-08-24 16:41:04 +02:00
|
|
|
let
|
|
|
|
accumulator = buildAccumulator(headers)
|
|
|
|
epochAccumulators = buildAccumulatorData(headers)
|
2022-06-23 21:00:59 +02:00
|
|
|
|
|
|
|
block: # Test valid headers
|
|
|
|
for i in headersToTest:
|
|
|
|
let header = headers[i]
|
|
|
|
let proofOpt =
|
|
|
|
if header.inCurrentEpoch(accumulator):
|
|
|
|
none(seq[Digest])
|
|
|
|
else:
|
2022-08-24 16:41:04 +02:00
|
|
|
let proof = buildProof(accumulator, epochAccumulators, header)
|
2022-06-23 21:00:59 +02:00
|
|
|
check proof.isOk()
|
|
|
|
|
|
|
|
some(proof.get())
|
|
|
|
|
2022-08-24 16:41:04 +02:00
|
|
|
check verifyHeader(accumulator, header, proofOpt).isOk()
|
2022-06-23 21:00:59 +02:00
|
|
|
|
|
|
|
block: # Test some invalid headers
|
|
|
|
# Test a header with block number > than latest in accumulator
|
|
|
|
let header = BlockHeader(blockNumber: 25000.stuint(256))
|
2022-08-24 16:41:04 +02:00
|
|
|
check verifyHeader(accumulator, header, none(seq[Digest])).isErr()
|
2022-06-23 21:00:59 +02:00
|
|
|
|
|
|
|
# Test different block headers by altering the difficulty
|
|
|
|
for i in headersToTest:
|
|
|
|
let header = BlockHeader(
|
|
|
|
blockNumber: i.stuint(256), difficulty: 2.stuint(256))
|
|
|
|
|
2022-08-24 16:41:04 +02:00
|
|
|
check verifyHeader(accumulator, header, none(seq[Digest])).isErr()
|
2022-08-04 08:34:53 +02:00
|
|
|
|
2022-09-09 13:12:09 +02:00
|
|
|
test "Header Accumulator Canonical Verification - No Historical Epochs":
|
|
|
|
const
|
|
|
|
# Amount of headers to be created and added to the accumulator
|
|
|
|
amount = epochSize
|
|
|
|
# Headers to test verification for
|
|
|
|
headersToTest = [
|
|
|
|
0,
|
|
|
|
epochSize - 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(
|
|
|
|
blockNumber: i.stuint(256), difficulty: 1.stuint(256)))
|
|
|
|
|
|
|
|
let
|
|
|
|
accumulator = buildAccumulator(headers)
|
|
|
|
epochAccumulators = buildAccumulatorData(headers)
|
|
|
|
|
|
|
|
block: # Test valid headers
|
|
|
|
for i in headersToTest:
|
|
|
|
let header = headers[i]
|
|
|
|
let proofOpt =
|
|
|
|
if header.inCurrentEpoch(accumulator):
|
|
|
|
none(seq[Digest])
|
|
|
|
else:
|
|
|
|
let proof = buildProof(accumulator, epochAccumulators, header)
|
|
|
|
check proof.isOk()
|
|
|
|
|
|
|
|
some(proof.get())
|
|
|
|
|
|
|
|
check verifyHeader(accumulator, header, proofOpt).isOk()
|
|
|
|
|
|
|
|
block: # Test some invalid headers
|
|
|
|
# Test a header with block number > than latest in accumulator
|
|
|
|
let header = BlockHeader(blockNumber: (amount).stuint(256))
|
|
|
|
check verifyHeader(accumulator, header, none(seq[Digest])).isErr()
|
|
|
|
|
|
|
|
# Test different block headers by altering the difficulty
|
|
|
|
for i in headersToTest:
|
|
|
|
let header = BlockHeader(
|
|
|
|
blockNumber: i.stuint(256), difficulty: 2.stuint(256))
|
|
|
|
|
|
|
|
check verifyHeader(accumulator, header, none(seq[Digest])).isErr()
|
|
|
|
|
|
|
|
test "Header Accumulator Blocknumber to Header Hash":
|
2022-08-24 22:12:56 +02:00
|
|
|
var acc = Accumulator.init()
|
2022-08-04 08:34:53 +02:00
|
|
|
|
2022-08-24 22:12:56 +02:00
|
|
|
let
|
|
|
|
numEpochs = 2
|
|
|
|
numHeadersInCurrentEpoch = 5
|
|
|
|
numHeaders = numEpochs * epochSize + numHeadersInCurrentEpoch
|
2022-08-04 08:34:53 +02:00
|
|
|
|
|
|
|
var headerHashes: seq[Hash256] = @[]
|
|
|
|
|
|
|
|
for i in 0..numHeaders:
|
|
|
|
var bh = BlockHeader()
|
|
|
|
bh.blockNumber = u256(i)
|
|
|
|
bh.difficulty = u256(1)
|
|
|
|
headerHashes.add(bh.blockHash())
|
|
|
|
acc.updateAccumulator(bh)
|
|
|
|
|
|
|
|
# get valid response for epoch 0
|
|
|
|
block:
|
|
|
|
for i in 0..epochSize-1:
|
|
|
|
let res = acc.getHeaderHashForBlockNumber(u256(i))
|
|
|
|
check:
|
|
|
|
res.kind == HEpoch
|
|
|
|
res.epochIndex == 0
|
|
|
|
|
|
|
|
# get valid response for epoch 1
|
|
|
|
block:
|
|
|
|
for i in epochSize..(2 * epochSize)-1:
|
|
|
|
let res = acc.getHeaderHashForBlockNumber(u256(i))
|
|
|
|
check:
|
|
|
|
res.kind == HEpoch
|
|
|
|
res.epochIndex == 1
|
|
|
|
|
|
|
|
# get valid response from current epoch (epoch 3)
|
|
|
|
block:
|
|
|
|
for i in (2 * epochSize)..(2 * epochSize) + numHeadersInCurrentEpoch:
|
|
|
|
let res = acc.getHeaderHashForBlockNumber(u256(i))
|
|
|
|
check:
|
|
|
|
res.kind == BHash
|
|
|
|
res.blockHash == headerHashes[i]
|
|
|
|
|
|
|
|
# get valid response when getting unknown hash
|
|
|
|
block:
|
|
|
|
let firstUknownBlockNumber = (2 * epochSize) + numHeadersInCurrentEpoch + 1
|
|
|
|
let res = acc.getHeaderHashForBlockNumber(u256(firstUknownBlockNumber))
|
|
|
|
|
|
|
|
check:
|
|
|
|
res.kind == UnknownBlockNumber
|
|
|
|
|
|
|
|
let res1 = acc.getHeaderHashForBlockNumber(u256(3 * epochSize))
|
|
|
|
check:
|
|
|
|
res1.kind == UnknownBlockNumber
|