2023-05-30 12:47:47 +01:00
|
|
|
# nimbus-eth1
|
Core db update storage root management for sub tries (#1964)
* Aristo: Re-phrase `LayerDelta` and `LayerFinal` as object references
why:
Avoids copying in some cases
* Fix copyright header
* Aristo: Verify `leafTie.root` function argument for `merge()` proc
why:
Zero root will lead to inconsistent DB entry
* Aristo: Update failure condition for hash labels compiler `hashify()`
why:
Node need not be rejected as long as links are on the schedule. In
that case, `redo[]` is to become `wff.base[]` at a later stage.
This amends an earlier fix, part of #1952 by also testing against
the target nodes of the `wff.base[]` sets.
* Aristo: Add storage root glue record to `hashify()` schedule
why:
An account leaf node might refer to a non-resolvable storage root ID.
Storage root node chains will end up at the storage root. So the link
`storage-root->account-leaf` needs an extra item in the schedule.
* Aristo: fix error code returned by `fetchPayload()`
details:
Final error code is implied by the error code form the `hikeUp()`
function.
* CoreDb: Discard `createOk` argument in API `getRoot()` function
why:
Not needed for the legacy DB. For the `Arsto` DB, a lazy approach is
implemented where a stprage root node is created on-the-fly.
* CoreDb: Prevent `$$` logging in some cases
why:
Logging the function `$$` is not useful when it is used for internal
use, i.e. retrieving an an error text for logging.
* CoreDb: Add `tryHashFn()` to API for pretty printing
why:
Pretty printing must not change the hashification status for the
`Aristo` DB. So there is an independent API wrapper for getting the
node hash which never updated the hashes.
* CoreDb: Discard `update` argument in API `hash()` function
why:
When calling the API function `hash()`, the latest state is always
wanted. For a version that uses the current state as-is without checking,
the function `tryHash()` was added to the backend.
* CoreDb: Update opaque vertex ID objects for the `Aristo` backend
why:
For `Aristo`, vID objects encapsulate a numeric `VertexID`
referencing a vertex (rather than a node hash as used on the
legacy backend.) For storage sub-tries, there might be no initial
vertex known when the descriptor is created. So opaque vertex ID
objects are supported without a valid `VertexID` which will be
initalised on-the-fly when the first item is merged.
* CoreDb: Add pretty printer for opaque vertex ID objects
* Cosmetics, printing profiling data
* CoreDb: Fix segfault in `Aristo` backend when creating MPT descriptor
why:
Missing initialisation error
* CoreDb: Allow MPT to inherit shared context on `Aristo` backend
why:
Creates descriptors with different storage roots for the same
shared `Aristo` DB descriptor.
* Cosmetics, update diagnostic message items for `Aristo` backend
* Fix Copyright year
2024-01-11 19:11:38 +00:00
|
|
|
# Copyright (c) 2023-2024 Status Research & Development GmbH
|
2023-05-30 12:47:47 +01:00
|
|
|
# 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.
|
|
|
|
|
2023-05-30 22:21:15 +01:00
|
|
|
## Aristo DB -- Patricia Trie builder, raw node insertion
|
|
|
|
## ======================================================
|
|
|
|
##
|
2023-10-27 22:36:51 +01:00
|
|
|
## This module merges `PathID` values as hexary lookup paths into the
|
2023-05-30 22:21:15 +01:00
|
|
|
## `Patricia Trie`. When changing vertices (aka nodes without Merkle hashes),
|
|
|
|
## associated (but separated) Merkle hashes will be deleted unless locked.
|
|
|
|
## Instead of deleting locked hashes error handling is applied.
|
|
|
|
##
|
|
|
|
## Also, nodes (vertices plus merkle hashes) can be added which is needed for
|
|
|
|
## boundary proofing after `snap/1` download. The vertices are split from the
|
|
|
|
## nodes and stored as-is on the table holding `Patricia Trie` entries. The
|
|
|
|
## hashes are stored iin a separate table and the vertices are labelled
|
|
|
|
## `locked`.
|
|
|
|
|
2023-05-30 12:47:47 +01:00
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
2024-06-18 11:14:02 +00:00
|
|
|
std/typetraits,
|
|
|
|
eth/common,
|
2023-09-12 19:45:12 +01:00
|
|
|
results,
|
2024-09-19 10:39:06 +02:00
|
|
|
"."/[aristo_desc, aristo_fetch, aristo_get, aristo_layers, aristo_vid]
|
2023-05-30 12:47:47 +01:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
|
|
|
|
proc layersPutLeaf(
|
|
|
|
db: AristoDbRef, rvid: RootedVertexID, path: NibblesBuf, payload: LeafPayload
|
|
|
|
): VertexRef =
|
|
|
|
let vtx = VertexRef(vType: Leaf, pfx: path, lData: payload)
|
|
|
|
db.layersPutVtx(rvid, vtx)
|
|
|
|
vtx
|
|
|
|
|
|
|
|
proc mergePayloadImpl(
|
|
|
|
db: AristoDbRef, # Database, top layer
|
|
|
|
root: VertexID, # MPT state root
|
|
|
|
path: openArray[byte], # Leaf item to add to the database
|
|
|
|
leaf: Opt[VertexRef],
|
|
|
|
payload: LeafPayload, # Payload value
|
|
|
|
): Result[(VertexRef, VertexRef, VertexRef), AristoError] =
|
|
|
|
## Merge the argument `(root,path)` key-value-pair into the top level vertex
|
|
|
|
## table of the database `db`. The `path` argument is used to address the
|
|
|
|
## leaf vertex with the payload. It is stored or updated on the database
|
|
|
|
## accordingly.
|
|
|
|
##
|
|
|
|
var
|
|
|
|
path = NibblesBuf.fromBytes(path)
|
|
|
|
cur = root
|
|
|
|
(vtx, _) = db.getVtxRc((root, cur)).valueOr:
|
|
|
|
if error != GetVtxNotFound:
|
|
|
|
return err(error)
|
|
|
|
|
|
|
|
# We're at the root vertex and there is no data - this must be a fresh
|
|
|
|
# VertexID!
|
|
|
|
return ok (db.layersPutLeaf((root, cur), path, payload), nil, nil)
|
|
|
|
steps: ArrayBuf[NibblesBuf.high + 1, VertexID]
|
|
|
|
|
|
|
|
template resetKeys() =
|
|
|
|
# Reset cached hashes of touched verticies
|
|
|
|
for i in 1..steps.len:
|
|
|
|
db.layersResKey((root, steps[^i]))
|
|
|
|
|
|
|
|
while path.len > 0:
|
|
|
|
# Clear existing merkle keys along the traversal path
|
|
|
|
steps.add cur
|
|
|
|
|
|
|
|
let n = path.sharedPrefixLen(vtx.pfx)
|
|
|
|
case vtx.vType
|
|
|
|
of Leaf:
|
|
|
|
let res =
|
|
|
|
if n == vtx.pfx.len:
|
|
|
|
# Same path - replace the current vertex with a new payload
|
|
|
|
|
|
|
|
if vtx.lData == payload:
|
|
|
|
return err(MergeNoAction)
|
|
|
|
|
|
|
|
let leafVtx = if root == VertexID(1):
|
|
|
|
var payload = payload.dup()
|
|
|
|
# TODO can we avoid this hack? it feels like the caller should already
|
|
|
|
# have set an appropriate stoID - this "fixup" feels risky,
|
|
|
|
# specially from a caching point of view
|
|
|
|
payload.stoID = vtx.lData.stoID
|
|
|
|
db.layersPutLeaf((root, cur), path, payload)
|
|
|
|
else:
|
|
|
|
db.layersPutLeaf((root, cur), path, payload)
|
|
|
|
(leafVtx, nil, nil)
|
|
|
|
else:
|
|
|
|
# Turn leaf into a branch (or extension) then insert the two leaves
|
|
|
|
# into the branch
|
|
|
|
let branch = VertexRef(vType: Branch, pfx: path.slice(0, n))
|
|
|
|
let other = block: # Copy of existing leaf node, now one level deeper
|
|
|
|
let local = db.vidFetch()
|
|
|
|
branch.bVid[vtx.pfx[n]] = local
|
|
|
|
db.layersPutLeaf((root, local), vtx.pfx.slice(n + 1), vtx.lData)
|
|
|
|
|
|
|
|
let leafVtx = block: # Newly inserted leaf node
|
|
|
|
let local = db.vidFetch()
|
|
|
|
branch.bVid[path[n]] = local
|
|
|
|
db.layersPutLeaf((root, local), path.slice(n + 1), payload)
|
|
|
|
|
|
|
|
# Put the branch at the vid where the leaf was
|
|
|
|
db.layersPutVtx((root, cur), branch)
|
|
|
|
|
|
|
|
# We need to return vtx here because its pfx member hasn't yet been
|
|
|
|
# sliced off and is therefore shared with the hike
|
|
|
|
(leafVtx, vtx, other)
|
|
|
|
|
|
|
|
resetKeys()
|
|
|
|
return ok(res)
|
|
|
|
of Branch:
|
|
|
|
if vtx.pfx.len == n:
|
|
|
|
# The existing branch is a prefix of the new entry
|
|
|
|
let
|
|
|
|
nibble = path[vtx.pfx.len]
|
|
|
|
next = vtx.bVid[nibble]
|
|
|
|
|
|
|
|
if next.isValid:
|
|
|
|
cur = next
|
|
|
|
path = path.slice(n + 1)
|
|
|
|
vtx =
|
|
|
|
if leaf.isSome and leaf[].isValid and leaf[].pfx == path:
|
|
|
|
leaf[]
|
|
|
|
else:
|
|
|
|
(?db.getVtxRc((root, next)))[0]
|
|
|
|
|
|
|
|
else:
|
|
|
|
# There's no vertex at the branch point - insert the payload as a new
|
|
|
|
# leaf and update the existing branch
|
|
|
|
let
|
|
|
|
local = db.vidFetch()
|
|
|
|
leafVtx = db.layersPutLeaf((root, local), path.slice(n + 1), payload)
|
|
|
|
brDup = vtx.dup()
|
|
|
|
|
|
|
|
brDup.bVid[nibble] = local
|
|
|
|
db.layersPutVtx((root, cur), brDup)
|
|
|
|
|
|
|
|
resetKeys()
|
|
|
|
return ok((leafVtx, nil, nil))
|
|
|
|
else:
|
|
|
|
# Partial path match - we need to split the existing branch at
|
|
|
|
# the point of divergence, inserting a new branch
|
|
|
|
let branch = VertexRef(vType: Branch, pfx: path.slice(0, n))
|
|
|
|
block: # Copy the existing vertex and add it to the new branch
|
|
|
|
let local = db.vidFetch()
|
|
|
|
branch.bVid[vtx.pfx[n]] = local
|
|
|
|
|
|
|
|
db.layersPutVtx(
|
|
|
|
(root, local),
|
|
|
|
VertexRef(vType: Branch, pfx: vtx.pfx.slice(n + 1), bVid: vtx.bVid),
|
|
|
|
)
|
|
|
|
|
|
|
|
let leafVtx = block: # add the new entry
|
|
|
|
let local = db.vidFetch()
|
|
|
|
branch.bVid[path[n]] = local
|
|
|
|
db.layersPutLeaf((root, local), path.slice(n + 1), payload)
|
|
|
|
|
|
|
|
db.layersPutVtx((root, cur), branch)
|
|
|
|
|
|
|
|
resetKeys()
|
|
|
|
return ok((leafVtx, nil, nil))
|
|
|
|
|
|
|
|
err(MergeHikeFailed)
|
2023-09-15 16:23:53 +01:00
|
|
|
|
2023-05-30 12:47:47 +01:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Public functions
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2024-06-27 09:01:26 +00:00
|
|
|
proc mergeAccountRecord*(
|
2023-07-04 19:24:03 +01:00
|
|
|
db: AristoDbRef; # Database, top layer
|
2024-10-01 21:03:10 +00:00
|
|
|
accPath: Hash32; # Even nibbled byte path
|
2024-06-27 09:01:26 +00:00
|
|
|
accRec: AristoAccount; # Account data
|
2024-06-18 11:14:02 +00:00
|
|
|
): Result[bool,AristoError] =
|
2024-07-12 13:12:25 +00:00
|
|
|
## Merge the key-value-pair argument `(accKey,accRec)` as an account
|
2024-06-18 11:14:02 +00:00
|
|
|
## ledger value, i.e. the the sub-tree starting at `VertexID(1)`.
|
2024-02-01 21:27:48 +00:00
|
|
|
##
|
2024-07-12 13:12:25 +00:00
|
|
|
## On success, the function returns `true` if the `accRec` argument was
|
|
|
|
## not on the database already or different from `accRec`, and `false`
|
|
|
|
## otherwise.
|
2024-06-18 19:30:01 +00:00
|
|
|
##
|
2024-06-18 11:14:02 +00:00
|
|
|
let
|
2024-07-14 12:02:05 +02:00
|
|
|
pyl = LeafPayload(pType: AccountData, account: accRec)
|
2024-09-19 10:39:06 +02:00
|
|
|
updated = db.mergePayloadImpl(
|
|
|
|
VertexID(1), accPath.data, db.cachedAccLeaf(accPath), pyl).valueOr:
|
|
|
|
if error == MergeNoAction:
|
|
|
|
return ok false
|
|
|
|
return err(error)
|
|
|
|
|
|
|
|
# Update leaf cache both of the merged value and potentially the displaced
|
|
|
|
# leaf resulting from splitting a leaf into a branch with two leaves
|
|
|
|
db.layersPutAccLeaf(accPath, updated[0])
|
|
|
|
if updated[1].isValid:
|
2024-09-29 14:37:09 +02:00
|
|
|
let otherPath = Hash32(getBytes(
|
2024-09-19 10:39:06 +02:00
|
|
|
NibblesBuf.fromBytes(accPath.data).replaceSuffix(updated[1].pfx)))
|
|
|
|
db.layersPutAccLeaf(otherPath, updated[2])
|
2024-02-01 21:27:48 +00:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
ok true
|
2023-09-15 16:23:53 +01:00
|
|
|
|
2024-06-18 11:14:02 +00:00
|
|
|
proc mergeGenericData*(
|
2023-09-15 16:23:53 +01:00
|
|
|
db: AristoDbRef; # Database, top layer
|
|
|
|
root: VertexID; # MPT state root
|
|
|
|
path: openArray[byte]; # Leaf item to add to the database
|
2023-11-08 12:18:32 +00:00
|
|
|
data: openArray[byte]; # Raw data payload value
|
2023-09-15 16:23:53 +01:00
|
|
|
): Result[bool,AristoError] =
|
2024-06-18 11:14:02 +00:00
|
|
|
## Variant of `mergeXXX()` for generic sub-trees, i.e. for arguments
|
|
|
|
## `root` greater than `VertexID(1)` and smaller than `LEAST_FREE_VID`.
|
|
|
|
##
|
2024-06-18 19:30:01 +00:00
|
|
|
## On success, the function returns `true` if the `data` argument was merged
|
|
|
|
## into the database ot updated, and `false` if it was on the database
|
|
|
|
## already.
|
|
|
|
##
|
2024-06-18 11:14:02 +00:00
|
|
|
# Verify that `root` is neither an accounts tree nor a strorage tree.
|
|
|
|
if not root.isValid:
|
|
|
|
return err(MergeRootVidMissing)
|
|
|
|
elif root == VertexID(1):
|
|
|
|
return err(MergeAccRootNotAccepted)
|
|
|
|
elif LEAST_FREE_VID <= root.distinctBase:
|
|
|
|
return err(MergeStoRootNotAccepted)
|
|
|
|
|
|
|
|
let
|
2024-07-14 12:02:05 +02:00
|
|
|
pyl = LeafPayload(pType: RawData, rawBlob: @data)
|
2024-02-01 21:27:48 +00:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
discard db.mergePayloadImpl(root, path, Opt.none(VertexRef), pyl).valueOr:
|
|
|
|
if error == MergeNoAction:
|
|
|
|
return ok false
|
|
|
|
return err error
|
|
|
|
|
|
|
|
ok true
|
2024-06-18 11:14:02 +00:00
|
|
|
|
|
|
|
proc mergeStorageData*(
|
2024-02-01 21:27:48 +00:00
|
|
|
db: AristoDbRef; # Database, top layer
|
2024-10-01 21:03:10 +00:00
|
|
|
accPath: Hash32; # Needed for accounts payload
|
|
|
|
stoPath: Hash32; # Storage data path (aka key)
|
2024-08-14 08:54:44 +00:00
|
|
|
stoData: UInt256; # Storage data payload value
|
2024-06-27 19:21:01 +00:00
|
|
|
): Result[void,AristoError] =
|
|
|
|
## Store the `stoData` data argument on the storage area addressed by
|
|
|
|
## `(accPath,stoPath)` where `accPath` is the account key (into the MPT)
|
|
|
|
## and `stoPath` is the slot path of the corresponding storage area.
|
2024-06-18 11:14:02 +00:00
|
|
|
##
|
2024-09-19 10:39:06 +02:00
|
|
|
var accHike: Hike
|
|
|
|
db.fetchAccountHike(accPath,accHike).isOkOr:
|
|
|
|
return err(MergeStoAccMissing)
|
2024-06-18 11:14:02 +00:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
let
|
|
|
|
stoID = accHike.legs[^1].wp.vtx.lData.stoID
|
2024-07-11 13:26:46 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
# Provide new storage ID when needed
|
|
|
|
useID =
|
|
|
|
if stoID.isValid: stoID # Use as is
|
|
|
|
elif stoID.vid.isValid: (true, stoID.vid) # Re-use previous vid
|
|
|
|
else: (true, db.vidFetch()) # Create new vid
|
|
|
|
mixPath = mixUp(accPath, stoPath)
|
|
|
|
# Call merge
|
|
|
|
pyl = LeafPayload(pType: StoData, stoData: stoData)
|
|
|
|
updated = db.mergePayloadImpl(
|
|
|
|
useID.vid, stoPath.data, db.cachedStoLeaf(mixPath), pyl).valueOr:
|
|
|
|
if error == MergeNoAction:
|
|
|
|
assert stoID.isValid # debugging only
|
|
|
|
return ok()
|
2024-07-11 13:26:46 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
return err(error)
|
2024-07-14 19:12:10 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
# Mark account path Merkle keys for update
|
|
|
|
db.layersResKeys(accHike)
|
2024-07-12 15:08:26 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
# Update leaf cache both of the merged value and potentially the displaced
|
|
|
|
# leaf resulting from splitting a leaf into a branch with two leaves
|
|
|
|
db.layersPutStoLeaf(mixPath, updated[0])
|
2024-07-11 13:26:46 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
if updated[1].isValid:
|
2024-09-29 14:37:09 +02:00
|
|
|
let otherPath = Hash32(getBytes(
|
2024-09-19 10:39:06 +02:00
|
|
|
NibblesBuf.fromBytes(stoPath.data).replaceSuffix(updated[1].pfx)))
|
|
|
|
db.layersPutStoLeaf(mixUp(accPath, otherPath), updated[2])
|
2024-07-11 13:26:46 +02:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
if not stoID.isValid:
|
|
|
|
# Make sure that there is an account that refers to that storage trie
|
|
|
|
let leaf = accHike.legs[^1].wp.vtx.dup # Dup on modify
|
|
|
|
leaf.lData.stoID = useID
|
|
|
|
db.layersPutAccLeaf(accPath, leaf)
|
|
|
|
db.layersPutVtx((VertexID(1), accHike.legs[^1].wp.vid), leaf)
|
2024-06-18 11:14:02 +00:00
|
|
|
|
2024-09-19 10:39:06 +02:00
|
|
|
ok()
|
2023-06-09 12:17:37 +01:00
|
|
|
|
2023-05-30 12:47:47 +01:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# End
|
|
|
|
# ------------------------------------------------------------------------------
|