# Nimbus # Copyright (c) 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. {.push raises: [].} import chronicles, std/tables, ../../common, ../../db/core_db, ../../evm/types, ../../evm/state, ../validate, ../executor/process_block export common, core_db type CursorDesc = object forkJunction: BlockNumber hash: Hash32 BlockDesc* = object blk*: Block receipts*: seq[Receipt] BaseDesc = object hash: Hash32 header: Header CanonicalDesc = object cursorHash: Hash32 header: Header ForkedChainRef* = ref object stagingTx: CoreDbTxRef db: CoreDbRef com: CommonRef blocks: Table[Hash32, BlockDesc] txRecords: Table[Hash32, (Hash32, uint64)] baseHash: Hash32 baseHeader: Header cursorHash: Hash32 cursorHeader: Header cursorHeads: seq[CursorDesc] extraValidation: bool baseDistance: uint64 const BaseDistance = 128 # ------------------------------------------------------------------------------ # Private # ------------------------------------------------------------------------------ template shouldNotKeyError(body: untyped) = try: body except KeyError as exc: raiseAssert exc.msg proc processBlock(c: ForkedChainRef, parent: Header, blk: Block): Result[seq[Receipt], string] = template header(): Header = blk.header let vmState = BaseVMState() vmState.init(parent, header, c.com) if c.extraValidation: ?c.com.validateHeaderAndKinship(blk, vmState.parent, checkSealOK = false) ?vmState.processBlock( blk, skipValidation = false, skipReceipts = false, skipUncles = true, ) # We still need to write header to database # because validateUncles still need it let blockHash = header.blockHash() if not c.db.persistHeader( blockHash, header, c.com.startOfHistory): return err("Could not persist header") # update currentBlock *after* we persist it # so the rpc return consistent result # between eth_blockNumber and eth_syncing c.com.syncCurrent = header.number ok(move(vmState.receipts)) func updateCursorHeads(c: ForkedChainRef, cursorHash: Hash32, header: Header) = # Example of cursorHeads and cursor # # -- A1 - A2 - A3 -- D5 - D6 # / / # base - B1 - B2 - B3 - B4 # \ # --- C3 - C4 # # A3, B4, C4, and D6, are in cursorHeads # Any one of them with blockHash == cursorHash # is the active chain with cursor pointing to the # latest block of that chain. for i in 0.. head.header.number: c.blocks.del(prevHash) else: break prevHash = header.parentHash if c.cursorHeads.len == 0: return # Update cursorHeads if indeed we trim for i in 0.. StagedBlocksThreshold # We need to persist some of the in-memory stuff # to a "staging area" or disk-backed memory but it must not afect `base`. # `base` is the point of no return, we only update it on finality. c.replaySegment(header.parentHash) c.validateBlock(c.cursorHeader, blk) proc forkChoice*(c: ForkedChainRef, headHash: Hash32, finalizedHash: Hash32): Result[void, string] = if headHash == c.cursorHash and finalizedHash == static(default(Hash32)): # Do nothing if the new head already our current head # and there is no request to new finality return ok() # If there are multiple heads, find which chain headHash belongs to let head = ?c.findCanonicalHead(headHash) if finalizedHash == static(default(Hash32)): # skip newBase calculation and skip chain finalization # if finalizedHash is zero c.updateHeadIfNecessary(head, headHash) return ok() # Finalized block must be part of canonical chain let finalizedHeader = ?c.canonicalChain(finalizedHash, headHash) let newBase = c.calculateNewBase( finalizedHeader, headHash, head.header) if newBase.hash == c.baseHash: # The base is not updated but the cursor maybe need update c.updateHeadIfNecessary(head, headHash) return ok() # At this point cursorHeader.number > baseHeader.number if newBase.hash == c.cursorHash: # Paranoid check, guaranteed by `newBase.hash == c.cursorHash` doAssert(not c.stagingTx.isNil) # CL decide to move backward and then forward? if c.cursorHeader.number < head.header.number: c.replaySegment(headHash, c.cursorHeader, c.cursorHash) # Current segment is canonical chain c.writeBaggage(newBase.hash) c.setHead(headHash, head.header.number) c.stagingTx.commit() c.stagingTx = nil # Move base to newBase c.updateBase(newBase.hash, c.cursorHeader, head.cursorHash) # Save and record the block number before the last saved block state. c.db.persistent(newBase.header.number).isOkOr: return err("Failed to save state: " & $$error) return ok() # At this point finalizedHeader.number is <= headHeader.number # and possibly switched to other chain beside the one with cursor doAssert(finalizedHeader.number <= head.header.number) doAssert(newBase.header.number <= finalizedHeader.number) # Write segment from base+1 to newBase into database c.stagingTx.rollback() c.stagingTx = c.db.ctx.newTransaction() if newBase.header.number > c.baseHeader.number: c.replaySegment(newBase.hash) c.writeBaggage(newBase.hash) c.stagingTx.commit() c.stagingTx = nil # Update base forward to newBase c.updateBase(newBase.hash, newBase.header, head.cursorHash) c.db.persistent(newBase.header.number).isOkOr: return err("Failed to save state: " & $$error) if c.stagingTx.isNil: # replaySegment or setHead below don't # go straight to db c.stagingTx = c.db.ctx.newTransaction() # Move chain state forward to current head if newBase.header.number < head.header.number: c.replaySegment(headHash) c.setHead(headHash, head.header.number) # Move cursor to current head c.trimCanonicalChain(head, headHash) if c.cursorHash != headHash: c.cursorHeader = head.header c.cursorHash = headHash ok() func haveBlockAndState*(c: ForkedChainRef, blockHash: Hash32): bool = if c.blocks.hasKey(blockHash): return true if c.baseHash == blockHash: return true false proc haveBlockLocally*(c: ForkedChainRef, blockHash: Hash32): bool = if c.blocks.hasKey(blockHash): return true if c.baseHash == blockHash: return true c.db.headerExists(blockHash) func stateReady*(c: ForkedChainRef, header: Header): bool = let blockHash = header.blockHash blockHash == c.cursorHash func com*(c: ForkedChainRef): CommonRef = c.com func db*(c: ForkedChainRef): CoreDbRef = c.db func latestHeader*(c: ForkedChainRef): Header = c.cursorHeader func latestNumber*(c: ForkedChainRef): BlockNumber = c.cursorHeader.number func latestHash*(c: ForkedChainRef): Hash32 = c.cursorHash func baseNumber*(c: ForkedChainRef): BlockNumber = c.baseHeader.number func baseHash*(c: ForkedChainRef): Hash32 = c.baseHash func txRecords*(c: ForkedChainRef, txHash: Hash32): (Hash32, uint64) = c.txRecords.getOrDefault(txHash, (Hash32.default, 0'u64)) func isInMemory*(c: ForkedChainRef, blockHash: Hash32): bool = c.blocks.hasKey(blockHash) func memoryBlock*(c: ForkedChainRef, blockHash: Hash32): BlockDesc = c.blocks.getOrDefault(blockHash) func memoryTransaction*(c: ForkedChainRef, txHash: Hash32): Opt[Transaction] = let (blockHash, index) = c.txRecords.getOrDefault(txHash, (Hash32.default, 0'u64)) c.blocks.withValue(blockHash, val) do: return Opt.some(val.blk.transactions[index]) return Opt.none(Transaction) proc latestBlock*(c: ForkedChainRef): Block = c.blocks.withValue(c.cursorHash, val) do: return val.blk do: # This can happen if block pointed by cursorHash is not loaded yet try: result = c.db.getEthBlock(c.cursorHash) c.blocks[c.cursorHash] = BlockDesc( blk: result, receipts: c.db.getReceipts(result.header.receiptsRoot), ) except BlockNotFound: doAssert(false, "Block should exists in database") except RlpError: doAssert(false, "Receipts should exists in database") proc headerByNumber*(c: ForkedChainRef, number: BlockNumber): Result[Header, string] = if number > c.cursorHeader.number: return err("Requested block number not exists: " & $number) if number == c.cursorHeader.number: return ok(c.cursorHeader) if number == c.baseHeader.number: return ok(c.baseHeader) if number < c.baseHeader.number: var header: Header if c.db.getBlockHeader(number, header): return ok(header) else: return err("Failed to get header with number: " & $number) shouldNotKeyError: var prevHash = c.cursorHeader.parentHash while prevHash != c.baseHash: let header = c.blocks[prevHash].blk.header if header.number == number: return ok(header) prevHash = header.parentHash doAssert(false, "headerByNumber: Unreachable code") proc headerByHash*(c: ForkedChainRef, blockHash: Hash32): Result[Header, string] = c.blocks.withValue(blockHash, val) do: return ok(val.blk.header) do: if c.baseHash == blockHash: return ok(c.baseHeader) var header: Header if c.db.getBlockHeader(blockHash, header): return ok(header) return err("Failed to get header with hash: " & $blockHash) proc blockByHash*(c: ForkedChainRef, blockHash: Hash32): Opt[Block] = # used by getPayloadBodiesByHash # https://github.com/ethereum/execution-apis/blob/v1.0.0-beta.4/src/engine/shanghai.md#specification-3 # 4. Client software MAY NOT respond to requests for finalized blocks by hash. c.blocks.withValue(blockHash, val) do: return Opt.some(val.blk) do: var header: Header body: BlockBody if c.db.getBlockHeader(blockHash, header) and c.db.getBlockBody(blockHash, body): return ok(Block.init(move(header), move(body))) else: return Opt.none(Block) proc blockByNumber*(c: ForkedChainRef, number: BlockNumber): Result[Block, string] = if number > c.cursorHeader.number: return err("Requested block number not exists: " & $number) if number < c.baseHeader.number: var header: Header body: BlockBody if c.db.getBlockHeader(number, header) and c.db.getBlockBody(header, body): return ok(Block.init(move(header), move(body))) else: return err("Failed to get block with number: " & $number) shouldNotKeyError: var prevHash = c.cursorHash while prevHash != c.baseHash: c.blocks.withValue(prevHash, item): if item.blk.header.number == number: return ok(item.blk) prevHash = item.blk.header.parentHash return err("Block not found, number = " & $number) func blockFromBaseTo*(c: ForkedChainRef, number: BlockNumber): seq[Block] = # return block in reverse order shouldNotKeyError: var prevHash = c.cursorHash while prevHash != c.baseHash: c.blocks.withValue(prevHash, item): if item.blk.header.number <= number: result.add item.blk prevHash = item.blk.header.parentHash func isCanonical*(c: ForkedChainRef, blockHash: Hash32): bool = shouldNotKeyError: var prevHash = c.cursorHash while prevHash != c.baseHash: c.blocks.withValue(prevHash, item): if blockHash == prevHash: return true prevHash = item.blk.header.parentHash proc isCanonicalAncestor*(c: ForkedChainRef, blockNumber: BlockNumber, blockHash: Hash32): bool = if blockNumber >= c.cursorHeader.number: return false if blockHash == c.cursorHash: return false if c.baseHeader.number < c.cursorHeader.number: # The current canonical chain in memory is headed by # cursorHeader shouldNotKeyError: var prevHash = c.cursorHeader.parentHash while prevHash != c.baseHash: var header = c.blocks[prevHash].blk.header if prevHash == blockHash and blockNumber == header.number: return true prevHash = header.parentHash # canonical chain in database should have a marker # and the marker is block number var canonHash: common.Hash32 c.db.getBlockHash(blockNumber, canonHash) and canonHash == blockHash