225 lines
6.7 KiB
Nim
Raw Normal View History

2023-07-09 09:16:22 +07:00
# Nimbus
# Copyright (c) 2023 Status Research & Development GmbH
# Licensed under either of
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
# at your option.
# This file may not be copied, modified, or distributed except according to
# those terms.
import
std/[os, json, strutils, times, typetraits, options],
stew/[byteutils, results],
eth/common,
2023-09-25 06:53:33 +07:00
json_rpc/rpcclient,
2023-07-09 09:16:22 +07:00
../sim_utils,
../../../tools/common/helpers as chp,
../../../tools/evmstate/helpers as ehp,
../../../tests/test_helpers,
../../../nimbus/beacon/web3_eth_conv,
../../../nimbus/beacon/execution_types,
../../../nimbus/beacon/payload_conv,
2023-09-25 06:53:33 +07:00
../../../nimbus/core/eip4844,
2023-07-09 09:16:22 +07:00
../engine/engine_client,
./test_env
2023-07-09 09:16:22 +07:00
const
baseFolder = "hive_integration/nodocker/pyspec"
2023-09-25 06:53:33 +07:00
#caseFolder = "tests/fixtures/eth_tests/EIPTests/Pyspecs/cancun"
2023-07-09 09:16:22 +07:00
caseFolder = baseFolder & "/testcases"
2023-09-25 06:53:33 +07:00
supportedNetwork = [
"Merge",
"Shanghai",
"MergeToShanghaiAtTime15k",
"Cancun",
"ShanghaiToCancunAtTime15k",
]
type
Payload = object
payload: ExecutionPayload
beaconRoot: Option[common.Hash256]
proc getPayload(node: JsonNode): Payload =
let
rlpBytes = hexToSeqByte(node.getStr)
blk = rlp.decode(rlpBytes, EthBlock)
Payload(
payload: executionPayload(blk),
beaconRoot: blk.header.parentBeaconBlockRoot,
)
2023-07-09 09:16:22 +07:00
proc validatePostState(node: JsonNode, t: TestEnv): bool =
# check nonce, balance & storage of accounts in final block against fixture values
for account, genesisAccount in postState(node["postState"]):
# get nonce & balance from last block (end of test execution)
let nonceRes = t.rpcClient.nonceAt(account)
if nonceRes.isErr:
echo "unable to call nonce from account: " & account.toHex
echo nonceRes.error
return false
let balanceRes = t.rpcClient.balanceAt(account)
if balanceRes.isErr:
echo "unable to call balance from account: " & account.toHex
echo balanceRes.error
return false
# check final nonce & balance matches expected in fixture
if genesisAccount.nonce != nonceRes.value:
echo "nonce recieved from account 0x",
account.toHex,
" doesn't match expected ",
genesisAccount.nonce,
" got ",
nonceRes.value
return false
if genesisAccount.balance != balanceRes.value:
echo "balance recieved from account 0x",
account.toHex,
" doesn't match expected ",
genesisAccount.balance,
" got ",
balanceRes.value
return false
# check final storage
if genesisAccount.storage.len > 0:
for slot, val in genesisAccount.storage:
let sRes = t.rpcClient.storageAt(account, slot)
if sRes.isErr:
echo "unable to call storage from account: 0x",
account.toHex,
" at slot 0x",
slot.toHex
echo sRes.error
return false
if val != sRes.value:
echo "storage recieved from account 0x",
account.toHex,
" at slot 0x",
slot.toHex,
" doesn't match expected 0x",
val.toHex,
" got 0x",
sRes.value.toHex
return false
return true
2023-09-25 06:53:33 +07:00
proc collectBlobHashes(list: openArray[Web3Tx]): seq[Web3Hash] =
for w3tx in list:
let tx = ethTx(w3Tx)
for h in tx.versionedHashes:
result.add w3Hash(h)
proc newPayload(client: RpcClient,
payload: ExecutionPayload,
beaconRoot: Option[common.Hash256]): Result[PayloadStatusV1, string] =
case payload.version
of Version.V1: return client.newPayloadV1(payload.V1)
of Version.V2: return client.newPayloadV2(payload.V2)
of Version.V3:
let versionedHashes = collectBlobHashes(payload.transactions)
return client.newPayloadV3(payload.V3,
versionedHashes,
w3Hash beaconRoot.get)
proc forkchoiceUpdated(client: RpcClient, version: Version,
update: ForkchoiceStateV1):
Result[ForkchoiceUpdatedResponse, string] =
case version
of Version.V1: client.forkchoiceUpdatedV1(update)
of Version.V2: client.forkchoiceUpdatedV2(update)
of Version.V3: client.forkchoiceUpdatedV3(update)
2023-07-09 09:16:22 +07:00
proc runTest(node: JsonNode, network: string): TestStatus =
let conf = getChainConfig(network)
var t = TestEnv(conf: makeTestConfig())
t.setupELClient(conf, node)
let blks = node["blocks"]
2023-09-25 06:53:33 +07:00
var
latestValidHash = common.Hash256()
latestVersion: Version
2023-07-09 09:16:22 +07:00
result = TestStatus.OK
for blkNode in blks:
let expectedStatus = if "expectException" in blkNode:
PayloadExecutionStatus.invalid
else:
PayloadExecutionStatus.valid
let payload = getPayload(blkNode["rlp"])
2023-09-25 06:53:33 +07:00
latestVersion = payload.payload.version
let res = t.rpcClient.newPayload(payload.payload, payload.beaconRoot)
2023-07-09 09:16:22 +07:00
if res.isErr:
result = TestStatus.Failed
2023-09-25 06:53:33 +07:00
echo "unable to send block ",
payload.payload.blockNumber.uint64, ": ", res.error
2023-07-09 09:16:22 +07:00
break
let pStatus = res.value
if pStatus.status == PayloadExecutionStatus.valid:
latestValidHash = ethHash pStatus.latestValidHash.get
2023-07-09 09:16:22 +07:00
if pStatus.status != expectedStatus:
result = TestStatus.Failed
2023-09-25 06:53:33 +07:00
echo "payload status mismatch for block ",
payload.payload.blockNumber.uint64,
", status: ", pStatus.status,
",expected: ", expectedStatus
2023-07-09 09:16:22 +07:00
if pStatus.validationError.isSome:
echo pStatus.validationError.get
break
block:
# only update head of beacon chain if valid response occurred
if latestValidHash != common.Hash256():
2023-07-09 09:16:22 +07:00
# update with latest valid response
let fcState = ForkchoiceStateV1(headBlockHash: BlockHash latestValidHash.data)
2023-09-25 06:53:33 +07:00
let res = t.rpcClient.forkchoiceUpdated(latestVersion, fcState)
2023-07-09 09:16:22 +07:00
if res.isErr:
result = TestStatus.Failed
echo "unable to update head of beacon chain: ", res.error
break
if not validatePostState(node, t):
result = TestStatus.Failed
break
t.stopELClient()
proc main() =
var stat: SimStat
let start = getTime()
2023-09-25 06:53:33 +07:00
let res = loadKzgTrustedSetup()
if res.isErr:
echo "FATAL: ", res.error
quit(QuitFailure)
2023-07-09 09:16:22 +07:00
for fileName in walkDirRec(caseFolder):
if not fileName.endsWith(".json"):
continue
let fixtureTests = json.parseFile(fileName)
for name, fixture in fixtureTests:
let network = fixture["network"].getStr
if network notin supportedNetwork:
# skip pre Merge tests
continue
2023-09-25 06:53:33 +07:00
try:
let status = runTest(fixture, network)
stat.inc(name, status)
except CatchableError as ex:
debugEcho ex.msg
stat.inc(name, TestStatus.Failed)
2023-07-09 09:16:22 +07:00
let elpd = getTime() - start
print(stat, elpd, "pyspec")
main()