mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-09 22:06:21 +00:00
6bcefc0e42
* fix state db lookup typo * fix randao reveal slot when proposing blocks * only store blocks that can be applied to a state * store state at every epoch boundary (yes, needs pruning!) * split out state advancement function when there's no block * default state sim to 0.9 attestation ratio
129 lines
4.7 KiB
Nim
129 lines
4.7 KiB
Nim
import
|
|
os, json, tables, options,
|
|
chronicles, json_serialization, eth/common/eth_types_json_serialization,
|
|
spec/[datatypes, digest, crypto],
|
|
eth/trie/db, ssz
|
|
|
|
type
|
|
BeaconChainDB* = ref object
|
|
backend: TrieDatabaseRef
|
|
|
|
DbKeyKind = enum
|
|
kHashToState
|
|
kHashToBlock
|
|
kHeadBlock # Pointer to the most recent block seen
|
|
kTailBlock # Pointer to the earliest finalized block
|
|
kSlotToBlockRoots
|
|
|
|
func subkey(kind: DbKeyKind): array[1, byte] =
|
|
result[0] = byte ord(kind)
|
|
|
|
func subkey[N: static int](kind: DbKeyKind, key: array[N, byte]):
|
|
array[N + 1, byte] =
|
|
result[0] = byte ord(kind)
|
|
result[1 .. ^1] = key
|
|
|
|
func subkey(kind: DbKeyKind, key: uint64): array[sizeof(key) + 1, byte] =
|
|
result[0] = byte ord(kind)
|
|
copyMem(addr result[1], unsafeAddr key, sizeof(key))
|
|
|
|
func subkey(kind: type BeaconState, key: Eth2Digest): auto =
|
|
subkey(kHashToState, key.data)
|
|
|
|
func subkey(kind: type BeaconBlock, key: Eth2Digest): auto =
|
|
subkey(kHashToBlock, key.data)
|
|
|
|
proc init*(T: type BeaconChainDB, backend: TrieDatabaseRef): BeaconChainDB =
|
|
new result
|
|
result.backend = backend
|
|
|
|
proc toSeq(v: openarray[byte], ofType: type): seq[ofType] =
|
|
if v.len != 0:
|
|
assert(v.len mod sizeof(ofType) == 0)
|
|
let sz = v.len div sizeof(ofType)
|
|
result = newSeq[ofType](sz)
|
|
copyMem(addr result[0], unsafeAddr v[0], v.len)
|
|
|
|
proc putBlock*(db: BeaconChainDB, key: Eth2Digest, value: BeaconBlock) =
|
|
let slotKey = subkey(kSlotToBlockRoots, value.slot)
|
|
var blockRootsBytes = db.backend.get(slotKey)
|
|
var blockRoots = blockRootsBytes.toSeq(Eth2Digest)
|
|
if key notin blockRoots:
|
|
db.backend.put(subkey(type value, key), ssz.serialize(value))
|
|
blockRootsBytes.setLen(blockRootsBytes.len + sizeof(key))
|
|
copyMem(addr blockRootsBytes[^sizeof(key)], unsafeAddr key, sizeof(key))
|
|
db.backend.put(slotKey, blockRootsBytes)
|
|
|
|
proc putHead*(db: BeaconChainDB, key: Eth2Digest) =
|
|
db.backend.put(subkey(kHeadBlock), key.data) # TODO head block?
|
|
|
|
proc putState*(db: BeaconChainDB, key: Eth2Digest, value: BeaconState) =
|
|
# TODO: prune old states
|
|
# TODO: it might be necessary to introduce the concept of a "last finalized
|
|
# state" to the storage, so that clients with limited storage have
|
|
# a natural state to start recovering from. One idea is to keep a
|
|
# special pointer to the state that has ben finalized, and prune all
|
|
# other states.
|
|
# One issue is that what will become a finalized is revealed only
|
|
# long after that state has passed, meaning that we need to keep
|
|
# a history of "finalized state candidates" or possibly replay from
|
|
# the previous finalized state, if we have that stored. To consider
|
|
# here is that the gap between finalized and present state might be
|
|
# significant (days), meaning replay might be expensive.
|
|
db.backend.put(subkey(type value, key), ssz.serialize(value))
|
|
|
|
proc putState*(db: BeaconChainDB, value: BeaconState) =
|
|
db.putState(hash_tree_root_final(value), value)
|
|
|
|
proc putBlock*(db: BeaconChainDB, value: BeaconBlock) =
|
|
db.putBlock(hash_tree_root_final(value), value)
|
|
|
|
proc putHeadBlock*(db: BeaconChainDB, key: Eth2Digest) =
|
|
db.backend.put(subkey(kHeadBlock), key.data) # TODO head block?
|
|
|
|
proc putTailBlock*(db: BeaconChainDB, key: Eth2Digest) =
|
|
db.backend.put(subkey(kTailBlock), key.data)
|
|
|
|
proc get(db: BeaconChainDB, key: auto, T: typedesc): Option[T] =
|
|
let res = db.backend.get(key)
|
|
if res.len != 0:
|
|
ssz.deserialize(res, T)
|
|
else:
|
|
none(T)
|
|
|
|
proc getBlock*(db: BeaconChainDB, key: Eth2Digest): Option[BeaconBlock] =
|
|
db.get(subkey(BeaconBlock, key), BeaconBlock)
|
|
|
|
proc getState*(db: BeaconChainDB, key: Eth2Digest): Option[BeaconState] =
|
|
db.get(subkey(BeaconState, key), BeaconState)
|
|
|
|
proc getHeadBlock*(db: BeaconChainDB): Option[Eth2Digest] =
|
|
db.get(subkey(kHeadBlock), Eth2Digest)
|
|
|
|
proc getTailBlock*(db: BeaconChainDB): Option[Eth2Digest] =
|
|
db.get(subkey(kTailBlock), Eth2Digest)
|
|
|
|
proc getBlockRootsForSlot*(db: BeaconChainDB, slot: uint64): seq[Eth2Digest] =
|
|
db.backend.get(subkey(kSlotToBlockRoots, slot)).toSeq(Eth2Digest)
|
|
|
|
proc containsBlock*(
|
|
db: BeaconChainDB, key: Eth2Digest): bool =
|
|
db.backend.contains(subkey(BeaconBlock, key))
|
|
|
|
proc containsState*(
|
|
db: BeaconChainDB, key: Eth2Digest): bool =
|
|
db.backend.contains(subkey(BeaconState, key))
|
|
|
|
iterator getAncestors*(db: BeaconChainDB, root: Eth2Digest):
|
|
tuple[root: Eth2Digest, blck: BeaconBlock] =
|
|
## Load a chain of ancestors for blck - returns a list of blocks with the
|
|
## oldest block last (blck will be at result[0]).
|
|
##
|
|
## The search will go on until the ancestor cannot be found.
|
|
|
|
var root = root
|
|
while (let blck = db.getBlock(root); blck.isSome()):
|
|
yield (root, blck.get())
|
|
|
|
root = blck.get().parent_root
|