/// /// @file /// This file provides macros for easier construction of openCypher query AST. /// The usage of macros is very similar to how one would write openCypher. For /// example: /// /// AstTreeStorage storage; // Macros rely on storage being in scope. /// // PROPERTY_LOOKUP and PROPERTY_PAIR macros also rely on a graph DB /// // accessor /// std::unique_ptr dba; /// /// QUERY(MATCH(PATTERN(NODE("n"), EDGE("e"), NODE("m"))), /// WHERE(LESS(PROPERTY_LOOKUP("e", edge_prop), LITERAL(3))), /// RETURN(SUM(PROPERTY_LOOKUP("m", prop)), AS("sum"), /// ORDER_BY(IDENT("sum")), /// SKIP(ADD(LITERAL(1), LITERAL(2))))); /// /// Each of the macros is accompanied by a function. The functions use overload /// resolution and template magic to provide a type safe way of constructing /// queries. Although the functions can be used by themselves, it is more /// convenient to use the macros. /// #pragma once #include #include #include #include #include #include #include "database/dbms.hpp" #include "database/graph_db_datatypes.hpp" #include "query/frontend/ast/ast.hpp" #include "query/interpret/awesome_memgraph_functions.hpp" #include "utils/string.hpp" namespace query { namespace test_common { template auto ToList(const TypedValue &t) { std::vector list; for (auto x : t.Value>()) { list.push_back(x.Value()); } return list; }; template auto ToMap(const TypedValue &t) { std::map map; for (const auto &kv : t.Value>()) map.emplace(kv.first, kv.second.Value()); return map; }; // Custom types for ORDER BY, SKIP, LIMIT, ON MATCH and ON CREATE expressions, // so that they can be used to resolve function calls. struct OrderBy { std::vector> expressions; }; struct Skip { Expression *expression = nullptr; }; struct Limit { Expression *expression = nullptr; }; struct OnMatch { std::vector set; }; struct OnCreate { std::vector set; }; // Helper functions for filling the OrderBy with expressions. auto FillOrderBy(OrderBy &order_by, Expression *expression, Ordering ordering = Ordering::ASC) { order_by.expressions.emplace_back(ordering, expression); } template auto FillOrderBy(OrderBy &order_by, Expression *expression, Ordering ordering, T... rest) { FillOrderBy(order_by, expression, ordering); FillOrderBy(order_by, rest...); } template auto FillOrderBy(OrderBy &order_by, Expression *expression, T... rest) { FillOrderBy(order_by, expression); FillOrderBy(order_by, rest...); } /// /// Create OrderBy expressions. /// /// The supported combination of arguments is: (Expression, [Ordering])+ /// Since the Ordering is optional, by default it is ascending. /// template auto GetOrderBy(T... exprs) { OrderBy order_by; FillOrderBy(order_by, exprs...); return order_by; } /// /// Create PropertyLookup with given name and property. /// /// Name is used to create the Identifier which is used for property lookup. /// auto GetPropertyLookup(AstTreeStorage &storage, std::unique_ptr &dba, const std::string &name, GraphDbTypes::Property property) { return storage.Create(storage.Create(name), dba->PropertyName(property), property); } auto GetPropertyLookup(AstTreeStorage &storage, std::unique_ptr &dba, Expression *expr, GraphDbTypes::Property property) { return storage.Create(expr, dba->PropertyName(property), property); } auto GetPropertyLookup( AstTreeStorage &storage, std::unique_ptr &, const std::string &name, const std::pair &prop_pair) { return storage.Create(storage.Create(name), prop_pair.first, prop_pair.second); } auto GetPropertyLookup( AstTreeStorage &storage, std::unique_ptr &, Expression *expr, const std::pair &prop_pair) { return storage.Create(expr, prop_pair.first, prop_pair.second); } /// /// Create an EdgeAtom with given name, edge_type and direction. /// /// Name is used to create the Identifier which is assigned to the edge. /// auto GetEdge(AstTreeStorage &storage, const std::string &name, GraphDbTypes::EdgeType edge_type = nullptr, EdgeAtom::Direction dir = EdgeAtom::Direction::BOTH) { auto edge = storage.Create(storage.Create(name), dir); if (edge_type) edge->edge_types_.emplace_back(edge_type); return edge; } /// /// Create an EdgeAtom with given name and direction. /// auto GetEdge(AstTreeStorage &storage, const std::string &name, EdgeAtom::Direction dir) { return GetEdge(storage, name, nullptr, dir); } /// /// Create a NodeAtom with given name and label. /// /// Name is used to create the Identifier which is assigned to the node. /// auto GetNode(AstTreeStorage &storage, const std::string &name, GraphDbTypes::Label label = nullptr) { auto node = storage.Create(storage.Create(name)); if (label) node->labels_.emplace_back(label); return node; } /// Generates a randomly chosen (uniformly) string from a population of 10 ** 20 std::string random_string() { std::string str = "rand_str_"; for (int i = 0; i < 20; i++) str += std::to_string(rand() % 10); return str; } /// /// Create a Pattern with given atoms. /// auto GetPattern(AstTreeStorage &storage, std::vector atoms) { auto pattern = storage.Create(); pattern->identifier_ = storage.Create(random_string(), false); pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end()); return pattern; } /// /// Create a Pattern with given name and atoms. /// auto GetPattern(AstTreeStorage &storage, const std::string &name, std::vector atoms) { auto pattern = storage.Create(); pattern->identifier_ = storage.Create(name, true); pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end()); return pattern; } /// /// This function fills an AST node which with given patterns. /// /// The function is most commonly used to create Match and Create clauses. /// template auto GetWithPatterns(TWithPatterns *with_patterns, std::vector patterns) { with_patterns->patterns_.insert(with_patterns->patterns_.begin(), patterns.begin(), patterns.end()); return with_patterns; } /// /// Create a query with given clauses. /// auto GetQuery(AstTreeStorage &storage, Clause *clause) { storage.query()->clauses_.emplace_back(clause); return storage.query(); } auto GetQuery(AstTreeStorage &storage, Match *match, Where *where) { match->where_ = where; storage.query()->clauses_.emplace_back(match); return storage.query(); } auto GetQuery(AstTreeStorage &storage, With *with, Where *where) { with->where_ = where; storage.query()->clauses_.emplace_back(with); return storage.query(); } template auto GetQuery(AstTreeStorage &storage, Match *match, Where *where, T *... clauses) { match->where_ = where; storage.query()->clauses_.emplace_back(match); return GetQuery(storage, clauses...); } template auto GetQuery(AstTreeStorage &storage, With *with, Where *where, T *... clauses) { with->where_ = where; storage.query()->clauses_.emplace_back(with); return GetQuery(storage, clauses...); } template auto GetQuery(AstTreeStorage &storage, Clause *clause, T *... clauses) { storage.query()->clauses_.emplace_back(clause); return GetQuery(storage, clauses...); } // Helper functions for constructing RETURN and WITH clauses. void FillReturnBody(AstTreeStorage &, ReturnBody &body, NamedExpression *named_expr) { body.named_expressions.emplace_back(named_expr); } void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, const std::string &name) { auto *ident = storage.Create(name); auto *named_expr = storage.Create(name, ident); body.named_expressions.emplace_back(named_expr); } void FillReturnBody(AstTreeStorage &, ReturnBody &body, Limit limit) { body.limit = limit.expression; } void FillReturnBody(AstTreeStorage &, ReturnBody &body, Skip skip, Limit limit = Limit{}) { body.skip = skip.expression; body.limit = limit.expression; } void FillReturnBody(AstTreeStorage &, ReturnBody &body, OrderBy order_by, Limit limit = Limit{}) { body.order_by = order_by.expressions; body.limit = limit.expression; } void FillReturnBody(AstTreeStorage &, ReturnBody &body, OrderBy order_by, Skip skip, Limit limit = Limit{}) { body.order_by = order_by.expressions; body.skip = skip.expression; body.limit = limit.expression; } void FillReturnBody(AstTreeStorage &, ReturnBody &body, Expression *expr, NamedExpression *named_expr) { // This overload supports `RETURN(expr, AS(name))` construct, since // NamedExpression does not inherit Expression. named_expr->expression_ = expr; body.named_expressions.emplace_back(named_expr); } void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, const std::string &name, NamedExpression *named_expr) { named_expr->expression_ = storage.Create(name); body.named_expressions.emplace_back(named_expr); } template void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, Expression *expr, NamedExpression *named_expr, T... rest) { named_expr->expression_ = expr; body.named_expressions.emplace_back(named_expr); FillReturnBody(storage, body, rest...); } template void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, NamedExpression *named_expr, T... rest) { body.named_expressions.emplace_back(named_expr); FillReturnBody(storage, body, rest...); } template void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, const std::string &name, NamedExpression *named_expr, T... rest) { named_expr->expression_ = storage.Create(name); body.named_expressions.emplace_back(named_expr); FillReturnBody(storage, body, rest...); } template void FillReturnBody(AstTreeStorage &storage, ReturnBody &body, const std::string &name, T... rest) { auto *ident = storage.Create(name); auto *named_expr = storage.Create(name, ident); body.named_expressions.emplace_back(named_expr); FillReturnBody(storage, body, rest...); } /// /// Create the return clause with given expressions. /// /// The supported expression combination of arguments is: /// /// (String | NamedExpression | (Expression NamedExpression))+ /// [OrderBy] [Skip] [Limit] /// /// When the pair (Expression NamedExpression) is given, the Expression will be /// moved inside the NamedExpression. This is done, so that the constructs like /// RETURN(expr, AS("name"), ...) are supported. Taking a String is a shorthand /// for RETURN(IDENT(string), AS(string), ....). /// /// @sa GetWith template auto GetReturn(AstTreeStorage &storage, bool distinct, T... exprs) { auto ret = storage.Create(); ret->body_.distinct = distinct; FillReturnBody(storage, ret->body_, exprs...); return ret; } /// /// Create the with clause with given expressions. /// /// The supported expression combination is the same as for @c GetReturn. /// /// @sa GetReturn template auto GetWith(AstTreeStorage &storage, bool distinct, T... exprs) { auto with = storage.Create(); with->body_.distinct = distinct; FillReturnBody(storage, with->body_, exprs...); return with; } /// /// Create the UNWIND clause with given named expression. /// auto GetUnwind(AstTreeStorage &storage, NamedExpression *named_expr) { return storage.Create(named_expr); } auto GetUnwind(AstTreeStorage &storage, Expression *expr, NamedExpression *as) { as->expression_ = expr; return GetUnwind(storage, as); } /// /// Create the delete clause with given named expressions. /// auto GetDelete(AstTreeStorage &storage, std::vector exprs, bool detach = false) { auto del = storage.Create(); del->expressions_.insert(del->expressions_.begin(), exprs.begin(), exprs.end()); del->detach_ = detach; return del; } /// /// Create a set property clause for given property lookup and the right hand /// side expression. /// auto GetSet(AstTreeStorage &storage, PropertyLookup *prop_lookup, Expression *expr) { return storage.Create(prop_lookup, expr); } /// /// Create a set properties clause for given identifier name and the right hand /// side expression. /// auto GetSet(AstTreeStorage &storage, const std::string &name, Expression *expr, bool update = false) { return storage.Create(storage.Create(name), expr, update); } /// /// Create a set labels clause for given identifier name and labels. /// auto GetSet(AstTreeStorage &storage, const std::string &name, std::vector labels) { return storage.Create(storage.Create(name), labels); } /// /// Create a remove property clause for given property lookup /// auto GetRemove(AstTreeStorage &storage, PropertyLookup *prop_lookup) { return storage.Create(prop_lookup); } /// /// Create a remove labels clause for given identifier name and labels. /// auto GetRemove(AstTreeStorage &storage, const std::string &name, std::vector labels) { return storage.Create(storage.Create(name), labels); } /// /// Create a Merge clause for given Pattern with optional OnMatch and OnCreate /// parts. /// auto GetMerge(AstTreeStorage &storage, Pattern *pattern, OnCreate on_create = OnCreate{}) { auto *merge = storage.Create(); merge->pattern_ = pattern; merge->on_create_ = on_create.set; return merge; } auto GetMerge(AstTreeStorage &storage, Pattern *pattern, OnMatch on_match, OnCreate on_create = OnCreate{}) { auto *merge = storage.Create(); merge->pattern_ = pattern; merge->on_match_ = on_match.set; merge->on_create_ = on_create.set; return merge; } } // namespace test_common } // namespace query /// /// All the following macros implicitly pass `storage` variable to functions. /// You need to have `AstTreeStorage storage;` somewhere in scope to use them. /// Refer to function documentation to see what the macro does. /// /// Example usage: /// /// // Create MATCH (n) -[r]- (m) RETURN m AS new_name /// AstTreeStorage storage; /// auto query = QUERY(MATCH(PATTERN(NODE("n"), EDGE("r"), NODE("m"))), /// RETURN(NEXPR("new_name"), IDENT("m"))); /// #define NODE(...) query::test_common::GetNode(storage, __VA_ARGS__) #define EDGE(...) query::test_common::GetEdge(storage, __VA_ARGS__) #define PATTERN(...) query::test_common::GetPattern(storage, {__VA_ARGS__}) #define NAMED_PATTERN(name, ...) \ query::test_common::GetPattern(storage, name, {__VA_ARGS__}) #define OPTIONAL_MATCH(...) \ query::test_common::GetWithPatterns(storage.Create(true), \ {__VA_ARGS__}) #define MATCH(...) \ query::test_common::GetWithPatterns(storage.Create(), \ {__VA_ARGS__}) #define WHERE(expr) storage.Create((expr)) #define CREATE(...) \ query::test_common::GetWithPatterns(storage.Create(), \ {__VA_ARGS__}) #define IDENT(name) storage.Create((name)) #define LITERAL(val) storage.Create((val)) #define LIST(...) \ storage.Create( \ std::vector{__VA_ARGS__}) #define MAP(...) \ storage.Create( \ std::unordered_map, \ query::Expression *>{__VA_ARGS__}) #define PROPERTY_PAIR(property_name) \ std::make_pair(property_name, dba->Property(property_name)) #define PROPERTY_LOOKUP(...) \ query::test_common::GetPropertyLookup(storage, dba, __VA_ARGS__) #define NEXPR(name, expr) storage.Create((name), (expr)) // AS is alternative to NEXPR which does not initialize NamedExpression with // Expression. It should be used with RETURN or WITH. For example: // RETURN(IDENT("n"), AS("n")) vs. RETURN(NEXPR("n", IDENT("n"))). #define AS(name) storage.Create((name)) #define RETURN(...) query::test_common::GetReturn(storage, false, __VA_ARGS__) #define WITH(...) query::test_common::GetWith(storage, false, __VA_ARGS__) #define RETURN_DISTINCT(...) \ query::test_common::GetReturn(storage, true, __VA_ARGS__) #define WITH_DISTINCT(...) \ query::test_common::GetWith(storage, true, __VA_ARGS__) #define UNWIND(...) query::test_common::GetUnwind(storage, __VA_ARGS__) #define ORDER_BY(...) query::test_common::GetOrderBy(__VA_ARGS__) #define SKIP(expr) \ query::test_common::Skip { (expr) } #define LIMIT(expr) \ query::test_common::Limit { (expr) } #define DELETE(...) query::test_common::GetDelete(storage, {__VA_ARGS__}) #define DETACH_DELETE(...) \ query::test_common::GetDelete(storage, {__VA_ARGS__}, true) #define SET(...) query::test_common::GetSet(storage, __VA_ARGS__) #define REMOVE(...) query::test_common::GetRemove(storage, __VA_ARGS__) #define MERGE(...) query::test_common::GetMerge(storage, __VA_ARGS__) #define ON_MATCH(...) \ query::test_common::OnMatch { \ std::vector { __VA_ARGS__ } \ } #define ON_CREATE(...) \ query::test_common::OnCreate { \ std::vector { __VA_ARGS__ } \ } #define CREATE_INDEX_ON(label, property) \ storage.Create((label), (property)) #define QUERY(...) query::test_common::GetQuery(storage, __VA_ARGS__) // Various operators #define ADD(expr1, expr2) \ storage.Create((expr1), (expr2)) #define LESS(expr1, expr2) storage.Create((expr1), (expr2)) #define LESS_EQ(expr1, expr2) \ storage.Create((expr1), (expr2)) #define GREATER(expr1, expr2) \ storage.Create((expr1), (expr2)) #define GREATER_EQ(expr1, expr2) \ storage.Create((expr1), (expr2)) #define SUM(expr) \ storage.Create((expr), nullptr, \ query::Aggregation::Op::SUM) #define COUNT(expr) \ storage.Create((expr), nullptr, \ query::Aggregation::Op::COUNT) #define EQ(expr1, expr2) storage.Create((expr1), (expr2)) #define NEQ(expr1, expr2) \ storage.Create((expr1), (expr2)) #define AND(expr1, expr2) storage.Create((expr1), (expr2)) #define OR(expr1, expr2) storage.Create((expr1), (expr2)) // Function call #define FN(function_name, ...) \ storage.Create( \ query::NameToFunction(utils::ToUpperCase(function_name)), \ std::vector{__VA_ARGS__}) // List slicing #define SLICE(list, lower_bound, upper_bound) \ storage.Create(list, lower_bound, upper_bound) // all(variable IN list WHERE predicate) #define ALL(variable, list, where) \ storage.Create(storage.Create(variable), \ list, where)