1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use std::fmt::{self, Debug, Formatter};
use std::sync::Arc;
use engine_traits::{self, IterOptions, Iterable, Peekable, ReadOptions, Result, Snapshot};
use rocksdb::rocksdb_options::UnsafeSnap;
use rocksdb::{DBIterator, DB};
use crate::db_vector::RocksDBVector;
use crate::options::RocksReadOptions;
use crate::util::get_cf_handle;
use crate::RocksEngineIterator;
pub struct RocksSnapshot {
db: Arc<DB>,
snap: UnsafeSnap,
}
unsafe impl Send for RocksSnapshot {}
unsafe impl Sync for RocksSnapshot {}
impl RocksSnapshot {
pub fn new(db: Arc<DB>) -> Self {
unsafe {
RocksSnapshot {
snap: db.unsafe_snap(),
db,
}
}
}
}
impl Snapshot for RocksSnapshot {
fn cf_names(&self) -> Vec<&str> {
self.db.cf_names()
}
}
impl Debug for RocksSnapshot {
fn fmt(&self, fmt: &mut Formatter<'_>) -> fmt::Result {
write!(fmt, "Engine Snapshot Impl")
}
}
impl Drop for RocksSnapshot {
fn drop(&mut self) {
unsafe {
self.db.release_snap(&self.snap);
}
}
}
impl Iterable for RocksSnapshot {
type Iterator = RocksEngineIterator;
fn iterator_opt(&self, opts: IterOptions) -> Result<Self::Iterator> {
let opt: RocksReadOptions = opts.into();
let mut opt = opt.into_raw();
unsafe {
opt.set_snapshot(&self.snap);
}
Ok(RocksEngineIterator::from_raw(DBIterator::new(
self.db.clone(),
opt,
)))
}
fn iterator_cf_opt(&self, cf: &str, opts: IterOptions) -> Result<Self::Iterator> {
let opt: RocksReadOptions = opts.into();
let mut opt = opt.into_raw();
unsafe {
opt.set_snapshot(&self.snap);
}
let handle = get_cf_handle(self.db.as_ref(), cf)?;
Ok(RocksEngineIterator::from_raw(DBIterator::new_cf(
self.db.clone(),
handle,
opt,
)))
}
}
impl Peekable for RocksSnapshot {
type DBVector = RocksDBVector;
fn get_value_opt(&self, opts: &ReadOptions, key: &[u8]) -> Result<Option<RocksDBVector>> {
let opt: RocksReadOptions = opts.into();
let mut opt = opt.into_raw();
unsafe {
opt.set_snapshot(&self.snap);
}
let v = self.db.get_opt(key, &opt)?;
Ok(v.map(RocksDBVector::from_raw))
}
fn get_value_cf_opt(
&self,
opts: &ReadOptions,
cf: &str,
key: &[u8],
) -> Result<Option<RocksDBVector>> {
let opt: RocksReadOptions = opts.into();
let mut opt = opt.into_raw();
unsafe {
opt.set_snapshot(&self.snap);
}
let handle = get_cf_handle(self.db.as_ref(), cf)?;
let v = self.db.get_cf_opt(handle, key, &opt)?;
Ok(v.map(RocksDBVector::from_raw))
}
}