# 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/os, eth/common/keys, eth/p2p as eth_p2p, chronos, json_rpc/[rpcserver, rpcclient], results, ../../../nimbus/[ config, constants, core/chain, core/tx_pool, core/tx_pool/tx_item, core/block_import, rpc, sync/handlers, beacon/beacon_engine, beacon/web3_eth_conv, common ], ../../../tests/test_helpers, web3/execution_types, ./engine_client, ./types from ./node import setBlock export results type EngineEnv* = ref object conf : NimbusConf com : CommonRef node : EthereumNode server : RpcHttpServer ttd : DifficultyInt client : RpcHttpClient txPool : TxPoolRef chain : ForkedChainRef const baseFolder = "hive_integration/nodocker/engine" genesisFile = baseFolder & "/init/genesis.json" sealerKey = baseFolder & "/init/sealer.key" chainFolder = baseFolder & "/chains" jwtSecret = "0x7365637265747365637265747365637265747365637265747365637265747365" proc makeCom*(conf: NimbusConf): CommonRef = CommonRef.new( newCoreDbRef DefaultDbMemory, conf.networkId, conf.networkParams ) proc envConfig*(): NimbusConf = makeConfig(@[ "--custom-network:" & genesisFile, "--listen-address: 127.0.0.1", ]) proc envConfig*(conf: ChainConfig): NimbusConf = result = envConfig() result.networkParams.config = conf proc newEngineEnv*(conf: var NimbusConf, chainFile: string, enableAuth: bool): EngineEnv = let ctx = newEthContext() ctx.am.importPrivateKey(sealerKey).isOkOr: echo error quit(QuitFailure) let node = setupEthNode(conf, ctx) com = makeCom(conf) head = com.db.getCanonicalHead() chain = newForkedChain(com, head) let txPool = TxPoolRef.new(com) node.addEthHandlerCapability( node.peerPool, chain, txPool) # txPool must be informed of active head # so it can know the latest account state doAssert txPool.smartHead(head, chain) var key: JwtSharedKey key.fromHex(jwtSecret).isOkOr: echo "JWT SECRET ERROR: ", error quit(QuitFailure) let hooks = if enableAuth: @[httpJwtAuth(key)] else: @[] server = newRpcHttpServerWithParams("127.0.0.1:" & $conf.httpPort, hooks).valueOr: echo "Failed to create rpc server: ", error quit(QuitFailure) beaconEngine = BeaconEngineRef.new(txPool, chain) serverApi = newServerAPI(chain, txPool) setupServerAPI(serverApi, server, ctx) setupEngineAPI(beaconEngine, server) # temporary disabled #setupDebugRpc(com, txPool, server) if chainFile.len > 0: importRlpBlocks(chainFolder / chainFile, chain, true).isOkOr: echo "Failed to import RLP blocks: ", error quit(QuitFailure) server.start() let client = newRpcHttpClient() waitFor client.connect("127.0.0.1", conf.httpPort, false) node.startListening() EngineEnv( conf : conf, com : com, node : node, server : server, client : client, txPool : txPool, chain : chain ) proc close*(env: EngineEnv) = waitFor env.node.closeWait() waitFor env.client.close() waitFor env.server.closeWait() proc setRealTTD*(env: EngineEnv) = let genesis = env.com.genesisHeader let realTTD = genesis.difficulty env.com.setTTD Opt.some(realTTD) env.ttd = realTTD func httpPort*(env: EngineEnv): Port = env.conf.httpPort func client*(env: EngineEnv): RpcHttpClient = env.client func ttd*(env: EngineEnv): UInt256 = env.ttd func com*(env: EngineEnv): CommonRef = env.com func node*(env: EngineEnv): ENode = env.node.listeningAddress proc connect*(env: EngineEnv, node: ENode) = waitFor env.node.connectToNode(node) func ID*(env: EngineEnv): string = $env.conf.httpPort proc peer*(env: EngineEnv): Peer = doAssert(env.node.numPeers > 0) for peer in env.node.peers: return peer proc getTxsInPool*(env: EngineEnv, txHashes: openArray[common.Hash32]): seq[Transaction] = result = newSeqOfCap[Transaction](txHashes.len) for txHash in txHashes: let res = env.txPool.getItem(txHash) if res.isErr: continue let item = res.get if item.reject == txInfoOk: result.add item.tx proc numTxsInPool*(env: EngineEnv): int = env.txPool.numTxs func version*(env: EngineEnv, time: EthTime): Version = if env.com.isPragueOrLater(time): Version.V4 elif env.com.isCancunOrLater(time): Version.V3 elif env.com.isShanghaiOrLater(time): Version.V2 else: Version.V1 func version*(env: EngineEnv, time: Web3Quantity): Version = env.version(time.EthTime) func version*(env: EngineEnv, time: uint64): Version = env.version(time.EthTime) proc setBlock*(env: EngineEnv, blk: common.EthBlock): bool = # env.chain.setBlock(blk).isOk() debugEcho "TODO: fix setBlock" false proc newPayload*(env: EngineEnv, payload: ExecutableData): Result[PayloadStatusV1, string] = let version = env.version(payload.basePayload.timestamp) env.client.newPayload(version, payload) proc newPayload*(env: EngineEnv, version: Version, payload: ExecutableData): Result[PayloadStatusV1, string] = env.client.newPayload(version, payload) proc getPayload*(env: EngineEnv, timestamp: uint64 | Web3Quantity | EthTime, payloadId: Bytes8): Result[GetPayloadResponse, string] = let version = env.version(timestamp) env.client.getPayload(version, payloadId) proc getPayload*(env: EngineEnv, version: Version, payloadId: Bytes8): Result[GetPayloadResponse, string] = env.client.getPayload(version, payloadId) proc forkchoiceUpdated*(env: EngineEnv, timestamp: uint64 | Web3Quantity | EthTime, update: ForkchoiceStateV1, attr = Opt.none(PayloadAttributes)): Result[ForkchoiceUpdatedResponse, string] = let version = env.version(timestamp) env.client.forkchoiceUpdated(version, update, attr) proc forkchoiceUpdated*(env: EngineEnv, timestamp: uint64 | Web3Quantity | EthTime, update: ForkchoiceStateV1, attr: PayloadAttributes): Result[ForkchoiceUpdatedResponse, string] = let version = env.version(timestamp) env.client.forkchoiceUpdated(version, update, Opt.some(attr)) proc forkchoiceUpdated*(env: EngineEnv, version: Version, update: ForkchoiceStateV1, attr = Opt.none(PayloadAttributes)): Result[ForkchoiceUpdatedResponse, string] = env.client.forkchoiceUpdated(version, update, attr) proc forkchoiceUpdated*(env: EngineEnv, version: Version, update: ForkchoiceStateV1, attr: PayloadAttributes): Result[ForkchoiceUpdatedResponse, string] = env.client.forkchoiceUpdated(version, update, Opt.some(attr))