05d19f6218
Summary: Replace includes in operator.hpp with forward declarations. Provide basic documentation for remaining operators. Use SwitchOld on accessors used in Expand and Filter operations. Add SwitchOld and SwitchNew to ExpressionEvaluator. Evaluate new or old state in operators. Test operators use correct accessors. Add some basic tests for cases where switching accessors to old and new matters. Reviewers: mislav.bradac, buda, florijan Reviewed By: florijan Subscribers: pullbot Differential Revision: https://phabricator.memgraph.io/D208
223 lines
7.8 KiB
C++
223 lines
7.8 KiB
C++
#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<PropertyLookup>(storage.Create<Identifier>(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<EdgeAtom>(storage.Create<Identifier>(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<NodeAtom>(storage.Create<Identifier>(name));
|
|
if (label) node->labels_.emplace_back(label);
|
|
return node;
|
|
}
|
|
|
|
///
|
|
/// Create a Pattern with given atoms.
|
|
///
|
|
auto GetPattern(AstTreeStorage &storage, std::vector<PatternAtom *> atoms) {
|
|
auto pattern = storage.Create<Pattern>();
|
|
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 <class TWithPatterns>
|
|
auto GetWithPatterns(AstTreeStorage &storage, std::vector<Pattern *> patterns) {
|
|
auto with_patterns = storage.Create<TWithPatterns>();
|
|
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 <class... T>
|
|
auto GetQuery(AstTreeStorage &storage, Clause *clause, T *... clauses) {
|
|
storage.query()->clauses_.emplace_back(clause);
|
|
return GetQuery(storage, clauses...);
|
|
}
|
|
template <class... T>
|
|
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 <class... T>
|
|
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 <class... T>
|
|
auto GetReturn(Return *ret, NamedExpression *named_expr, T *... rest) {
|
|
ret->named_expressions_.emplace_back(named_expr);
|
|
return GetReturn(ret, rest...);
|
|
}
|
|
template <class... T>
|
|
auto GetReturn(AstTreeStorage &storage, T *... exprs) {
|
|
auto ret = storage.Create<Return>();
|
|
return GetReturn(ret, exprs...);
|
|
}
|
|
|
|
///
|
|
/// Create the delete clause with given named expressions.
|
|
///
|
|
auto GetDelete(AstTreeStorage &storage, std::vector<Expression *> exprs,
|
|
bool detach = false) {
|
|
auto del = storage.Create<Delete>();
|
|
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<SetProperty>(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<SetProperties>(storage.Create<Identifier>(name), expr,
|
|
update);
|
|
}
|
|
|
|
///
|
|
/// Create a set labels clause for given identifier name and labels.
|
|
///
|
|
auto GetSet(AstTreeStorage &storage, const std::string &name,
|
|
std::vector<GraphDbTypes::Label> labels) {
|
|
return storage.Create<SetLabels>(storage.Create<Identifier>(name), labels);
|
|
}
|
|
|
|
///
|
|
/// Create a remove property clause for given property lookup
|
|
///
|
|
auto GetRemove(AstTreeStorage &storage, PropertyLookup *prop_lookup) {
|
|
return storage.Create<RemoveProperty>(prop_lookup);
|
|
}
|
|
|
|
///
|
|
/// Create a remove labels clause for given identifier name and labels.
|
|
///
|
|
auto GetRemove(AstTreeStorage &storage, const std::string &name,
|
|
std::vector<GraphDbTypes::Label> labels) {
|
|
return storage.Create<RemoveLabels>(storage.Create<Identifier>(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<query::Match>(storage, {__VA_ARGS__})
|
|
#define WHERE(expr) storage.Create<query::Where>((expr))
|
|
#define CREATE(...) \
|
|
query::test_common::GetWithPatterns<query::Create>(storage, {__VA_ARGS__})
|
|
#define IDENT(name) storage.Create<query::Identifier>((name))
|
|
#define LITERAL(val) storage.Create<query::Literal>((val))
|
|
#define PROPERTY_LOOKUP(...) \
|
|
query::test_common::GetPropertyLookup(storage, __VA_ARGS__)
|
|
#define NEXPR(name, expr) storage.Create<query::NamedExpression>((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<query::NamedExpression>((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<query::AdditionOperator>((expr1), (expr2))
|
|
#define LESS(expr1, expr2) storage.Create<query::LessOperator>((expr1), (expr2))
|