// // 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; use rocksdb::{DB, MergeOperands, Options, Writable}; // fn snapshot_test() { // let path = "_rust_rocksdb_iteratortest"; // { // let mut db = DB::open_default(path).unwrap(); // let p = db.put(b"k1", b"v1111"); // assert!(p.is_ok()); // let p = db.put(b"k2", b"v2222"); // assert!(p.is_ok()); // let p = db.put(b"k3", b"v3333"); // assert!(p.is_ok()); // let mut snap = db.snapshot(); // let mut view1 = snap.iterator(); // println!("See the output of the first iter"); // for (k,v) in view1.from_start() { // println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), // std::str::from_utf8(v).unwrap()); // }; // for (k,v) in view1.from_start() { // println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), // std::str::from_utf8(v).unwrap()); // }; // for (k,v) in view1.from_end() { // println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), // std::str::from_utf8(v).unwrap()); // }; // } // let opts = Options::new(); // assert!(DB::destroy(&opts, path).is_ok()); // } #[cfg(not(feature = "valgrind"))] fn main() { let path = "/tmp/rust-rocksdb"; let db = DB::open_default(path).unwrap(); assert!(db.put(b"my key", b"my value").is_ok()); match db.get(b"my key") { Ok(Some(value)) => { match value.to_utf8() { Some(v) => println!("retrieved utf8 value: {}", v), None => println!("did not read valid utf-8 out of the db"), } } Err(e) => println!("error retrieving value: {}", e), _ => panic!("value not present!"), } assert!(db.delete(b"my key").is_ok()); custom_merge(); } fn concat_merge(_: &[u8], existing_val: Option<&[u8]>, operands: &mut MergeOperands) -> Vec { let mut result: Vec = Vec::with_capacity(operands.size_hint().0); match existing_val { Some(v) => for e in v { result.push(*e) }, None => (), } for op in operands { for e in op { result.push(*e); } } result } fn custom_merge() { let path = "_rust_rocksdb_mergetest"; let mut opts = Options::new(); opts.create_if_missing(true); opts.add_merge_operator("test operator", concat_merge); { let db = DB::open(&opts, path).unwrap(); db.put(b"k1", b"a").unwrap(); db.merge(b"k1", b"b").unwrap(); db.merge(b"k1", b"c").unwrap(); db.merge(b"k1", b"d").unwrap(); db.merge(b"k1", b"efg").unwrap(); db.merge(b"k1", b"h").unwrap(); match db.get(b"k1") { Ok(Some(value)) => { match value.to_utf8() { Some(v) => println!("retrieved utf8 value: {}", v), None => println!("did not read valid utf-8 out of the db"), } } Err(e) => println!("error retrieving value: {}", e), _ => panic!("value not present!"), } } DB::destroy(&opts, path).is_ok(); } #[cfg(feature = "valgrind")] fn main() { let path = "_rust_rocksdb_valgrind"; let mut opts = Options::new(); opts.create_if_missing(true); opts.add_merge_operator("test operator", concat_merge); let db = DB::open(&opts, path).unwrap(); loop { db.put(b"k1", b"a"); db.merge(b"k1", b"b"); db.merge(b"k1", b"c"); db.merge(b"k1", b"d"); db.merge(b"k1", b"efg"); db.merge(b"k1", b"h"); db.get(b"k1") .map(|value| { match value.to_utf8() { Some(v) => (), None => panic!("value corrupted"), } }) .or_else(|e| panic!("error retrieving value: {}", e)); db.delete(b"k1"); } } #[cfg(test)] mod tests { use std::thread::sleep_ms; use rocksdb::{BlockBasedOptions, DB, MergeOperands, Options, Writable, new_bloom_filter}; use rocksdb::DBCompactionStyle::DBUniversalCompaction; fn tuned_for_somebody_elses_disk(path: &str, opts: &mut Options, blockopts: &mut BlockBasedOptions) -> DB { opts.create_if_missing(true); opts.set_max_open_files(10000); opts.set_use_fsync(false); opts.set_bytes_per_sync(8388608); opts.set_disable_data_sync(false); opts.set_block_cache_size_mb(1024); opts.set_table_cache_num_shard_bits(6); opts.set_max_write_buffer_number(32); opts.set_write_buffer_size(536870912); opts.set_target_file_size_base(1073741824); opts.set_min_write_buffer_number_to_merge(4); opts.set_level_zero_stop_writes_trigger(2000); opts.set_level_zero_slowdown_writes_trigger(0); opts.set_compaction_style(DBUniversalCompaction); opts.set_max_background_compactions(4); opts.set_max_background_flushes(4); opts.set_filter_deletes(false); blockopts.set_block_size(524288); opts.set_block_based_table_factory(blockopts); opts.set_disable_auto_compactions(true); let filter = new_bloom_filter(10); // opts.set_filter(filter); DB::open(&opts, path).unwrap() } // TODO(tyler) unstable // #[bench] // fn a_writes(b: &mut Bencher) { // dirty hack due to parallel tests causing contention. // sleep_ms(1000); // let path = "_rust_rocksdb_optimizetest"; // let mut opts = Options::new(); // let mut blockopts = BlockBasedOptions::new(); // let mut db = tuned_for_somebody_elses_disk(path, &mut opts, &mut blockopts); // let mut i = 0 as u64; // b.iter(|| { // db.put(i.to_string().as_bytes(), b"v1111"); // i += 1; // }); // } // // #[bench] // fn b_reads(b: &mut Bencher) { // let path = "_rust_rocksdb_optimizetest"; // let mut opts = Options::new(); // let mut blockopts = BlockBasedOptions::new(); // { // let db = tuned_for_somebody_elses_disk(path, &mut opts, &mut blockopts); // let mut i = 0 as u64; // b.iter(|| { // db.get(i.to_string().as_bytes()).on_error( |e| { // println!("error: {}", e); // e // }); // i += 1; // }); // } // DB::destroy(&opts, path).is_ok(); // } // }