rocksdb-node v1.7.2
RocksDB
Node native binding for RocksDB. Aims to be a full nodejs mapping for https://github.com/facebook/rocksdb/wiki. Developed against RocksDB 5.2.
Installation
First install RocksDB:
Mac: brew install rocksdb
Linux and other platforms, see RocksDB install guide.
Then npm install rocksdb-node:
npm i rocksdb-node --save
Sample Usage
Sync examples:
const rocksdb = require('rocksdb-node')
const db = rocksdb.open({create_if_missing: true}, '/tmp/my-rocks-database')
db.put('node', 'rocks')
const value = db.get('node')
db.del('node')
// both keys and vals as buffers
const key = fs.readFileSync('./test/fixtures/beach-thumb.jpg')
const val = fs.readFileSync('./test/fixtures/beach.jpg')
db.put(key, val)
const beach = db.get({buffer: true}, key)
fs.writeFileSync('/tmp/beach2.jpg', beach)
Async examples:
const rocksdb = require('rocksdb-node')
const db = rocksdb.open({create_if_missing: true}, '/tmp/my-rocks-database')
db.put('node', 'rocks', function(err) {
if (err) return console.error(err);
db.get('node', function (err, val) {
if (err) return console.error(err);
console.log(val)
db.del('node', function(err){
if (err) return console.error(err);
})
})
})
API
'rocksdb-node' currently exports the following functions:
{
open,
openDBWithTTL,
listColumnFamilies
destroyDB
}
The open
function opens a database and returns an object that you use to call further functions on the database.
Open
const rocksdb = require('rocksdb-node')
const db = rocksdb.open({create_if_missing: true}, '/tmp/my-rocks-database')
Note that passing an options object is required, even if it's empty.
All boolean and int options as defined in DBOptions are supported, e.g.
const opts = {
create_if_missing: true,
error_if_exists: false,
create_missing_column_families: false,
paranoid_checks: true,
use_fsync: false,
allow_mmap_reads: false,
allow_mmap_writes: false,
use_direct_reads: false,
allow_fallocate: true,
is_fd_close_on_exec: true,
advise_random_on_open: true,
new_table_reader_for_compaction_inputs: true,
use_adaptive_mutex: false,
enable_thread_tracking: false,
allow_concurrent_memtable_write: true,
enable_write_thread_adaptive_yield: true,
skip_stats_update_on_db_open: false,
allow_2pc: false,
fail_if_options_file_error: false,
dump_malloc_stats: false,
avoid_flush_during_recovery: false,
avoid_flush_during_shutdown: false,
max_open_files: -1,
max_file_opening_threads: 16,
max_total_wal_size: 0,
base_background_compactions: 1,
max_background_compactions: 1,
max_subcompactions: 1,
max_background_flushes: 1,
max_log_file_size: 0,
log_file_time_to_roll: 0,
keep_log_file_num: 1000,
recycle_log_file_num: 0,
table_cache_numshardbits: 0,
WAL_ttl_seconds: 0,
WAL_size_limit_MB: 0,
manifest_preallocation_size: 4 * 1024 * 1024,
db_write_buffer_size: 0,
compaction_readahead_size: 0,
random_access_max_buffer_size: 1024 * 1024,
writable_file_max_buffer_size: 1024 * 1024,
bytes_per_sync: 0,
wal_bytes_per_sync: 0,
write_thread_max_yield_usec: 100,
write_thread_slow_yield_usec: 3
}
Open ReadOnly
Open database for read only. All DB interfaces that modify data, like put/delete, will return error.
const dbRO = rocksdb.open({readOnly: true}, './myrocks') // myrocks must already exist
Put
db.put(<options>, <column-family>, key, value, <callback>)
where <options>
, <column-family>
and <callback>
are optional. If no callback is passed the method is synchronous.
try {
db.put('foo', 'bar')
} catch(e)...
or
db.put('foo', 'bar', function(err){...})
The key and value params for put
can both be either a string or a buffer.
const key = fs.readFileSync('./test/fixtures/beach-thumb.jpg')
const val = fs.readFileSync('./test/fixtures/beach.jpg')
db.put(key, val)
// or
db.put('foo', val)
// or
db.put(key, 'bar')
put
optionally takes an options object as it's first parameter. All boolean and int options as defined in WriteOptions are supported, e.g.
const writeOpts = {
sync: false,
disableWAL: false,
ignore_missing_column_families: false,
no_slowdown: false
}
db.put(writeOpts, 'foo', 'bar')
Column Families are also supported for put:
db.createColumnFamily('myFamily')
db.put('myFamily', 'foo', 'bar')
Get
db.get(<options>, <column-family>, key, <callback>)
where <options>
, <column-family>
and <callback>
are optional. If no callback is passed the method is synchronous.
try {
const value = db.get('foo')
} catch(e)...
or
db.get('foo', function(err, value){...})
The key param for get
can be either a string or a buffer, and if the value you are expecting to get is a buffer, you must pass the option buffer:true
, e.g.
const key = fs.readFileSync('./test/fixtures/beach-thumb.jpg')
const valueBuffer = db.get({buffer:true}, key)
get
optionally takes an options object as it's first parameter. All boolean and int options as defined in ReadOptions are supported, e.g.
const readOpts = {
verify_checksums: true,
fill_cache: true,
tailing: false,
managed: false,
total_order_seek: false,
prefix_same_as_start: false,
pin_data: false,
background_purge_on_iterator_cleanup: false,
readahead_size: 0,
ignore_range_deletions: false
}
const bar = db.get(readOpts, 'foo')
Column Families are also supported for put:
db.createColumnFamily('myFamily')
db.put('myFamily', 'foo', 'bar')
var value = db.get('myFamily', 'foo')
MultiGet
The MultiGet API allows an application to retrieve a bunch of keys from the database.
db.multiGet(<options>, <column-family>, [keys], <callback>)
where <options>
, <column-family>
and <callback>
are optional. If no callback is passed the method is synchronous.
try {
const values = db.multiGet(['foo', 'foo1'])
console.log('foo - ', values[0])
} catch(e)...
or
db.multiGet(['foo'], function(err, values){...})
As with get
, key params be either string or buffer. If the values you are expecting are buffers, you must pass the option buffer:true
, e.g.
const key = fs.readFileSync('./test/fixtures/beach-thumb.jpg')
const valueBuffers = db.multiGet({buffer:true}, [key])
multiGet
optionally takes an options object as it's first parameter. All boolean and int options as defined in ReadOptions are supported, e.g.
const readOpts = {
verify_checksums: true,
fill_cache: true,
tailing: false,
managed: false,
total_order_seek: false,
prefix_same_as_start: false,
pin_data: false,
background_purge_on_iterator_cleanup: false,
readahead_size: 0,
ignore_range_deletions: false
}
const vals = db.multiGet(readOpts, ['foo1', 'foo2'])
Column Families are also supported:
db.createColumnFamily('myFamily')
db.put('myFamily', 'foo', 'bar')
var values = db.multiGet('myFamily', ['foo'])
Delete
db.del(<options>, <column-family>, key, <callback>)
where <options>
, <column-family>
and <callback>
are optional. If no callback is passed the method is synchronous.
try {
db.del('foo')
} catch(e)...
or
db.del('foo', function(err){...})
The key param for del
can be either a string or a buffer, e.g.
const key = fs.readFileSync('./test/fixtures/beach-thumb.jpg')
db.del(key)
del
optionally takes an options object as it's first parameter, it also takes a WriteOptions (like put
), e.g.
const writeOpts = {
sync: false,
disableWAL: false,
ignore_missing_column_families: false,
no_slowdown: false
}
db.del(writeOpts, 'foo')
Column Families are also supported for del:
db.createColumnFamily('myFamily')
db.put('myFamily', 'foo', 'bar')
db.del('myFamily', 'foo')
Close
db.cose()
Closes an open database. Any subsequent usage of the db
will result in a 'Database is not open' Error thrown.
Destroy Database
rocksdb.destroyDB(path, <options>)
where <options>
is optional argument. This destroys a database, fully removing it from the file system. Database must already be closed before it can be destroyed.
const path = os.tmpdir() + '/rocksdbDestroyTest'
const db = rocksdb.open({create_if_missing: true}, path)
...
...
db.close()
rocksdb.destroyDB(path)
t.ok(!fs.existsSync(path))
Iteration
The Iterator API matches the Rocks Iterator API. Note the Iterator API is synchronous.
const it = db.newIterator()
for (it.seekToFirst(); it.valid(); it.next()) {
console.log(it.key(), it.value())
}
const err = it.status()
if (err) throw err;
The following Rocks Iterator API is supported (documentation has been copied from there):
db.newIterator()
db.newIterator(<read-options>, <column-family>)
where and optional.
Creates a new Iterator for the current database. Optionally takes ReadOptions or a ColumnFamily name.
const readOpts = {
verify_checksums: true
}
const iter = db.newIterator(readOpts)
Can also optionally take a Column Family Name, e.g.
db.createColumnFamily('foo')
db.put('foo', 'hello', 'world')
const iter = db.newIterator('foo')
...
iterator.valid()
An iterator is either positioned at a key/value pair, or not valid. This method returns true iff the iterator is valid.
iterator.seekToFirst()
Position at the first key in the source. The iterator is Valid() after this call iff the source is not empty.
iterator.seekToLast()
Position at the last key in the source. The iterator is Valid() after this call iff the source is not empty.
iterator.seek()
Position at the first key in the source that at or past target. The iterator is Valid() after this call iff the source contains an entry that comes at or past target.
db.put('1', 'one')
db.put('2', 'two')
db.put('3', 'three')
const it = db.newIterator()
for (it.seek('2'); it.valid(); it.next()) {
console.log(iter.key(), iter.value())
}
iterator.seekForPrev()
Position at the last key in the source that at or before target. The iterator is Valid() after this call iff the source contains an entry that comes at or before target.
iterator.next()
Moves to the next entry in the source. After this call, Valid() is true iff the iterator was not positioned at the last entry in the source.
iterator.prev()
Moves to the previous entry in the source. After this call, Valid() is true iff the iterator was not positioned at the first entry in source.
iterator.key()
Return the key for the current entry. The underlying storage for the returned slice is valid only until the next modification of the iterator.
Note if the key is a buffer, you need to pass the buffer:true
flag
for (it.seekToFirst(); it.valid(); it.next()) {
const k = it.key({buffer: true})
const v = it.value({buffer: true})
...
iterator.value()
Return the value for the current entry. The underlying storage for the returned slice is valid only until the next modification of the iterator.
As with iterator.key
if the value is a buffer, you need to pass the buffer:true
flag
iterator.status()
If an error has occurred, return it. A javascript Error object is returned if an error occurred, otherwise null.
const err = it.status()
if (err) throw err;
db.releaseIterator()
db.releaseIterator(iterator)
, releases the iterator, this should be called when you are finished iterating.
const iter = db.newIterator(readOpts)
...
...
db.releaseIterator(iter)
Column Families
The Column Family API mostly matches the Rocks Column Families, with some additional utility methods. Open, Put, Get, Delete all support Column Families. When a database is opened, it is queried for it's Column Families and all are opened.
getColumnFamilies
Get all the Column Families in an open database. Returns a javascript array containing all the Column Family names:
db = rocksdb.open({create_if_missing: true}, './myrocks')
const families = db.getColumnFamilies();
console.log(families)
createColumnFamily
Creates a new Column Family:
db.createColumnFamily('myFamily')
db.put('myFamily', 'foo', 'bar')
dropColumnFamily
Drops a Column Family:
db.dropColumnFamily('myFamily')
Compact Range
Support for Compaction.
db.compactRange(<compactrange-options>, <column-family>, <from>, <to>, <callback>)
where all parameters are optional, i.e. the following combinations are all possible:
db.compactRange()
db.compactRange(options)
db.compactRange(callback)
db.compactRange(columnFamily)
db.compactRange(from, to)
db.compactRange(options, callback)
db.compactRange(options, from, to)
db.compactRange(columnFamily, from, to)
db.compactRange(options, columnFamily, callback)
db.compactRange(options, columnFamily, from, to)
db.compactRange(options, from, to, callback)
db.compactRange(columnFamily, from, to, callback)
db.compactRange(options, columnFamily, from, to, callback)
The following options are supported:
const opts = {
// If true, no other compaction will run at the same time as this manual compaction
exclusive_manual_compaction: true,
// If true, compacted files will be moved to the minimum level capable of holding the data or given level (specified non-negative target_level).
change_level: false,
// If change_level is true and target_level have non-negative value, compacted files will be moved to target_level.
target_level: -1,
// Compaction outputs will be placed in options.db_paths[target_path_id]. Behavior is undefined if target_path_id is out of range.
target_path_id: 0
}
Example usage:
db.put('foo1', 'bar1')
db.put('foo2', 'bar2')
db.put('foo3', 'bar3')
db.compactRange()
db.compactRange('foo1', 'foo3')
db.compactRange((err) => {
...
openDBWithTTL
Opens a DBWithTTL. This database is a specialized form of the main RocksDB database (returned by open
), it inherits all the regular RocksDB database functions.
rocksdb.openDBWithTTL(<options>, <path>, <ttl>)
, where <ttl>
is measured in seconds. E.g.
const dbWithTTL = rocksdb.openDBWithTTL({create_if_missing: true}, path, 1)
All the same options that can be passed to #open() can be passed here.
Note: the TTL passed applies to all Column Families.
Example:
const db = rocksdb.openDBWithTTL({create_if_missing: true}, path, 1) // 1 second ttl..
db.put('foo', 'bar')
console.log(db.get('foo'))
// sleep for 2 seconds, then foo should be removed (after compact)
setTimeout(() => {
db.compactRange()
console.log(db.get('foo'))
db.close()
}, 2000);
List Column Families
It's also possible to query a database for it's Column Families without opening the database:
const rocksdb = require('rocksdb-node')
const families = rocksdb.listColumnFamilies('./myrocks')
Note listColumnFamilies
can also take the same options that you can pass to open, e.g.
const families = rocksdb.listColumnFamilies({paranoid_checks: true}, './myrocks')
Batch Updates
Support for Batch Updates:
const batch = db.batch()
batch.put('k1', 'v1')
batch.del('k1')
db.write(batch)
Note that the batch API is synchronous. Column Families are supported:
db.createColumnFamily('foo')
const batch = db.batch()
batch.put('foo', 'k1', 'v1')
batch.del('foo', 'k1')
db.write(batch)
Note also that WriteOptions
can be passed to write
, e.g.
const opts = {
sync: false,
}
const batch = db.batch()
batch.put('k2', 'v2')
db.write(opts, batch)
Snapshots
Support for Snapshots. Usage:
const db = rocksdb.open({create_if_missing: true}, path)
db.put('foo', 'bar')
const readOptions = {}
readOptions.snapshot = db.getSnapshot()
// these will not be part of the iterator
db.put('foo2', 'bar2')
const it = db.newIterator(readOptions)
for (it.seekToFirst(); it.valid(); it.next())
....
....
db.releaseIterator(it)
db.releaseSnapshot(readOptions.snapshot)
Creating and Ingesting SST Files
Support for Creating and Ingesting SST Files. Usage:
const db = rocksdb.open({create_if_missing: true}, path)
const sstFileWriter = db.getSstFileWriter()
sstFileWriter.open('/tmp/file1')
sstFileWriter.add('a', 'b')
...
sstFileWriter.finish()
db.ingestExternalFile('/tmp/file1')
console.log(db.get('a')) // 'b' now exists