Add tests for table mappers; Bugfix on-disk mapper (#3408)
add tests to `kvstore::mapper::{disk, memory}` modules fix bug in disk mapper uncovered by tests use `tempdir` crate for unit test-directories
This commit is contained in:
parent
4247fa946e
commit
9b0bf5ad66
|
@ -2242,6 +2242,7 @@ dependencies = [
|
||||||
"rand 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
"rand 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"serde 1.0.89 (registry+https://github.com/rust-lang/crates.io-index)",
|
"serde 1.0.89 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"serde_derive 1.0.89 (registry+https://github.com/rust-lang/crates.io-index)",
|
"serde_derive 1.0.89 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"tempfile 3.0.7 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
|
@ -18,3 +18,5 @@ rand = "0.6.5"
|
||||||
serde = "1.0.89"
|
serde = "1.0.89"
|
||||||
serde_derive = "1.0.88"
|
serde_derive = "1.0.88"
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
tempfile = "3.0.7"
|
||||||
|
|
|
@ -369,7 +369,6 @@ fn is_lvl0_full(tables: &[BTreeMap<Key, SSTable>], config: &Config) -> bool {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(test, debug_assertions))]
|
|
||||||
pub mod test {
|
pub mod test {
|
||||||
pub mod gen {
|
pub mod gen {
|
||||||
use crate::Key;
|
use crate::Key;
|
||||||
|
|
|
@ -50,7 +50,7 @@ impl Disk {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize)]
|
#[derive(Clone, PartialEq, Debug, Serialize, Deserialize)]
|
||||||
pub struct PathInfo {
|
pub struct PathInfo {
|
||||||
pub data: PathBuf,
|
pub data: PathBuf,
|
||||||
pub index: PathBuf,
|
pub index: PathBuf,
|
||||||
|
@ -213,3 +213,124 @@ fn next_id(kind: Kind) -> Id {
|
||||||
kind,
|
kind,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::mapper::Kind;
|
||||||
|
use crate::sstable::{Key, Value};
|
||||||
|
use crate::test::gen;
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::thread;
|
||||||
|
use tempfile::tempdir;
|
||||||
|
|
||||||
|
const DATA_SIZE: usize = 128;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_table_management() {
|
||||||
|
let tempdir = tempdir().unwrap();
|
||||||
|
let mapper = Arc::new(Disk::single(tempdir.path()));
|
||||||
|
let records: BTreeMap<_, _> = gen_records().take(1024).collect();
|
||||||
|
|
||||||
|
let mut threads = vec![];
|
||||||
|
let mut number_of_tables = 4;
|
||||||
|
|
||||||
|
for kind in [Kind::Active, Kind::Garbage, Kind::Compaction].iter() {
|
||||||
|
let records = records.clone();
|
||||||
|
let mapper = Arc::clone(&mapper);
|
||||||
|
|
||||||
|
let child = thread::spawn(move || {
|
||||||
|
for _ in 0..number_of_tables {
|
||||||
|
mapper
|
||||||
|
.make_table(*kind, &mut |mut data_writer, mut index_writer| {
|
||||||
|
SSTable::create(
|
||||||
|
&mut records.iter(),
|
||||||
|
0,
|
||||||
|
&mut data_writer,
|
||||||
|
&mut index_writer,
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
number_of_tables *= 2;
|
||||||
|
threads.push(child);
|
||||||
|
}
|
||||||
|
|
||||||
|
threads.into_iter().for_each(|child| child.join().unwrap());
|
||||||
|
let count_kind = |kind, mapper: &Disk| {
|
||||||
|
mapper
|
||||||
|
.mappings
|
||||||
|
.read()
|
||||||
|
.unwrap()
|
||||||
|
.keys()
|
||||||
|
.filter(|id| id.kind == kind)
|
||||||
|
.count()
|
||||||
|
};
|
||||||
|
assert_eq!(count_kind(Kind::Active, &mapper), 4);
|
||||||
|
assert_eq!(count_kind(Kind::Garbage, &mapper), 8);
|
||||||
|
assert_eq!(count_kind(Kind::Compaction, &mapper), 16);
|
||||||
|
|
||||||
|
mapper.empty_trash().unwrap();
|
||||||
|
assert_eq!(count_kind(Kind::Garbage, &mapper), 0);
|
||||||
|
|
||||||
|
mapper.rotate_tables().unwrap();
|
||||||
|
assert_eq!(count_kind(Kind::Active, &mapper), 16);
|
||||||
|
assert_eq!(count_kind(Kind::Garbage, &mapper), 4);
|
||||||
|
assert_eq!(count_kind(Kind::Compaction, &mapper), 0);
|
||||||
|
|
||||||
|
let active_set = mapper.active_set().unwrap();
|
||||||
|
assert_eq!(active_set.len(), 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_state() {
|
||||||
|
let tempdir = tempdir().unwrap();
|
||||||
|
let dirs_1: Vec<_> = (0..4).map(|i| tempdir.path().join(i.to_string())).collect();
|
||||||
|
let dirs_2: Vec<_> = (4..8).map(|i| tempdir.path().join(i.to_string())).collect();
|
||||||
|
|
||||||
|
let mapper_1 = Arc::new(Disk::new(&dirs_1));
|
||||||
|
let records: BTreeMap<_, _> = gen_records().take(1024).collect();
|
||||||
|
|
||||||
|
for (i, &kind) in [Kind::Active, Kind::Compaction, Kind::Garbage]
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
{
|
||||||
|
for _ in 0..(i * 3) {
|
||||||
|
mapper_1
|
||||||
|
.make_table(kind, &mut |mut data_writer, mut index_writer| {
|
||||||
|
SSTable::create(
|
||||||
|
&mut records.iter(),
|
||||||
|
0,
|
||||||
|
&mut data_writer,
|
||||||
|
&mut index_writer,
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let state_path = tempdir.path().join("state");
|
||||||
|
mapper_1.serialize_state_to(&state_path).unwrap();
|
||||||
|
assert!(state_path.exists());
|
||||||
|
|
||||||
|
let mapper_2 = Arc::new(Disk::new(&dirs_2));
|
||||||
|
mapper_2.load_state_from(&state_path).unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
&*mapper_1.mappings.read().unwrap(),
|
||||||
|
&*mapper_2.mappings.read().unwrap()
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
&*mapper_1.storage_dirs.read().unwrap(),
|
||||||
|
&*mapper_2.storage_dirs.read().unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn gen_records() -> impl Iterator<Item = (Key, Value)> {
|
||||||
|
gen::pairs(DATA_SIZE).map(|(key, data)| (key, Value::new(0, Some(data))))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
@ -62,18 +62,16 @@ impl Mapper for Memory {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn rotate_tables(&self) -> Result<()> {
|
fn rotate_tables(&self) -> Result<()> {
|
||||||
use std::mem::swap;
|
let (mut active, mut compaction, mut garbage) = (
|
||||||
|
|
||||||
let (mut active, mut compact, mut garbage) = (
|
|
||||||
self.tables.write().expect(BACKING_ERR_MSG),
|
self.tables.write().expect(BACKING_ERR_MSG),
|
||||||
self.compaction.write().expect(BACKING_ERR_MSG),
|
self.compaction.write().expect(BACKING_ERR_MSG),
|
||||||
self.garbage.write().expect(BACKING_ERR_MSG),
|
self.garbage.write().expect(BACKING_ERR_MSG),
|
||||||
);
|
);
|
||||||
|
|
||||||
// compacted tables => active set
|
|
||||||
swap(&mut active, &mut compact);
|
|
||||||
// old active set => garbage
|
// old active set => garbage
|
||||||
garbage.extend(compact.drain());
|
garbage.extend(active.drain());
|
||||||
|
// compacted tables => new active set
|
||||||
|
active.extend(compaction.drain());
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -142,3 +140,87 @@ fn get_table(id: Id, map: &TableMap) -> Result<SSTable> {
|
||||||
fn next_id() -> Id {
|
fn next_id() -> Id {
|
||||||
rand::thread_rng().gen()
|
rand::thread_rng().gen()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::mapper::Kind;
|
||||||
|
use crate::sstable::{Key, Value};
|
||||||
|
use crate::test::gen;
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::thread;
|
||||||
|
|
||||||
|
const DATA_SIZE: usize = 128;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_table_management() {
|
||||||
|
let mapper = Arc::new(Memory::new());
|
||||||
|
let records: BTreeMap<_, _> = gen_records().take(1024).collect();
|
||||||
|
|
||||||
|
let mut threads = vec![];
|
||||||
|
let mut number_of_tables = 4;
|
||||||
|
|
||||||
|
for kind in [Kind::Active, Kind::Garbage, Kind::Compaction].iter() {
|
||||||
|
let records = records.clone();
|
||||||
|
let mapper = Arc::clone(&mapper);
|
||||||
|
|
||||||
|
let child = thread::spawn(move || {
|
||||||
|
for _ in 0..number_of_tables {
|
||||||
|
mapper
|
||||||
|
.make_table(*kind, &mut |mut data_writer, mut index_writer| {
|
||||||
|
SSTable::create(
|
||||||
|
&mut records.iter(),
|
||||||
|
0,
|
||||||
|
&mut data_writer,
|
||||||
|
&mut index_writer,
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
number_of_tables *= 2;
|
||||||
|
threads.push(child);
|
||||||
|
}
|
||||||
|
|
||||||
|
threads.into_iter().for_each(|child| child.join().unwrap());
|
||||||
|
assert_eq!(mapper.tables.read().unwrap().len(), 4);
|
||||||
|
assert_eq!(mapper.garbage.read().unwrap().len(), 8);
|
||||||
|
assert_eq!(mapper.compaction.read().unwrap().len(), 16);
|
||||||
|
|
||||||
|
mapper.empty_trash().unwrap();
|
||||||
|
assert_eq!(mapper.garbage.read().unwrap().len(), 0);
|
||||||
|
|
||||||
|
mapper.rotate_tables().unwrap();
|
||||||
|
assert_eq!(mapper.tables.read().unwrap().len(), 16);
|
||||||
|
assert_eq!(mapper.garbage.read().unwrap().len(), 4);
|
||||||
|
assert!(mapper.compaction.read().unwrap().is_empty());
|
||||||
|
|
||||||
|
let active_set = mapper.active_set().unwrap();
|
||||||
|
assert_eq!(active_set.len(), 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_no_state() {
|
||||||
|
let tempdir = tempfile::tempdir().unwrap();
|
||||||
|
let mapper = Arc::new(Memory::new());
|
||||||
|
let records: BTreeMap<_, _> = gen_records().take(1024).collect();
|
||||||
|
|
||||||
|
mapper
|
||||||
|
.make_table(Kind::Active, &mut |mut data_writer, mut index_writer| {
|
||||||
|
SSTable::create(&mut records.iter(), 0, &mut data_writer, &mut index_writer);
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let state_path = tempdir.path().join("state");
|
||||||
|
mapper.serialize_state_to(&state_path).unwrap();
|
||||||
|
mapper.load_state_from(&state_path).unwrap();
|
||||||
|
assert!(!state_path.exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
fn gen_records() -> impl Iterator<Item = (Key, Value)> {
|
||||||
|
gen::pairs(DATA_SIZE).map(|(key, data)| (key, Value::new(0, Some(data))))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue