2018-01-10 19:57:44 +08:00
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include "query/typed_value.hpp"
|
2018-11-07 01:15:55 +08:00
|
|
|
#include "storage/common/types/property_value_store.hpp"
|
|
|
|
#include "storage/common/types/types.hpp"
|
2018-10-04 21:23:07 +08:00
|
|
|
#include "storage/distributed/edge.hpp"
|
2019-05-03 21:26:15 +08:00
|
|
|
#include "storage/distributed/mvcc/version_list.hpp"
|
2018-11-07 01:15:55 +08:00
|
|
|
#include "storage/distributed/rpc/serialization.hpp"
|
2018-10-04 21:23:07 +08:00
|
|
|
#include "storage/distributed/vertex.hpp"
|
2018-10-05 18:48:57 +08:00
|
|
|
#include "transactions/distributed/engine_single_node.hpp"
|
2018-01-10 19:57:44 +08:00
|
|
|
|
2019-05-03 21:26:15 +08:00
|
|
|
#include "slk_common.hpp"
|
|
|
|
|
2018-01-16 17:09:15 +08:00
|
|
|
using namespace storage;
|
2018-01-10 19:57:44 +08:00
|
|
|
|
|
|
|
template <typename TAddress>
|
|
|
|
TAddress ToGlobal(const TAddress &address, int worker_id) {
|
|
|
|
if (address.is_remote()) return address;
|
|
|
|
return TAddress{address.local()->gid_, worker_id};
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CHECK_RETURN(condition) \
|
|
|
|
{ \
|
|
|
|
if (!(condition)) return false; \
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckEdges(const Edges &e1, int w1, const Edges &e2, int w2) {
|
|
|
|
CHECK_RETURN(e1.size() == e2.size());
|
|
|
|
auto e1_it = e1.begin();
|
|
|
|
for (auto e2_it = e2.begin(); e2_it != e2.end(); ++e1_it, ++e2_it) {
|
|
|
|
CHECK_RETURN(ToGlobal(e1_it->vertex, w1) == ToGlobal(e2_it->vertex, w2));
|
|
|
|
CHECK_RETURN(ToGlobal(e1_it->edge, w1) == ToGlobal(e2_it->edge, w2));
|
|
|
|
CHECK_RETURN(e1_it->edge_type == e2_it->edge_type);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckProperties(const PropertyValueStore &p1,
|
|
|
|
const PropertyValueStore &p2) {
|
|
|
|
CHECK_RETURN(p1.size() == p2.size());
|
|
|
|
auto p1_it = p1.begin();
|
|
|
|
for (auto p2_it = p2.begin(); p2_it != p2.end(); ++p1_it, ++p2_it) {
|
|
|
|
CHECK_RETURN(p1_it->first == p2_it->first);
|
|
|
|
auto tv =
|
|
|
|
query::TypedValue(p1_it->second) == query::TypedValue(p2_it->second);
|
|
|
|
CHECK_RETURN(tv.IsBool());
|
|
|
|
CHECK_RETURN(tv.ValueBool());
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckVertex(const Vertex &v1, int w1, const Vertex &v2, int w2) {
|
|
|
|
CHECK_RETURN(CheckEdges(v1.in_, w1, v2.in_, w2));
|
|
|
|
CHECK_RETURN(CheckEdges(v1.out_, w1, v2.out_, w2));
|
|
|
|
CHECK_RETURN(v1.labels_ == v2.labels_);
|
|
|
|
CHECK_RETURN(CheckProperties(v1.properties_, v2.properties_));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckEdge(const Edge &e1, int w1, const Edge &e2, int w2) {
|
|
|
|
CHECK_RETURN(ToGlobal(e1.from_, w1) == ToGlobal(e2.from_, w2));
|
|
|
|
CHECK_RETURN(ToGlobal(e1.to_, w1) == ToGlobal(e2.to_, w2));
|
|
|
|
CHECK_RETURN(e1.edge_type_ == e2.edge_type_);
|
|
|
|
CHECK_RETURN(CheckProperties(e1.properties_, e2.properties_));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef CHECK_RETURN
|
|
|
|
|
2019-05-03 21:26:15 +08:00
|
|
|
#define SAVE_AND_LOAD_VERTEX(name, element) \
|
|
|
|
Vertex name; \
|
|
|
|
{ \
|
|
|
|
slk::Loopback loopback; \
|
|
|
|
auto builder = loopback.GetBuilder(); \
|
|
|
|
slk::Save(element, builder, 0); \
|
|
|
|
auto reader = loopback.GetReader(); \
|
|
|
|
slk::Load(&name, reader); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SAVE_AND_LOAD_EDGE(name, element) \
|
|
|
|
std::unique_ptr<Edge> name; \
|
|
|
|
{ \
|
|
|
|
slk::Loopback loopback; \
|
|
|
|
auto builder = loopback.GetBuilder(); \
|
|
|
|
slk::Save(element, builder, 0); \
|
|
|
|
auto reader = loopback.GetReader(); \
|
|
|
|
slk::Load(&name, reader); \
|
2018-01-10 19:57:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(DistributedSerialization, Empty) {
|
|
|
|
Vertex v;
|
|
|
|
int w_id{0};
|
2019-05-03 21:26:15 +08:00
|
|
|
SAVE_AND_LOAD_VERTEX(v_recovered, v)
|
|
|
|
EXPECT_TRUE(CheckVertex(v, w_id, v_recovered, w_id));
|
2018-01-10 19:57:44 +08:00
|
|
|
}
|
|
|
|
|
2019-05-03 21:26:15 +08:00
|
|
|
#define UPDATE_AND_CHECK_V(v, action) \
|
|
|
|
{ \
|
|
|
|
SAVE_AND_LOAD_VERTEX(before, v) \
|
|
|
|
EXPECT_TRUE(CheckVertex(v, 0, before, 0)); \
|
|
|
|
action; \
|
|
|
|
EXPECT_FALSE(CheckVertex(v, 0, before, 0)); \
|
|
|
|
SAVE_AND_LOAD_VERTEX(after, v) \
|
|
|
|
EXPECT_TRUE(CheckVertex(v, 0, after, 0)); \
|
2018-01-10 19:57:44 +08:00
|
|
|
}
|
|
|
|
|
2019-05-03 21:26:15 +08:00
|
|
|
#define UPDATE_AND_CHECK_E(e, action) \
|
|
|
|
{ \
|
|
|
|
SAVE_AND_LOAD_EDGE(before, e) \
|
|
|
|
EXPECT_TRUE(CheckEdge(e, 0, *before, 0)); \
|
|
|
|
action; \
|
|
|
|
EXPECT_FALSE(CheckEdge(e, 0, *before, 0)); \
|
|
|
|
SAVE_AND_LOAD_EDGE(after, e) \
|
|
|
|
EXPECT_TRUE(CheckEdge(e, 0, *after, 0)); \
|
|
|
|
}
|
2018-01-10 19:57:44 +08:00
|
|
|
|
|
|
|
TEST(DistributedSerialization, VertexLabels) {
|
|
|
|
Vertex v;
|
|
|
|
UPDATE_AND_CHECK_V(v, v.labels_.emplace_back(Label(1)));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.labels_.emplace_back(Label(2)));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.labels_.resize(1));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.labels_.clear());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(DistributedSerialization, VertexProperties) {
|
|
|
|
Vertex v;
|
|
|
|
UPDATE_AND_CHECK_V(v, v.properties_.set(Property(1), true));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.properties_.set(Property(1), "string"));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.properties_.set(Property(2), 42));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.properties_.erase(Property(1)));
|
|
|
|
UPDATE_AND_CHECK_V(v, v.properties_.clear());
|
|
|
|
}
|
|
|
|
|
|
|
|
class DistributedSerializationMvcc : public ::testing::Test {
|
|
|
|
protected:
|
2018-09-05 02:30:58 +08:00
|
|
|
tx::EngineSingleNode engine;
|
2018-01-10 19:57:44 +08:00
|
|
|
tx::Transaction *tx = engine.Begin();
|
2018-07-05 03:32:07 +08:00
|
|
|
mvcc::VersionList<Vertex> v1_vlist{*tx, 0, 0};
|
2018-01-10 19:57:44 +08:00
|
|
|
Vertex &v1 = *v1_vlist.Oldest();
|
2018-07-05 03:32:07 +08:00
|
|
|
mvcc::VersionList<Vertex> v2_vlist{*tx, 1, 1};
|
2018-01-10 19:57:44 +08:00
|
|
|
Vertex &v2 = *v2_vlist.Oldest();
|
2018-07-05 03:32:07 +08:00
|
|
|
mvcc::VersionList<Edge> e1_vlist{*tx,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
storage::VertexAddress(&v1_vlist),
|
|
|
|
storage::VertexAddress(&v2_vlist),
|
|
|
|
EdgeType(0)};
|
2018-01-10 19:57:44 +08:00
|
|
|
Edge &e1 = *e1_vlist.Oldest();
|
2018-07-05 03:32:07 +08:00
|
|
|
mvcc::VersionList<Edge> e2_vlist{*tx,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
storage::VertexAddress(&v2_vlist),
|
|
|
|
storage::VertexAddress(&v1_vlist),
|
|
|
|
EdgeType(2)};
|
2018-01-10 19:57:44 +08:00
|
|
|
Edge &e2 = *e2_vlist.Oldest();
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DistributedSerializationMvcc, VertexEdges) {
|
2018-07-05 03:32:07 +08:00
|
|
|
UPDATE_AND_CHECK_V(
|
|
|
|
v1, v1.out_.emplace(storage::VertexAddress(&v2_vlist),
|
|
|
|
storage::EdgeAddress(&e1_vlist), EdgeType(0)));
|
|
|
|
UPDATE_AND_CHECK_V(
|
|
|
|
v2, v2.in_.emplace(storage::VertexAddress(&v1_vlist),
|
|
|
|
storage::EdgeAddress(&e1_vlist), EdgeType(0)));
|
|
|
|
UPDATE_AND_CHECK_V(
|
|
|
|
v1, v1.in_.emplace(storage::VertexAddress(&v2_vlist),
|
|
|
|
storage::EdgeAddress(&e2_vlist), EdgeType(2)));
|
|
|
|
UPDATE_AND_CHECK_V(
|
|
|
|
v2, v2.out_.emplace(storage::VertexAddress(&v1_vlist),
|
|
|
|
storage::EdgeAddress(&e2_vlist), EdgeType(2)));
|
2018-01-10 19:57:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedSerializationMvcc, EdgeFromAndTo) {
|
2018-07-05 03:32:07 +08:00
|
|
|
UPDATE_AND_CHECK_E(e1, e1.from_ = storage::VertexAddress(&v2_vlist));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.to_ = storage::VertexAddress(&v1_vlist));
|
2018-01-10 19:57:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedSerializationMvcc, EdgeType) {
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.edge_type_ = EdgeType(123));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.edge_type_ = EdgeType(55));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedSerializationMvcc, EdgeProperties) {
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.properties_.set(Property(1), true));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.properties_.set(Property(1), "string"));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.properties_.set(Property(2), 42));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.properties_.erase(Property(1)));
|
|
|
|
UPDATE_AND_CHECK_E(e1, e1.properties_.clear());
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef UPDATE_AND_CHECK_E
|
|
|
|
#undef UPDATE_AND_CHECK_V
|
|
|
|
#undef UPDATE_AND_CHECK
|
|
|
|
#undef SAVE_AND_LOAD
|