#include #include #include #include #include #include "gmock/gmock.h" #include "gtest/gtest.h" #include "database/graph_db_accessor.hpp" #include "query/context.hpp" #include "query/frontend/ast/ast.hpp" #include "query/frontend/opencypher/parser.hpp" #include "query/interpret/awesome_memgraph_functions.hpp" #include "query/interpret/eval.hpp" #include "query/interpret/frame.hpp" #include "query/path.hpp" #include "storage/types.hpp" #include "utils/string.hpp" #include "query_common.hpp" using namespace query; using query::test_common::ToList; using testing::ElementsAre; using testing::UnorderedElementsAre; namespace { struct NoContextExpressionEvaluator { NoContextExpressionEvaluator() {} Frame frame{128}; database::SingleNode db; std::unique_ptr dba{db.Access()}; Context ctx{*dba}; ExpressionEvaluator eval{frame, &ctx, GraphView::OLD}; }; TypedValue EvaluateFunction(const std::string &function_name, const std::vector &args, Context *context) { AstStorage storage; Frame frame{128}; ExpressionEvaluator eval{frame, context, GraphView::OLD}; std::vector expressions; for (const auto &arg : args) { expressions.push_back(storage.Create(arg)); } auto *op = storage.Create(function_name, expressions); return op->Accept(eval); } TypedValue EvaluateFunction(const std::string &function_name, const std::vector &args) { database::SingleNode db; auto dba = db.Access(); Context ctx{*dba}; return EvaluateFunction(function_name, args, &ctx); } TEST(ExpressionEvaluator, OrOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(true), storage.Create(false)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(true), storage.Create(true)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), true); } TEST(ExpressionEvaluator, XorOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(true), storage.Create(false)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(true), storage.Create(true)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), false); } TEST(ExpressionEvaluator, AndOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(true), storage.Create(true)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(false), storage.Create(true)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), false); } TEST(ExpressionEvaluator, AndOperatorShortCircuit) { AstStorage storage; NoContextExpressionEvaluator eval; { auto *op = storage.Create(storage.Create(false), storage.Create(5)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), false); } { auto *op = storage.Create(storage.Create(5), storage.Create(false)); // We are evaluating left to right, so we don't short circuit here and raise // due to `5`. This differs from neo4j, where they evaluate both sides and // return `false` without checking for type of the first expression. EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } } TEST(ExpressionEvaluator, AndOperatorNull) { AstStorage storage; NoContextExpressionEvaluator eval; { // Null doesn't short circuit auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(5)); EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } { auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(true)); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } { auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(false)); auto value = op->Accept(eval.eval); ASSERT_TRUE(value.IsBool()); EXPECT_EQ(value.Value(), false); } } TEST(ExpressionEvaluator, AdditionOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create( storage.Create(2), storage.Create(3)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 5); } TEST(ExpressionEvaluator, SubtractionOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create( storage.Create(2), storage.Create(3)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), -1); } TEST(ExpressionEvaluator, MultiplicationOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create( storage.Create(2), storage.Create(3)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 6); } TEST(ExpressionEvaluator, DivisionOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(50), storage.Create(10)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 5); } TEST(ExpressionEvaluator, ModOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(65), storage.Create(10)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 5); } TEST(ExpressionEvaluator, EqualOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), false); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), true); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), false); } TEST(ExpressionEvaluator, NotEqualOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), true); } TEST(ExpressionEvaluator, LessOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), false); } TEST(ExpressionEvaluator, GreaterOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), false); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), true); } TEST(ExpressionEvaluator, LessEqualOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), true); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), false); } TEST(ExpressionEvaluator, GreaterEqualOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create( storage.Create(10), storage.Create(15)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), false); op = storage.Create( storage.Create(15), storage.Create(15)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), true); op = storage.Create( storage.Create(20), storage.Create(15)); auto val3 = op->Accept(eval.eval); ASSERT_EQ(val3.Value(), true); } TEST(ExpressionEvaluator, InListOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *list_literal = storage.Create(std::vector{ storage.Create(1), storage.Create(2), storage.Create("a")}); { // Element exists in list. auto *op = storage.Create( storage.Create(2), list_literal); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), true); } { // Element doesn't exist in list. auto *op = storage.Create( storage.Create("x"), list_literal); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), false); } { auto *list_literal = storage.Create(std::vector{ storage.Create(TypedValue::Null), storage.Create(2), storage.Create("a")}); // Element doesn't exist in list with null element. auto *op = storage.Create( storage.Create("x"), list_literal); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } { // Null list. auto *op = storage.Create( storage.Create("x"), storage.Create(TypedValue::Null)); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } { // Null literal. auto *op = storage.Create( storage.Create(TypedValue::Null), list_literal); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } { // Null literal, empty list. auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(std::vector())); auto value = op->Accept(eval.eval); EXPECT_FALSE(value.ValueBool()); } } TEST(ExpressionEvaluator, ListIndexing) { AstStorage storage; NoContextExpressionEvaluator eval; auto *list_literal = storage.Create(std::vector{ storage.Create(1), storage.Create(2), storage.Create(3), storage.Create(4)}); { // Legal indexing. auto *op = storage.Create( list_literal, storage.Create(2)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), 3); } { // Out of bounds indexing. auto *op = storage.Create( list_literal, storage.Create(4)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::Null); } { // Out of bounds indexing with negative bound. auto *op = storage.Create( list_literal, storage.Create(-100)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::Null); } { // Legal indexing with negative index. auto *op = storage.Create( list_literal, storage.Create(-2)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), 3); } { // Indexing with one operator being null. auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(-2)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::Null); } { // Indexing with incompatible type. auto *op = storage.Create( list_literal, storage.Create("bla")); EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } } TEST(ExpressionEvaluator, MapIndexing) { AstStorage storage; NoContextExpressionEvaluator eval; database::SingleNode db; auto dba_ptr = db.Access(); auto &dba = *dba_ptr; auto *map_literal = storage.Create( std::unordered_map, Expression *>{ {PROPERTY_PAIR("a"), storage.Create(1)}, {PROPERTY_PAIR("b"), storage.Create(2)}, {PROPERTY_PAIR("c"), storage.Create(3)}}); { // Legal indexing. auto *op = storage.Create( map_literal, storage.Create("b")); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), 2); } { // Legal indexing, non-existing key. auto *op = storage.Create( map_literal, storage.Create("z")); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } { // Wrong key type. auto *op = storage.Create( map_literal, storage.Create(42)); EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } { // Indexing with Null. auto *op = storage.Create( map_literal, storage.Create(TypedValue::Null)); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } } TEST(ExpressionEvaluator, VertexAndEdgeIndexing) { AstStorage storage; NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto edge_type = dba.EdgeType("edge_type"); auto prop = dba.Property("prop"); auto v1 = dba.InsertVertex(); auto e11 = dba.InsertEdge(v1, v1, edge_type); v1.PropsSet(prop, 42); e11.PropsSet(prop, 43); auto *vertex_literal = storage.Create(v1); auto *edge_literal = storage.Create(e11); { // Legal indexing. auto *op1 = storage.Create( vertex_literal, storage.Create("prop")); auto value1 = op1->Accept(eval.eval); EXPECT_EQ(value1.Value(), 42); auto *op2 = storage.Create( edge_literal, storage.Create("prop")); auto value2 = op2->Accept(eval.eval); EXPECT_EQ(value2.Value(), 43); } { // Legal indexing, non-existing key. auto *op1 = storage.Create( vertex_literal, storage.Create("blah")); auto value1 = op1->Accept(eval.eval); EXPECT_TRUE(value1.IsNull()); auto *op2 = storage.Create( edge_literal, storage.Create("blah")); auto value2 = op2->Accept(eval.eval); EXPECT_TRUE(value2.IsNull()); } { // Wrong key type. auto *op1 = storage.Create( vertex_literal, storage.Create(1)); EXPECT_THROW(op1->Accept(eval.eval), QueryRuntimeException); auto *op2 = storage.Create( edge_literal, storage.Create(1)); EXPECT_THROW(op2->Accept(eval.eval), QueryRuntimeException); } { // Indexing with Null. auto *op1 = storage.Create( vertex_literal, storage.Create(TypedValue::Null)); auto value1 = op1->Accept(eval.eval); EXPECT_TRUE(value1.IsNull()); auto *op2 = storage.Create( edge_literal, storage.Create(TypedValue::Null)); auto value2 = op2->Accept(eval.eval); EXPECT_TRUE(value2.IsNull()); } } TEST(ExpressionEvaluator, ListSlicingOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *list_literal = storage.Create(std::vector{ storage.Create(1), storage.Create(2), storage.Create(3), storage.Create(4)}); auto extract_ints = [](TypedValue list) { std::vector int_list; for (auto x : list.Value>()) { int_list.push_back(x.Value()); } return int_list; }; { // Legal slicing with both bounds defined. auto *op = storage.Create( list_literal, storage.Create(2), storage.Create(4)); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre(3, 4)); } { // Legal slicing with negative bound. auto *op = storage.Create( list_literal, storage.Create(2), storage.Create(-1)); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre(3)); } { // Lower bound larger than upper bound. auto *op = storage.Create( list_literal, storage.Create(2), storage.Create(-4)); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre()); } { // Bounds ouf or range. auto *op = storage.Create( list_literal, storage.Create(-100), storage.Create(10)); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre(1, 2, 3, 4)); } { // Lower bound undefined. auto *op = storage.Create( list_literal, nullptr, storage.Create(3)); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre(1, 2, 3)); } { // Upper bound undefined. auto *op = storage.Create( list_literal, storage.Create(-2), nullptr); auto value = op->Accept(eval.eval); EXPECT_THAT(extract_ints(value), ElementsAre(3, 4)); } { // Bound of illegal type and null value bound. auto *op = storage.Create( list_literal, storage.Create(TypedValue::Null), storage.Create("mirko")); EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } { // List of illegal type. auto *op = storage.Create( storage.Create("a"), storage.Create(-2), nullptr); EXPECT_THROW(op->Accept(eval.eval), QueryRuntimeException); } { // Null value list with undefined upper bound. auto *op = storage.Create( storage.Create(TypedValue::Null), storage.Create(-2), nullptr); auto value = op->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::Null); } { // Null value index. auto *op = storage.Create( list_literal, storage.Create(-2), storage.Create(TypedValue::Null)); auto value = op->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::Null); } } TEST(ExpressionEvaluator, IfOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *then_expression = storage.Create(10); auto *else_expression = storage.Create(20); { auto *condition_true = storage.Create(storage.Create(2), storage.Create(2)); auto *op = storage.Create(condition_true, then_expression, else_expression); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 10); } { auto *condition_false = storage.Create(storage.Create(2), storage.Create(3)); auto *op = storage.Create(condition_false, then_expression, else_expression); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 20); } { auto *condition_exception = storage.Create(storage.Create(2), storage.Create(3)); auto *op = storage.Create(condition_exception, then_expression, else_expression); ASSERT_THROW(op->Accept(eval.eval), QueryRuntimeException); } } TEST(ExpressionEvaluator, NotOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(false)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), true); } TEST(ExpressionEvaluator, UnaryPlusOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(5)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), 5); } TEST(ExpressionEvaluator, UnaryMinusOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(5)); auto value = op->Accept(eval.eval); ASSERT_EQ(value.Value(), -5); } TEST(ExpressionEvaluator, IsNullOperator) { AstStorage storage; NoContextExpressionEvaluator eval; auto *op = storage.Create(storage.Create(1)); auto val1 = op->Accept(eval.eval); ASSERT_EQ(val1.Value(), false); op = storage.Create( storage.Create(TypedValue::Null)); auto val2 = op->Accept(eval.eval); ASSERT_EQ(val2.Value(), true); } class ExpressionEvaluatorPropertyLookup : public testing::Test { protected: AstStorage storage; NoContextExpressionEvaluator eval; database::SingleNode db; std::unique_ptr dba_ptr{db.Access()}; database::GraphDbAccessor &dba{*dba_ptr}; std::pair prop_age = PROPERTY_PAIR("age"); std::pair prop_height = PROPERTY_PAIR("height"); Expression *identifier = storage.Create("element"); Symbol symbol = eval.ctx.symbol_table_.CreateSymbol("element", true); void SetUp() { eval.ctx.symbol_table_[*identifier] = symbol; } auto Value(std::pair property) { auto *op = storage.Create(identifier, property); return op->Accept(eval.eval); } }; TEST_F(ExpressionEvaluatorPropertyLookup, Vertex) { auto v1 = dba.InsertVertex(); v1.PropsSet(prop_age.second, 10); eval.frame[symbol] = v1; EXPECT_EQ(Value(prop_age).Value(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, Edge) { auto v1 = dba.InsertVertex(); auto v2 = dba.InsertVertex(); auto e12 = dba.InsertEdge(v1, v2, dba.EdgeType("edge_type")); e12.PropsSet(prop_age.second, 10); eval.frame[symbol] = e12; EXPECT_EQ(Value(prop_age).Value(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, Null) { eval.frame[symbol] = TypedValue::Null; EXPECT_TRUE(Value(prop_age).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, MapLiteral) { eval.frame[symbol] = std::map{{prop_age.first, 10}}; EXPECT_EQ(Value(prop_age).Value(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } TEST(ExpressionEvaluator, LabelsTest) { AstStorage storage; NoContextExpressionEvaluator eval; database::SingleNode db; auto dba = db.Access(); auto v1 = dba->InsertVertex(); v1.add_label(dba->Label("ANIMAL")); v1.add_label(dba->Label("DOG")); v1.add_label(dba->Label("NICE_DOG")); auto *identifier = storage.Create("n"); auto node_symbol = eval.ctx.symbol_table_.CreateSymbol("n", true); eval.ctx.symbol_table_[*identifier] = node_symbol; eval.frame[node_symbol] = v1; { auto *op = storage.Create( identifier, std::vector{dba->Label("DOG"), dba->Label("ANIMAL")}); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), true); } { auto *op = storage.Create( identifier, std::vector{dba->Label("DOG"), dba->Label("BAD_DOG"), dba->Label("ANIMAL")}); auto value = op->Accept(eval.eval); EXPECT_EQ(value.Value(), false); } { eval.frame[node_symbol] = TypedValue::Null; auto *op = storage.Create( identifier, std::vector{dba->Label("DOG"), dba->Label("BAD_DOG"), dba->Label("ANIMAL")}); auto value = op->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } } TEST(ExpressionEvaluator, Aggregation) { AstStorage storage; auto aggr = storage.Create(storage.Create(42), nullptr, Aggregation::Op::COUNT); database::SingleNode db; auto dba = db.Access(); Context ctx(*dba); auto aggr_sym = ctx.symbol_table_.CreateSymbol("aggr", true); ctx.symbol_table_[*aggr] = aggr_sym; Frame frame{ctx.symbol_table_.max_position()}; frame[aggr_sym] = TypedValue(1); Parameters parameters; ExpressionEvaluator eval{frame, &ctx, GraphView::OLD}; auto value = aggr->Accept(eval); EXPECT_EQ(value.Value(), 1); } TEST(ExpressionEvaluator, ListLiteral) { AstStorage storage; NoContextExpressionEvaluator eval; auto *list_literal = storage.Create( std::vector{storage.Create(1), storage.Create("bla"), storage.Create(true)}); TypedValue result = list_literal->Accept(eval.eval); ASSERT_EQ(result.type(), TypedValue::Type::List); auto &result_elems = result.Value>(); ASSERT_EQ(3, result_elems.size()); EXPECT_EQ(result_elems[0].type(), TypedValue::Type::Int); EXPECT_EQ(result_elems[1].type(), TypedValue::Type::String); EXPECT_EQ(result_elems[2].type(), TypedValue::Type::Bool); } TEST(ExpressionEvaluator, FunctionCoalesce) { ASSERT_THROW(EvaluateFunction("COALESCE", {}), QueryRuntimeException); ASSERT_EQ( EvaluateFunction("COALESCE", {TypedValue::Null, TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ( EvaluateFunction("COALESCE", {TypedValue::Null, 2, 3}).Value(), 2); } TEST(ExpressionEvaluator, FunctionEndNode) { ASSERT_THROW(EvaluateFunction("ENDNODE", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("ENDNODE", {TypedValue::Null}).type(), TypedValue::Type::Null); database::SingleNode db; auto dba = db.Access(); auto v1 = dba->InsertVertex(); v1.add_label(dba->Label("label1")); auto v2 = dba->InsertVertex(); v2.add_label(dba->Label("label2")); auto e = dba->InsertEdge(v1, v2, dba->EdgeType("t")); ASSERT_TRUE(EvaluateFunction("ENDNODE", {e}) .Value() .has_label(dba->Label("label2"))); ASSERT_THROW(EvaluateFunction("ENDNODE", {2}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionHead) { ASSERT_THROW(EvaluateFunction("HEAD", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("HEAD", {TypedValue::Null}).type(), TypedValue::Type::Null); std::vector arguments; arguments.push_back(std::vector{3, 4, 5}); ASSERT_EQ(EvaluateFunction("HEAD", arguments).Value(), 3); arguments[0].Value>().clear(); ASSERT_EQ(EvaluateFunction("HEAD", arguments).type(), TypedValue::Type::Null); ASSERT_THROW(EvaluateFunction("HEAD", {2}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionProperties) { ASSERT_THROW(EvaluateFunction("PROPERTIES", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("PROPERTIES", {TypedValue::Null}).type(), TypedValue::Type::Null); NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto v1 = dba.InsertVertex(); v1.PropsSet(dba.Property("height"), 5); v1.PropsSet(dba.Property("age"), 10); auto v2 = dba.InsertVertex(); auto e = dba.InsertEdge(v1, v2, dba.EdgeType("type1")); e.PropsSet(dba.Property("height"), 3); e.PropsSet(dba.Property("age"), 15); auto prop_values_to_int = [](TypedValue t) { std::unordered_map properties; for (auto property : t.Value>()) { properties[property.first] = property.second.Value(); } return properties; }; ASSERT_THAT( prop_values_to_int(EvaluateFunction("PROPERTIES", {v1}, &eval.ctx)), UnorderedElementsAre(testing::Pair("height", 5), testing::Pair("age", 10))); ASSERT_THAT( prop_values_to_int(EvaluateFunction("PROPERTIES", {e}, &eval.ctx)), UnorderedElementsAre(testing::Pair("height", 3), testing::Pair("age", 15))); ASSERT_THROW(EvaluateFunction("PROPERTIES", {2}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionLast) { ASSERT_THROW(EvaluateFunction("LAST", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("LAST", {TypedValue::Null}).type(), TypedValue::Type::Null); std::vector arguments; arguments.push_back(std::vector{3, 4, 5}); ASSERT_EQ(EvaluateFunction("LAST", arguments).Value(), 5); arguments[0].Value>().clear(); ASSERT_EQ(EvaluateFunction("LAST", arguments).type(), TypedValue::Type::Null); ASSERT_THROW(EvaluateFunction("LAST", {5}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionSize) { ASSERT_THROW(EvaluateFunction("SIZE", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("SIZE", {TypedValue::Null}).type(), TypedValue::Type::Null); std::vector arguments; arguments.push_back(std::vector{3, 4, 5}); ASSERT_EQ(EvaluateFunction("SIZE", arguments).Value(), 3); ASSERT_EQ(EvaluateFunction("SIZE", {"john"}).Value(), 4); ASSERT_EQ(EvaluateFunction("SIZE", {std::map{ {"a", 5}, {"b", true}, {"c", "123"}}}) .Value(), 3); ASSERT_THROW(EvaluateFunction("SIZE", {5}), QueryRuntimeException); database::SingleNode db; auto dba = db.Access(); auto v0 = dba->InsertVertex(); query::Path path(v0); EXPECT_EQ(EvaluateFunction("SIZE", {path}).ValueInt(), 0); auto v1 = dba->InsertVertex(); path.Expand(dba->InsertEdge(v0, v1, dba->EdgeType("type"))); path.Expand(v1); EXPECT_EQ(EvaluateFunction("SIZE", {path}).ValueInt(), 1); } TEST(ExpressionEvaluator, FunctionStartNode) { ASSERT_THROW(EvaluateFunction("STARTNODE", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("STARTNODE", {TypedValue::Null}).type(), TypedValue::Type::Null); database::SingleNode db; auto dba = db.Access(); auto v1 = dba->InsertVertex(); v1.add_label(dba->Label("label1")); auto v2 = dba->InsertVertex(); v2.add_label(dba->Label("label2")); auto e = dba->InsertEdge(v1, v2, dba->EdgeType("t")); ASSERT_TRUE(EvaluateFunction("STARTNODE", {e}) .Value() .has_label(dba->Label("label1"))); ASSERT_THROW(EvaluateFunction("STARTNODE", {2}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionDegree) { ASSERT_THROW(EvaluateFunction("DEGREE", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("DEGREE", {TypedValue::Null}).type(), TypedValue::Type::Null); database::SingleNode db; auto dba = db.Access(); auto v1 = dba->InsertVertex(); auto v2 = dba->InsertVertex(); auto v3 = dba->InsertVertex(); auto e12 = dba->InsertEdge(v1, v2, dba->EdgeType("t")); dba->InsertEdge(v3, v2, dba->EdgeType("t")); ASSERT_EQ(EvaluateFunction("DEGREE", {v1}).Value(), 1); ASSERT_EQ(EvaluateFunction("DEGREE", {v2}).Value(), 2); ASSERT_EQ(EvaluateFunction("DEGREE", {v3}).Value(), 1); ASSERT_THROW(EvaluateFunction("DEGREE", {2}), QueryRuntimeException); ASSERT_THROW(EvaluateFunction("DEGREE", {e12}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionToBoolean) { ASSERT_THROW(EvaluateFunction("TOBOOLEAN", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {123}).ValueBool(), true); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {-213}).ValueBool(), true); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {0}).ValueBool(), false); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {" trUE \n\t"}).Value(), true); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {"\n\tFalsE "}).Value(), false); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {"\n\tFALSEA "}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {true}).Value(), true); ASSERT_EQ(EvaluateFunction("TOBOOLEAN", {false}).Value(), false); } TEST(ExpressionEvaluator, FunctionToFloat) { ASSERT_THROW(EvaluateFunction("TOFLOAT", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("TOFLOAT", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOFLOAT", {" -3.5 \n\t"}).Value(), -3.5); ASSERT_EQ(EvaluateFunction("TOFLOAT", {"\n\t0.5e-1"}).Value(), 0.05); ASSERT_EQ(EvaluateFunction("TOFLOAT", {"\n\t3.4e-3X "}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOFLOAT", {-3.5}).Value(), -3.5); ASSERT_EQ(EvaluateFunction("TOFLOAT", {-3}).Value(), -3.0); ASSERT_THROW(EvaluateFunction("TOFLOAT", {true}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionToInteger) { ASSERT_THROW(EvaluateFunction("TOINTEGER", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("TOINTEGER", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOINTEGER", {false}).Value(), 0); ASSERT_EQ(EvaluateFunction("TOINTEGER", {true}).Value(), 1); ASSERT_EQ(EvaluateFunction("TOINTEGER", {"\n\t3"}).Value(), 3); ASSERT_EQ(EvaluateFunction("TOINTEGER", {" -3.5 \n\t"}).Value(), -3); ASSERT_EQ(EvaluateFunction("TOINTEGER", {"\n\t3X "}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("TOINTEGER", {-3.5}).Value(), -3); ASSERT_EQ(EvaluateFunction("TOINTEGER", {3.5}).Value(), 3); } TEST(ExpressionEvaluator, FunctionType) { ASSERT_THROW(EvaluateFunction("TYPE", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("TYPE", {TypedValue::Null}).type(), TypedValue::Type::Null); NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto v1 = dba.InsertVertex(); v1.add_label(dba.Label("label1")); auto v2 = dba.InsertVertex(); v2.add_label(dba.Label("label2")); auto e = dba.InsertEdge(v1, v2, dba.EdgeType("type1")); ASSERT_EQ(EvaluateFunction("TYPE", {e}, &eval.ctx).Value(), "type1"); ASSERT_THROW(EvaluateFunction("TYPE", {2}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionLabels) { ASSERT_THROW(EvaluateFunction("LABELS", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("LABELS", {TypedValue::Null}).type(), TypedValue::Type::Null); NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto v = dba.InsertVertex(); v.add_label(dba.Label("label1")); v.add_label(dba.Label("label2")); std::vector labels; auto _labels = EvaluateFunction("LABELS", {v}, &eval.ctx) .Value>(); for (auto label : _labels) { labels.push_back(label.Value()); } ASSERT_THAT(labels, UnorderedElementsAre("label1", "label2")); ASSERT_THROW(EvaluateFunction("LABELS", {2}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionNodesRelationships) { EXPECT_THROW(EvaluateFunction("NODES", {}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RELATIONSHIPS", {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction("NODES", {TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("RELATIONSHIPS", {TypedValue::Null}).IsNull()); { NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto v1 = dba.InsertVertex(); auto v2 = dba.InsertVertex(); auto v3 = dba.InsertVertex(); auto e1 = dba.InsertEdge(v1, v2, dba.EdgeType("Type")); auto e2 = dba.InsertEdge(v2, v3, dba.EdgeType("Type")); query::Path path(v1, e1, v2, e2, v3); auto _nodes = EvaluateFunction("NODES", {path}).ValueList(); std::vector nodes; for (const auto &node : _nodes) { nodes.push_back(node.ValueVertex()); } EXPECT_THAT(nodes, ElementsAre(v1, v2, v3)); auto _edges = EvaluateFunction("RELATIONSHIPS", {path}).ValueList(); std::vector edges; for (const auto &edge : _edges) { edges.push_back(edge.ValueEdge()); } EXPECT_THAT(edges, ElementsAre(e1, e2)); } EXPECT_THROW(EvaluateFunction("NODES", {2}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RELATIONSHIPS", {2}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionRange) { EXPECT_THROW(EvaluateFunction("RANGE", {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction("RANGE", {1, 2, TypedValue::Null}).IsNull()); EXPECT_THROW(EvaluateFunction("RANGE", {1, TypedValue::Null, 1.3}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RANGE", {1, 2, 0}), QueryRuntimeException); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {1, 3})), ElementsAre(1, 2, 3)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {-1, 5, 2})), ElementsAre(-1, 1, 3, 5)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {2, 10, 3})), ElementsAre(2, 5, 8)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {2, 2, 2})), ElementsAre(2)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {3, 0, 5})), ElementsAre()); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {5, 1, -2})), ElementsAre(5, 3, 1)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {6, 1, -2})), ElementsAre(6, 4, 2)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {2, 2, -3})), ElementsAre(2)); EXPECT_THAT(ToList(EvaluateFunction("RANGE", {-2, 4, -1})), ElementsAre()); } TEST(ExpressionEvaluator, FunctionKeys) { ASSERT_THROW(EvaluateFunction("KEYS", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("KEYS", {TypedValue::Null}).type(), TypedValue::Type::Null); NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto v1 = dba.InsertVertex(); v1.PropsSet(dba.Property("height"), 5); v1.PropsSet(dba.Property("age"), 10); auto v2 = dba.InsertVertex(); auto e = dba.InsertEdge(v1, v2, dba.EdgeType("type1")); e.PropsSet(dba.Property("width"), 3); e.PropsSet(dba.Property("age"), 15); auto prop_keys_to_string = [](TypedValue t) { std::vector keys; for (auto property : t.Value>()) { keys.push_back(property.Value()); } return keys; }; ASSERT_THAT(prop_keys_to_string(EvaluateFunction("KEYS", {v1}, &eval.ctx)), UnorderedElementsAre("height", "age")); ASSERT_THAT(prop_keys_to_string(EvaluateFunction("KEYS", {e}, &eval.ctx)), UnorderedElementsAre("width", "age")); ASSERT_THROW(EvaluateFunction("KEYS", {2}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionTail) { ASSERT_THROW(EvaluateFunction("TAIL", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("TAIL", {TypedValue::Null}).type(), TypedValue::Type::Null); std::vector arguments; arguments.push_back(std::vector{}); ASSERT_EQ(EvaluateFunction("TAIL", arguments) .Value>() .size(), 0U); arguments[0] = std::vector{3, 4, true, "john"}; auto list = EvaluateFunction("TAIL", arguments).Value>(); ASSERT_EQ(list.size(), 3U); ASSERT_EQ(list[0].Value(), 4); ASSERT_EQ(list[1].Value(), true); ASSERT_EQ(list[2].Value(), "john"); ASSERT_THROW(EvaluateFunction("TAIL", {2}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionAbs) { ASSERT_THROW(EvaluateFunction("ABS", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("ABS", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("ABS", {-2}).Value(), 2); ASSERT_EQ(EvaluateFunction("ABS", {-2.5}).Value(), 2.5); ASSERT_THROW(EvaluateFunction("ABS", {true}), QueryRuntimeException); } // Test if log works. If it does then all functions wrapped with // WRAP_CMATH_FLOAT_FUNCTION macro should work and are not gonna be tested for // correctnes.. TEST(ExpressionEvaluator, FunctionLog) { ASSERT_THROW(EvaluateFunction("LOG", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("LOG", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_DOUBLE_EQ(EvaluateFunction("LOG", {2}).Value(), log(2)); ASSERT_DOUBLE_EQ(EvaluateFunction("LOG", {1.5}).Value(), log(1.5)); // Not portable, but should work on most platforms. ASSERT_TRUE(std::isnan(EvaluateFunction("LOG", {-1.5}).Value())); ASSERT_THROW(EvaluateFunction("LOG", {true}), QueryRuntimeException); } // Function Round wraps round from cmath and will work if FunctionLog test // passes. This test is used to show behavior of round since it differs from // neo4j's round. TEST(ExpressionEvaluator, FunctionRound) { ASSERT_THROW(EvaluateFunction("ROUND", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("ROUND", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("ROUND", {-2}).Value(), -2); ASSERT_EQ(EvaluateFunction("ROUND", {-2.4}).Value(), -2); ASSERT_EQ(EvaluateFunction("ROUND", {-2.5}).Value(), -3); ASSERT_EQ(EvaluateFunction("ROUND", {-2.6}).Value(), -3); ASSERT_EQ(EvaluateFunction("ROUND", {2.4}).Value(), 2); ASSERT_EQ(EvaluateFunction("ROUND", {2.5}).Value(), 3); ASSERT_EQ(EvaluateFunction("ROUND", {2.6}).Value(), 3); ASSERT_THROW(EvaluateFunction("ROUND", {true}), QueryRuntimeException); } // Check if wrapped functions are callable (check if everything was spelled // correctly...). Wrapper correctnes is checked in FunctionLog test. TEST(ExpressionEvaluator, FunctionWrappedMathFunctions) { for (auto function_name : {"FLOOR", "CEIL", "ROUND", "EXP", "LOG", "LOG10", "SQRT", "ACOS", "ASIN", "ATAN", "COS", "SIN", "TAN"}) { EvaluateFunction(function_name, {0.5}); } } TEST(ExpressionEvaluator, FunctionAtan2) { ASSERT_THROW(EvaluateFunction("ATAN2", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("ATAN2", {TypedValue::Null, 1}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("ATAN2", {1, TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_DOUBLE_EQ(EvaluateFunction("ATAN2", {2, -1.0}).Value(), atan2(2, -1)); ASSERT_THROW(EvaluateFunction("ATAN2", {3.0, true}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionSign) { ASSERT_THROW(EvaluateFunction("SIGN", {}), QueryRuntimeException); ASSERT_EQ(EvaluateFunction("SIGN", {TypedValue::Null}).type(), TypedValue::Type::Null); ASSERT_EQ(EvaluateFunction("SIGN", {-2}).Value(), -1); ASSERT_EQ(EvaluateFunction("SIGN", {-0.2}).Value(), -1); ASSERT_EQ(EvaluateFunction("SIGN", {0.0}).Value(), 0); ASSERT_EQ(EvaluateFunction("SIGN", {2.5}).Value(), 1); ASSERT_THROW(EvaluateFunction("SIGN", {true}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionE) { ASSERT_THROW(EvaluateFunction("E", {1}), QueryRuntimeException); ASSERT_DOUBLE_EQ(EvaluateFunction("E", {}).Value(), M_E); } TEST(ExpressionEvaluator, FunctionPi) { ASSERT_THROW(EvaluateFunction("PI", {1}), QueryRuntimeException); ASSERT_DOUBLE_EQ(EvaluateFunction("PI", {}).Value(), M_PI); } TEST(ExpressionEvaluator, FunctionRand) { ASSERT_THROW(EvaluateFunction("RAND", {1}), QueryRuntimeException); ASSERT_GE(EvaluateFunction("RAND", {}).Value(), 0.0); ASSERT_LT(EvaluateFunction("RAND", {}).Value(), 1.0); } TEST(ExpressionEvaluator, FunctionStartsWith) { EXPECT_THROW(EvaluateFunction(kStartsWith, {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kStartsWith, {"a", TypedValue::Null}).IsNull()); EXPECT_THROW(EvaluateFunction(kStartsWith, {TypedValue::Null, 1.3}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kStartsWith, {"abc", "abc"}).Value()); EXPECT_TRUE(EvaluateFunction(kStartsWith, {"abcdef", "abc"}).Value()); EXPECT_FALSE(EvaluateFunction(kStartsWith, {"abcdef", "aBc"}).Value()); EXPECT_FALSE(EvaluateFunction(kStartsWith, {"abc", "abcd"}).Value()); } TEST(ExpressionEvaluator, FunctionEndsWith) { EXPECT_THROW(EvaluateFunction(kEndsWith, {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kEndsWith, {"a", TypedValue::Null}).IsNull()); EXPECT_THROW(EvaluateFunction(kEndsWith, {TypedValue::Null, 1.3}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kEndsWith, {"abc", "abc"}).Value()); EXPECT_TRUE(EvaluateFunction(kEndsWith, {"abcdef", "def"}).Value()); EXPECT_FALSE(EvaluateFunction(kEndsWith, {"abcdef", "dEf"}).Value()); EXPECT_FALSE(EvaluateFunction(kEndsWith, {"bcd", "abcd"}).Value()); } TEST(ExpressionEvaluator, FunctionContains) { EXPECT_THROW(EvaluateFunction(kContains, {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kContains, {"a", TypedValue::Null}).IsNull()); EXPECT_THROW(EvaluateFunction(kContains, {TypedValue::Null, 1.3}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction(kContains, {"abc", "abc"}).Value()); EXPECT_TRUE(EvaluateFunction(kContains, {"abcde", "bcd"}).Value()); EXPECT_FALSE(EvaluateFunction(kContains, {"cde", "abcdef"}).Value()); EXPECT_FALSE(EvaluateFunction(kContains, {"abcdef", "dEf"}).Value()); } TEST(ExpressionEvaluator, FunctionAll) { AstStorage storage; auto *ident_x = IDENT("x"); auto *all = ALL("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*all->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = all->Accept(eval.eval); ASSERT_EQ(value.type(), TypedValue::Type::Bool); EXPECT_FALSE(value.Value()); } TEST(ExpressionEvaluator, FunctionAllNullList) { AstStorage storage; auto *all = ALL("x", LITERAL(TypedValue::Null), WHERE(LITERAL(true))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*all->identifier_] = x_sym; auto value = all->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } TEST(ExpressionEvaluator, FunctionAllWhereWrongType) { AstStorage storage; auto *all = ALL("x", LIST(LITERAL(1)), WHERE(LITERAL(2))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*all->identifier_] = x_sym; EXPECT_THROW(all->Accept(eval.eval), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionSingle) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*single->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = single->Accept(eval.eval); ASSERT_EQ(value.type(), TypedValue::Type::Bool); EXPECT_TRUE(value.Value()); } TEST(ExpressionEvaluator, FunctionSingle2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(GREATER(ident_x, LITERAL(0)))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*single->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = single->Accept(eval.eval); ASSERT_EQ(value.type(), TypedValue::Type::Bool); EXPECT_FALSE(value.Value()); } TEST(ExpressionEvaluator, FunctionSingleNullList) { AstStorage storage; auto *single = SINGLE("x", LITERAL(TypedValue::Null), WHERE(LITERAL(true))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*single->identifier_] = x_sym; auto value = single->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } TEST(ExpressionEvaluator, FunctionReduce) { AstStorage storage; auto *ident_sum = IDENT("sum"); auto *ident_x = IDENT("x"); auto *reduce = REDUCE("sum", LITERAL(0), "x", LIST(LITERAL(1), LITERAL(2)), ADD(ident_sum, ident_x)); NoContextExpressionEvaluator eval; const auto sum_sym = eval.ctx.symbol_table_.CreateSymbol("sum", true); eval.ctx.symbol_table_[*reduce->accumulator_] = sum_sym; eval.ctx.symbol_table_[*ident_sum] = sum_sym; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*reduce->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = reduce->Accept(eval.eval); ASSERT_EQ(value.type(), TypedValue::Type::Int); EXPECT_EQ(value.Value(), 3); } TEST(ExpressionEvaluator, FunctionExtract) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LIST(LITERAL(1), LITERAL(2), LITERAL(TypedValue::Null)), ADD(ident_x, LITERAL(1))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*extract->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = extract->Accept(eval.eval); EXPECT_EQ(value.type(), TypedValue::Type::List); auto result = value.ValueList(); EXPECT_EQ(result[0].ValueInt(), 2); EXPECT_EQ(result[1].ValueInt(), 3); EXPECT_TRUE(result[2].IsNull()); } TEST(ExpressionEvaluator, FunctionExtractNull) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LITERAL(TypedValue::Null), ADD(ident_x, LITERAL(1))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*extract->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; auto value = extract->Accept(eval.eval); EXPECT_TRUE(value.IsNull()); } TEST(ExpressionEvaluator, FunctionExtractExceptions) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LITERAL("bla"), ADD(ident_x, LITERAL(1))); NoContextExpressionEvaluator eval; const auto x_sym = eval.ctx.symbol_table_.CreateSymbol("x", true); eval.ctx.symbol_table_[*extract->identifier_] = x_sym; eval.ctx.symbol_table_[*ident_x] = x_sym; EXPECT_THROW(extract->Accept(eval.eval), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionAssert) { // Invalid calls. ASSERT_THROW(EvaluateFunction("ASSERT", {}), QueryRuntimeException); ASSERT_THROW(EvaluateFunction("ASSERT", {false, false}), QueryRuntimeException); ASSERT_THROW(EvaluateFunction("ASSERT", {"string", false}), QueryRuntimeException); ASSERT_THROW(EvaluateFunction("ASSERT", {false, "reason", true}), QueryRuntimeException); // Valid calls, assertion fails. ASSERT_THROW(EvaluateFunction("ASSERT", {false}), QueryRuntimeException); ASSERT_THROW(EvaluateFunction("ASSERT", {false, "message"}), QueryRuntimeException); try { EvaluateFunction("ASSERT", {false, "bbgba"}); } catch (QueryRuntimeException &e) { ASSERT_TRUE(utils::EndsWith(e.what(), "bbgba")); } // Valid calls, assertion passes. ASSERT_TRUE(EvaluateFunction("ASSERT", {true}).ValueBool()); ASSERT_TRUE(EvaluateFunction("ASSERT", {true, "message"}).ValueBool()); } TEST(ExpressionEvaluator, ParameterLookup) { NoContextExpressionEvaluator eval; eval.ctx.parameters_.Add(0, 42); AstStorage storage; auto *param_lookup = storage.Create(0); auto value = param_lookup->Accept(eval.eval); ASSERT_EQ(value.type(), TypedValue::Type::Int); EXPECT_EQ(value.Value(), 42); } TEST(ExpressionEvaluator, FunctionCounter) { NoContextExpressionEvaluator eval; EXPECT_THROW(EvaluateFunction("COUNTER", {}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("COUNTER", {"a", "b"}, &eval.ctx), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 0); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 1); EXPECT_EQ(EvaluateFunction("COUNTER", {"c2"}, &eval.ctx).ValueInt(), 0); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 2); EXPECT_EQ(EvaluateFunction("COUNTER", {"c2"}, &eval.ctx).ValueInt(), 1); } TEST(ExpressionEvaluator, FunctionCounterSet) { NoContextExpressionEvaluator eval; EXPECT_THROW(EvaluateFunction("COUNTERSET", {}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("COUNTERSET", {"a"}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("COUNTERSET", {"a", "b"}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("COUNTERSET", {"a", 11, 12}, &eval.ctx), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 0); EvaluateFunction("COUNTERSET", {"c1", 12}, &eval.ctx); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 12); EvaluateFunction("COUNTERSET", {"c2", 42}, &eval.ctx); EXPECT_EQ(EvaluateFunction("COUNTER", {"c2"}, &eval.ctx).ValueInt(), 42); EXPECT_EQ(EvaluateFunction("COUNTER", {"c1"}, &eval.ctx).ValueInt(), 13); EXPECT_EQ(EvaluateFunction("COUNTER", {"c2"}, &eval.ctx).ValueInt(), 43); } TEST(ExpressionEvaluator, FunctionIndexInfo) { NoContextExpressionEvaluator eval; EXPECT_THROW(EvaluateFunction("INDEXINFO", {1}, &eval.ctx), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("INDEXINFO", {}, &eval.ctx).ValueList().size(), 0); auto &dba = *eval.dba; dba.InsertVertex().add_label(dba.Label("l1")); { auto info = ToList(EvaluateFunction("INDEXINFO", {}, &eval.ctx)); EXPECT_EQ(info.size(), 1); EXPECT_EQ(info[0], ":l1"); } { dba.BuildIndex(dba.Label("l1"), dba.Property("prop")); auto info = ToList(EvaluateFunction("INDEXINFO", {}, &eval.ctx)); EXPECT_EQ(info.size(), 2); EXPECT_THAT(info, testing::UnorderedElementsAre(":l1", ":l1(prop)")); } } TEST(ExpressionEvaluator, FunctionId) { NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto va = dba.InsertVertex(); auto ea = dba.InsertEdge(va, va, dba.EdgeType("edge")); auto vb = dba.InsertVertex(); EXPECT_EQ(EvaluateFunction("ID", {va}, &eval.ctx).Value(), 0); EXPECT_EQ(EvaluateFunction("ID", {ea}, &eval.ctx).Value(), 0); EXPECT_EQ(EvaluateFunction("ID", {vb}, &eval.ctx).Value(), 1024); EXPECT_THROW(EvaluateFunction("ID", {}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("ID", {0}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("ID", {va, ea}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionWorkerIdException) { database::SingleNode db; NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto va = dba.InsertVertex(); EXPECT_THROW(EvaluateFunction("WORKERID", {}, &eval.ctx), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("WORKERID", {va, va}, &eval.ctx), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionWorkerIdSingleNode) { NoContextExpressionEvaluator eval; auto &dba = *eval.dba; auto va = dba.InsertVertex(); EXPECT_EQ(EvaluateFunction("WORKERID", {va}, &eval.ctx).Value(), 0); } TEST(ExpressionEvaluator, FunctionToStringNull) { EXPECT_TRUE(EvaluateFunction("TOSTRING", {TypedValue::Null}).IsNull()); } TEST(ExpressionEvaluator, FunctionToStringString) { EXPECT_EQ(EvaluateFunction("TOSTRING", {""}).ValueString(), ""); EXPECT_EQ(EvaluateFunction("TOSTRING", {"this is a string"}).ValueString(), "this is a string"); } TEST(ExpressionEvaluator, FunctionToStringInteger) { EXPECT_EQ(EvaluateFunction("TOSTRING", {-23321312}).ValueString(), "-23321312"); EXPECT_EQ(EvaluateFunction("TOSTRING", {0}).ValueString(), "0"); EXPECT_EQ(EvaluateFunction("TOSTRING", {42}).ValueString(), "42"); } TEST(ExpressionEvaluator, FunctionToStringDouble) { EXPECT_EQ(EvaluateFunction("TOSTRING", {-42.42}).ValueString(), "-42.420000"); EXPECT_EQ(EvaluateFunction("TOSTRING", {0.0}).ValueString(), "0.000000"); EXPECT_EQ(EvaluateFunction("TOSTRING", {238910.2313217}).ValueString(), "238910.231322"); } TEST(ExpressionEvaluator, FunctionToStringBool) { EXPECT_EQ(EvaluateFunction("TOSTRING", {true}).ValueString(), "true"); EXPECT_EQ(EvaluateFunction("TOSTRING", {false}).ValueString(), "false"); } TEST(ExpressionEvaluator, FunctionToStringExceptions) { EXPECT_THROW(EvaluateFunction("TOSTRING", {1, 2, 3}), QueryRuntimeException); std::vector l{1, 2, 3}; EXPECT_THROW(EvaluateFunction("TOSTRING", l), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionTimestamp) { NoContextExpressionEvaluator eval; eval.ctx.timestamp_ = 42; EXPECT_EQ(EvaluateFunction("TIMESTAMP", {}, &eval.ctx).ValueInt(), 42); } TEST(ExpressionEvaluator, FunctionTimestampExceptions) { NoContextExpressionEvaluator eval; eval.ctx.timestamp_ = 42; EXPECT_THROW(EvaluateFunction("TIMESTAMP", {1}, &eval.ctx).ValueInt(), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionLeft) { EXPECT_THROW(EvaluateFunction("LEFT", {}), QueryRuntimeException); EXPECT_TRUE( EvaluateFunction("LEFT", {TypedValue::Null, TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("LEFT", {TypedValue::Null, 10}).IsNull()); EXPECT_THROW(EvaluateFunction("LEFT", {TypedValue::Null, -10}), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("LEFT", {"memgraph", 0}).ValueString(), ""); EXPECT_EQ(EvaluateFunction("LEFT", {"memgraph", 3}).ValueString(), "mem"); EXPECT_EQ(EvaluateFunction("LEFT", {"memgraph", 1000}).ValueString(), "memgraph"); EXPECT_THROW(EvaluateFunction("LEFT", {"memgraph", -10}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("LEFT", {"memgraph", "graph"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("LEFT", {132, 10}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionRight) { EXPECT_THROW(EvaluateFunction("RIGHT", {}), QueryRuntimeException); EXPECT_TRUE( EvaluateFunction("RIGHT", {TypedValue::Null, TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("RIGHT", {TypedValue::Null, 10}).IsNull()); EXPECT_THROW(EvaluateFunction("RIGHT", {TypedValue::Null, -10}), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("RIGHT", {"memgraph", 0}).ValueString(), ""); EXPECT_EQ(EvaluateFunction("RIGHT", {"memgraph", 3}).ValueString(), "aph"); EXPECT_EQ(EvaluateFunction("RIGHT", {"memgraph", 1000}).ValueString(), "memgraph"); EXPECT_THROW(EvaluateFunction("RIGHT", {"memgraph", -10}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RIGHT", {"memgraph", "graph"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RIGHT", {132, 10}), QueryRuntimeException); } TEST(ExpressionEvaluator, Trimming) { EXPECT_TRUE(EvaluateFunction("LTRIM", {TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("RTRIM", {TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("TRIM", {TypedValue::Null}).IsNull()); EXPECT_EQ(EvaluateFunction("LTRIM", {" abc "}).ValueString(), "abc "); EXPECT_EQ(EvaluateFunction("RTRIM", {" abc "}).ValueString(), " abc"); EXPECT_EQ(EvaluateFunction("TRIM", {"abc"}).ValueString(), "abc"); EXPECT_THROW(EvaluateFunction("LTRIM", {"x", "y"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("RTRIM", {"x", "y"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("TRIM", {"x", "y"}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionReverse) { EXPECT_TRUE(EvaluateFunction("REVERSE", {TypedValue::Null}).IsNull()); EXPECT_EQ(EvaluateFunction("REVERSE", {"abc"}).ValueString(), "cba"); EXPECT_THROW(EvaluateFunction("REVERSE", {"x", "y"}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionReplace) { EXPECT_THROW(EvaluateFunction("REPLACE", {}), QueryRuntimeException); EXPECT_TRUE( EvaluateFunction("REPLACE", {TypedValue::Null, "l", "w"}).IsNull()); EXPECT_TRUE( EvaluateFunction("REPLACE", {"hello", TypedValue::Null, "w"}).IsNull()); EXPECT_TRUE( EvaluateFunction("REPLACE", {"hello", "l", TypedValue::Null}).IsNull()); EXPECT_EQ(EvaluateFunction("REPLACE", {"hello", "l", "w"}).ValueString(), "hewwo"); EXPECT_THROW(EvaluateFunction("REPLACE", {1, "l", "w"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("REPLACE", {"hello", 1, "w"}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("REPLACE", {"hello", "l", 1}), QueryRuntimeException); } TEST(ExpressionEvaluator, FunctionSplit) { EXPECT_THROW(EvaluateFunction("SPLIT", {}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("SPLIT", {"one,two", 1}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("SPLIT", {1, "one,two"}), QueryRuntimeException); EXPECT_TRUE( EvaluateFunction("SPLIT", {TypedValue::Null, TypedValue::Null}).IsNull()); EXPECT_TRUE( EvaluateFunction("SPLIT", {"one,two", TypedValue::Null}).IsNull()); EXPECT_TRUE(EvaluateFunction("SPLIT", {TypedValue::Null, ","}).IsNull()); auto result = EvaluateFunction("SPLIT", {"one,two", ","}); EXPECT_TRUE(result.IsList()); EXPECT_EQ(result.ValueList()[0].ValueString(), "one"); EXPECT_EQ(result.ValueList()[1].ValueString(), "two"); } TEST(ExpressionEvaluator, FunctionSubstring) { EXPECT_THROW(EvaluateFunction("SUBSTRING", {}), QueryRuntimeException); EXPECT_TRUE( EvaluateFunction("SUBSTRING", {TypedValue::Null, 0, 10}).IsNull()); EXPECT_THROW( EvaluateFunction("SUBSTRING", {TypedValue::Null, TypedValue::Null}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("SUBSTRING", {TypedValue::Null, -10}), QueryRuntimeException); EXPECT_THROW( EvaluateFunction("SUBSTRING", {TypedValue::Null, 0, TypedValue::Null}), QueryRuntimeException); EXPECT_THROW(EvaluateFunction("SUBSTRING", {TypedValue::Null, 0, -10}), QueryRuntimeException); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 2}).ValueString(), "llo"); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 10}).ValueString(), ""); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 2, 0}).ValueString(), ""); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 1, 3}).ValueString(), "ell"); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 1, 4}).ValueString(), "ello"); EXPECT_EQ(EvaluateFunction("SUBSTRING", {"hello", 1, 10}).ValueString(), "ello"); } TEST(ExpressionEvaluator, FunctionToLower) { EXPECT_THROW(EvaluateFunction("TOLOWER", {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction("TOLOWER", {TypedValue::Null}).IsNull()); EXPECT_EQ(EvaluateFunction("TOLOWER", {"Ab__C"}).ValueString(), "ab__c"); } TEST(ExpressionEvaluator, FunctionToUpper) { EXPECT_THROW(EvaluateFunction("TOUPPER", {}), QueryRuntimeException); EXPECT_TRUE(EvaluateFunction("TOUPPER", {TypedValue::Null}).IsNull()); EXPECT_EQ(EvaluateFunction("TOUPPER", {"Ab__C"}).ValueString(), "AB__C"); } } // namespace