2018-02-15 17:47:50 +08:00
|
|
|
#include <algorithm>
|
2018-10-16 16:58:41 +08:00
|
|
|
#include <memory>
|
2018-02-15 17:47:50 +08:00
|
|
|
#include <mutex>
|
2017-12-06 21:12:26 +08:00
|
|
|
#include <unordered_set>
|
2018-02-15 17:47:50 +08:00
|
|
|
#include <vector>
|
2017-12-06 21:12:26 +08:00
|
|
|
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
2018-09-05 02:30:58 +08:00
|
|
|
#include "transactions/distributed/engine_master.hpp"
|
|
|
|
#include "transactions/distributed/engine_worker.hpp"
|
2017-12-06 21:12:26 +08:00
|
|
|
|
2018-10-16 15:12:19 +08:00
|
|
|
#include "test_coordination.hpp"
|
|
|
|
|
2017-12-06 21:12:26 +08:00
|
|
|
using namespace tx;
|
2018-01-24 19:16:14 +08:00
|
|
|
using namespace communication::rpc;
|
2018-03-21 17:26:43 +08:00
|
|
|
using namespace distributed;
|
2017-12-06 21:12:26 +08:00
|
|
|
|
|
|
|
class WorkerEngineTest : public testing::Test {
|
|
|
|
protected:
|
2018-10-16 16:58:41 +08:00
|
|
|
void SetUp() override {
|
|
|
|
master_coordination_ = std::make_unique<TestMasterCoordination>();
|
|
|
|
master_ = std::make_unique<EngineMaster>(master_coordination_.get());
|
2018-10-16 18:14:48 +08:00
|
|
|
master_coordination_->Start();
|
2018-10-16 16:58:41 +08:00
|
|
|
|
|
|
|
worker_coordination_ = std::make_unique<TestWorkerCoordination>(
|
|
|
|
master_coordination_->GetServerEndpoint(), 1);
|
|
|
|
worker_ = std::make_unique<EngineWorker>(worker_coordination_.get());
|
2018-10-16 18:14:48 +08:00
|
|
|
worker_coordination_->Start();
|
2018-10-16 16:58:41 +08:00
|
|
|
}
|
|
|
|
|
2018-09-27 21:07:46 +08:00
|
|
|
void TearDown() override {
|
2018-10-16 16:58:41 +08:00
|
|
|
std::thread thread([this] { worker_coordination_->Stop(); });
|
|
|
|
master_coordination_->Stop();
|
2018-10-16 15:12:19 +08:00
|
|
|
if (thread.joinable()) thread.join();
|
2018-09-27 21:07:46 +08:00
|
|
|
}
|
|
|
|
|
2018-10-16 16:58:41 +08:00
|
|
|
std::unique_ptr<TestMasterCoordination> master_coordination_;
|
|
|
|
std::unique_ptr<EngineMaster> master_;
|
2018-03-21 17:26:43 +08:00
|
|
|
|
2018-10-16 16:58:41 +08:00
|
|
|
std::unique_ptr<TestWorkerCoordination> worker_coordination_;
|
|
|
|
std::unique_ptr<EngineWorker> worker_;
|
2017-12-06 21:12:26 +08:00
|
|
|
};
|
|
|
|
|
2018-02-01 17:58:56 +08:00
|
|
|
TEST_F(WorkerEngineTest, BeginOnWorker) {
|
2018-10-16 16:58:41 +08:00
|
|
|
worker_->Begin();
|
|
|
|
auto second = worker_->Begin();
|
|
|
|
EXPECT_EQ(master_->RunningTransaction(second->id_)->snapshot().size(), 1);
|
2018-02-01 17:58:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, AdvanceOnWorker) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto tx = worker_->Begin();
|
2018-02-01 17:58:56 +08:00
|
|
|
auto cid = tx->cid();
|
2018-10-16 16:58:41 +08:00
|
|
|
EXPECT_EQ(worker_->Advance(tx->id_), cid + 1);
|
2018-02-01 17:58:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, CommitOnWorker) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto tx = worker_->Begin();
|
2018-02-01 17:58:56 +08:00
|
|
|
auto tx_id = tx->id_;
|
2018-10-16 16:58:41 +08:00
|
|
|
worker_->Commit(*tx);
|
|
|
|
EXPECT_TRUE(master_->Info(tx_id).is_committed());
|
2018-02-01 17:58:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, AbortOnWorker) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto tx = worker_->Begin();
|
2018-02-01 17:58:56 +08:00
|
|
|
auto tx_id = tx->id_;
|
2018-10-16 16:58:41 +08:00
|
|
|
worker_->Abort(*tx);
|
|
|
|
EXPECT_TRUE(master_->Info(tx_id).is_aborted());
|
2018-02-01 17:58:56 +08:00
|
|
|
}
|
|
|
|
|
2018-01-19 18:18:13 +08:00
|
|
|
TEST_F(WorkerEngineTest, RunningTransaction) {
|
2018-10-16 16:58:41 +08:00
|
|
|
master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
worker_->RunningTransaction(1);
|
|
|
|
worker_->RunningTransaction(2);
|
2017-12-06 21:12:26 +08:00
|
|
|
int count = 0;
|
2018-10-16 16:58:41 +08:00
|
|
|
worker_->LocalForEachActiveTransaction([&count](Transaction &t) {
|
2017-12-06 21:12:26 +08:00
|
|
|
++count;
|
|
|
|
if (t.id_ == 1) {
|
2018-09-27 21:07:46 +08:00
|
|
|
EXPECT_EQ(t.snapshot(), tx::Snapshot(std::vector<tx::TransactionId>{}));
|
2017-12-06 21:12:26 +08:00
|
|
|
} else {
|
|
|
|
EXPECT_EQ(t.snapshot(), tx::Snapshot({1}));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
EXPECT_EQ(count, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, Info) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto *tx_1 = master_->Begin();
|
|
|
|
auto *tx_2 = master_->Begin();
|
2017-12-06 21:12:26 +08:00
|
|
|
// We can't check active transactions in the worker (see comments there for
|
|
|
|
// info).
|
2018-10-16 16:58:41 +08:00
|
|
|
master_->Commit(*tx_1);
|
|
|
|
EXPECT_TRUE(master_->Info(1).is_committed());
|
|
|
|
EXPECT_TRUE(worker_->Info(1).is_committed());
|
|
|
|
master_->Abort(*tx_2);
|
|
|
|
EXPECT_TRUE(master_->Info(2).is_aborted());
|
|
|
|
EXPECT_TRUE(worker_->Info(2).is_aborted());
|
2017-12-06 21:12:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalGcSnapshot) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto *tx_1 = master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
master_->Commit(*tx_1);
|
|
|
|
EXPECT_EQ(master_->GlobalGcSnapshot(), tx::Snapshot({1, 2}));
|
|
|
|
EXPECT_EQ(worker_->GlobalGcSnapshot(), master_->GlobalGcSnapshot());
|
2017-12-06 21:12:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalActiveTransactions) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto *tx_1 = master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
auto *tx_3 = master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
master_->Commit(*tx_1);
|
|
|
|
master_->Abort(*tx_3);
|
|
|
|
EXPECT_EQ(worker_->GlobalActiveTransactions(), tx::Snapshot({2, 4}));
|
2017-12-06 21:12:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, LocalLast) {
|
2018-10-16 16:58:41 +08:00
|
|
|
master_->Begin();
|
|
|
|
EXPECT_EQ(worker_->LocalLast(), 0);
|
|
|
|
worker_->RunningTransaction(1);
|
|
|
|
EXPECT_EQ(worker_->LocalLast(), 1);
|
|
|
|
master_->Begin();
|
|
|
|
EXPECT_EQ(worker_->LocalLast(), 1);
|
|
|
|
master_->Begin();
|
|
|
|
EXPECT_EQ(worker_->LocalLast(), 1);
|
|
|
|
master_->Begin();
|
|
|
|
worker_->RunningTransaction(4);
|
|
|
|
EXPECT_EQ(worker_->LocalLast(), 4);
|
2017-12-06 21:12:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, LocalForEachActiveTransaction) {
|
2018-10-16 16:58:41 +08:00
|
|
|
master_->Begin();
|
|
|
|
worker_->RunningTransaction(1);
|
|
|
|
master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
master_->Begin();
|
|
|
|
worker_->RunningTransaction(4);
|
2018-04-19 21:32:41 +08:00
|
|
|
std::unordered_set<tx::TransactionId> local;
|
2018-10-16 16:58:41 +08:00
|
|
|
worker_->LocalForEachActiveTransaction(
|
2017-12-06 21:12:26 +08:00
|
|
|
[&local](Transaction &t) { local.insert(t.id_); });
|
2018-04-19 21:32:41 +08:00
|
|
|
EXPECT_EQ(local, std::unordered_set<tx::TransactionId>({1, 4}));
|
2017-12-06 21:12:26 +08:00
|
|
|
}
|
2018-03-22 17:44:38 +08:00
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, EnsureTxIdGreater) {
|
2018-10-16 16:58:41 +08:00
|
|
|
ASSERT_LE(master_->Begin()->id_, 40);
|
|
|
|
worker_->EnsureNextIdGreater(42);
|
|
|
|
EXPECT_EQ(master_->Begin()->id_, 43);
|
|
|
|
EXPECT_EQ(worker_->Begin()->id_, 44);
|
2018-03-22 17:44:38 +08:00
|
|
|
}
|
2018-03-23 20:57:29 +08:00
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalNext) {
|
2018-10-16 16:58:41 +08:00
|
|
|
auto tx = master_->Begin();
|
|
|
|
EXPECT_NE(worker_->LocalLast(), worker_->GlobalLast());
|
|
|
|
EXPECT_EQ(master_->LocalLast(), worker_->GlobalLast());
|
|
|
|
EXPECT_EQ(worker_->GlobalLast(), tx->id_);
|
2018-03-23 20:57:29 +08:00
|
|
|
}
|