2018-03-02 00:39:22 +08:00
|
|
|
#include <functional>
|
2018-03-14 17:53:20 +08:00
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
2018-03-02 00:39:22 +08:00
|
|
|
|
2018-10-05 18:37:23 +08:00
|
|
|
#include "database/distributed/graph_db_accessor.hpp"
|
2018-03-23 22:21:46 +08:00
|
|
|
#include "distributed/updates_rpc_clients.hpp"
|
|
|
|
#include "distributed/updates_rpc_server.hpp"
|
2018-03-14 17:53:20 +08:00
|
|
|
#include "query/typed_value.hpp"
|
2018-11-07 01:15:55 +08:00
|
|
|
#include "storage/common/types/property_value.hpp"
|
2018-02-05 16:48:45 +08:00
|
|
|
|
|
|
|
#include "distributed_common.hpp"
|
|
|
|
|
2018-02-06 23:26:29 +08:00
|
|
|
class DistributedUpdateTest : public DistributedGraphDbTest {
|
|
|
|
protected:
|
2018-07-05 16:55:00 +08:00
|
|
|
DistributedUpdateTest() : DistributedGraphDbTest("update") {}
|
|
|
|
|
2018-02-06 23:26:29 +08:00
|
|
|
std::unique_ptr<database::GraphDbAccessor> dba1;
|
|
|
|
std::unique_ptr<database::GraphDbAccessor> dba2;
|
|
|
|
storage::Label label;
|
|
|
|
std::unique_ptr<VertexAccessor> v1_dba1;
|
|
|
|
std::unique_ptr<VertexAccessor> v1_dba2;
|
|
|
|
|
|
|
|
void SetUp() override {
|
|
|
|
DistributedGraphDbTest::SetUp();
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba_tx1 = worker(1).Access();
|
|
|
|
auto v = dba_tx1->InsertVertex();
|
2018-02-06 23:26:29 +08:00
|
|
|
auto v_ga = v.GlobalAddress();
|
2018-07-26 15:08:21 +08:00
|
|
|
dba_tx1->Commit();
|
2018-02-06 23:26:29 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
dba1 = worker(1).Access();
|
|
|
|
dba2 = worker(2).Access(dba1->transaction_id());
|
2018-02-06 23:26:29 +08:00
|
|
|
|
|
|
|
v1_dba1 = std::make_unique<VertexAccessor>(v_ga, *dba1);
|
|
|
|
v1_dba2 = std::make_unique<VertexAccessor>(v_ga, *dba2);
|
|
|
|
ASSERT_FALSE(v1_dba2->address().is_local());
|
|
|
|
label = dba1->Label("l");
|
|
|
|
v1_dba2->add_label(label);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TearDown() override {
|
|
|
|
dba2 = nullptr;
|
|
|
|
dba1 = nullptr;
|
|
|
|
DistributedGraphDbTest::TearDown();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#define EXPECT_LABEL(var, old_result, new_result) \
|
|
|
|
{ \
|
|
|
|
var->SwitchOld(); \
|
|
|
|
EXPECT_EQ(var->has_label(label), old_result); \
|
|
|
|
var->SwitchNew(); \
|
|
|
|
EXPECT_EQ(var->has_label(label), new_result); \
|
|
|
|
}
|
|
|
|
|
2018-03-23 22:21:46 +08:00
|
|
|
TEST_F(DistributedUpdateTest, UpdateLocalOnly) {
|
2018-02-06 23:26:29 +08:00
|
|
|
EXPECT_LABEL(v1_dba2, false, true);
|
|
|
|
EXPECT_LABEL(v1_dba1, false, false);
|
|
|
|
}
|
|
|
|
|
2018-03-23 22:21:46 +08:00
|
|
|
TEST_F(DistributedUpdateTest, UpdateApply) {
|
2018-02-06 23:26:29 +08:00
|
|
|
EXPECT_LABEL(v1_dba1, false, false);
|
2018-03-23 22:21:46 +08:00
|
|
|
worker(1).updates_server().Apply(dba1->transaction_id());
|
2018-02-06 23:26:29 +08:00
|
|
|
EXPECT_LABEL(v1_dba1, false, true);
|
2018-02-05 16:48:45 +08:00
|
|
|
}
|
2018-02-06 23:26:29 +08:00
|
|
|
|
|
|
|
#undef EXPECT_LABEL
|
2018-02-20 21:48:36 +08:00
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
class DistributedGraphDbSimpleUpdatesTest : public DistributedGraphDbTest {
|
|
|
|
public:
|
|
|
|
DistributedGraphDbSimpleUpdatesTest()
|
|
|
|
: DistributedGraphDbTest("simple_updates") {}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, CreateVertex) {
|
2018-02-20 21:48:36 +08:00
|
|
|
gid::Gid gid;
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(1).Access();
|
2018-08-28 22:28:35 +08:00
|
|
|
auto v = database::InsertVertexIntoRemote(dba.get(), 2, {}, {},
|
|
|
|
std::experimental::nullopt);
|
2018-02-20 21:48:36 +08:00
|
|
|
gid = v.gid();
|
2018-07-26 15:08:21 +08:00
|
|
|
dba->Commit();
|
2018-02-20 21:48:36 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(2).Access();
|
|
|
|
auto v = dba->FindVertexOptional(gid, false);
|
2018-02-20 21:48:36 +08:00
|
|
|
ASSERT_TRUE(v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, CreateVertexWithUpdate) {
|
2018-02-20 21:48:36 +08:00
|
|
|
gid::Gid gid;
|
|
|
|
storage::Property prop;
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(1).Access();
|
2018-08-28 22:28:35 +08:00
|
|
|
auto v = database::InsertVertexIntoRemote(dba.get(), 2, {}, {},
|
|
|
|
std::experimental::nullopt);
|
2018-02-20 21:48:36 +08:00
|
|
|
gid = v.gid();
|
2018-07-26 15:08:21 +08:00
|
|
|
prop = dba->Property("prop");
|
2018-02-20 21:48:36 +08:00
|
|
|
v.PropsSet(prop, 42);
|
2018-07-26 15:08:21 +08:00
|
|
|
worker(2).updates_server().Apply(dba->transaction_id());
|
|
|
|
dba->Commit();
|
2018-02-20 21:48:36 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(2).Access();
|
|
|
|
auto v = dba->FindVertexOptional(gid, false);
|
2018-02-20 21:48:36 +08:00
|
|
|
ASSERT_TRUE(v);
|
|
|
|
EXPECT_EQ(v->PropsAt(prop).Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, CreateVertexWithData) {
|
2018-02-20 21:48:36 +08:00
|
|
|
gid::Gid gid;
|
|
|
|
storage::Label l1;
|
|
|
|
storage::Label l2;
|
|
|
|
storage::Property prop;
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(1).Access();
|
|
|
|
l1 = dba->Label("l1");
|
|
|
|
l2 = dba->Label("l2");
|
|
|
|
prop = dba->Property("prop");
|
2018-08-28 22:28:35 +08:00
|
|
|
auto v = database::InsertVertexIntoRemote(
|
|
|
|
dba.get(), 2, {l1, l2}, {{prop, 42}}, std::experimental::nullopt);
|
2018-02-20 21:48:36 +08:00
|
|
|
gid = v.gid();
|
|
|
|
|
|
|
|
// Check local visibility before commit.
|
|
|
|
EXPECT_TRUE(v.has_label(l1));
|
|
|
|
EXPECT_TRUE(v.has_label(l2));
|
|
|
|
EXPECT_EQ(v.PropsAt(prop).Value<int64_t>(), 42);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
worker(2).updates_server().Apply(dba->transaction_id());
|
|
|
|
dba->Commit();
|
2018-02-20 21:48:36 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(2).Access();
|
|
|
|
auto v = dba->FindVertexOptional(gid, false);
|
2018-02-20 21:48:36 +08:00
|
|
|
ASSERT_TRUE(v);
|
|
|
|
// Check remote data after commit.
|
|
|
|
EXPECT_TRUE(v->has_label(l1));
|
|
|
|
EXPECT_TRUE(v->has_label(l2));
|
|
|
|
EXPECT_EQ(v->PropsAt(prop).Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-22 21:29:20 +08:00
|
|
|
// Checks if expiring a local record for a local update before applying a remote
|
|
|
|
// update delta causes a problem
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, UpdateVertexRemoteAndLocal) {
|
2018-02-22 21:29:20 +08:00
|
|
|
gid::Gid gid;
|
|
|
|
storage::Label l1;
|
|
|
|
storage::Label l2;
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = worker(1).Access();
|
|
|
|
auto v = dba->InsertVertex();
|
2018-02-22 21:29:20 +08:00
|
|
|
gid = v.gid();
|
2018-07-26 15:08:21 +08:00
|
|
|
l1 = dba->Label("label1");
|
|
|
|
l2 = dba->Label("label2");
|
|
|
|
dba->Commit();
|
2018-02-22 21:29:20 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v_local = dba1->FindVertex(gid, false);
|
|
|
|
auto v_remote = VertexAccessor(storage::VertexAddress(gid, 1), *dba0);
|
2018-02-22 21:29:20 +08:00
|
|
|
|
|
|
|
v_remote.add_label(l2);
|
|
|
|
v_local.add_label(l1);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
auto result = worker(1).updates_server().Apply(dba0->transaction_id());
|
2018-03-23 22:21:46 +08:00
|
|
|
EXPECT_EQ(result, distributed::UpdateResult::DONE);
|
2018-02-22 21:29:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, AddSameLabelRemoteAndLocal) {
|
2018-02-26 21:55:55 +08:00
|
|
|
auto v_address = InsertVertex(worker(1));
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v_local = dba1->FindVertex(v_address.gid(), false);
|
|
|
|
auto v_remote = VertexAccessor(v_address, *dba0);
|
|
|
|
auto l1 = dba1->Label("label");
|
2018-02-26 21:55:55 +08:00
|
|
|
v_remote.add_label(l1);
|
|
|
|
v_local.add_label(l1);
|
2018-07-26 15:08:21 +08:00
|
|
|
worker(1).updates_server().Apply(dba0->transaction_id());
|
|
|
|
dba0->Commit();
|
2018-02-26 21:55:55 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v = dba1->FindVertex(v_address.gid(), false);
|
2018-02-26 21:55:55 +08:00
|
|
|
EXPECT_EQ(v.labels().size(), 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, IndexGetsUpdatedRemotely) {
|
2018-02-26 21:55:55 +08:00
|
|
|
storage::VertexAddress v_remote = InsertVertex(worker(1));
|
|
|
|
storage::Label label;
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
label = dba0->Label("label");
|
|
|
|
VertexAccessor va(v_remote, *dba0);
|
2018-02-26 21:55:55 +08:00
|
|
|
va.add_label(label);
|
2018-07-26 15:08:21 +08:00
|
|
|
worker(1).updates_server().Apply(dba0->transaction_id());
|
|
|
|
dba0->Commit();
|
2018-02-26 21:55:55 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba1 = worker(1).Access();
|
|
|
|
auto vertices = dba1->Vertices(label, false);
|
2018-02-26 21:55:55 +08:00
|
|
|
EXPECT_EQ(std::distance(vertices.begin(), vertices.end()), 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, DeleteVertexRemoteCommit) {
|
2018-02-28 17:36:48 +08:00
|
|
|
auto v_address = InsertVertex(worker(1));
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v_remote = VertexAccessor(v_address, *dba0);
|
|
|
|
dba0->RemoveVertex(v_remote);
|
|
|
|
EXPECT_TRUE(dba1->FindVertexOptional(v_address.gid(), true));
|
|
|
|
EXPECT_EQ(worker(1).updates_server().Apply(dba0->transaction_id()),
|
2018-03-23 22:21:46 +08:00
|
|
|
distributed::UpdateResult::DONE);
|
2018-07-26 15:08:21 +08:00
|
|
|
EXPECT_FALSE(dba1->FindVertexOptional(v_address.gid(), true));
|
2018-02-28 17:36:48 +08:00
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, DeleteVertexRemoteBothDelete) {
|
2018-02-28 17:36:48 +08:00
|
|
|
auto v_address = InsertVertex(worker(1));
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v_local = dba1->FindVertex(v_address.gid(), false);
|
|
|
|
auto v_remote = VertexAccessor(v_address, *dba0);
|
|
|
|
EXPECT_TRUE(dba1->RemoveVertex(v_local));
|
|
|
|
EXPECT_TRUE(dba0->RemoveVertex(v_remote));
|
|
|
|
EXPECT_EQ(worker(1).updates_server().Apply(dba0->transaction_id()),
|
2018-03-23 22:21:46 +08:00
|
|
|
distributed::UpdateResult::DONE);
|
2018-07-26 15:08:21 +08:00
|
|
|
EXPECT_FALSE(dba1->FindVertexOptional(v_address.gid(), true));
|
2018-02-28 17:36:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-05 16:55:00 +08:00
|
|
|
TEST_F(DistributedGraphDbSimpleUpdatesTest, DeleteVertexRemoteStillConnected) {
|
2018-02-28 17:36:48 +08:00
|
|
|
auto v_address = InsertVertex(worker(1));
|
|
|
|
auto e_address = InsertEdge(v_address, v_address, "edge");
|
|
|
|
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto v_remote = VertexAccessor(v_address, *dba0);
|
|
|
|
dba0->RemoveVertex(v_remote);
|
|
|
|
EXPECT_EQ(worker(1).updates_server().Apply(dba0->transaction_id()),
|
2018-03-23 22:21:46 +08:00
|
|
|
distributed::UpdateResult::UNABLE_TO_DELETE_VERTEX_ERROR);
|
2018-07-26 15:08:21 +08:00
|
|
|
EXPECT_TRUE(dba1->FindVertexOptional(v_address.gid(), true));
|
2018-02-28 17:36:48 +08:00
|
|
|
}
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto e_local = dba1->FindEdge(e_address.gid(), false);
|
2019-02-15 22:57:14 +08:00
|
|
|
EXPECT_TRUE(dba1->FindVertexOptional(v_address.gid(), false));
|
2018-07-26 15:08:21 +08:00
|
|
|
auto v_remote = VertexAccessor(v_address, *dba0);
|
2018-02-28 17:36:48 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
dba1->RemoveEdge(e_local);
|
|
|
|
dba0->RemoveVertex(v_remote);
|
2018-02-28 17:36:48 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
EXPECT_EQ(worker(1).updates_server().Apply(dba0->transaction_id()),
|
2018-03-23 22:21:46 +08:00
|
|
|
distributed::UpdateResult::DONE);
|
2018-07-26 15:08:21 +08:00
|
|
|
EXPECT_FALSE(dba1->FindVertexOptional(v_address.gid(), true));
|
2018-02-28 17:36:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-02 00:39:22 +08:00
|
|
|
class DistributedDetachDeleteTest : public DistributedGraphDbTest {
|
|
|
|
protected:
|
2018-07-05 16:55:00 +08:00
|
|
|
DistributedDetachDeleteTest() : DistributedGraphDbTest("detach_delete") {}
|
|
|
|
|
2018-03-02 00:39:22 +08:00
|
|
|
storage::VertexAddress w1_a;
|
|
|
|
storage::VertexAddress w1_b;
|
|
|
|
storage::VertexAddress w2_a;
|
|
|
|
|
|
|
|
void SetUp() override {
|
|
|
|
DistributedGraphDbTest::SetUp();
|
|
|
|
w1_a = InsertVertex(worker(1));
|
|
|
|
w1_b = InsertVertex(worker(1));
|
|
|
|
w2_a = InsertVertex(worker(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename TF>
|
|
|
|
void Run(storage::VertexAddress v_address, TF check_func) {
|
|
|
|
for (int i : {0, 1, 2}) {
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba0 = master().Access();
|
|
|
|
auto dba1 = worker(1).Access(dba0->transaction_id());
|
|
|
|
auto dba2 = worker(2).Access(dba0->transaction_id());
|
2018-03-02 00:39:22 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> dba{
|
|
|
|
*dba0, *dba1, *dba2};
|
2019-02-14 15:36:40 +08:00
|
|
|
std::vector<database::GraphDb *> dbs{&master(), &worker(1),
|
2018-07-26 15:08:21 +08:00
|
|
|
&worker(2)};
|
2018-03-02 00:39:22 +08:00
|
|
|
|
|
|
|
auto &accessor = dba[i].get();
|
|
|
|
auto v_accessor = VertexAccessor(v_address, accessor);
|
|
|
|
accessor.DetachRemoveVertex(v_accessor);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
for (auto *db : dbs) {
|
|
|
|
ASSERT_EQ(db->updates_server().Apply(dba[0].get().transaction_id()),
|
2018-03-23 22:21:46 +08:00
|
|
|
distributed::UpdateResult::DONE);
|
2018-03-02 00:39:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
check_func(dba);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DistributedDetachDeleteTest, VertexCycle) {
|
|
|
|
auto e_address = InsertEdge(w1_a, w1_a, "edge");
|
|
|
|
Run(w1_a,
|
|
|
|
[this, e_address](
|
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> &dba) {
|
2018-03-21 17:19:23 +08:00
|
|
|
EXPECT_FALSE(dba[1].get().FindVertexOptional(w1_a.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindEdgeOptional(e_address.gid(), true));
|
2018-03-02 00:39:22 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedDetachDeleteTest, TwoVerticesDifferentWorkers) {
|
|
|
|
auto e_address = InsertEdge(w1_a, w2_a, "edge");
|
|
|
|
|
|
|
|
// Delete from
|
|
|
|
Run(w1_a,
|
|
|
|
[this, e_address](
|
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> &dba) {
|
2018-03-21 17:19:23 +08:00
|
|
|
EXPECT_FALSE(dba[1].get().FindVertexOptional(w1_a.gid(), true));
|
|
|
|
EXPECT_TRUE(dba[2].get().FindVertexOptional(w2_a.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindEdgeOptional(e_address.gid(), true));
|
2018-03-02 00:39:22 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
// Delete to
|
|
|
|
Run(w2_a,
|
|
|
|
[this, e_address](
|
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> &dba) {
|
2018-03-21 17:19:23 +08:00
|
|
|
EXPECT_TRUE(dba[1].get().FindVertexOptional(w1_a.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[2].get().FindVertexOptional(w2_a.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindEdgeOptional(e_address.gid(), true));
|
2018-03-02 00:39:22 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedDetachDeleteTest, TwoVerticesSameWorkers) {
|
|
|
|
auto e_address = InsertEdge(w1_a, w1_b, "edge");
|
|
|
|
|
|
|
|
// Delete from
|
|
|
|
Run(w1_a,
|
|
|
|
[this, e_address](
|
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> &dba) {
|
2018-03-21 17:19:23 +08:00
|
|
|
EXPECT_FALSE(dba[1].get().FindVertexOptional(w1_a.gid(), true));
|
|
|
|
EXPECT_TRUE(dba[1].get().FindVertexOptional(w1_b.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindEdgeOptional(e_address.gid(), true));
|
2018-03-02 00:39:22 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
// Delete to
|
|
|
|
Run(w1_b,
|
|
|
|
[this, e_address](
|
|
|
|
std::vector<std::reference_wrapper<database::GraphDbAccessor>> &dba) {
|
2018-03-21 17:19:23 +08:00
|
|
|
EXPECT_TRUE(dba[1].get().FindVertexOptional(w1_a.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindVertexOptional(w1_b.gid(), true));
|
|
|
|
EXPECT_FALSE(dba[1].get().FindEdgeOptional(e_address.gid(), true));
|
2018-03-02 00:39:22 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-02-20 21:48:36 +08:00
|
|
|
class DistributedEdgeCreateTest : public DistributedGraphDbTest {
|
|
|
|
protected:
|
2018-07-05 16:55:00 +08:00
|
|
|
DistributedEdgeCreateTest() : DistributedGraphDbTest("edge_create") {}
|
|
|
|
|
2018-02-20 21:48:36 +08:00
|
|
|
storage::VertexAddress w1_a;
|
|
|
|
storage::VertexAddress w1_b;
|
|
|
|
storage::VertexAddress w2_a;
|
2018-03-14 17:53:20 +08:00
|
|
|
std::unordered_map<std::string, PropertyValue> props{{"p1", 42},
|
|
|
|
{"p2", true}};
|
2018-02-20 21:48:36 +08:00
|
|
|
storage::EdgeAddress e_ga;
|
|
|
|
|
|
|
|
void SetUp() override {
|
|
|
|
DistributedGraphDbTest::SetUp();
|
|
|
|
w1_a = InsertVertex(worker(1));
|
|
|
|
w1_b = InsertVertex(worker(1));
|
|
|
|
w2_a = InsertVertex(worker(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CreateEdge(database::GraphDb &creator, storage::VertexAddress from_addr,
|
2018-05-30 18:59:22 +08:00
|
|
|
storage::VertexAddress to_addr) {
|
2018-02-26 19:38:37 +08:00
|
|
|
CHECK(from_addr.is_remote() && to_addr.is_remote())
|
|
|
|
<< "Local address given to CreateEdge";
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = creator.Access();
|
|
|
|
auto edge_type = dba->EdgeType("et");
|
|
|
|
VertexAccessor v1{from_addr, *dba};
|
|
|
|
VertexAccessor v2{to_addr, *dba};
|
|
|
|
auto edge = dba->InsertEdge(v1, v2, edge_type);
|
2018-03-14 17:53:20 +08:00
|
|
|
e_ga = edge.GlobalAddress();
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
for (auto &kv : props) edge.PropsSet(dba->Property(kv.first), kv.second);
|
2018-03-14 17:53:20 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
master().updates_server().Apply(dba->transaction_id());
|
|
|
|
worker(1).updates_server().Apply(dba->transaction_id());
|
|
|
|
worker(2).updates_server().Apply(dba->transaction_id());
|
|
|
|
dba->Commit();
|
2018-02-20 21:48:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CheckState(database::GraphDb &db, bool edge_is_local,
|
|
|
|
storage::VertexAddress from_addr,
|
|
|
|
storage::VertexAddress to_addr) {
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = db.Access();
|
2018-02-20 21:48:36 +08:00
|
|
|
|
|
|
|
// Check edge data.
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
EdgeAccessor edge{e_ga, *dba};
|
2018-02-20 21:48:36 +08:00
|
|
|
EXPECT_EQ(edge.address().is_local(), edge_is_local);
|
|
|
|
EXPECT_EQ(edge.GlobalAddress(), e_ga);
|
|
|
|
auto from = edge.from();
|
|
|
|
EXPECT_EQ(from.GlobalAddress(), from_addr);
|
2018-05-04 20:50:30 +08:00
|
|
|
EXPECT_EQ(edge.from_addr().is_local(), from.is_local());
|
2018-02-20 21:48:36 +08:00
|
|
|
auto to = edge.to();
|
|
|
|
EXPECT_EQ(to.GlobalAddress(), to_addr);
|
2018-05-04 20:50:30 +08:00
|
|
|
EXPECT_EQ(edge.to_addr().is_local(), to.is_local());
|
2018-03-14 17:53:20 +08:00
|
|
|
|
|
|
|
EXPECT_EQ(edge.Properties().size(), props.size());
|
|
|
|
for (auto &kv : props) {
|
2018-07-26 15:08:21 +08:00
|
|
|
auto equality = edge.PropsAt(dba->Property(kv.first)) ==
|
2018-03-14 17:53:20 +08:00
|
|
|
query::TypedValue(kv.second);
|
|
|
|
EXPECT_TRUE(equality.IsBool() && equality.ValueBool());
|
|
|
|
}
|
2018-02-20 21:48:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
auto edges = [](auto iterable) {
|
|
|
|
std::vector<EdgeAccessor> res;
|
|
|
|
for (auto edge : iterable) res.emplace_back(edge);
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Check `from` data.
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
VertexAccessor from{from_addr, *dba};
|
2018-02-20 21:48:36 +08:00
|
|
|
ASSERT_EQ(edges(from.out()).size(), 1);
|
|
|
|
EXPECT_EQ(edges(from.out())[0].GlobalAddress(), e_ga);
|
|
|
|
// In case of cycles we have 1 in the `in` edges.
|
|
|
|
EXPECT_EQ(edges(from.in()).size(), from_addr == to_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check `to` data.
|
|
|
|
{
|
2018-07-26 15:08:21 +08:00
|
|
|
VertexAccessor to{to_addr, *dba};
|
2018-02-20 21:48:36 +08:00
|
|
|
// In case of cycles we have 1 in the `out` edges.
|
|
|
|
EXPECT_EQ(edges(to.out()).size(), from_addr == to_addr);
|
|
|
|
ASSERT_EQ(edges(to.in()).size(), 1);
|
|
|
|
EXPECT_EQ(edges(to.in())[0].GlobalAddress(), e_ga);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckAll(storage::VertexAddress from_addr,
|
|
|
|
storage::VertexAddress to_addr) {
|
|
|
|
int edge_worker = from_addr.worker_id();
|
2018-03-14 17:53:20 +08:00
|
|
|
EXPECT_EQ(EdgeCount(master()), edge_worker == 0);
|
|
|
|
EXPECT_EQ(EdgeCount(worker(1)), edge_worker == 1);
|
|
|
|
EXPECT_EQ(EdgeCount(worker(2)), edge_worker == 2);
|
2018-02-20 21:48:36 +08:00
|
|
|
CheckState(master(), edge_worker == 0, from_addr, to_addr);
|
|
|
|
CheckState(worker(1), edge_worker == 1, from_addr, to_addr);
|
|
|
|
CheckState(worker(2), edge_worker == 2, from_addr, to_addr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeCreateTest, LocalRemote) {
|
|
|
|
CreateEdge(worker(1), w1_a, w2_a);
|
|
|
|
CheckAll(w1_a, w2_a);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeCreateTest, RemoteLocal) {
|
|
|
|
CreateEdge(worker(2), w1_a, w2_a);
|
|
|
|
CheckAll(w1_a, w2_a);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeCreateTest, RemoteRemoteDifferentWorkers) {
|
|
|
|
CreateEdge(master(), w1_a, w2_a);
|
|
|
|
CheckAll(w1_a, w2_a);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeCreateTest, RemoteRemoteSameWorkers) {
|
|
|
|
CreateEdge(master(), w1_a, w1_b);
|
|
|
|
CheckAll(w1_a, w1_b);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeCreateTest, RemoteRemoteCycle) {
|
|
|
|
CreateEdge(master(), w1_a, w1_a);
|
|
|
|
CheckAll(w1_a, w1_a);
|
|
|
|
}
|
2018-02-28 21:30:21 +08:00
|
|
|
|
|
|
|
class DistributedEdgeRemoveTest : public DistributedGraphDbTest {
|
|
|
|
protected:
|
2018-07-05 16:55:00 +08:00
|
|
|
DistributedEdgeRemoveTest() : DistributedGraphDbTest("edge_remove") {}
|
|
|
|
|
2018-02-28 21:30:21 +08:00
|
|
|
storage::VertexAddress from_addr;
|
|
|
|
storage::VertexAddress to_addr;
|
|
|
|
storage::EdgeAddress edge_addr;
|
|
|
|
|
|
|
|
void Create(database::GraphDb &from_db, database::GraphDb &to_db) {
|
|
|
|
from_addr = InsertVertex(from_db);
|
|
|
|
to_addr = InsertVertex(to_db);
|
|
|
|
edge_addr = InsertEdge(from_addr, to_addr, "edge_type");
|
|
|
|
}
|
|
|
|
|
|
|
|
void Delete(database::GraphDb &db) {
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = db.Access();
|
|
|
|
EdgeAccessor edge{edge_addr, *dba};
|
|
|
|
dba->RemoveEdge(edge);
|
|
|
|
master().updates_server().Apply(dba->transaction_id());
|
|
|
|
worker(1).updates_server().Apply(dba->transaction_id());
|
|
|
|
worker(2).updates_server().Apply(dba->transaction_id());
|
|
|
|
dba->Commit();
|
2018-02-28 21:30:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename TIterable>
|
|
|
|
auto Size(TIterable iterable) {
|
2019-02-26 15:41:16 +08:00
|
|
|
size_t size = 0;
|
|
|
|
for (auto i = iterable.begin(); i != iterable.end(); ++i) {
|
|
|
|
++size;
|
|
|
|
}
|
|
|
|
return size;
|
2018-02-28 21:30:21 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void CheckCreation() {
|
|
|
|
auto wid = from_addr.worker_id();
|
|
|
|
ASSERT_TRUE(wid >= 0 && wid < 3);
|
|
|
|
ASSERT_EQ(EdgeCount(master()), wid == 0);
|
|
|
|
ASSERT_EQ(EdgeCount(worker(1)), wid == 1);
|
|
|
|
ASSERT_EQ(EdgeCount(worker(2)), wid == 2);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = master().Access();
|
|
|
|
VertexAccessor from{from_addr, *dba};
|
2018-02-28 21:30:21 +08:00
|
|
|
EXPECT_EQ(Size(from.out()), 1);
|
|
|
|
EXPECT_EQ(Size(from.in()), 0);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
VertexAccessor to{to_addr, *dba};
|
2018-02-28 21:30:21 +08:00
|
|
|
EXPECT_EQ(Size(to.out()), 0);
|
|
|
|
EXPECT_EQ(Size(to.in()), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckDeletion() {
|
|
|
|
EXPECT_EQ(EdgeCount(master()), 0);
|
|
|
|
EXPECT_EQ(EdgeCount(worker(1)), 0);
|
|
|
|
EXPECT_EQ(EdgeCount(worker(2)), 0);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
auto dba = master().Access();
|
2018-02-28 21:30:21 +08:00
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
VertexAccessor from{from_addr, *dba};
|
2018-02-28 21:30:21 +08:00
|
|
|
EXPECT_EQ(Size(from.out()), 0);
|
|
|
|
EXPECT_EQ(Size(from.in()), 0);
|
|
|
|
|
2018-07-26 15:08:21 +08:00
|
|
|
VertexAccessor to{to_addr, *dba};
|
2018-02-28 21:30:21 +08:00
|
|
|
EXPECT_EQ(Size(to.out()), 0);
|
|
|
|
EXPECT_EQ(Size(to.in()), 0);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeRemoveTest, DifferentVertexOwnersRemoteDelete) {
|
|
|
|
Create(worker(1), worker(2));
|
|
|
|
CheckCreation();
|
|
|
|
Delete(master());
|
|
|
|
CheckDeletion();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeRemoveTest, DifferentVertexOwnersFromDelete) {
|
|
|
|
Create(worker(1), worker(2));
|
|
|
|
CheckCreation();
|
|
|
|
Delete(worker(1));
|
|
|
|
CheckDeletion();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeRemoveTest, DifferentVertexOwnersToDelete) {
|
|
|
|
Create(worker(1), worker(2));
|
|
|
|
CheckCreation();
|
|
|
|
Delete(worker(2));
|
|
|
|
CheckDeletion();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DistributedEdgeRemoveTest, SameVertexOwnersRemoteDelete) {
|
|
|
|
Create(worker(1), worker(1));
|
|
|
|
CheckCreation();
|
|
|
|
Delete(worker(2));
|
|
|
|
CheckDeletion();
|
|
|
|
}
|