Merge pull request #88 from alexreg/cleanup

Mainly the addition of the `ffi_try!` macro, plus a few minor things @kaedroho mentioned in the comments for the PR you already accepted.
master
Tyler Neely 8 years ago committed by GitHub
commit cfc3cbb551
  1. 1
      CHANGELOG.txt
  2. 10
      README.md
  3. 1
      rocksdb-sys/src/lib.rs
  4. 1
      rocksdb-sys/src/test.rs
  5. 1
      rocksdb-sys/tests/ffi.rs
  6. 38
      src/ffi_util.rs
  7. 8
      src/lib.rs
  8. 109
      src/rocksdb.rs
  9. 15
      test/test.rs
  10. 1
      test/test_column_family.rs
  11. 15
      test/test_iterator.rs
  12. 15
      test/test_multithreaded.rs
  13. 16
      test/test_rocksdb_options.rs

@ -4,5 +4,6 @@ Changelog
0.5 (in development) 0.5 (in development)
~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~
* All imports of `rocksdb::rocksdb::*` should now be simply `rocksdb::*` (alexreg)
* Add iterator_cf to snapshot (jezell) * Add iterator_cf to snapshot (jezell)
* Changelog started * Changelog started

@ -3,9 +3,9 @@ rust-rocksdb
[![Build Status](https://travis-ci.org/spacejam/rust-rocksdb.svg?branch=master)](https://travis-ci.org/spacejam/rust-rocksdb) [![Build Status](https://travis-ci.org/spacejam/rust-rocksdb.svg?branch=master)](https://travis-ci.org/spacejam/rust-rocksdb)
[![crates.io](http://meritbadge.herokuapp.com/rocksdb)](https://crates.io/crates/rocksdb) [![crates.io](http://meritbadge.herokuapp.com/rocksdb)](https://crates.io/crates/rocksdb)
This library has been tested against RocksDB 3.13.1 on linux and OSX. The 0.4.1 crate should work with the Rust 1.9 stable and nightly releases as of 7/1/16. This library has been tested against RocksDB 3.13.1 on Linux and OS X. The 0.4.1 crate should work with the Rust 1.9 stable and nightly releases as of 7/1/16.
### status ### Status
- [x] basic open/put/get/delete/close - [x] basic open/put/get/delete/close
- [x] rustic merge operator - [x] rustic merge operator
- [x] write batch (thanks @dgrnbrg!) - [x] write batch (thanks @dgrnbrg!)
@ -46,7 +46,7 @@ fn main() {
} }
``` ```
###### Doing an atomic commit of several writes ###### Making an atomic commit of several writes
```rust ```rust
extern crate rocksdb; extern crate rocksdb;
use rocksdb::{DB, WriteBatch, Writable}; use rocksdb::{DB, WriteBatch, Writable};
@ -64,7 +64,7 @@ fn main() {
} }
``` ```
###### Getting an Iterator ###### Getting an `Iterator`
```rust ```rust
extern crate rocksdb; extern crate rocksdb;
use rocksdb::{DB, Direction, IteratorMode}; use rocksdb::{DB, Direction, IteratorMode};
@ -93,7 +93,7 @@ fn main() {
} }
``` ```
###### Getting an Iterator from a Snapshot ###### Getting an `Iterator` from a `Snapshot`
```rust ```rust
extern crate rocksdb; extern crate rocksdb;
use rocksdb::{DB, Direction}; use rocksdb::{DB, Direction};

@ -1,4 +1,3 @@
//
// Copyright 2014 Tyler Neely, 2016 Alex Regueiro // Copyright 2014 Tyler Neely, 2016 Alex Regueiro
// //
// Licensed under the Apache License, Version 2.0 (the "License"); // Licensed under the Apache License, Version 2.0 (the "License");

@ -1,4 +1,3 @@
//
// Copyright 2014 Tyler Neely // Copyright 2014 Tyler Neely
// //
// Licensed under the Apache License, Version 2.0 (the "License"); // Licensed under the Apache License, Version 2.0 (the "License");

@ -1,4 +1,3 @@
//
// Copyright 2016 Alex Regueiro // Copyright 2016 Alex Regueiro
// //
// Licensed under the Apache License, Version 2.0 (the "License"); // Licensed under the Apache License, Version 2.0 (the "License");

@ -0,0 +1,38 @@
// Copyright 2016 Alex Regueiro
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
use libc::{self, c_char, c_void};
use std::ffi::{CStr};
use std::str;
pub fn error_message(ptr: *const c_char) -> String {
let cstr = unsafe { CStr::from_ptr(ptr as *const _) };
let s = str::from_utf8(cstr.to_bytes()).unwrap().to_owned();
unsafe {
libc::free(ptr as *mut c_void);
}
s
}
macro_rules! ffi_try {
( $($function:ident)::*( $( $arg:expr ),* ) ) => ({
let mut err: *mut ::libc::c_char = ::std::ptr::null_mut();
let result = $($function)::*($($arg),*, &mut err);
if !err.is_null() {
return Err(Error::new($crate::ffi_util::error_message(err)));
}
result
})
}

@ -16,13 +16,15 @@
extern crate libc; extern crate libc;
extern crate rocksdb_sys as ffi; extern crate rocksdb_sys as ffi;
pub mod merge_operator; #[macro_use]
pub mod comparator; mod ffi_util;
pub mod comparator;
pub mod merge_operator;
mod rocksdb; mod rocksdb;
mod rocksdb_options; mod rocksdb_options;
pub use rocksdb::{DB, DBCompressionType, DBCompactionStyle, DBRecoveryMode, DBIterator, DBVector, Direction, IteratorMode, Writable, WriteBatch, Error, new_bloom_filter}; pub use rocksdb::{DB, DBCompressionType, DBCompactionStyle, DBRecoveryMode, DBIterator, DBVector, Direction, Error, IteratorMode, Writable, WriteBatch, Snapshot, new_bloom_filter};
pub use merge_operator::MergeOperands; pub use merge_operator::MergeOperands;
pub struct BlockBasedOptions { pub struct BlockBasedOptions {

@ -14,29 +14,20 @@
// //
use std::collections::BTreeMap; use std::collections::BTreeMap;
use std::ffi::{CStr, CString}; use std::ffi::{CString};
use std::fmt;
use std::fs; use std::fs;
use std::ops::Deref; use std::ops::Deref;
use std::path::{Path, PathBuf}; use std::path::{Path, PathBuf};
use std::ptr; use std::ptr;
use std::slice; use std::slice;
use std::str::from_utf8; use std::str;
use std::fmt;
use libc::{self, c_char, c_uchar, c_int, c_void, size_t}; use libc::{self, c_char, c_uchar, c_int, c_void, size_t};
use {Options, WriteOptions}; use {Options, WriteOptions};
use ffi; use ffi;
fn error_message(ptr: *const i8) -> String {
let c_str = unsafe { CStr::from_ptr(ptr as *const _) };
let s = from_utf8(c_str.to_bytes()).unwrap().to_owned();
unsafe {
libc::free(ptr as *mut c_void);
}
s
}
pub fn new_bloom_filter(bits: c_int) -> *mut ffi::rocksdb_filterpolicy_t { pub fn new_bloom_filter(bits: c_int) -> *mut ffi::rocksdb_filterpolicy_t {
unsafe { ffi::rocksdb_filterpolicy_create_bloom(bits) } unsafe { ffi::rocksdb_filterpolicy_create_bloom(bits) }
} }
@ -317,13 +308,12 @@ impl DB {
return Err(Error::new(format!("Failed to create rocksdb directory: {:?}", e))); return Err(Error::new(format!("Failed to create rocksdb directory: {:?}", e)));
} }
let mut err: *mut c_char = ptr::null_mut();
let db: *mut ffi::rocksdb_t; let db: *mut ffi::rocksdb_t;
let mut cf_map = BTreeMap::new(); let mut cf_map = BTreeMap::new();
if cfs.len() == 0 { if cfs.len() == 0 {
unsafe { unsafe {
db = ffi::rocksdb_open(opts.inner, cpath_ptr as *const _, &mut err); db = ffi_try!(ffi::rocksdb_open(opts.inner, cpath_ptr as *const _));
} }
} else { } else {
let mut cfs_v = cfs.to_vec(); let mut cfs_v = cfs.to_vec();
@ -344,7 +334,7 @@ impl DB {
let cfopts: Vec<_> = cfs_v.iter().map(|_| unsafe { ffi::rocksdb_options_create() as *const _ }).collect(); let cfopts: Vec<_> = cfs_v.iter().map(|_| unsafe { ffi::rocksdb_options_create() as *const _ }).collect();
unsafe { unsafe {
db = ffi::rocksdb_open_column_families(opts.inner, cpath_ptr as *const _, cfs_v.len() as c_int, cfnames.as_ptr() as *const _, cfopts.as_ptr(), cfhandles.as_mut_ptr(), &mut err); db = ffi_try!(ffi::rocksdb_open_column_families(opts.inner, cpath_ptr as *const _, cfs_v.len() as c_int, cfnames.as_ptr() as *const _, cfopts.as_ptr(), cfhandles.as_mut_ptr()));
} }
for handle in &cfhandles { for handle in &cfhandles {
@ -358,9 +348,6 @@ impl DB {
} }
} }
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
if db.is_null() { if db.is_null() {
return Err(Error::new("Could not initialize database.".to_owned())); return Err(Error::new("Could not initialize database.".to_owned()));
} }
@ -374,24 +361,16 @@ impl DB {
pub fn destroy<P: AsRef<Path>>(opts: &Options, path: P) -> Result<(), Error> { pub fn destroy<P: AsRef<Path>>(opts: &Options, path: P) -> Result<(), Error> {
let cpath = CString::new(path.as_ref().to_string_lossy().as_bytes()).unwrap(); let cpath = CString::new(path.as_ref().to_string_lossy().as_bytes()).unwrap();
let mut err: *mut c_char = ptr::null_mut();
unsafe { unsafe {
ffi::rocksdb_destroy_db(opts.inner, cpath.as_ptr(), &mut err); ffi_try!(ffi::rocksdb_destroy_db(opts.inner, cpath.as_ptr()));
}
if !err.is_null() {
return Err(Error::new(error_message(err)));
} }
Ok(()) Ok(())
} }
pub fn repair<P: AsRef<Path>>(opts: Options, path: P) -> Result<(), Error> { pub fn repair<P: AsRef<Path>>(opts: Options, path: P) -> Result<(), Error> {
let cpath = CString::new(path.as_ref().to_string_lossy().as_bytes()).unwrap(); let cpath = CString::new(path.as_ref().to_string_lossy().as_bytes()).unwrap();
let mut err: *mut c_char = ptr::null_mut();
unsafe { unsafe {
ffi::rocksdb_repair_db(opts.inner, cpath.as_ptr(), &mut err); ffi_try!(ffi::rocksdb_repair_db(opts.inner, cpath.as_ptr()));
}
if !err.is_null() {
return Err(Error::new(error_message(err)));
} }
Ok(()) Ok(())
} }
@ -401,12 +380,8 @@ impl DB {
} }
pub fn write_opt(&self, batch: WriteBatch, writeopts: &WriteOptions) -> Result<(), Error> { pub fn write_opt(&self, batch: WriteBatch, writeopts: &WriteOptions) -> Result<(), Error> {
let mut err: *mut c_char = ptr::null_mut();
unsafe { unsafe {
ffi::rocksdb_write(self.inner, writeopts.inner, batch.inner, &mut err); ffi_try!(ffi::rocksdb_write(self.inner, writeopts.inner, batch.inner));
}
if !err.is_null() {
return Err(Error::new(error_message(err)));
} }
Ok(()) Ok(())
} }
@ -428,11 +403,7 @@ impl DB {
unsafe { unsafe {
let mut val_len: size_t = 0; let mut val_len: size_t = 0;
let mut err: *mut c_char = ptr::null_mut(); let val = ffi_try!(ffi::rocksdb_get(self.inner, readopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, &mut val_len)) as *mut u8;
let val = ffi::rocksdb_get(self.inner, readopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, &mut val_len, &mut err) as *mut u8;
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
if val.is_null() { if val.is_null() {
Ok(None) Ok(None)
} else { } else {
@ -453,11 +424,7 @@ impl DB {
unsafe { unsafe {
let mut val_len: size_t = 0; let mut val_len: size_t = 0;
let mut err: *mut c_char = ptr::null_mut(); let val = ffi_try!(ffi::rocksdb_get_cf(self.inner, readopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t, &mut val_len)) as *mut u8;
let val = ffi::rocksdb_get_cf(self.inner, readopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t, &mut val_len, &mut err) as *mut u8;
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
if val.is_null() { if val.is_null() {
Ok(None) Ok(None)
} else { } else {
@ -477,15 +444,11 @@ impl DB {
return Err(Error::new("Failed to convert path to CString when opening rocksdb".to_owned())) return Err(Error::new("Failed to convert path to CString when opening rocksdb".to_owned()))
} }
}; };
let mut err: *mut c_char = ptr::null_mut();
let cf_handler = unsafe { let cf_handler = unsafe {
let cf_handler = ffi::rocksdb_create_column_family(self.inner, opts.inner, cname.as_ptr(), &mut err); let cf_handler = ffi_try!(ffi::rocksdb_create_column_family(self.inner, opts.inner, cname.as_ptr()));
self.cfs.insert(name.to_string(), cf_handler); self.cfs.insert(name.to_string(), cf_handler);
cf_handler cf_handler
}; };
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(cf_handler) Ok(cf_handler)
} }
@ -494,12 +457,8 @@ impl DB {
if cf.is_none() { if cf.is_none() {
return Err(Error::new(format!("Invalid column family: {}", name).to_owned())); return Err(Error::new(format!("Invalid column family: {}", name).to_owned()));
} }
let mut err: *mut c_char = ptr::null_mut();
unsafe { unsafe {
ffi::rocksdb_drop_column_family(self.inner, *cf.unwrap(), &mut err); ffi_try!(ffi::rocksdb_drop_column_family(self.inner, *cf.unwrap()));
}
if !err.is_null() {
return Err(Error::new(error_message(err)));
} }
Ok(()) Ok(())
} }
@ -525,66 +484,42 @@ impl DB {
pub fn put_opt(&self, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { pub fn put_opt(&self, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_put(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t));
ffi::rocksdb_put(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
pub fn put_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { pub fn put_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_put_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t));
ffi::rocksdb_put_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
pub fn merge_opt(&self, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { pub fn merge_opt(&self, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_merge(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t));
ffi::rocksdb_merge(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, value.as_ptr() as *const c_char, value.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
fn merge_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { fn merge_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], value: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_merge_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const i8, key.len() as size_t, value.as_ptr() as *const i8, value.len() as size_t));
ffi::rocksdb_merge_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const i8, key.len() as size_t, value.as_ptr() as *const i8, value.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
fn delete_opt(&self, key: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { fn delete_opt(&self, key: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_delete(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t));
ffi::rocksdb_delete(self.inner, writeopts.inner, key.as_ptr() as *const c_char, key.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
fn delete_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], writeopts: &WriteOptions) -> Result<(), Error> { fn delete_cf_opt(&self, cf: *mut ffi::rocksdb_column_family_handle_t, key: &[u8], writeopts: &WriteOptions) -> Result<(), Error> {
unsafe { unsafe {
let mut err: *mut c_char = ptr::null_mut(); ffi_try!(ffi::rocksdb_delete_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t));
ffi::rocksdb_delete_cf(self.inner, writeopts.inner, cf, key.as_ptr() as *const c_char, key.len() as size_t, &mut err);
if !err.is_null() {
return Err(Error::new(error_message(err)));
}
Ok(()) Ok(())
} }
} }
@ -774,7 +709,7 @@ impl DBVector {
} }
pub fn to_utf8(&self) -> Option<&str> { pub fn to_utf8(&self) -> Option<&str> {
from_utf8(self.deref()).ok() str::from_utf8(self.deref()).ok()
} }
} }
@ -798,9 +733,9 @@ fn external() {
#[test] #[test]
fn errors_do_stuff() { fn errors_do_stuff() {
let path = "_rust_rocksdb_error"; let path = "_rust_rocksdb_error";
let db = DB::open_default(path).unwrap(); let _db = DB::open_default(path).unwrap();
let opts = Options::default(); let opts = Options::default();
// The DB will still be open when we try to destroy and the lock should fail // The DB will still be open when we try to destroy it and the lock should fail.
match DB::destroy(&opts, path) { match DB::destroy(&opts, path) {
Err(s) => { Err(s) => {
assert!(s == Error::new("IO error: lock _rust_rocksdb_error/LOCK: No locks available".to_owned())) assert!(s == Error::new("IO error: lock _rust_rocksdb_error/LOCK: No locks available".to_owned()))
@ -857,7 +792,7 @@ fn iterator_test() {
assert!(p.is_ok()); assert!(p.is_ok());
let iter = db.iterator(IteratorMode::Start); let iter = db.iterator(IteratorMode::Start);
for (k, v) in iter { for (k, v) in iter {
println!("Hello {}: {}", from_utf8(&*k).unwrap(), from_utf8(&*v).unwrap()); println!("Hello {}: {}", str::from_utf8(&*k).unwrap(), str::from_utf8(&*v).unwrap());
} }
} }
let opts = Options::default(); let opts = Options::default();

@ -1,3 +1,18 @@
// Copyright 2014 Tyler Neely
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
extern crate rocksdb; extern crate rocksdb;
mod test_iterator; mod test_iterator;

@ -12,6 +12,7 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
// //
use rocksdb::{DB, MergeOperands, Options, Writable}; use rocksdb::{DB, MergeOperands, Options, Writable};
#[test] #[test]

@ -1,3 +1,18 @@
// Copyright 2014 Tyler Neely
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
use rocksdb::{DB, Direction, IteratorMode, Options, Writable}; use rocksdb::{DB, Direction, IteratorMode, Options, Writable};
fn cba(input: &Box<[u8]>) -> Box<[u8]> { fn cba(input: &Box<[u8]>) -> Box<[u8]> {

@ -1,3 +1,18 @@
// Copyright 2014 Tyler Neely
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
use rocksdb::{DB, Options, Writable}; use rocksdb::{DB, Options, Writable};
use std::thread; use std::thread;
use std::sync::Arc; use std::sync::Arc;

@ -1,5 +1,19 @@
use rocksdb::{DB, Options}; // Copyright 2014 Tyler Neely
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
use rocksdb::{DB, Options};
#[test] #[test]
fn test_set_num_levels() { fn test_set_num_levels() {

Loading…
Cancel
Save