2023-06-30 23:22:33 +01:00
|
|
|
# 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],
|
2023-11-08 12:18:32 +00:00
|
|
|
eth/[common, trie/nibbles],
|
2023-06-30 23:22:33 +01:00
|
|
|
stew/interval_set,
|
2023-08-10 21:01:28 +01:00
|
|
|
../../aristo,
|
|
|
|
../aristo_walk/persistent,
|
2023-11-08 12:18:32 +00:00
|
|
|
".."/[aristo_desc, aristo_get, aristo_vid]
|
2023-06-30 23:22:33 +01:00
|
|
|
|
|
|
|
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
|
|
|
|
|
2023-08-11 18:23:57 +01:00
|
|
|
proc toNodeBE(
|
2023-06-30 23:22:33 +01:00
|
|
|
vtx: VertexRef; # Vertex to convert
|
2023-07-04 19:24:03 +01:00
|
|
|
db: AristoDbRef; # Database, top layer
|
2023-06-30 23:22:33 +01:00
|
|
|
): Result[NodeRef,VertexID] =
|
|
|
|
## Similar to `toNode()` but fetching from the backend only
|
|
|
|
case vtx.vType:
|
|
|
|
of Leaf:
|
2023-07-05 21:27:48 +01:00
|
|
|
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:
|
2023-08-11 18:23:57 +01:00
|
|
|
let rc = db.getKeyBE vid
|
2023-07-05 21:27:48 +01:00
|
|
|
if rc.isErr or not rc.value.isValid:
|
|
|
|
return err(vid)
|
|
|
|
node.key[0] = rc.value
|
|
|
|
return ok node
|
2023-06-30 23:22:33 +01:00
|
|
|
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:
|
2023-08-11 18:23:57 +01:00
|
|
|
let rc = db.getKeyBE vid
|
2023-06-30 23:22:33 +01:00
|
|
|
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
|
2023-08-11 18:23:57 +01:00
|
|
|
rc = db.getKeyBE vid
|
2023-06-30 23:22:33 +01:00
|
|
|
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
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2023-08-11 18:23:57 +01:00
|
|
|
proc checkBE*[T: RdbBackendRef|MemBackendRef|VoidBackendRef](
|
2023-08-10 21:01:28 +01:00
|
|
|
_: type T;
|
2023-07-04 19:24:03 +01:00
|
|
|
db: AristoDbRef; # Database, top layer
|
2023-06-30 23:22:33 +01:00
|
|
|
relax: bool; # Not compiling hashes if `true`
|
2023-09-11 21:38:49 +01:00
|
|
|
cache: bool; # Also verify against top layer cache
|
|
|
|
fifos = true; # Also verify cascaded filter fifos
|
2023-06-30 23:22:33 +01:00
|
|
|
): 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))
|
|
|
|
|
2023-08-10 21:01:28 +01:00
|
|
|
for (_,vid,vtx) in T.walkVtxBE db:
|
2023-06-30 23:22:33 +01:00
|
|
|
if not vtx.isValid:
|
|
|
|
return err((vid,CheckBeVtxInvalid))
|
2023-08-11 18:23:57 +01:00
|
|
|
let rc = db.getKeyBE vid
|
2023-06-30 23:22:33 +01:00
|
|
|
if rc.isErr or not rc.value.isValid:
|
|
|
|
return err((vid,CheckBeKeyMissing))
|
2023-11-08 12:18:32 +00:00
|
|
|
case vtx.vType:
|
|
|
|
of Leaf:
|
|
|
|
discard
|
|
|
|
of Branch:
|
|
|
|
block check42Links:
|
|
|
|
var seen = false
|
|
|
|
for n in 0 .. 15:
|
|
|
|
if vtx.bVid[n].isValid:
|
|
|
|
if seen:
|
|
|
|
break check42Links
|
|
|
|
seen = true
|
|
|
|
return err((vid,CheckBeVtxBranchLinksMissing))
|
|
|
|
of Extension:
|
|
|
|
if vtx.ePfx.len == 0:
|
|
|
|
return err((vid,CheckBeVtxExtPfxMissing))
|
2023-06-30 23:22:33 +01:00
|
|
|
|
2023-08-10 21:01:28 +01:00
|
|
|
for (_,vid,key) in T.walkKeyBE db:
|
2023-06-30 23:22:33 +01:00
|
|
|
if not key.isvalid:
|
|
|
|
return err((vid,CheckBeKeyInvalid))
|
2023-08-11 18:23:57 +01:00
|
|
|
let rc = db.getVtxBE vid
|
2023-06-30 23:22:33 +01:00
|
|
|
if rc.isErr or not rc.value.isValid:
|
|
|
|
return err((vid,CheckBeVtxMissing))
|
2023-08-11 18:23:57 +01:00
|
|
|
let rx = rc.value.toNodeBE db # backend only
|
2023-06-30 23:22:33 +01:00
|
|
|
if rx.isErr:
|
|
|
|
return err((vid,CheckBeKeyCantCompile))
|
|
|
|
if not relax:
|
2023-11-08 12:18:32 +00:00
|
|
|
let expected = rx.value.digestTo(HashKey)
|
2023-06-30 23:22:33 +01:00
|
|
|
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
|
2023-08-21 15:58:30 +01:00
|
|
|
let vGen = block:
|
|
|
|
let rc = db.getIdgBE()
|
|
|
|
if rc.isOk:
|
|
|
|
rc.value.toHashSet
|
|
|
|
elif rc.error == GetIdgNotFound:
|
|
|
|
EmptyVidSeq.toHashSet
|
|
|
|
else:
|
|
|
|
return err((VertexID(0),rc.error))
|
2023-06-30 23:22:33 +01:00
|
|
|
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))
|
|
|
|
|
2023-09-11 21:38:49 +01:00
|
|
|
# Check top layer cache against backend
|
2023-06-30 23:22:33 +01:00
|
|
|
if cache:
|
2023-08-17 14:42:01 +01:00
|
|
|
if db.top.dirty:
|
|
|
|
return err((VertexID(0),CheckBeCacheIsDirty))
|
|
|
|
|
2023-06-30 23:22:33 +01:00
|
|
|
# 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
|
2023-08-11 18:23:57 +01:00
|
|
|
if db.getVtxBE(vid).isErr:
|
2023-06-30 23:22:33 +01:00
|
|
|
return err((vid,CheckBeCacheVidUnsynced))
|
|
|
|
# Register deleted vid against backend generator state
|
|
|
|
discard vids.merge Interval[VertexID,uint64].new(vid,vid)
|
|
|
|
|
2023-09-11 21:38:49 +01:00
|
|
|
# Check cascaded fifos
|
2023-09-15 16:23:53 +01:00
|
|
|
if fifos and
|
|
|
|
not db.backend.isNil and
|
|
|
|
not db.backend.filters.isNil:
|
2023-09-11 21:38:49 +01:00
|
|
|
var lastTrg = db.getKeyUBE(VertexID(1)).get(otherwise = VOID_HASH_KEY)
|
2023-11-08 12:18:32 +00:00
|
|
|
.to(Hash256)
|
2023-09-15 16:23:53 +01:00
|
|
|
for (qid,filter) in db.backend.T.walkFifoBe: # walk in fifo order
|
2023-09-11 21:38:49 +01:00
|
|
|
if filter.src != lastTrg:
|
|
|
|
return err((VertexID(0),CheckBeFifoSrcTrgMismatch))
|
2023-11-08 12:18:32 +00:00
|
|
|
if filter.trg != filter.kMap.getOrVoid(VertexID 1).to(Hash256):
|
2023-09-11 21:38:49 +01:00
|
|
|
return err((VertexID(1),CheckBeFifoTrgNotStateRoot))
|
|
|
|
lastTrg = filter.trg
|
|
|
|
|
2023-06-30 23:22:33 +01:00
|
|
|
# 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))
|
2023-11-08 12:18:32 +00:00
|
|
|
let expected = rc.value.digestTo(HashKey)
|
2023-06-30 23:22:33 +01:00
|
|
|
if expected != lbl.key:
|
|
|
|
return err((vid,CheckBeCacheKeyMismatch))
|
|
|
|
|
|
|
|
# Check vGen
|
|
|
|
let
|
2023-08-10 21:01:28 +01:00
|
|
|
vGen = db.top.vGen.vidReorg.toHashSet
|
2023-06-30 23:22:33 +01:00
|
|
|
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:
|
2023-11-08 12:18:32 +00:00
|
|
|
let delta = delta.toSeq
|
|
|
|
# As happens with Merkle signature calculator: `root=VertexID(2)`
|
|
|
|
if delta.len != 1 or delta[0] != VertexID(1) or VertexID(1) in vGen:
|
|
|
|
return err((delta.sorted[^1],CheckBeCacheGarbledVGen))
|
2023-06-30 23:22:33 +01:00
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# End
|
|
|
|
# ------------------------------------------------------------------------------
|