mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-28 13:05:18 +00:00
58cde36656
This kind of data is not used except in tests where it is used only to create databases that don't match actual usage of aristo. Removing simplifies future optimizations that can focus on processing specific leaf types more efficiently. A casualty of this removal is some test code as well as some proof generation code that is unused - on the surface, it looks like it should be possible to port both of these to the more specific data types - doing so would ensure that a database written by one part of the codebase can interact with the other - as it stands, there is confusion on this point since using the proof generation code will result in a database of a shape that is incompatible with the rest of eth1.
310 lines
9.5 KiB
Nim
310 lines
9.5 KiB
Nim
# nimbus-eth1
|
|
# 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
|
|
std/[sets, sequtils],
|
|
eth/common,
|
|
results,
|
|
".."/[aristo_desc, aristo_get, aristo_vid],
|
|
./part_desc
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Private helpers
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc read(rlp: var Rlp; T: type PrfNode): T {.gcsafe, raises: [RlpError].} =
|
|
## Mixin for RLP reader. The decoder with error return code in a `Leaf`
|
|
## node if needed.
|
|
##
|
|
func readError(error: AristoError): PrfNode =
|
|
## Prettify return code expression
|
|
PrfNode(vtx: VertexRef(vType: Leaf), prfType: isError, error: error)
|
|
|
|
if not rlp.isList:
|
|
# Otherwise `rlp.items` would raise a `Defect`
|
|
return readError(PartRlp2Or17ListEntries)
|
|
|
|
var
|
|
blobs = newSeq[seq[byte]](2) # temporary, cache
|
|
links: array[16,HashKey] # reconstruct branch node
|
|
top = 0 # count entries and positions
|
|
|
|
# Collect lists of either 2 or 17 blob entries.
|
|
for w in rlp.items:
|
|
case top
|
|
of 0, 1:
|
|
if not w.isBlob:
|
|
return readError(PartRlpBlobExpected)
|
|
blobs[top] = rlp.read(seq[byte])
|
|
of 2 .. 15:
|
|
let blob = rlp.read(seq[byte])
|
|
links[top] = HashKey.fromBytes(blob).valueOr:
|
|
return readError(PartRlpBranchHashKeyExpected)
|
|
of 16:
|
|
if not w.isBlob or 0 < rlp.read(seq[byte]).len:
|
|
return readError(PartRlpEmptyBlobExpected)
|
|
else:
|
|
return readError(PartRlp2Or17ListEntries)
|
|
top.inc
|
|
|
|
# Verify extension data
|
|
case top
|
|
of 2:
|
|
if blobs[0].len == 0:
|
|
return readError(PartRlpNonEmptyBlobExpected)
|
|
let (isLeaf, pathSegment) = NibblesBuf.fromHexPrefix blobs[0]
|
|
if isLeaf:
|
|
return PrfNode(
|
|
prfType: ignore, )
|
|
|
|
# TODO interpret the blob (?)
|
|
# vtx: VertexRef(
|
|
# vType: Leaf,
|
|
# pfx: pathSegment,
|
|
# lData: LeafPayload(
|
|
# pType: RawData,
|
|
# rawBlob: blobs[1])))
|
|
else:
|
|
var node = PrfNode(
|
|
prfType: isExtension,
|
|
vtx: VertexRef(
|
|
vType: Branch,
|
|
pfx: pathSegment))
|
|
node.key[0] = HashKey.fromBytes(blobs[1]).valueOr:
|
|
return readError(PartRlpExtHashKeyExpected)
|
|
return node
|
|
of 17:
|
|
for n in [0,1]:
|
|
links[n] = HashKey.fromBytes(blobs[n]).valueOr:
|
|
return readError(PartRlpBranchHashKeyExpected)
|
|
return PrfNode(
|
|
vtx: VertexRef(
|
|
vType: Branch),
|
|
prfType: ignore,
|
|
key: links)
|
|
else:
|
|
discard
|
|
|
|
readError(PartRlp2Or17ListEntries)
|
|
|
|
proc read(rlp: var Rlp; T: type PrfPayload): T {.gcsafe, raises: [RlpError].} =
|
|
## Mixin for RLP reader decoding `Account` or storage slot payload.
|
|
##
|
|
case rlp.listLen:
|
|
of 1:
|
|
result.prfType = isStoValue
|
|
result.num = rlp.read UInt256
|
|
of 4:
|
|
result.prfType = isAccount
|
|
result.acc = rlp.read Account
|
|
else:
|
|
result.prfType = isError
|
|
result.error = PartRlp1r4ListEntries
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public functions
|
|
# ------------------------------------------------------------------------------
|
|
|
|
func toNodesTab*(
|
|
proof: openArray[seq[byte]]; # List of RLP encoded nodes
|
|
mode: PartStateMode; # How to decode `Leaf` nodes
|
|
): Result[TableRef[HashKey,PrfNode],AristoError] =
|
|
## Convert RLP encoded argument list `proof` to a nodes table indexed by
|
|
## the `HashKey` values of the `proof` list entries.
|
|
var
|
|
exts: Table[HashKey,PrfNode] # need to be processed separately
|
|
nodes = newTable[HashKey,PrfNode]()
|
|
|
|
# populate tables
|
|
for w in proof:
|
|
# Decode blob `w`
|
|
let nd = block:
|
|
try: rlp.decode(w, PrfNode)
|
|
except RlpError:
|
|
return err(PartRlpNodeException)
|
|
|
|
case nd.prfType:
|
|
of isExtension:
|
|
# For a moment, hold extensions on a separate cache
|
|
exts[w.digestTo HashKey] = nd
|
|
|
|
of ignore:
|
|
# Store `Branch` and `Leaf` nodes in final lookup table
|
|
nodes[w.digestTo HashKey] = nd
|
|
|
|
# Special decoding for account `Leaf` nodes
|
|
if nd.vtx.vType == Leaf and mode != ForceGenericPayload:
|
|
# Decode payload to deficated format for storage or accounts
|
|
var pyl: PrfPayload
|
|
try:
|
|
# TODO interpret the blob
|
|
# pyl = rlp.decode(nd.vtx.lData.rawBlob, PrfPayload)
|
|
pyl = PrfPayload(prfType: isError, error: PartRlpPayloadException)
|
|
except RlpError:
|
|
pyl = PrfPayload(prfType: isError, error: PartRlpPayloadException)
|
|
|
|
case pyl.prfType:
|
|
of isStoValue:
|
|
# Single value encoding might not be unique so it cannot be
|
|
# automatically detected
|
|
if mode != AutomaticPayload:
|
|
nd.vtx.lData = LeafPayload(pType: StoData, stoData: pyl.num)
|
|
of isAccount:
|
|
nd.key[0] = pyl.acc.storageRoot.to(HashKey)
|
|
nd.vtx.lData = LeafPayload(
|
|
pType: AccountData,
|
|
account: AristoAccount(
|
|
nonce: pyl.acc.nonce,
|
|
balance: pyl.acc.balance,
|
|
codeHash: pyl.acc.codeHash))
|
|
elif mode == AutomaticPayload:
|
|
discard
|
|
else:
|
|
return err(pyl.error)
|
|
else:
|
|
return err(nd.error)
|
|
|
|
# Postprocess extension nodes
|
|
for (xKey,xNode) in exts.pairs:
|
|
# Combine `xNode + nodes[w.ePfx]`
|
|
let nd = nodes.getOrDefault xNode.key[0]
|
|
if nd.isNil:
|
|
# Need to store raw extension
|
|
nodes[xKey] = xNode
|
|
continue
|
|
if nd.vtx.pfx.len != 0:
|
|
return err(PartGarbledExtsInProofs)
|
|
# Move extended `nd` branch node
|
|
nd.prfType = ignore
|
|
nd.vtx.pfx = xNode.vtx.pfx
|
|
nodes.del xNode.key[0]
|
|
nodes[xKey] = nd
|
|
|
|
ok nodes
|
|
|
|
|
|
proc backLinks*(nTab: TableRef[HashKey,PrfNode]): PrfBackLinks =
|
|
## tuple[chains: seq[seq[HashKey]], links: Table[HashKey,HashKey]] =
|
|
## Classify argument table
|
|
##
|
|
## * chains: key list of back chains
|
|
## * links: `(child,parent)` lookup table
|
|
##
|
|
new result
|
|
|
|
# Collect predecessor list
|
|
for (key,nd) in nTab.pairs:
|
|
if nd.vtx.vType == Leaf:
|
|
if nd.vtx.lData.pType == AccountData and nd.key[0].isValid:
|
|
result.links[nd.key[0]] = key
|
|
elif nd.prfType == isExtension:
|
|
result.links[nd.key[0]] = key
|
|
else:
|
|
for w in nd.key:
|
|
if w.isValid:
|
|
result.links[w] = key
|
|
|
|
# Compute leafs list, i.e. keys without children in `nTab[]`
|
|
var leafs = nTab.keys.toSeq.toHashSet
|
|
for (child,parent) in result.links.pairs:
|
|
if child in nTab:
|
|
leafs.excl parent # `parent` has `child` => not a leaf
|
|
|
|
# Compute chains starting at leafs
|
|
for leaf in leafs:
|
|
var q = @[leaf]
|
|
while true:
|
|
let up = result.links.getOrVoid q[^1]
|
|
if up.isValid:
|
|
q.add up
|
|
else:
|
|
break
|
|
result.chains.add q
|
|
|
|
|
|
proc getTreeRootVid*(
|
|
ps: PartStateRef;
|
|
key: HashKey;
|
|
): Result[VertexID,AristoError] =
|
|
## Find root ID in `ps[]` or create a new ID
|
|
##
|
|
# Use root from `ps` descriptor
|
|
let rvid = ps[key]
|
|
if rvid.isValid:
|
|
return ok(rvid.vid)
|
|
|
|
# Try next free VID
|
|
for n in 2 .. LEAST_FREE_VID:
|
|
let rvid = (VertexID(n),VertexID(n))
|
|
if not ps.db.getVtx(rvid).isValid:
|
|
ps[key] = rvid
|
|
return ok(VertexID n)
|
|
|
|
err(PartNoMoreRootVidsLeft)
|
|
|
|
|
|
proc getRvid*(
|
|
ps: PartStateRef;
|
|
root: VertexID;
|
|
key: HashKey;
|
|
): Result[tuple[rvid: RootedVertexID, fromStateDb: bool],AristoError] =
|
|
## Find key in `ps[]` or create a new key. the value `onStateDb` is
|
|
## return `false` if a new entry was created.
|
|
##
|
|
var (rvid, fromStateDb) = (ps[key], true)
|
|
if not rvid.isValid:
|
|
# Create new one
|
|
(rvid, fromStateDb) = ((root, ps.db.vidFetch()), false)
|
|
ps[key] = rvid
|
|
elif root != rvid.root:
|
|
# Oops
|
|
return err(PartRootVidsDontMatch)
|
|
ok((rvid, fromStateDb))
|
|
|
|
|
|
proc updateAccountsTree*(
|
|
ps: PartStateRef; # Partial database descriptor
|
|
nodes: TableRef[HashKey,PrfNode]; # Node lookup table
|
|
bl: PrfBackLinks; # Uplink lists
|
|
mode: PartStateMode; # Try accounts, otherwise generic
|
|
): Result[void,AristoError] =
|
|
## Check wether the chain has an accounts leaf node and update the
|
|
## argument descriptor `ps` accordingly.
|
|
##
|
|
if mode == ForceGenericPayload or VertexID(1) in ps.core:
|
|
return ok()
|
|
|
|
var accRootKey = VOID_HASH_KEY
|
|
for chain in bl.chains:
|
|
for key in chain:
|
|
nodes[].withValue(key,node):
|
|
if node.vtx.vType == Leaf and node.vtx.lData.pType == AccountData:
|
|
|
|
# Ok, got an accounts leaf node
|
|
if not accRootKey.isValid:
|
|
# Register accounts root
|
|
accRootKey = chain[^1]
|
|
ps[accRootKey] = (VertexID(1),VertexID(1))
|
|
elif accRootKey != chain[^1]:
|
|
# Two account chains with different root keys
|
|
return err(PartRootKeysDontMatch)
|
|
|
|
# Register storage root (if any)
|
|
if node.key[0].isValid:
|
|
let vid = ps.db.vidFetch()
|
|
ps[node.key[0]] = (vid,vid)
|
|
ok()
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# End
|
|
# ------------------------------------------------------------------------------
|