#include "database/graph_db_datatypes.hpp" namespace query { namespace test_common { /// /// Create PropertyLookup with given name and property. /// /// Name is used to create the Identifier which is used for property lookup. /// auto GetPropertyLookup(AstTreeStorage &storage, const std::string &name, GraphDbTypes::Property property) { return storage.Create(storage.Create(name), property); } /// /// 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; } /// /// Create a Pattern with given atoms. /// auto GetPattern(AstTreeStorage &storage, std::vector atoms) { auto pattern = storage.Create(); pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end()); return pattern; } /// /// This function creates an AST node which can store patterns and fills them /// with given patterns. /// /// The function is most commonly used to create Match and Create clauses. /// template auto GetWithPatterns(AstTreeStorage &storage, std::vector patterns) { auto with_patterns = storage.Create(); 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(); } template auto GetQuery(AstTreeStorage &storage, Clause *clause, T *... clauses) { storage.query()->clauses_.emplace_back(clause); return GetQuery(storage, clauses...); } template auto GetQuery(AstTreeStorage &storage, Match *match, Where *where, T *... clauses) { match->where_ = where; storage.query()->clauses_.emplace_back(match); return GetQuery(storage, clauses...); } /// /// Create the return clause with given named expressions. /// auto GetReturn(Return *ret, NamedExpression *named_expr) { ret->named_expressions_.emplace_back(named_expr); return ret; } auto GetReturn(Return *ret, Expression *expr, NamedExpression *named_expr) { // This overload supports `RETURN(expr, AS(name))` construct, since // NamedExpression does not inherit Expression. named_expr->expression_ = expr; ret->named_expressions_.emplace_back(named_expr); return ret; } template auto GetReturn(Return *ret, Expression *expr, NamedExpression *named_expr, T *... rest) { named_expr->expression_ = expr; ret->named_expressions_.emplace_back(named_expr); return GetReturn(ret, rest...); } template auto GetReturn(Return *ret, NamedExpression *named_expr, T *... rest) { ret->named_expressions_.emplace_back(named_expr); return GetReturn(ret, rest...); } template auto GetReturn(AstTreeStorage &storage, T *... exprs) { auto ret = storage.Create(); return GetReturn(ret, exprs...); } /// /// 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); } } // 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 MATCH(...) \ query::test_common::GetWithPatterns(storage, {__VA_ARGS__}) #define WHERE(expr) storage.Create((expr)) #define CREATE(...) \ query::test_common::GetWithPatterns(storage, {__VA_ARGS__}) #define IDENT(name) storage.Create((name)) #define LITERAL(val) storage.Create((val)) #define PROPERTY_LOOKUP(...) \ query::test_common::GetPropertyLookup(storage, __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. 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, __VA_ARGS__) #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 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))