nimbus-eth1/hive_integration/nodocker/engine/withdrawals/wd_reorg_spec.nim

343 lines
13 KiB
Nim

# Nimbus
# Copyright (c) 2023-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/tables,
stint,
chronos,
chronicles,
eth/common,
./wd_base_spec,
./wd_history,
../test_env,
../engine_client,
../types,
../base_spec,
../../../../nimbus/beacon/web3_eth_conv,
../../../../nimbus/utils/utils
# Withdrawals re-org spec:
# Specifies a withdrawals test where the withdrawals re-org can happen
# even to a point before withdrawals were enabled, or simply to a previous
# withdrawals block.
type
ReorgSpec* = ref object of WDBaseSpec
# How many blocks the re-org will replace, including the head
reOrgBlockCount* : int
# Whether the client should fetch the sidechain by syncing from the secondary client
reOrgViaSync* : bool
sidechainTimeIncrements*: int
Sidechain = ref object
startAccount: UInt256
nextIndex : int
wdHistory : WDHistory
sidechain : Table[uint64, ExecutableData]
payloadId : Bytes8
height : uint64
attr : Opt[PayloadAttributes]
Canonical = ref object
startAccount: UInt256
nextIndex : int
proc getSidechainSplitHeight(ws: ReorgSpec): int =
doAssert(ws.reOrgBlockCount <= ws.getTotalPayloadCount())
return ws.getTotalPayloadCount() + 1 - ws.reOrgBlockCount
proc getSidechainBlockTimeIncrements(ws: ReorgSpec): int=
if ws.sidechainTimeIncrements == 0:
return ws.getBlockTimeIncrements()
ws.sidechainTimeIncrements
proc getSidechainforkHeight(ws: ReorgSpec): int =
if ws.getSidechainBlockTimeIncrements() != ws.getBlockTimeIncrements():
# Block timestamp increments in both chains are different so need to
# calculate different heights, only if split happens before fork.
# We cannot split by having two different genesis blocks.
doAssert(ws.getSidechainSplitHeight() != 0, "invalid sidechain split height")
if ws.getSidechainSplitHeight() <= ws.forkHeight:
# We need to calculate the height of the fork on the sidechain
let sidechainSplitBlocktimestamp = (ws.getSidechainSplitHeight() - 1) * ws.getBlockTimeIncrements()
let remainingTime = ws.getWithdrawalsGenesisTimeDelta() - sidechainSplitBlocktimestamp
if remainingTime == 0 :
return ws.getSidechainSplitHeight()
return ((remainingTime - 1) div ws.sidechainTimeIncrements) + ws.getSidechainSplitHeight()
return ws.forkHeight
proc execute*(ws: ReorgSpec, env: TestEnv): bool =
result = true
testCond waitFor env.clMock.waitForTTD()
# Spawn a secondary client which will produce the sidechain
let sec = env.addEngine(addToCL = false)
var
canonical = Canonical(
startAccount: u256(0x1000),
nextIndex : 0,
)
sidechain = Sidechain(
startAccount: 1.u256 shl 160,
nextIndex : 0,
wdHistory : WDHistory(),
sidechain : Table[uint64, ExecutableData]()
)
# Sidechain withdraws on the max account value range 0xffffffffffffffffffffffffffffffffffffffff
sidechain.startAccount -= u256(ws.getWithdrawableAccountCount()+1)
let numBlocks = ws.getPreWithdrawalsBlockCount()+ws.wdBlockCount
let pbRes = env.clMock.produceBlocks(numBlocks, BlockProcessCallbacks(
onPayloadProducerSelected: proc(): bool =
env.clMock.nextWithdrawals = Opt.none(seq[WithdrawalV1])
if env.clMock.currentPayloadNumber >= ws.forkHeight.uint64:
# Prepare some withdrawals
let wfb = ws.generateWithdrawalsForBlock(canonical.nextIndex, canonical.startAccount)
env.clMock.nextWithdrawals = Opt.some(w3Withdrawals wfb.wds)
canonical.nextIndex = wfb.nextIndex
ws.wdHistory.put(env.clMock.currentPayloadNumber, wfb.wds)
if env.clMock.currentPayloadNumber >= ws.getSidechainSplitHeight().uint64:
# We have split
if env.clMock.currentPayloadNumber >= ws.getSidechainforkHeight().uint64:
# And we are past the withdrawals fork on the sidechain
let wfb = ws.generateWithdrawalsForBlock(sidechain.nextIndex, sidechain.startAccount)
sidechain.wdHistory.put(env.clMock.currentPayloadNumber, wfb.wds)
sidechain.nextIndex = wfb.nextIndex
else:
if env.clMock.nextWithdrawals.isSome:
let wds = ethWithdrawals env.clMock.nextWithdrawals.get()
sidechain.wdHistory.put(env.clMock.currentPayloadNumber, wds)
sidechain.nextIndex = canonical.nextIndex
return true
,
onRequestNextPayload: proc(): bool =
# Send transactions to be included in the payload
let txs = env.makeTxs(
BaseTx(
recipient: Opt.some(prevRandaoContractAddr),
amount: 1.u256,
txType: ws.txType,
gasLimit: 75000.GasInt,
),
ws.getTransactionCountPerPayload()
)
testCond env.sendTxs(env.clMock.nextBlockProducer, txs):
error "Error trying to send transaction"
# Error will be ignored here since the tx could have been already relayed
discard env.sendTxs(sec, txs)
if env.clMock.currentPayloadNumber >= ws.getSidechainSplitHeight().uint64:
# Also request a payload from the sidechain
var fcState = ForkchoiceStateV1(
headBlockHash: env.clMock.latestForkchoice.headBlockHash,
)
if env.clMock.currentPayloadNumber > ws.getSidechainSplitHeight().uint64:
let lastSidePayload = sidechain.sidechain[env.clMock.currentPayloadNumber-1]
fcState.headBlockHash = lastSidePayload.blockHash
var attr = PayloadAttributes(
prevRandao: env.clMock.latestPayloadAttributes.prevRandao,
suggestedFeeRecipient: env.clMock.latestPayloadAttributes.suggestedFeeRecipient,
)
if env.clMock.currentPayloadNumber > ws.getSidechainSplitHeight().uint64:
attr.timestamp = w3Qty(sidechain.sidechain[env.clMock.currentPayloadNumber-1].timestamp, ws.getSidechainBlockTimeIncrements())
elif env.clMock.currentPayloadNumber == ws.getSidechainSplitHeight().uint64:
attr.timestamp = w3Qty(env.clMock.latestHeader.timestamp, ws.getSidechainBlockTimeIncrements())
else:
attr.timestamp = env.clMock.latestPayloadAttributes.timestamp
if env.clMock.currentPayloadNumber >= ws.getSidechainforkHeight().uint64:
# Withdrawals
let rr = sidechain.wdHistory.get(env.clMock.currentPayloadNumber)
testCond rr.isOk:
error "sidechain wd", msg=rr.error
attr.withdrawals = Opt.some(w3Withdrawals rr.get)
info "Requesting sidechain payload",
number=env.clMock.currentPayloadNumber
sidechain.attr = Opt.some(attr)
let r = sec.forkchoiceUpdated(attr.timestamp, fcState, attr)
r.expectNoError()
r.expectPayloadStatus(PayloadExecutionStatus.valid)
testCond r.get().payloadId.isSome:
error "Unable to get a payload ID on the sidechain"
sidechain.payloadId = r.get().payloadId.get()
return true
,
onGetPayload: proc(): bool =
var
payload: ExecutableData
if env.clMock.latestPayloadBuilt.blockNumber.uint64 >= ws.getSidechainSplitHeight().uint64:
# This payload is built by the secondary client, hence need to manually fetch it here
doAssert(sidechain.attr.isSome)
let attr = sidechain.attr.get()
let timeVer = attr.timestamp
let r = sec.getPayload(timeVer, sidechain.payloadId)
r.expectNoError()
payload = r.get().toExecutableData(attr)
sidechain.sidechain[payload.blockNumber.uint64] = payload
else:
# This block is part of both chains, simply forward it to the secondary client
payload = env.clMock.latestExecutedPayload
let r = sec.newPayload(payload)
r.expectStatus(PayloadExecutionStatus.valid)
let fcState = ForkchoiceStateV1(
headBlockHash: payload.blockHash,
)
let p = sec.forkchoiceUpdated(payload.timestamp, fcState)
p.expectPayloadStatus(PayloadExecutionStatus.valid)
return true
))
testCond pbRes
sidechain.height = env.clMock.latestExecutedPayload.blockNumber.uint64
if ws.forkHeight < ws.getSidechainforkHeight():
# This means the canonical chain forked before the sidechain.
# Therefore we need to produce more sidechain payloads to reach
# at least`ws.WithdrawalsBlockCount` withdrawals payloads produced on
# the sidechain.
let height = ws.getSidechainforkHeight()-ws.forkHeight
for i in 0..<height:
let
wfb = ws.generateWithdrawalsForBlock(sidechain.nextIndex, sidechain.startAccount)
sidechain.wdHistory.put(sidechain.height+1, wfb.wds)
sidechain.nextIndex = wfb.nextIndex
let wds = sidechain.wdHistory.get(sidechain.height+1).valueOr:
echo "get wd history error ", error
return false
let
attr = PayloadAttributes(
timestamp: w3Qty(sidechain.sidechain[sidechain.height].timestamp, ws.getSidechainBlockTimeIncrements()),
prevRandao: env.clMock.latestPayloadAttributes.prevRandao,
suggestedFeeRecipient: env.clMock.latestPayloadAttributes.suggestedFeeRecipient,
withdrawals: Opt.some(w3Withdrawals wds),
)
fcState = ForkchoiceStateV1(
headBlockHash: sidechain.sidechain[sidechain.height].blockHash,
)
let r = sec.client.forkchoiceUpdatedV2(fcState, Opt.some(attr))
r.expectPayloadStatus(PayloadExecutionStatus.valid)
let p = sec.client.getPayloadV2(r.get().payloadId.get)
p.expectNoError()
let z = p.get()
let s = sec.client.newPayloadV2(z.executionPayload)
s.expectStatus(PayloadExecutionStatus.valid)
let fs = ForkchoiceStateV1(headBlockHash: z.executionPayload.blockHash)
let q = sec.client.forkchoiceUpdatedV2(fs)
q.expectPayloadStatus(PayloadExecutionStatus.valid)
inc sidechain.height
let tmp = executionPayload(z.executionPayload)
sidechain.sidechain[sidechain.height] = tmp.toExecutableData(attr)
# Check the withdrawals on the latest
let res = ws.wdHistory.verifyWithdrawals(sidechain.height, Opt.none(uint64), env.client)
testCond res.isOk
if ws.reOrgViaSync:
# Send latest sidechain payload as NewPayload + FCU and wait for sync
let
payload = sidechain.sidechain[sidechain.height]
sideHash = sidechain.sidechain[sidechain.height].blockHash
sleep = DefaultSleep
period = chronos.seconds(sleep)
var loop = 0
if ws.timeoutSeconds == 0:
ws.timeoutSeconds = DefaultTimeout
while loop < ws.timeoutSeconds:
let r = env.client.newPayloadV2(payload.basePayload.V2)
r.expectNoError()
let fcState = ForkchoiceStateV1(headBlockHash: sideHash)
let p = env.client.forkchoiceUpdatedV2(fcState)
p.expectNoError()
let status = p.get().payloadStatus.status
if status == PayloadExecutionStatus.invalid:
error "Primary client invalidated side chain"
return false
let b = env.client.latestHeader()
testCond b.isOk
let header = b.get
if header.blockHash == sideHash:
# sync successful
break
waitFor sleepAsync(period)
loop += sleep
else:
# Send all payloads one by one to the primary client
var payloadNumber = ws.getSidechainSplitHeight()
while payloadNumber.uint64 <= sidechain.height:
let payload = sidechain.sidechain[payloadNumber.uint64]
var version = Version.V1
if payloadNumber >= ws.getSidechainforkHeight():
version = Version.V2
info "Sending sidechain",
payloadNumber,
hash=payload.blockHash.short,
parentHash=payload.parentHash.short
let r = env.engine.newPayload(version, payload)
r.expectStatusEither([PayloadExecutionStatus.valid, PayloadExecutionStatus.accepted])
let fcState = ForkchoiceStateV1(headBlockHash: payload.blockHash)
let p = env.engine.forkchoiceUpdated(version, fcState)
p.expectPayloadStatus(PayloadExecutionStatus.valid)
inc payloadNumber
# Verify withdrawals changed
let r2 = sidechain.wdHistory.verifyWithdrawals(sidechain.height, Opt.none(uint64), env.client)
testCond r2.isOk
# Verify all balances of accounts in the original chain didn't increase
# after the fork.
# We are using different accounts credited between the canonical chain
# and the fork.
# We check on `latest`.
let r3 = ws.wdHistory.verifyWithdrawals(uint64(ws.forkHeight-1), Opt.none(uint64), env.client)
testCond r3.isOk
# Re-Org back to the canonical chain
let fcState = ForkchoiceStateV1(headBlockHash: env.clMock.latestPayloadBuilt.blockHash)
let r = env.client.forkchoiceUpdatedV2(fcState)
r.expectPayloadStatus(PayloadExecutionStatus.valid)