mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-10 04:15:54 +00:00
3e88589eb1
* Split off `ReadOnlyStateDB` from `AccountStateDB` from `state_db.nim` why: Apart from testing, applications use `ReadOnlyStateDB` as an easy way to access the accounts ledger. This is well supported by the `Aristo` db, but writable mode is only parially supported. The writable AccountStateDB` object for modifying accounts is not used by production code. So, for lecgacy and testing apps, the full support of the previous `AccountStateDB` is now enabled by `import db/state_db/read_write` and the `import db/state_db` provides read-only mode. * Encapsulate `AccountStateDB` as `GenesisLedgerRef` or genesis creation why: `AccountStateDB` has poor support for `Aristo` and is not widely used in favour of `AccountsLedger` (which will be abstracted as `ledger`.) Currently, using other than the `AccountStateDB` ledgers within the `GenesisLedgerRef` wrapper is experimental and test only. Eventually, the wrapper should disappear so that the `Ledger` object (which encapsulates `AccountsCache` and `AccountsLedger`) will prevail. * For the `Ledger`, provide access to raw accounts `MPT` why: This gives to the `CoreDbMptRef` descriptor from the `CoreDb` (which is the legacy version of CoreDxMptRef`.) For the new `ledger` API, the accounts are based on the `CoreDxMAccRef` descriptor which uses a particular sub-system for accounts while legacy applications use the `CoreDbPhkRef` equivalent of the `SecureHexaryTrie`. The only place where this feature will currently be used is the `genesis.nim` source file. * Fix `Aristo` bugs, missing boundary checks, typos, etc. * Verify root vertex in `MPT` and account constructors why: Was missing so far, in particular the accounts constructor must verify `VertexID(1) * Fix include file
1015 lines
36 KiB
Nim
1015 lines
36 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2018-2023 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/typetraits,
|
|
chronicles,
|
|
eth/common,
|
|
results,
|
|
"../.."/[constants, errors],
|
|
../aristo/aristo_constants, # `EmptyBlob`
|
|
./base/[base_desc, validate]
|
|
|
|
export
|
|
CoreDbAccBackendRef,
|
|
CoreDbAccount,
|
|
CoreDbApiError,
|
|
CoreDbBackendRef,
|
|
CoreDbCaptFlags,
|
|
CoreDbErrorCode,
|
|
CoreDbErrorRef,
|
|
CoreDbKvtBackendRef,
|
|
CoreDbMptBackendRef,
|
|
CoreDbPersistentTypes,
|
|
CoreDbRef,
|
|
CoreDbSaveFlags,
|
|
CoreDbType,
|
|
CoreDbVidRef,
|
|
CoreDxAccRef,
|
|
CoreDxCaptRef,
|
|
CoreDxKvtRef,
|
|
CoreDxMptRef,
|
|
CoreDxPhkRef,
|
|
CoreDxTxRef
|
|
|
|
when defined(release):
|
|
const AutoValidateDescriptors = false
|
|
else:
|
|
const AutoValidateDescriptors = true
|
|
|
|
const
|
|
ProvideCoreDbLegacyAPI* = true # and false
|
|
|
|
EnableApiTracking = true # and false
|
|
## When enabled, functions using this tracking facility need to import
|
|
## `chronicles`, as well. Tracking is enabled by setting the `trackLegaApi`
|
|
## and/or the `trackNewApi` flags to `true`.
|
|
|
|
# Annotation helpers
|
|
{.pragma: noRaise, gcsafe, raises: [].}
|
|
{.pragma: apiRaise, gcsafe, raises: [CoreDbApiError].}
|
|
{.pragma: catchRaise, gcsafe, raises: [CatchableError].}
|
|
|
|
when ProvideCoreDbLegacyAPI:
|
|
type
|
|
TxWrapperApiError* = object of CoreDbApiError
|
|
## For re-routing exception on tx/action template
|
|
|
|
CoreDbKvtRef* = distinct CoreDxKvtRef ## Let methods defect on error
|
|
CoreDbMptRef* = distinct CoreDxMptRef ## ...
|
|
CoreDbPhkRef* = distinct CoreDxPhkRef
|
|
CoreDbTxRef* = distinct CoreDxTxRef
|
|
CoreDbTxID* = distinct CoreDxTxID
|
|
CoreDbCaptRef* = distinct CoreDxCaptRef
|
|
|
|
CoreDbTrieRefs* = CoreDbMptRef | CoreDbPhkRef
|
|
## Shortcut, *MPT* modules for (legacy API)
|
|
|
|
CoreDbChldRefs* = CoreDbKvtRef | CoreDbTrieRefs | CoreDbTxRef | CoreDbTxID |
|
|
CoreDbCaptRef
|
|
## Shortcut, all modules with a `parent` entry (for legacy API)
|
|
|
|
type
|
|
CoreDxTrieRefs = CoreDxMptRef | CoreDxPhkRef | CoreDxAccRef
|
|
## Shortcut, *MPT* descriptors
|
|
|
|
CoreDxTrieRelated = CoreDxTrieRefs | CoreDxTxRef | CoreDxTxID | CoreDxCaptRef
|
|
## Shortcut, descriptors for sub-modules running on an *MPT*
|
|
|
|
CoreDbBackends = CoreDbBackendRef | CoreDbKvtBackendRef |
|
|
CoreDbMptBackendRef | CoreDbAccBackendRef
|
|
## Shortcut, all backend descriptors.
|
|
|
|
CoreDxChldRefs = CoreDxKvtRef | CoreDxTrieRelated | CoreDbVidRef |
|
|
CoreDbBackends | CoreDbErrorRef
|
|
## Shortcut, all descriptors with a `parent` entry.
|
|
|
|
proc `$$`*(e: CoreDbErrorRef): string {.gcsafe.}
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Private helpers
|
|
# ------------------------------------------------------------------------------
|
|
|
|
template logTxt(info: static[string]): static[string] =
|
|
"CoreDb " & info
|
|
|
|
template itNotImplemented(db: CoreDbRef, name: string) =
|
|
warn logTxt "iterator not implemented", dbType=db.dbType, meth=name
|
|
|
|
# ---------
|
|
|
|
when EnableApiTracking:
|
|
import std/[sequtils, strutils], stew/byteutils
|
|
{.warning: "*** Provided API logging for CoreDB (disabled by default)".}
|
|
|
|
func getParent(w: CoreDxChldRefs): auto =
|
|
## Avoida inifinite call to `parent()` in `ifTrack*Api()` tmplates
|
|
w.parent
|
|
|
|
when ProvideCoreDbLegacyAPI:
|
|
template legaApiTxt(info: static[string]): static[string] =
|
|
logTxt "legacy API " & info
|
|
|
|
template setTrackLegaApiOnly(w: CoreDbChldRefs|CoreDbRef) =
|
|
when typeof(w) is CoreDbRef:
|
|
let db = w
|
|
else:
|
|
let db = w.distinctBase.getParent
|
|
let save = db.trackNewApi
|
|
# Prevent from cascaded logging
|
|
db.trackNewApi = false
|
|
defer: db.trackNewApi = save
|
|
|
|
template ifTrackLegaApi(w: CoreDbChldRefs|CoreDbRef; code: untyped) =
|
|
block:
|
|
when typeof(w) is CoreDbRef:
|
|
let db = w
|
|
else:
|
|
let db = w.distinctBase.getParent
|
|
if db.trackLegaApi:
|
|
code
|
|
|
|
proc toStr(w: CoreDbKvtRef): string =
|
|
if w.distinctBase.isNil: "kvtRef(nil)" else: "kvtRef"
|
|
|
|
# End LegacyAPI
|
|
|
|
template newApiTxt(info: static[string]): static[string] =
|
|
logTxt "new API " & info
|
|
|
|
template ifTrackNewApi(w: CoreDxChldRefs|CoreDbRef; code: untyped) =
|
|
block:
|
|
when typeof(w) is CoreDbRef:
|
|
let db = w
|
|
else:
|
|
if w.isNil: break
|
|
let db = w.getParent
|
|
if db.trackNewApi:
|
|
code
|
|
|
|
proc oaToStr(w: openArray[byte]): string =
|
|
w.toHex.toLowerAscii
|
|
|
|
proc toStr(w: Hash256): string =
|
|
if w == EMPTY_ROOT_HASH: "EMPTY_ROOT_HASH" else: w.data.oaToStr
|
|
|
|
proc toStr(p: CoreDbVidRef): string =
|
|
if p.isNil:
|
|
"vidRef(nil)"
|
|
elif not p.ready:
|
|
"vidRef(not-ready)"
|
|
else:
|
|
let val = p.parent.methods.vidHashFn(p,false).valueOr: EMPTY_ROOT_HASH
|
|
if val != EMPTY_ROOT_HASH:
|
|
"vidRef(some-hash)"
|
|
else:
|
|
"vidRef(empty-hash)"
|
|
|
|
proc toStr(w: Blob): string =
|
|
if 0 < w.len and w.len < 5: "<" & w.oaToStr & ">"
|
|
else: "Blob[" & $w.len & "]"
|
|
|
|
proc toStr(w: openArray[byte]): string =
|
|
w.oaToStr
|
|
|
|
proc toStr(w: set[CoreDbCaptFlags]): string =
|
|
"Flags[" & $w.len & "]"
|
|
|
|
proc toStr(rc: CoreDbRc[bool]): string =
|
|
if rc.isOk: "ok(" & $rc.value & ")" else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr(rc: CoreDbRc[void]): string =
|
|
if rc.isOk: "ok()" else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr(rc: CoreDbRc[Blob]): string =
|
|
if rc.isOk: "ok(Blob[" & $rc.value.len & "])"
|
|
else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr(rc: CoreDbRc[Hash256]): string =
|
|
if rc.isOk: "ok(" & rc.value.toStr & ")" else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr(rc: CoreDbRc[Account]): string =
|
|
if rc.isOk: "ok(Account)" else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr[T](rc: CoreDbRc[T]; ifOk: static[string]): string =
|
|
if rc.isOk: "ok(" & ifOk & ")" else: "err(" & $$rc.error & ")"
|
|
|
|
proc toStr(rc: CoreDbRc[CoreDbRef]): string = rc.toStr "dbRef"
|
|
proc toStr(rc: CoreDbRc[CoreDbVidRef]): string = rc.toStr "vidRef"
|
|
proc toStr(rc: CoreDbRc[CoreDbAccount]): string = rc.toStr "accRef"
|
|
proc toStr(rc: CoreDbRc[CoreDxTxID]): string = rc.toStr "txId"
|
|
proc toStr(rc: CoreDbRc[CoreDxTxRef]): string = rc.toStr "txRef"
|
|
proc toStr(rc: CoreDbRc[CoreDxCaptRef]): string = rc.toStr "captRef"
|
|
|
|
else:
|
|
when ProvideCoreDbLegacyAPI:
|
|
template setTrackLegaApiOnly(w: CoreDbChldRefs|CoreDbRef) = discard
|
|
template ifTrackLegaApi(w: CoreDbChldRefs|CoreDbRef; c: untyped) = discard
|
|
template ifTrackNewApi(w: CoreDxChldRefs|CoreDbRef; code: untyped) = discard
|
|
|
|
# ---------
|
|
|
|
func toCoreDxPhkRef(mpt: CoreDxMptRef): CoreDxPhkRef =
|
|
## MPT => pre-hashed MPT (aka PHK)
|
|
result = CoreDxPhkRef(
|
|
fromMpt: mpt,
|
|
methods: mpt.methods)
|
|
|
|
result.methods.fetchFn =
|
|
proc(k: openArray[byte]): CoreDbRc[Blob] =
|
|
mpt.methods.fetchFn(k.keccakHash.data)
|
|
|
|
result.methods.deleteFn =
|
|
proc(k: openArray[byte]): CoreDbRc[void] =
|
|
mpt.methods.deleteFn(k.keccakHash.data)
|
|
|
|
result.methods.mergeFn =
|
|
proc(k:openArray[byte]; v: openArray[byte]): CoreDbRc[void] =
|
|
mpt.methods.mergeFn(k.keccakHash.data, v)
|
|
|
|
result.methods.hasPathFn =
|
|
proc(k: openArray[byte]): CoreDbRc[bool] =
|
|
mpt.methods.hasPathFn(k.keccakHash.data)
|
|
|
|
result.methods.pairsIt =
|
|
iterator(): (Blob, Blob) {.apiRaise.} =
|
|
mpt.parent.itNotImplemented("phk/pairs()")
|
|
|
|
result.methods.replicateIt =
|
|
iterator(): (Blob, Blob) {.apiRaise.} =
|
|
mpt.parent.itNotImplemented("phk/replicate()")
|
|
|
|
when AutoValidateDescriptors:
|
|
result.validate
|
|
|
|
|
|
func parent(phk: CoreDxPhkRef): CoreDbRef =
|
|
phk.fromMpt.parent
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public constructor helper
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc bless*(db: CoreDbRef): CoreDbRef =
|
|
## Verify descriptor
|
|
when AutoValidateDescriptors:
|
|
db.validate
|
|
db
|
|
|
|
|
|
proc bless*(db: CoreDbRef; child: CoreDbVidRef): CoreDbVidRef =
|
|
## Complete sub-module descriptor, fill in `parent` and actvate it.
|
|
child.parent = db
|
|
child.ready = true
|
|
when AutoValidateDescriptors:
|
|
child.validate
|
|
child
|
|
|
|
|
|
proc bless*(db: CoreDbRef; child: CoreDxKvtRef): CoreDxKvtRef =
|
|
## Complete sub-module descriptor, fill in `parent` and de-actvate
|
|
## iterator for persistent database.
|
|
child.parent = db
|
|
|
|
# Disable interator for non-memory instances
|
|
if db.dbType in CoreDbPersistentTypes:
|
|
child.methods.pairsIt = iterator(): (Blob, Blob) =
|
|
db.itNotImplemented "pairs/kvt"
|
|
|
|
when AutoValidateDescriptors:
|
|
child.validate
|
|
child
|
|
|
|
|
|
proc bless*[T: CoreDxTrieRelated | CoreDbBackends](
|
|
db: CoreDbRef;
|
|
child: T;
|
|
): auto =
|
|
## Complete sub-module descriptor, fill in `parent`.
|
|
child.parent = db
|
|
when AutoValidateDescriptors:
|
|
child.validate
|
|
child
|
|
|
|
|
|
proc bless*(
|
|
db: CoreDbRef;
|
|
error: CoreDbErrorCode;
|
|
child: CoreDbErrorRef;
|
|
): CoreDbErrorRef =
|
|
child.parent = db
|
|
child.error = error
|
|
when AutoValidateDescriptors:
|
|
child.validate
|
|
child
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public main descriptor methods
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc dbType*(db: CoreDbRef): CoreDbType =
|
|
## Getter
|
|
result = db.dbType
|
|
db.ifTrackNewApi: debug newApiTxt "dbType()", result
|
|
|
|
proc compensateLegacySetup*(db: CoreDbRef) =
|
|
## On the persistent legacy hexary trie, this function is needed for
|
|
## bootstrapping and Genesis setup when the `purge` flag is activated.
|
|
## Otherwise the database backend may defect on an internal inconsistency.
|
|
db.methods.legacySetupFn()
|
|
db.ifTrackNewApi: debug newApiTxt "compensateLegacySetup()"
|
|
|
|
proc parent*(cld: CoreDxChldRefs): CoreDbRef =
|
|
## Getter, common method for all sub-modules
|
|
result = cld.parent
|
|
|
|
proc backend*(dsc: CoreDxKvtRef | CoreDxTrieRelated | CoreDbRef): auto =
|
|
## Getter, retrieves the *raw* backend object for special/localised support.
|
|
result = dsc.methods.backendFn()
|
|
dsc.ifTrackNewApi: debug newApiTxt "backend()"
|
|
|
|
proc finish*(db: CoreDbRef; flush = false) =
|
|
## Database destructor. If the argument `flush` is set `false`, the database
|
|
## is left as-is and only the in-memory handlers are cleaned up.
|
|
##
|
|
## Otherwise the destructor is allowed to remove the database. This feature
|
|
## depends on the backend database. Currently, only the `AristoDbRocks` type
|
|
## backend removes the database on `true`.
|
|
db.methods.destroyFn flush
|
|
db.ifTrackNewApi: debug newApiTxt "finish()"
|
|
|
|
proc `$$`*(e: CoreDbErrorRef): string =
|
|
## Pretty print error symbol, note that this directive may have side effects
|
|
## as it calls a backend function.
|
|
result = $e.error & "(" & e.parent.methods.errorPrintFn(e) & ")"
|
|
e.ifTrackNewApi: debug newApiTxt "$$()", result
|
|
|
|
proc hash*(vid: CoreDbVidRef; update: bool): CoreDbRc[Hash256] =
|
|
## Getter (well, sort of), retrieves the hash for a `vid` argument. The
|
|
## function might fail if there is currently no hash available (e.g. on
|
|
## `Aristo`.) Note that this is different from succeeding with an
|
|
## `EMPTY_ROOT_HASH` value.
|
|
##
|
|
## The value `EMPTY_ROOT_HASH` is also returned on an empty `vid` argument
|
|
## `CoreDbVidRef(nil)`, say.
|
|
##
|
|
result = block:
|
|
if not vid.isNil and vid.ready:
|
|
vid.parent.methods.vidHashFn(vid, update)
|
|
else:
|
|
ok EMPTY_ROOT_HASH
|
|
# Note: tracker will be silent if `vid` is NIL
|
|
vid.ifTrackNewApi:
|
|
debug newApiTxt "hash()", vid=vid.toStr, result=result.toStr
|
|
|
|
proc hashOrEmpty*(vid: CoreDbVidRef): Hash256 =
|
|
## Convenience wrapper, returns `EMPTY_ROOT_HASH` where `hash()` would fail.
|
|
vid.hash(update = true).valueOr: EMPTY_ROOT_HASH
|
|
|
|
proc recast*(account: CoreDbAccount; update: bool): CoreDbRc[Account] =
|
|
## Convert the argument `account` to the portable Ethereum representation
|
|
## of an account. This conversion may fail if the storage root hash (see
|
|
## `hash()` above) is currently unavailable.
|
|
##
|
|
## Note that for the legacy backend, this function always succeeds.
|
|
##
|
|
let vid = account.storageVid
|
|
result = block:
|
|
let rc = vid.hash(update)
|
|
if rc.isOk:
|
|
ok Account(
|
|
nonce: account.nonce,
|
|
balance: account.balance,
|
|
codeHash: account.codeHash,
|
|
storageRoot: rc.value)
|
|
else:
|
|
err(rc.error)
|
|
vid.ifTrackNewApi: debug newApiTxt "recast()", result=result.toStr
|
|
|
|
proc getRoot*(
|
|
db: CoreDbRef;
|
|
root: Hash256;
|
|
createOk = false;
|
|
): CoreDbRc[CoreDbVidRef] =
|
|
## Find root node with argument hash `root` in database and return the
|
|
## corresponding `CoreDbVidRef` object. If the `root` arguent is set
|
|
## `EMPTY_CODE_HASH`, this function always succeeds, otherwise it fails
|
|
## unless a root node with the corresponding hash exists.
|
|
##
|
|
## This function is intended to open a virtual accounts trie database as in:
|
|
## ::
|
|
## proc openAccountLedger(db: CoreDbRef, rootHash: Hash256): CoreDxMptRef =
|
|
## let root = db.getRoot(rootHash).valueOr:
|
|
## # some error handling
|
|
## return
|
|
## db.newAccMpt root
|
|
##
|
|
result = db.methods.getRootFn(root, createOk)
|
|
db.ifTrackNewApi:
|
|
debug newApiTxt "getRoot()", root=root.toStr, result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public key-value table methods
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc newKvt*(db: CoreDbRef; saveMode = AutoSave): CoreDxKvtRef =
|
|
## Constructor, will defect on failure.
|
|
##
|
|
## Depending on the argument `saveMode`, the contructed object will have
|
|
## the following properties.
|
|
##
|
|
## * `Cached`
|
|
## Subscribe to the common base object shared with other subscribed
|
|
## `AutoSave` or `Cached` descriptors. So any changes are immediately
|
|
## visible among subscribers. On automatic destruction (when the
|
|
## constructed object gets out of scope), changes are not saved to the
|
|
## backend database but are still available to subscribers.
|
|
##
|
|
## * `AutoSave`
|
|
## This mode works similar to `Cached` with the difference that changes
|
|
## are saved to the backend database on automatic destruction when this
|
|
## is permissible, i.e. there is a backend available and there is no
|
|
## pending transaction on the common base object.
|
|
##
|
|
## * `Companion`
|
|
## The contructed object will be a new descriptor separate from the common
|
|
## base object. It will be a copy of the current state of the common
|
|
## base object available to subscribers. On automatic destruction, changes
|
|
## will be discarded.
|
|
##
|
|
## The constructed object can be manually descructed (see `destroy()`) where
|
|
## the `saveMode` behaviour can be overridden.
|
|
##
|
|
## The legacy backend always assumes `AutoSave` mode regardless of the
|
|
## function argument.
|
|
##
|
|
result = db.methods.newKvtFn(saveMode).valueOr:
|
|
raiseAssert $$error
|
|
db.ifTrackNewApi: debug newApiTxt "newKvt()", saveMode
|
|
|
|
proc get*(kvt: CoreDxKvtRef; key: openArray[byte]): CoreDbRc[Blob] =
|
|
## This function always returns a non-empty `Blob` or an error code.
|
|
result = kvt.methods.getFn key
|
|
kvt.ifTrackNewApi:
|
|
debug newApiTxt "get()", key=key.toStr, result=result.toStr
|
|
|
|
proc getOrEmpty*(kvt: CoreDxKvtRef; key: openArray[byte]): CoreDbRc[Blob] =
|
|
## This function sort of mimics the behaviour of the legacy database
|
|
## returning an empty `Blob` if the argument `key` is not found on the
|
|
## database.
|
|
result = kvt.methods.getFn key
|
|
if result.isErr and result.error.error == KvtNotFound:
|
|
result = CoreDbRc[Blob].ok(EmptyBlob)
|
|
kvt.ifTrackNewApi:
|
|
debug newApiTxt "getOrEmpty()", key=key.toStr, result=result.toStr
|
|
|
|
proc del*(kvt: CoreDxKvtRef; key: openArray[byte]): CoreDbRc[void] =
|
|
result = kvt.methods.delFn key
|
|
kvt.ifTrackNewApi:
|
|
debug newApiTxt "del()", key=key.toStr, result=result.toStr
|
|
|
|
proc put*(
|
|
kvt: CoreDxKvtRef;
|
|
key: openArray[byte];
|
|
val: openArray[byte];
|
|
): CoreDbRc[void] =
|
|
result = kvt.methods.putFn(key, val)
|
|
kvt.ifTrackNewApi: debug newApiTxt "put()",
|
|
key=key.toStr, val=val.toSeq.toStr, result=result.toStr
|
|
|
|
proc hasKey*(kvt: CoreDxKvtRef; key: openArray[byte]): CoreDbRc[bool] =
|
|
## Would be named `contains` if it returned `bool` rather than `Result[]`.
|
|
result = kvt.methods.hasKeyFn key
|
|
kvt.ifTrackNewApi:
|
|
debug newApiTxt "kvt/hasKey()", key=key.toStr, result=result.toStr
|
|
|
|
proc destroy*(dsc: CoreDxKvtRef; saveMode = AutoSave): CoreDbRc[void] =
|
|
## For the legacy database, this function has no effect and succeeds always.
|
|
##
|
|
## The function explicitely destructs the descriptor `dsc`. If the function
|
|
## argument `saveMode` is not `AutoSave` the data object behind the argument
|
|
## descriptor `dsc` is just discarded and the function returns success.
|
|
##
|
|
## Otherwise, the state of the descriptor object is saved to the database
|
|
## backend if that is possible, or an error is returned.
|
|
##
|
|
## Subject to change
|
|
## -----------------
|
|
## * Saving an object which was created with the `Companion` flag (see
|
|
## `newKvt()`), the common base object will not reveal any change although
|
|
## the backend database will have persistently stored the data.
|
|
## * Subsequent saving of the common base object may override that.
|
|
##
|
|
## When returnng an error, the argument descriptor `dsc` will have been
|
|
## disposed nevertheless.
|
|
##
|
|
result = dsc.methods.destroyFn saveMode
|
|
dsc.ifTrackNewApi: debug newApiTxt "destroy()", saveMode, result=result.toStr
|
|
|
|
iterator pairs*(kvt: CoreDxKvtRef): (Blob, Blob) {.apiRaise.} =
|
|
## Iterator supported on memory DB (otherwise implementation dependent)
|
|
for k,v in kvt.methods.pairsIt():
|
|
yield (k,v)
|
|
kvt.ifTrackNewApi: debug newApiTxt "kvt/pairs()"
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public Merkle Patricia Tree, hexary trie constructors
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc newMpt*(
|
|
db: CoreDbRef;
|
|
root: CoreDbVidRef;
|
|
prune = true;
|
|
saveMode = AutoSave;
|
|
): CoreDxMptRef =
|
|
## Constructor, will defect on failure. The argument `prune` is currently
|
|
## effective only for the legacy backend.
|
|
##
|
|
## See the discussion at `newKvt()` for an explanation of the `saveMode`
|
|
## argument.
|
|
##
|
|
## The constructed object can be manually descructed (see `destroy()`) where
|
|
## the `saveMode` behaviour can be overridden.
|
|
##
|
|
## The legacy backend always assumes `AutoSave` mode regardless of the
|
|
## function argument.
|
|
##
|
|
result = db.methods.newMptFn(root, prune, saveMode).valueOr:
|
|
raiseAssert $$error
|
|
db.ifTrackNewApi:
|
|
debug newApiTxt "newMpt()", root=root.toStr, prune, saveMode
|
|
|
|
proc newMpt*(db: CoreDbRef; prune = true; saveMode = AutoSave): CoreDxMptRef =
|
|
## Shortcut for `db.newMpt CoreDbVidRef()`
|
|
let root = CoreDbVidRef()
|
|
result = db.methods.newMptFn(root, prune, saveMode).valueOr:
|
|
raiseAssert $$error
|
|
db.ifTrackNewApi: debug newApiTxt "newMpt()", root=root.toStr, prune, saveMode
|
|
|
|
proc newMpt*(acc: CoreDxAccRef): CoreDxMptRef =
|
|
## Constructor, will defect on failure. The argument `prune` is currently
|
|
## effective only for the legacy backend.
|
|
##
|
|
## Variant of `newMpt()` where the input arguments are taken from the
|
|
## current `acc` descriptor settings.
|
|
##
|
|
result = acc.methods.newMptFn().valueOr:
|
|
raiseAssert $$error
|
|
acc.ifTrackNewApi: debug newApiTxt "acc/toMpt()"
|
|
|
|
proc newAccMpt*(
|
|
db: CoreDbRef;
|
|
root: CoreDbVidRef;
|
|
prune = true;
|
|
saveMode = AutoSave;
|
|
): CoreDxAccRef =
|
|
## This function works similar to `newMpt()` for handling accounts. Although
|
|
## this sub-trie can be emulated by means of `newMpt(..).toPhk()`, it is
|
|
## recommended using this particular constructor for accounts because it
|
|
## provides its own subset of methods to handle accounts.
|
|
##
|
|
## The argument `prune` is currently effective only for the legacy backend.
|
|
##
|
|
## See the discussion at `newKvt()` for an explanation of the `saveMode`
|
|
## argument.
|
|
##
|
|
## The constructed object can be manually descructed (see `destroy()`) where
|
|
## the `saveMode` behaviour can be overridden.
|
|
##
|
|
## The legacy backend always assumes `AutoSave` mode regardless of the
|
|
## function argument.
|
|
##
|
|
result = db.methods.newAccFn(root, prune, saveMode).valueOr:
|
|
raiseAssert $$error
|
|
db.ifTrackNewApi:
|
|
debug newApiTxt "newAccMpt()", root=root.toStr, prune, saveMode
|
|
|
|
proc toMpt*(phk: CoreDxPhkRef): CoreDxMptRef =
|
|
## Replaces the pre-hashed argument trie `phk` by the non pre-hashed *MPT*.
|
|
## Note that this does not apply to an accounts trie that was created by
|
|
## `newAccMpt()`.
|
|
result = phk.fromMpt
|
|
phk.ifTrackNewApi: debug newApiTxt "phk/toMpt()"
|
|
|
|
proc toPhk*(mpt: CoreDxMptRef): CoreDxPhkRef =
|
|
## Replaces argument `mpt` by a pre-hashed *MPT*.
|
|
## Note that this does not apply to an accounts trie that was created by
|
|
## `newAaccMpt()`.
|
|
result = mpt.toCoreDxPhkRef
|
|
mpt.ifTrackNewApi: debug newApiTxt "mpt/toPhk()"
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public common methods for all hexary trie databases (`mpt`, `phk`, or `acc`)
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc isPruning*(dsc: CoreDxTrieRefs | CoreDxAccRef): bool =
|
|
## Getter
|
|
result = dsc.methods.isPruningFn()
|
|
dsc.ifTrackNewApi: debug newApiTxt "isPruning()", result
|
|
|
|
proc rootVid*(dsc: CoreDxTrieRefs | CoreDxAccRef): CoreDbVidRef =
|
|
## Getter, result is not `nil`
|
|
result = dsc.methods.rootVidFn()
|
|
dsc.ifTrackNewApi: debug newApiTxt "rootVid()", result=result.toStr
|
|
|
|
proc destroy*(
|
|
dsc: CoreDxTrieRefs | CoreDxAccRef;
|
|
saveMode = AutoSave;
|
|
): CoreDbRc[void]
|
|
{.discardable.} =
|
|
## For the legacy database, this function has no effect and succeeds always.
|
|
##
|
|
## See the discussion at `destroy()` for `CoreDxKvtRef` for an explanation
|
|
## of the `saveMode` argument.
|
|
##
|
|
result = dsc.methods.destroyFn saveMode
|
|
dsc.ifTrackNewApi: debug newApiTxt "destroy()", result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public generic hexary trie database methods (`mpt` or `phk`)
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc fetch*(trie: CoreDxTrieRefs; key: openArray[byte]): CoreDbRc[Blob] =
|
|
## Fetch data from the argument `trie`. The function always returns a
|
|
## non-empty `Blob` or an error code.
|
|
result = trie.methods.fetchFn(key)
|
|
trie.ifTrackNewApi:
|
|
debug newApiTxt "trie/fetch()", key=key.toStr, result=result.toStr
|
|
|
|
proc fetchOrEmpty*(trie: CoreDxTrieRefs; key: openArray[byte]): CoreDbRc[Blob] =
|
|
## This function returns an empty `Blob` if the argument `key` is not found
|
|
## on the database.
|
|
result = trie.methods.fetchFn(key)
|
|
if result.isErr and result.error.error == MptNotFound:
|
|
result = ok(EmptyBlob)
|
|
trie.ifTrackNewApi:
|
|
debug newApiTxt "trie/fetchOrEmpty()", key=key.toStr, result=result.toStr
|
|
|
|
proc delete*(trie: CoreDxTrieRefs; key: openArray[byte]): CoreDbRc[void] =
|
|
result = trie.methods.deleteFn key
|
|
trie.ifTrackNewApi:
|
|
debug newApiTxt "trie/delete()", key=key.toStr, result=result.toStr
|
|
|
|
proc merge*(
|
|
trie: CoreDxTrieRefs;
|
|
key: openArray[byte];
|
|
val: openArray[byte];
|
|
): CoreDbRc[void] =
|
|
when trie is CoreDxMptRef:
|
|
const info = "mpt/merge()"
|
|
else:
|
|
const info = "phk/merge()"
|
|
result = trie.methods.mergeFn(key, val)
|
|
trie.ifTrackNewApi: debug newApiTxt info,
|
|
key=key.toStr, val=val.toSeq.toStr, result=result.toStr
|
|
|
|
proc hasPath*(trie: CoreDxTrieRefs; key: openArray[byte]): CoreDbRc[bool] =
|
|
## Would be named `contains` if it returned `bool` rather than `Result[]`.
|
|
result = trie.methods.hasPathFn key
|
|
trie.ifTrackNewApi:
|
|
debug newApiTxt "trie/hasKey()", key=key.toStr, result=result.toStr
|
|
|
|
iterator pairs*(mpt: CoreDxMptRef): (Blob, Blob) {.apiRaise.} =
|
|
## Trie traversal, only supported for `CoreDxMptRef`
|
|
for k,v in mpt.methods.pairsIt():
|
|
yield (k,v)
|
|
mpt.ifTrackNewApi: debug newApiTxt "mpt/pairs()"
|
|
|
|
iterator replicate*(mpt: CoreDxMptRef): (Blob, Blob) {.apiRaise.} =
|
|
## Low level trie dump, only supported for `CoreDxMptRef`
|
|
for k,v in mpt.methods.replicateIt():
|
|
yield (k,v)
|
|
mpt.ifTrackNewApi: debug newApiTxt "mpt/replicate()"
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public trie database methods for accounts
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc fetch*(acc: CoreDxAccRef; address: EthAddress): CoreDbRc[CoreDbAccount] =
|
|
## Fetch data from the argument `trie`.
|
|
result = acc.methods.fetchFn address
|
|
acc.ifTrackNewApi:
|
|
debug newApiTxt "acc/fetch()", address=address.toStr, result=result.toStr
|
|
|
|
proc delete*(acc: CoreDxAccRef; address: EthAddress): CoreDbRc[void] =
|
|
result = acc.methods.deleteFn address
|
|
acc.ifTrackNewApi:
|
|
debug newApiTxt "acc/delete()", address=address.toStr, result=result.toStr
|
|
|
|
proc merge*(
|
|
acc: CoreDxAccRef;
|
|
address: EthAddress;
|
|
account: CoreDbAccount;
|
|
): CoreDbRc[void] =
|
|
result = acc.methods.mergeFn(address, account)
|
|
acc.ifTrackNewApi:
|
|
debug newApiTxt "acc/merge()", address=address.toStr, result=result.toStr
|
|
|
|
proc hasPath*(acc: CoreDxAccRef; address: EthAddress): CoreDbRc[bool] =
|
|
## Would be named `contains` if it returned `bool` rather than `Result[]`.
|
|
result = acc.methods.hasPathFn address
|
|
acc.ifTrackNewApi:
|
|
debug newApiTxt "acc/hasKey()", address=address.toStr, result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public transaction related methods
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc newTransaction*(db: CoreDbRef): CoreDbRc[CoreDxTxRef] =
|
|
## Constructor
|
|
result = db.methods.beginFn()
|
|
db.ifTrackNewApi: debug newApiTxt "newTransaction()", result=result.toStr
|
|
|
|
proc commit*(tx: CoreDxTxRef, applyDeletes = true): CoreDbRc[void] =
|
|
result = tx.methods.commitFn applyDeletes
|
|
tx.ifTrackNewApi: debug newApiTxt "tx/commit()", result=result.toStr
|
|
|
|
proc rollback*(tx: CoreDxTxRef): CoreDbRc[void] =
|
|
result = tx.methods.rollbackFn()
|
|
tx.ifTrackNewApi: debug newApiTxt "tx/rollback()", result=result.toStr
|
|
|
|
proc dispose*(tx: CoreDxTxRef): CoreDbRc[void] =
|
|
result = tx.methods.disposeFn()
|
|
tx.ifTrackNewApi: debug newApiTxt "tx/dispose()", result=result.toStr
|
|
|
|
proc safeDispose*(tx: CoreDxTxRef): CoreDbRc[void] =
|
|
result = tx.methods.safeDisposeFn()
|
|
tx.ifTrackNewApi: debug newApiTxt "tx/safeDispose()", result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public tracer methods
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc newCapture*(
|
|
db: CoreDbRef;
|
|
flags: set[CoreDbCaptFlags] = {};
|
|
): CoreDbRc[CoreDxCaptRef] =
|
|
## Constructor
|
|
result = db.methods.captureFn flags
|
|
db.ifTrackNewApi: debug newApiTxt "db/capture()", result=result.toStr
|
|
|
|
proc recorder*(cp: CoreDxCaptRef): CoreDbRc[CoreDbRef] =
|
|
## Getter
|
|
result = cp.methods.recorderFn()
|
|
cp.ifTrackNewApi: debug newApiTxt "capt/recorder()", result=result.toStr
|
|
|
|
proc logDb*(cp: CoreDxCaptRef): CoreDbRc[CoreDbRef] =
|
|
result = cp.methods.logDbFn()
|
|
cp.ifTrackNewApi: debug newApiTxt "capt/logDb()", result=result.toStr
|
|
|
|
proc flags*(cp: CoreDxCaptRef): set[CoreDbCaptFlags] =
|
|
## Getter
|
|
result = cp.methods.getFlagsFn()
|
|
cp.ifTrackNewApi: debug newApiTxt "capt/flags()", result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public methods, legacy API
|
|
# ------------------------------------------------------------------------------
|
|
|
|
when ProvideCoreDbLegacyAPI:
|
|
|
|
proc parent*(cld: CoreDbChldRefs): CoreDbRef =
|
|
## Getter, common method for all sub-modules
|
|
result = cld.distinctBase.parent
|
|
|
|
proc backend*(dsc: CoreDbChldRefs): auto =
|
|
dsc.setTrackLegaApiOnly
|
|
result = dsc.distinctBase.backend
|
|
dsc.ifTrackLegaApi: debug legaApiTxt "parent()"
|
|
|
|
# ----------------
|
|
|
|
proc kvt*(db: CoreDbRef): CoreDbKvtRef =
|
|
## Legacy pseudo constructor, see `toKvt()` for production constructor
|
|
db.setTrackLegaApiOnly
|
|
result = db.newKvt().CoreDbKvtRef
|
|
db.ifTrackLegaApi: debug legaApiTxt "kvt()", result=result.toStr
|
|
|
|
proc get*(kvt: CoreDbKvtRef; key: openArray[byte]): Blob =
|
|
kvt.setTrackLegaApiOnly
|
|
const info = "kvt/get()"
|
|
result = kvt.distinctBase.getOrEmpty(key).expect info
|
|
kvt.ifTrackLegaApi:
|
|
debug legaApiTxt info, key=key.toStr, result=result.toStr
|
|
|
|
proc del*(kvt: CoreDbKvtRef; key: openArray[byte]): void =
|
|
kvt.setTrackLegaApiOnly
|
|
const info = "kvt/del()"
|
|
kvt.distinctBase.del(key).expect info
|
|
kvt.ifTrackLegaApi: debug legaApiTxt info, key=key.toStr
|
|
|
|
proc put*(kvt: CoreDbKvtRef; key: openArray[byte]; val: openArray[byte]) =
|
|
kvt.setTrackLegaApiOnly
|
|
const info = "kvt/put()"
|
|
let w = kvt.distinctBase.parent.newKvt()
|
|
w.put(key, val).expect info
|
|
#kvt.distinctBase.put(key, val).expect info
|
|
kvt.ifTrackLegaApi:
|
|
debug legaApiTxt info, key=key.toStr, val=val.toSeq.toStr
|
|
|
|
proc contains*(kvt: CoreDbKvtRef; key: openArray[byte]): bool =
|
|
kvt.setTrackLegaApiOnly
|
|
const info = "kvt/contains()"
|
|
result = kvt.distinctBase.hasKey(key).expect info
|
|
kvt.ifTrackLegaApi: debug legaApiTxt info, key=key.toStr, result
|
|
|
|
iterator pairs*(kvt: CoreDbKvtRef): (Blob, Blob) {.apiRaise.} =
|
|
kvt.setTrackLegaApiOnly
|
|
for k,v in kvt.distinctBase.pairs():
|
|
yield (k,v)
|
|
kvt.ifTrackLegaApi: debug legaApiTxt "kvt/pairs()"
|
|
|
|
# ----------------
|
|
|
|
proc toMpt*(phk: CoreDbPhkRef): CoreDbMptRef =
|
|
phk.setTrackLegaApiOnly
|
|
result = phk.distinctBase.toMpt.CoreDbMptRef
|
|
phk.ifTrackLegaApi: debug legaApiTxt "phk/toMpt()"
|
|
|
|
proc mptPrune*(db: CoreDbRef; root: Hash256; prune = true): CoreDbMptRef =
|
|
db.setTrackLegaApiOnly
|
|
const info = "mptPrune()"
|
|
let vid = db.getRoot(root, createOk=true).expect info
|
|
result = db.newMpt(vid, prune).CoreDbMptRef
|
|
db.ifTrackLegaApi: debug legaApiTxt info, root=root.toStr, prune
|
|
|
|
proc mptPrune*(db: CoreDbRef; prune = true): CoreDbMptRef =
|
|
db.newMpt(CoreDbVidRef(nil), prune).CoreDbMptRef
|
|
|
|
# ----------------
|
|
|
|
proc toPhk*(mpt: CoreDbMptRef): CoreDbPhkRef =
|
|
mpt.setTrackLegaApiOnly
|
|
result = mpt.distinctBase.toPhk.CoreDbPhkRef
|
|
mpt.ifTrackLegaApi: debug legaApiTxt "mpt/toMpt()"
|
|
|
|
proc phkPrune*(db: CoreDbRef; root: Hash256; prune = true): CoreDbPhkRef =
|
|
db.setTrackLegaApiOnly
|
|
const info = "phkPrune()"
|
|
let vid = db.getRoot(root, createOk=true).expect info
|
|
result = db.newMpt(vid, prune).toCoreDxPhkRef.CoreDbPhkRef
|
|
db.ifTrackLegaApi: debug legaApiTxt info, root=root.toStr, prune
|
|
|
|
proc phkPrune*(db: CoreDbRef; prune = true): CoreDbPhkRef =
|
|
db.newMpt(CoreDbVidRef(nil), prune).toCoreDxPhkRef.CoreDbPhkRef
|
|
|
|
# ----------------
|
|
|
|
proc isPruning*(trie: CoreDbTrieRefs): bool =
|
|
trie.setTrackLegaApiOnly
|
|
result = trie.distinctBase.isPruning()
|
|
trie.ifTrackLegaApi: debug legaApiTxt "trie/isPruning()", result
|
|
|
|
proc get*(trie: CoreDbTrieRefs; key: openArray[byte]): Blob =
|
|
trie.setTrackLegaApiOnly
|
|
const info = "trie/get()"
|
|
result = trie.distinctBase.fetchOrEmpty(key).expect info
|
|
trie.ifTrackLegaApi:
|
|
debug legaApiTxt info, key=key.toStr, result=result.toStr
|
|
|
|
proc del*(trie: CoreDbTrieRefs; key: openArray[byte]) =
|
|
trie.setTrackLegaApiOnly
|
|
const info = "trie/del()"
|
|
trie.distinctBase.delete(key).expect info
|
|
trie.ifTrackLegaApi: debug legaApiTxt info, key=key.toStr
|
|
|
|
proc put*(trie: CoreDbTrieRefs; key: openArray[byte]; val: openArray[byte]) =
|
|
trie.setTrackLegaApiOnly
|
|
when trie is CoreDbMptRef:
|
|
const info = "mpt/put()"
|
|
else:
|
|
const info = "phk/put()"
|
|
trie.distinctBase.merge(key, val).expect info
|
|
trie.ifTrackLegaApi:
|
|
debug legaApiTxt info, key=key.toStr, val=val.toSeq.toStr
|
|
|
|
proc contains*(trie: CoreDbTrieRefs; key: openArray[byte]): bool =
|
|
trie.setTrackLegaApiOnly
|
|
const info = "trie/contains()"
|
|
result = trie.distinctBase.hasPath(key).expect info
|
|
trie.ifTrackLegaApi: debug legaApiTxt info, key=key.toStr, result
|
|
|
|
proc rootHash*(trie: CoreDbTrieRefs): Hash256 =
|
|
trie.setTrackLegaApiOnly
|
|
const info = "trie/rootHash()"
|
|
result = trie.distinctBase.rootVid().hash(update=true).expect info
|
|
trie.ifTrackLegaApi: debug legaApiTxt info, result=result.toStr
|
|
|
|
iterator pairs*(mpt: CoreDbMptRef): (Blob, Blob) {.apiRaise.} =
|
|
## Trie traversal, not supported for `CoreDbPhkRef`
|
|
mpt.setTrackLegaApiOnly
|
|
for k,v in mpt.distinctBase.pairs():
|
|
yield (k,v)
|
|
mpt.ifTrackLegaApi: debug legaApiTxt "mpt/pairs()"
|
|
|
|
iterator replicate*(mpt: CoreDbMptRef): (Blob, Blob) {.apiRaise.} =
|
|
## Low level trie dump, not supported for `CoreDbPhkRef`
|
|
mpt.setTrackLegaApiOnly
|
|
for k,v in mpt.distinctBase.replicate():
|
|
yield (k,v)
|
|
mpt.ifTrackLegaApi: debug legaApiTxt "mpt/replicate()"
|
|
|
|
# ----------------
|
|
|
|
proc getTransactionID*(db: CoreDbRef): CoreDbTxID =
|
|
db.setTrackLegaApiOnly
|
|
const info = "getTransactionID()"
|
|
result = db.methods.getIdFn().expect(info).CoreDbTxID
|
|
db.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc shortTimeReadOnly*(
|
|
id: CoreDbTxID;
|
|
action: proc() {.catchRaise.};
|
|
) {.catchRaise.} =
|
|
id.setTrackLegaApiOnly
|
|
const info = "txId/shortTimeReadOnly()"
|
|
var oops = none(ref CatchableError)
|
|
proc safeFn() =
|
|
try:
|
|
action()
|
|
except CatchableError as e:
|
|
oops = some(e)
|
|
# Action has finished now
|
|
|
|
id.distinctBase.methods.roWrapperFn(safeFn).expect info
|
|
|
|
# Delayed exception
|
|
if oops.isSome:
|
|
let
|
|
e = oops.unsafeGet
|
|
msg = "delayed and reraised" &
|
|
", name=\"" & $e.name & "\", msg=\"" & e.msg & "\""
|
|
raise (ref TxWrapperApiError)(msg: msg)
|
|
id.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc beginTransaction*(db: CoreDbRef): CoreDbTxRef =
|
|
db.setTrackLegaApiOnly
|
|
const info = "newTransaction()"
|
|
result = (db.distinctBase.newTransaction().expect info).CoreDbTxRef
|
|
db.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc commit*(tx: CoreDbTxRef, applyDeletes = true) =
|
|
tx.setTrackLegaApiOnly
|
|
const info = "tx/commit()"
|
|
tx.distinctBase.commit(applyDeletes).expect info
|
|
tx.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc rollback*(tx: CoreDbTxRef) =
|
|
tx.setTrackLegaApiOnly
|
|
const info = "tx/rollback()"
|
|
tx.distinctBase.rollback().expect info
|
|
tx.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc dispose*(tx: CoreDbTxRef) =
|
|
tx.setTrackLegaApiOnly
|
|
const info = "tx/dispose()"
|
|
tx.distinctBase.dispose().expect info
|
|
tx.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc safeDispose*(tx: CoreDbTxRef) =
|
|
tx.setTrackLegaApiOnly
|
|
const info = "tx/safeDispose()"
|
|
tx.distinctBase.safeDispose().expect info
|
|
tx.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
# ----------------
|
|
|
|
proc capture*(
|
|
db: CoreDbRef;
|
|
flags: set[CoreDbCaptFlags] = {};
|
|
): CoreDbCaptRef =
|
|
db.setTrackLegaApiOnly
|
|
const info = "db/capture()"
|
|
result = db.newCapture(flags).expect(info).CoreDbCaptRef
|
|
db.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc recorder*(cp: CoreDbCaptRef): CoreDbRef =
|
|
cp.setTrackLegaApiOnly
|
|
const info = "capt/recorder()"
|
|
result = cp.distinctBase.recorder().expect info
|
|
cp.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc logDb*(cp: CoreDbCaptRef): CoreDbRef =
|
|
cp.setTrackLegaApiOnly
|
|
const info = "capt/logDb()"
|
|
result = cp.distinctBase.logDb().expect info
|
|
cp.ifTrackLegaApi: debug legaApiTxt info
|
|
|
|
proc flags*(cp: CoreDbCaptRef): set[CoreDbCaptFlags] =
|
|
cp.setTrackLegaApiOnly
|
|
result = cp.distinctBase.flags()
|
|
cp.ifTrackLegaApi: debug legaApiTxt "capt/flags()", result=result.toStr
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# End
|
|
# ------------------------------------------------------------------------------
|