Move BlockHeaderWithProof content to content key selector 0 (#1379)

* Move BlockHeaderWithProof content to content key selector 0

- Remove as content type with content key selector 4
- Replace regular block header with BlockHeaderWithProof at
content key selector 0

* Apply blockHeader content key also to bridge

* Add tests for header with proof generation and verification
This commit is contained in:
Kim De Mey 2023-01-20 23:04:58 +01:00 committed by GitHub
parent 6fb48517ba
commit 12f66ae598
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 166 additions and 114 deletions

View File

@ -5,7 +5,10 @@
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0). # * 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. # at your option. This file may not be copied, modified, or distributed except according to those terms.
{.push raises: [Defect].} when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import import
json_serialization, json_serialization/std/tables, json_serialization, json_serialization/std/tables,
@ -33,14 +36,16 @@ type
BlockDataTable* = Table[string, BlockData] BlockDataTable* = Table[string, BlockData]
proc toString(v: IoErrorCode): string =
try: ioErrorMsg(v)
except Exception as e: raiseAssert e.msg
proc readJsonType*(dataFile: string, T: type): Result[T, string] = proc readJsonType*(dataFile: string, T: type): Result[T, string] =
let data = readAllFile(dataFile) let data = ? readAllFile(dataFile).mapErr(toString)
if data.isErr(): # TODO: map errors
return err("Failed reading data-file")
let decoded = let decoded =
try: try:
Json.decode(data.get(), T) Json.decode(data, T)
except SerializationError as e: except SerializationError as e:
return err("Failed decoding json data-file: " & e.msg) return err("Failed decoding json data-file: " & e.msg)

View File

@ -181,8 +181,8 @@ proc historyPropagateHeadersWithProof*(
let let
content = headerWithProof.get() content = headerWithProof.get()
contentKey = ContentKey( contentKey = ContentKey(
contentType: blockHeaderWithProof, contentType: blockHeader,
blockHeaderWithProofKey: BlockKey(blockHash: header.blockHash())) blockHeaderKey: BlockKey(blockHash: header.blockHash()))
encKey = history_content.encode(contentKey) encKey = history_content.encode(contentKey)
contentId = history_content.toContentId(encKey) contentId = history_content.toContentId(encKey)
encodedContent = SSZ.encode(content) encodedContent = SSZ.encode(content)

View File

@ -32,7 +32,6 @@ type
blockBody = 0x01 blockBody = 0x01
receipts = 0x02 receipts = 0x02
epochAccumulator = 0x03 epochAccumulator = 0x03
blockHeaderWithProof = 0x04
BlockKey* = object BlockKey* = object
blockHash*: BlockHash blockHash*: BlockHash
@ -50,8 +49,6 @@ type
receiptsKey*: BlockKey receiptsKey*: BlockKey
of epochAccumulator: of epochAccumulator:
epochAccumulatorKey*: EpochAccumulatorKey epochAccumulatorKey*: EpochAccumulatorKey
of blockHeaderWithProof:
blockHeaderWithProofKey*: BlockKey
func init*( func init*(
T: type ContentKey, contentType: ContentType, T: type ContentKey, contentType: ContentType,
@ -70,10 +67,6 @@ func init*(
ContentKey( ContentKey(
contentType: contentType, contentType: contentType,
epochAccumulatorKey: EpochAccumulatorKey(epochHash: hash)) epochAccumulatorKey: EpochAccumulatorKey(epochHash: hash))
of blockHeaderWithProof:
ContentKey(
contentType: contentType,
blockHeaderWithProofKey: BlockKey(blockHash: hash))
func encode*(contentKey: ContentKey): ByteList = func encode*(contentKey: ContentKey): ByteList =
ByteList.init(SSZ.encode(contentKey)) ByteList.init(SSZ.encode(contentKey))
@ -111,8 +104,6 @@ func `$`*(x: ContentKey): string =
of epochAccumulator: of epochAccumulator:
let key = x.epochAccumulatorKey let key = x.epochAccumulatorKey
res.add("epochHash: " & $key.epochHash) res.add("epochHash: " & $key.epochHash)
of blockHeaderWithProof:
res.add($x.blockHeaderWithProofKey)
res.add(")") res.add(")")

View File

@ -260,7 +260,7 @@ proc getVerifiedBlockHeader*(
n: HistoryNetwork, hash: BlockHash): n: HistoryNetwork, hash: BlockHash):
Future[Opt[BlockHeader]] {.async.} = Future[Opt[BlockHeader]] {.async.} =
let let
contentKey = ContentKey.init(blockHeaderWithProof, hash).encode() contentKey = ContentKey.init(blockHeader, hash).encode()
contentId = contentKey.toContentId() contentId = contentKey.toContentId()
logScope: logScope:
@ -310,49 +310,6 @@ proc getVerifiedBlockHeader*(
# Headers were requested `requestRetries` times and all failed on validation # Headers were requested `requestRetries` times and all failed on validation
return Opt.none(BlockHeader) return Opt.none(BlockHeader)
# TODO: To be deprecated or not? Should there be the case for requesting a
# block header without proofs?
proc getBlockHeader*(
n: HistoryNetwork, hash: BlockHash):
Future[Opt[BlockHeader]] {.async.} =
let
contentKey = ContentKey.init(blockHeader, hash).encode()
contentId = contentKey.toContentId()
logScope:
hash
contentKey
let headerFromDb = n.getContentFromDb(BlockHeader, contentId)
if headerFromDb.isSome():
info "Fetched block header from database"
return headerFromDb
for i in 0..<requestRetries:
let
headerContent = (await n.portalProtocol.contentLookup(
contentKey, contentId)).valueOr:
warn "Failed fetching block header from the network"
return Opt.none(BlockHeader)
header = validateBlockHeaderBytes(headerContent.content, hash).valueOr:
warn "Validation of block header failed", error
continue
info "Fetched valid block header from the network"
# Content is valid, it can be stored and propagated to interested peers
n.portalProtocol.storeContent(contentKey, contentId, headerContent.content)
n.portalProtocol.triggerPoke(
headerContent.nodesInterestedInContent,
contentKey,
headerContent.content
)
return Opt.some(header)
# Headers were requested `requestRetries` times and all failed on validation
return Opt.none(BlockHeader)
proc getBlockBody*( proc getBlockBody*(
n: HistoryNetwork, hash: BlockHash, header: BlockHeader): n: HistoryNetwork, hash: BlockHash, header: BlockHeader):
Future[Opt[BlockBody]] {.async.} = Future[Opt[BlockBody]] {.async.} =
@ -528,19 +485,20 @@ proc validateContent(
case key.contentType: case key.contentType:
of blockHeader: of blockHeader:
# Note: For now we still accept regular block header type to remain let
# compatible with the current specs. However, a verification is done by headerWithProof = decodeSsz(content, BlockHeaderWithProof).valueOr:
# basically requesting the header with proofs from somewhere else. warn "Failed decoding header with proof", error
# This all doesn't make much sense aside from compatibility and should return false
# eventually be removed. header = validateBlockHeaderBytes(
let header = validateBlockHeaderBytes( headerWithProof.header.asSeq(),
content, key.blockHeaderKey.blockHash).valueOr: key.blockHeaderKey.blockHash).valueOr:
warn "Invalid block header offered", error warn "Invalid block header offered", error
return false return false
let res = await n.getVerifiedBlockHeader(key.blockHeaderKey.blockHash) let res = n.verifyHeader(header, headerWithProof.proof)
if res.isNone(): if res.isErr():
warn "Block header failed canonical verification" warn "Failed on check if header is part of canonical chain",
error = res.error
return false return false
else: else:
return true return true
@ -594,25 +552,6 @@ proc validateContent(
else: else:
return true return true
of blockHeaderWithProof:
let
headerWithProof = decodeSsz(content, BlockHeaderWithProof).valueOr:
warn "Failed decoding header with proof", error
return false
header = validateBlockHeaderBytes(
headerWithProof.header.asSeq(),
key.blockHeaderWithProofKey.blockHash).valueOr:
warn "Invalid block header offered", error
return false
let res = n.verifyHeader(header, headerWithProof.proof)
if res.isErr():
warn "Failed on check if header is part of canonical chain",
error = res.error
return false
else:
return true
proc new*( proc new*(
T: type HistoryNetwork, T: type HistoryNetwork,
baseProtocol: protocol.Protocol, baseProtocol: protocol.Protocol,

View File

@ -19,7 +19,8 @@ import
history_data_json_store, history_data_json_store,
history_data_ssz_e2s], history_data_ssz_e2s],
../network/history/[history_content, accumulator], ../network/history/[history_content, accumulator],
../seed_db ../seed_db,
../tests/test_history_util
type type
FutureCallback[A] = proc (): Future[A] {.gcsafe, raises: [Defect].} FutureCallback[A] = proc (): Future[A] {.gcsafe, raises: [Defect].}
@ -42,24 +43,6 @@ type
desc: "Port of the JSON-RPC service of the bootstrap (first) node" desc: "Port of the JSON-RPC service of the bootstrap (first) node"
name: "base-rpc-port" .}: uint16 name: "base-rpc-port" .}: uint16
proc buildHeadersWithProof*(
blockHeaders: seq[BlockHeader],
epochAccumulator: EpochAccumulatorCached):
Result[seq[(seq[byte], seq[byte])], string] =
var blockHeadersWithProof: seq[(seq[byte], seq[byte])]
for header in blockHeaders:
if header.isPreMerge():
let
content = ? buildHeaderWithProof(header, epochAccumulator)
contentKey = ContentKey(
contentType: blockHeaderWithProof,
blockHeaderWithProofKey: BlockKey(blockHash: header.blockHash()))
blockHeadersWithProof.add(
(encode(contentKey).asSeq(), SSZ.encode(content)))
ok(blockHeadersWithProof)
proc connectToRpcServers(config: PortalTestnetConf): proc connectToRpcServers(config: PortalTestnetConf):
Future[seq[RpcClient]] {.async.} = Future[seq[RpcClient]] {.async.} =
var clients: seq[RpcClient] var clients: seq[RpcClient]

View File

@ -10,6 +10,7 @@
import import
./test_portal_wire_encoding, ./test_portal_wire_encoding,
./test_history_content, ./test_history_content,
./test_headers_with_proof,
./test_header_content, ./test_header_content,
./test_state_content, ./test_state_content,
./test_accumulator_root ./test_accumulator_root

View File

@ -0,0 +1,97 @@
# Nimbus
# Copyright (c) 2023 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.}
when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import
unittest2, stew/byteutils,
eth/common/eth_types_rlp,
../../../network_metadata,
../../../eth_data/[history_data_json_store, history_data_ssz_e2s],
../../../network/history/[history_content, history_network, accumulator],
../../test_history_util
type
JsonHeaderContent* = object
content_key*: string
content_value*: string
JsonHeaderContentTable* = Table[string, JsonHeaderContent]
suite "Headers with Proof":
test "HeaderWithProof Decoding and Verifying":
const dataFile =
"./vendor/portal-spec-tests/tests/mainnet/history/headers_with_proof/1000001-1000010.json"
let accumulator =
try:
SSZ.decode(finishedAccumulator, FinishedAccumulator)
except SszError as err:
raiseAssert "Invalid baked-in accumulator: " & err.msg
let res = readJsonType(dataFile, JsonHeaderContentTable)
check res.isOk()
let headerContent = res.get()
for k, v in headerContent:
let
# TODO: strange assignment failure when using try/except ValueError
# for the hexToSeqByte() here.
contentKey = decodeSsz(
v.content_key.hexToSeqByte(), ContentKey)
contentValue = decodeSsz(
v.content_value.hexToSeqByte(), BlockHeaderWithProof)
check:
contentKey.isOk()
contentValue.isOk()
let blockHeaderWithProof = contentValue.get()
let res = decodeRlp(blockHeaderWithProof.header.asSeq(), BlockHeader)
check res.isOk()
let header = res.get()
check accumulator.verifyHeader(header, blockHeaderWithProof.proof).isOk()
test "HeaderWithProof Building and Encoding":
const
headerFile = "./vendor/portal-spec-tests/tests/mainnet/history/headers/1000001-1000010.e2s"
accumulatorFile = "./vendor/portal-spec-tests/tests/mainnet/history/accumulator/epoch-accumulator-00122.ssz"
headersWithProofFile = "./vendor/portal-spec-tests/tests/mainnet/history/headers_with_proof/1000001-1000010.json"
let
blockHeaders = readBlockHeaders(headerFile).valueOr:
raiseAssert "Invalid header file: " & headerFile
epochAccumulator = readEpochAccumulatorCached(accumulatorFile).valueOr:
raiseAssert "Invalid epoch accumulator file: " & accumulatorFile
blockHeadersWithProof =
buildHeadersWithProof(blockHeaders, epochAccumulator).valueOr:
raiseAssert "Could not build headers with proof"
let res = readJsonType(headersWithProofFile, JsonHeaderContentTable)
check res.isOk()
let headerContent = res.get()
# Go over all content keys and headers with generated proofs and compare
# them with the ones from the test vectors.
for i, (headerContentKey, headerWithProof) in blockHeadersWithProof:
let
blockNumber = blockHeaders[i].blockNumber
contentKey =
headerContent[blockNumber.toString()].content_key.hexToSeqByte()
contentValue =
headerContent[blockNumber.toString()].content_value.hexToSeqByte()
check:
contentKey == headerContentKey
contentValue == headerWithProof

View File

@ -70,7 +70,7 @@ proc headersToContentInfo(
headerHash = header.blockHash() headerHash = header.blockHash()
blockKey = BlockKey(blockHash: headerHash) blockKey = BlockKey(blockHash: headerHash)
contentKey = encode(ContentKey( contentKey = encode(ContentKey(
contentType: blockHeaderWithProof, blockHeaderWithProofKey: blockKey)) contentType: blockHeader, blockHeaderKey: blockKey))
contentInfo = ContentInfo( contentInfo = ContentInfo(
contentKey: contentKey, content: SSZ.encode(headerWithProof)) contentKey: contentKey, content: SSZ.encode(headerWithProof))
contentInfos.add(contentInfo) contentInfos.add(contentInfo)
@ -119,7 +119,7 @@ procSuite "History Content Network":
headerHash = header.blockHash() headerHash = header.blockHash()
blockKey = BlockKey(blockHash: headerHash) blockKey = BlockKey(blockHash: headerHash)
contentKey = ContentKey( contentKey = ContentKey(
contentType: blockHeaderWithProof, blockHeaderWithProofKey: blockKey) contentType: blockHeader, blockHeaderKey: blockKey)
encKey = encode(contentKey) encKey = encode(contentKey)
contentId = toContentId(contentKey) contentId = toContentId(contentKey)
historyNode2.portalProtocol().storeContent( historyNode2.portalProtocol().storeContent(

View File

@ -0,0 +1,36 @@
# Nimbus
# Copyright (c) 2023 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.
when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import
stew/results,
eth/common/eth_types_rlp,
../network/history/[history_content, accumulator]
export results, accumulator, history_content
proc buildHeadersWithProof*(
blockHeaders: seq[BlockHeader],
epochAccumulator: EpochAccumulatorCached):
Result[seq[(seq[byte], seq[byte])], string] =
var blockHeadersWithProof: seq[(seq[byte], seq[byte])]
for header in blockHeaders:
if header.isPreMerge():
let
content = ? buildHeaderWithProof(header, epochAccumulator)
contentKey = ContentKey(
contentType: blockHeader,
blockHeaderKey: BlockKey(blockHash: header.blockHash()))
blockHeadersWithProof.add(
(encode(contentKey).asSeq(), SSZ.encode(content)))
ok(blockHeadersWithProof)

View File

@ -231,7 +231,7 @@ proc run() {.raises: [Exception, Defect].} =
blockhash = history_content.`$`hash blockhash = history_content.`$`hash
block: # gossip header block: # gossip header
let contentKey = ContentKey.init(blockHeaderWithProof, hash) let contentKey = ContentKey.init(blockHeader, hash)
let encodedContentKey = contentKey.encode.asSeq() let encodedContentKey = contentKey.encode.asSeq()
try: try:

@ -1 +1 @@
Subproject commit 5a1d2e553d97c04339b6227624d4ebab4da88701 Subproject commit ee863fecdc6ec9cc81effe355e74459cd5dda28d