web3-developer 947f629903
Fluffy State Bridge - State Gossip via Portal JSON-RPC (#2535)
* Create block offers queue and collect account preimages.

* Implement iterators to return account and storage proofs and bytecode from updatedCaches.

* Implement building offers from proofs.

* Refactor BlockDataRef type to only include required fields.

* Store block data in database.

* Improve state diff types.

* Implement start state backfill from specific block.

* Record last persisted block number in database.

* Persist account preimages in db.

* Apply state updates for DAO hard fork.

* Implement state gossip of block offers via portal JSON RPC.
2024-07-30 22:56:21 +08:00

222 lines
6.4 KiB
Nim

# Fluffy
# Copyright (c) 2024 Status Research & Development GmbH
# Licensed and distributed under either of
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
# at your option. This file may not be copied, modified, or distributed except according to those terms.
{.push raises: [].}
import std/[os, tables, sequtils], results, eth/trie/db, rocksdb
export results, db
const COL_FAMILY_NAME_ACCOUNTS = "A"
const COL_FAMILY_NAME_STORAGE = "S"
const COL_FAMILY_NAME_BYTECODE = "B"
const COL_FAMILY_NAME_PREIMAGES = "P"
const COL_FAMILY_NAMES = [
COL_FAMILY_NAME_ACCOUNTS, COL_FAMILY_NAME_STORAGE, COL_FAMILY_NAME_BYTECODE,
COL_FAMILY_NAME_PREIMAGES,
]
type
AccountsBackendRef = ref object of RootObj
cfHandle: ColFamilyHandleRef
tx: TransactionRef
updatedCache: TrieDatabaseRef
StorageBackendRef = ref object of RootObj
cfHandle: ColFamilyHandleRef
tx: TransactionRef
updatedCache: TrieDatabaseRef
BytecodeBackendRef = ref object of RootObj
cfHandle: ColFamilyHandleRef
tx: TransactionRef
updatedCache: TrieDatabaseRef
PreimagesBackendRef = ref object of RootObj
cfHandle: ColFamilyHandleRef
tx: TransactionRef
updatedCache: TrieDatabaseRef
DatabaseBackendRef =
AccountsBackendRef | StorageBackendRef | BytecodeBackendRef | PreimagesBackendRef
DatabaseRef* = ref object
rocksDb: OptimisticTxDbRef
pendingTransaction: TransactionRef
accountsBackend: AccountsBackendRef
storageBackend: StorageBackendRef
bytecodeBackend: BytecodeBackendRef
preimagesBackend: PreimagesBackendRef
proc init*(T: type DatabaseRef, baseDir: string): Result[T, string] =
let dbPath = baseDir / "db"
try:
createDir(dbPath)
except OSError, IOError:
return err("DatabaseRef: cannot create database directory")
let cfOpts = defaultColFamilyOptions(autoClose = true)
cfOpts.`compression=` Compression.lz4Compression
cfOpts.`bottommostCompression=` Compression.zstdCompression
let
db =
?openOptimisticTxDb(
dbPath,
columnFamilies = COL_FAMILY_NAMES.mapIt(initColFamilyDescriptor(it, cfOpts)),
)
accountsBackend = AccountsBackendRef(
cfHandle: db.getColFamilyHandle(COL_FAMILY_NAME_ACCOUNTS).get()
)
storageBackend =
StorageBackendRef(cfHandle: db.getColFamilyHandle(COL_FAMILY_NAME_STORAGE).get())
bytecodeBackend = BytecodeBackendRef(
cfHandle: db.getColFamilyHandle(COL_FAMILY_NAME_BYTECODE).get()
)
preimagesBackend = PreimagesBackendRef(
cfHandle: db.getColFamilyHandle(COL_FAMILY_NAME_PREIMAGES).get()
)
ok(
T(
rocksDb: db,
pendingTransaction: nil,
accountsBackend: accountsBackend,
storageBackend: storageBackend,
bytecodeBackend: bytecodeBackend,
preimagesBackend: preimagesBackend,
)
)
proc onData(data: openArray[byte]) {.gcsafe, raises: [].} =
discard # noop used to check if key exists
proc contains(
dbBackend: DatabaseBackendRef, key: openArray[byte]
): bool {.gcsafe, raises: [].} =
dbBackend.tx.get(key, onData, dbBackend.cfHandle).get()
proc put(
dbBackend: DatabaseBackendRef, key, val: openArray[byte]
) {.gcsafe, raises: [].} =
doAssert dbBackend.tx.put(key, val, dbBackend.cfHandle).isOk()
if not dbBackend.updatedCache.isNil():
dbBackend.updatedCache.put(key, val)
proc get(
dbBackend: DatabaseBackendRef, key: openArray[byte]
): seq[byte] {.gcsafe, raises: [].} =
if dbBackend.contains(key):
dbBackend.tx.get(key, dbBackend.cfHandle).get()
else:
@[]
proc del(
dbBackend: DatabaseBackendRef, key: openArray[byte]
): bool {.gcsafe, raises: [].} =
if dbBackend.contains(key):
doAssert dbBackend.tx.delete(key, dbBackend.cfHandle).isOk()
true
else:
false
proc getAccountsBackend*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
trieDB(db.accountsBackend)
proc getStorageBackend*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
trieDB(db.storageBackend)
proc getBytecodeBackend*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
trieDB(db.bytecodeBackend)
proc getPreimagesBackend*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
trieDB(db.preimagesBackend)
proc getAccountsUpdatedCache*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
db.accountsBackend.updatedCache
proc getStorageUpdatedCache*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
db.storageBackend.updatedCache
proc getBytecodeUpdatedCache*(db: DatabaseRef): TrieDatabaseRef {.inline.} =
db.bytecodeBackend.updatedCache
proc put*(db: DatabaseRef, key, val: openArray[byte]) =
let tx = db.rocksDb.beginTransaction()
defer:
tx.close()
# using default column family
doAssert tx.put(key, val).isOk()
doAssert tx.commit().isOk()
proc get*(db: DatabaseRef, key: openArray[byte]): seq[byte] =
let tx = db.rocksDb.beginTransaction()
defer:
tx.close()
# using default column family
if tx.get(key, onData).get():
tx.get(key).get()
else:
@[]
proc beginTransaction*(db: DatabaseRef): Result[void, string] =
if not db.pendingTransaction.isNil():
return err("DatabaseRef: Pending transaction already in progress")
let tx = db.rocksDb.beginTransaction()
db.pendingTransaction = tx
db.accountsBackend.tx = tx
db.storageBackend.tx = tx
db.bytecodeBackend.tx = tx
db.preimagesBackend.tx = tx
db.accountsBackend.updatedCache = newMemoryDB()
db.storageBackend.updatedCache = newMemoryDB()
db.bytecodeBackend.updatedCache = newMemoryDB()
db.preimagesBackend.updatedCache = nil # not used
ok()
proc commitTransaction*(db: DatabaseRef): Result[void, string] =
if db.pendingTransaction.isNil():
return err("DatabaseRef: No pending transaction")
?db.pendingTransaction.commit()
db.pendingTransaction.close()
db.pendingTransaction = nil
ok()
proc rollbackTransaction*(db: DatabaseRef): Result[void, string] =
if db.pendingTransaction.isNil():
return err("DatabaseRef: No pending transaction")
?db.pendingTransaction.rollback()
db.pendingTransaction.close()
db.pendingTransaction = nil
ok()
template withTransaction*(db: DatabaseRef, body: untyped): auto =
db.beginTransaction().expect("Transaction should be started")
try:
body
finally:
db.commitTransaction().expect("Transaction should be commited")
proc close*(db: DatabaseRef) =
if not db.pendingTransaction.isNil():
discard db.rollbackTransaction()
db.rocksDb.close()