Implement execution of pattern comprehension.

Simple query works with one result works.
When multiple rows in return - doesn't work.
When after pattern comprehension, returing property - doesn't work.
This commit is contained in:
Aidar Samerkhanov 2024-03-17 02:00:18 +04:00
parent 24f8a14b43
commit c6cd2134a0
10 changed files with 173 additions and 51 deletions

View File

@ -1105,8 +1105,15 @@ class ExpressionEvaluator : public ExpressionVisitor<TypedValue> {
} }
} }
TypedValue Visit(PatternComprehension & /*pattern_comprehension*/) override { TypedValue Visit(PatternComprehension &pattern_comprehension) override {
throw utils::NotYetImplemented("Expression evaluator can not handle pattern comprehension."); TypedValue &frame_pattern_comprehension_value = frame_->at(symbol_table_->at(pattern_comprehension));
if (!frame_pattern_comprehension_value.IsList()) [[unlikely]] {
throw QueryRuntimeException(
"Unexpected behavior: Pattern Comprehension expected a list, got {}. Please report the problem on GitHub "
"issues",
frame_pattern_comprehension_value.type());
}
return frame_pattern_comprehension_value;
} }
private: private:

View File

@ -138,6 +138,7 @@ extern const Event EvaluatePatternFilterOperator;
extern const Event ApplyOperator; extern const Event ApplyOperator;
extern const Event IndexedJoinOperator; extern const Event IndexedJoinOperator;
extern const Event HashJoinOperator; extern const Event HashJoinOperator;
extern const Event RollUpApplyOperator;
} // namespace memgraph::metrics } // namespace memgraph::metrics
namespace memgraph::query::plan { namespace memgraph::query::plan {
@ -5741,16 +5742,15 @@ UniqueCursorPtr HashJoin::MakeCursor(utils::MemoryResource *mem) const {
return MakeUniqueCursorPtr<HashJoinCursor>(mem, *this, mem); return MakeUniqueCursorPtr<HashJoinCursor>(mem, *this, mem);
} }
RollUpApply::RollUpApply(const std::shared_ptr<LogicalOperator> &input, RollUpApply::RollUpApply(const std::shared_ptr<LogicalOperator> &&input,
std::shared_ptr<LogicalOperator> &&second_branch) std::shared_ptr<LogicalOperator> &&list_collection_branch,
: input_(input), list_collection_branch_(second_branch) {} const std::vector<Symbol> &list_collection_symbols, Symbol result_symbol)
: input_(input),
list_collection_branch_(list_collection_branch),
list_collection_symbols_(list_collection_symbols),
result_symbol_(result_symbol) {}
std::vector<Symbol> RollUpApply::OutputSymbols(const SymbolTable & /*symbol_table*/) const { std::vector<Symbol> RollUpApply::ModifiedSymbols(const SymbolTable &table) const { return {result_symbol_}; }
std::vector<Symbol> symbols;
return symbols;
}
std::vector<Symbol> RollUpApply::ModifiedSymbols(const SymbolTable &table) const { return OutputSymbols(table); }
bool RollUpApply::Accept(HierarchicalLogicalOperatorVisitor &visitor) { bool RollUpApply::Accept(HierarchicalLogicalOperatorVisitor &visitor) {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
@ -5762,4 +5762,68 @@ bool RollUpApply::Accept(HierarchicalLogicalOperatorVisitor &visitor) {
return visitor.PostVisit(*this); return visitor.PostVisit(*this);
} }
namespace {
class RollUpApplyCursor : public Cursor {
public:
RollUpApplyCursor(const RollUpApply &self, utils::MemoryResource *mem)
: self_(self),
input_cursor_(self.input_->MakeCursor(mem)),
list_collection_cursor_(self_.list_collection_branch_->MakeCursor(mem)) {
MG_ASSERT(input_cursor_ != nullptr, "RollUpApplyCursor: Missing left operator cursor.");
MG_ASSERT(list_collection_cursor_ != nullptr, "RollUpApplyCursor: Missing right operator cursor.");
MG_ASSERT(self_.list_collection_symbols_.size() == 1U, "Expected a single list collection symbol.");
}
bool Pull(Frame &frame, ExecutionContext &context) override {
OOMExceptionEnabler oom_exception;
SCOPED_PROFILE_OP_BY_REF(self_);
auto clear_frame_change_collector = [&context](const auto &symbols) {
for (const auto &symbol : symbols) {
if (context.frame_change_collector && context.frame_change_collector->IsKeyTracked(symbol.name())) {
context.frame_change_collector->ResetTrackingValue(symbol.name());
}
}
};
TypedValue result(std::vector<TypedValue>(), context.evaluation_context.memory);
if (input_cursor_->Pull(frame, context)) {
while (list_collection_cursor_->Pull(frame, context)) {
// collect values from the list collection branch
for (const auto &output_symbol : self_.list_collection_symbols_) {
result.ValueList().emplace_back(frame[output_symbol]);
}
}
clear_frame_change_collector(self_.list_collection_symbols_);
frame[self_.result_symbol_] = result;
} else {
return false;
}
return true;
}
void Shutdown() override {
input_cursor_->Shutdown();
list_collection_cursor_->Shutdown();
}
void Reset() override {
input_cursor_->Reset();
list_collection_cursor_->Reset();
}
private:
const RollUpApply &self_;
const UniqueCursorPtr input_cursor_;
const UniqueCursorPtr list_collection_cursor_;
};
} // namespace
UniqueCursorPtr RollUpApply::MakeCursor(utils::MemoryResource *mem) const {
memgraph::metrics::IncrementCounter(memgraph::metrics::RollUpApplyOperator);
return MakeUniqueCursorPtr<RollUpApplyCursor>(mem, *this, mem);
}
} // namespace memgraph::query::plan } // namespace memgraph::query::plan

View File

@ -2682,28 +2682,30 @@ class RollUpApply : public memgraph::query::plan::LogicalOperator {
const utils::TypeInfo &GetTypeInfo() const override { return kType; } const utils::TypeInfo &GetTypeInfo() const override { return kType; }
RollUpApply() = default; RollUpApply() = default;
RollUpApply(const std::shared_ptr<LogicalOperator> &input, std::shared_ptr<LogicalOperator> &&second_branch); RollUpApply(const std::shared_ptr<LogicalOperator> &&input, std::shared_ptr<LogicalOperator> &&list_collection_branch,
const std::vector<Symbol> &list_collection_symbols, Symbol result_symbol);
bool HasSingleInput() const override { return false; } bool HasSingleInput() const override { return false; }
std::shared_ptr<LogicalOperator> input() const override { return input_; } std::shared_ptr<LogicalOperator> input() const override { return input_; }
void set_input(std::shared_ptr<LogicalOperator> input) override { input_ = input; } void set_input(std::shared_ptr<LogicalOperator> input) override { input_ = input; }
bool Accept(HierarchicalLogicalOperatorVisitor &visitor) override; bool Accept(HierarchicalLogicalOperatorVisitor &visitor) override;
UniqueCursorPtr MakeCursor(utils::MemoryResource *) const override { UniqueCursorPtr MakeCursor(utils::MemoryResource *) const override;
throw utils::NotYetImplemented("Execution of Pattern comprehension is currently unsupported.");
}
std::vector<Symbol> OutputSymbols(const SymbolTable &) const override;
std::vector<Symbol> ModifiedSymbols(const SymbolTable &) const override; std::vector<Symbol> ModifiedSymbols(const SymbolTable &) const override;
std::unique_ptr<LogicalOperator> Clone(AstStorage *storage) const override { std::unique_ptr<LogicalOperator> Clone(AstStorage *storage) const override {
auto object = std::make_unique<RollUpApply>(); auto object = std::make_unique<RollUpApply>();
object->input_ = input_ ? input_->Clone(storage) : nullptr; object->input_ = input_ ? input_->Clone(storage) : nullptr;
object->list_collection_branch_ = list_collection_branch_ ? list_collection_branch_->Clone(storage) : nullptr; object->list_collection_branch_ = list_collection_branch_ ? list_collection_branch_->Clone(storage) : nullptr;
object->list_collection_symbols_ = list_collection_symbols_;
object->result_symbol_ = result_symbol_;
return object; return object;
} }
std::shared_ptr<memgraph::query::plan::LogicalOperator> input_; std::shared_ptr<memgraph::query::plan::LogicalOperator> input_;
std::shared_ptr<memgraph::query::plan::LogicalOperator> list_collection_branch_; std::shared_ptr<memgraph::query::plan::LogicalOperator> list_collection_branch_;
std::vector<Symbol> list_collection_symbols_;
Symbol result_symbol_;
}; };
} // namespace plan } // namespace plan

View File

@ -702,6 +702,7 @@ void PatternVisitor::Visit(PatternComprehension &op) {
AddMatching({op.pattern_}, op.filter_, symbol_table_, storage_, matching); AddMatching({op.pattern_}, op.filter_, symbol_table_, storage_, matching);
matching.result_expr = storage_.Create<NamedExpression>(symbol_table_.at(op).name(), op.resultExpr_); matching.result_expr = storage_.Create<NamedExpression>(symbol_table_.at(op).name(), op.resultExpr_);
matching.result_expr->MapTo(symbol_table_.at(op)); matching.result_expr->MapTo(symbol_table_.at(op));
matching.result_symbol = symbol_table_.at(op);
pattern_comprehension_matchings_.push_back(std::move(matching)); pattern_comprehension_matchings_.push_back(std::move(matching));
} }

View File

@ -503,6 +503,7 @@ inline auto Filters::IdFilters(const Symbol &symbol) const -> std::vector<Filter
struct PatternComprehensionMatching : Matching { struct PatternComprehensionMatching : Matching {
/// Pattern comprehension result named expression /// Pattern comprehension result named expression
NamedExpression *result_expr = nullptr; NamedExpression *result_expr = nullptr;
Symbol result_symbol;
}; };
/// @brief Represents a read (+ write) part of a query. Parts are split on /// @brief Represents a read (+ write) part of a query. Parts are split on

View File

@ -465,6 +465,18 @@ class EdgeTypeIndexRewriter final : public HierarchicalLogicalOperatorVisitor {
return true; return true;
} }
bool PreVisit(RollUpApply &op) override {
prev_ops_.push_back(&op);
op.input()->Accept(*this);
RewriteBranch(&op.list_collection_branch_);
return false;
}
bool PostVisit(RollUpApply &) override {
prev_ops_.pop_back();
return true;
}
std::shared_ptr<LogicalOperator> new_root_; std::shared_ptr<LogicalOperator> new_root_;
private: private:

View File

@ -19,6 +19,7 @@
#include <unordered_set> #include <unordered_set>
#include "query/frontend/ast/ast.hpp" #include "query/frontend/ast/ast.hpp"
#include "query/frontend/semantic/symbol_table.hpp"
#include "query/plan/operator.hpp" #include "query/plan/operator.hpp"
#include "query/plan/preprocess.hpp" #include "query/plan/preprocess.hpp"
#include "utils/algorithm.hpp" #include "utils/algorithm.hpp"
@ -43,8 +44,7 @@ namespace {
class ReturnBodyContext : public HierarchicalTreeVisitor { class ReturnBodyContext : public HierarchicalTreeVisitor {
public: public:
ReturnBodyContext(const ReturnBody &body, SymbolTable &symbol_table, const std::unordered_set<Symbol> &bound_symbols, ReturnBodyContext(const ReturnBody &body, SymbolTable &symbol_table, const std::unordered_set<Symbol> &bound_symbols,
AstStorage &storage, std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops, AstStorage &storage, PatternComprehensionDataMap &pc_ops, Where *where = nullptr)
Where *where = nullptr)
: body_(body), symbol_table_(symbol_table), bound_symbols_(bound_symbols), storage_(storage), where_(where) { : body_(body), symbol_table_(symbol_table), bound_symbols_(bound_symbols), storage_(storage), where_(where) {
// Collect symbols from named expressions. // Collect symbols from named expressions.
output_symbols_.reserve(body_.named_expressions.size()); output_symbols_.reserve(body_.named_expressions.size());
@ -57,9 +57,11 @@ class ReturnBodyContext : public HierarchicalTreeVisitor {
output_symbols_.emplace_back(symbol_table_.at(*named_expr)); output_symbols_.emplace_back(symbol_table_.at(*named_expr));
named_expr->Accept(*this); named_expr->Accept(*this);
named_expressions_.emplace_back(named_expr); named_expressions_.emplace_back(named_expr);
if (pattern_comprehension_) {
// Pattern comprehension can be filled during named expression traversion
if (has_pattern_comprehension()) {
if (auto it = pc_ops.find(named_expr->name_); it != pc_ops.end()) { if (auto it = pc_ops.find(named_expr->name_); it != pc_ops.end()) {
pattern_comprehension_op_ = std::move(it->second); pattern_comprehension_data_.op = std::move(it->second.op);
pc_ops.erase(it); pc_ops.erase(it);
} else { } else {
throw utils::NotYetImplemented("Operation on top of pattern comprehension"); throw utils::NotYetImplemented("Operation on top of pattern comprehension");
@ -399,18 +401,19 @@ class ReturnBodyContext : public HierarchicalTreeVisitor {
} }
bool PreVisit(PatternComprehension & /*unused*/) override { bool PreVisit(PatternComprehension & /*unused*/) override {
pattern_compression_aggregations_start_index_ = has_aggregation_.size(); aggregations_start_index_ = has_aggregation_.size();
return true; return true;
} }
bool PostVisit(PatternComprehension &pattern_comprehension) override { bool PostVisit(PatternComprehension &pattern_comprehension) override {
bool has_aggr = false; bool has_aggr = false;
for (auto i = has_aggregation_.size(); i > pattern_compression_aggregations_start_index_; --i) { for (auto i = has_aggregation_.size(); i > aggregations_start_index_; --i) {
has_aggr |= has_aggregation_.back(); has_aggr |= has_aggregation_.back();
has_aggregation_.pop_back(); has_aggregation_.pop_back();
} }
has_aggregation_.emplace_back(has_aggr); has_aggregation_.emplace_back(has_aggr);
pattern_comprehension_ = &pattern_comprehension; pattern_comprehension_data_.pattern_comprehension = &pattern_comprehension;
pattern_comprehension_data_.result_symbol = symbol_table_.at(pattern_comprehension);
return true; return true;
} }
@ -468,9 +471,11 @@ class ReturnBodyContext : public HierarchicalTreeVisitor {
// named_expressions. // named_expressions.
const auto &output_symbols() const { return output_symbols_; } const auto &output_symbols() const { return output_symbols_; }
const auto *pattern_comprehension() const { return pattern_comprehension_; } const bool has_pattern_comprehension() const { return pattern_comprehension_data_.pattern_comprehension != nullptr; }
std::shared_ptr<LogicalOperator> pattern_comprehension_op() const { return pattern_comprehension_op_; } const PatternComprehensionData pattern_comprehension_data() const { return pattern_comprehension_data_; }
const SymbolTable &symbol_table() const { return symbol_table_; }
private: private:
const ReturnBody &body_; const ReturnBody &body_;
@ -493,9 +498,8 @@ class ReturnBodyContext : public HierarchicalTreeVisitor {
// group by it. // group by it.
std::list<bool> has_aggregation_; std::list<bool> has_aggregation_;
std::vector<NamedExpression *> named_expressions_; std::vector<NamedExpression *> named_expressions_;
PatternComprehension *pattern_comprehension_ = nullptr; PatternComprehensionData pattern_comprehension_data_;
std::shared_ptr<LogicalOperator> pattern_comprehension_op_; size_t aggregations_start_index_ = 0;
size_t pattern_compression_aggregations_start_index_ = 0;
}; };
std::unique_ptr<LogicalOperator> GenReturnBody(std::unique_ptr<LogicalOperator> input_op, bool advance_command, std::unique_ptr<LogicalOperator> GenReturnBody(std::unique_ptr<LogicalOperator> input_op, bool advance_command,
@ -514,8 +518,11 @@ std::unique_ptr<LogicalOperator> GenReturnBody(std::unique_ptr<LogicalOperator>
last_op = std::make_unique<Aggregate>(std::move(last_op), body.aggregations(), body.group_by(), remember); last_op = std::make_unique<Aggregate>(std::move(last_op), body.aggregations(), body.group_by(), remember);
} }
if (body.pattern_comprehension()) { if (body.has_pattern_comprehension()) {
last_op = std::make_unique<RollUpApply>(std::move(last_op), body.pattern_comprehension_op()); auto list_collection_data = body.pattern_comprehension_data();
auto list_collection_symbols = list_collection_data.op->ModifiedSymbols(body.symbol_table());
last_op = std::make_unique<RollUpApply>(std::move(last_op), std::move(list_collection_data.op),
list_collection_symbols, list_collection_data.result_symbol);
} }
last_op = std::make_unique<Produce>(std::move(last_op), body.named_expressions()); last_op = std::make_unique<Produce>(std::move(last_op), body.named_expressions());
@ -580,9 +587,9 @@ Expression *ExtractFilters(const std::unordered_set<Symbol> &bound_symbols, Filt
return filter_expr; return filter_expr;
} }
std::unordered_set<Symbol> GetSubqueryBoundSymbols( std::unordered_set<Symbol> GetSubqueryBoundSymbols(const std::vector<SingleQueryPart> &single_query_parts,
const std::vector<SingleQueryPart> &single_query_parts, SymbolTable &symbol_table, AstStorage &storage, SymbolTable &symbol_table, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops) { PatternComprehensionDataMap &pc_ops) {
const auto &query = single_query_parts[0]; const auto &query = single_query_parts[0];
if (!query.matching.expansions.empty() || query.remaining_clauses.empty()) { if (!query.matching.expansions.empty() || query.remaining_clauses.empty()) {
@ -622,7 +629,7 @@ std::unique_ptr<LogicalOperator> GenNamedPaths(std::unique_ptr<LogicalOperator>
std::unique_ptr<LogicalOperator> GenReturn(Return &ret, std::unique_ptr<LogicalOperator> input_op, std::unique_ptr<LogicalOperator> GenReturn(Return &ret, std::unique_ptr<LogicalOperator> input_op,
SymbolTable &symbol_table, bool is_write, SymbolTable &symbol_table, bool is_write,
const std::unordered_set<Symbol> &bound_symbols, AstStorage &storage, const std::unordered_set<Symbol> &bound_symbols, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops) { PatternComprehensionDataMap &pc_ops) {
// Similar to WITH clause, but we want to accumulate when the query writes to // Similar to WITH clause, but we want to accumulate when the query writes to
// the database. This way we handle the case when we want to return // the database. This way we handle the case when we want to return
// expressions with the latest updated results. For example, `MATCH (n) -- () // expressions with the latest updated results. For example, `MATCH (n) -- ()
@ -638,7 +645,7 @@ std::unique_ptr<LogicalOperator> GenReturn(Return &ret, std::unique_ptr<LogicalO
std::unique_ptr<LogicalOperator> GenWith(With &with, std::unique_ptr<LogicalOperator> input_op, std::unique_ptr<LogicalOperator> GenWith(With &with, std::unique_ptr<LogicalOperator> input_op,
SymbolTable &symbol_table, bool is_write, SymbolTable &symbol_table, bool is_write,
std::unordered_set<Symbol> &bound_symbols, AstStorage &storage, std::unordered_set<Symbol> &bound_symbols, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops) { PatternComprehensionDataMap &pc_ops) {
// WITH clause is Accumulate/Aggregate (advance_command) + Produce and // WITH clause is Accumulate/Aggregate (advance_command) + Produce and
// optional Filter. In case of update and aggregation, we want to accumulate // optional Filter. In case of update and aggregation, we want to accumulate
// first, so that when aggregating, we get the latest results. Similar to // first, so that when aggregating, we get the latest results. Similar to

View File

@ -27,6 +27,18 @@
namespace memgraph::query::plan { namespace memgraph::query::plan {
struct PatternComprehensionData {
PatternComprehensionData() = default;
PatternComprehensionData(std::shared_ptr<LogicalOperator> lop, Symbol res_symbol)
: op(std::move(lop)), result_symbol(res_symbol) {}
PatternComprehension *pattern_comprehension = nullptr;
std::shared_ptr<LogicalOperator> op;
Symbol result_symbol;
};
using PatternComprehensionDataMap = std::unordered_map<std::string, PatternComprehensionData>;
/// @brief Context which contains variables commonly used during planning. /// @brief Context which contains variables commonly used during planning.
template <class TDbAccessor> template <class TDbAccessor>
struct PlanningContext { struct PlanningContext {
@ -88,9 +100,9 @@ bool HasBoundFilterSymbols(const std::unordered_set<Symbol> &bound_symbols, cons
// Returns the set of symbols for the subquery that are actually referenced from the outer scope and // Returns the set of symbols for the subquery that are actually referenced from the outer scope and
// used in the subquery. // used in the subquery.
std::unordered_set<Symbol> GetSubqueryBoundSymbols( std::unordered_set<Symbol> GetSubqueryBoundSymbols(const std::vector<SingleQueryPart> &single_query_parts,
const std::vector<SingleQueryPart> &single_query_parts, SymbolTable &symbol_table, AstStorage &storage, SymbolTable &symbol_table, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops); PatternComprehensionDataMap &pc_ops);
Symbol GetSymbol(NodeAtom *atom, const SymbolTable &symbol_table); Symbol GetSymbol(NodeAtom *atom, const SymbolTable &symbol_table);
Symbol GetSymbol(EdgeAtom *atom, const SymbolTable &symbol_table); Symbol GetSymbol(EdgeAtom *atom, const SymbolTable &symbol_table);
@ -145,12 +157,12 @@ std::unique_ptr<LogicalOperator> GenNamedPaths(std::unique_ptr<LogicalOperator>
std::unique_ptr<LogicalOperator> GenReturn(Return &ret, std::unique_ptr<LogicalOperator> input_op, std::unique_ptr<LogicalOperator> GenReturn(Return &ret, std::unique_ptr<LogicalOperator> input_op,
SymbolTable &symbol_table, bool is_write, SymbolTable &symbol_table, bool is_write,
const std::unordered_set<Symbol> &bound_symbols, AstStorage &storage, const std::unordered_set<Symbol> &bound_symbols, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops); PatternComprehensionDataMap &pc_ops);
std::unique_ptr<LogicalOperator> GenWith(With &with, std::unique_ptr<LogicalOperator> input_op, std::unique_ptr<LogicalOperator> GenWith(With &with, std::unique_ptr<LogicalOperator> input_op,
SymbolTable &symbol_table, bool is_write, SymbolTable &symbol_table, bool is_write,
std::unordered_set<Symbol> &bound_symbols, AstStorage &storage, std::unordered_set<Symbol> &bound_symbols, AstStorage &storage,
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops); PatternComprehensionDataMap &pc_ops);
std::unique_ptr<LogicalOperator> GenUnion(const CypherUnion &cypher_union, std::shared_ptr<LogicalOperator> left_op, std::unique_ptr<LogicalOperator> GenUnion(const CypherUnion &cypher_union, std::shared_ptr<LogicalOperator> left_op,
std::shared_ptr<LogicalOperator> right_op, SymbolTable &symbol_table); std::shared_ptr<LogicalOperator> right_op, SymbolTable &symbol_table);
@ -194,7 +206,7 @@ class RuleBasedPlanner {
uint64_t merge_id = 0; uint64_t merge_id = 0;
uint64_t subquery_id = 0; uint64_t subquery_id = 0;
std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pattern_comprehension_ops; PatternComprehensionDataMap pattern_comprehension_ops;
if (single_query_part.pattern_comprehension_matchings.size() > 1) { if (single_query_part.pattern_comprehension_matchings.size() > 1) {
throw utils::NotYetImplemented("Multiple pattern comprehensions."); throw utils::NotYetImplemented("Multiple pattern comprehensions.");
@ -204,7 +216,8 @@ class RuleBasedPlanner {
MatchContext match_ctx{matching.second, *context.symbol_table, context.bound_symbols}; MatchContext match_ctx{matching.second, *context.symbol_table, context.bound_symbols};
new_input = PlanMatching(match_ctx, std::move(new_input)); new_input = PlanMatching(match_ctx, std::move(new_input));
new_input = std::make_unique<Produce>(std::move(new_input), std::vector{matching.second.result_expr}); new_input = std::make_unique<Produce>(std::move(new_input), std::vector{matching.second.result_expr});
pattern_comprehension_ops.emplace(matching.first, std::move(new_input)); pattern_comprehension_ops.emplace(
matching.first, PatternComprehensionData(std::move(new_input), matching.second.result_symbol));
} }
for (const auto &clause : single_query_part.remaining_clauses) { for (const auto &clause : single_query_part.remaining_clauses) {
@ -875,9 +888,9 @@ class RuleBasedPlanner {
symbol); symbol);
} }
std::unique_ptr<LogicalOperator> HandleSubquery( std::unique_ptr<LogicalOperator> HandleSubquery(std::unique_ptr<LogicalOperator> last_op,
std::unique_ptr<LogicalOperator> last_op, std::shared_ptr<QueryParts> subquery, SymbolTable &symbol_table, std::shared_ptr<QueryParts> subquery, SymbolTable &symbol_table,
AstStorage &storage, std::unordered_map<std::string, std::shared_ptr<LogicalOperator>> pc_ops) { AstStorage &storage, PatternComprehensionDataMap &pc_ops) {
std::unordered_set<Symbol> outer_scope_bound_symbols; std::unordered_set<Symbol> outer_scope_bound_symbols;
outer_scope_bound_symbols.insert(std::make_move_iterator(context_->bound_symbols.begin()), outer_scope_bound_symbols.insert(std::make_move_iterator(context_->bound_symbols.begin()),
std::make_move_iterator(context_->bound_symbols.end())); std::make_move_iterator(context_->bound_symbols.end()));

View File

@ -1,4 +1,4 @@
// Copyright 2023 Memgraph Ltd. // Copyright 2024 Memgraph Ltd.
// //
// Use of this software is governed by the Business Source License // 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 // included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
@ -57,6 +57,7 @@
M(ApplyOperator, Operator, "Number of times ApplyOperator operator was used.") \ M(ApplyOperator, Operator, "Number of times ApplyOperator operator was used.") \
M(IndexedJoinOperator, Operator, "Number of times IndexedJoin operator was used.") \ M(IndexedJoinOperator, Operator, "Number of times IndexedJoin operator was used.") \
M(HashJoinOperator, Operator, "Number of times HashJoin operator was used.") \ M(HashJoinOperator, Operator, "Number of times HashJoin operator was used.") \
M(RollUpApplyOperator, Operator, "Number of times RollUpApply operator was used.") \
\ \
M(ActiveLabelIndices, Index, "Number of active label indices in the system.") \ M(ActiveLabelIndices, Index, "Number of active label indices in the system.") \
M(ActiveLabelPropertyIndices, Index, "Number of active label property indices in the system.") \ M(ActiveLabelPropertyIndices, Index, "Number of active label property indices in the system.") \

View File

@ -287,10 +287,20 @@ Feature: List operators
MATCH (keanu:Person {name: 'Keanu Reeves'}) MATCH (keanu:Person {name: 'Keanu Reeves'})
RETURN [(keanu)-->(b:Movie) WHERE b.title CONTAINS 'Matrix' | b.released] AS years RETURN [(keanu)-->(b:Movie) WHERE b.title CONTAINS 'Matrix' | b.released] AS years
""" """
Then an error should be raised Then the result should be:
# Then the result should be: | years |
# | years | | [2003, 2003, 1999, 2021] |
# | [2021,2003,2003,1999] |
Scenario: List pattern comprehension and property
Given graph "graph_keanu"
When executing query:
"""
MATCH (keanu:Person {name: 'Keanu Reeves'})
RETURN [(keanu)-->(b:Movie) WHERE b.title CONTAINS 'Matrix' | b.released] AS years, keanu.name
"""
Then the result should be:
| years | keanu.name |
| [2003, 2003, 1999, 2021] | "Keanu Reeves" |
Scenario: Multiple entries with list pattern comprehension Scenario: Multiple entries with list pattern comprehension
Given graph "graph_keanu" Given graph "graph_keanu"
@ -299,7 +309,11 @@ Feature: List operators
MATCH (n:Person) MATCH (n:Person)
RETURN n.name, [(n)-->(b:Movie) WHERE b.title CONTAINS 'Matrix' | b.released] AS years RETURN n.name, [(n)-->(b:Movie) WHERE b.title CONTAINS 'Matrix' | b.released] AS years
""" """
Then an error should be raised Then the result should be:
| n.name | years |
| "Keanu Reeves" | [2003, 2003, 1999, 2021] |
| "Carrie-Anne Moss" | [2003,1999] |
| "Laurence Fishburne" | [1999] |
Scenario: Multiple list pattern comprehensions in Return Scenario: Multiple list pattern comprehensions in Return
Given graph "graph_keanu" Given graph "graph_keanu"