# nimbus-eth1 # Copyright (c) 2021 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 std/[algorithm, sequtils, sets, tables], eth/common, stew/interval_set, ../aristo_init/[aristo_memory, aristo_rocksdb], ".."/[aristo_desc, aristo_get, aristo_vid, aristo_transcode, aristo_utils] const Vid2 = @[VertexID(2)].toHashSet # ------------------------------------------------------------------------------ # Private helper # ------------------------------------------------------------------------------ proc invTo(s: IntervalSetRef[VertexID,uint64]; T: type HashSet[VertexID]): T = ## Convert the complement of the argument list `s` to a set of vertex IDs ## as it would appear with a vertex generator state list. if s.total < high(uint64): for w in s.increasing: if w.maxPt == high(VertexID): result.incl w.minPt # last interval else: for pt in w.minPt .. w.maxPt: result.incl pt proc toNodeBe( vtx: VertexRef; # Vertex to convert db: AristoDbRef; # Database, top layer ): Result[NodeRef,VertexID] = ## Similar to `toNode()` but fetching from the backend only case vtx.vType: of Leaf: let node = NodeRef(vType: Leaf, lPfx: vtx.lPfx, lData: vtx.lData) if vtx.lData.pType == AccountData: let vid = vtx.lData.account.storageID if vid.isValid: let rc = db.getKeyBackend vid if rc.isErr or not rc.value.isValid: return err(vid) node.key[0] = rc.value return ok node of Branch: let node = NodeRef(vType: Branch, bVid: vtx.bVid) var missing: seq[VertexID] for n in 0 .. 15: let vid = vtx.bVid[n] if vid.isValid: let rc = db.getKeyBackend vid if rc.isOk and rc.value.isValid: node.key[n] = rc.value else: return err(vid) else: node.key[n] = VOID_HASH_KEY return ok node of Extension: let vid = vtx.eVid rc = db.getKeyBackend vid if rc.isOk and rc.value.isValid: let node = NodeRef(vType: Extension, ePfx: vtx.ePfx, eVid: vid) node.key[0] = rc.value return ok node return err(vid) # ------------------------------------------------------------------------------ # Public functions # ------------------------------------------------------------------------------ proc checkBE*[T]( be: T; # backend descriptor db: AristoDbRef; # Database, top layer relax: bool; # Not compiling hashes if `true` cache: bool; # Also verify cache ): Result[void,(VertexID,AristoError)] = ## Make sure that each vertex has a Merkle hash and vice versa. Also check ## the vertex ID generator state. let vids = IntervalSetRef[VertexID,uint64].init() discard vids.merge Interval[VertexID,uint64].new(VertexID(1),high(VertexID)) for (_,vid,vtx) in be.walkVtx: if not vtx.isValid: return err((vid,CheckBeVtxInvalid)) let rc = db.getKeyBackend vid if rc.isErr or not rc.value.isValid: return err((vid,CheckBeKeyMissing)) for (_,vid,key) in be.walkKey: if not key.isvalid: return err((vid,CheckBeKeyInvalid)) let rc = db.getVtxBackend vid if rc.isErr or not rc.value.isValid: return err((vid,CheckBeVtxMissing)) let rx = rc.value.toNodeBe db # backend only if rx.isErr: return err((vid,CheckBeKeyCantCompile)) if not relax: let expected = rx.value.to(HashKey) if expected != key: return err((vid,CheckBeKeyMismatch)) discard vids.reduce Interval[VertexID,uint64].new(vid,vid) # Compare calculated state against database state block: # Extract vertex ID generator state var vGen: HashSet[VertexID] for (_,_,w) in be.walkIdg: vGen = vGen + w.toHashSet let vGenExpected = vids.invTo(HashSet[VertexID]) delta = vGenExpected -+- vGen # symmetric difference if 0 < delta.len: # Exclude fringe case when there is a single root vertex only if vGenExpected != Vid2 or 0 < vGen.len: return err((delta.toSeq.sorted[^1],CheckBeGarbledVGen)) # Check cache against backend if cache: # Check structural table for (vid,vtx) in db.top.sTab.pairs: # A `kMap[]` entry must exist. if not db.top.kMap.hasKey vid: return err((vid,CheckBeCacheKeyMissing)) if vtx.isValid: # Register existing vid against backend generator state discard vids.reduce Interval[VertexID,uint64].new(vid,vid) else: # Some vertex is to be deleted, the key must be empty let lbl = db.top.kMap.getOrVoid vid if lbl.isValid: return err((vid,CheckBeCacheKeyNonEmpty)) # There must be a representation on the backend DB if db.getVtxBackend(vid).isErr: return err((vid,CheckBeCacheVidUnsynced)) # Register deleted vid against backend generator state discard vids.merge Interval[VertexID,uint64].new(vid,vid) # Check key table for (vid,lbl) in db.top.kMap.pairs: let vtx = db.getVtx vid if not db.top.sTab.hasKey(vid) and not vtx.isValid: return err((vid,CheckBeCacheKeyDangling)) if lbl.isValid and not relax: if not vtx.isValid: return err((vid,CheckBeCacheVtxDangling)) let rc = vtx.toNode db # compile cache first if rc.isErr: return err((vid,CheckBeCacheKeyCantCompile)) let expected = rc.value.to(HashKey) if expected != lbl.key: return err((vid,CheckBeCacheKeyMismatch)) # Check vGen var tmp = AristoDbRef(top: AristoLayerRef(vGen: db.top.vGen)) tmp.vidReorg() let vGen = tmp.top.vGen.toHashSet vGenExpected = vids.invTo(HashSet[VertexID]) delta = vGenExpected -+- vGen # symmetric difference if 0 < delta.len: # Exclude fringe case when there is a single root vertex only if vGenExpected != Vid2 or 0 < vGen.len: return err((delta.toSeq.sorted[^1],CheckBeCacheGarbledVGen)) ok() # ------------------------------------------------------------------------------ # End # ------------------------------------------------------------------------------