mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-21 09:39:09 +00:00
3d58393b4c
In block processing, depending on the complexity of a transaction and
hotness of caches etc, signature checking can actually make up the
majority of time needed to process a transaction (60% observed in some
randomly sampled block ranges).
Fortunately, this is a task that trivially can be offloaded to a task
pool similar to how nimbus-eth2 does it.
This PR introduces taskpools in the most simple way possible, by
performing signature checking concurrently with other TX processing,
assigning a taskpool task per TX effectively.
With this little trick, we're in gigagas land 🎉 on my laptop!
```
INF 2024-12-10 21:05:35.170+01:00 Imported blocks
blockNumber=3874817 b... mgps=1222.707 ...
```
Tests don't use the taskpool for now because it needs manual cleanup and
we don't have a good mechanism in place. Future PR:s should address this
by creating a common shutdown sequence that also closes and cleans up
other resources like the DB.
Co-authored-by: andri lim <jangko128@gmail.com>
249 lines
8.2 KiB
Nim
249 lines
8.2 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2022-2024 Status Research & Development GmbH
|
|
# Licensed under either of
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
|
|
# http://www.apache.org/licenses/LICENSE-2.0)
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT) or
|
|
# http://opensource.org/licenses/MIT)
|
|
# at your option. This file may not be copied, modified, or distributed except
|
|
# according to those terms.
|
|
|
|
import
|
|
std/[algorithm, os, sequtils, strformat, tables, times, json],
|
|
../../nimbus/core/[chain, tx_pool], # must be early (compilation annoyance)
|
|
../../nimbus/common/common,
|
|
../../nimbus/[config, constants],
|
|
../../nimbus/utils/ec_recover,
|
|
../../nimbus/core/tx_pool/[tx_chain, tx_item],
|
|
../../nimbus/transaction,
|
|
eth/common/transaction_utils,
|
|
./helpers,
|
|
eth/[keys, p2p],
|
|
stew/[keyed_queue, byteutils]
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Private functions
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc setStatus(xp: TxPoolRef; item: TxItemRef; status: TxItemStatus)
|
|
{.gcsafe,raises: [CatchableError].} =
|
|
## Change/update the status of the transaction item.
|
|
if status != item.status:
|
|
discard xp.txDB.reassign(item, status)
|
|
|
|
type
|
|
TxEnv = object
|
|
chainId: ChainID
|
|
rng: ref HmacDrbgContext
|
|
signers: Table[Address, PrivateKey]
|
|
map: Table[Address, Address]
|
|
txs: seq[Transaction]
|
|
|
|
Signer = object
|
|
address: Address
|
|
signer: PrivateKey
|
|
|
|
const
|
|
genesisFile = "tests/customgenesis/cancun123.json"
|
|
|
|
proc initTxEnv(chainId: ChainID): TxEnv =
|
|
result.rng = newRng()
|
|
result.chainId = chainId
|
|
|
|
proc getSigner(env: var TxEnv, address: Address): Signer =
|
|
env.map.withValue(address, val) do:
|
|
let newAddress = val[]
|
|
return Signer(address: newAddress, signer: env.signers[newAddress])
|
|
do:
|
|
let key = PrivateKey.random(env.rng[])
|
|
let newAddress = toCanonicalAddress(key.toPublicKey)
|
|
env.map[address] = newAddress
|
|
env.signers[newAddress] = key
|
|
return Signer(address: newAddress, signer: key)
|
|
|
|
proc fillGenesis(env: var TxEnv, param: NetworkParams) =
|
|
const txFile = "tests/test_txpool/transactions.json"
|
|
let n = json.parseFile(txFile)
|
|
|
|
var map: Table[Address, UInt256]
|
|
|
|
for z in n:
|
|
let bytes = hexToSeqByte(z.getStr)
|
|
let tx = rlp.decode(bytes, Transaction)
|
|
let sender = tx.recoverSender().expect("valid signature")
|
|
let bal = map.getOrDefault(sender, 0.u256)
|
|
if bal + tx.value > 0:
|
|
map[sender] = bal + tx.value
|
|
env.txs.add(tx)
|
|
|
|
for k, v in map:
|
|
let s = env.getSigner(k)
|
|
param.genesis.alloc[s.address] = GenesisAccount(
|
|
balance: v + v,
|
|
)
|
|
|
|
proc setupTxPool*(getStatus: proc(): TxItemStatus): (CommonRef, TxPoolRef, int) =
|
|
let
|
|
conf = makeConfig(@[
|
|
"--custom-network:" & genesisFile
|
|
])
|
|
|
|
var txEnv = initTxEnv(conf.networkParams.config.chainId)
|
|
txEnv.fillGenesis(conf.networkParams)
|
|
|
|
let com = CommonRef.new(
|
|
newCoreDbRef DefaultDbMemory, nil,
|
|
conf.networkId,
|
|
conf.networkParams
|
|
)
|
|
|
|
let txPool = TxPoolRef.new(com)
|
|
|
|
for n, tx in txEnv.txs:
|
|
let s = txEnv.getSigner(tx.recoverSender().expect("valid signature"))
|
|
let status = statusInfo[getStatus()]
|
|
let info = &"{n}/{txEnv.txs.len} {status}"
|
|
let signedTx = signTransaction(tx, s.signer, eip155 = true)
|
|
txPool.add(PooledTransaction(tx: signedTx), info)
|
|
|
|
(com, txPool, txEnv.txs.len)
|
|
|
|
proc toTxPool*(
|
|
com: CommonRef; ## to be modified, initialisier for `TxPool`
|
|
itList: seq[TxItemRef]; ## import items into new `TxPool` (read only)
|
|
baseFee = 0.GasPrice; ## initalise with `baseFee` (unless 0)
|
|
local: seq[Address] = @[]; ## local addresses
|
|
noisy = true): TxPoolRef =
|
|
|
|
doAssert not com.isNil
|
|
|
|
result = TxPoolRef.new(com)
|
|
result.baseFee = baseFee
|
|
result.maxRejects = itList.len
|
|
|
|
let noLocals = local.len == 0
|
|
var localAddr: Table[Address,bool]
|
|
for a in local:
|
|
localAddr[a] = true
|
|
|
|
noisy.showElapsed(&"Loading {itList.len} transactions"):
|
|
for item in itList:
|
|
if noLocals:
|
|
result.add(item.pooledTx, item.info)
|
|
elif localAddr.hasKey(item.sender):
|
|
doAssert result.addLocal(item.pooledTx, true).isOk
|
|
else:
|
|
doAssert result.addRemote(item.pooledTx, true).isOk
|
|
doAssert result.nItems.total == itList.len
|
|
|
|
|
|
proc toTxPool*(
|
|
com: CommonRef; ## to be modified, initialisier for `TxPool`
|
|
timeGap: var Time; ## to be set, time in the middle of time gap
|
|
nGapItems: var int; ## to be set, # items before time gap
|
|
itList: var seq[TxItemRef]; ## import items into new `TxPool` (read only)
|
|
baseFee = 0.GasPrice; ## initalise with `baseFee` (unless 0)
|
|
itemsPC = 30; ## % number if items befor time gap
|
|
delayMSecs = 200; ## size of time vap
|
|
local: seq[EAddress] = @[]; ## local addresses
|
|
noisy = true): TxPoolRef =
|
|
## Variant of `toTxPoolFromSeq()` with a time gap between consecutive
|
|
## items on the `remote` queue
|
|
doAssert not com.isNil
|
|
doAssert 0 < itemsPC and itemsPC < 100
|
|
|
|
result = TxPoolRef.new(com)
|
|
result.baseFee = baseFee
|
|
result.maxRejects = itList.len
|
|
|
|
let noLocals = local.len == 0
|
|
var localAddr: Table[Address,bool]
|
|
for a in local:
|
|
localAddr[a] = true
|
|
|
|
let
|
|
delayAt = itList.len * itemsPC div 100
|
|
middleOfTimeGap = initDuration(milliSeconds = delayMSecs div 2)
|
|
const
|
|
tFmt = "yyyy-MM-dd'T'HH-mm-ss'.'fff"
|
|
|
|
noisy.showElapsed(&"Loading {itList.len} transactions"):
|
|
for n in 0 ..< itList.len:
|
|
let item = itList[n]
|
|
if noLocals:
|
|
result.add(item.pooledTx, item.info)
|
|
elif localAddr.hasKey(item.sender):
|
|
doAssert result.addLocal(item.pooledTx, true).isOk
|
|
else:
|
|
doAssert result.addRemote(item.pooledTx, true).isOk
|
|
if n < 3 or delayAt-3 <= n and n <= delayAt+3 or itList.len-4 < n:
|
|
let t = result.getItem(item.itemID).value.timeStamp.format(tFmt, utc())
|
|
noisy.say &"added item {n} time={t}"
|
|
if delayAt == n:
|
|
nGapItems = n # pass back value
|
|
let itemID = item.itemID
|
|
doAssert result.nItems.disposed == 0
|
|
timeGap = result.getItem(itemID).value.timeStamp + middleOfTimeGap
|
|
let t = timeGap.format(tFmt, utc())
|
|
noisy.say &"{delayMSecs}ms time gap centered around {t}"
|
|
delayMSecs.sleep
|
|
|
|
doAssert result.nItems.total == itList.len
|
|
doAssert result.nItems.disposed == 0
|
|
|
|
|
|
proc toItems*(xp: TxPoolRef): seq[TxItemRef] =
|
|
toSeq(xp.txDB.byItemID.nextValues)
|
|
|
|
proc toItems*(xp: TxPoolRef; label: TxItemStatus): seq[TxItemRef] =
|
|
for (_,nonceList) in xp.txDB.decAccount(label):
|
|
result.add toSeq(nonceList.incNonce)
|
|
|
|
proc setItemStatusFromInfo*(xp: TxPoolRef) =
|
|
## Re-define status from last character of info field. Note that this might
|
|
## violate boundary conditions regarding nonces.
|
|
for item in xp.toItems:
|
|
let w = TxItemStatus.toSeq.filterIt(statusInfo[it][0] == item.info[^1])
|
|
if w.len > 0:
|
|
xp.setStatus(item, w[0])
|
|
|
|
|
|
proc getBackHeader*(xp: TxPoolRef; nTxs, nAccounts: int):
|
|
(BlockHeader, seq[Transaction], seq[Address]) {.inline.} =
|
|
## back track the block chain for at least `nTxs` transactions and
|
|
## `nAccounts` sender accounts
|
|
var
|
|
accTab: Table[Address,bool]
|
|
txsLst: seq[Transaction]
|
|
backHash = xp.head.blockHash
|
|
backHeader = xp.head
|
|
backBody = xp.chain.com.db.getBlockBody(backHash)
|
|
|
|
while true:
|
|
# count txs and step behind last block
|
|
txsLst.add backBody.transactions
|
|
backHash = backHeader.parentHash
|
|
if not xp.chain.com.db.getBlockHeader(backHash, backHeader) or
|
|
not xp.chain.com.db.getBlockBody(backHash, backBody):
|
|
break
|
|
|
|
# collect accounts unless max reached
|
|
if accTab.len < nAccounts:
|
|
for tx in backBody.transactions:
|
|
let rc = tx.ecRecover
|
|
if rc.isOK:
|
|
if xp.txDB.bySender.eq(rc.value).isOk:
|
|
accTab[rc.value] = true
|
|
if nAccounts <= accTab.len:
|
|
break
|
|
|
|
if nTxs <= txsLst.len and nAccounts <= accTab.len:
|
|
break
|
|
# otherwise get next block
|
|
|
|
(backHeader, txsLst.reversed, toSeq(accTab.keys))
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# End
|
|
# ------------------------------------------------------------------------------
|