2024-03-05 03:12:37 +00:00
|
|
|
# Nim-RocksDB
|
|
|
|
# Copyright 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)
|
|
|
|
# * GPL license, version 2.0, ([LICENSE-GPLv2](LICENSE-GPLv2) or https://www.gnu.org/licenses/old-licenses/gpl-2.0.en.html)
|
|
|
|
#
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
## A `RocksDBRef` represents a reference to a RocksDB instance. It can be opened
|
|
|
|
## in read-only or read-write mode in which case a `RocksDbReadOnlyRef` or
|
|
|
|
## `RocksDbReadWriteRef` will be returned respectively. The `RocksDbReadOnlyRef`
|
|
|
|
## type doesn't support any of the write operations such as `put`, `delete` or
|
|
|
|
## `write`.
|
|
|
|
##
|
|
|
|
## Many of the operations on these types can potentially fail for various reasons,
|
|
|
|
## in which case a `RocksDbResult` containing an error will be returned.
|
|
|
|
##
|
|
|
|
## The types wrap and hold a handle to a c pointer which needs to be freed
|
|
|
|
## so `close` should be called to prevent a memory leak after use.
|
|
|
|
##
|
|
|
|
## Most of the procs below support passing in the name of the column family
|
|
|
|
## which should be used for the operation. The default column family will be
|
|
|
|
## used if none is provided.
|
|
|
|
|
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
|
|
|
std/[sequtils, locks],
|
|
|
|
./lib/librocksdb,
|
|
|
|
./options/[dbopts, readopts, writeopts],
|
|
|
|
./columnfamily/[cfopts, cfdescriptor, cfhandle],
|
|
|
|
./internal/[cftable, utils],
|
|
|
|
./rocksiterator,
|
|
|
|
./rocksresult,
|
|
|
|
./writebatch
|
|
|
|
|
2024-06-26 15:00:10 +00:00
|
|
|
export rocksresult, dbopts, readopts, writeopts, cfdescriptor, rocksiterator, writebatch
|
2024-03-05 03:12:37 +00:00
|
|
|
|
|
|
|
type
|
|
|
|
RocksDbPtr* = ptr rocksdb_t
|
|
|
|
IngestExternalFilesOptionsPtr = ptr rocksdb_ingestexternalfileoptions_t
|
|
|
|
|
|
|
|
RocksDbRef* = ref object of RootObj
|
|
|
|
lock: Lock
|
|
|
|
cPtr: RocksDbPtr
|
|
|
|
path: string
|
|
|
|
dbOpts: DbOptionsRef
|
|
|
|
readOpts: ReadOptionsRef
|
2024-06-28 02:04:37 +00:00
|
|
|
cfDescriptors: seq[ColFamilyDescriptor]
|
2024-06-26 17:31:39 +00:00
|
|
|
defaultCfHandle: ColFamilyHandleRef
|
2024-03-05 03:12:37 +00:00
|
|
|
cfTable: ColFamilyTableRef
|
|
|
|
|
|
|
|
RocksDbReadOnlyRef* = ref object of RocksDbRef
|
|
|
|
|
|
|
|
RocksDbReadWriteRef* = ref object of RocksDbRef
|
|
|
|
writeOpts: WriteOptionsRef
|
|
|
|
ingestOptsPtr: IngestExternalFilesOptionsPtr
|
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
proc listColumnFamilies*(path: string): RocksDBResult[seq[string]] =
|
2024-06-13 15:43:33 +00:00
|
|
|
## List exisiting column families on disk. This might be used to find out
|
|
|
|
## whether there were some columns missing with the version on disk.
|
|
|
|
##
|
|
|
|
## Column families previously used must be declared when re-opening an
|
|
|
|
## existing database. So this function can be used to add some CFs
|
|
|
|
## on-the-fly to the opener list of CFs
|
|
|
|
##
|
|
|
|
## Note that the on-the-fly adding might not be needed in the way described
|
|
|
|
## above once rocksdb has been upgraded to the latest version, see comments
|
|
|
|
## at the end of ./columnfamily/cfhandle.nim.
|
|
|
|
|
|
|
|
var
|
2024-06-28 02:04:37 +00:00
|
|
|
cfLen: csize_t
|
2024-06-13 15:43:33 +00:00
|
|
|
errors: cstring
|
2024-06-28 02:04:37 +00:00
|
|
|
let
|
|
|
|
dbOpts = defaultDbOptions(autoClose = true)
|
|
|
|
cfList = rocksdb_list_column_families(
|
|
|
|
dbOpts.cPtr, path.cstring, addr cfLen, cast[cstringArray](errors.addr)
|
|
|
|
)
|
|
|
|
bailOnErrors(errors, dbOpts)
|
2024-06-13 15:43:33 +00:00
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
if cfList.isNil or cfLen == 0:
|
|
|
|
return ok(newSeqOfCap[string](0))
|
2024-06-13 15:43:33 +00:00
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
defer:
|
|
|
|
rocksdb_list_column_families_destroy(cfList, cfLen)
|
|
|
|
dbOpts.close()
|
2024-06-13 15:43:33 +00:00
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
var colFamilyNames = newSeqOfCap[string](cfLen)
|
|
|
|
for i in 0 ..< cfLen:
|
|
|
|
colFamilyNames.add($cfList[i])
|
2024-06-13 15:43:33 +00:00
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
ok(colFamilyNames)
|
2024-06-13 15:43:33 +00:00
|
|
|
|
2024-03-05 03:12:37 +00:00
|
|
|
proc openRocksDb*(
|
2024-06-26 15:00:10 +00:00
|
|
|
path: string,
|
2024-06-28 02:04:37 +00:00
|
|
|
dbOpts = defaultDbOptions(autoClose = true),
|
|
|
|
readOpts = defaultReadOptions(autoClose = true),
|
|
|
|
writeOpts = defaultWriteOptions(autoClose = true),
|
2024-06-26 15:00:10 +00:00
|
|
|
columnFamilies: openArray[ColFamilyDescriptor] = [],
|
|
|
|
): RocksDBResult[RocksDbReadWriteRef] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Open a RocksDB instance in read-write mode. If `columnFamilies` is empty
|
|
|
|
## then it will open the default column family. If `dbOpts`, `readOpts`, or
|
|
|
|
## `writeOpts` are not supplied then the default options will be used.
|
2024-06-28 02:04:37 +00:00
|
|
|
## These default options will be closed when the database is closed.
|
|
|
|
## If any options are provided, they will need to be closed manually.
|
|
|
|
##
|
2024-03-05 03:12:37 +00:00
|
|
|
## By default, column families will be created if they don't yet exist.
|
|
|
|
## All existing column families must be specified if the database has
|
2024-03-07 01:02:24 +00:00
|
|
|
## previously created any column families.
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-03-07 01:02:24 +00:00
|
|
|
var cfs = columnFamilies.toSeq()
|
|
|
|
if DEFAULT_COLUMN_FAMILY_NAME notin columnFamilies.mapIt(it.name()):
|
2024-06-28 02:04:37 +00:00
|
|
|
cfs.add(defaultColFamilyDescriptor(autoClose = true))
|
2024-03-05 03:12:37 +00:00
|
|
|
|
|
|
|
var
|
2024-03-07 01:02:24 +00:00
|
|
|
cfNames = cfs.mapIt(it.name().cstring)
|
|
|
|
cfOpts = cfs.mapIt(it.options.cPtr)
|
|
|
|
cfHandles = newSeq[ColFamilyHandlePtr](cfs.len)
|
2024-03-05 03:12:37 +00:00
|
|
|
errors: cstring
|
|
|
|
let rocksDbPtr = rocksdb_open_column_families(
|
2024-06-28 02:04:37 +00:00
|
|
|
dbOpts.cPtr,
|
2024-06-26 15:00:10 +00:00
|
|
|
path.cstring,
|
|
|
|
cfNames.len().cint,
|
|
|
|
cast[cstringArray](cfNames[0].addr),
|
|
|
|
cfOpts[0].addr,
|
|
|
|
cfHandles[0].addr,
|
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-06-28 02:04:37 +00:00
|
|
|
bailOnErrors(errors, dbOpts, readOpts, writeOpts, cfDescriptors = cfs)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-06-13 15:43:33 +00:00
|
|
|
let
|
2024-06-26 17:31:39 +00:00
|
|
|
cfTable = newColFamilyTable(cfNames.mapIt($it), cfHandles)
|
2024-06-13 15:43:33 +00:00
|
|
|
db = RocksDbReadWriteRef(
|
2024-03-05 03:12:37 +00:00
|
|
|
lock: createLock(),
|
|
|
|
cPtr: rocksDbPtr,
|
|
|
|
path: path,
|
|
|
|
dbOpts: dbOpts,
|
|
|
|
readOpts: readOpts,
|
|
|
|
writeOpts: writeOpts,
|
2024-06-28 02:04:37 +00:00
|
|
|
cfDescriptors: cfs,
|
2024-03-05 03:12:37 +00:00
|
|
|
ingestOptsPtr: rocksdb_ingestexternalfileoptions_create(),
|
2024-06-26 17:31:39 +00:00
|
|
|
defaultCfHandle: cfTable.get(DEFAULT_COLUMN_FAMILY_NAME),
|
|
|
|
cfTable: cfTable,
|
2024-06-26 15:00:10 +00:00
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
ok(db)
|
|
|
|
|
|
|
|
proc openRocksDbReadOnly*(
|
2024-06-26 15:00:10 +00:00
|
|
|
path: string,
|
2024-06-28 02:04:37 +00:00
|
|
|
dbOpts = defaultDbOptions(autoClose = true),
|
|
|
|
readOpts = defaultReadOptions(autoClose = true),
|
2024-06-26 15:00:10 +00:00
|
|
|
columnFamilies: openArray[ColFamilyDescriptor] = [],
|
|
|
|
errorIfWalFileExists = false,
|
|
|
|
): RocksDBResult[RocksDbReadOnlyRef] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Open a RocksDB instance in read-only mode. If `columnFamilies` is empty
|
|
|
|
## then it will open the default column family. If `dbOpts` or `readOpts` are
|
2024-06-28 02:04:37 +00:00
|
|
|
## not supplied then the default options will be used.
|
|
|
|
## These default options will be closed when the database is closed.
|
|
|
|
## If any options are provided, they will need to be closed manually.
|
|
|
|
##
|
|
|
|
## By default, column families will be created if they don't yet exist.
|
|
|
|
## If the database already contains any column families, then all or
|
|
|
|
## a subset of the existing column families can be opened for reading.
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-03-07 01:02:24 +00:00
|
|
|
var cfs = columnFamilies.toSeq()
|
|
|
|
if DEFAULT_COLUMN_FAMILY_NAME notin columnFamilies.mapIt(it.name()):
|
2024-06-28 02:04:37 +00:00
|
|
|
cfs.add(defaultColFamilyDescriptor(autoClose = true))
|
2024-03-05 03:12:37 +00:00
|
|
|
|
|
|
|
var
|
2024-03-07 01:02:24 +00:00
|
|
|
cfNames = cfs.mapIt(it.name().cstring)
|
|
|
|
cfOpts = cfs.mapIt(it.options.cPtr)
|
|
|
|
cfHandles = newSeq[ColFamilyHandlePtr](cfs.len)
|
2024-03-05 03:12:37 +00:00
|
|
|
errors: cstring
|
|
|
|
let rocksDbPtr = rocksdb_open_for_read_only_column_families(
|
2024-06-28 02:04:37 +00:00
|
|
|
dbOpts.cPtr,
|
2024-06-26 15:00:10 +00:00
|
|
|
path.cstring,
|
|
|
|
cfNames.len().cint,
|
|
|
|
cast[cstringArray](cfNames[0].addr),
|
|
|
|
cfOpts[0].addr,
|
|
|
|
cfHandles[0].addr,
|
|
|
|
errorIfWalFileExists.uint8,
|
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-06-28 02:04:37 +00:00
|
|
|
bailOnErrors(errors, dbOpts, readOpts, cfDescriptors = cfs)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-06-13 15:43:33 +00:00
|
|
|
let
|
2024-06-26 17:31:39 +00:00
|
|
|
cfTable = newColFamilyTable(cfNames.mapIt($it), cfHandles)
|
2024-06-13 15:43:33 +00:00
|
|
|
db = RocksDbReadOnlyRef(
|
2024-03-05 03:12:37 +00:00
|
|
|
lock: createLock(),
|
|
|
|
cPtr: rocksDbPtr,
|
|
|
|
path: path,
|
|
|
|
dbOpts: dbOpts,
|
|
|
|
readOpts: readOpts,
|
2024-06-28 02:04:37 +00:00
|
|
|
cfDescriptors: cfs,
|
2024-06-26 17:31:39 +00:00
|
|
|
defaultCfHandle: cfTable.get(DEFAULT_COLUMN_FAMILY_NAME),
|
|
|
|
cfTable: cfTable,
|
2024-06-26 15:00:10 +00:00
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
ok(db)
|
|
|
|
|
2024-06-26 17:31:39 +00:00
|
|
|
proc getColFamilyHandle*(
|
|
|
|
db: RocksDbRef, name: string
|
|
|
|
): RocksDBResult[ColFamilyHandleRef] =
|
|
|
|
let cfHandle = db.cfTable.get(name)
|
|
|
|
if cfHandle.isNil():
|
|
|
|
err("rocksdb: unknown column family")
|
|
|
|
else:
|
|
|
|
ok(cfHandle)
|
|
|
|
|
2024-03-05 03:12:37 +00:00
|
|
|
proc isClosed*(db: RocksDbRef): bool {.inline.} =
|
|
|
|
## Returns `true` if the database has been closed and `false` otherwise.
|
|
|
|
db.cPtr.isNil()
|
|
|
|
|
|
|
|
proc cPtr*(db: RocksDbRef): RocksDbPtr {.inline.} =
|
|
|
|
## Get the underlying database pointer.
|
|
|
|
doAssert not db.isClosed()
|
|
|
|
db.cPtr
|
|
|
|
|
|
|
|
proc get*(
|
|
|
|
db: RocksDbRef,
|
|
|
|
key: openArray[byte],
|
|
|
|
onData: DataProc,
|
2024-06-26 17:31:39 +00:00
|
|
|
cfHandle = db.defaultCfHandle,
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[bool] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Get the value for the given key from the specified column family.
|
|
|
|
## If the value does not exist, `false` will be returned in the result
|
|
|
|
## and `onData` will not be called. If the value does exist, `true` will be
|
|
|
|
## returned in the result and `onData` will be called with the value.
|
|
|
|
## The `onData` callback reduces the number of copies and therefore should be
|
|
|
|
## preferred if performance is required.
|
|
|
|
|
|
|
|
if key.len() == 0:
|
|
|
|
return err("rocksdb: key is empty")
|
|
|
|
|
|
|
|
var
|
|
|
|
len: csize_t
|
|
|
|
errors: cstring
|
|
|
|
let data = rocksdb_get_cf(
|
2024-06-26 15:00:10 +00:00
|
|
|
db.cPtr,
|
|
|
|
db.readOpts.cPtr,
|
|
|
|
cfHandle.cPtr,
|
|
|
|
cast[cstring](unsafeAddr key[0]),
|
|
|
|
csize_t(key.len),
|
|
|
|
len.addr,
|
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
bailOnErrors(errors)
|
|
|
|
|
|
|
|
if data.isNil():
|
|
|
|
doAssert len == 0
|
|
|
|
ok(false)
|
|
|
|
else:
|
|
|
|
onData(toOpenArrayByte(data, 0, len.int - 1))
|
|
|
|
rocksdb_free(data)
|
|
|
|
ok(true)
|
|
|
|
|
|
|
|
proc get*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbRef, key: openArray[byte], cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[seq[byte]] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Get the value for the given key from the specified column family.
|
|
|
|
## If the value does not exist, an empty error will be returned in the result.
|
|
|
|
## If the value does exist, the value will be returned in the result.
|
|
|
|
|
|
|
|
var dataRes: RocksDBResult[seq[byte]]
|
2024-06-26 15:00:10 +00:00
|
|
|
proc onData(data: openArray[byte]) =
|
|
|
|
dataRes.ok(@data)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-06-26 17:31:39 +00:00
|
|
|
let res = db.get(key, onData, cfHandle)
|
2024-03-05 03:12:37 +00:00
|
|
|
if res.isOk():
|
|
|
|
return dataRes
|
|
|
|
|
|
|
|
dataRes.err(res.error())
|
|
|
|
|
|
|
|
proc put*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbReadWriteRef, key, val: openArray[byte], cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[void] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Put the value for the given key into the specified column family.
|
|
|
|
|
|
|
|
if key.len() == 0:
|
|
|
|
return err("rocksdb: key is empty")
|
|
|
|
|
|
|
|
var errors: cstring
|
|
|
|
rocksdb_put_cf(
|
2024-06-26 15:00:10 +00:00
|
|
|
db.cPtr,
|
|
|
|
db.writeOpts.cPtr,
|
|
|
|
cfHandle.cPtr,
|
|
|
|
cast[cstring](unsafeAddr key[0]),
|
|
|
|
csize_t(key.len),
|
|
|
|
cast[cstring](if val.len > 0:
|
|
|
|
unsafeAddr val[0]
|
|
|
|
else:
|
|
|
|
nil
|
|
|
|
),
|
|
|
|
csize_t(val.len),
|
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
bailOnErrors(errors)
|
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
proc keyExists*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbRef, key: openArray[byte], cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[bool] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Check if the key exists in the specified column family.
|
|
|
|
## Returns a result containing `true` if the key exists or a result
|
|
|
|
## containing `false` otherwise.
|
|
|
|
|
|
|
|
# TODO: Call rocksdb_key_may_exist_cf to improve performance for the case
|
|
|
|
# when the key does not exist
|
|
|
|
|
2024-06-26 15:00:10 +00:00
|
|
|
db.get(
|
|
|
|
key,
|
|
|
|
proc(data: openArray[byte]) =
|
|
|
|
discard
|
|
|
|
,
|
2024-06-26 17:31:39 +00:00
|
|
|
cfHandle,
|
2024-06-26 15:00:10 +00:00
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
|
|
|
proc delete*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbReadWriteRef, key: openArray[byte], cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[void] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Delete the value for the given key from the specified column family.
|
|
|
|
## If the value does not exist, the delete will be a no-op.
|
|
|
|
## To check if the value exists before or after a delete, use `keyExists`.
|
|
|
|
|
|
|
|
if key.len() == 0:
|
|
|
|
return err("rocksdb: key is empty")
|
|
|
|
|
|
|
|
var errors: cstring
|
|
|
|
rocksdb_delete_cf(
|
2024-06-26 15:00:10 +00:00
|
|
|
db.cPtr,
|
|
|
|
db.writeOpts.cPtr,
|
|
|
|
cfHandle.cPtr,
|
|
|
|
cast[cstring](unsafeAddr key[0]),
|
|
|
|
csize_t(key.len),
|
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
bailOnErrors(errors)
|
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
proc openIterator*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbRef, cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[RocksIteratorRef] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Opens an `RocksIteratorRef` for the specified column family.
|
|
|
|
doAssert not db.isClosed()
|
|
|
|
|
2024-06-26 15:00:10 +00:00
|
|
|
let rocksIterPtr =
|
|
|
|
rocksdb_create_iterator_cf(db.cPtr, db.readOpts.cPtr, cfHandle.cPtr)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
|
|
|
ok(newRocksIterator(rocksIterPtr))
|
|
|
|
|
|
|
|
proc openWriteBatch*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbReadWriteRef, cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): WriteBatchRef =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Opens a `WriteBatchRef` which defaults to using the specified column family.
|
|
|
|
doAssert not db.isClosed()
|
|
|
|
|
2024-06-26 17:31:39 +00:00
|
|
|
newWriteBatch(cfHandle)
|
2024-03-05 03:12:37 +00:00
|
|
|
|
2024-06-26 15:00:10 +00:00
|
|
|
proc write*(db: RocksDbReadWriteRef, updates: WriteBatchRef): RocksDBResult[void] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Apply the updates in the `WriteBatchRef` to the database.
|
|
|
|
doAssert not db.isClosed()
|
|
|
|
|
|
|
|
var errors: cstring
|
|
|
|
rocksdb_write(
|
2024-06-26 15:00:10 +00:00
|
|
|
db.cPtr, db.writeOpts.cPtr, updates.cPtr, cast[cstringArray](errors.addr)
|
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
bailOnErrors(errors)
|
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
proc ingestExternalFile*(
|
2024-06-26 17:31:39 +00:00
|
|
|
db: RocksDbReadWriteRef, filePath: string, cfHandle = db.defaultCfHandle
|
2024-06-26 15:00:10 +00:00
|
|
|
): RocksDBResult[void] =
|
2024-03-05 03:12:37 +00:00
|
|
|
## Ingest an external sst file into the database. The file will be ingested
|
|
|
|
## into the specified column family or the default column family if none is
|
|
|
|
## provided.
|
|
|
|
doAssert not db.isClosed()
|
|
|
|
|
|
|
|
var
|
|
|
|
sstPath = filePath.cstring
|
|
|
|
errors: cstring
|
|
|
|
rocksdb_ingest_external_file_cf(
|
|
|
|
db.cPtr,
|
|
|
|
cfHandle.cPtr,
|
2024-06-26 15:00:10 +00:00
|
|
|
cast[cstringArray](sstPath.addr),
|
|
|
|
csize_t(1),
|
2024-03-05 03:12:37 +00:00
|
|
|
db.ingestOptsPtr,
|
2024-06-26 15:00:10 +00:00
|
|
|
cast[cstringArray](errors.addr),
|
|
|
|
)
|
2024-03-05 03:12:37 +00:00
|
|
|
bailOnErrors(errors)
|
|
|
|
|
|
|
|
ok()
|
|
|
|
|
|
|
|
proc close*(db: RocksDbRef) =
|
|
|
|
## Close the `RocksDbRef` which will release the connection to the database
|
|
|
|
## and free the memory associated with it. `close` is idempotent and can
|
|
|
|
## safely be called multple times. `close` is a no-op if the `RocksDbRef`
|
|
|
|
## is already closed.
|
|
|
|
|
|
|
|
withLock(db.lock):
|
|
|
|
if not db.isClosed():
|
2024-06-28 02:04:37 +00:00
|
|
|
# the column families should be closed before the database
|
2024-03-05 03:12:37 +00:00
|
|
|
db.cfTable.close()
|
|
|
|
|
2024-06-28 02:04:37 +00:00
|
|
|
rocksdb_close(db.cPtr)
|
|
|
|
db.cPtr = nil
|
|
|
|
|
|
|
|
# opts should be closed after the database is closed
|
|
|
|
if db.dbOpts.autoClose:
|
|
|
|
db.dbOpts.close()
|
|
|
|
if db.readOpts.autoClose:
|
|
|
|
db.readOpts.close()
|
|
|
|
|
|
|
|
for cfDesc in db.cfDescriptors:
|
|
|
|
if cfDesc.autoClose:
|
|
|
|
cfDesc.close()
|
|
|
|
|
2024-03-05 03:12:37 +00:00
|
|
|
if db of RocksDbReadWriteRef:
|
|
|
|
let db = RocksDbReadWriteRef(db)
|
2024-06-28 02:04:37 +00:00
|
|
|
if db.writeOpts.autoClose:
|
|
|
|
db.writeOpts.close()
|
|
|
|
|
2024-03-05 03:12:37 +00:00
|
|
|
rocksdb_ingestexternalfileoptions_destroy(db.ingestOptsPtr)
|
|
|
|
db.ingestOptsPtr = nil
|