# 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/[json, strutils, sets, tables, options, streams], chronicles, eth/common/keys, eth/common/transaction_utils, stew/byteutils, results, stint, ../../nimbus/[evm/types, evm/state], ../../nimbus/db/ledger, ../../nimbus/transaction, ../../nimbus/core/executor, ../../nimbus/common/common, ../../nimbus/evm/tracer/json_tracer, ../../nimbus/core/eip4844, ../../nimbus/utils/state_dump, ../common/helpers as chp, "."/[config, helpers], ../common/state_clearing type StateContext = object name: string parent: Header header: Header tx: Transaction expectedHash: Hash32 expectedLogs: Hash32 forkStr: string chainConfig: ChainConfig index: int tracerFlags: set[TracerFlags] error: string StateResult = object name : string pass : bool root : Hash32 fork : string error: string state: StateDump TestVMState = ref object of BaseVMState proc extractNameAndFixture(ctx: var StateContext, n: JsonNode): JsonNode = for label, child in n: result = child ctx.name = label return doAssert(false, "unreachable") proc toBytes(x: string): seq[byte] = result = newSeq[byte](x.len) for i in 0.. 0 proc prepareAndRun(inputFile: string, conf: StateConf): bool = var ctx: StateContext let fixture = json.parseFile(inputFile) n = ctx.extractNameAndFixture(fixture) txData = n["transaction"] post = n["post"] pre = n["pre"] ctx.parent = parseParentHeader(n["env"]) ctx.header = parseHeader(n["env"]) if conf.debugEnabled or conf.jsonEnabled: ctx.tracerFlags = toTracerFlags(conf) var stateRes = newSeqOfCap[StateResult](post.len) index = 1 hasError = false template prepareFork(forkName: string) = try: ctx.forkStr = forkName ctx.chainConfig = getChainConfig(forkName) except ValueError as ex: debugEcho ex.msg return false ctx.index = index inc index template runSubTest(subTest: JsonNode) = ctx.expectedHash = Hash32.fromJson(subTest["hash"]) ctx.expectedLogs = Hash32.fromJson(subTest["logs"]) ctx.tx = parseTx(txData, subTest["indexes"]) let res = ctx.runExecution(conf, pre) stateRes.add res hasError = hasError or ctx.hasError if conf.fork.len > 0: if not post.hasKey(conf.fork): stdout.writeLine("selected fork not available: " & conf.fork) return false let forkData = post[conf.fork] prepareFork(conf.fork) if conf.index.isNone: for subTest in forkData: runSubTest(subTest) else: let index = conf.index.get() if index > forkData.len or index < 0: stdout.writeLine("selected index out of range(0-$1), requested $2" % [$forkData.len, $index]) return false let subTest = forkData[index] runSubTest(subTest) else: for forkName, forkData in post: prepareFork(forkName) for subTest in forkData: runSubTest(subTest) writeResultToStdout(stateRes) not hasError when defined(chronicles_runtime_filtering): type Lev = chronicles.LogLevel proc toLogLevel(v: int): Lev = case v of 1: Lev.ERROR of 2: Lev.WARN of 3: Lev.INFO of 4: Lev.DEBUG of 5: Lev.TRACE else: Lev.NONE proc setVerbosity(v: int) = let level = v.toLogLevel setLogLevel(level) proc main() = let conf = StateConf.init() when defined(chronicles_runtime_filtering): setVerbosity(conf.verbosity) loadKzgTrustedSetup().isOkOr: echo "FATAL: ", error quit(QuitFailure) if conf.inputFile.len > 0: if not prepareAndRun(conf.inputFile, conf): quit(QuitFailure) else: var noError = true for inputFile in lines(stdin): let res = prepareAndRun(inputFile, conf) noError = noError and res if not noError: quit(QuitFailure) main()