160 lines
5.5 KiB
Nim
160 lines
5.5 KiB
Nim
# nimbus_verified_proxy
|
|
# 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.
|
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
{.push raises: [Defect].}
|
|
else:
|
|
{.push raises: [].}
|
|
|
|
import
|
|
std/typetraits,
|
|
eth/common/eth_types as etypes,
|
|
eth/[trie, rlp, trie/db],
|
|
stint,
|
|
web3
|
|
|
|
type
|
|
ExecutionData* = object
|
|
parentHash*: BlockHash
|
|
feeRecipient*: Address
|
|
stateRoot*: BlockHash
|
|
receiptsRoot*: BlockHash
|
|
logsBloom*: FixedBytes[256]
|
|
prevRandao*: FixedBytes[32]
|
|
blockNumber*: Quantity
|
|
gasLimit*: Quantity
|
|
gasUsed*: Quantity
|
|
timestamp*: Quantity
|
|
extraData*: DynamicBytes[0, 32]
|
|
baseFeePerGas*: UInt256
|
|
blockHash*: BlockHash
|
|
transactions*: seq[TypedTransaction]
|
|
withdrawals*: seq[WithdrawalV1]
|
|
|
|
proc asExecutionData*(
|
|
payload: ExecutionPayloadV1 | ExecutionPayloadV2 | ExecutionPayloadV3): ExecutionData =
|
|
when payload is ExecutionPayloadV1:
|
|
return ExecutionData(
|
|
parentHash: payload.parentHash,
|
|
feeRecipient: payload.feeRecipient,
|
|
stateRoot: payload.stateRoot,
|
|
receiptsRoot: payload.receiptsRoot,
|
|
logsBloom: payload.logsBloom,
|
|
prevRandao: payload.prevRandao,
|
|
blockNumber: payload.blockNumber,
|
|
gasLimit: payload.gasLimit,
|
|
gasUsed: payload.gasUsed,
|
|
timestamp: payload.timestamp,
|
|
extraData: payload.extraData,
|
|
baseFeePerGas: payload.baseFeePerGas,
|
|
blockHash: payload.blockHash,
|
|
transactions: payload.transactions,
|
|
withdrawals: @[]
|
|
)
|
|
else:
|
|
# TODO: Deal with different payload types
|
|
return ExecutionData(
|
|
parentHash: payload.parentHash,
|
|
feeRecipient: payload.feeRecipient,
|
|
stateRoot: payload.stateRoot,
|
|
receiptsRoot: payload.receiptsRoot,
|
|
logsBloom: payload.logsBloom,
|
|
prevRandao: payload.prevRandao,
|
|
blockNumber: payload.blockNumber,
|
|
gasLimit: payload.gasLimit,
|
|
gasUsed: payload.gasUsed,
|
|
timestamp: payload.timestamp,
|
|
extraData: payload.extraData,
|
|
baseFeePerGas: payload.baseFeePerGas,
|
|
blockHash: payload.blockHash,
|
|
transactions: payload.transactions,
|
|
withdrawals: payload.withdrawals
|
|
)
|
|
|
|
template unsafeQuantityToInt64(q: Quantity): int64 =
|
|
int64 q
|
|
|
|
func toFixedBytes(d: MDigest[256]): FixedBytes[32] =
|
|
FixedBytes[32](d.data)
|
|
|
|
template asEthHash(hash: BlockHash): Hash256 =
|
|
Hash256(data: distinctBase(hash))
|
|
|
|
proc calculateTransactionData(
|
|
items: openArray[TypedTransaction]):
|
|
(etypes.Hash256, seq[TxHash], uint64) {.raises: [Defect, RlpError].} =
|
|
## returns tuple composed of
|
|
## - root of transactions trie
|
|
## - list of transactions hashes
|
|
## - total size of transactions in block
|
|
var tr = initHexaryTrie(newMemoryDB())
|
|
var txHashes: seq[TxHash]
|
|
var txSize: uint64
|
|
for i, t in items:
|
|
let tx = distinctBase(t)
|
|
txSize = txSize + uint64(len(tx))
|
|
tr.put(rlp.encode(i), tx)
|
|
txHashes.add(toFixedBytes(keccakHash(tx)))
|
|
return (tr.rootHash(), txHashes, txSize)
|
|
|
|
func blockHeaderSize(
|
|
payload: ExecutionData, txRoot: etypes.Hash256): uint64 =
|
|
let bh = etypes.BlockHeader(
|
|
parentHash : payload.parentHash.asEthHash,
|
|
ommersHash : etypes.EMPTY_UNCLE_HASH,
|
|
coinbase : etypes.EthAddress payload.feeRecipient,
|
|
stateRoot : payload.stateRoot.asEthHash,
|
|
txRoot : txRoot,
|
|
receiptRoot : payload.receiptsRoot.asEthHash,
|
|
bloom : distinctBase(payload.logsBloom),
|
|
difficulty : default(etypes.DifficultyInt),
|
|
blockNumber : payload.blockNumber.distinctBase.u256,
|
|
gasLimit : payload.gasLimit.unsafeQuantityToInt64,
|
|
gasUsed : payload.gasUsed.unsafeQuantityToInt64,
|
|
timestamp : fromUnix payload.timestamp.unsafeQuantityToInt64,
|
|
extraData : bytes payload.extraData,
|
|
mixDigest : payload.prevRandao.asEthHash,
|
|
nonce : default(etypes.BlockNonce),
|
|
fee : some payload.baseFeePerGas
|
|
)
|
|
return uint64(len(rlp.encode(bh)))
|
|
|
|
proc asBlockObject*(
|
|
p: ExecutionData): BlockObject {.raises: [Defect, RlpError].} =
|
|
# TODO: currently we always calculate txHashes as BlockObject does not have
|
|
# option of returning full transactions. It needs fixing at nim-web3 library
|
|
# level
|
|
let (txRoot, txHashes, txSize) = calculateTransactionData(p.transactions)
|
|
let headerSize = blockHeaderSize(p, txRoot)
|
|
let blockSize = txSize + headerSize
|
|
BlockObject(
|
|
number: p.blockNumber,
|
|
hash: p.blockHash,
|
|
parentHash: p.parentHash,
|
|
sha3Uncles: FixedBytes(etypes.EMPTY_UNCLE_HASH.data),
|
|
logsBloom: p.logsBloom,
|
|
transactionsRoot: toFixedBytes(txRoot),
|
|
stateRoot: p.stateRoot,
|
|
receiptsRoot: p.receiptsRoot,
|
|
miner: p.feeRecipient,
|
|
difficulty: UInt256.zero,
|
|
extraData: p.extraData.toHex,
|
|
gasLimit: p.gasLimit,
|
|
gasUsed: p.gasUsed,
|
|
timestamp: p.timestamp,
|
|
nonce: some(default(FixedBytes[8])),
|
|
size: Quantity(blockSize),
|
|
# TODO: It does not matter what we put here after merge blocks.
|
|
# Other projects like `helios` return `0`, data providers like alchemy return
|
|
# transition difficulty. For now retruning `0` as this is a bit easier to do.
|
|
totalDifficulty: UInt256.zero,
|
|
transactions: txHashes,
|
|
uncles: @[],
|
|
baseFeePerGas: some(p.baseFeePerGas)
|
|
)
|
|
|