2018-02-15 17:47:50 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#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-01-24 19:16:14 +08:00
|
|
|
#include "communication/rpc/server.hpp"
|
2018-01-15 21:03:07 +08:00
|
|
|
#include "io/network/endpoint.hpp"
|
2017-12-06 21:12:26 +08:00
|
|
|
#include "transactions/engine_master.hpp"
|
2017-12-22 21:36:25 +08:00
|
|
|
#include "transactions/engine_rpc_messages.hpp"
|
2017-12-06 21:12:26 +08:00
|
|
|
#include "transactions/engine_worker.hpp"
|
2018-02-15 17:47:50 +08:00
|
|
|
#include "transactions/tx_end_listener.hpp"
|
2017-12-06 21:12:26 +08:00
|
|
|
|
|
|
|
using namespace tx;
|
2018-01-24 19:16:14 +08:00
|
|
|
using namespace communication::rpc;
|
2017-12-06 21:12:26 +08:00
|
|
|
|
|
|
|
class WorkerEngineTest : public testing::Test {
|
|
|
|
protected:
|
|
|
|
const std::string local{"127.0.0.1"};
|
|
|
|
|
2018-01-15 21:03:07 +08:00
|
|
|
System master_system_{{local, 0}};
|
2018-01-10 22:10:22 +08:00
|
|
|
MasterEngine master_{master_system_};
|
2017-12-06 21:12:26 +08:00
|
|
|
|
2018-01-24 19:16:14 +08:00
|
|
|
WorkerEngine worker_{master_system_.endpoint()};
|
2017-12-06 21:12:26 +08:00
|
|
|
};
|
|
|
|
|
2018-02-01 17:58:56 +08:00
|
|
|
TEST_F(WorkerEngineTest, BeginOnWorker) {
|
|
|
|
worker_.Begin();
|
|
|
|
auto second = worker_.Begin();
|
|
|
|
EXPECT_EQ(master_.RunningTransaction(second->id_)->snapshot().size(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, AdvanceOnWorker) {
|
|
|
|
auto tx = worker_.Begin();
|
|
|
|
auto cid = tx->cid();
|
|
|
|
EXPECT_EQ(worker_.Advance(tx->id_), cid + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, CommitOnWorker) {
|
|
|
|
auto tx = worker_.Begin();
|
|
|
|
auto tx_id = tx->id_;
|
|
|
|
worker_.Commit(*tx);
|
|
|
|
EXPECT_TRUE(master_.Info(tx_id).is_committed());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, AbortOnWorker) {
|
|
|
|
auto tx = worker_.Begin();
|
|
|
|
auto tx_id = tx->id_;
|
|
|
|
worker_.Abort(*tx);
|
|
|
|
EXPECT_TRUE(master_.Info(tx_id).is_aborted());
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:18:13 +08:00
|
|
|
TEST_F(WorkerEngineTest, RunningTransaction) {
|
2017-12-06 21:12:26 +08:00
|
|
|
master_.Begin();
|
|
|
|
master_.Begin();
|
2018-01-19 18:18:13 +08:00
|
|
|
worker_.RunningTransaction(1);
|
|
|
|
worker_.RunningTransaction(2);
|
2017-12-06 21:12:26 +08:00
|
|
|
int count = 0;
|
|
|
|
worker_.LocalForEachActiveTransaction([&count](Transaction &t) {
|
|
|
|
++count;
|
|
|
|
if (t.id_ == 1) {
|
|
|
|
EXPECT_EQ(t.snapshot(),
|
|
|
|
tx::Snapshot(std::vector<tx::transaction_id_t>{}));
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(t.snapshot(), tx::Snapshot({1}));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
EXPECT_EQ(count, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, Info) {
|
|
|
|
auto *tx_1 = master_.Begin();
|
|
|
|
auto *tx_2 = master_.Begin();
|
|
|
|
// We can't check active transactions in the worker (see comments there for
|
|
|
|
// info).
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalGcSnapshot) {
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalActiveTransactions) {
|
|
|
|
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}));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, GlobalIsActive) {
|
|
|
|
auto *tx_1 = master_.Begin();
|
|
|
|
master_.Begin();
|
|
|
|
auto *tx_3 = master_.Begin();
|
|
|
|
master_.Begin();
|
|
|
|
master_.Commit(*tx_1);
|
|
|
|
master_.Abort(*tx_3);
|
|
|
|
EXPECT_FALSE(worker_.GlobalIsActive(1));
|
|
|
|
EXPECT_TRUE(worker_.GlobalIsActive(2));
|
|
|
|
EXPECT_FALSE(worker_.GlobalIsActive(3));
|
|
|
|
EXPECT_TRUE(worker_.GlobalIsActive(4));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, LocalLast) {
|
|
|
|
master_.Begin();
|
|
|
|
EXPECT_EQ(worker_.LocalLast(), 0);
|
2018-01-19 18:18:13 +08:00
|
|
|
worker_.RunningTransaction(1);
|
2017-12-06 21:12:26 +08:00
|
|
|
EXPECT_EQ(worker_.LocalLast(), 1);
|
|
|
|
master_.Begin();
|
|
|
|
EXPECT_EQ(worker_.LocalLast(), 1);
|
|
|
|
master_.Begin();
|
|
|
|
EXPECT_EQ(worker_.LocalLast(), 1);
|
|
|
|
master_.Begin();
|
2018-01-19 18:18:13 +08:00
|
|
|
worker_.RunningTransaction(4);
|
2017-12-06 21:12:26 +08:00
|
|
|
EXPECT_EQ(worker_.LocalLast(), 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, LocalForEachActiveTransaction) {
|
|
|
|
master_.Begin();
|
2018-01-19 18:18:13 +08:00
|
|
|
worker_.RunningTransaction(1);
|
2017-12-06 21:12:26 +08:00
|
|
|
master_.Begin();
|
|
|
|
master_.Begin();
|
|
|
|
master_.Begin();
|
2018-01-19 18:18:13 +08:00
|
|
|
worker_.RunningTransaction(4);
|
2017-12-06 21:12:26 +08:00
|
|
|
std::unordered_set<tx::transaction_id_t> local;
|
|
|
|
worker_.LocalForEachActiveTransaction(
|
|
|
|
[&local](Transaction &t) { local.insert(t.id_); });
|
|
|
|
EXPECT_EQ(local, std::unordered_set<tx::transaction_id_t>({1, 4}));
|
|
|
|
}
|
2018-02-15 17:47:50 +08:00
|
|
|
|
|
|
|
TEST_F(WorkerEngineTest, TxEndListener) {
|
|
|
|
std::atomic<int> has_expired{0};
|
|
|
|
TxEndListener worker_end_listner{
|
|
|
|
worker_, [&has_expired](transaction_id_t tid) {
|
|
|
|
std::cout << "asdasdadas: " << tid << std::endl;
|
|
|
|
++has_expired; }};
|
|
|
|
|
|
|
|
auto sleep_period =
|
|
|
|
WorkerEngine::kCacheReleasePeriod + std::chrono::milliseconds(200);
|
|
|
|
auto t1 = master_.Begin();
|
|
|
|
auto t2 = master_.Begin();
|
|
|
|
std::this_thread::sleep_for(sleep_period);
|
|
|
|
EXPECT_EQ(has_expired.load(), 0);
|
|
|
|
master_.Commit(*t1);
|
|
|
|
master_.Abort(*t2);
|
|
|
|
std::this_thread::sleep_for(sleep_period);
|
|
|
|
EXPECT_EQ(has_expired.load(), 2);
|
|
|
|
}
|