/* 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. */ #![feature(test)] extern crate rocksdb; extern crate test; use rocksdb::{Options, RocksDB, MergeOperands, new_bloom_filter, Writable, DBIterator, SubDBIterator }; use rocksdb::RocksDBCompactionStyle::RocksDBUniversalCompaction; fn iterator_test() { let path = "_rust_rocksdb_iteratortest"; { let mut db = RocksDB::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 view1 = db.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 mut view2 = db.iterator(); let p = db.put(b"k4", b"v4444"); assert!(p.is_ok()); let mut view3 = db.iterator(); println!("See the output of the second iter"); for (k,v) in view2.from_start() { println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), std::str::from_utf8(v).unwrap()); } println!("See the output of the third iter"); for (k,v) in view3.from_start() { println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), std::str::from_utf8(v).unwrap()); } println!("now the 3rd iter from k2 fwd"); for (k,v) in view3.from(b"k2", rocksdb::Direction::forward) { println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), std::str::from_utf8(v).unwrap()); } println!("now the 3rd iter from k2 and back"); for (k,v) in view3.from(b"k2", rocksdb::Direction::reverse) { println!("Hello {}: {}", std::str::from_utf8(k).unwrap(), std::str::from_utf8(v).unwrap()); } } let opts = Options::new(); assert!(RocksDB::destroy(&opts, path).is_ok()); } #[cfg(not(feature = "valgrind"))] fn main() { iterator_test(); let path = "/tmp/rust-rocksdb"; let mut db = RocksDB::open_default(path).unwrap(); assert!(db.put(b"my key", b"my value").is_ok()); db.get(b"my key").map( |value| { match value.to_utf8() { Some(v) => println!("retrieved utf8 value: {}", v), None => println!("did not read valid utf-8 out of the db"), } }) .on_absent( || { println!("value not found") }) .on_error( |e| { println!("error retrieving value: {}", e) }); assert!(db.delete(b"my key").is_ok()); custom_merge(); } fn concat_merge(new_key: &[u8], existing_val: Option<&[u8]>, mut operands: &mut MergeOperands) -> Vec { let mut result: Vec = Vec::with_capacity(operands.size_hint().0); match existing_val { Some(v) => result.extend(v), None => (), } for op in operands { result.extend(op); } 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 mut db = RocksDB::open(&opts, path).unwrap(); 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) => println!("retrieved utf8 value: {}", v), None => println!("did not read valid utf-8 out of the db"), } }) .on_absent( || { println!("value not found") }) .on_error( |e| { println!("error retrieving value: {}", e) }); } RocksDB::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 = RocksDB::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"), } }) .on_absent( || { panic!("value not found") }) .on_error( |e| { panic!("error retrieving value: {}", e) }); db.delete(b"k1"); } } #[cfg(test)] mod tests { use test::Bencher; use std::thread::sleep_ms; use rocksdb::{BlockBasedOptions, Options, RocksDB, MergeOperands, new_bloom_filter, Writable }; use rocksdb::RocksDBCompactionStyle::RocksDBUniversalCompaction; fn tuned_for_somebody_elses_disk(path: &str, opts: & mut Options, blockopts: &mut BlockBasedOptions) -> RocksDB { 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(RocksDBUniversalCompaction); 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); RocksDB::open(&opts, path).unwrap() } #[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; }); } RocksDB::destroy(&opts, path).is_ok(); } }