|
|
|
@ -43,13 +43,14 @@ class TtlTest { |
|
|
|
|
|
|
|
|
|
// Open database with TTL support when TTL not provided with db_ttl_ pointer
|
|
|
|
|
void OpenTtl() { |
|
|
|
|
assert(db_ttl_ == nullptr); // db should be closed before opening again
|
|
|
|
|
ASSERT_TRUE(db_ttl_ == |
|
|
|
|
nullptr); // db should be closed before opening again
|
|
|
|
|
ASSERT_OK(UtilityDB::OpenTtlDB(options_, dbname_, &db_ttl_)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Open database with TTL support when TTL provided with db_ttl_ pointer
|
|
|
|
|
void OpenTtl(int32_t ttl) { |
|
|
|
|
assert(db_ttl_ == nullptr); |
|
|
|
|
ASSERT_TRUE(db_ttl_ == nullptr); |
|
|
|
|
ASSERT_OK(UtilityDB::OpenTtlDB(options_, dbname_, &db_ttl_, ttl)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -63,7 +64,7 @@ class TtlTest { |
|
|
|
|
|
|
|
|
|
// Open database with TTL support in read_only mode
|
|
|
|
|
void OpenReadOnlyTtl(int32_t ttl) { |
|
|
|
|
assert(db_ttl_ == nullptr); |
|
|
|
|
ASSERT_TRUE(db_ttl_ == nullptr); |
|
|
|
|
ASSERT_OK(UtilityDB::OpenTtlDB(options_, dbname_, &db_ttl_, ttl, true)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -97,7 +98,7 @@ class TtlTest { |
|
|
|
|
|
|
|
|
|
// Makes a write-batch with key-vals from kvmap_ and 'Write''s it
|
|
|
|
|
void MakePutWriteBatch(const BatchOperation* batch_ops, int num_ops) { |
|
|
|
|
assert(num_ops <= (int)kvmap_.size()); |
|
|
|
|
ASSERT_LE(num_ops, (int)kvmap_.size()); |
|
|
|
|
static WriteOptions wopts; |
|
|
|
|
static FlushOptions flush_opts; |
|
|
|
|
WriteBatch batch; |
|
|
|
@ -111,7 +112,7 @@ class TtlTest { |
|
|
|
|
batch.Delete(kv_it_->first); |
|
|
|
|
break; |
|
|
|
|
default: |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
db_ttl_->Write(wopts, &batch); |
|
|
|
@ -119,26 +120,38 @@ class TtlTest { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Puts num_entries starting from start_pos_map from kvmap_ into the database
|
|
|
|
|
void PutValues(int start_pos_map, int num_entries, bool flush = true) { |
|
|
|
|
assert(db_ttl_); |
|
|
|
|
void PutValues(int start_pos_map, int num_entries, bool flush = true, |
|
|
|
|
ColumnFamilyHandle* cf = nullptr) { |
|
|
|
|
ASSERT_TRUE(db_ttl_); |
|
|
|
|
ASSERT_LE(start_pos_map + num_entries, (int)kvmap_.size()); |
|
|
|
|
static WriteOptions wopts; |
|
|
|
|
static FlushOptions flush_opts; |
|
|
|
|
kv_it_ = kvmap_.begin(); |
|
|
|
|
advance(kv_it_, start_pos_map); |
|
|
|
|
for (int i = 0; kv_it_ != kvmap_.end() && i < num_entries; i++, kv_it_++) { |
|
|
|
|
ASSERT_OK(db_ttl_->Put(wopts, kv_it_->first, kv_it_->second)); |
|
|
|
|
ASSERT_OK(cf == nullptr |
|
|
|
|
? db_ttl_->Put(wopts, kv_it_->first, kv_it_->second) |
|
|
|
|
: db_ttl_->Put(wopts, cf, kv_it_->first, kv_it_->second)); |
|
|
|
|
} |
|
|
|
|
// Put a mock kv at the end because CompactionFilter doesn't delete last key
|
|
|
|
|
ASSERT_OK(db_ttl_->Put(wopts, "keymock", "valuemock")); |
|
|
|
|
ASSERT_OK(cf == nullptr ? db_ttl_->Put(wopts, "keymock", "valuemock") |
|
|
|
|
: db_ttl_->Put(wopts, cf, "keymock", "valuemock")); |
|
|
|
|
if (flush) { |
|
|
|
|
if (cf == nullptr) { |
|
|
|
|
db_ttl_->Flush(flush_opts); |
|
|
|
|
} else { |
|
|
|
|
db_ttl_->Flush(flush_opts, cf); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Runs a manual compaction
|
|
|
|
|
void ManualCompact() { |
|
|
|
|
void ManualCompact(ColumnFamilyHandle* cf = nullptr) { |
|
|
|
|
if (cf == nullptr) { |
|
|
|
|
db_ttl_->CompactRange(nullptr, nullptr); |
|
|
|
|
} else { |
|
|
|
|
db_ttl_->CompactRange(cf, nullptr, nullptr); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// checks the whole kvmap_ to return correct values using KeyMayExist
|
|
|
|
@ -151,12 +164,12 @@ class TtlTest { |
|
|
|
|
if (ret == false || value_found == false) { |
|
|
|
|
fprintf(stderr, "KeyMayExist could not find key=%s in the database but" |
|
|
|
|
" should have\n", kv.first.c_str()); |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} else if (val.compare(kv.second) != 0) { |
|
|
|
|
fprintf(stderr, " value for key=%s present in database is %s but" |
|
|
|
|
" should be %s\n", kv.first.c_str(), val.c_str(), |
|
|
|
|
kv.second.c_str()); |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
@ -167,16 +180,18 @@ class TtlTest { |
|
|
|
|
// Also checks that value that we got is the same as inserted; and =kNewValue
|
|
|
|
|
// if test_compaction_change is true
|
|
|
|
|
void SleepCompactCheck(int slp_tim, int st_pos, int span, bool check = true, |
|
|
|
|
bool test_compaction_change = false) { |
|
|
|
|
assert(db_ttl_); |
|
|
|
|
bool test_compaction_change = false, |
|
|
|
|
ColumnFamilyHandle* cf = nullptr) { |
|
|
|
|
ASSERT_TRUE(db_ttl_); |
|
|
|
|
sleep(slp_tim); |
|
|
|
|
ManualCompact(); |
|
|
|
|
ManualCompact(cf); |
|
|
|
|
static ReadOptions ropts; |
|
|
|
|
kv_it_ = kvmap_.begin(); |
|
|
|
|
advance(kv_it_, st_pos); |
|
|
|
|
std::string v; |
|
|
|
|
for (int i = 0; kv_it_ != kvmap_.end() && i < span; i++, kv_it_++) { |
|
|
|
|
Status s = db_ttl_->Get(ropts, kv_it_->first, &v); |
|
|
|
|
Status s = (cf == nullptr) ? db_ttl_->Get(ropts, kv_it_->first, &v) |
|
|
|
|
: db_ttl_->Get(ropts, cf, kv_it_->first, &v); |
|
|
|
|
if (s.ok() != check) { |
|
|
|
|
fprintf(stderr, "key=%s ", kv_it_->first.c_str()); |
|
|
|
|
if (!s.ok()) { |
|
|
|
@ -184,18 +199,18 @@ class TtlTest { |
|
|
|
|
} else { |
|
|
|
|
fprintf(stderr, "is present in db but was expected to be absent\n"); |
|
|
|
|
} |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} else if (s.ok()) { |
|
|
|
|
if (test_compaction_change && v.compare(kNewValue_) != 0) { |
|
|
|
|
fprintf(stderr, " value for key=%s present in database is %s but " |
|
|
|
|
" should be %s\n", kv_it_->first.c_str(), v.c_str(), |
|
|
|
|
kNewValue_.c_str()); |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} else if (!test_compaction_change && v.compare(kv_it_->second) !=0) { |
|
|
|
|
fprintf(stderr, " value for key=%s present in database is %s but " |
|
|
|
|
" should be %s\n", kv_it_->first.c_str(), v.c_str(), |
|
|
|
|
kv_it_->second.c_str()); |
|
|
|
|
assert(false); |
|
|
|
|
ASSERT_TRUE(false); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
@ -203,7 +218,7 @@ class TtlTest { |
|
|
|
|
|
|
|
|
|
// Similar as SleepCompactCheck but uses TtlIterator to read from db
|
|
|
|
|
void SleepCompactCheckIter(int slp, int st_pos, int span, bool check=true) { |
|
|
|
|
assert(db_ttl_); |
|
|
|
|
ASSERT_TRUE(db_ttl_); |
|
|
|
|
sleep(slp); |
|
|
|
|
ManualCompact(); |
|
|
|
|
static ReadOptions ropts; |
|
|
|
@ -301,10 +316,10 @@ class TtlTest { |
|
|
|
|
|
|
|
|
|
// Choose carefully so that Put, Gets & Compaction complete in 1 second buffer
|
|
|
|
|
const int64_t kSampleSize_ = 100; |
|
|
|
|
|
|
|
|
|
private: |
|
|
|
|
std::string dbname_; |
|
|
|
|
StackableDB* db_ttl_; |
|
|
|
|
|
|
|
|
|
private: |
|
|
|
|
Options options_; |
|
|
|
|
KVMap kvmap_; |
|
|
|
|
KVMap::iterator kv_it_; |
|
|
|
@ -496,6 +511,54 @@ TEST(TtlTest, KeyMayExist) { |
|
|
|
|
CloseTtl(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
TEST(TtlTest, ColumnFamiliesTest) { |
|
|
|
|
DB* db; |
|
|
|
|
Options options; |
|
|
|
|
options.create_if_missing = true; |
|
|
|
|
|
|
|
|
|
DB::Open(options, dbname_, &db); |
|
|
|
|
ColumnFamilyHandle* handle; |
|
|
|
|
ASSERT_OK(db->CreateColumnFamily(ColumnFamilyOptions(options), |
|
|
|
|
"ttl_column_family", &handle)); |
|
|
|
|
|
|
|
|
|
delete handle; |
|
|
|
|
delete db; |
|
|
|
|
|
|
|
|
|
std::vector<ColumnFamilyDescriptor> column_families; |
|
|
|
|
column_families.push_back(ColumnFamilyDescriptor( |
|
|
|
|
kDefaultColumnFamilyName, ColumnFamilyOptions(options))); |
|
|
|
|
column_families.push_back(ColumnFamilyDescriptor( |
|
|
|
|
"ttl_column_family", ColumnFamilyOptions(options))); |
|
|
|
|
|
|
|
|
|
std::vector<ColumnFamilyHandle*> handles; |
|
|
|
|
|
|
|
|
|
ASSERT_OK(UtilityDB::OpenTtlDB(DBOptions(options), dbname_, column_families, |
|
|
|
|
&handles, &db_ttl_, {2, 4}, false)); |
|
|
|
|
ASSERT_EQ(handles.size(), 2U); |
|
|
|
|
|
|
|
|
|
MakeKVMap(kSampleSize_); |
|
|
|
|
PutValues(0, kSampleSize_, false, handles[0]); |
|
|
|
|
PutValues(0, kSampleSize_, false, handles[1]); |
|
|
|
|
|
|
|
|
|
// everything should be there after 1 second
|
|
|
|
|
SleepCompactCheck(1, 0, kSampleSize_, true, false, handles[0]); |
|
|
|
|
SleepCompactCheck(0, 0, kSampleSize_, true, false, handles[1]); |
|
|
|
|
|
|
|
|
|
// only column family 1 should be alive after 3 seconds
|
|
|
|
|
SleepCompactCheck(2, 0, kSampleSize_, false, false, handles[0]); |
|
|
|
|
SleepCompactCheck(0, 0, kSampleSize_, true, false, handles[1]); |
|
|
|
|
|
|
|
|
|
// nothing should be there after 5 seconds
|
|
|
|
|
SleepCompactCheck(2, 0, kSampleSize_, false, false, handles[0]); |
|
|
|
|
SleepCompactCheck(0, 0, kSampleSize_, false, false, handles[1]); |
|
|
|
|
|
|
|
|
|
for (auto h : handles) { |
|
|
|
|
delete h; |
|
|
|
|
} |
|
|
|
|
delete db_ttl_; |
|
|
|
|
db_ttl_ = nullptr; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
} // namespace rocksdb
|
|
|
|
|
|
|
|
|
|
// A black-box test for the ttl wrapper around rocksdb
|
|
|
|
|