#include "gtest/gtest.h" #include "database/graph_db.hpp" #include "database/graph_db_datatypes.hpp" #include "database/indexes/label_property_index.hpp" #include "database/dbms.hpp" #include "mvcc_gc_common.hpp" class LabelPropertyIndexComplexTest : public ::testing::Test { protected: virtual void SetUp() { auto accessor = dbms.active(); label = accessor->label("label"); property = accessor->property("property"); label2 = accessor->label("label2"); property2 = accessor->property("property2"); key = new LabelPropertyIndex::Key(label, property); EXPECT_EQ(index.CreateIndex(*key), true); index.IndexFinishedBuilding(*key); t = engine.Begin(); vlist = new mvcc::VersionList(*t); engine.Advance(t->id_); vertex = vlist->find(*t); ASSERT_NE(vertex, nullptr); vertex->labels_.push_back(label); vertex->properties_.set(property, 0); EXPECT_EQ(index.Count(*key), 0); } virtual void TearDown() { delete key; delete vlist; } public: Dbms dbms; LabelPropertyIndex index; LabelPropertyIndex::Key *key; tx::Engine engine; tx::Transaction *t{nullptr}; mvcc::VersionList *vlist; Vertex *vertex; GraphDbTypes::Label label; GraphDbTypes::Property property; GraphDbTypes::Label label2; GraphDbTypes::Property property2; }; TEST(LabelPropertyIndex, CreateIndex) { Dbms dbms; auto accessor = dbms.active(); LabelPropertyIndex::Key key(accessor->label("test"), accessor->property("test2")); LabelPropertyIndex index; EXPECT_EQ(index.CreateIndex(key), true); EXPECT_EQ(index.CreateIndex(key), false); } TEST(LabelPropertyIndex, IndexExistance) { Dbms dbms; auto accessor = dbms.active(); LabelPropertyIndex::Key key(accessor->label("test"), accessor->property("test2")); LabelPropertyIndex index; EXPECT_EQ(index.CreateIndex(key), true); // Index doesn't exist - and can't be used untill it's been notified as built. EXPECT_EQ(index.IndexExists(key), false); index.IndexFinishedBuilding(key); EXPECT_EQ(index.IndexExists(key), true); } TEST(LabelPropertyIndex, Count) { Dbms dbms; auto accessor = dbms.active(); auto label = accessor->label("label"); auto property = accessor->property("property"); LabelPropertyIndex::Key key(label, property); LabelPropertyIndex index; ::testing::FLAGS_gtest_death_test_style = "threadsafe"; EXPECT_DEATH(index.Count(key), "Index doesn't exist."); EXPECT_EQ(index.CreateIndex(key), true); EXPECT_DEATH(index.Count(key), "Index not yet ready."); index.IndexFinishedBuilding(key); EXPECT_EQ(index.Count(key), 0); } // Add on label+property to index. TEST_F(LabelPropertyIndexComplexTest, UpdateOnLabelPropertyTrue) { index.UpdateOnLabelProperty(vlist, vertex); EXPECT_EQ(index.Count(*key), 1); } // Try adding on label+property but fail because labels are clear. TEST_F(LabelPropertyIndexComplexTest, UpdateOnLabelPropertyFalse) { vertex->labels_.clear(); index.UpdateOnLabelProperty(vlist, vertex); EXPECT_EQ(index.Count(*key), 0); } // Add on label to index. TEST_F(LabelPropertyIndexComplexTest, UpdateOnLabelTrue) { index.UpdateOnLabel(label, vlist, vertex); EXPECT_EQ(index.Count(*key), 1); } // Try adding on label but fail because label is wrong. TEST_F(LabelPropertyIndexComplexTest, UpdateOnLabelFalse) { index.UpdateOnLabel(label2, vlist, vertex); EXPECT_EQ(index.Count(*key), 0); } // Add on property to index. TEST_F(LabelPropertyIndexComplexTest, UpdateOnPropertyTrue) { index.UpdateOnProperty(property, vlist, vertex); EXPECT_EQ(index.Count(*key), 1); } // Try adding on property but fail because property is wrong. TEST_F(LabelPropertyIndexComplexTest, UpdateOnPropertyFalse) { index.UpdateOnProperty(property2, vlist, vertex); EXPECT_EQ(index.Count(*key), 0); } // Test index does it insert everything uniquely TEST_F(LabelPropertyIndexComplexTest, UniqueInsert) { index.UpdateOnLabelProperty(vlist, vertex); index.UpdateOnLabelProperty(vlist, vertex); EXPECT_EQ(index.Count(*key), 1); } // Check if index filters duplicates. TEST_F(LabelPropertyIndexComplexTest, UniqueFilter) { index.UpdateOnLabelProperty(vlist, vertex); t->Commit(); auto t2 = engine.Begin(); auto vertex2 = vlist->update(*t2); t2->Commit(); index.UpdateOnLabelProperty(vlist, vertex2); EXPECT_EQ(index.Count(*key), 2); auto t3 = engine.Begin(); auto iter = index.GetVlists(*key, *t3, false); EXPECT_EQ(std::distance(iter.begin(), iter.end()), 1); t3->Commit(); } // Remove label and check if index vertex is not returned now. TEST_F(LabelPropertyIndexComplexTest, RemoveLabel) { index.UpdateOnLabelProperty(vlist, vertex); auto iter1 = index.GetVlists(*key, *t, false); EXPECT_EQ(std::distance(iter1.begin(), iter1.end()), 1); vertex->labels_.clear(); auto iter2 = index.GetVlists(*key, *t, false); EXPECT_EQ(std::distance(iter2.begin(), iter2.end()), 0); } // Remove property and check if vertex is not returned now. TEST_F(LabelPropertyIndexComplexTest, RemoveProperty) { index.UpdateOnLabelProperty(vlist, vertex); auto iter1 = index.GetVlists(*key, *t, false); EXPECT_EQ(std::distance(iter1.begin(), iter1.end()), 1); vertex->properties_.clear(); auto iter2 = index.GetVlists(*key, *t, false); EXPECT_EQ(std::distance(iter2.begin(), iter2.end()), 0); } // Refresh with a vertex that looses its labels and properties. TEST_F(LabelPropertyIndexComplexTest, Refresh) { index.UpdateOnLabelProperty(vlist, vertex); t->Commit(); EXPECT_EQ(index.Count(*key), 1); vertex->labels_.clear(); vertex->properties_.clear(); index.Refresh(GcSnapshot(engine, nullptr), engine); auto iter = index.GetVlists(*key, *t, false); EXPECT_EQ(std::distance(iter.begin(), iter.end()), 0); } int main(int argc, char **argv) { ::testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); }