// Copyright 2023 Memgraph Ltd. // // 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. #include "bfs_common.hpp" using namespace memgraph::query; using namespace memgraph::query::plan; class SingleNodeDb : public Database { public: SingleNodeDb() : db_() {} memgraph::storage::Storage::Accessor Access() override { return db_.Access(); } std::unique_ptr MakeBfsOperator(Symbol source_sym, Symbol sink_sym, Symbol edge_sym, EdgeAtom::Direction direction, const std::vector &edge_types, const std::shared_ptr &input, bool existing_node, Expression *lower_bound, Expression *upper_bound, const ExpansionLambda &filter_lambda) override { return std::make_unique(input, source_sym, sink_sym, edge_sym, EdgeAtom::Type::BREADTH_FIRST, direction, edge_types, false, lower_bound, upper_bound, existing_node, filter_lambda, std::nullopt, std::nullopt); } std::pair, std::vector> BuildGraph( memgraph::query::DbAccessor *dba, const std::vector &vertex_locations, const std::vector> &edges) override { std::vector vertex_addr; std::vector edge_addr; for (size_t id = 0; id < vertex_locations.size(); ++id) { auto vertex = dba->InsertVertex(); MG_ASSERT( vertex.SetProperty(dba->NameToProperty("id"), memgraph::storage::PropertyValue(static_cast(id))) .HasValue()); vertex_addr.push_back(vertex); } for (auto e : edges) { int u, v; std::string type; std::tie(u, v, type) = e; auto &from = vertex_addr[u]; auto &to = vertex_addr[v]; auto edge = dba->InsertEdge(&from, &to, dba->NameToEdgeType(type)); MG_ASSERT(edge->SetProperty(dba->NameToProperty("from"), memgraph::storage::PropertyValue(u)).HasValue()); MG_ASSERT(edge->SetProperty(dba->NameToProperty("to"), memgraph::storage::PropertyValue(v)).HasValue()); edge_addr.push_back(*edge); } return std::make_pair(vertex_addr, edge_addr); } protected: memgraph::storage::Storage db_; }; class SingleNodeBfsTest : public ::testing::TestWithParam< std::tuple, bool, FilterLambdaType>> { public: static void SetUpTestCase() { db_ = std::make_unique(); } static void TearDownTestCase() { db_ = nullptr; } protected: static std::unique_ptr db_; }; TEST_P(SingleNodeBfsTest, All) { int lower_bound; int upper_bound; EdgeAtom::Direction direction; std::vector edge_types; bool known_sink; FilterLambdaType filter_lambda_type; std::tie(lower_bound, upper_bound, direction, edge_types, known_sink, filter_lambda_type) = GetParam(); BfsTest(db_.get(), lower_bound, upper_bound, direction, edge_types, known_sink, filter_lambda_type); } std::unique_ptr SingleNodeBfsTest::db_{nullptr}; INSTANTIATE_TEST_SUITE_P(DirectionAndExpansionDepth, SingleNodeBfsTest, testing::Combine(testing::Range(-1, kVertexCount), testing::Range(-1, kVertexCount), testing::Values(EdgeAtom::Direction::OUT, EdgeAtom::Direction::IN, EdgeAtom::Direction::BOTH), testing::Values(std::vector{}), testing::Bool(), testing::Values(FilterLambdaType::NONE))); INSTANTIATE_TEST_SUITE_P( EdgeType, SingleNodeBfsTest, testing::Combine(testing::Values(-1), testing::Values(-1), testing::Values(EdgeAtom::Direction::OUT, EdgeAtom::Direction::IN, EdgeAtom::Direction::BOTH), testing::Values(std::vector{}, std::vector{"a"}, std::vector{"b"}, std::vector{"a", "b"}), testing::Bool(), testing::Values(FilterLambdaType::NONE))); INSTANTIATE_TEST_SUITE_P(FilterLambda, SingleNodeBfsTest, testing::Combine(testing::Values(-1), testing::Values(-1), testing::Values(EdgeAtom::Direction::OUT, EdgeAtom::Direction::IN, EdgeAtom::Direction::BOTH), testing::Values(std::vector{}), testing::Bool(), testing::Values(FilterLambdaType::NONE, FilterLambdaType::USE_FRAME, FilterLambdaType::USE_FRAME_NULL, FilterLambdaType::USE_CTX, FilterLambdaType::ERROR)));