2017-03-29 18:37:58 +08:00
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
#include <memory>
|
2017-04-14 23:32:59 +08:00
|
|
|
#include <thread>
|
2017-03-29 18:37:58 +08:00
|
|
|
|
2017-06-21 17:29:13 +08:00
|
|
|
#include <glog/logging.h>
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2017-03-29 18:37:58 +08:00
|
|
|
#include "data_structures/concurrent/skiplist.hpp"
|
|
|
|
#include "mvcc/record.hpp"
|
|
|
|
#include "mvcc/version_list.hpp"
|
|
|
|
#include "storage/garbage_collector.hpp"
|
|
|
|
#include "storage/vertex.hpp"
|
|
|
|
#include "transactions/engine.hpp"
|
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
#include "mvcc_gc_common.hpp"
|
2017-03-29 18:37:58 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
class MvccGcTest : public ::testing::Test {
|
|
|
|
protected:
|
2017-03-29 18:37:58 +08:00
|
|
|
tx::Engine engine;
|
2017-06-21 17:29:13 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
private:
|
|
|
|
tx::Transaction *t0 = engine.Begin();
|
2017-06-21 17:29:13 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
protected:
|
|
|
|
std::atomic<int> record_destruction_count{0};
|
|
|
|
mvcc::VersionList<DestrCountRec> version_list{*t0, record_destruction_count};
|
|
|
|
std::vector<tx::Transaction *> transactions{t0};
|
|
|
|
|
|
|
|
void SetUp() override { t0->Commit(); }
|
|
|
|
|
|
|
|
void MakeUpdates(int update_count, bool commit) {
|
|
|
|
for (int i = 0; i < update_count; i++) {
|
|
|
|
auto t = engine.Begin();
|
|
|
|
version_list.update(*t);
|
|
|
|
if (commit)
|
|
|
|
t->Commit();
|
|
|
|
else
|
|
|
|
t->Abort();
|
|
|
|
}
|
2017-03-29 18:37:58 +08:00
|
|
|
}
|
|
|
|
|
2017-06-21 17:29:13 +08:00
|
|
|
auto GcDeleted(tx::Transaction *latest = nullptr) {
|
2017-06-12 16:21:19 +08:00
|
|
|
return version_list.GcDeleted(GcSnapshot(engine, latest), engine);
|
2017-05-18 21:23:08 +08:00
|
|
|
}
|
2017-06-12 16:21:19 +08:00
|
|
|
};
|
2017-04-14 23:32:59 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
TEST_F(MvccGcTest, RemoveAndAbort) {
|
|
|
|
auto t = engine.Begin();
|
|
|
|
version_list.remove(*t);
|
|
|
|
t->Abort();
|
|
|
|
auto ret = GcDeleted();
|
|
|
|
EXPECT_EQ(ret.first, false);
|
|
|
|
EXPECT_EQ(ret.second, nullptr);
|
|
|
|
EXPECT_EQ(record_destruction_count, 0);
|
|
|
|
}
|
2017-04-14 23:32:59 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
TEST_F(MvccGcTest, UpdateAndAbort) {
|
|
|
|
MakeUpdates(1, false);
|
|
|
|
auto ret = GcDeleted();
|
|
|
|
EXPECT_EQ(ret.first, false);
|
|
|
|
EXPECT_EQ(ret.second, nullptr);
|
|
|
|
EXPECT_EQ(record_destruction_count, 0);
|
|
|
|
|
|
|
|
MakeUpdates(3, false);
|
|
|
|
ret = GcDeleted();
|
|
|
|
EXPECT_EQ(ret.first, false);
|
|
|
|
EXPECT_EQ(ret.second, nullptr);
|
|
|
|
EXPECT_EQ(record_destruction_count, 0);
|
|
|
|
}
|
2017-04-14 23:32:59 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
TEST_F(MvccGcTest, RemoveAndCommit) {
|
|
|
|
auto t = engine.Begin();
|
|
|
|
version_list.remove(*t);
|
|
|
|
t->Commit();
|
|
|
|
auto ret = GcDeleted();
|
|
|
|
EXPECT_EQ(ret.first, true);
|
|
|
|
EXPECT_NE(ret.second, nullptr);
|
|
|
|
delete ret.second;
|
|
|
|
EXPECT_EQ(record_destruction_count, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MvccGcTest, UpdateAndCommit) {
|
|
|
|
MakeUpdates(4, true);
|
|
|
|
auto ret = GcDeleted();
|
|
|
|
EXPECT_EQ(ret.first, false);
|
|
|
|
EXPECT_NE(ret.second, nullptr);
|
|
|
|
delete ret.second;
|
|
|
|
EXPECT_EQ(record_destruction_count, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MvccGcTest, OldestTransactionSnapshot) {
|
|
|
|
// this test validates that we can't delete
|
|
|
|
// a record that has been expired by a transaction (t1)
|
|
|
|
// committed before GC starts (when t2 is oldest),
|
|
|
|
// if t1 is in t2's snapshot.
|
|
|
|
// this is because there could exist transcation t3
|
|
|
|
// that also has t1 in it's snapshot, and consequently
|
|
|
|
// does not see the expiration and sees the record
|
|
|
|
auto t1 = engine.Begin();
|
|
|
|
auto t2 = engine.Begin();
|
|
|
|
version_list.remove(*t1);
|
|
|
|
t1->Commit();
|
|
|
|
|
|
|
|
auto ret = GcDeleted(t2);
|
|
|
|
EXPECT_EQ(ret.first, false);
|
|
|
|
EXPECT_EQ(ret.second, nullptr);
|
|
|
|
EXPECT_EQ(record_destruction_count, 0);
|
2017-03-29 18:37:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-12 16:21:19 +08:00
|
|
|
* Test integration of garbage collector with MVCC GC. Delete version lists
|
|
|
|
* which are
|
2017-03-29 18:37:58 +08:00
|
|
|
* empty (not visible from any future transaction) from the skiplist.
|
|
|
|
*/
|
2017-04-10 21:44:36 +08:00
|
|
|
TEST(GarbageCollector, GcClean) {
|
2017-06-12 16:21:19 +08:00
|
|
|
SkipList<mvcc::VersionList<DestrCountRec> *> skiplist;
|
2017-03-29 18:37:58 +08:00
|
|
|
tx::Engine engine;
|
2017-06-12 16:21:19 +08:00
|
|
|
DeferredDeleter<DestrCountRec> deleter;
|
|
|
|
DeferredDeleter<mvcc::VersionList<DestrCountRec>> vlist_deleter;
|
|
|
|
GarbageCollector<DestrCountRec> gc(skiplist, deleter, vlist_deleter);
|
|
|
|
|
|
|
|
// create a version list in transaction t1
|
|
|
|
auto t1 = engine.Begin();
|
|
|
|
std::atomic<int> record_destruction_count{0};
|
|
|
|
auto vl = new mvcc::VersionList<DestrCountRec>(*t1, record_destruction_count);
|
2017-03-29 18:37:58 +08:00
|
|
|
auto access = skiplist.access();
|
|
|
|
access.insert(vl);
|
2017-06-12 16:21:19 +08:00
|
|
|
t1->Commit();
|
|
|
|
|
|
|
|
// run garbage collection that has nothing co collect
|
|
|
|
gc.Run(GcSnapshot(engine, nullptr), engine);
|
|
|
|
EXPECT_EQ(deleter.Count(), 0);
|
|
|
|
EXPECT_EQ(vlist_deleter.Count(), 0);
|
|
|
|
EXPECT_EQ(record_destruction_count, 0);
|
2017-03-29 18:37:58 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
// delete the only record in the version-list in transaction t2
|
|
|
|
auto t2 = engine.Begin();
|
2017-05-16 18:22:28 +08:00
|
|
|
vl->remove(*t2);
|
2017-06-12 16:21:19 +08:00
|
|
|
t2->Commit();
|
|
|
|
gc.Run(GcSnapshot(engine, nullptr), engine);
|
2017-04-14 23:32:59 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
// check that we destroyed the record
|
2017-04-14 23:32:59 +08:00
|
|
|
EXPECT_EQ(deleter.Count(), 1);
|
2017-06-12 16:21:19 +08:00
|
|
|
deleter.FreeExpiredObjects(engine.Count() + 1);
|
2017-04-14 23:32:59 +08:00
|
|
|
EXPECT_EQ(deleter.Count(), 0);
|
2017-06-12 16:21:19 +08:00
|
|
|
EXPECT_EQ(record_destruction_count, 1);
|
2017-04-14 23:32:59 +08:00
|
|
|
|
2017-06-12 16:21:19 +08:00
|
|
|
// check that we destroyed the version list
|
2017-04-14 23:32:59 +08:00
|
|
|
EXPECT_EQ(vlist_deleter.Count(), 1);
|
2017-06-12 16:21:19 +08:00
|
|
|
vlist_deleter.FreeExpiredObjects(engine.Count() + 1);
|
2017-04-14 23:32:59 +08:00
|
|
|
EXPECT_EQ(vlist_deleter.Count(), 0);
|
2017-03-29 18:37:58 +08:00
|
|
|
|
2017-09-27 22:53:11 +08:00
|
|
|
EXPECT_EQ(access.size(), 0U);
|
2017-03-29 18:37:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
2017-06-21 17:29:13 +08:00
|
|
|
google::InitGoogleLogging(argv[0]);
|
2017-03-29 18:37:58 +08:00
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|