mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-18 16:21:12 +00:00
fdb048ed21
* Add eth_getLogs json-rpc endpoint in Fluffy and Nimbus
374 lines
11 KiB
Nim
374 lines
11 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2021-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.
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
import
|
|
std/options,
|
|
stew/results, chronos, chronicles,
|
|
eth/[common/eth_types, rlp],
|
|
eth/p2p/discoveryv5/[protocol, enr],
|
|
../../content_db,
|
|
../../../nimbus/[utils, constants],
|
|
../wire/[portal_protocol, portal_stream, portal_protocol_config],
|
|
./history_content
|
|
|
|
logScope:
|
|
topics = "portal_hist"
|
|
|
|
const
|
|
historyProtocolId* = [byte 0x50, 0x0B]
|
|
|
|
# TODO: Extract common parts from the different networks
|
|
type
|
|
HistoryNetwork* = ref object
|
|
portalProtocol*: PortalProtocol
|
|
contentDB*: ContentDB
|
|
|
|
Block* = (BlockHeader, BlockBody)
|
|
|
|
func setStreamTransport*(n: HistoryNetwork, transport: UtpDiscv5Protocol) =
|
|
setTransport(n.portalProtocol.stream, transport)
|
|
|
|
proc toContentIdHandler(contentKey: ByteList): Option[ContentId] =
|
|
some(toContentId(contentKey))
|
|
|
|
func encodeKey(k: ContentKey): (ByteList, ContentId) =
|
|
let keyEncoded = encode(k)
|
|
return (keyEncoded, toContentId(keyEncoded))
|
|
|
|
func getEncodedKeyForContent(
|
|
cType: ContentType, chainId: uint16, hash: BlockHash):
|
|
(ByteList, ContentId) =
|
|
let contentKeyType = BlockKey(chainId: chainId, blockHash: hash)
|
|
|
|
let contentKey =
|
|
case cType
|
|
of blockHeader:
|
|
ContentKey(contentType: cType, blockHeaderKey: contentKeyType)
|
|
of blockBody:
|
|
ContentKey(contentType: cType, blockBodyKey: contentKeyType)
|
|
of receipts:
|
|
ContentKey(contentType: cType, receiptsKey: contentKeyType)
|
|
of epochAccumulator:
|
|
raiseAssert("Not implemented")
|
|
of masterAccumulator:
|
|
raiseAssert("Not implemented")
|
|
|
|
return encodeKey(contentKey)
|
|
|
|
proc getContentFromBytes(bytes: openArray[byte], T: type): Result[T, string] =
|
|
var rlp = rlpFromBytes(bytes)
|
|
try:
|
|
let content = rlp.read(T)
|
|
ok[T](content)
|
|
except RlpError as e:
|
|
err(e.msg)
|
|
|
|
proc validateHeaderBytes*(
|
|
bytes: openArray[byte], hash: BlockHash): Option[BlockHeader] =
|
|
|
|
let headerResult = getContentFromBytes(bytes, BlockHeader)
|
|
|
|
if headerResult.isErr():
|
|
error "Failed to decode header ", msg = headerResult.error()
|
|
return none(BlockHeader)
|
|
|
|
let header = headerResult.unsafeGet()
|
|
|
|
if not (header.blockHash() == hash):
|
|
# TODO: Header with different hash than expected, maybe we should punish
|
|
# peer which sent us this ?
|
|
return none(BlockHeader)
|
|
|
|
return some(header)
|
|
|
|
proc validateExpectedBody(
|
|
bb: BlockBody,
|
|
txRoot: KeccakHash,
|
|
ommersHash: KeccakHash): Result[void, string] =
|
|
try:
|
|
let calculatedTxRoot = calcTxRoot(bb.transactions)
|
|
let calculatedOmmersHash = rlpHash(bb.uncles)
|
|
|
|
if calculatedTxRoot != txRoot:
|
|
return err("Unexpected transaction root")
|
|
elif calculatedOmmersHash != ommersHash:
|
|
return err("Unexpected ommers hash")
|
|
else:
|
|
return ok()
|
|
except RlpError as e:
|
|
return err(e.msg)
|
|
|
|
proc validateBodyBytes*(
|
|
bytes: openArray[byte],
|
|
txRoot: KeccakHash,
|
|
ommersHash: KeccakHash):Option[BlockBody] =
|
|
|
|
let bodyResult = getContentFromBytes(bytes, BlockBody)
|
|
|
|
if bodyResult.isErr():
|
|
error "Failed to decode block body", msg = bodyResult.error()
|
|
return none(BlockBody)
|
|
|
|
let blockBody = bodyResult.unsafeGet()
|
|
|
|
let expectedResult = validateExpectedBody(blockBody, txRoot, ommersHash)
|
|
|
|
if expectedResult.isErr():
|
|
error "Failed to validate if block body matches header",
|
|
msg = expectedResult.error()
|
|
|
|
# we got block body (bundle of transactions and uncles) which do not match
|
|
# header. For now just ignore it, but maybe we should penalize peer
|
|
# sending us such data?
|
|
return none(BlockBody)
|
|
|
|
return some(blockBody)
|
|
|
|
proc getContentFromDb(
|
|
h: HistoryNetwork, T: type, contentId: ContentId): Option[T] =
|
|
if h.portalProtocol.inRange(contentId):
|
|
let contentFromDB = h.contentDB.get(contentId)
|
|
if contentFromDB.isSome():
|
|
var rlp = rlpFromBytes(contentFromDB.unsafeGet())
|
|
try:
|
|
let content = rlp.read(T)
|
|
return some(content)
|
|
except CatchableError as e:
|
|
# Content in db should always have valid formatting, so this should not
|
|
# happen
|
|
raiseAssert(e.msg)
|
|
else:
|
|
return none(T)
|
|
else:
|
|
return none(T)
|
|
|
|
proc getBlockHeader*(
|
|
h: HistoryNetwork, chainId: uint16, hash: BlockHash):
|
|
Future[Option[BlockHeader]] {.async.} =
|
|
let (keyEncoded, contentId) = getEncodedKeyForContent(blockHeader, chainId, hash)
|
|
|
|
let maybeHeaderFromDb = h.getContentFromDb(BlockHeader, contentId)
|
|
|
|
if maybeHeaderFromDb.isSome():
|
|
info "Fetched block header from database", hash
|
|
return maybeHeaderFromDb
|
|
|
|
let maybeHeaderContent = await h.portalProtocol.contentLookup(keyEncoded, contentId)
|
|
|
|
if maybeHeaderContent.isNone():
|
|
warn "Failed fetching block header from the network", hash
|
|
return none(BlockHeader)
|
|
|
|
let headerContent = maybeHeaderContent.unsafeGet()
|
|
|
|
let maybeHeader = validateHeaderBytes(headerContent.content, hash)
|
|
|
|
if maybeHeader.isSome():
|
|
info "Fetched block header from the network", hash
|
|
# Content is valid we can propagate it to interested peers
|
|
h.portalProtocol.triggerPoke(
|
|
headerContent.nodesInterestedInContent,
|
|
keyEncoded,
|
|
headerContent.content
|
|
)
|
|
|
|
h.portalProtocol.storeContent(contentId, headerContent.content)
|
|
|
|
return maybeHeader
|
|
|
|
proc getBlockBody*(
|
|
h: HistoryNetwork,
|
|
chainId: uint16,
|
|
hash: BlockHash,
|
|
header: BlockHeader):Future[Option[BlockBody]] {.async.} =
|
|
|
|
let (keyEncoded, contentId) = getEncodedKeyForContent(blockBody, chainId, hash)
|
|
|
|
let maybeBodyFromDb = h.getContentFromDb(BlockBody, contentId)
|
|
|
|
if maybeBodyFromDb.isSome():
|
|
info "Fetched block body from database", hash
|
|
return some[BlockBody](maybeBodyFromDb.unsafeGet())
|
|
|
|
let maybeBodyContent = await h.portalProtocol.contentLookup(keyEncoded, contentId)
|
|
|
|
if maybeBodyContent.isNone():
|
|
warn "Failed fetching block body from the network", hash
|
|
return none(BlockBody)
|
|
|
|
let bodyContent = maybeBodyContent.unsafeGet()
|
|
|
|
let maybeBody = validateBodyBytes(bodyContent.content, header.txRoot, header.ommersHash)
|
|
|
|
if maybeBody.isNone():
|
|
return none(BlockBody)
|
|
|
|
info "Fetched block body from the network", hash
|
|
|
|
let blockBody = maybeBody.unsafeGet()
|
|
|
|
# body is valid, propagate it to interested peers
|
|
h.portalProtocol.triggerPoke(
|
|
bodyContent.nodesInterestedInContent,
|
|
keyEncoded,
|
|
bodyContent.content
|
|
)
|
|
|
|
h.portalProtocol.storeContent(contentId, bodyContent.content)
|
|
|
|
return some(blockBody)
|
|
|
|
proc getBlock*(
|
|
h: HistoryNetwork, chainId: uint16, hash: BlockHash):
|
|
Future[Option[Block]] {.async.} =
|
|
let maybeHeader = await h.getBlockHeader(chainId, hash)
|
|
|
|
if maybeHeader.isNone():
|
|
# we do not have header for given hash,so we would not be able to validate
|
|
# that received body really belong to it
|
|
return none(Block)
|
|
|
|
let header = maybeHeader.unsafeGet()
|
|
|
|
let maybeBody = await h.getBlockBody(chainId, hash, header)
|
|
|
|
if maybeBody.isNone():
|
|
return none(Block)
|
|
|
|
let body = maybeBody.unsafeGet()
|
|
|
|
return some[Block]((header, body))
|
|
|
|
proc validateExpectedReceipts(
|
|
receipts: seq[Receipt],
|
|
receiptRoot: KeccakHash): Result[void, string] =
|
|
try:
|
|
let calculatedReceiptRoot = calcReceiptRoot(receipts)
|
|
|
|
if calculatedReceiptRoot != receiptRoot:
|
|
return err("Unexpected receipt root")
|
|
else:
|
|
return ok()
|
|
except RlpError as e:
|
|
return err(e.msg)
|
|
|
|
proc validateReceiptsBytes*(
|
|
bytes: openArray[byte],
|
|
receiptRoot: KeccakHash): Option[seq[Receipt]] =
|
|
|
|
let receiptResult = getContentFromBytes(bytes, seq[Receipt])
|
|
|
|
if receiptResult.isErr():
|
|
error "Failed to decode receipts", msg = receiptResult.error()
|
|
return none(seq[Receipt])
|
|
|
|
let receipts = receiptResult.unsafeGet()
|
|
|
|
let expectedReceiptsResult = validateExpectedReceipts(receipts, receiptRoot)
|
|
|
|
if expectedReceiptsResult.isErr():
|
|
error "Failed to validate if receipts matches header",
|
|
msg = expectedReceiptsResult.error()
|
|
|
|
# we got receipts which do not match
|
|
# header. For now just ignore it, but maybe we should penalize peer
|
|
# sending us such data?
|
|
return none(seq[Receipt])
|
|
|
|
return some(receipts)
|
|
|
|
proc getReceipts*(
|
|
h: HistoryNetwork,
|
|
chainId: uint16,
|
|
hash: BlockHash,
|
|
header: BlockHeader): Future[Option[seq[Receipt]]] {.async.} =
|
|
# header does not have any receipts, return early and do not save empty bytes
|
|
# into the database
|
|
if header.receiptRoot == BLANK_ROOT_HASH:
|
|
return some(newSeq[Receipt]())
|
|
|
|
let (keyEncoded, contentId) = getEncodedKeyForContent(receipts, chainId, hash)
|
|
|
|
let maybeReceiptsFromDb = h.getContentFromDb(seq[Receipt], contentId)
|
|
|
|
if maybeReceiptsFromDb.isSome():
|
|
info "Fetched receipts from database", hash
|
|
return some(maybeReceiptsFromDb.unsafeGet())
|
|
|
|
let maybeReceiptsContent = await h.portalProtocol.contentLookup(keyEncoded, contentId)
|
|
|
|
if maybeReceiptsContent.isNone():
|
|
warn "Failed fetching receipts from the network", hash
|
|
return none[seq[Receipt]]()
|
|
|
|
let receiptsContent = maybeReceiptsContent.unsafeGet()
|
|
|
|
let maybeReceipts = validateReceiptsBytes(receiptsContent.content, header.receiptRoot)
|
|
|
|
if maybeReceipts.isNone():
|
|
return none[seq[Receipt]]()
|
|
|
|
info "Fetched receipts from the network", hash
|
|
|
|
let receipts = maybeReceipts.unsafeGet()
|
|
|
|
# receips are valid, propagate it to interested peers
|
|
h.portalProtocol.triggerPoke(
|
|
receiptsContent.nodesInterestedInContent,
|
|
keyEncoded,
|
|
receiptsContent.content
|
|
)
|
|
|
|
h.portalProtocol.storeContent(contentId, receiptsContent.content)
|
|
|
|
return some(receipts)
|
|
|
|
proc validateContent(content: openArray[byte], contentKey: ByteList): bool =
|
|
let keyOpt = contentKey.decode()
|
|
|
|
if keyOpt.isNone():
|
|
return false
|
|
|
|
let key = keyOpt.get()
|
|
|
|
case key.contentType:
|
|
of blockHeader:
|
|
validateHeaderBytes(content, key.blockHeaderKey.blockHash).isSome()
|
|
of blockBody:
|
|
true
|
|
# TODO: Need to get the header from the db or the network for this. Or how
|
|
# to deal with this?
|
|
of receipts:
|
|
true
|
|
of epochAccumulator:
|
|
true
|
|
of masterAccumulator:
|
|
true
|
|
|
|
proc new*(
|
|
T: type HistoryNetwork,
|
|
baseProtocol: protocol.Protocol,
|
|
contentDB: ContentDB,
|
|
bootstrapRecords: openArray[Record] = [],
|
|
portalConfig: PortalProtocolConfig = defaultPortalProtocolConfig): T =
|
|
let portalProtocol = PortalProtocol.new(
|
|
baseProtocol, historyProtocolId, contentDB,
|
|
toContentIdHandler, validateContent, bootstrapRecords,
|
|
config = portalConfig)
|
|
|
|
return HistoryNetwork(portalProtocol: portalProtocol, contentDB: contentDB)
|
|
|
|
proc start*(p: HistoryNetwork) =
|
|
info "Starting Portal execution history network",
|
|
protocolId = p.portalProtocol.protocolId
|
|
p.portalProtocol.start()
|
|
|
|
proc stop*(p: HistoryNetwork) =
|
|
p.portalProtocol.stop()
|