/// @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: /// /// AstStorage storage; // Macros rely on storage being in scope. /// // PROPERTY_LOOKUP and PROPERTY_PAIR macros /// // rely on a DbAccessor *reference* named dba. /// database::GraphDb db; /// auto dba_ptr = db.Access(); /// auto &dba = *dba_ptr; /// /// 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 "query/frontend/ast/ast.hpp" #include "storage/common/types.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.push_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. template auto GetPropertyLookup(AstStorage &storage, TDbAccessor &dba, const std::string &name, storage::Property property) { return storage.Create(storage.Create(name), dba.PropertyName(property), property); } template auto GetPropertyLookup(AstStorage &storage, TDbAccessor &dba, Expression *expr, storage::Property property) { return storage.Create(expr, dba.PropertyName(property), property); } template auto GetPropertyLookup( AstStorage &storage, TDbAccessor &, const std::string &name, const std::pair &prop_pair) { return storage.Create(storage.Create(name), prop_pair.first, prop_pair.second); } template auto GetPropertyLookup( AstStorage &storage, TDbAccessor &, Expression *expr, const std::pair &prop_pair) { return storage.Create(expr, prop_pair.first, prop_pair.second); } /// Create an EdgeAtom with given name, direction and edge_type. /// /// Name is used to create the Identifier which is assigned to the edge. auto GetEdge(AstStorage &storage, const std::string &name, EdgeAtom::Direction dir = EdgeAtom::Direction::BOTH, const std::vector &edge_types = {}) { return storage.Create(storage.Create(name), EdgeAtom::Type::SINGLE, dir, edge_types); } /// Create a variable length expansion EdgeAtom with given name, direction and /// edge_type. /// /// Name is used to create the Identifier which is assigned to the edge. auto GetEdgeVariable(AstStorage &storage, const std::string &name, EdgeAtom::Direction dir = EdgeAtom::Direction::BOTH, const std::vector &edge_types = {}, Identifier *inner_edge = nullptr, Identifier *inner_node = nullptr) { auto r_val = storage.Create(storage.Create(name), EdgeAtom::Type::DEPTH_FIRST, dir, edge_types); r_val->filter_lambda_.inner_edge = inner_edge ? inner_edge : storage.Create(utils::RandomString(20)); r_val->filter_lambda_.inner_node = inner_node ? inner_node : storage.Create(utils::RandomString(20)); return r_val; } /// Create a NodeAtom with given name and label. /// /// Name is used to create the Identifier which is assigned to the node. auto GetNode(AstStorage &storage, const std::string &name, std::experimental::optional label = std::experimental::nullopt) { auto node = storage.Create(storage.Create(name)); if (label) node->labels_.emplace_back(*label); return node; } /// Create a Pattern with given atoms. auto GetPattern(AstStorage &storage, std::vector atoms) { auto pattern = storage.Create(); pattern->identifier_ = storage.Create(utils::RandomString(20), false); pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end()); return pattern; } /// Create a Pattern with given name and atoms. auto GetPattern(AstStorage &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 GetSingleQuery(SingleQuery *single_query, Clause *clause) { single_query->clauses_.emplace_back(clause); return single_query; } auto GetSingleQuery(SingleQuery *single_query, Match *match, Where *where) { match->where_ = where; single_query->clauses_.emplace_back(match); return single_query; } auto GetSingleQuery(SingleQuery *single_query, With *with, Where *where) { with->where_ = where; single_query->clauses_.emplace_back(with); return single_query; } template auto GetSingleQuery(SingleQuery *single_query, Match *match, Where *where, T *... clauses) { match->where_ = where; single_query->clauses_.emplace_back(match); return GetSingleQuery(single_query, clauses...); } template auto GetSingleQuery(SingleQuery *single_query, With *with, Where *where, T *... clauses) { with->where_ = where; single_query->clauses_.emplace_back(with); return GetSingleQuery(single_query, clauses...); } template auto GetSingleQuery(SingleQuery *single_query, Clause *clause, T *... clauses) { single_query->clauses_.emplace_back(clause); return GetSingleQuery(single_query, clauses...); } auto GetCypherUnion(CypherUnion *cypher_union, SingleQuery *single_query) { cypher_union->single_query_ = single_query; return cypher_union; } auto GetQuery(AstStorage &storage, SingleQuery *single_query) { auto *query = storage.Create(); query->single_query_ = single_query; return query; } template auto GetQuery(AstStorage &storage, SingleQuery *single_query, T *... cypher_unions) { auto *query = storage.Create(); query->single_query_ = single_query; query->cypher_unions_ = std::vector{cypher_unions...}; return query; } // Helper functions for constructing RETURN and WITH clauses. void FillReturnBody(AstStorage &, ReturnBody &body, NamedExpression *named_expr) { body.named_expressions.emplace_back(named_expr); } void FillReturnBody(AstStorage &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(AstStorage &, ReturnBody &body, Limit limit) { body.limit = limit.expression; } void FillReturnBody(AstStorage &, ReturnBody &body, Skip skip, Limit limit = Limit{}) { body.skip = skip.expression; body.limit = limit.expression; } void FillReturnBody(AstStorage &, ReturnBody &body, OrderBy order_by, Limit limit = Limit{}) { body.order_by = order_by.expressions; body.limit = limit.expression; } void FillReturnBody(AstStorage &, 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(AstStorage &, 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(AstStorage &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(AstStorage &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(AstStorage &storage, ReturnBody &body, NamedExpression *named_expr, T... rest) { body.named_expressions.emplace_back(named_expr); FillReturnBody(storage, body, rest...); } template void FillReturnBody(AstStorage &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(AstStorage &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(AstStorage &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(AstStorage &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(AstStorage &storage, NamedExpression *named_expr) { return storage.Create(named_expr); } auto GetUnwind(AstStorage &storage, Expression *expr, NamedExpression *as) { as->expression_ = expr; return GetUnwind(storage, as); } /// Create the delete clause with given named expressions. auto GetDelete(AstStorage &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(AstStorage &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(AstStorage &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(AstStorage &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(AstStorage &storage, PropertyLookup *prop_lookup) { return storage.Create(prop_lookup); } /// Create a remove labels clause for given identifier name and labels. auto GetRemove(AstStorage &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(AstStorage &storage, Pattern *pattern, OnCreate on_create = OnCreate{}) { auto *merge = storage.Create(); merge->pattern_ = pattern; merge->on_create_ = on_create.set; return merge; } auto GetMerge(AstStorage &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 `AstStorage 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 /// AstStorage 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 EDGE_VARIABLE(...) \ query::test_common::GetEdgeVariable(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 PARAMETER_LOOKUP(token_position) \ storage.Create((token_position)) #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__) #define SINGLE_QUERY(...) \ query::test_common::GetSingleQuery(storage.Create(), __VA_ARGS__) #define UNION(...) \ query::test_common::GetCypherUnion(storage.Create(true), \ __VA_ARGS__) #define UNION_ALL(...) \ query::test_common::GetCypherUnion(storage.Create(false), \ __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 AVG(expr) \ storage.Create((expr), nullptr, \ query::Aggregation::Op::AVG) #define COLLECT_LIST(expr) \ storage.Create((expr), nullptr, \ query::Aggregation::Op::COLLECT_LIST) #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)) #define IN_LIST(expr1, expr2) \ storage.Create((expr1), (expr2)) #define IF(cond, then, else) \ storage.Create((cond), (then), (else)) // Function call #define FN(function_name, ...) \ storage.Create( \ 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) #define SINGLE(variable, list, where) \ storage.Create(storage.Create(variable), \ list, where) #define REDUCE(accumulator, initializer, variable, list, expr) \ storage.Create( \ storage.Create(accumulator), initializer, \ storage.Create(variable), list, expr) #define EXTRACT(variable, list, expr) \ storage.Create(storage.Create(variable), \ list, expr) #define AUTH_QUERY(action, user, role, user_or_role, password, privileges) \ storage.Create((action), (user), (role), (user_or_role), \ password, (privileges)) #define DROP_USER(usernames) storage.Create((usernames)) #define CREATE_STREAM(stream_name, stream_uri, stream_topic, transform_uri, \ batch_interval, batch_size) \ storage.Create( \ (stream_name), LITERAL(stream_uri), LITERAL(stream_topic), \ LITERAL(transform_uri), (batch_interval), (batch_size)) #define DROP_STREAM(stream_name) \ storage.Create((stream_name)) #define SHOW_STREAMS storage.Create() #define START_STREAM(stream_name, limit_batches) \ storage.Create((stream_name), true, (limit_batches)) #define STOP_STREAM(stream_name) \ storage.Create((stream_name), false, nullptr) #define START_ALL_STREAMS storage.Create(true) #define STOP_ALL_STREAMS storage.Create(false) #define TEST_STREAM(stream_name, limit_batches) \ storage.Create((stream_name), (limit_batches))