2023-03-31 21:24:02 +08:00
|
|
|
// Copyright 2023 Memgraph Ltd.
|
2021-10-26 14:53:56 +08:00
|
|
|
//
|
|
|
|
// Use of this software is governed by the Business Source License
|
|
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
|
|
//
|
|
|
|
// As of the Change Date specified in that file, in accordance with
|
|
|
|
// the Business Source License, use of this software will be governed
|
|
|
|
// by the Apache License, Version 2.0, included in the file
|
|
|
|
// licenses/APL.txt.
|
|
|
|
|
2017-06-01 20:23:01 +08:00
|
|
|
#include <algorithm>
|
2021-09-09 18:39:13 +08:00
|
|
|
#include <variant>
|
2017-06-01 20:23:01 +08:00
|
|
|
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "query/frontend/semantic/symbol_generator.hpp"
|
|
|
|
#include "query/frontend/semantic/symbol_table.hpp"
|
|
|
|
#include "query/plan/planner.hpp"
|
|
|
|
#include "utils/algorithm.hpp"
|
|
|
|
|
|
|
|
#include "query_plan_common.hpp"
|
|
|
|
|
2019-11-21 02:36:56 +08:00
|
|
|
#include "formatters.hpp"
|
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
using namespace memgraph::query::plan;
|
|
|
|
using memgraph::query::AstStorage;
|
|
|
|
using Type = memgraph::query::EdgeAtom::Type;
|
|
|
|
using Direction = memgraph::query::EdgeAtom::Direction;
|
2017-06-01 20:23:01 +08:00
|
|
|
|
2019-11-21 02:36:56 +08:00
|
|
|
// Functions for printing resulting rows from a query.
|
|
|
|
template <class TAccessor>
|
|
|
|
std::string ToString(const std::vector<TypedValue> &row, const TAccessor &acc) {
|
|
|
|
std::ostringstream os;
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::utils::PrintIterable(os, row, ", ", [&](auto &stream, const auto &item) { stream << ToString(item, acc); });
|
2019-11-21 02:36:56 +08:00
|
|
|
return os.str();
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
2019-11-21 02:36:56 +08:00
|
|
|
template <class TAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
std::string ToString(const std::vector<std::vector<TypedValue>> &rows, const TAccessor &acc) {
|
2019-11-21 02:36:56 +08:00
|
|
|
std::ostringstream os;
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::utils::PrintIterable(os, rows, "\n",
|
|
|
|
[&](auto &stream, const auto &item) { stream << ToString(item, acc); });
|
2019-11-21 02:36:56 +08:00
|
|
|
return os.str();
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2019-11-21 02:36:56 +08:00
|
|
|
template <class TAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
void AssertRows(const std::vector<std::vector<TypedValue>> &datum, std::vector<std::vector<TypedValue>> expected,
|
2019-11-21 02:36:56 +08:00
|
|
|
const TAccessor &acc) {
|
2017-06-01 20:23:01 +08:00
|
|
|
auto row_equal = [](const auto &row1, const auto &row2) {
|
|
|
|
if (row1.size() != row2.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
TypedValue::BoolEqual value_eq;
|
|
|
|
auto row1_it = row1.begin();
|
2021-02-18 22:32:43 +08:00
|
|
|
for (auto row2_it = row2.begin(); row2_it != row2.end(); ++row1_it, ++row2_it) {
|
2017-06-01 20:23:01 +08:00
|
|
|
if (!value_eq(*row1_it, *row2_it)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
2021-02-18 22:32:43 +08:00
|
|
|
ASSERT_TRUE(std::is_permutation(datum.begin(), datum.end(), expected.begin(), expected.end(), row_equal))
|
2017-06-01 20:23:01 +08:00
|
|
|
<< "Actual rows:" << std::endl
|
2019-11-21 02:36:56 +08:00
|
|
|
<< ToString(datum, acc) << std::endl
|
2017-06-01 20:23:01 +08:00
|
|
|
<< "Expected rows:" << std::endl
|
2019-11-21 02:36:56 +08:00
|
|
|
<< ToString(expected, acc);
|
2017-06-01 20:23:01 +08:00
|
|
|
};
|
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
void CheckPlansProduce(size_t expected_plan_count, memgraph::query::CypherQuery *query, AstStorage &storage,
|
|
|
|
memgraph::query::DbAccessor *dba,
|
2021-02-18 22:32:43 +08:00
|
|
|
std::function<void(const std::vector<std::vector<TypedValue>> &)> check) {
|
2022-02-22 20:33:45 +08:00
|
|
|
auto symbol_table = memgraph::query::MakeSymbolTable(query);
|
2021-02-18 22:32:43 +08:00
|
|
|
auto planning_context = MakePlanningContext(&storage, &symbol_table, query, dba);
|
2018-10-10 21:19:34 +08:00
|
|
|
auto query_parts = CollectQueryParts(symbol_table, storage, query);
|
2017-11-29 20:55:02 +08:00
|
|
|
EXPECT_TRUE(query_parts.query_parts.size() > 0);
|
2023-03-31 21:24:02 +08:00
|
|
|
auto plans = MakeLogicalPlanForSingleQuery<VariableStartPlanner>(query_parts, &planning_context);
|
2017-06-01 20:23:01 +08:00
|
|
|
EXPECT_EQ(std::distance(plans.begin(), plans.end()), expected_plan_count);
|
|
|
|
for (const auto &plan : plans) {
|
|
|
|
auto *produce = dynamic_cast<Produce *>(plan.get());
|
|
|
|
ASSERT_TRUE(produce);
|
2019-11-21 02:36:56 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, dba);
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
auto results = CollectProduce(*produce, &context);
|
2017-06-01 20:23:01 +08:00
|
|
|
check(results);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, MatchReturn) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-06-01 20:23:01 +08:00
|
|
|
// Make a graph (v1) -[:r]-> (v2)
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r")).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-06-01 20:23:01 +08:00
|
|
|
// Test MATCH (n) -[r]-> (m) RETURN n
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m"))), RETURN("n")));
|
2017-06-01 20:23:01 +08:00
|
|
|
// We have 2 nodes `n` and `m` from which we could start, so expect 2 plans.
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(2, query, storage, &dba, [&](const auto &results) {
|
|
|
|
// We expect to produce only a single (v1) node.
|
2022-02-22 20:33:45 +08:00
|
|
|
AssertRows(results, {{TypedValue(memgraph::query::VertexAccessor(v1))}}, dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, MatchTripletPatternReturn) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-06-01 20:23:01 +08:00
|
|
|
// Make a graph (v1) -[:r]-> (v2) -[:r]-> (v3)
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
auto v3 = dba.InsertVertex();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r")).HasValue());
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r")).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-06-01 20:23:01 +08:00
|
|
|
{
|
|
|
|
// Test `MATCH (n) -[r]-> (m) -[e]-> (l) RETURN n`
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(
|
2021-02-18 22:32:43 +08:00
|
|
|
MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m"), EDGE("e", Direction::OUT), NODE("l"))),
|
2017-11-29 20:55:02 +08:00
|
|
|
RETURN("n")));
|
2017-06-01 20:23:01 +08:00
|
|
|
// We have 3 nodes: `n`, `m` and `l` from which we could start.
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(3, query, storage, &dba, [&](const auto &results) {
|
|
|
|
// We expect to produce only a single (v1) node.
|
2022-02-22 20:33:45 +08:00
|
|
|
AssertRows(results, {{TypedValue(memgraph::query::VertexAccessor(v1))}}, dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
{
|
|
|
|
// Equivalent to `MATCH (n) -[r]-> (m), (m) -[e]-> (l) RETURN n`.
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m")),
|
|
|
|
PATTERN(NODE("m"), EDGE("e", Direction::OUT), NODE("l"))),
|
|
|
|
RETURN("n")));
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(3, query, storage, &dba, [&](const auto &results) {
|
2022-02-22 20:33:45 +08:00
|
|
|
AssertRows(results, {{TypedValue(memgraph::query::VertexAccessor(v1))}}, dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, MatchOptionalMatchReturn) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-06-01 20:23:01 +08:00
|
|
|
// Make a graph (v1) -[:r]-> (v2) -[:r]-> (v3)
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
auto v3 = dba.InsertVertex();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r")).HasValue());
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r")).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-06-01 20:23:01 +08:00
|
|
|
// Test MATCH (n) -[r]-> (m) OPTIONAL MATCH (m) -[e]-> (l) RETURN n, l
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m"))),
|
|
|
|
OPTIONAL_MATCH(PATTERN(NODE("m"), EDGE("e", Direction::OUT), NODE("l"))), RETURN("n", "l")));
|
2017-06-01 20:23:01 +08:00
|
|
|
// We have 2 nodes `n` and `m` from which we could start the MATCH, and 2
|
|
|
|
// nodes for OPTIONAL MATCH. This should produce 2 * 2 plans.
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(4, query, storage, &dba, [&](const auto &results) {
|
|
|
|
// We expect to produce 2 rows:
|
|
|
|
// * (v1), (v3)
|
|
|
|
// * (v2), null
|
|
|
|
AssertRows(results,
|
2022-02-22 20:33:45 +08:00
|
|
|
{{TypedValue(memgraph::query::VertexAccessor(v1)), TypedValue(memgraph::query::VertexAccessor(v3))},
|
|
|
|
{TypedValue(memgraph::query::VertexAccessor(v2)), TypedValue()}},
|
2019-11-22 01:25:25 +08:00
|
|
|
dba);
|
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, MatchOptionalMatchMergeReturn) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-06-01 20:23:01 +08:00
|
|
|
// Graph (v1) -[:r]-> (v2)
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::VertexAccessor v1(dba.InsertVertex());
|
|
|
|
memgraph::query::VertexAccessor v2(dba.InsertVertex());
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
auto r_type_name = "r";
|
2019-11-22 01:25:25 +08:00
|
|
|
auto r_type = dba.NameToEdgeType(r_type_name);
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, r_type).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-06-01 20:23:01 +08:00
|
|
|
// Test MATCH (n) -[r]-> (m) OPTIONAL MATCH (m) -[e]-> (l)
|
|
|
|
// MERGE (u) -[q:r]-> (v) RETURN n, m, l, u, v
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m"))),
|
|
|
|
OPTIONAL_MATCH(PATTERN(NODE("m"), EDGE("e", Direction::OUT), NODE("l"))),
|
|
|
|
MERGE(PATTERN(NODE("u"), EDGE("q", Direction::OUT, {r_type_name}), NODE("v"))),
|
|
|
|
RETURN("n", "m", "l", "u", "v")));
|
2017-06-01 20:23:01 +08:00
|
|
|
// Since MATCH, OPTIONAL MATCH and MERGE each have 2 nodes from which we can
|
|
|
|
// start, we generate 2 * 2 * 2 plans.
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(8, query, storage, &dba, [&](const auto &results) {
|
|
|
|
// We expect to produce a single row: (v1), (v2), null, (v1), (v2)
|
2021-02-18 22:32:43 +08:00
|
|
|
AssertRows(results, {{TypedValue(v1), TypedValue(v2), TypedValue(), TypedValue(v1), TypedValue(v2)}}, dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, MatchWithMatchReturn) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-06-01 20:23:01 +08:00
|
|
|
// Graph (v1) -[:r]-> (v2)
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::VertexAccessor v1(dba.InsertVertex());
|
|
|
|
memgraph::query::VertexAccessor v2(dba.InsertVertex());
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r")).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-06-01 20:23:01 +08:00
|
|
|
// Test MATCH (n) -[r]-> (m) WITH n MATCH (m) -[r]-> (l) RETURN n, m, l
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), EDGE("r", Direction::OUT), NODE("m"))), WITH("n"),
|
|
|
|
MATCH(PATTERN(NODE("m"), EDGE("r", Direction::OUT), NODE("l"))), RETURN("n", "m", "l")));
|
2017-06-01 20:23:01 +08:00
|
|
|
// We can start from 2 nodes in each match. Since WITH separates query parts,
|
|
|
|
// we expect to get 2 plans for each, which totals 2 * 2.
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(4, query, storage, &dba, [&](const auto &results) {
|
|
|
|
// We expect to produce a single row: (v1), (v1), (v2)
|
2021-02-18 22:32:43 +08:00
|
|
|
AssertRows(results, {{TypedValue(v1), TypedValue(v1), TypedValue(v2)}}, dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
});
|
2017-06-01 20:23:01 +08:00
|
|
|
}
|
|
|
|
|
2017-08-21 15:41:26 +08:00
|
|
|
TEST(TestVariableStartPlanner, MatchVariableExpand) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2017-08-21 15:41:26 +08:00
|
|
|
// Graph (v1) -[:r1]-> (v2) -[:r2]-> (v3)
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
auto v3 = dba.InsertVertex();
|
2019-11-22 01:25:25 +08:00
|
|
|
auto r1 = *dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1"));
|
|
|
|
auto r2 = *dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r2"));
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-08-21 15:41:26 +08:00
|
|
|
// Test MATCH (n) -[r*]-> (m) RETURN r
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2018-10-18 17:16:32 +08:00
|
|
|
auto edge = EDGE_VARIABLE("r", Type::DEPTH_FIRST, Direction::OUT);
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), edge, NODE("m"))), RETURN("r")));
|
2017-08-21 15:41:26 +08:00
|
|
|
// We expect to get a single column with the following rows:
|
2019-06-06 19:16:37 +08:00
|
|
|
TypedValue r1_list(std::vector<TypedValue>{TypedValue(r1)}); // [r1]
|
|
|
|
TypedValue r2_list(std::vector<TypedValue>{TypedValue(r2)}); // [r2]
|
|
|
|
// [r1, r2]
|
2021-02-18 22:32:43 +08:00
|
|
|
TypedValue r1_r2_list(std::vector<TypedValue>{TypedValue(r1), TypedValue(r2)});
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(2, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results, {{r1_list}, {r2_list}, {r1_r2_list}}, dba);
|
|
|
|
});
|
2017-08-21 15:41:26 +08:00
|
|
|
}
|
|
|
|
|
2017-08-22 22:24:40 +08:00
|
|
|
TEST(TestVariableStartPlanner, MatchVariableExpandReferenceNode) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
auto id = dba.NameToProperty("id");
|
2017-08-22 22:24:40 +08:00
|
|
|
// Graph (v1 {id:1}) -[:r1]-> (v2 {id: 2}) -[:r2]-> (v3 {id: 3})
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v1.SetProperty(id, memgraph::storage::PropertyValue(1)).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v2 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v2.SetProperty(id, memgraph::storage::PropertyValue(2)).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v3 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v3.SetProperty(id, memgraph::storage::PropertyValue(3)).HasValue());
|
2019-11-22 01:25:25 +08:00
|
|
|
auto r1 = *dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1"));
|
|
|
|
auto r2 = *dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r2"));
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-08-22 22:24:40 +08:00
|
|
|
// Test MATCH (n) -[r*..n.id]-> (m) RETURN r
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2018-10-18 17:16:32 +08:00
|
|
|
auto edge = EDGE_VARIABLE("r", Type::DEPTH_FIRST, Direction::OUT);
|
2017-08-22 22:24:40 +08:00
|
|
|
edge->upper_bound_ = PROPERTY_LOOKUP("n", id);
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), edge, NODE("m"))), RETURN("r")));
|
2017-08-22 22:24:40 +08:00
|
|
|
// We expect to get a single column with the following rows:
|
2019-06-06 19:16:37 +08:00
|
|
|
// [r1] (v1 -[*..1]-> v2)
|
|
|
|
TypedValue r1_list(std::vector<TypedValue>{TypedValue(r1)});
|
|
|
|
// [r2] (v2 -[*..2]-> v3)
|
|
|
|
TypedValue r2_list(std::vector<TypedValue>{TypedValue(r2)});
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(2, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results, {{r1_list}, {r2_list}}, dba);
|
|
|
|
});
|
2017-08-22 22:24:40 +08:00
|
|
|
}
|
|
|
|
|
2017-09-04 20:09:24 +08:00
|
|
|
TEST(TestVariableStartPlanner, MatchVariableExpandBoth) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
auto id = dba.NameToProperty("id");
|
2017-09-04 20:09:24 +08:00
|
|
|
// Graph (v1 {id:1}) -[:r1]-> (v2) -[:r2]-> (v3)
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v1.SetProperty(id, memgraph::storage::PropertyValue(1)).HasValue());
|
2019-04-15 17:36:43 +08:00
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
auto v3 = dba.InsertVertex();
|
2019-11-22 01:25:25 +08:00
|
|
|
auto r1 = *dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1"));
|
|
|
|
auto r2 = *dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r2"));
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-09-04 20:09:24 +08:00
|
|
|
// Test MATCH (n {id:1}) -[r*]- (m) RETURN r
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2018-10-18 17:16:32 +08:00
|
|
|
auto edge = EDGE_VARIABLE("r", Type::DEPTH_FIRST, Direction::BOTH);
|
2017-09-04 20:09:24 +08:00
|
|
|
auto node_n = NODE("n");
|
2021-09-09 18:39:13 +08:00
|
|
|
std::get<0>(node_n->properties_)[storage.GetPropertyIx("id")] = LITERAL(1);
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(node_n, edge, NODE("m"))), RETURN("r")));
|
2017-09-04 20:09:24 +08:00
|
|
|
// We expect to get a single column with the following rows:
|
2019-06-06 19:16:37 +08:00
|
|
|
TypedValue r1_list(std::vector<TypedValue>{TypedValue(r1)}); // [r1]
|
|
|
|
// [r1, r2]
|
2021-02-18 22:32:43 +08:00
|
|
|
TypedValue r1_r2_list(std::vector<TypedValue>{TypedValue(r1), TypedValue(r2)});
|
2019-11-22 01:25:25 +08:00
|
|
|
CheckPlansProduce(2, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results, {{r1_list}, {r1_r2_list}}, dba);
|
|
|
|
});
|
2017-09-04 20:09:24 +08:00
|
|
|
}
|
|
|
|
|
2017-08-22 22:24:40 +08:00
|
|
|
TEST(TestVariableStartPlanner, MatchBfs) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::storage::Storage db;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto storage_dba = db.Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
2019-11-22 01:25:25 +08:00
|
|
|
auto id = dba.NameToProperty("id");
|
2017-08-22 22:24:40 +08:00
|
|
|
// Graph (v1 {id:1}) -[:r1]-> (v2 {id: 2}) -[:r2]-> (v3 {id: 3})
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v1 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v1.SetProperty(id, memgraph::storage::PropertyValue(1)).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v2 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v2.SetProperty(id, memgraph::storage::PropertyValue(2)).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v3 = dba.InsertVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
ASSERT_TRUE(v3.SetProperty(id, memgraph::storage::PropertyValue(3)).HasValue());
|
2019-11-22 01:25:25 +08:00
|
|
|
auto r1 = *dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1"));
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("r2")).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-10-05 17:25:52 +08:00
|
|
|
// Test MATCH (n) -[r *bfs..10](r, n | n.id <> 3)]-> (m) RETURN r
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2022-02-22 20:33:45 +08:00
|
|
|
auto *bfs = storage.Create<memgraph::query::EdgeAtom>(IDENT("r"), EdgeAtom::Type::BREADTH_FIRST, Direction::OUT,
|
|
|
|
std::vector<memgraph::query::EdgeTypeIx>{});
|
2018-02-08 19:57:12 +08:00
|
|
|
bfs->filter_lambda_.inner_edge = IDENT("r");
|
|
|
|
bfs->filter_lambda_.inner_node = IDENT("n");
|
|
|
|
bfs->filter_lambda_.expression = NEQ(PROPERTY_LOOKUP("n", id), LITERAL(3));
|
2017-09-27 20:57:41 +08:00
|
|
|
bfs->upper_bound_ = LITERAL(10);
|
2021-02-18 22:32:43 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"), bfs, NODE("m"))), RETURN("r")));
|
2017-08-22 22:24:40 +08:00
|
|
|
// We expect to get a single column with the following rows:
|
2019-06-06 19:16:37 +08:00
|
|
|
TypedValue r1_list(std::vector<TypedValue>{TypedValue(r1)}); // [r1]
|
2021-02-18 22:32:43 +08:00
|
|
|
CheckPlansProduce(2, query, storage, &dba, [&](const auto &results) { AssertRows(results, {{r1_list}}, dba); });
|
2017-08-22 22:24:40 +08:00
|
|
|
}
|
|
|
|
|
2023-03-31 21:24:02 +08:00
|
|
|
TEST(TestVariableStartPlanner, TestBasicSubquery) {
|
|
|
|
memgraph::storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
|
|
|
AstStorage storage;
|
|
|
|
|
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
|
|
|
|
|
|
|
auto *subquery = SINGLE_QUERY(MATCH(PATTERN(NODE("m"))), RETURN("m"));
|
|
|
|
|
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), CALL_SUBQUERY(subquery), RETURN("n", "m")));
|
|
|
|
|
|
|
|
CheckPlansProduce(1, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results,
|
|
|
|
{{TypedValue(v1), TypedValue(v1)},
|
|
|
|
{TypedValue(v1), TypedValue(v2)},
|
|
|
|
{TypedValue(v2), TypedValue(v1)},
|
|
|
|
{TypedValue(v2), TypedValue(v2)}},
|
|
|
|
dba);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, TestBasicSubqueryWithMatching) {
|
|
|
|
memgraph::storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
|
|
|
AstStorage storage;
|
|
|
|
|
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1")).HasValue());
|
|
|
|
|
|
|
|
dba.AdvanceCommand();
|
|
|
|
|
|
|
|
auto *subquery =
|
|
|
|
SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))), RETURN("m2"));
|
|
|
|
|
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("m1"), EDGE("r1", EdgeAtom::Direction::OUT), NODE("n1"))),
|
|
|
|
CALL_SUBQUERY(subquery), RETURN("m1", "m2")));
|
|
|
|
|
|
|
|
CheckPlansProduce(4, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results, {{TypedValue(v1), TypedValue(v1)}}, dba);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, TestSubqueryWithUnion) {
|
|
|
|
memgraph::storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
|
|
|
AstStorage storage;
|
|
|
|
auto id = dba.NameToProperty("id");
|
|
|
|
|
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
ASSERT_TRUE(v1.SetProperty(id, memgraph::storage::PropertyValue(1)).HasValue());
|
|
|
|
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
ASSERT_TRUE(v2.SetProperty(id, memgraph::storage::PropertyValue(2)).HasValue());
|
|
|
|
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1")).HasValue());
|
|
|
|
|
|
|
|
dba.AdvanceCommand();
|
|
|
|
|
|
|
|
auto *subquery =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))), RETURN("n2")),
|
|
|
|
UNION_ALL(SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))),
|
|
|
|
RETURN("n2"))));
|
|
|
|
|
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("m1"), EDGE("r1", EdgeAtom::Direction::OUT), NODE("n1"))),
|
|
|
|
CALL_SUBQUERY(subquery), RETURN("m1", "n2")));
|
|
|
|
|
|
|
|
CheckPlansProduce(8, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results, {{TypedValue(v1), TypedValue(v2)}, {TypedValue(v1), TypedValue(v2)}}, dba);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(TestVariableStartPlanner, TestSubqueryWithTripleUnion) {
|
|
|
|
memgraph::storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
memgraph::query::DbAccessor dba(&storage_dba);
|
|
|
|
AstStorage storage;
|
|
|
|
auto id = dba.NameToProperty("id");
|
|
|
|
|
|
|
|
auto v1 = dba.InsertVertex();
|
|
|
|
ASSERT_TRUE(v1.SetProperty(id, memgraph::storage::PropertyValue(1)).HasValue());
|
|
|
|
|
|
|
|
auto v2 = dba.InsertVertex();
|
|
|
|
ASSERT_TRUE(v2.SetProperty(id, memgraph::storage::PropertyValue(2)).HasValue());
|
|
|
|
|
|
|
|
ASSERT_TRUE(dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("r1")).HasValue());
|
|
|
|
|
|
|
|
dba.AdvanceCommand();
|
|
|
|
|
|
|
|
auto *subquery =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))), RETURN("n2")),
|
|
|
|
UNION_ALL(SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))),
|
|
|
|
RETURN("n2"))),
|
|
|
|
UNION_ALL(SINGLE_QUERY(MATCH(PATTERN(NODE("m2"), EDGE("r2", EdgeAtom::Direction::OUT), NODE("n2"))),
|
|
|
|
RETURN("n2"))));
|
|
|
|
|
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("m1"), EDGE("r1", EdgeAtom::Direction::OUT), NODE("n1"))),
|
|
|
|
CALL_SUBQUERY(subquery), RETURN("m1", "n2")));
|
|
|
|
|
|
|
|
CheckPlansProduce(16, query, storage, &dba, [&](const auto &results) {
|
|
|
|
AssertRows(results,
|
|
|
|
{{TypedValue(v1), TypedValue(v2)}, {TypedValue(v1), TypedValue(v2)}, {TypedValue(v1), TypedValue(v2)}},
|
|
|
|
dba);
|
|
|
|
});
|
|
|
|
}
|
2017-06-01 20:23:01 +08:00
|
|
|
} // namespace
|