2017-04-24 19:51:16 +08:00
|
|
|
/// @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:
|
|
|
|
///
|
2018-05-22 22:45:52 +08:00
|
|
|
/// AstStorage storage; // Macros rely on storage being in scope.
|
2018-06-19 19:52:11 +08:00
|
|
|
/// // PROPERTY_LOOKUP and PROPERTY_PAIR macros
|
2018-07-26 15:08:21 +08:00
|
|
|
/// // rely on a DbAccessor *reference* named dba.
|
2018-10-09 17:09:10 +08:00
|
|
|
/// database::GraphDb db;
|
2018-07-26 15:08:21 +08:00
|
|
|
/// auto dba_ptr = db.Access();
|
|
|
|
/// auto &dba = *dba_ptr;
|
2017-04-24 19:51:16 +08:00
|
|
|
///
|
|
|
|
/// 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.
|
|
|
|
|
2017-05-30 21:19:38 +08:00
|
|
|
#pragma once
|
|
|
|
|
2017-09-18 20:40:36 +08:00
|
|
|
#include <map>
|
2018-11-07 17:54:35 +08:00
|
|
|
#include <sstream>
|
2017-09-18 20:40:36 +08:00
|
|
|
#include <string>
|
2017-10-02 16:34:38 +08:00
|
|
|
#include <unordered_map>
|
2017-04-24 19:51:16 +08:00
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
2017-04-24 22:01:00 +08:00
|
|
|
#include "query/frontend/ast/ast.hpp"
|
2018-11-07 17:54:35 +08:00
|
|
|
#include "query/frontend/ast/pretty_print.hpp"
|
2019-11-22 00:24:01 +08:00
|
|
|
#include "storage/v2/id_types.hpp"
|
2019-11-22 01:38:01 +08:00
|
|
|
#include "utils/string.hpp"
|
2019-11-22 00:24:01 +08:00
|
|
|
|
2017-03-24 23:50:42 +08:00
|
|
|
namespace query {
|
|
|
|
|
|
|
|
namespace test_common {
|
|
|
|
|
2019-08-22 20:50:57 +08:00
|
|
|
auto ToIntList(const TypedValue &t) {
|
|
|
|
std::vector<int64_t> list;
|
2019-05-21 22:06:05 +08:00
|
|
|
for (auto x : t.ValueList()) {
|
2019-08-22 20:50:57 +08:00
|
|
|
list.push_back(x.ValueInt());
|
2017-05-19 21:49:25 +08:00
|
|
|
}
|
|
|
|
return list;
|
|
|
|
};
|
|
|
|
|
2019-08-22 20:50:57 +08:00
|
|
|
auto ToIntMap(const TypedValue &t) {
|
|
|
|
std::map<std::string, int64_t> map;
|
2021-02-18 22:32:43 +08:00
|
|
|
for (const auto &kv : t.ValueMap()) map.emplace(kv.first, kv.second.ValueInt());
|
Collect Map added
Summary:
Tests are on the way. Please first comment if you're OK with this implementation, some points are discussable.
What works now:
```
bash:MEMGRAPH_ROOT/build/>./tests/manual/console 10
MG>MATCH (n) RETURN COLLECT("age_" + n.age, n.height)
+-----------------------------------------------------------------------------------------------------------------------------------+
| COLLECT("age_" + n.age, n.height) |
+-----------------------------------------------------------------------------------------------------------------------------------+
| {age_10: 176, age_13: 180, age_24: 172, age_25: 179, age_32: 123, age_33: 186, age_37: 147, age_43: 162, age_49: 126, age_6: 170} |
+-----------------------------------------------------------------------------------------------------------------------------------+
```
Reviewers: mislav.bradac, teon.banek, buda
Reviewed By: mislav.bradac, buda
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D695
2017-08-23 16:43:45 +08:00
|
|
|
return map;
|
|
|
|
};
|
|
|
|
|
2019-01-16 17:43:32 +08:00
|
|
|
std::string ToString(Expression *expr) {
|
2018-11-07 17:54:35 +08:00
|
|
|
std::ostringstream ss;
|
2019-01-16 17:43:32 +08:00
|
|
|
PrintExpression(expr, &ss);
|
2018-11-07 17:54:35 +08:00
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2019-01-16 17:43:32 +08:00
|
|
|
std::string ToString(NamedExpression *expr) {
|
2018-11-07 17:54:35 +08:00
|
|
|
std::ostringstream ss;
|
2019-01-16 17:43:32 +08:00
|
|
|
PrintExpression(expr, &ss);
|
2018-11-07 17:54:35 +08:00
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
2017-04-26 19:49:41 +08:00
|
|
|
// Custom types for ORDER BY, SKIP, LIMIT, ON MATCH and ON CREATE expressions,
|
|
|
|
// so that they can be used to resolve function calls.
|
2017-04-24 19:51:16 +08:00
|
|
|
struct OrderBy {
|
2018-10-04 17:57:23 +08:00
|
|
|
std::vector<SortItem> expressions;
|
2017-04-24 19:51:16 +08:00
|
|
|
};
|
2017-04-20 17:20:20 +08:00
|
|
|
struct Skip {
|
2017-04-24 19:51:16 +08:00
|
|
|
Expression *expression = nullptr;
|
2017-04-20 17:20:20 +08:00
|
|
|
};
|
|
|
|
struct Limit {
|
2017-04-24 19:51:16 +08:00
|
|
|
Expression *expression = nullptr;
|
2017-04-20 17:20:20 +08:00
|
|
|
};
|
2017-04-26 19:49:41 +08:00
|
|
|
struct OnMatch {
|
|
|
|
std::vector<Clause *> set;
|
|
|
|
};
|
|
|
|
struct OnCreate {
|
|
|
|
std::vector<Clause *> set;
|
|
|
|
};
|
2017-04-20 17:20:20 +08:00
|
|
|
|
2017-04-24 19:51:16 +08:00
|
|
|
// Helper functions for filling the OrderBy with expressions.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto FillOrderBy(OrderBy &order_by, Expression *expression, Ordering ordering = Ordering::ASC) {
|
2018-10-04 17:57:23 +08:00
|
|
|
order_by.expressions.push_back({ordering, expression});
|
2017-04-24 19:51:16 +08:00
|
|
|
}
|
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto FillOrderBy(OrderBy &order_by, Expression *expression, Ordering ordering, T... rest) {
|
2017-04-24 19:51:16 +08:00
|
|
|
FillOrderBy(order_by, expression, ordering);
|
|
|
|
FillOrderBy(order_by, rest...);
|
|
|
|
}
|
|
|
|
template <class... T>
|
|
|
|
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 <class... T>
|
|
|
|
auto GetOrderBy(T... exprs) {
|
|
|
|
OrderBy order_by;
|
|
|
|
FillOrderBy(order_by, exprs...);
|
|
|
|
return order_by;
|
|
|
|
}
|
|
|
|
|
2017-03-24 23:50:42 +08:00
|
|
|
/// Create PropertyLookup with given name and property.
|
|
|
|
///
|
|
|
|
/// Name is used to create the Identifier which is used for property lookup.
|
2019-11-22 00:24:01 +08:00
|
|
|
template <class TDbAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPropertyLookup(AstStorage &storage, TDbAccessor &dba, const std::string &name, storage::PropertyId property) {
|
|
|
|
return storage.Create<PropertyLookup>(storage.Create<Identifier>(name),
|
|
|
|
storage.GetPropertyIx(dba.PropertyToName(property)));
|
2019-11-22 00:24:01 +08:00
|
|
|
}
|
2018-06-19 19:52:11 +08:00
|
|
|
|
2019-11-22 00:24:01 +08:00
|
|
|
template <class TDbAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPropertyLookup(AstStorage &storage, TDbAccessor &dba, Expression *expr, storage::PropertyId property) {
|
|
|
|
return storage.Create<PropertyLookup>(expr, storage.GetPropertyIx(dba.PropertyToName(property)));
|
2019-11-22 00:24:01 +08:00
|
|
|
}
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
|
|
|
|
template <class TDbAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPropertyLookup(AstStorage &storage, TDbAccessor &dba, Expression *expr, const std::string &property) {
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
return storage.Create<PropertyLookup>(expr, storage.GetPropertyIx(property));
|
2017-08-08 19:43:42 +08:00
|
|
|
}
|
2018-06-19 19:52:11 +08:00
|
|
|
|
|
|
|
template <class TDbAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPropertyLookup(AstStorage &storage, TDbAccessor &, const std::string &name,
|
|
|
|
const std::pair<std::string, storage::PropertyId> &prop_pair) {
|
|
|
|
return storage.Create<PropertyLookup>(storage.Create<Identifier>(name), storage.GetPropertyIx(prop_pair.first));
|
2017-08-08 19:43:42 +08:00
|
|
|
}
|
2018-06-19 19:52:11 +08:00
|
|
|
|
|
|
|
template <class TDbAccessor>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPropertyLookup(AstStorage &storage, TDbAccessor &, Expression *expr,
|
|
|
|
const std::pair<std::string, storage::PropertyId> &prop_pair) {
|
|
|
|
return storage.Create<PropertyLookup>(expr, storage.GetPropertyIx(prop_pair.first));
|
2017-05-05 21:34:07 +08:00
|
|
|
}
|
2017-03-24 23:50:42 +08:00
|
|
|
|
2017-10-05 17:25:52 +08:00
|
|
|
/// Create an EdgeAtom with given name, direction and edge_type.
|
2017-03-24 23:50:42 +08:00
|
|
|
///
|
|
|
|
/// Name is used to create the Identifier which is assigned to the edge.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetEdge(AstStorage &storage, const std::string &name, EdgeAtom::Direction dir = EdgeAtom::Direction::BOTH,
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
const std::vector<std::string> &edge_types = {}) {
|
|
|
|
std::vector<EdgeTypeIx> types;
|
|
|
|
types.reserve(edge_types.size());
|
|
|
|
for (const auto &type : edge_types) {
|
|
|
|
types.push_back(storage.GetEdgeTypeIx(type));
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
return storage.Create<EdgeAtom>(storage.Create<Identifier>(name), EdgeAtom::Type::SINGLE, dir, types);
|
2017-10-05 17:25:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a variable length expansion EdgeAtom with given name, direction and
|
|
|
|
/// edge_type.
|
2017-03-24 23:50:42 +08:00
|
|
|
///
|
2017-10-05 17:25:52 +08:00
|
|
|
/// Name is used to create the Identifier which is assigned to the edge.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetEdgeVariable(AstStorage &storage, const std::string &name, EdgeAtom::Type type = EdgeAtom::Type::DEPTH_FIRST,
|
2017-10-05 17:25:52 +08:00
|
|
|
EdgeAtom::Direction dir = EdgeAtom::Direction::BOTH,
|
2021-02-18 22:32:43 +08:00
|
|
|
const std::vector<std::string> &edge_types = {}, Identifier *flambda_inner_edge = nullptr,
|
|
|
|
Identifier *flambda_inner_node = nullptr, Identifier *wlambda_inner_edge = nullptr,
|
|
|
|
Identifier *wlambda_inner_node = nullptr, Expression *wlambda_expression = nullptr,
|
2018-10-18 17:16:32 +08:00
|
|
|
Identifier *total_weight = nullptr) {
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
std::vector<EdgeTypeIx> types;
|
|
|
|
types.reserve(edge_types.size());
|
|
|
|
for (const auto &type : edge_types) {
|
|
|
|
types.push_back(storage.GetEdgeTypeIx(type));
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
auto r_val = storage.Create<EdgeAtom>(storage.Create<Identifier>(name), type, dir, types);
|
2018-10-18 17:16:32 +08:00
|
|
|
|
2018-02-08 19:57:12 +08:00
|
|
|
r_val->filter_lambda_.inner_edge =
|
2021-02-18 22:32:43 +08:00
|
|
|
flambda_inner_edge ? flambda_inner_edge : storage.Create<Identifier>(utils::RandomString(20));
|
2018-02-08 19:57:12 +08:00
|
|
|
r_val->filter_lambda_.inner_node =
|
2021-02-18 22:32:43 +08:00
|
|
|
flambda_inner_node ? flambda_inner_node : storage.Create<Identifier>(utils::RandomString(20));
|
2018-10-18 17:16:32 +08:00
|
|
|
|
|
|
|
if (type == EdgeAtom::Type::WEIGHTED_SHORTEST_PATH) {
|
|
|
|
r_val->weight_lambda_.inner_edge =
|
2021-02-18 22:32:43 +08:00
|
|
|
wlambda_inner_edge ? wlambda_inner_edge : storage.Create<Identifier>(utils::RandomString(20));
|
2018-10-18 17:16:32 +08:00
|
|
|
r_val->weight_lambda_.inner_node =
|
2021-02-18 22:32:43 +08:00
|
|
|
wlambda_inner_node ? wlambda_inner_node : storage.Create<Identifier>(utils::RandomString(20));
|
2018-10-18 17:16:32 +08:00
|
|
|
r_val->weight_lambda_.expression =
|
2021-02-18 22:32:43 +08:00
|
|
|
wlambda_expression ? wlambda_expression : storage.Create<query::PrimitiveLiteral>(1);
|
2018-10-18 17:16:32 +08:00
|
|
|
|
|
|
|
r_val->total_weight_ = total_weight;
|
|
|
|
}
|
|
|
|
|
2017-10-05 17:25:52 +08:00
|
|
|
return r_val;
|
2017-03-24 23:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a NodeAtom with given name and label.
|
|
|
|
///
|
|
|
|
/// Name is used to create the Identifier which is assigned to the node.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetNode(AstStorage &storage, const std::string &name, std::optional<std::string> label = std::nullopt) {
|
2017-03-24 23:50:42 +08:00
|
|
|
auto node = storage.Create<NodeAtom>(storage.Create<Identifier>(name));
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
if (label) node->labels_.emplace_back(storage.GetLabelIx(*label));
|
2017-03-24 23:50:42 +08:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a Pattern with given atoms.
|
2018-05-22 22:45:52 +08:00
|
|
|
auto GetPattern(AstStorage &storage, std::vector<PatternAtom *> atoms) {
|
2017-03-24 23:50:42 +08:00
|
|
|
auto pattern = storage.Create<Pattern>();
|
2021-02-18 22:32:43 +08:00
|
|
|
pattern->identifier_ = storage.Create<Identifier>(utils::RandomString(20), false);
|
2017-09-18 20:40:36 +08:00
|
|
|
pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end());
|
|
|
|
return pattern;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a Pattern with given name and atoms.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetPattern(AstStorage &storage, const std::string &name, std::vector<PatternAtom *> atoms) {
|
2017-09-18 20:40:36 +08:00
|
|
|
auto pattern = storage.Create<Pattern>();
|
|
|
|
pattern->identifier_ = storage.Create<Identifier>(name, true);
|
2017-03-24 23:50:42 +08:00
|
|
|
pattern->atoms_.insert(pattern->atoms_.begin(), atoms.begin(), atoms.end());
|
|
|
|
return pattern;
|
|
|
|
}
|
|
|
|
|
2017-04-28 16:37:49 +08:00
|
|
|
/// This function fills an AST node which with given patterns.
|
2017-03-24 23:50:42 +08:00
|
|
|
///
|
|
|
|
/// The function is most commonly used to create Match and Create clauses.
|
|
|
|
template <class TWithPatterns>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetWithPatterns(TWithPatterns *with_patterns, std::vector<Pattern *> patterns) {
|
|
|
|
with_patterns->patterns_.insert(with_patterns->patterns_.begin(), patterns.begin(), patterns.end());
|
2017-03-24 23:50:42 +08:00
|
|
|
return with_patterns;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a query with given clauses.
|
2017-11-29 20:55:02 +08:00
|
|
|
|
|
|
|
auto GetSingleQuery(SingleQuery *single_query, Clause *clause) {
|
|
|
|
single_query->clauses_.emplace_back(clause);
|
|
|
|
return single_query;
|
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
|
|
|
}
|
2017-11-29 20:55:02 +08:00
|
|
|
auto GetSingleQuery(SingleQuery *single_query, Match *match, Where *where) {
|
2017-04-05 20:19:14 +08:00
|
|
|
match->where_ = where;
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(match);
|
|
|
|
return single_query;
|
2017-04-05 20:19:14 +08:00
|
|
|
}
|
2017-11-29 20:55:02 +08:00
|
|
|
auto GetSingleQuery(SingleQuery *single_query, With *with, Where *where) {
|
2017-04-05 20:19:14 +08:00
|
|
|
with->where_ = where;
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(with);
|
|
|
|
return single_query;
|
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>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSingleQuery(SingleQuery *single_query, Match *match, Where *where, T *...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
|
|
|
match->where_ = where;
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(match);
|
|
|
|
return GetSingleQuery(single_query, clauses...);
|
2017-03-24 23:50:42 +08:00
|
|
|
}
|
2017-04-05 20:19:14 +08:00
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSingleQuery(SingleQuery *single_query, With *with, Where *where, T *...clauses) {
|
2017-04-05 20:19:14 +08:00
|
|
|
with->where_ = where;
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(with);
|
|
|
|
return GetSingleQuery(single_query, clauses...);
|
2017-04-05 20:19:14 +08:00
|
|
|
}
|
2017-11-29 20:55:02 +08:00
|
|
|
|
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSingleQuery(SingleQuery *single_query, Clause *clause, T *...clauses) {
|
2017-11-29 20:55:02 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-05-22 22:45:52 +08:00
|
|
|
auto GetQuery(AstStorage &storage, SingleQuery *single_query) {
|
2018-10-19 22:18:44 +08:00
|
|
|
auto *query = storage.Create<CypherQuery>();
|
2018-10-10 21:19:34 +08:00
|
|
|
query->single_query_ = single_query;
|
|
|
|
return query;
|
2017-11-29 20:55:02 +08:00
|
|
|
}
|
|
|
|
|
2017-04-05 20:19:14 +08:00
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetQuery(AstStorage &storage, SingleQuery *single_query, T *...cypher_unions) {
|
2018-10-19 22:18:44 +08:00
|
|
|
auto *query = storage.Create<CypherQuery>();
|
2018-10-10 21:19:34 +08:00
|
|
|
query->single_query_ = single_query;
|
|
|
|
query->cypher_unions_ = std::vector<CypherUnion *>{cypher_unions...};
|
|
|
|
return query;
|
2017-04-05 20:19:14 +08:00
|
|
|
}
|
2017-03-24 23:50:42 +08:00
|
|
|
|
2017-04-24 19:51:16 +08:00
|
|
|
// Helper functions for constructing RETURN and WITH clauses.
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, NamedExpression *named_expr) {
|
2017-05-31 20:00:30 +08:00
|
|
|
body.named_expressions.emplace_back(named_expr);
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, const std::string &name) {
|
2018-10-18 17:16:32 +08:00
|
|
|
if (name == "*") {
|
|
|
|
body.all_identifiers = true;
|
|
|
|
} else {
|
|
|
|
auto *ident = storage.Create<query::Identifier>(name);
|
|
|
|
auto *named_expr = storage.Create<query::NamedExpression>(name, ident);
|
|
|
|
body.named_expressions.emplace_back(named_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
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, Limit limit) { body.limit = limit.expression; }
|
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, Skip skip, Limit limit = Limit{}) {
|
2017-04-24 19:51:16 +08:00
|
|
|
body.skip = skip.expression;
|
|
|
|
body.limit = limit.expression;
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, OrderBy order_by, Limit limit = Limit{}) {
|
2017-04-24 19:51:16 +08:00
|
|
|
body.order_by = order_by.expressions;
|
|
|
|
body.limit = limit.expression;
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, OrderBy order_by, Skip skip, Limit limit = Limit{}) {
|
2017-04-24 19:51:16 +08:00
|
|
|
body.order_by = order_by.expressions;
|
|
|
|
body.skip = skip.expression;
|
|
|
|
body.limit = limit.expression;
|
2017-04-20 17:20:20 +08:00
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &, ReturnBody &body, Expression *expr, NamedExpression *named_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
|
|
|
// This overload supports `RETURN(expr, AS(name))` construct, since
|
|
|
|
// NamedExpression does not inherit Expression.
|
|
|
|
named_expr->expression_ = expr;
|
2017-04-24 19:51:16 +08:00
|
|
|
body.named_expressions.emplace_back(named_expr);
|
2017-03-24 23:50:42 +08:00
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, const std::string &name, NamedExpression *named_expr) {
|
2017-05-31 20:00:30 +08:00
|
|
|
named_expr->expression_ = storage.Create<query::Identifier>(name);
|
|
|
|
body.named_expressions.emplace_back(named_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
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, Expression *expr, NamedExpression *named_expr, T... rest) {
|
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
|
|
|
named_expr->expression_ = expr;
|
2017-04-24 19:51:16 +08:00
|
|
|
body.named_expressions.emplace_back(named_expr);
|
2017-05-31 20:00:30 +08:00
|
|
|
FillReturnBody(storage, body, rest...);
|
|
|
|
}
|
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, NamedExpression *named_expr, T... rest) {
|
2017-05-31 20:00:30 +08:00
|
|
|
body.named_expressions.emplace_back(named_expr);
|
|
|
|
FillReturnBody(storage, body, rest...);
|
|
|
|
}
|
|
|
|
template <class... T>
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, const std::string &name, NamedExpression *named_expr,
|
2017-05-31 20:00:30 +08:00
|
|
|
T... rest) {
|
|
|
|
named_expr->expression_ = storage.Create<query::Identifier>(name);
|
|
|
|
body.named_expressions.emplace_back(named_expr);
|
|
|
|
FillReturnBody(storage, body, rest...);
|
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>
|
2021-02-18 22:32:43 +08:00
|
|
|
void FillReturnBody(AstStorage &storage, ReturnBody &body, const std::string &name, T... rest) {
|
2017-05-31 20:00:30 +08:00
|
|
|
auto *ident = storage.Create<query::Identifier>(name);
|
|
|
|
auto *named_expr = storage.Create<query::NamedExpression>(name, ident);
|
2017-04-24 19:51:16 +08:00
|
|
|
body.named_expressions.emplace_back(named_expr);
|
2017-05-31 20:00:30 +08:00
|
|
|
FillReturnBody(storage, body, rest...);
|
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
|
|
|
}
|
2017-04-24 19:51:16 +08:00
|
|
|
|
|
|
|
/// Create the return clause with given expressions.
|
|
|
|
///
|
|
|
|
/// The supported expression combination of arguments is:
|
|
|
|
///
|
2017-05-31 20:00:30 +08:00
|
|
|
/// (String | NamedExpression | (Expression NamedExpression))+
|
|
|
|
/// [OrderBy] [Skip] [Limit]
|
2017-04-24 19:51:16 +08:00
|
|
|
///
|
|
|
|
/// When the pair (Expression NamedExpression) is given, the Expression will be
|
|
|
|
/// moved inside the NamedExpression. This is done, so that the constructs like
|
2017-05-31 20:00:30 +08:00
|
|
|
/// RETURN(expr, AS("name"), ...) are supported. Taking a String is a shorthand
|
|
|
|
/// for RETURN(IDENT(string), AS(string), ....).
|
2017-04-24 19:51:16 +08:00
|
|
|
///
|
|
|
|
/// @sa GetWith
|
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>
|
2018-05-22 22:45:52 +08:00
|
|
|
auto GetReturn(AstStorage &storage, bool distinct, T... exprs) {
|
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 ret = storage.Create<Return>();
|
2017-05-03 20:57:46 +08:00
|
|
|
ret->body_.distinct = distinct;
|
2017-05-31 20:00:30 +08:00
|
|
|
FillReturnBody(storage, ret->body_, exprs...);
|
2017-04-24 19:51:16 +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
|
|
|
}
|
2017-03-24 23:50:42 +08:00
|
|
|
|
2017-04-24 19:51:16 +08:00
|
|
|
/// Create the with clause with given expressions.
|
2017-04-05 20:19:14 +08:00
|
|
|
///
|
2017-04-24 19:51:16 +08:00
|
|
|
/// The supported expression combination is the same as for @c GetReturn.
|
|
|
|
///
|
|
|
|
/// @sa GetReturn
|
2017-04-05 20:19:14 +08:00
|
|
|
template <class... T>
|
2018-05-22 22:45:52 +08:00
|
|
|
auto GetWith(AstStorage &storage, bool distinct, T... exprs) {
|
2017-04-05 20:19:14 +08:00
|
|
|
auto with = storage.Create<With>();
|
2017-05-03 20:57:46 +08:00
|
|
|
with->body_.distinct = distinct;
|
2017-05-31 20:00:30 +08:00
|
|
|
FillReturnBody(storage, with->body_, exprs...);
|
2017-04-24 19:51:16 +08:00
|
|
|
return with;
|
2017-04-05 20:19:14 +08:00
|
|
|
}
|
|
|
|
|
2017-05-02 21:21:04 +08:00
|
|
|
/// Create the UNWIND clause with given named expression.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetUnwind(AstStorage &storage, NamedExpression *named_expr) { return storage.Create<query::Unwind>(named_expr); }
|
2018-05-22 22:45:52 +08:00
|
|
|
auto GetUnwind(AstStorage &storage, Expression *expr, NamedExpression *as) {
|
2017-05-02 21:21:04 +08:00
|
|
|
as->expression_ = expr;
|
|
|
|
return GetUnwind(storage, as);
|
|
|
|
}
|
|
|
|
|
2017-03-27 20:23:31 +08:00
|
|
|
/// Create the delete clause with given named expressions.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetDelete(AstStorage &storage, std::vector<Expression *> exprs, bool detach = false) {
|
2017-03-27 20:23:31 +08:00
|
|
|
auto del = storage.Create<Delete>();
|
2021-02-18 22:32:43 +08:00
|
|
|
del->expressions_.insert(del->expressions_.begin(), exprs.begin(), exprs.end());
|
2017-03-27 20:23:31 +08:00
|
|
|
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.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSet(AstStorage &storage, PropertyLookup *prop_lookup, Expression *expr) {
|
2017-03-28 17:04:28 +08:00
|
|
|
return storage.Create<SetProperty>(prop_lookup, expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a set properties clause for given identifier name and the right hand
|
|
|
|
/// side expression.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSet(AstStorage &storage, const std::string &name, Expression *expr, bool update = false) {
|
|
|
|
return storage.Create<SetProperties>(storage.Create<Identifier>(name), expr, update);
|
2017-03-28 17:04:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a set labels clause for given identifier name and labels.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetSet(AstStorage &storage, const std::string &name, std::vector<std::string> label_names) {
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
std::vector<LabelIx> labels;
|
|
|
|
labels.reserve(label_names.size());
|
|
|
|
for (const auto &label : label_names) {
|
|
|
|
labels.push_back(storage.GetLabelIx(label));
|
|
|
|
}
|
2017-03-28 17:04:28 +08:00
|
|
|
return storage.Create<SetLabels>(storage.Create<Identifier>(name), labels);
|
|
|
|
}
|
|
|
|
|
2017-03-30 14:44:56 +08:00
|
|
|
/// Create a remove property clause for given property lookup
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetRemove(AstStorage &storage, PropertyLookup *prop_lookup) { return storage.Create<RemoveProperty>(prop_lookup); }
|
2017-03-30 14:44:56 +08:00
|
|
|
|
|
|
|
/// Create a remove labels clause for given identifier name and labels.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetRemove(AstStorage &storage, const std::string &name, std::vector<std::string> label_names) {
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
std::vector<LabelIx> labels;
|
|
|
|
labels.reserve(label_names.size());
|
|
|
|
for (const auto &label : label_names) {
|
|
|
|
labels.push_back(storage.GetLabelIx(label));
|
|
|
|
}
|
2017-03-30 14:44:56 +08:00
|
|
|
return storage.Create<RemoveLabels>(storage.Create<Identifier>(name), labels);
|
|
|
|
}
|
|
|
|
|
2017-04-26 19:49:41 +08:00
|
|
|
/// Create a Merge clause for given Pattern with optional OnMatch and OnCreate
|
|
|
|
/// parts.
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetMerge(AstStorage &storage, Pattern *pattern, OnCreate on_create = OnCreate{}) {
|
2017-04-26 19:49:41 +08:00
|
|
|
auto *merge = storage.Create<query::Merge>();
|
|
|
|
merge->pattern_ = pattern;
|
|
|
|
merge->on_create_ = on_create.set;
|
|
|
|
return merge;
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
auto GetMerge(AstStorage &storage, Pattern *pattern, OnMatch on_match, OnCreate on_create = OnCreate{}) {
|
2017-04-26 19:49:41 +08:00
|
|
|
auto *merge = storage.Create<query::Merge>();
|
|
|
|
merge->pattern_ = pattern;
|
|
|
|
merge->on_match_ = on_match.set;
|
|
|
|
merge->on_create_ = on_create.set;
|
|
|
|
return merge;
|
|
|
|
}
|
|
|
|
|
2017-03-24 23:50:42 +08:00
|
|
|
} // namespace test_common
|
|
|
|
|
|
|
|
} // namespace query
|
|
|
|
|
|
|
|
/// All the following macros implicitly pass `storage` variable to functions.
|
2018-05-22 22:45:52 +08:00
|
|
|
/// You need to have `AstStorage 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
|
2018-05-22 22:45:52 +08:00
|
|
|
/// AstStorage storage;
|
2017-03-24 23:50:42 +08:00
|
|
|
/// 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__)
|
2021-02-18 22:32:43 +08:00
|
|
|
#define EDGE_VARIABLE(...) query::test_common::GetEdgeVariable(storage, __VA_ARGS__)
|
2017-03-24 23:50:42 +08:00
|
|
|
#define PATTERN(...) query::test_common::GetPattern(storage, {__VA_ARGS__})
|
2021-02-18 22:32:43 +08:00
|
|
|
#define NAMED_PATTERN(name, ...) query::test_common::GetPattern(storage, name, {__VA_ARGS__})
|
|
|
|
#define OPTIONAL_MATCH(...) query::test_common::GetWithPatterns(storage.Create<query::Match>(true), {__VA_ARGS__})
|
|
|
|
#define MATCH(...) query::test_common::GetWithPatterns(storage.Create<query::Match>(), {__VA_ARGS__})
|
2017-03-27 18:10:50 +08:00
|
|
|
#define WHERE(expr) storage.Create<query::Where>((expr))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define CREATE(...) query::test_common::GetWithPatterns(storage.Create<query::Create>(), {__VA_ARGS__})
|
2017-03-27 19:09:14 +08:00
|
|
|
#define IDENT(name) storage.Create<query::Identifier>((name))
|
2017-04-28 18:06:18 +08:00
|
|
|
#define LITERAL(val) storage.Create<query::PrimitiveLiteral>((val))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define LIST(...) storage.Create<query::ListLiteral>(std::vector<query::Expression *>{__VA_ARGS__})
|
|
|
|
#define MAP(...) \
|
|
|
|
storage.Create<query::MapLiteral>(std::unordered_map<query::PropertyIx, query::Expression *>{__VA_ARGS__})
|
|
|
|
#define PROPERTY_PAIR(property_name) std::make_pair(property_name, dba.NameToProperty(property_name))
|
|
|
|
#define PROPERTY_LOOKUP(...) query::test_common::GetPropertyLookup(storage, dba, __VA_ARGS__)
|
|
|
|
#define PARAMETER_LOOKUP(token_position) storage.Create<query::ParameterLookup>((token_position))
|
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
|
2017-04-05 20:19:14 +08:00
|
|
|
// Expression. It should be used with RETURN or WITH. For example:
|
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
|
|
|
// RETURN(IDENT("n"), AS("n")) vs. RETURN(NEXPR("n", IDENT("n"))).
|
|
|
|
#define AS(name) storage.Create<query::NamedExpression>((name))
|
2017-05-03 20:57:46 +08:00
|
|
|
#define RETURN(...) query::test_common::GetReturn(storage, false, __VA_ARGS__)
|
|
|
|
#define WITH(...) query::test_common::GetWith(storage, false, __VA_ARGS__)
|
2021-02-18 22:32:43 +08:00
|
|
|
#define RETURN_DISTINCT(...) query::test_common::GetReturn(storage, true, __VA_ARGS__)
|
|
|
|
#define WITH_DISTINCT(...) query::test_common::GetWith(storage, true, __VA_ARGS__)
|
2017-05-02 21:21:04 +08:00
|
|
|
#define UNWIND(...) query::test_common::GetUnwind(storage, __VA_ARGS__)
|
2017-04-24 19:51:16 +08:00
|
|
|
#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) }
|
2017-03-27 20:23:31 +08:00
|
|
|
#define DELETE(...) query::test_common::GetDelete(storage, {__VA_ARGS__})
|
2021-02-18 22:32:43 +08:00
|
|
|
#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__)
|
2017-03-30 14:44:56 +08:00
|
|
|
#define REMOVE(...) query::test_common::GetRemove(storage, __VA_ARGS__)
|
2017-04-26 19:49:41 +08:00
|
|
|
#define MERGE(...) query::test_common::GetMerge(storage, __VA_ARGS__)
|
|
|
|
#define ON_MATCH(...) \
|
|
|
|
query::test_common::OnMatch { \
|
|
|
|
std::vector<query::Clause *> { __VA_ARGS__ } \
|
|
|
|
}
|
|
|
|
#define ON_CREATE(...) \
|
|
|
|
query::test_common::OnCreate { \
|
|
|
|
std::vector<query::Clause *> { __VA_ARGS__ } \
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
#define CREATE_INDEX_ON(label, property) \
|
|
|
|
storage.Create<query::IndexQuery>(query::IndexQuery::Action::CREATE, (label), \
|
|
|
|
std::vector<query::PropertyIx>{(property)})
|
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__)
|
2021-02-18 22:32:43 +08:00
|
|
|
#define SINGLE_QUERY(...) query::test_common::GetSingleQuery(storage.Create<SingleQuery>(), __VA_ARGS__)
|
|
|
|
#define UNION(...) query::test_common::GetCypherUnion(storage.Create<CypherUnion>(true), __VA_ARGS__)
|
|
|
|
#define UNION_ALL(...) query::test_common::GetCypherUnion(storage.Create<CypherUnion>(false), __VA_ARGS__)
|
2017-04-03 20:32:29 +08:00
|
|
|
// Various operators
|
2018-11-07 17:54:35 +08:00
|
|
|
#define NOT(expr) storage.Create<query::NotOperator>((expr))
|
|
|
|
#define UPLUS(expr) storage.Create<query::UnaryPlusOperator>((expr))
|
|
|
|
#define UMINUS(expr) storage.Create<query::UnaryMinusOperator>((expr))
|
|
|
|
#define IS_NULL(expr) storage.Create<query::IsNullOperator>((expr))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define ADD(expr1, expr2) storage.Create<query::AdditionOperator>((expr1), (expr2))
|
2017-03-27 18:10:50 +08:00
|
|
|
#define LESS(expr1, expr2) storage.Create<query::LessOperator>((expr1), (expr2))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define LESS_EQ(expr1, expr2) storage.Create<query::LessEqualOperator>((expr1), (expr2))
|
|
|
|
#define GREATER(expr1, expr2) storage.Create<query::GreaterOperator>((expr1), (expr2))
|
|
|
|
#define GREATER_EQ(expr1, expr2) storage.Create<query::GreaterEqualOperator>((expr1), (expr2))
|
|
|
|
#define SUM(expr) storage.Create<query::Aggregation>((expr), nullptr, query::Aggregation::Op::SUM)
|
|
|
|
#define COUNT(expr) storage.Create<query::Aggregation>((expr), nullptr, query::Aggregation::Op::COUNT)
|
|
|
|
#define AVG(expr) storage.Create<query::Aggregation>((expr), nullptr, query::Aggregation::Op::AVG)
|
|
|
|
#define COLLECT_LIST(expr) storage.Create<query::Aggregation>((expr), nullptr, query::Aggregation::Op::COLLECT_LIST)
|
2017-05-05 21:34:07 +08:00
|
|
|
#define EQ(expr1, expr2) storage.Create<query::EqualOperator>((expr1), (expr2))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define NEQ(expr1, expr2) storage.Create<query::NotEqualOperator>((expr1), (expr2))
|
2017-05-05 21:34:07 +08:00
|
|
|
#define AND(expr1, expr2) storage.Create<query::AndOperator>((expr1), (expr2))
|
|
|
|
#define OR(expr1, expr2) storage.Create<query::OrOperator>((expr1), (expr2))
|
2021-02-18 22:32:43 +08:00
|
|
|
#define IN_LIST(expr1, expr2) storage.Create<query::InListOperator>((expr1), (expr2))
|
|
|
|
#define IF(cond, then, else) storage.Create<query::IfOperator>((cond), (then), (else))
|
2017-06-05 17:50:47 +08:00
|
|
|
// Function call
|
2021-02-18 22:32:43 +08:00
|
|
|
#define FN(function_name, ...) \
|
|
|
|
storage.Create<query::Function>(utils::ToUpperCase(function_name), std::vector<query::Expression *>{__VA_ARGS__})
|
2017-06-05 17:50:47 +08:00
|
|
|
// List slicing
|
2021-02-18 22:32:43 +08:00
|
|
|
#define SLICE(list, lower_bound, upper_bound) storage.Create<query::ListSlicingOperator>(list, lower_bound, upper_bound)
|
2017-07-25 19:01:08 +08:00
|
|
|
// all(variable IN list WHERE predicate)
|
2021-02-18 22:32:43 +08:00
|
|
|
#define ALL(variable, list, where) storage.Create<query::All>(storage.Create<query::Identifier>(variable), list, where)
|
|
|
|
#define SINGLE(variable, list, where) \
|
|
|
|
storage.Create<query::Single>(storage.Create<query::Identifier>(variable), list, where)
|
|
|
|
#define ANY(variable, list, where) storage.Create<query::Any>(storage.Create<query::Identifier>(variable), list, where)
|
|
|
|
#define NONE(variable, list, where) \
|
|
|
|
storage.Create<query::None>(storage.Create<query::Identifier>(variable), list, where)
|
|
|
|
#define REDUCE(accumulator, initializer, variable, list, expr) \
|
|
|
|
storage.Create<query::Reduce>(storage.Create<query::Identifier>(accumulator), initializer, \
|
|
|
|
storage.Create<query::Identifier>(variable), list, expr)
|
|
|
|
#define COALESCE(...) storage.Create<query::Coalesce>(std::vector<query::Expression *>{__VA_ARGS__})
|
|
|
|
#define EXTRACT(variable, list, expr) \
|
|
|
|
storage.Create<query::Extract>(storage.Create<query::Identifier>(variable), list, expr)
|
|
|
|
#define AUTH_QUERY(action, user, role, user_or_role, password, privileges) \
|
|
|
|
storage.Create<query::AuthQuery>((action), (user), (role), (user_or_role), password, (privileges))
|
2018-06-14 22:02:27 +08:00
|
|
|
#define DROP_USER(usernames) storage.Create<query::DropUser>((usernames))
|