2017-03-24 16:52:25 +08:00
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
2017-04-14 23:32:59 +08:00
|
|
|
#include "mvcc/id.hpp"
|
2017-03-24 16:52:25 +08:00
|
|
|
#include "mvcc/record.hpp"
|
|
|
|
#include "mvcc/version_list.hpp"
|
|
|
|
#include "storage/vertex.hpp"
|
|
|
|
#include "transactions/engine.hpp"
|
|
|
|
|
2017-04-14 23:32:59 +08:00
|
|
|
#include "gc_common.hpp"
|
2017-03-24 16:52:25 +08:00
|
|
|
|
|
|
|
TEST(MVCC, Case1Test3) {
|
|
|
|
tx::Engine engine;
|
|
|
|
auto t1 = engine.begin();
|
2017-04-03 17:12:15 +08:00
|
|
|
mvcc::VersionList<Prop> version_list(*t1);
|
2017-03-24 16:52:25 +08:00
|
|
|
t1->commit();
|
|
|
|
|
|
|
|
auto t2 = engine.begin();
|
2017-04-10 18:12:40 +08:00
|
|
|
version_list.update(*t2);
|
2017-03-24 16:52:25 +08:00
|
|
|
t2->commit();
|
|
|
|
|
|
|
|
auto t3 = engine.begin();
|
|
|
|
auto t4 = engine.begin();
|
|
|
|
version_list.update(*t4);
|
|
|
|
t4->commit();
|
2017-04-28 16:33:46 +08:00
|
|
|
EXPECT_THROW(version_list.remove(*t3), mvcc::SerializationError);
|
2017-03-24 16:52:25 +08:00
|
|
|
}
|
|
|
|
|
2017-04-10 18:12:40 +08:00
|
|
|
TEST(MVCC, InSnapshotSerializationError) {
|
2017-03-24 16:52:25 +08:00
|
|
|
tx::Engine engine;
|
|
|
|
auto t1 = engine.begin();
|
2017-04-03 17:12:15 +08:00
|
|
|
mvcc::VersionList<Prop> version_list(*t1);
|
2017-03-24 16:52:25 +08:00
|
|
|
t1->commit();
|
|
|
|
|
2017-03-29 18:37:58 +08:00
|
|
|
auto t2 = engine.begin();
|
|
|
|
version_list.update(*t2);
|
|
|
|
auto t3 = engine.begin(); // t2 is in snapshot of t3
|
|
|
|
t2->commit();
|
|
|
|
|
2017-04-28 16:33:46 +08:00
|
|
|
EXPECT_THROW(version_list.update(*t3), mvcc::SerializationError);
|
2017-04-10 18:12:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that we don't delete records when we re-link.
|
|
|
|
TEST(MVCC, UpdateDontDelete) {
|
|
|
|
std::atomic<int> count{0};
|
|
|
|
{
|
|
|
|
tx::Engine engine;
|
|
|
|
auto t1 = engine.begin();
|
|
|
|
mvcc::VersionList<PropCount> version_list(*t1, count);
|
|
|
|
t1->commit();
|
|
|
|
|
|
|
|
auto t2 = engine.begin();
|
|
|
|
version_list.update(*t2);
|
|
|
|
t2->abort();
|
|
|
|
EXPECT_EQ(count, 0);
|
|
|
|
|
|
|
|
auto t3 = engine.begin();
|
|
|
|
|
|
|
|
// Update re-links the node and shouldn't clear it yet.
|
|
|
|
version_list.update(*t3);
|
|
|
|
EXPECT_EQ(count, 0);
|
|
|
|
|
|
|
|
t3->commit();
|
|
|
|
}
|
|
|
|
EXPECT_EQ(count, 3);
|
2017-03-24 16:52:25 +08:00
|
|
|
}
|
|
|
|
|
2017-04-14 23:32:59 +08:00
|
|
|
// Check that we get the oldest record.
|
|
|
|
TEST(MVCC, Oldest) {
|
|
|
|
tx::Engine engine;
|
|
|
|
auto t1 = engine.begin();
|
|
|
|
mvcc::VersionList<Prop> version_list(*t1);
|
|
|
|
auto first = version_list.Oldest();
|
|
|
|
EXPECT_NE(first, nullptr);
|
|
|
|
for (int i = 0; i < 10; ++i) {
|
|
|
|
engine.advance(t1->id);
|
|
|
|
version_list.update(*t1);
|
|
|
|
EXPECT_EQ(version_list.Oldest(), first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-24 16:52:25 +08:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|