2017-03-24 23:50:42 +08:00
|
|
|
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,
|
|
|
|
GraphDb::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,
|
|
|
|
GraphDb::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,
|
|
|
|
GraphDb::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.
|
|
|
|
///
|
Make Where and NamedExpression macros easier to use
Summary:
`Where` can now be constructed in a `QUERY`, instead of requiring manual
addition to `Match`. For example:
auto query = QUERY(MATCH(pattern), WHERE(expr), ...);
compared to:
auto match = MATCH(pattern);
match->where_ = WHERE(expr);
auto query = QUERY(match, ...);
Similarly, `AS` can be used instead of `NEXPR` to create
`NamedExpressions` only with a name. This is meant to be used with
`RETURN` which will look at the previous `Expression` and store it
inside `NamedExpression`. For example:
auto ret = RETURN(IDENT("n"), AS("n"),
PROPERTY_LOOKUP("n", prop), AS("prop_val"));
compared to:
auto ret = RETURN(NEXPR("n", IDENT("n")),
NEXPR("prop_val", PROPERTY_LOOKUP("n", prop)));
Reviewers: florijan, mislav.bradac
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D195
2017-03-28 21:39:18 +08:00
|
|
|
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...);
|
2017-03-24 23:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Create the return clause with given named expressions.
|
|
|
|
///
|
Make Where and NamedExpression macros easier to use
Summary:
`Where` can now be constructed in a `QUERY`, instead of requiring manual
addition to `Match`. For example:
auto query = QUERY(MATCH(pattern), WHERE(expr), ...);
compared to:
auto match = MATCH(pattern);
match->where_ = WHERE(expr);
auto query = QUERY(match, ...);
Similarly, `AS` can be used instead of `NEXPR` to create
`NamedExpressions` only with a name. This is meant to be used with
`RETURN` which will look at the previous `Expression` and store it
inside `NamedExpression`. For example:
auto ret = RETURN(IDENT("n"), AS("n"),
PROPERTY_LOOKUP("n", prop), AS("prop_val"));
compared to:
auto ret = RETURN(NEXPR("n", IDENT("n")),
NEXPR("prop_val", PROPERTY_LOOKUP("n", prop)));
Reviewers: florijan, mislav.bradac
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D195
2017-03-28 21:39:18 +08:00
|
|
|
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);
|
2017-03-24 23:50:42 +08:00
|
|
|
return ret;
|
|
|
|
}
|
Make Where and NamedExpression macros easier to use
Summary:
`Where` can now be constructed in a `QUERY`, instead of requiring manual
addition to `Match`. For example:
auto query = QUERY(MATCH(pattern), WHERE(expr), ...);
compared to:
auto match = MATCH(pattern);
match->where_ = WHERE(expr);
auto query = QUERY(match, ...);
Similarly, `AS` can be used instead of `NEXPR` to create
`NamedExpressions` only with a name. This is meant to be used with
`RETURN` which will look at the previous `Expression` and store it
inside `NamedExpression`. For example:
auto ret = RETURN(IDENT("n"), AS("n"),
PROPERTY_LOOKUP("n", prop), AS("prop_val"));
compared to:
auto ret = RETURN(NEXPR("n", IDENT("n")),
NEXPR("prop_val", PROPERTY_LOOKUP("n", prop)));
Reviewers: florijan, mislav.bradac
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D195
2017-03-28 21:39:18 +08:00
|
|
|
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...);
|
|
|
|
}
|
2017-03-24 23:50:42 +08:00
|
|
|
|
2017-03-27 20:23:31 +08:00
|
|
|
///
|
|
|
|
/// 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;
|
|
|
|
}
|
|
|
|
|
2017-03-28 17:04:28 +08:00
|
|
|
///
|
|
|
|
/// 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<GraphDb::Label> labels) {
|
|
|
|
return storage.Create<SetLabels>(storage.Create<Identifier>(name), labels);
|
|
|
|
}
|
|
|
|
|
2017-03-24 23:50:42 +08:00
|
|
|
} // namespace test_common
|
|
|
|
|
|
|
|
} // namespace query
|
|
|
|
|
|
|
|
///
|
|
|
|
/// All the following macros implicitly pass `storage` variable to functions.
|
2017-03-27 18:10:50 +08:00
|
|
|
/// You need to have `AstTreeStorage storage;` somewhere in scope to use them.
|
2017-03-24 23:50:42 +08:00
|
|
|
/// 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(...) \
|
2017-03-27 19:09:14 +08:00
|
|
|
query::test_common::GetWithPatterns<query::Match>(storage, {__VA_ARGS__})
|
2017-03-27 18:10:50 +08:00
|
|
|
#define WHERE(expr) storage.Create<query::Where>((expr))
|
2017-03-24 23:50:42 +08:00
|
|
|
#define CREATE(...) \
|
2017-03-27 19:09:14 +08:00
|
|
|
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))
|
2017-03-27 18:10:50 +08:00
|
|
|
#define PROPERTY_LOOKUP(...) \
|
|
|
|
query::test_common::GetPropertyLookup(storage, __VA_ARGS__)
|
2017-03-27 19:09:14 +08:00
|
|
|
#define NEXPR(name, expr) storage.Create<query::NamedExpression>((name), (expr))
|
Make Where and NamedExpression macros easier to use
Summary:
`Where` can now be constructed in a `QUERY`, instead of requiring manual
addition to `Match`. For example:
auto query = QUERY(MATCH(pattern), WHERE(expr), ...);
compared to:
auto match = MATCH(pattern);
match->where_ = WHERE(expr);
auto query = QUERY(match, ...);
Similarly, `AS` can be used instead of `NEXPR` to create
`NamedExpressions` only with a name. This is meant to be used with
`RETURN` which will look at the previous `Expression` and store it
inside `NamedExpression`. For example:
auto ret = RETURN(IDENT("n"), AS("n"),
PROPERTY_LOOKUP("n", prop), AS("prop_val"));
compared to:
auto ret = RETURN(NEXPR("n", IDENT("n")),
NEXPR("prop_val", PROPERTY_LOOKUP("n", prop)));
Reviewers: florijan, mislav.bradac
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D195
2017-03-28 21:39:18 +08:00
|
|
|
// 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__)
|
2017-03-27 20:23:31 +08:00
|
|
|
#define DELETE(...) query::test_common::GetDelete(storage, {__VA_ARGS__})
|
|
|
|
#define DETACH_DELETE(...) \
|
|
|
|
query::test_common::GetDelete(storage, {__VA_ARGS__}, true)
|
2017-03-28 17:04:28 +08:00
|
|
|
#define SET(...) query::test_common::GetSet(storage, __VA_ARGS__)
|
Make Where and NamedExpression macros easier to use
Summary:
`Where` can now be constructed in a `QUERY`, instead of requiring manual
addition to `Match`. For example:
auto query = QUERY(MATCH(pattern), WHERE(expr), ...);
compared to:
auto match = MATCH(pattern);
match->where_ = WHERE(expr);
auto query = QUERY(match, ...);
Similarly, `AS` can be used instead of `NEXPR` to create
`NamedExpressions` only with a name. This is meant to be used with
`RETURN` which will look at the previous `Expression` and store it
inside `NamedExpression`. For example:
auto ret = RETURN(IDENT("n"), AS("n"),
PROPERTY_LOOKUP("n", prop), AS("prop_val"));
compared to:
auto ret = RETURN(NEXPR("n", IDENT("n")),
NEXPR("prop_val", PROPERTY_LOOKUP("n", prop)));
Reviewers: florijan, mislav.bradac
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D195
2017-03-28 21:39:18 +08:00
|
|
|
#define QUERY(...) query::test_common::GetQuery(storage, __VA_ARGS__)
|
2017-03-27 18:10:50 +08:00
|
|
|
#define LESS(expr1, expr2) storage.Create<query::LessOperator>((expr1), (expr2))
|