DeleteRange read path end-to-end tests
Summary: Closes https://github.com/facebook/rocksdb/pull/1592 Differential Revision: D4246260 Pulled By: ajkr fbshipit-source-id: ce03fa2
This commit is contained in:
parent
2f4fc539c6
commit
b821984d31
@ -430,6 +430,179 @@ TEST_F(DBRangeDelTest, ObsoleteTombstoneCleanup) {
|
|||||||
}
|
}
|
||||||
#endif // ROCKSDB_LITE
|
#endif // ROCKSDB_LITE
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, GetCoveredKeyFromMutableMemtable) {
|
||||||
|
db_->Put(WriteOptions(), "key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
|
||||||
|
ReadOptions read_opts;
|
||||||
|
std::string value;
|
||||||
|
ASSERT_TRUE(db_->Get(read_opts, "key", &value).IsNotFound());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, GetCoveredKeyFromImmutableMemtable) {
|
||||||
|
Options opts = CurrentOptions();
|
||||||
|
opts.max_write_buffer_number = 3;
|
||||||
|
opts.min_write_buffer_number_to_merge = 2;
|
||||||
|
// SpecialSkipListFactory lets us specify maximum number of elements the
|
||||||
|
// memtable can hold. It switches the active memtable to immutable (flush is
|
||||||
|
// prevented by the above options) upon inserting an element that would
|
||||||
|
// overflow the memtable.
|
||||||
|
opts.memtable_factory.reset(new SpecialSkipListFactory(1));
|
||||||
|
Reopen(opts);
|
||||||
|
|
||||||
|
db_->Put(WriteOptions(), "key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
db_->Put(WriteOptions(), "blah", "val");
|
||||||
|
|
||||||
|
ReadOptions read_opts;
|
||||||
|
std::string value;
|
||||||
|
ASSERT_TRUE(db_->Get(read_opts, "key", &value).IsNotFound());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, GetCoveredKeyFromSst) {
|
||||||
|
db_->Put(WriteOptions(), "key", "val");
|
||||||
|
// snapshot prevents key from being deleted during flush
|
||||||
|
const Snapshot* snapshot = db_->GetSnapshot();
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
ASSERT_OK(db_->Flush(FlushOptions()));
|
||||||
|
|
||||||
|
ReadOptions read_opts;
|
||||||
|
std::string value;
|
||||||
|
ASSERT_TRUE(db_->Get(read_opts, "key", &value).IsNotFound());
|
||||||
|
db_->ReleaseSnapshot(snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, GetIgnoresRangeDeletions) {
|
||||||
|
Options opts = CurrentOptions();
|
||||||
|
opts.max_write_buffer_number = 4;
|
||||||
|
opts.min_write_buffer_number_to_merge = 3;
|
||||||
|
opts.memtable_factory.reset(new SpecialSkipListFactory(1));
|
||||||
|
Reopen(opts);
|
||||||
|
|
||||||
|
db_->Put(WriteOptions(), "sst_key", "val");
|
||||||
|
// snapshot prevents key from being deleted during flush
|
||||||
|
const Snapshot* snapshot = db_->GetSnapshot();
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
ASSERT_OK(db_->Flush(FlushOptions()));
|
||||||
|
db_->Put(WriteOptions(), "imm_key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
db_->Put(WriteOptions(), "mem_key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
|
||||||
|
ReadOptions read_opts;
|
||||||
|
read_opts.ignore_range_deletions = true;
|
||||||
|
for (std::string key : {"sst_key", "imm_key", "mem_key"}) {
|
||||||
|
std::string value;
|
||||||
|
ASSERT_OK(db_->Get(read_opts, key, &value));
|
||||||
|
}
|
||||||
|
db_->ReleaseSnapshot(snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, IteratorRemovesCoveredKeys) {
|
||||||
|
const int kNum = 200, kRangeBegin = 50, kRangeEnd = 150, kNumPerFile = 25;
|
||||||
|
Options opts = CurrentOptions();
|
||||||
|
opts.comparator = test::Uint64Comparator();
|
||||||
|
opts.memtable_factory.reset(new SpecialSkipListFactory(kNumPerFile));
|
||||||
|
Reopen(opts);
|
||||||
|
|
||||||
|
// Write half of the keys before the tombstone and half after the tombstone.
|
||||||
|
// Only covered keys (i.e., within the range and older than the tombstone)
|
||||||
|
// should be deleted.
|
||||||
|
for (int i = 0; i < kNum; ++i) {
|
||||||
|
if (i == kNum / 2) {
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(),
|
||||||
|
GetNumericStr(kRangeBegin), GetNumericStr(kRangeEnd));
|
||||||
|
}
|
||||||
|
db_->Put(WriteOptions(), GetNumericStr(i), "val");
|
||||||
|
}
|
||||||
|
ReadOptions read_opts;
|
||||||
|
auto* iter = db_->NewIterator(read_opts);
|
||||||
|
|
||||||
|
int expected = 0;
|
||||||
|
for (iter->SeekToFirst(); iter->Valid(); iter->Next()) {
|
||||||
|
ASSERT_EQ(GetNumericStr(expected), iter->key());
|
||||||
|
if (expected == kRangeBegin - 1) {
|
||||||
|
expected = kNum / 2;
|
||||||
|
} else {
|
||||||
|
++expected;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ASSERT_EQ(kNum, expected);
|
||||||
|
delete iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, IteratorOverUserSnapshot) {
|
||||||
|
const int kNum = 200, kRangeBegin = 50, kRangeEnd = 150, kNumPerFile = 25;
|
||||||
|
Options opts = CurrentOptions();
|
||||||
|
opts.comparator = test::Uint64Comparator();
|
||||||
|
opts.memtable_factory.reset(new SpecialSkipListFactory(kNumPerFile));
|
||||||
|
Reopen(opts);
|
||||||
|
|
||||||
|
const Snapshot* snapshot = nullptr;
|
||||||
|
// Put a snapshot before the range tombstone, verify an iterator using that
|
||||||
|
// snapshot sees all inserted keys.
|
||||||
|
for (int i = 0; i < kNum; ++i) {
|
||||||
|
if (i == kNum / 2) {
|
||||||
|
snapshot = db_->GetSnapshot();
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(),
|
||||||
|
GetNumericStr(kRangeBegin), GetNumericStr(kRangeEnd));
|
||||||
|
}
|
||||||
|
db_->Put(WriteOptions(), GetNumericStr(i), "val");
|
||||||
|
}
|
||||||
|
ReadOptions read_opts;
|
||||||
|
read_opts.snapshot = snapshot;
|
||||||
|
auto* iter = db_->NewIterator(read_opts);
|
||||||
|
|
||||||
|
int expected = 0;
|
||||||
|
for (iter->SeekToFirst(); iter->Valid(); iter->Next()) {
|
||||||
|
ASSERT_EQ(GetNumericStr(expected), iter->key());
|
||||||
|
++expected;
|
||||||
|
}
|
||||||
|
ASSERT_EQ(kNum / 2, expected);
|
||||||
|
delete iter;
|
||||||
|
db_->ReleaseSnapshot(snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DBRangeDelTest, IteratorIgnoresRangeDeletions) {
|
||||||
|
Options opts = CurrentOptions();
|
||||||
|
opts.max_write_buffer_number = 4;
|
||||||
|
opts.min_write_buffer_number_to_merge = 3;
|
||||||
|
opts.memtable_factory.reset(new SpecialSkipListFactory(1));
|
||||||
|
Reopen(opts);
|
||||||
|
|
||||||
|
db_->Put(WriteOptions(), "sst_key", "val");
|
||||||
|
// snapshot prevents key from being deleted during flush
|
||||||
|
const Snapshot* snapshot = db_->GetSnapshot();
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
ASSERT_OK(db_->Flush(FlushOptions()));
|
||||||
|
db_->Put(WriteOptions(), "imm_key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
db_->Put(WriteOptions(), "mem_key", "val");
|
||||||
|
ASSERT_OK(
|
||||||
|
db_->DeleteRange(WriteOptions(), db_->DefaultColumnFamily(), "a", "z"));
|
||||||
|
|
||||||
|
ReadOptions read_opts;
|
||||||
|
read_opts.ignore_range_deletions = true;
|
||||||
|
auto* iter = db_->NewIterator(read_opts);
|
||||||
|
int i = 0;
|
||||||
|
std::string expected[] = {"imm_key", "mem_key", "sst_key"};
|
||||||
|
for (iter->SeekToFirst(); iter->Valid(); iter->Next(), ++i) {
|
||||||
|
std::string key;
|
||||||
|
ASSERT_EQ(expected[i], iter->key());
|
||||||
|
}
|
||||||
|
ASSERT_EQ(3, i);
|
||||||
|
delete iter;
|
||||||
|
db_->ReleaseSnapshot(snapshot);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace rocksdb
|
} // namespace rocksdb
|
||||||
|
|
||||||
int main(int argc, char** argv) {
|
int main(int argc, char** argv) {
|
||||||
|
Loading…
Reference in New Issue
Block a user