Refactor ExpressionEvaluator to classic visitor

Reviewers: florijan, mislav.bradac, buda

Reviewed By: mislav.bradac

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D368
This commit is contained in:
Teon Banek 2017-05-16 10:55:02 +02:00
parent f74ba586b2
commit 23ab2b41a3
5 changed files with 318 additions and 304 deletions

View File

@ -14,10 +14,14 @@ namespace query {
class AstTreeStorage; class AstTreeStorage;
class Tree : public ::utils::Visitable<HierarchicalTreeVisitor> { class Tree : public ::utils::Visitable<HierarchicalTreeVisitor>,
::utils::Visitable<TreeVisitor<TypedValue>> {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
using ::utils::Visitable<HierarchicalTreeVisitor>::Accept;
using ::utils::Visitable<TreeVisitor<TypedValue>>::Accept;
int uid() const { return uid_; } int uid() const { return uid_; }
protected: protected:
@ -63,6 +67,7 @@ class OrOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -78,6 +83,7 @@ class XorOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -93,6 +99,7 @@ class AndOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -112,6 +119,7 @@ class FilterAndOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -127,6 +135,7 @@ class AdditionOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -142,6 +151,7 @@ class SubtractionOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -157,6 +167,7 @@ class MultiplicationOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -172,6 +183,7 @@ class DivisionOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -187,6 +199,7 @@ class ModOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -202,6 +215,7 @@ class NotEqualOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -217,6 +231,7 @@ class EqualOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -232,6 +247,7 @@ class LessOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -247,6 +263,7 @@ class GreaterOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -262,6 +279,7 @@ class LessEqualOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -277,6 +295,7 @@ class GreaterEqualOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -292,6 +311,7 @@ class InListOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -307,6 +327,7 @@ class ListIndexingOperator : public BinaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression1_->Accept(visitor) && expression2_->Accept(visitor); expression1_->Accept(visitor) && expression2_->Accept(visitor);
@ -322,6 +343,7 @@ class ListSlicingOperator : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
bool cont = list_->Accept(visitor); bool cont = list_->Accept(visitor);
@ -352,6 +374,7 @@ class NotOperator : public UnaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -367,6 +390,7 @@ class UnaryPlusOperator : public UnaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -382,6 +406,7 @@ class UnaryMinusOperator : public UnaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -397,6 +422,7 @@ class IsNullOperator : public UnaryOperator {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -419,6 +445,7 @@ class PrimitiveLiteral : public BaseLiteral {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
DEFVISITABLE(HierarchicalTreeVisitor); DEFVISITABLE(HierarchicalTreeVisitor);
TypedValue value_; TypedValue value_;
@ -433,6 +460,7 @@ class ListLiteral : public BaseLiteral {
public: public:
const std::vector<Expression *> elements_; const std::vector<Expression *> elements_;
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto expr_ptr : elements_) for (auto expr_ptr : elements_)
@ -451,6 +479,7 @@ class Identifier : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
DEFVISITABLE(HierarchicalTreeVisitor); DEFVISITABLE(HierarchicalTreeVisitor);
std::string name_; std::string name_;
bool user_declared_ = true; bool user_declared_ = true;
@ -465,6 +494,7 @@ class PropertyLookup : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -492,6 +522,7 @@ class LabelsTest : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -512,6 +543,7 @@ class EdgeTypeTest : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -535,6 +567,7 @@ class Function : public Expression {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto *argument : arguments_) { for (auto *argument : arguments_) {
@ -567,6 +600,7 @@ class Aggregation : public UnaryOperator {
static const constexpr char *const kSum = "SUM"; static const constexpr char *const kSum = "SUM";
static const constexpr char *const kAvg = "AVG"; static const constexpr char *const kAvg = "AVG";
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
if (expression_) { if (expression_) {
@ -590,6 +624,7 @@ class NamedExpression : public Tree {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -623,6 +658,7 @@ class NodeAtom : public PatternAtom {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
identifier_->Accept(visitor); identifier_->Accept(visitor);
@ -647,6 +683,7 @@ class EdgeAtom : public PatternAtom {
// necessarily go from left to right // necessarily go from left to right
enum class Direction { LEFT, RIGHT, BOTH }; enum class Direction { LEFT, RIGHT, BOTH };
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
identifier_->Accept(visitor); identifier_->Accept(visitor);
@ -676,6 +713,7 @@ class Pattern : public Tree {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto &part : atoms_) { for (auto &part : atoms_) {
@ -695,6 +733,7 @@ class Query : public Tree {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto &clause : clauses_) { for (auto &clause : clauses_) {
@ -715,6 +754,7 @@ class Create : public Clause {
public: public:
Create(int uid) : Clause(uid) {} Create(int uid) : Clause(uid) {}
std::vector<Pattern *> patterns_; std::vector<Pattern *> patterns_;
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto &pattern : patterns_) { for (auto &pattern : patterns_) {
@ -729,6 +769,7 @@ class Where : public Tree {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
expression_->Accept(visitor); expression_->Accept(visitor);
@ -746,6 +787,7 @@ class Match : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
bool cont = true; bool cont = true;
@ -795,6 +837,7 @@ class Return : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
bool cont = true; bool cont = true;
@ -828,6 +871,7 @@ class With : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
bool cont = true; bool cont = true;
@ -863,6 +907,7 @@ class Delete : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
for (auto &expr : expressions_) { for (auto &expr : expressions_) {
@ -882,6 +927,7 @@ class SetProperty : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
property_lookup_->Accept(visitor) && expression_->Accept(visitor); property_lookup_->Accept(visitor) && expression_->Accept(visitor);
@ -903,6 +949,7 @@ class SetProperties : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
identifier_->Accept(visitor) && expression_->Accept(visitor); identifier_->Accept(visitor) && expression_->Accept(visitor);
@ -927,6 +974,7 @@ class SetLabels : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
identifier_->Accept(visitor); identifier_->Accept(visitor);
@ -947,6 +995,7 @@ class RemoveProperty : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
property_lookup_->Accept(visitor); property_lookup_->Accept(visitor);
@ -965,6 +1014,7 @@ class RemoveLabels : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
identifier_->Accept(visitor); identifier_->Accept(visitor);
@ -985,6 +1035,7 @@ class Merge : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
bool cont = pattern_->Accept(visitor); bool cont = pattern_->Accept(visitor);
@ -1020,6 +1071,7 @@ class Unwind : public Clause {
friend class AstTreeStorage; friend class AstTreeStorage;
public: public:
DEFVISITABLE(TreeVisitor<TypedValue>);
bool Accept(HierarchicalTreeVisitor &visitor) override { bool Accept(HierarchicalTreeVisitor &visitor) override {
if (visitor.PreVisit(*this)) { if (visitor.PreVisit(*this)) {
named_expression_->Accept(visitor); named_expression_->Accept(visitor);

View File

@ -77,4 +77,17 @@ class HierarchicalTreeVisitor : public TreeCompositeVisitor,
using TreeLeafVisitor::Visit; using TreeLeafVisitor::Visit;
}; };
template <typename TResult>
using TreeVisitor = ::utils::Visitor<
TResult, Query, NamedExpression, OrOperator, XorOperator, AndOperator,
FilterAndOperator, NotOperator, AdditionOperator, SubtractionOperator,
MultiplicationOperator, DivisionOperator, ModOperator, NotEqualOperator,
EqualOperator, LessOperator, GreaterOperator, LessEqualOperator,
GreaterEqualOperator, InListOperator, ListIndexingOperator,
ListSlicingOperator, UnaryPlusOperator, UnaryMinusOperator, IsNullOperator,
ListLiteral, PropertyLookup, LabelsTest, EdgeTypeTest, Aggregation,
Function, Create, Match, Return, With, Pattern, NodeAtom, EdgeAtom, Delete,
Where, SetProperty, SetProperties, SetLabels, RemoveProperty, RemoveLabels,
Merge, Unwind, Identifier, PrimitiveLiteral>;
} // namespace query } // namespace query

View File

@ -16,7 +16,7 @@
namespace query { namespace query {
class ExpressionEvaluator : public HierarchicalTreeVisitor { class ExpressionEvaluator : public TreeVisitor<TypedValue> {
public: public:
ExpressionEvaluator(Frame &frame, const SymbolTable &symbol_table, ExpressionEvaluator(Frame &frame, const SymbolTable &symbol_table,
GraphDbAccessor &db_accessor, GraphDbAccessor &db_accessor,
@ -26,50 +26,56 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
db_accessor_(db_accessor), db_accessor_(db_accessor),
graph_view_(graph_view) {} graph_view_(graph_view) {}
/** using TreeVisitor<TypedValue>::Visit;
* Removes and returns the last value from the result stack.
* Consumers of this function are PostVisit functions for #define BLOCK_VISIT(TREE_TYPE) \
* expressions that consume subexpressions, as well as top TypedValue Visit(TREE_TYPE &) override { \
* level expression consumers. permanent_fail("ExpressionEvaluator should not visit " #TREE_TYPE); \
*/
auto PopBack() {
debug_assert(result_stack_.size() > 0, "Result stack empty");
auto last = result_stack_.back();
result_stack_.pop_back();
return last;
} }
using HierarchicalTreeVisitor::PreVisit; BLOCK_VISIT(Query);
using typename HierarchicalTreeVisitor::ReturnType; BLOCK_VISIT(Create);
using HierarchicalTreeVisitor::Visit; BLOCK_VISIT(Match);
using HierarchicalTreeVisitor::PostVisit; BLOCK_VISIT(Return);
BLOCK_VISIT(With);
BLOCK_VISIT(Pattern);
BLOCK_VISIT(NodeAtom);
BLOCK_VISIT(EdgeAtom);
BLOCK_VISIT(Delete);
BLOCK_VISIT(Where);
BLOCK_VISIT(SetProperty);
BLOCK_VISIT(SetProperties);
BLOCK_VISIT(SetLabels);
BLOCK_VISIT(RemoveProperty);
BLOCK_VISIT(RemoveLabels);
BLOCK_VISIT(Merge);
BLOCK_VISIT(Unwind);
bool PostVisit(NamedExpression &named_expression) override { #undef BLOCK_VISIT
auto symbol = symbol_table_.at(named_expression);
frame_[symbol] = PopBack(); TypedValue Visit(NamedExpression &named_expression) override {
return true; const auto &symbol = symbol_table_.at(named_expression);
auto value = named_expression.expression_->Accept(*this);
frame_[symbol] = value;
return value;
} }
ReturnType Visit(Identifier &ident) override { TypedValue Visit(Identifier &ident) override {
auto value = frame_[symbol_table_.at(ident)]; auto value = frame_[symbol_table_.at(ident)];
SwitchAccessors(value); SwitchAccessors(value);
result_stack_.emplace_back(std::move(value)); return value;
return true;
} }
#define BINARY_OPERATOR_VISITOR(OP_NODE, CPP_OP) \ #define BINARY_OPERATOR_VISITOR(OP_NODE, CPP_OP) \
bool PostVisit(OP_NODE &) override { \ TypedValue Visit(OP_NODE &op) override { \
auto expression2 = PopBack(); \ auto val1 = op.expression1_->Accept(*this); \
auto expression1 = PopBack(); \ auto val2 = op.expression2_->Accept(*this); \
result_stack_.push_back(expression1 CPP_OP expression2); \ return val1 CPP_OP val2; \
return true; \
} }
#define UNARY_OPERATOR_VISITOR(OP_NODE, CPP_OP) \ #define UNARY_OPERATOR_VISITOR(OP_NODE, CPP_OP) \
bool PostVisit(OP_NODE &) override { \ TypedValue Visit(OP_NODE &op) override { \
auto expression = PopBack(); \ return CPP_OP op.expression_->Accept(*this); \
result_stack_.push_back(CPP_OP expression); \
return true; \
} }
BINARY_OPERATOR_VISITOR(OrOperator, ||); BINARY_OPERATOR_VISITOR(OrOperator, ||);
@ -94,34 +100,27 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
#undef BINARY_OPERATOR_VISITOR #undef BINARY_OPERATOR_VISITOR
#undef UNARY_OPERATOR_VISITOR #undef UNARY_OPERATOR_VISITOR
bool PreVisit(FilterAndOperator &op) override { TypedValue Visit(FilterAndOperator &op) override {
op.expression1_->Accept(*this); auto value1 = op.expression1_->Accept(*this);
auto expression1 = PopBack(); if (value1.IsNull() || !value1.Value<bool>()) {
if (expression1.IsNull() || !expression1.Value<bool>()) {
// If first expression is null or false, don't execute the second one. // If first expression is null or false, don't execute the second one.
result_stack_.emplace_back(expression1); return value1;
return false;
} }
op.expression2_->Accept(*this); return op.expression2_->Accept(*this);
auto expression2 = PopBack();
result_stack_.emplace_back(expression2);
return false;
} }
bool PostVisit(InListOperator &) override { TypedValue Visit(InListOperator &in_list) override {
auto _list = PopBack(); auto literal = in_list.expression1_->Accept(*this);
auto literal = PopBack(); auto _list = in_list.expression2_->Accept(*this);
if (_list.IsNull()) { if (_list.IsNull()) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
return true;
} }
// Exceptions have higher priority than returning null. // Exceptions have higher priority than returning null.
// We need to convert list to its type before checking if literal is null, // We need to convert list to its type before checking if literal is null,
// because conversion will throw exception if list conversion fails. // because conversion will throw exception if list conversion fails.
auto list = _list.Value<std::vector<TypedValue>>(); auto list = _list.Value<std::vector<TypedValue>>();
if (literal.IsNull()) { if (literal.IsNull()) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
return true;
} }
auto has_null = false; auto has_null = false;
for (const auto &element : list) { for (const auto &element : list) {
@ -129,34 +128,30 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
if (result.IsNull()) { if (result.IsNull()) {
has_null = true; has_null = true;
} else if (result.Value<bool>()) { } else if (result.Value<bool>()) {
result_stack_.emplace_back(true);
return true; return true;
} }
} }
if (has_null) { if (has_null) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
} else {
result_stack_.emplace_back(false);
} }
return true; return false;
} }
bool PostVisit(ListIndexingOperator &) override { TypedValue Visit(ListIndexingOperator &list_indexing) override {
// TODO: implement this for maps // TODO: implement this for maps
auto _index = PopBack(); auto _list = list_indexing.expression1_->Accept(*this);
if (_index.type() != TypedValue::Type::Int &&
_index.type() != TypedValue::Type::Null) {
throw TypedValueException("Incompatible type in list lookup");
}
auto _list = PopBack();
if (_list.type() != TypedValue::Type::List && if (_list.type() != TypedValue::Type::List &&
_list.type() != TypedValue::Type::Null) { _list.type() != TypedValue::Type::Null) {
throw TypedValueException("Incompatible type in list lookup"); throw TypedValueException("Incompatible type in list lookup");
} }
auto _index = list_indexing.expression2_->Accept(*this);
if (_index.type() != TypedValue::Type::Int &&
_index.type() != TypedValue::Type::Null) {
throw TypedValueException("Incompatible type in list lookup");
}
if (_index.type() == TypedValue::Type::Null || if (_index.type() == TypedValue::Type::Null ||
_list.type() == TypedValue::Type::Null) { _list.type() == TypedValue::Type::Null) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
return true;
} }
auto index = _index.Value<int64_t>(); auto index = _index.Value<int64_t>();
const auto &list = _list.Value<std::vector<TypedValue>>(); const auto &list = _list.Value<std::vector<TypedValue>>();
@ -164,20 +159,18 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
index = static_cast<int64_t>(list.size()) + index; index = static_cast<int64_t>(list.size()) + index;
} }
if (index >= static_cast<int64_t>(list.size()) || index < 0) { if (index >= static_cast<int64_t>(list.size()) || index < 0) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
return true;
} }
result_stack_.emplace_back(list[index]); return list[index];
return true;
} }
bool PostVisit(ListSlicingOperator &op) override { TypedValue Visit(ListSlicingOperator &op) override {
// If some type is null we can't return null, because throwing exception on // If some type is null we can't return null, because throwing exception on
// illegal type has higher priority. // illegal type has higher priority.
auto is_null = false; auto is_null = false;
auto get_bound = [&](bool is_defined, int64_t default_value) { auto get_bound = [&](Expression *bound_expr, int64_t default_value) {
if (is_defined) { if (bound_expr) {
auto bound = PopBack(); auto bound = bound_expr->Accept(*this);
if (bound.type() == TypedValue::Type::Null) { if (bound.type() == TypedValue::Type::Null) {
is_null = true; is_null = true;
} else if (bound.type() != TypedValue::Type::Int) { } else if (bound.type() != TypedValue::Type::Int) {
@ -191,7 +184,7 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
get_bound(op.upper_bound_, std::numeric_limits<int64_t>::max()); get_bound(op.upper_bound_, std::numeric_limits<int64_t>::max());
auto _lower_bound = get_bound(op.lower_bound_, 0); auto _lower_bound = get_bound(op.lower_bound_, 0);
auto _list = PopBack(); auto _list = op.list_->Accept(*this);
if (_list.type() == TypedValue::Type::Null) { if (_list.type() == TypedValue::Type::Null) {
is_null = true; is_null = true;
} else if (_list.type() != TypedValue::Type::List) { } else if (_list.type() != TypedValue::Type::List) {
@ -199,8 +192,7 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
} }
if (is_null) { if (is_null) {
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
return true;
} }
const auto &list = _list.Value<std::vector<TypedValue>>(); const auto &list = _list.Value<std::vector<TypedValue>>();
auto normalise_bound = [&](int64_t bound) { auto normalise_bound = [&](int64_t bound) {
@ -213,131 +205,105 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
auto lower_bound = normalise_bound(_lower_bound.Value<int64_t>()); auto lower_bound = normalise_bound(_lower_bound.Value<int64_t>());
auto upper_bound = normalise_bound(_upper_bound.Value<int64_t>()); auto upper_bound = normalise_bound(_upper_bound.Value<int64_t>());
if (upper_bound <= lower_bound) { if (upper_bound <= lower_bound) {
result_stack_.emplace_back(std::vector<TypedValue>()); return std::vector<TypedValue>();
return true;
} }
result_stack_.emplace_back(std::vector<TypedValue>( return std::vector<TypedValue>(list.begin() + lower_bound,
list.begin() + lower_bound, list.begin() + upper_bound)); list.begin() + upper_bound);
return true;
} }
bool PostVisit(IsNullOperator &) override { TypedValue Visit(IsNullOperator &is_null) override {
auto expression = PopBack(); auto value = is_null.expression_->Accept(*this);
result_stack_.push_back(TypedValue(expression.IsNull())); return value.IsNull();
return true;
} }
bool PostVisit(PropertyLookup &property_lookup) override { TypedValue Visit(PropertyLookup &property_lookup) override {
auto expression_result = PopBack(); auto expression_result = property_lookup.expression_->Accept(*this);
switch (expression_result.type()) { switch (expression_result.type()) {
case TypedValue::Type::Null: case TypedValue::Type::Null:
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
break;
case TypedValue::Type::Vertex: case TypedValue::Type::Vertex:
result_stack_.emplace_back( return expression_result.Value<VertexAccessor>().PropsAt(
expression_result.Value<VertexAccessor>().PropsAt( property_lookup.property_);
property_lookup.property_)); case TypedValue::Type::Edge:
break; return expression_result.Value<EdgeAccessor>().PropsAt(
case TypedValue::Type::Edge: { property_lookup.property_);
result_stack_.emplace_back(
expression_result.Value<EdgeAccessor>().PropsAt(
property_lookup.property_));
break;
}
case TypedValue::Type::Map: case TypedValue::Type::Map:
// TODO implement me // TODO implement me
throw utils::NotYetImplemented(); throw utils::NotYetImplemented(
break; "Not yet implemented property lookup on map");
default: default:
throw TypedValueException( throw TypedValueException(
"Expected Node, Edge or Map for property lookup"); "Expected Node, Edge or Map for property lookup");
} }
return true;
} }
bool PostVisit(LabelsTest &labels_test) override { TypedValue Visit(LabelsTest &labels_test) override {
auto expression_result = PopBack(); auto expression_result = labels_test.expression_->Accept(*this);
switch (expression_result.type()) { switch (expression_result.type()) {
case TypedValue::Type::Null: case TypedValue::Type::Null:
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
break;
case TypedValue::Type::Vertex: { case TypedValue::Type::Vertex: {
auto vertex = expression_result.Value<VertexAccessor>(); auto vertex = expression_result.Value<VertexAccessor>();
for (const auto label : labels_test.labels_) { for (const auto label : labels_test.labels_) {
if (!vertex.has_label(label)) { if (!vertex.has_label(label)) {
result_stack_.emplace_back(false); return false;
return true;
} }
} }
result_stack_.emplace_back(true); return true;
break;
} }
default: default:
throw TypedValueException("Expected Node in labels test"); throw TypedValueException("Expected Node in labels test");
} }
return true;
} }
bool PostVisit(EdgeTypeTest &edge_type_test) override { TypedValue Visit(EdgeTypeTest &edge_type_test) override {
auto expression_result = PopBack(); auto expression_result = edge_type_test.expression_->Accept(*this);
switch (expression_result.type()) { switch (expression_result.type()) {
case TypedValue::Type::Null: case TypedValue::Type::Null:
result_stack_.emplace_back(TypedValue::Null); return TypedValue::Null;
break;
case TypedValue::Type::Edge: { case TypedValue::Type::Edge: {
auto real_edge_type = auto real_edge_type =
expression_result.Value<EdgeAccessor>().edge_type(); expression_result.Value<EdgeAccessor>().edge_type();
for (const auto edge_type : edge_type_test.edge_types_) { for (const auto edge_type : edge_type_test.edge_types_) {
if (edge_type == real_edge_type) { if (edge_type == real_edge_type) {
result_stack_.emplace_back(true);
return true; return true;
} }
} }
result_stack_.emplace_back(false); return false;
break;
} }
default: default:
throw TypedValueException("Expected Edge in edge type test"); throw TypedValueException("Expected Edge in edge type test");
} }
return true;
} }
ReturnType Visit(PrimitiveLiteral &literal) override { TypedValue Visit(PrimitiveLiteral &literal) override {
// TODO: no need to evaluate constants, we can write it to frame in one // TODO: no need to evaluate constants, we can write it to frame in one
// of the previous phases. // of the previous phases.
result_stack_.push_back(literal.value_); return literal.value_;
return true;
} }
bool PostVisit(ListLiteral &literal) override { TypedValue Visit(ListLiteral &literal) override {
std::vector<TypedValue> result; std::vector<TypedValue> result;
result.reserve(literal.elements_.size()); result.reserve(literal.elements_.size());
for (size_t i = 0; i < literal.elements_.size(); i++) for (const auto &expression : literal.elements_)
result.emplace_back(PopBack()); result.emplace_back(expression->Accept(*this));
std::reverse(result.begin(), result.end()); return result;
result_stack_.emplace_back(std::move(result));
return true;
} }
bool PreVisit(Aggregation &aggregation) override { TypedValue Visit(Aggregation &aggregation) override {
auto value = frame_[symbol_table_.at(aggregation)]; auto value = frame_[symbol_table_.at(aggregation)];
// Aggregation is probably always simple type, but let's switch accessor // Aggregation is probably always simple type, but let's switch accessor
// just to be sure. // just to be sure.
SwitchAccessors(value); SwitchAccessors(value);
result_stack_.emplace_back(std::move(value)); return value;
// Prevent evaluation of expressions inside the aggregation.
return false;
} }
bool PostVisit(Function &function) override { TypedValue Visit(Function &function) override {
std::vector<TypedValue> arguments; std::vector<TypedValue> arguments;
for (int i = 0; i < static_cast<int>(function.arguments_.size()); ++i) { for (const auto &argument : function.arguments_) {
arguments.push_back(PopBack()); arguments.emplace_back(argument->Accept(*this));
} }
reverse(arguments.begin(), arguments.end()); return function.function_(arguments, db_accessor_);
result_stack_.emplace_back(function.function_(arguments, db_accessor_));
return true;
} }
private: private:
@ -394,6 +360,6 @@ class ExpressionEvaluator : public HierarchicalTreeVisitor {
GraphDbAccessor &db_accessor_; GraphDbAccessor &db_accessor_;
// which switching approach should be used when evaluating // which switching approach should be used when evaluating
const GraphView graph_view_; const GraphView graph_view_;
std::list<TypedValue> result_stack_;
}; };
}
} // namespace query

View File

@ -66,8 +66,7 @@ void CreateNode::CreateNodeCursor::Create(Frame &frame,
// setting properties on new nodes. // setting properties on new nodes.
ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW); ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW);
for (auto &kv : self_.node_atom_->properties_) { for (auto &kv : self_.node_atom_->properties_) {
kv.second->Accept(evaluator); new_node.PropsSet(kv.first, kv.second->Accept(evaluator));
new_node.PropsSet(kv.first, evaluator.PopBack());
} }
frame[symbol_table.at(*self_.node_atom_->identifier_)] = new_node; frame[symbol_table.at(*self_.node_atom_->identifier_)] = new_node;
} }
@ -142,8 +141,7 @@ VertexAccessor &CreateExpand::CreateExpandCursor::OtherVertex(
auto node = db_.insert_vertex(); auto node = db_.insert_vertex();
for (auto label : self_.node_atom_->labels_) node.add_label(label); for (auto label : self_.node_atom_->labels_) node.add_label(label);
for (auto kv : self_.node_atom_->properties_) { for (auto kv : self_.node_atom_->properties_) {
kv.second->Accept(evaluator); node.PropsSet(kv.first, kv.second->Accept(evaluator));
node.PropsSet(kv.first, evaluator.PopBack());
} }
auto symbol = symbol_table.at(*self_.node_atom_->identifier_); auto symbol = symbol_table.at(*self_.node_atom_->identifier_);
frame[symbol] = node; frame[symbol] = node;
@ -157,8 +155,7 @@ void CreateExpand::CreateExpandCursor::CreateEdge(
EdgeAccessor edge = EdgeAccessor edge =
db_.insert_edge(from, to, self_.edge_atom_->edge_types_[0]); db_.insert_edge(from, to, self_.edge_atom_->edge_types_[0]);
for (auto kv : self_.edge_atom_->properties_) { for (auto kv : self_.edge_atom_->properties_) {
kv.second->Accept(evaluator); edge.PropsSet(kv.first, kv.second->Accept(evaluator));
edge.PropsSet(kv.first, evaluator.PopBack());
} }
frame[symbol_table.at(*self_.edge_atom_->identifier_)] = edge; frame[symbol_table.at(*self_.edge_atom_->identifier_)] = edge;
} }
@ -369,8 +366,7 @@ bool Filter::FilterCursor::Pull(Frame &frame, const SymbolTable &symbol_table) {
// and edges. // and edges.
ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::OLD); ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::OLD);
while (input_cursor_->Pull(frame, symbol_table)) { while (input_cursor_->Pull(frame, symbol_table)) {
self_.expression_->Accept(evaluator); TypedValue result = self_.expression_->Accept(evaluator);
TypedValue result = evaluator.PopBack();
if (result.IsNull() || !result.Value<bool>()) continue; if (result.IsNull() || !result.Value<bool>()) continue;
return true; return true;
} }
@ -444,8 +440,7 @@ bool Delete::DeleteCursor::Pull(Frame &frame, const SymbolTable &symbol_table) {
std::vector<TypedValue> expression_results; std::vector<TypedValue> expression_results;
expression_results.reserve(self_.expressions_.size()); expression_results.reserve(self_.expressions_.size());
for (Expression *expression : self_.expressions_) { for (Expression *expression : self_.expressions_) {
expression->Accept(evaluator); expression_results.emplace_back(expression->Accept(evaluator));
expression_results.emplace_back(evaluator.PopBack());
} }
// delete edges first // delete edges first
@ -503,10 +498,8 @@ bool SetProperty::SetPropertyCursor::Pull(Frame &frame,
// Set, just like Create needs to see the latest changes. // Set, just like Create needs to see the latest changes.
ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW); ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW);
self_.lhs_->expression_->Accept(evaluator); TypedValue lhs = self_.lhs_->expression_->Accept(evaluator);
TypedValue lhs = evaluator.PopBack(); TypedValue rhs = self_.rhs_->Accept(evaluator);
self_.rhs_->Accept(evaluator);
TypedValue rhs = evaluator.PopBack();
// TODO the following code uses implicit TypedValue to PropertyValue // TODO the following code uses implicit TypedValue to PropertyValue
// conversion which throws a TypedValueException if impossible // conversion which throws a TypedValueException if impossible
@ -551,8 +544,7 @@ bool SetProperties::SetPropertiesCursor::Pull(Frame &frame,
// Set, just like Create needs to see the latest changes. // Set, just like Create needs to see the latest changes.
ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW); ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW);
self_.rhs_->Accept(evaluator); TypedValue rhs = self_.rhs_->Accept(evaluator);
TypedValue rhs = evaluator.PopBack();
switch (lhs.type()) { switch (lhs.type()) {
case TypedValue::Type::Vertex: case TypedValue::Type::Vertex:
@ -657,8 +649,7 @@ bool RemoveProperty::RemovePropertyCursor::Pull(
// Remove, just like Delete needs to see the latest changes. // Remove, just like Delete needs to see the latest changes.
ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW); ExpressionEvaluator evaluator(frame, symbol_table, db_, GraphView::NEW);
self_.lhs_->expression_->Accept(evaluator); TypedValue lhs = self_.lhs_->expression_->Accept(evaluator);
TypedValue lhs = evaluator.PopBack();
switch (lhs.type()) { switch (lhs.type()) {
case TypedValue::Type::Vertex: case TypedValue::Type::Vertex:
@ -910,8 +901,7 @@ void Aggregate::AggregateCursor::ProcessOne(Frame &frame,
// create the group-by list of values // create the group-by list of values
std::list<TypedValue> group_by; std::list<TypedValue> group_by;
for (Expression *expression : self_.group_by_) { for (Expression *expression : self_.group_by_) {
expression->Accept(evaluator); group_by.emplace_back(expression->Accept(evaluator));
group_by.emplace_back(evaluator.PopBack());
} }
AggregationValue &agg_value = aggregation_[group_by]; AggregationValue &agg_value = aggregation_[group_by];
@ -961,8 +951,7 @@ void Aggregate::AggregateCursor::Update(
continue; continue;
} }
input_expr_ptr->Accept(evaluator); TypedValue input_value = input_expr_ptr->Accept(evaluator);
TypedValue input_value = evaluator.PopBack();
// Aggregations skip Null input values. // Aggregations skip Null input values.
if (input_value.IsNull()) continue; if (input_value.IsNull()) continue;
@ -1086,8 +1075,7 @@ bool Skip::SkipCursor::Pull(Frame &frame, const SymbolTable &symbol_table) {
// first successful pull from the input // first successful pull from the input
// evaluate the skip expression // evaluate the skip expression
ExpressionEvaluator evaluator(frame, symbol_table, db_); ExpressionEvaluator evaluator(frame, symbol_table, db_);
self_.expression_->Accept(evaluator); TypedValue to_skip = self_.expression_->Accept(evaluator);
TypedValue to_skip = evaluator.PopBack();
if (to_skip.type() != TypedValue::Type::Int) if (to_skip.type() != TypedValue::Type::Int)
throw QueryRuntimeException("Result of SKIP expression must be an int"); throw QueryRuntimeException("Result of SKIP expression must be an int");
@ -1134,8 +1122,7 @@ bool Limit::LimitCursor::Pull(Frame &frame, const SymbolTable &symbol_table) {
// is not allowed to contain any identifiers // is not allowed to contain any identifiers
if (limit_ == -1) { if (limit_ == -1) {
ExpressionEvaluator evaluator(frame, symbol_table, db_); ExpressionEvaluator evaluator(frame, symbol_table, db_);
self_.expression_->Accept(evaluator); TypedValue limit = self_.expression_->Accept(evaluator);
TypedValue limit = evaluator.PopBack();
if (limit.type() != TypedValue::Type::Int) if (limit.type() != TypedValue::Type::Int)
throw QueryRuntimeException("Result of LIMIT expression must be an int"); throw QueryRuntimeException("Result of LIMIT expression must be an int");
@ -1194,8 +1181,7 @@ bool OrderBy::OrderByCursor::Pull(Frame &frame,
// collect the order_by elements // collect the order_by elements
std::list<TypedValue> order_by; std::list<TypedValue> order_by;
for (auto expression_ptr : self_.order_by_) { for (auto expression_ptr : self_.order_by_) {
expression_ptr->Accept(evaluator); order_by.emplace_back(expression_ptr->Accept(evaluator));
order_by.emplace_back(evaluator.PopBack());
} }
// collect the output elements // collect the output elements
@ -1456,8 +1442,7 @@ bool Unwind::UnwindCursor::Pull(Frame &frame, const SymbolTable &symbol_table) {
// successful pull from input, initialize value and iterator // successful pull from input, initialize value and iterator
ExpressionEvaluator evaluator(frame, symbol_table, db_); ExpressionEvaluator evaluator(frame, symbol_table, db_);
self_.input_expression_->Accept(evaluator); TypedValue input_value = self_.input_expression_->Accept(evaluator);
TypedValue input_value = evaluator.PopBack();
if (input_value.type() != TypedValue::Type::List) if (input_value.type() != TypedValue::Type::List)
throw QueryRuntimeException("UNWIND only accepts list values"); throw QueryRuntimeException("UNWIND only accepts list values");
input_value_ = input_value.Value<std::vector<TypedValue>>(); input_value_ = input_value.Value<std::vector<TypedValue>>();

View File

@ -44,8 +44,7 @@ TypedValue EvaluateFunction(const std::string &function_name,
} }
auto *op = auto *op =
storage.Create<Function>(NameToFunction(function_name), expressions); storage.Create<Function>(NameToFunction(function_name), expressions);
op->Accept(eval.eval); return op->Accept(eval.eval);
return eval.eval.PopBack();
} }
TEST(ExpressionEvaluator, OrOperator) { TEST(ExpressionEvaluator, OrOperator) {
@ -54,12 +53,12 @@ TEST(ExpressionEvaluator, OrOperator) {
auto *op = auto *op =
storage.Create<OrOperator>(storage.Create<PrimitiveLiteral>(true), storage.Create<OrOperator>(storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(false)); storage.Create<PrimitiveLiteral>(false));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<OrOperator>(storage.Create<PrimitiveLiteral>(true), op = storage.Create<OrOperator>(storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(true)); storage.Create<PrimitiveLiteral>(true));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val2.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, XorOperator) { TEST(ExpressionEvaluator, XorOperator) {
@ -68,12 +67,12 @@ TEST(ExpressionEvaluator, XorOperator) {
auto *op = auto *op =
storage.Create<XorOperator>(storage.Create<PrimitiveLiteral>(true), storage.Create<XorOperator>(storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(false)); storage.Create<PrimitiveLiteral>(false));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<XorOperator>(storage.Create<PrimitiveLiteral>(true), op = storage.Create<XorOperator>(storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(true)); storage.Create<PrimitiveLiteral>(true));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val2.Value<bool>(), false);
} }
TEST(ExpressionEvaluator, AndOperator) { TEST(ExpressionEvaluator, AndOperator) {
@ -82,12 +81,12 @@ TEST(ExpressionEvaluator, AndOperator) {
auto *op = auto *op =
storage.Create<AndOperator>(storage.Create<PrimitiveLiteral>(true), storage.Create<AndOperator>(storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(true)); storage.Create<PrimitiveLiteral>(true));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<AndOperator>(storage.Create<PrimitiveLiteral>(false), op = storage.Create<AndOperator>(storage.Create<PrimitiveLiteral>(false),
storage.Create<PrimitiveLiteral>(true)); storage.Create<PrimitiveLiteral>(true));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val2.Value<bool>(), false);
} }
TEST(ExpressionEvaluator, FilterAndOperator) { TEST(ExpressionEvaluator, FilterAndOperator) {
@ -97,22 +96,22 @@ TEST(ExpressionEvaluator, FilterAndOperator) {
auto *op = storage.Create<FilterAndOperator>( auto *op = storage.Create<FilterAndOperator>(
storage.Create<PrimitiveLiteral>(true), storage.Create<PrimitiveLiteral>(true),
storage.Create<PrimitiveLiteral>(true)); storage.Create<PrimitiveLiteral>(true));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), true); EXPECT_EQ(value.Value<bool>(), true);
} }
{ {
auto *op = storage.Create<FilterAndOperator>( auto *op = storage.Create<FilterAndOperator>(
storage.Create<PrimitiveLiteral>(false), storage.Create<PrimitiveLiteral>(false),
storage.Create<PrimitiveLiteral>(5)); storage.Create<PrimitiveLiteral>(5));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), false); EXPECT_EQ(value.Value<bool>(), false);
} }
{ {
auto *op = storage.Create<FilterAndOperator>( auto *op = storage.Create<FilterAndOperator>(
storage.Create<PrimitiveLiteral>(TypedValue::Null), storage.Create<PrimitiveLiteral>(TypedValue::Null),
storage.Create<PrimitiveLiteral>(5)); storage.Create<PrimitiveLiteral>(5));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
} }
@ -121,8 +120,8 @@ TEST(ExpressionEvaluator, AdditionOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = storage.Create<AdditionOperator>( auto *op = storage.Create<AdditionOperator>(
storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3)); storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), 5); ASSERT_EQ(value.Value<int64_t>(), 5);
} }
TEST(ExpressionEvaluator, SubtractionOperator) { TEST(ExpressionEvaluator, SubtractionOperator) {
@ -130,8 +129,8 @@ TEST(ExpressionEvaluator, SubtractionOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = storage.Create<SubtractionOperator>( auto *op = storage.Create<SubtractionOperator>(
storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3)); storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), -1); ASSERT_EQ(value.Value<int64_t>(), -1);
} }
TEST(ExpressionEvaluator, MultiplicationOperator) { TEST(ExpressionEvaluator, MultiplicationOperator) {
@ -139,8 +138,8 @@ TEST(ExpressionEvaluator, MultiplicationOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = storage.Create<MultiplicationOperator>( auto *op = storage.Create<MultiplicationOperator>(
storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3)); storage.Create<PrimitiveLiteral>(2), storage.Create<PrimitiveLiteral>(3));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), 6); ASSERT_EQ(value.Value<int64_t>(), 6);
} }
TEST(ExpressionEvaluator, DivisionOperator) { TEST(ExpressionEvaluator, DivisionOperator) {
@ -149,8 +148,8 @@ TEST(ExpressionEvaluator, DivisionOperator) {
auto *op = auto *op =
storage.Create<DivisionOperator>(storage.Create<PrimitiveLiteral>(50), storage.Create<DivisionOperator>(storage.Create<PrimitiveLiteral>(50),
storage.Create<PrimitiveLiteral>(10)); storage.Create<PrimitiveLiteral>(10));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), 5); ASSERT_EQ(value.Value<int64_t>(), 5);
} }
TEST(ExpressionEvaluator, ModOperator) { TEST(ExpressionEvaluator, ModOperator) {
@ -158,8 +157,8 @@ TEST(ExpressionEvaluator, ModOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = storage.Create<ModOperator>(storage.Create<PrimitiveLiteral>(65), auto *op = storage.Create<ModOperator>(storage.Create<PrimitiveLiteral>(65),
storage.Create<PrimitiveLiteral>(10)); storage.Create<PrimitiveLiteral>(10));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), 5); ASSERT_EQ(value.Value<int64_t>(), 5);
} }
TEST(ExpressionEvaluator, EqualOperator) { TEST(ExpressionEvaluator, EqualOperator) {
@ -168,16 +167,16 @@ TEST(ExpressionEvaluator, EqualOperator) {
auto *op = auto *op =
storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(10), storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val1.Value<bool>(), false);
op = storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(15), op = storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val2.Value<bool>(), true);
op = storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(20), op = storage.Create<EqualOperator>(storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val3.Value<bool>(), false);
} }
TEST(ExpressionEvaluator, NotEqualOperator) { TEST(ExpressionEvaluator, NotEqualOperator) {
@ -186,16 +185,16 @@ TEST(ExpressionEvaluator, NotEqualOperator) {
auto *op = auto *op =
storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(10), storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(15), op = storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val2.Value<bool>(), false);
op = storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(20), op = storage.Create<NotEqualOperator>(storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val3.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, LessOperator) { TEST(ExpressionEvaluator, LessOperator) {
@ -203,16 +202,16 @@ TEST(ExpressionEvaluator, LessOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(10), auto *op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(15), op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val2.Value<bool>(), false);
op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(20), op = storage.Create<LessOperator>(storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val3.Value<bool>(), false);
} }
TEST(ExpressionEvaluator, GreaterOperator) { TEST(ExpressionEvaluator, GreaterOperator) {
@ -221,16 +220,16 @@ TEST(ExpressionEvaluator, GreaterOperator) {
auto *op = auto *op =
storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(10), storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val1.Value<bool>(), false);
op = storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(15), op = storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val2.Value<bool>(), false);
op = storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(20), op = storage.Create<GreaterOperator>(storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val3.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, LessEqualOperator) { TEST(ExpressionEvaluator, LessEqualOperator) {
@ -239,16 +238,16 @@ TEST(ExpressionEvaluator, LessEqualOperator) {
auto *op = auto *op =
storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(10), storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val1.Value<bool>(), true);
op = storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(15), op = storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val2.Value<bool>(), true);
op = storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(20), op = storage.Create<LessEqualOperator>(storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val3.Value<bool>(), false);
} }
TEST(ExpressionEvaluator, GreaterEqualOperator) { TEST(ExpressionEvaluator, GreaterEqualOperator) {
@ -257,18 +256,18 @@ TEST(ExpressionEvaluator, GreaterEqualOperator) {
auto *op = storage.Create<GreaterEqualOperator>( auto *op = storage.Create<GreaterEqualOperator>(
storage.Create<PrimitiveLiteral>(10), storage.Create<PrimitiveLiteral>(10),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val1.Value<bool>(), false);
op = storage.Create<GreaterEqualOperator>( op = storage.Create<GreaterEqualOperator>(
storage.Create<PrimitiveLiteral>(15), storage.Create<PrimitiveLiteral>(15),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val2.Value<bool>(), true);
op = storage.Create<GreaterEqualOperator>( op = storage.Create<GreaterEqualOperator>(
storage.Create<PrimitiveLiteral>(20), storage.Create<PrimitiveLiteral>(20),
storage.Create<PrimitiveLiteral>(15)); storage.Create<PrimitiveLiteral>(15));
op->Accept(eval.eval); auto val3 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val3.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, InListOperator) { TEST(ExpressionEvaluator, InListOperator) {
@ -281,15 +280,15 @@ TEST(ExpressionEvaluator, InListOperator) {
// Element exists in list. // Element exists in list.
auto *op = storage.Create<InListOperator>( auto *op = storage.Create<InListOperator>(
storage.Create<PrimitiveLiteral>(2), list_literal); storage.Create<PrimitiveLiteral>(2), list_literal);
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), true); EXPECT_EQ(value.Value<bool>(), true);
} }
{ {
// Element doesn't exist in list. // Element doesn't exist in list.
auto *op = storage.Create<InListOperator>( auto *op = storage.Create<InListOperator>(
storage.Create<PrimitiveLiteral>("x"), list_literal); storage.Create<PrimitiveLiteral>("x"), list_literal);
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), false); EXPECT_EQ(value.Value<bool>(), false);
} }
{ {
auto *list_literal = storage.Create<ListLiteral>(std::vector<Expression *>{ auto *list_literal = storage.Create<ListLiteral>(std::vector<Expression *>{
@ -299,16 +298,16 @@ TEST(ExpressionEvaluator, InListOperator) {
// Element doesn't exist in list with null element. // Element doesn't exist in list with null element.
auto *op = storage.Create<InListOperator>( auto *op = storage.Create<InListOperator>(
storage.Create<PrimitiveLiteral>("x"), list_literal); storage.Create<PrimitiveLiteral>("x"), list_literal);
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
{ {
// Null list. // Null list.
auto *op = storage.Create<InListOperator>( auto *op = storage.Create<InListOperator>(
storage.Create<PrimitiveLiteral>("x"), storage.Create<PrimitiveLiteral>("x"),
storage.Create<PrimitiveLiteral>(TypedValue::Null)); storage.Create<PrimitiveLiteral>(TypedValue::Null));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
} }
@ -323,37 +322,37 @@ TEST(ExpressionEvaluator, ListIndexingOperator) {
// Legal indexing. // Legal indexing.
auto *op = storage.Create<ListIndexingOperator>( auto *op = storage.Create<ListIndexingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(2)); list_literal, storage.Create<PrimitiveLiteral>(2));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<int64_t>(), 3); EXPECT_EQ(value.Value<int64_t>(), 3);
} }
{ {
// Out of bounds indexing. // Out of bounds indexing.
auto *op = storage.Create<ListIndexingOperator>( auto *op = storage.Create<ListIndexingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(4)); list_literal, storage.Create<PrimitiveLiteral>(4));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().type(), TypedValue::Type::Null); EXPECT_EQ(value.type(), TypedValue::Type::Null);
} }
{ {
// Out of bounds indexing with negative bound. // Out of bounds indexing with negative bound.
auto *op = storage.Create<ListIndexingOperator>( auto *op = storage.Create<ListIndexingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(-100)); list_literal, storage.Create<PrimitiveLiteral>(-100));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().type(), TypedValue::Type::Null); EXPECT_EQ(value.type(), TypedValue::Type::Null);
} }
{ {
// Legal indexing with negative index. // Legal indexing with negative index.
auto *op = storage.Create<ListIndexingOperator>( auto *op = storage.Create<ListIndexingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(-2)); list_literal, storage.Create<PrimitiveLiteral>(-2));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<int64_t>(), 3); EXPECT_EQ(value.Value<int64_t>(), 3);
} }
{ {
// Indexing with one operator being null. // Indexing with one operator being null.
auto *op = storage.Create<ListIndexingOperator>( auto *op = storage.Create<ListIndexingOperator>(
storage.Create<PrimitiveLiteral>(TypedValue::Null), storage.Create<PrimitiveLiteral>(TypedValue::Null),
storage.Create<PrimitiveLiteral>(-2)); storage.Create<PrimitiveLiteral>(-2));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().type(), TypedValue::Type::Null); EXPECT_EQ(value.type(), TypedValue::Type::Null);
} }
{ {
// Indexing with incompatible type. // Indexing with incompatible type.
@ -384,46 +383,46 @@ TEST(ExpressionEvaluator, ListSlicingOperator) {
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(2), list_literal, storage.Create<PrimitiveLiteral>(2),
storage.Create<PrimitiveLiteral>(4)); storage.Create<PrimitiveLiteral>(4));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre(3, 4)); EXPECT_THAT(extract_ints(value), ElementsAre(3, 4));
} }
{ {
// Legal slicing with negative bound. // Legal slicing with negative bound.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(2), list_literal, storage.Create<PrimitiveLiteral>(2),
storage.Create<PrimitiveLiteral>(-1)); storage.Create<PrimitiveLiteral>(-1));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre(3)); EXPECT_THAT(extract_ints(value), ElementsAre(3));
} }
{ {
// Lower bound larger than upper bound. // Lower bound larger than upper bound.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(2), list_literal, storage.Create<PrimitiveLiteral>(2),
storage.Create<PrimitiveLiteral>(-4)); storage.Create<PrimitiveLiteral>(-4));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre()); EXPECT_THAT(extract_ints(value), ElementsAre());
} }
{ {
// Bounds ouf or range. // Bounds ouf or range.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(-100), list_literal, storage.Create<PrimitiveLiteral>(-100),
storage.Create<PrimitiveLiteral>(10)); storage.Create<PrimitiveLiteral>(10));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre(1, 2, 3, 4)); EXPECT_THAT(extract_ints(value), ElementsAre(1, 2, 3, 4));
} }
{ {
// Lower bound undefined. // Lower bound undefined.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, nullptr, storage.Create<PrimitiveLiteral>(3)); list_literal, nullptr, storage.Create<PrimitiveLiteral>(3));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre(1, 2, 3)); EXPECT_THAT(extract_ints(value), ElementsAre(1, 2, 3));
} }
{ {
// Upper bound undefined. // Upper bound undefined.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(-2), nullptr); list_literal, storage.Create<PrimitiveLiteral>(-2), nullptr);
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_THAT(extract_ints(eval.eval.PopBack()), ElementsAre(3, 4)); EXPECT_THAT(extract_ints(value), ElementsAre(3, 4));
} }
{ {
// Bound of illegal type and null value bound. // Bound of illegal type and null value bound.
@ -444,16 +443,16 @@ TEST(ExpressionEvaluator, ListSlicingOperator) {
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
storage.Create<PrimitiveLiteral>(TypedValue::Null), storage.Create<PrimitiveLiteral>(TypedValue::Null),
storage.Create<PrimitiveLiteral>(-2), nullptr); storage.Create<PrimitiveLiteral>(-2), nullptr);
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().type(), TypedValue::Type::Null); EXPECT_EQ(value.type(), TypedValue::Type::Null);
} }
{ {
// Null value index. // Null value index.
auto *op = storage.Create<ListSlicingOperator>( auto *op = storage.Create<ListSlicingOperator>(
list_literal, storage.Create<PrimitiveLiteral>(-2), list_literal, storage.Create<PrimitiveLiteral>(-2),
storage.Create<PrimitiveLiteral>(TypedValue::Null)); storage.Create<PrimitiveLiteral>(TypedValue::Null));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().type(), TypedValue::Type::Null); EXPECT_EQ(value.type(), TypedValue::Type::Null);
} }
} }
@ -462,8 +461,8 @@ TEST(ExpressionEvaluator, NotOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = auto *op =
storage.Create<NotOperator>(storage.Create<PrimitiveLiteral>(false)); storage.Create<NotOperator>(storage.Create<PrimitiveLiteral>(false));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(value.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, UnaryPlusOperator) { TEST(ExpressionEvaluator, UnaryPlusOperator) {
@ -471,8 +470,8 @@ TEST(ExpressionEvaluator, UnaryPlusOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = auto *op =
storage.Create<UnaryPlusOperator>(storage.Create<PrimitiveLiteral>(5)); storage.Create<UnaryPlusOperator>(storage.Create<PrimitiveLiteral>(5));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), 5); ASSERT_EQ(value.Value<int64_t>(), 5);
} }
TEST(ExpressionEvaluator, UnaryMinusOperator) { TEST(ExpressionEvaluator, UnaryMinusOperator) {
@ -480,8 +479,8 @@ TEST(ExpressionEvaluator, UnaryMinusOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = auto *op =
storage.Create<UnaryMinusOperator>(storage.Create<PrimitiveLiteral>(5)); storage.Create<UnaryMinusOperator>(storage.Create<PrimitiveLiteral>(5));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<int64_t>(), -5); ASSERT_EQ(value.Value<int64_t>(), -5);
} }
TEST(ExpressionEvaluator, IsNullOperator) { TEST(ExpressionEvaluator, IsNullOperator) {
@ -489,12 +488,12 @@ TEST(ExpressionEvaluator, IsNullOperator) {
NoContextExpressionEvaluator eval; NoContextExpressionEvaluator eval;
auto *op = auto *op =
storage.Create<IsNullOperator>(storage.Create<PrimitiveLiteral>(1)); storage.Create<IsNullOperator>(storage.Create<PrimitiveLiteral>(1));
op->Accept(eval.eval); auto val1 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), false); ASSERT_EQ(val1.Value<bool>(), false);
op = storage.Create<IsNullOperator>( op = storage.Create<IsNullOperator>(
storage.Create<PrimitiveLiteral>(TypedValue::Null)); storage.Create<PrimitiveLiteral>(TypedValue::Null));
op->Accept(eval.eval); auto val2 = op->Accept(eval.eval);
ASSERT_EQ(eval.eval.PopBack().Value<bool>(), true); ASSERT_EQ(val2.Value<bool>(), true);
} }
TEST(ExpressionEvaluator, PropertyLookup) { TEST(ExpressionEvaluator, PropertyLookup) {
@ -510,20 +509,20 @@ TEST(ExpressionEvaluator, PropertyLookup) {
eval.frame[node_symbol] = v1; eval.frame[node_symbol] = v1;
{ {
auto *op = storage.Create<PropertyLookup>(identifier, dba->property("age")); auto *op = storage.Create<PropertyLookup>(identifier, dba->property("age"));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<int64_t>(), 10); EXPECT_EQ(value.Value<int64_t>(), 10);
} }
{ {
auto *op = auto *op =
storage.Create<PropertyLookup>(identifier, dba->property("height")); storage.Create<PropertyLookup>(identifier, dba->property("height"));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
{ {
eval.frame[node_symbol] = TypedValue::Null; eval.frame[node_symbol] = TypedValue::Null;
auto *op = storage.Create<PropertyLookup>(identifier, dba->property("age")); auto *op = storage.Create<PropertyLookup>(identifier, dba->property("age"));
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
} }
@ -544,16 +543,16 @@ TEST(ExpressionEvaluator, LabelsTest) {
auto *op = storage.Create<LabelsTest>( auto *op = storage.Create<LabelsTest>(
identifier, std::vector<GraphDbTypes::Label>{dba->label("DOG"), identifier, std::vector<GraphDbTypes::Label>{dba->label("DOG"),
dba->label("ANIMAL")}); dba->label("ANIMAL")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), true); EXPECT_EQ(value.Value<bool>(), true);
} }
{ {
auto *op = storage.Create<LabelsTest>( auto *op = storage.Create<LabelsTest>(
identifier, identifier,
std::vector<GraphDbTypes::Label>{ std::vector<GraphDbTypes::Label>{
dba->label("DOG"), dba->label("BAD_DOG"), dba->label("ANIMAL")}); dba->label("DOG"), dba->label("BAD_DOG"), dba->label("ANIMAL")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), false); EXPECT_EQ(value.Value<bool>(), false);
} }
{ {
eval.frame[node_symbol] = TypedValue::Null; eval.frame[node_symbol] = TypedValue::Null;
@ -561,8 +560,8 @@ TEST(ExpressionEvaluator, LabelsTest) {
identifier, identifier,
std::vector<GraphDbTypes::Label>{ std::vector<GraphDbTypes::Label>{
dba->label("DOG"), dba->label("BAD_DOG"), dba->label("ANIMAL")}); dba->label("DOG"), dba->label("BAD_DOG"), dba->label("ANIMAL")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
} }
@ -583,23 +582,23 @@ TEST(ExpressionEvaluator, EdgeTypeTest) {
identifier, std::vector<GraphDbTypes::EdgeType>{ identifier, std::vector<GraphDbTypes::EdgeType>{
dba->edge_type("TYPE0"), dba->edge_type("TYPE1"), dba->edge_type("TYPE0"), dba->edge_type("TYPE1"),
dba->edge_type("TYPE2")}); dba->edge_type("TYPE2")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), true); EXPECT_EQ(value.Value<bool>(), true);
} }
{ {
auto *op = storage.Create<EdgeTypeTest>( auto *op = storage.Create<EdgeTypeTest>(
identifier, std::vector<GraphDbTypes::EdgeType>{ identifier, std::vector<GraphDbTypes::EdgeType>{
dba->edge_type("TYPE0"), dba->edge_type("TYPE2")}); dba->edge_type("TYPE0"), dba->edge_type("TYPE2")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_EQ(eval.eval.PopBack().Value<bool>(), false); EXPECT_EQ(value.Value<bool>(), false);
} }
{ {
eval.frame[edge_symbol] = TypedValue::Null; eval.frame[edge_symbol] = TypedValue::Null;
auto *op = storage.Create<EdgeTypeTest>( auto *op = storage.Create<EdgeTypeTest>(
identifier, std::vector<GraphDbTypes::EdgeType>{ identifier, std::vector<GraphDbTypes::EdgeType>{
dba->edge_type("TYPE0"), dba->edge_type("TYPE2")}); dba->edge_type("TYPE0"), dba->edge_type("TYPE2")});
op->Accept(eval.eval); auto value = op->Accept(eval.eval);
EXPECT_TRUE(eval.eval.PopBack().IsNull()); EXPECT_TRUE(value.IsNull());
} }
} }
@ -615,8 +614,8 @@ TEST(ExpressionEvaluator, Aggregation) {
Dbms dbms; Dbms dbms;
auto dba = dbms.active(); auto dba = dbms.active();
ExpressionEvaluator eval{frame, symbol_table, *dba}; ExpressionEvaluator eval{frame, symbol_table, *dba};
aggr->Accept(eval); auto value = aggr->Accept(eval);
EXPECT_EQ(eval.PopBack().Value<int64_t>(), 1); EXPECT_EQ(value.Value<int64_t>(), 1);
} }
TEST(ExpressionEvaluator, ListLiteral) { TEST(ExpressionEvaluator, ListLiteral) {
@ -626,8 +625,7 @@ TEST(ExpressionEvaluator, ListLiteral) {
std::vector<Expression *>{storage.Create<PrimitiveLiteral>(1), std::vector<Expression *>{storage.Create<PrimitiveLiteral>(1),
storage.Create<PrimitiveLiteral>("bla"), storage.Create<PrimitiveLiteral>("bla"),
storage.Create<PrimitiveLiteral>(true)}); storage.Create<PrimitiveLiteral>(true)});
list_literal->Accept(eval.eval); TypedValue result = list_literal->Accept(eval.eval);
TypedValue result = eval.eval.PopBack();
ASSERT_EQ(result.type(), TypedValue::Type::List); ASSERT_EQ(result.type(), TypedValue::Type::List);
auto &result_elems = result.Value<std::vector<TypedValue>>(); auto &result_elems = result.Value<std::vector<TypedValue>>();
ASSERT_EQ(3, result_elems.size()); ASSERT_EQ(3, result_elems.size());