2017-02-16 22:47:55 +08:00
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "database/graph_db.hpp"
|
|
|
|
#include "database/graph_db_accessor.hpp"
|
2017-02-18 18:54:37 +08:00
|
|
|
#include "dbms/dbms.hpp"
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
#include "storage/edge_accessor.hpp"
|
2017-02-18 18:54:37 +08:00
|
|
|
#include "storage/vertex_accessor.hpp"
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
size_t CountVertices(GraphDbAccessor &db_accessor) {
|
|
|
|
size_t r_val = 0;
|
2017-03-08 18:13:49 +08:00
|
|
|
for ([[gnu::unused]] auto va : db_accessor.vertices()) r_val++;
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
return r_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t CountEdges(GraphDbAccessor &db_accessor) {
|
|
|
|
size_t r_val = 0;
|
2017-03-08 18:13:49 +08:00
|
|
|
for ([[gnu::unused]] auto va : db_accessor.edges()) r_val++;
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
return r_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, DbmsCreateDefault) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor = dbms.active();
|
|
|
|
EXPECT_EQ(accessor->name(), "default");
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, InsertVertex) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountVertices(*accessor), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
accessor->insert_vertex();
|
2017-03-22 20:40:50 +08:00
|
|
|
accessor->advance_command();
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountVertices(*accessor), 1);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
accessor->insert_vertex();
|
2017-03-22 20:40:50 +08:00
|
|
|
accessor->advance_command();
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountVertices(*accessor), 2);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, RemoveVertexSameTransaction) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountVertices(*accessor), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
auto va1 = accessor->insert_vertex();
|
2017-03-22 20:40:50 +08:00
|
|
|
accessor->advance_command();
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountVertices(*accessor), 1);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_TRUE(accessor->remove_vertex(va1));
|
|
|
|
EXPECT_EQ(CountVertices(*accessor), 1);
|
|
|
|
accessor->advance_command();
|
|
|
|
EXPECT_EQ(CountVertices(*accessor), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, RemoveVertexDifferentTransaction) {
|
|
|
|
Dbms dbms;
|
|
|
|
|
|
|
|
// first transaction creates a vertex
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor1 = dbms.active();
|
|
|
|
accessor1->insert_vertex();
|
|
|
|
accessor1->commit();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// second transaction checks that it sees it, and deletes it
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor2 = dbms.active();
|
|
|
|
EXPECT_EQ(CountVertices(*accessor2), 1);
|
|
|
|
for (auto vertex_accessor : accessor2->vertices())
|
|
|
|
accessor2->remove_vertex(vertex_accessor);
|
|
|
|
accessor2->commit();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// third transaction checks that it does not see the vertex
|
2017-03-14 20:26:48 +08:00
|
|
|
auto accessor3 = dbms.active();
|
|
|
|
EXPECT_EQ(CountVertices(*accessor3), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, InsertEdge) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto dba = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
auto va1 = dba->insert_vertex();
|
|
|
|
auto va2 = dba->insert_vertex();
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(va1.in_degree(), 0);
|
|
|
|
EXPECT_EQ(va1.out_degree(), 0);
|
|
|
|
EXPECT_EQ(va2.in_degree(), 0);
|
|
|
|
EXPECT_EQ(va2.out_degree(), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// setup (v1) - [:likes] -> (v2)
|
2017-03-14 20:26:48 +08:00
|
|
|
dba->insert_edge(va1, va2, dba->edge_type("likes"));
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountEdges(*dba), 1);
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(va1.out().begin()->to(), va2);
|
|
|
|
EXPECT_EQ(va2.in().begin()->from(), va1);
|
|
|
|
EXPECT_EQ(va1.in_degree(), 0);
|
|
|
|
EXPECT_EQ(va1.out_degree(), 1);
|
|
|
|
EXPECT_EQ(va2.in_degree(), 1);
|
|
|
|
EXPECT_EQ(va2.out_degree(), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// setup (v1) - [:likes] -> (v2) <- [:hates] - (v3)
|
2017-03-14 20:26:48 +08:00
|
|
|
auto va3 = dba->insert_vertex();
|
|
|
|
dba->insert_edge(va3, va2, dba->edge_type("hates"));
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(CountEdges(*dba), 2);
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(va3.out().begin()->to(), va2);
|
|
|
|
EXPECT_EQ(va1.in_degree(), 0);
|
|
|
|
EXPECT_EQ(va1.out_degree(), 1);
|
|
|
|
EXPECT_EQ(va2.in_degree(), 2);
|
|
|
|
EXPECT_EQ(va2.out_degree(), 0);
|
|
|
|
EXPECT_EQ(va3.in_degree(), 0);
|
|
|
|
EXPECT_EQ(va3.out_degree(), 1);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, RemoveEdge) {
|
|
|
|
Dbms dbms;
|
2017-03-22 20:40:50 +08:00
|
|
|
auto dba = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// setup (v1) - [:likes] -> (v2) <- [:hates] - (v3)
|
2017-03-22 20:40:50 +08:00
|
|
|
auto va1 = dba->insert_vertex();
|
|
|
|
auto va2 = dba->insert_vertex();
|
|
|
|
auto va3 = dba->insert_vertex();
|
|
|
|
dba->insert_edge(va1, va2, dba->edge_type("likes"));
|
|
|
|
dba->insert_edge(va3, va2, dba->edge_type("hates"));
|
|
|
|
dba->advance_command();
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 2);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// remove all [:hates] edges
|
2017-03-22 20:40:50 +08:00
|
|
|
EXPECT_EQ(CountEdges(*dba), 2);
|
|
|
|
for (auto edge : dba->edges())
|
|
|
|
if (edge.edge_type() == dba->edge_type("hates")) dba->remove_edge(edge);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// current state: (v1) - [:likes] -> (v2), (v3)
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 1);
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 3);
|
|
|
|
for (auto edge : dba->edges()) {
|
|
|
|
EXPECT_EQ(edge.edge_type(), dba->edge_type("likes"));
|
2017-02-16 22:47:55 +08:00
|
|
|
auto v1 = edge.from();
|
|
|
|
auto v2 = edge.to();
|
|
|
|
|
|
|
|
// ensure correct connectivity for all the vertices
|
2017-03-22 20:40:50 +08:00
|
|
|
for (auto vertex : dba->vertices()) {
|
2017-02-16 22:47:55 +08:00
|
|
|
if (vertex == v1) {
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(vertex.in_degree(), 0);
|
|
|
|
EXPECT_EQ(vertex.out_degree(), 1);
|
2017-02-16 22:47:55 +08:00
|
|
|
} else if (vertex == v2) {
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(vertex.in_degree(), 1);
|
|
|
|
EXPECT_EQ(vertex.out_degree(), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
} else {
|
2017-03-03 20:59:38 +08:00
|
|
|
EXPECT_EQ(vertex.in_degree(), 0);
|
|
|
|
EXPECT_EQ(vertex.out_degree(), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, DetachRemoveVertex) {
|
|
|
|
Dbms dbms;
|
2017-03-22 20:40:50 +08:00
|
|
|
auto dba = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// setup (v1) - [:likes] -> (v2) <- [:hates] - (v3)
|
2017-03-22 20:40:50 +08:00
|
|
|
auto va1 = dba->insert_vertex();
|
|
|
|
auto va2 = dba->insert_vertex();
|
|
|
|
auto va3 = dba->insert_vertex();
|
|
|
|
dba->insert_edge(va1, va2, dba->edge_type("likes"));
|
|
|
|
dba->insert_edge(va1, va3, dba->edge_type("likes"));
|
|
|
|
dba->advance_command();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// ensure that plain remove does NOT work
|
2017-03-22 20:40:50 +08:00
|
|
|
EXPECT_EQ(CountVertices(*dba), 3);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 2);
|
|
|
|
EXPECT_FALSE(dba->remove_vertex(va1));
|
|
|
|
EXPECT_FALSE(dba->remove_vertex(va2));
|
|
|
|
EXPECT_FALSE(dba->remove_vertex(va3));
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 3);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 2);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// make a new transaction because at the moment deletions
|
|
|
|
// in the same transaction are not visible
|
|
|
|
// DETACH REMOVE V3
|
|
|
|
// new situation: (v1) - [:likes] -> (v2)
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->detach_remove_vertex(va3);
|
|
|
|
dba->advance_command();
|
|
|
|
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 2);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 1);
|
|
|
|
for (auto va : dba->vertices()) EXPECT_FALSE(dba->remove_vertex(va));
|
|
|
|
|
|
|
|
dba->advance_command();
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 2);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 1);
|
|
|
|
|
|
|
|
for (auto va : dba->vertices()) {
|
|
|
|
EXPECT_FALSE(dba->remove_vertex(va));
|
|
|
|
dba->detach_remove_vertex(va);
|
2017-02-16 22:47:55 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 1);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// remove the last vertex, it has no connections
|
|
|
|
// so that should work
|
2017-03-22 20:40:50 +08:00
|
|
|
for (auto va : dba->vertices()) EXPECT_TRUE(dba->remove_vertex(va));
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-22 20:40:50 +08:00
|
|
|
dba->advance_command();
|
|
|
|
EXPECT_EQ(CountVertices(*dba), 0);
|
|
|
|
EXPECT_EQ(CountEdges(*dba), 0);
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, Labels) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto dba1 = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
GraphDb::Label label_friend = dba1->label("friend");
|
|
|
|
EXPECT_EQ(label_friend, dba1->label("friend"));
|
|
|
|
EXPECT_NE(label_friend, dba1->label("friend2"));
|
|
|
|
EXPECT_EQ(dba1->label_name(label_friend), "friend");
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// test that getting labels through a different accessor works
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(label_friend, dbms.active()->label("friend"));
|
|
|
|
EXPECT_NE(label_friend, dbms.active()->label("friend2"));
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, EdgeTypes) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto dba1 = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
GraphDb::EdgeType edge_type = dba1->edge_type("likes");
|
|
|
|
EXPECT_EQ(edge_type, dba1->edge_type("likes"));
|
|
|
|
EXPECT_NE(edge_type, dba1->edge_type("hates"));
|
|
|
|
EXPECT_EQ(dba1->edge_type_name(edge_type), "likes");
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// test that getting labels through a different accessor works
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(edge_type, dbms.active()->edge_type("likes"));
|
|
|
|
EXPECT_NE(edge_type, dbms.active()->edge_type("hates"));
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GraphDbAccessorTest, Properties) {
|
|
|
|
Dbms dbms;
|
2017-03-14 20:26:48 +08:00
|
|
|
auto dba1 = dbms.active();
|
2017-02-16 22:47:55 +08:00
|
|
|
|
2017-03-14 20:26:48 +08:00
|
|
|
GraphDb::EdgeType prop = dba1->property("name");
|
|
|
|
EXPECT_EQ(prop, dba1->property("name"));
|
|
|
|
EXPECT_NE(prop, dba1->property("surname"));
|
|
|
|
EXPECT_EQ(dba1->property_name(prop), "name");
|
2017-02-16 22:47:55 +08:00
|
|
|
|
|
|
|
// test that getting labels through a different accessor works
|
2017-03-14 20:26:48 +08:00
|
|
|
EXPECT_EQ(prop, dbms.active()->property("name"));
|
|
|
|
EXPECT_NE(prop, dbms.active()->property("surname"));
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
2017-02-16 22:47:55 +08:00
|
|
|
}
|