2023-05-11 14:25:29 +00:00
|
|
|
# Nimbus - Types, data structures and shared utilities used in network sync
|
|
|
|
#
|
|
|
|
# Copyright (c) 2018-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.
|
|
|
|
|
|
|
|
import
|
|
|
|
std/sequtils,
|
|
|
|
eth/common,
|
|
|
|
rocksdb,
|
2023-06-09 11:17:37 +00:00
|
|
|
../../nimbus/db/aristo/[
|
|
|
|
aristo_constants, aristo_debug, aristo_desc, aristo_merge],
|
2023-05-11 14:25:29 +00:00
|
|
|
../../nimbus/db/kvstore_rocksdb,
|
2023-06-02 10:04:29 +00:00
|
|
|
../../nimbus/sync/protocol/snap/snap_types,
|
|
|
|
../test_sync_snap/test_types,
|
|
|
|
../replay/[pp, undump_accounts, undump_storages]
|
|
|
|
|
|
|
|
type
|
|
|
|
ProofTrieData* = object
|
|
|
|
root*: NodeKey
|
|
|
|
id*: int
|
|
|
|
proof*: seq[SnapProof]
|
2023-06-09 11:17:37 +00:00
|
|
|
kvpLst*: seq[LeafSubKVP]
|
2023-06-02 10:04:29 +00:00
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Private helpers
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2023-06-09 11:17:37 +00:00
|
|
|
proc toPfx(indent: int): string =
|
|
|
|
"\n" & " ".repeat(indent)
|
2023-05-11 14:25:29 +00:00
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
2023-06-09 11:17:37 +00:00
|
|
|
# Public pretty printing
|
2023-05-11 14:25:29 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2023-06-09 11:17:37 +00:00
|
|
|
proc pp*(w: ProofTrieData; db: var AristoDb; indent = 4): string =
|
|
|
|
let pfx = indent.toPfx
|
|
|
|
result = "(" & w.root.pp(db) & "," & $w.id & ",[" & $w.proof.len & "],"
|
|
|
|
result &= pfx & " ["
|
|
|
|
for n,kvp in w.kvpLst:
|
|
|
|
if 0 < n:
|
|
|
|
result &= "," & pfx & " "
|
|
|
|
result &= "(" & kvp.leafKey.pp(db) & "," & $kvp.payload.pType & ")"
|
|
|
|
result &= "])"
|
|
|
|
|
|
|
|
proc pp*(w: ProofTrieData; indent = 4): string =
|
|
|
|
var db = AristoDB()
|
|
|
|
w.pp(db, indent)
|
|
|
|
|
|
|
|
proc pp*(w: openArray[ProofTrieData]; db: var AristoDb; indent = 4): string =
|
|
|
|
let pfx = indent.toPfx
|
|
|
|
"[" & w.mapIt(it.pp(db, indent + 1)).join("," & pfx & " ") & "]"
|
|
|
|
|
|
|
|
proc pp*(w: openArray[ProofTrieData]; indent = 4): string =
|
|
|
|
let pfx = indent.toPfx
|
|
|
|
"[" & w.mapIt(it.pp(indent + 1)).join("," & pfx & " ") & "]"
|
|
|
|
|
|
|
|
# ----------
|
|
|
|
|
2023-05-11 14:25:29 +00:00
|
|
|
proc say*(noisy = false; pfx = "***"; args: varargs[string, `$`]) =
|
|
|
|
if noisy:
|
|
|
|
if args.len == 0:
|
|
|
|
echo "*** ", pfx
|
|
|
|
elif 0 < pfx.len and pfx[^1] != ' ':
|
|
|
|
echo pfx, " ", args.toSeq.join
|
|
|
|
else:
|
|
|
|
echo pfx, args.toSeq.join
|
|
|
|
|
2023-06-09 11:17:37 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Public helpers
|
|
|
|
# ------------------------------------------------------------------------------
|
2023-06-02 10:04:29 +00:00
|
|
|
|
|
|
|
proc to*(sample: AccountsSample; T: type seq[UndumpAccounts]): T =
|
|
|
|
## Convert test data into usable in-memory format
|
|
|
|
let file = sample.file.findFilePath.value
|
|
|
|
var root: Hash256
|
|
|
|
for w in file.undumpNextAccount:
|
|
|
|
let n = w.seenAccounts - 1
|
|
|
|
if n < sample.firstItem:
|
|
|
|
continue
|
|
|
|
if sample.lastItem < n:
|
|
|
|
break
|
|
|
|
if sample.firstItem == n:
|
|
|
|
root = w.root
|
|
|
|
elif w.root != root:
|
|
|
|
break
|
|
|
|
result.add w
|
|
|
|
|
|
|
|
proc to*(sample: AccountsSample; T: type seq[UndumpStorages]): T =
|
|
|
|
## Convert test data into usable in-memory format
|
|
|
|
let file = sample.file.findFilePath.value
|
|
|
|
var root: Hash256
|
|
|
|
for w in file.undumpNextStorages:
|
|
|
|
let n = w.seenAccounts - 1 # storages selector based on accounts
|
|
|
|
if n < sample.firstItem:
|
|
|
|
continue
|
|
|
|
if sample.lastItem < n:
|
|
|
|
break
|
|
|
|
if sample.firstItem == n:
|
|
|
|
root = w.root
|
|
|
|
elif w.root != root:
|
|
|
|
break
|
|
|
|
result.add w
|
|
|
|
|
2023-06-09 11:17:37 +00:00
|
|
|
proc to*(ua: seq[UndumpAccounts]; T: type seq[ProofTrieData]): T =
|
|
|
|
var (rootKey, rootVid) = (EMPTY_ROOT_KEY, VertexID(0))
|
|
|
|
for w in ua:
|
|
|
|
let thisRoot = w.root.to(NodeKey)
|
|
|
|
if rootKey != thisRoot:
|
|
|
|
(rootKey, rootVid) = (thisRoot, VertexID(rootVid.uint64 + 1))
|
|
|
|
result.add ProofTrieData(
|
|
|
|
root: rootKey,
|
|
|
|
proof: w.data.proof,
|
|
|
|
kvpLst: w.data.accounts.mapIt(LeafSubKVP(
|
|
|
|
leafKey: LeafKey(root: rootVid, path: it.accKey.to(NodeTag)),
|
|
|
|
payload: PayloadRef(pType: BlobData, blob: it.accBlob))))
|
|
|
|
|
|
|
|
proc to*(us: seq[UndumpStorages]; T: type seq[ProofTrieData]): T =
|
|
|
|
var (rootKey, rootVid) = (EMPTY_ROOT_KEY, VertexID(0))
|
|
|
|
for n,s in us:
|
|
|
|
for w in s.data.storages:
|
|
|
|
let thisRoot = w.account.storageRoot.to(NodeKey)
|
|
|
|
if rootKey != thisRoot:
|
|
|
|
(rootKey, rootVid) = (thisRoot, VertexID(rootVid.uint64 + 1))
|
|
|
|
result.add ProofTrieData(
|
|
|
|
root: thisRoot,
|
|
|
|
id: n + 1,
|
|
|
|
kvpLst: w.data.mapIt(LeafSubKVP(
|
|
|
|
leafKey: LeafKey(root: rootVid, path: it.slotHash.to(NodeTag)),
|
|
|
|
payload: PayloadRef(pType: BlobData, blob: it.slotData))))
|
|
|
|
if 0 < result.len:
|
|
|
|
result[^1].proof = s.data.proof
|
2023-06-02 10:04:29 +00:00
|
|
|
|
2023-06-09 11:17:37 +00:00
|
|
|
proc mapRootVid*(a: openArray[LeafSubKVP]; toVid: VertexID): seq[LeafSubKVP] =
|
|
|
|
a.mapIt(LeafSubKVP(
|
|
|
|
leafKey: LeafKey(root: toVid, path: it.leafKey.path),
|
|
|
|
payload: it.payload))
|
2023-06-02 10:04:29 +00:00
|
|
|
|
2023-05-11 14:25:29 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Public iterators
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
iterator walkAllDb*(rocky: RocksStoreRef): (int,Blob,Blob) =
|
|
|
|
## Walk over all key-value pairs of the database (`RocksDB` only.)
|
|
|
|
let
|
|
|
|
rop = rocky.store.readOptions
|
|
|
|
rit = rocky.store.db.rocksdb_create_iterator(rop)
|
|
|
|
defer:
|
|
|
|
rit.rocksdb_iter_destroy()
|
|
|
|
|
|
|
|
rit.rocksdb_iter_seek_to_first()
|
|
|
|
var count = -1
|
|
|
|
|
|
|
|
while rit.rocksdb_iter_valid() != 0:
|
|
|
|
count .inc
|
|
|
|
|
|
|
|
# Read key-value pair
|
|
|
|
var
|
|
|
|
kLen, vLen: csize_t
|
|
|
|
let
|
|
|
|
kData = rit.rocksdb_iter_key(addr kLen)
|
|
|
|
vData = rit.rocksdb_iter_value(addr vLen)
|
|
|
|
|
|
|
|
# Fetch data
|
|
|
|
let
|
|
|
|
key = if kData.isNil: EmptyBlob
|
|
|
|
else: kData.toOpenArrayByte(0,int(kLen)-1).toSeq
|
|
|
|
value = if vData.isNil: EmptyBlob
|
|
|
|
else: vData.toOpenArrayByte(0,int(vLen)-1).toSeq
|
|
|
|
|
|
|
|
yield (count, key, value)
|
|
|
|
|
|
|
|
# Update Iterator (might overwrite kData/vdata)
|
|
|
|
rit.rocksdb_iter_next()
|
|
|
|
# End while
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# End
|
|
|
|
# ------------------------------------------------------------------------------
|