2023-06-29 17:44:55 +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.
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "disk_test_utils.hpp"
|
2019-09-11 22:10:53 +08:00
|
|
|
#include "query_plan_common.hpp"
|
|
|
|
|
2022-09-02 23:12:07 +08:00
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2019-09-11 22:10:53 +08:00
|
|
|
#include "query/frontend/semantic/symbol_table.hpp"
|
|
|
|
#include "query/plan/operator.hpp"
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "storage/v2/disk/storage.hpp"
|
|
|
|
#include "storage/v2/inmemory/storage.hpp"
|
2019-09-11 22:10:53 +08:00
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
template <typename StorageType>
|
|
|
|
class QueryPlan : public testing::Test {
|
|
|
|
public:
|
|
|
|
const std::string testSuite = "query_plan_v2_create_set_remove_delete";
|
|
|
|
memgraph::storage::Config config = disk_test_utils::GenerateOnDiskConfig(testSuite);
|
|
|
|
std::unique_ptr<memgraph::storage::Storage> db = std::make_unique<StorageType>(config);
|
|
|
|
|
|
|
|
void TearDown() override {
|
|
|
|
if (std::is_same<StorageType, memgraph::storage::DiskStorage>::value) {
|
|
|
|
disk_test_utils::RemoveRocksDbDirs(testSuite);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
using StorageTypes = ::testing::Types<memgraph::storage::InMemoryStorage, memgraph::storage::DiskStorage>;
|
|
|
|
TYPED_TEST_CASE(QueryPlan, StorageTypes);
|
|
|
|
|
|
|
|
TYPED_TEST(QueryPlan, CreateNodeWithAttributes) {
|
|
|
|
auto dba = this->db->Access();
|
2019-09-11 22:10:53 +08:00
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
auto label = memgraph::storage::LabelId::FromInt(42);
|
|
|
|
auto property = memgraph::storage::PropertyId::FromInt(1);
|
2019-09-11 22:10:53 +08:00
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::AstStorage ast;
|
|
|
|
memgraph::query::SymbolTable symbol_table;
|
2019-09-11 22:10:53 +08:00
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::NodeCreationInfo node;
|
2019-09-11 22:10:53 +08:00
|
|
|
node.symbol = symbol_table.CreateSymbol("n", true);
|
|
|
|
node.labels.emplace_back(label);
|
2022-02-22 20:33:45 +08:00
|
|
|
std::get<std::vector<std::pair<memgraph::storage::PropertyId, Expression *>>>(node.properties)
|
2021-09-09 18:39:13 +08:00
|
|
|
.emplace_back(property, ast.Create<PrimitiveLiteral>(42));
|
2019-09-11 22:10:53 +08:00
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::CreateNode create_node(nullptr, node);
|
2023-06-29 17:44:55 +08:00
|
|
|
DbAccessor execution_dba(dba.get());
|
2019-09-11 22:10:53 +08:00
|
|
|
auto context = MakeContext(ast, symbol_table, &execution_dba);
|
|
|
|
Frame frame(context.symbol_table.max_position());
|
2022-02-22 20:33:45 +08:00
|
|
|
auto cursor = create_node.MakeCursor(memgraph::utils::NewDeleteResource());
|
2019-09-11 22:10:53 +08:00
|
|
|
int count = 0;
|
|
|
|
while (cursor->Pull(frame, context)) {
|
|
|
|
++count;
|
|
|
|
const auto &node_value = frame[node.symbol];
|
|
|
|
EXPECT_EQ(node_value.type(), TypedValue::Type::Vertex);
|
|
|
|
const auto &v = node_value.ValueVertex();
|
2022-02-22 20:33:45 +08:00
|
|
|
EXPECT_TRUE(*v.HasLabel(memgraph::storage::View::NEW, label));
|
|
|
|
EXPECT_EQ(v.GetProperty(memgraph::storage::View::NEW, property)->ValueInt(), 42);
|
2023-06-29 17:44:55 +08:00
|
|
|
dba->PrefetchInEdges(v.impl_);
|
|
|
|
dba->PrefetchOutEdges(v.impl_);
|
2023-09-08 23:12:25 +08:00
|
|
|
EXPECT_EQ(CountIterable(v.InEdges(memgraph::storage::View::NEW)->edges), 0);
|
|
|
|
EXPECT_EQ(CountIterable(v.OutEdges(memgraph::storage::View::NEW)->edges), 0);
|
2019-09-11 22:10:53 +08:00
|
|
|
// Invokes LOG(FATAL) instead of erroring out.
|
2022-02-22 20:33:45 +08:00
|
|
|
// EXPECT_TRUE(v.HasLabel(label, memgraph::storage::View::OLD).IsError());
|
2019-09-11 22:10:53 +08:00
|
|
|
}
|
|
|
|
EXPECT_EQ(count, 1);
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(QueryPlan, ScanAllEmpty) {
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::AstStorage ast;
|
|
|
|
memgraph::query::SymbolTable symbol_table;
|
2023-06-29 17:44:55 +08:00
|
|
|
auto dba = this->db->Access();
|
|
|
|
DbAccessor execution_dba(dba.get());
|
2019-09-11 22:10:53 +08:00
|
|
|
auto node_symbol = symbol_table.CreateSymbol("n", true);
|
|
|
|
{
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::ScanAll scan_all(nullptr, node_symbol, memgraph::storage::View::OLD);
|
2019-09-11 22:10:53 +08:00
|
|
|
auto context = MakeContext(ast, symbol_table, &execution_dba);
|
|
|
|
Frame frame(context.symbol_table.max_position());
|
2022-02-22 20:33:45 +08:00
|
|
|
auto cursor = scan_all.MakeCursor(memgraph::utils::NewDeleteResource());
|
2019-09-11 22:10:53 +08:00
|
|
|
int count = 0;
|
|
|
|
while (cursor->Pull(frame, context)) ++count;
|
|
|
|
EXPECT_EQ(count, 0);
|
|
|
|
}
|
|
|
|
{
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::ScanAll scan_all(nullptr, node_symbol, memgraph::storage::View::NEW);
|
2019-09-11 22:10:53 +08:00
|
|
|
auto context = MakeContext(ast, symbol_table, &execution_dba);
|
|
|
|
Frame frame(context.symbol_table.max_position());
|
2022-02-22 20:33:45 +08:00
|
|
|
auto cursor = scan_all.MakeCursor(memgraph::utils::NewDeleteResource());
|
2019-09-11 22:10:53 +08:00
|
|
|
int count = 0;
|
|
|
|
while (cursor->Pull(frame, context)) ++count;
|
|
|
|
EXPECT_EQ(count, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(QueryPlan, ScanAll) {
|
2019-09-11 22:10:53 +08:00
|
|
|
{
|
2023-06-29 17:44:55 +08:00
|
|
|
auto dba = this->db->Access();
|
|
|
|
for (int i = 0; i < 42; ++i) dba->CreateVertex();
|
|
|
|
EXPECT_FALSE(dba->Commit().HasError());
|
2019-09-11 22:10:53 +08:00
|
|
|
}
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::AstStorage ast;
|
|
|
|
memgraph::query::SymbolTable symbol_table;
|
2023-06-29 17:44:55 +08:00
|
|
|
auto dba = this->db->Access();
|
|
|
|
DbAccessor execution_dba(dba.get());
|
2019-09-11 22:10:53 +08:00
|
|
|
auto node_symbol = symbol_table.CreateSymbol("n", true);
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::ScanAll scan_all(nullptr, node_symbol);
|
2019-09-11 22:10:53 +08:00
|
|
|
auto context = MakeContext(ast, symbol_table, &execution_dba);
|
|
|
|
Frame frame(context.symbol_table.max_position());
|
2022-02-22 20:33:45 +08:00
|
|
|
auto cursor = scan_all.MakeCursor(memgraph::utils::NewDeleteResource());
|
2019-09-11 22:10:53 +08:00
|
|
|
int count = 0;
|
|
|
|
while (cursor->Pull(frame, context)) ++count;
|
|
|
|
EXPECT_EQ(count, 42);
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(QueryPlan, ScanAllByLabel) {
|
|
|
|
auto label = this->db->NameToLabel("label");
|
|
|
|
ASSERT_FALSE(this->db->CreateIndex(label).HasError());
|
2019-09-11 22:10:53 +08:00
|
|
|
{
|
2023-06-29 17:44:55 +08:00
|
|
|
auto dba = this->db->Access();
|
2019-09-11 22:10:53 +08:00
|
|
|
// Add some unlabeled vertices
|
2023-06-29 17:44:55 +08:00
|
|
|
for (int i = 0; i < 12; ++i) dba->CreateVertex();
|
2019-09-11 22:10:53 +08:00
|
|
|
// Add labeled vertices
|
|
|
|
for (int i = 0; i < 42; ++i) {
|
2023-06-29 17:44:55 +08:00
|
|
|
auto v = dba->CreateVertex();
|
2019-09-11 22:10:53 +08:00
|
|
|
ASSERT_TRUE(v.AddLabel(label).HasValue());
|
|
|
|
}
|
2023-06-29 17:44:55 +08:00
|
|
|
EXPECT_FALSE(dba->Commit().HasError());
|
2019-09-11 22:10:53 +08:00
|
|
|
}
|
2023-06-29 17:44:55 +08:00
|
|
|
auto dba = this->db->Access();
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::AstStorage ast;
|
|
|
|
memgraph::query::SymbolTable symbol_table;
|
2019-09-11 22:10:53 +08:00
|
|
|
auto node_symbol = symbol_table.CreateSymbol("n", true);
|
2023-06-29 17:44:55 +08:00
|
|
|
DbAccessor execution_dba(dba.get());
|
2022-02-22 20:33:45 +08:00
|
|
|
memgraph::query::plan::ScanAllByLabel scan_all(nullptr, node_symbol, label);
|
2019-09-11 22:10:53 +08:00
|
|
|
auto context = MakeContext(ast, symbol_table, &execution_dba);
|
|
|
|
Frame frame(context.symbol_table.max_position());
|
2022-02-22 20:33:45 +08:00
|
|
|
auto cursor = scan_all.MakeCursor(memgraph::utils::NewDeleteResource());
|
2019-09-11 22:10:53 +08:00
|
|
|
int count = 0;
|
|
|
|
while (cursor->Pull(frame, context)) ++count;
|
|
|
|
EXPECT_EQ(count, 42);
|
|
|
|
}
|