// // Copyright 2017 Memgraph // Created by Florijan Stamenkovic on 14.03.17. // #include #include #include #include #include "gmock/gmock.h" #include "gtest/gtest.h" #include "communication/result_stream_faker.hpp" #include "query/context.hpp" #include "query/exceptions.hpp" #include "query/plan/operator.hpp" #include "query_plan_common.hpp" using namespace query; using namespace query::plan; TEST(QueryPlan, Skip) { database::GraphDb db; auto dba = db.Access(); AstStorage storage; SymbolTable symbol_table; auto n = MakeScanAll(storage, symbol_table, "n1"); auto skip = std::make_shared(n.op_, LITERAL(2)); EXPECT_EQ(0, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(0, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(0, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(1, PullAll(skip, *dba, symbol_table)); for (int i = 0; i < 10; ++i) dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(11, PullAll(skip, *dba, symbol_table)); } TEST(QueryPlan, Limit) { database::GraphDb db; auto dba = db.Access(); AstStorage storage; SymbolTable symbol_table; auto n = MakeScanAll(storage, symbol_table, "n1"); auto skip = std::make_shared(n.op_, LITERAL(2)); EXPECT_EQ(0, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(1, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(2, PullAll(skip, *dba, symbol_table)); dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(2, PullAll(skip, *dba, symbol_table)); for (int i = 0; i < 10; ++i) dba->InsertVertex(); dba->AdvanceCommand(); EXPECT_EQ(2, PullAll(skip, *dba, symbol_table)); } TEST(QueryPlan, CreateLimit) { // CREATE (n), (m) // MATCH (n) CREATE (m) LIMIT 1 // in the end we need to have 3 vertices in the db database::GraphDb db; auto dba = db.Access(); dba->InsertVertex(); dba->InsertVertex(); dba->AdvanceCommand(); AstStorage storage; SymbolTable symbol_table; auto n = MakeScanAll(storage, symbol_table, "n1"); auto m = NODE("m"); symbol_table[*m->identifier_] = symbol_table.CreateSymbol("m", true); auto c = std::make_shared(n.op_, m); auto skip = std::make_shared(c, LITERAL(1)); EXPECT_EQ(1, PullAll(skip, *dba, symbol_table)); dba->AdvanceCommand(); EXPECT_EQ(3, CountIterable(dba->Vertices(false))); } TEST(QueryPlan, OrderBy) { database::GraphDb db; auto dba_ptr = db.Access(); auto &dba = *dba_ptr; AstStorage storage; SymbolTable symbol_table; auto prop = dba.Property("prop"); // contains a series of tests // each test defines the ordering a vector of values in the desired order auto Null = PropertyValue::Null; std::vector>> orderable{ {Ordering::ASC, {0, 0, 0.5, 1, 2, 12.6, 42, Null, Null}}, {Ordering::ASC, {false, false, true, true, Null, Null}}, {Ordering::ASC, {"A", "B", "a", "a", "aa", "ab", "aba", Null, Null}}, {Ordering::DESC, {Null, Null, 33, 33, 32.5, 32, 2.2, 2.1, 0}}, {Ordering::DESC, {Null, true, false}}, {Ordering::DESC, {Null, "zorro", "borro"}}}; for (const auto &order_value_pair : orderable) { const auto &values = order_value_pair.second; // empty database for (auto &vertex : dba.Vertices(false)) dba.DetachRemoveVertex(vertex); dba.AdvanceCommand(); ASSERT_EQ(0, CountIterable(dba.Vertices(false))); // take some effort to shuffle the values // because we are testing that something not ordered gets ordered // and need to take care it does not happen by accident std::vector shuffled(values.begin(), values.end()); auto order_equal = [&values, &shuffled]() { return std::equal(values.begin(), values.end(), shuffled.begin(), TypedValue::BoolEqual{}); }; for (int i = 0; i < 50 && order_equal(); ++i) { std::random_shuffle(shuffled.begin(), shuffled.end()); } ASSERT_FALSE(order_equal()); // create the vertices for (const auto &value : shuffled) dba.InsertVertex().PropsSet(prop, value); dba.AdvanceCommand(); // order by and collect results auto n = MakeScanAll(storage, symbol_table, "n"); auto n_p = PROPERTY_LOOKUP("n", prop); symbol_table[*n_p->expression_] = n.sym_; auto order_by = std::make_shared( n.op_, std::vector{{order_value_pair.first, n_p}}, std::vector{n.sym_}); auto n_p_ne = NEXPR("n.p", n_p); symbol_table[*n_p_ne] = symbol_table.CreateSymbol("n.p", true); auto produce = MakeProduce(order_by, n_p_ne); auto results = CollectProduce(produce.get(), symbol_table, dba); ASSERT_EQ(values.size(), results.size()); for (int j = 0; j < results.size(); ++j) EXPECT_TRUE(TypedValue::BoolEqual{}(results[j][0], values[j])); } } TEST(QueryPlan, OrderByMultiple) { database::GraphDb db; auto dba_ptr = db.Access(); auto &dba = *dba_ptr; AstStorage storage; SymbolTable symbol_table; auto p1 = dba.Property("p1"); auto p2 = dba.Property("p2"); // create a bunch of vertices that in two properties // have all the variations (with repetition) of N values. // ensure that those vertices are not created in the // "right" sequence, but randomized const int N = 20; std::vector> prop_values; for (int i = 0; i < N * N; ++i) prop_values.emplace_back(i % N, i / N); std::random_shuffle(prop_values.begin(), prop_values.end()); for (const auto &pair : prop_values) { auto v = dba.InsertVertex(); v.PropsSet(p1, pair.first); v.PropsSet(p2, pair.second); } dba.AdvanceCommand(); // order by and collect results auto n = MakeScanAll(storage, symbol_table, "n"); auto n_p1 = PROPERTY_LOOKUP("n", p1); symbol_table[*n_p1->expression_] = n.sym_; auto n_p2 = PROPERTY_LOOKUP("n", p2); symbol_table[*n_p2->expression_] = n.sym_; // order the results so we get // (p1: 0, p2: N-1) // (p1: 0, p2: N-2) // ... // (p1: N-1, p2:0) auto order_by = std::make_shared(n.op_, std::vector{ {Ordering::ASC, n_p1}, {Ordering::DESC, n_p2}, }, std::vector{n.sym_}); auto n_p1_ne = NEXPR("n.p1", n_p1); symbol_table[*n_p1_ne] = symbol_table.CreateSymbol("n.p1", true); auto n_p2_ne = NEXPR("n.p2", n_p2); symbol_table[*n_p2_ne] = symbol_table.CreateSymbol("n.p2", true); auto produce = MakeProduce(order_by, n_p1_ne, n_p2_ne); auto results = CollectProduce(produce.get(), symbol_table, dba); ASSERT_EQ(N * N, results.size()); for (int j = 0; j < N * N; ++j) { ASSERT_EQ(results[j][0].type(), TypedValue::Type::Int); EXPECT_EQ(results[j][0].Value(), j / N); ASSERT_EQ(results[j][1].type(), TypedValue::Type::Int); EXPECT_EQ(results[j][1].Value(), N - 1 - j % N); } } TEST(QueryPlan, OrderByExceptions) { database::GraphDb db; auto dba_ptr = db.Access(); auto &dba = *dba_ptr; AstStorage storage; SymbolTable symbol_table; auto prop = dba.Property("prop"); // a vector of pairs of typed values that should result // in an exception when trying to order on them std::vector> exception_pairs{ {42, true}, {42, "bla"}, {42, std::vector{42}}, {true, "bla"}, {true, std::vector{true}}, {"bla", std::vector{"bla"}}, // illegal comparisons of same-type values {std::vector{42}, std::vector{42}}}; for (const auto &pair : exception_pairs) { // empty database for (auto &vertex : dba.Vertices(false)) dba.DetachRemoveVertex(vertex); dba.AdvanceCommand(); ASSERT_EQ(0, CountIterable(dba.Vertices(false))); // make two vertices, and set values dba.InsertVertex().PropsSet(prop, pair.first); dba.InsertVertex().PropsSet(prop, pair.second); dba.AdvanceCommand(); ASSERT_EQ(2, CountIterable(dba.Vertices(false))); for (const auto &va : dba.Vertices(false)) ASSERT_NE(va.PropsAt(prop).type(), PropertyValue::Type::Null); // order by and expect an exception auto n = MakeScanAll(storage, symbol_table, "n"); auto n_p = PROPERTY_LOOKUP("n", prop); symbol_table[*n_p->expression_] = n.sym_; auto order_by = std::make_shared( n.op_, std::vector{{Ordering::ASC, n_p}}, std::vector{}); EXPECT_THROW(PullAll(order_by, dba, symbol_table), QueryRuntimeException); } }