// Copyright 2022 Memgraph Ltd. // // Use of this software is governed by the Business Source License // included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source // License, and you may not use this file except in compliance with the Business Source License. // // As of the Change Date specified in that file, in accordance with // the Business Source License, use of this software will be governed // by the Apache License, Version 2.0, included in the file // licenses/APL.txt. #include #include #include #include #include #include "common/types.hpp" #include "exceptions.hpp" #include "storage/v3/bindings/ast/ast.hpp" #include "storage/v3/bindings/bindings.hpp" #include "storage/v3/bindings/db_accessor.hpp" #include "storage/v3/bindings/eval.hpp" #include "storage/v3/bindings/frame.hpp" #include "storage/v3/bindings/symbol_table.hpp" #include "storage/v3/bindings/typed_value.hpp" #include "storage/v3/id_types.hpp" #include "storage/v3/key_store.hpp" #include "storage/v3/property_value.hpp" #include "storage/v3/schemas.hpp" #include "storage/v3/shard.hpp" #include "utils/string.hpp" using testing::ElementsAre; using testing::UnorderedElementsAre; #define NODE(...) test_common::GetNode(storage, __VA_ARGS__) #define EDGE(...) test_common::GetEdge(storage, __VA_ARGS__) #define EDGE_VARIABLE(...) test_common::GetEdgeVariable(storage, __VA_ARGS__) #define PATTERN(...) test_common::GetPattern(storage, {__VA_ARGS__}) #define NAMED_PATTERN(name, ...) test_common::GetPattern(storage, name, {__VA_ARGS__}) #define OPTIONAL_MATCH(...) test_common::GetWithPatterns(storage.Create(true), {__VA_ARGS__}) #define MATCH(...) test_common::GetWithPatterns(storage.Create(), {__VA_ARGS__}) #define WHERE(expr) storage.Create((expr)) #define CREATE(...) test_common::GetWithPatterns(storage.Create(), {__VA_ARGS__}) #define IDENT(...) storage.Create(__VA_ARGS__) #define LITERAL(val) storage.Create((val)) #define LIST(...) storage.Create(std::vector{__VA_ARGS__}) #define MAP(...) storage.Create(std::unordered_map{__VA_ARGS__}) #define PROPERTY_PAIR(property_name) std::make_pair(property_name, dba.NameToProperty(property_name)) #define PROPERTY_LOOKUP(...) test_common::GetPropertyLookup(storage, dba, __VA_ARGS__) #define PARAMETER_LOOKUP(token_position) storage.Create((token_position)) #define NEXPR(name, expr) storage.Create((name), (expr)) // AS is alternative to NEXPR which does not initialize NamedExpression with // Expression. It should be used with RETURN or WITH. For example: // RETURN(IDENT("n"), AS("n")) vs. RETURN(NEXPR("n", IDENT("n"))). #define AS(name) storage.Create((name)) #define RETURN(...) test_common::GetReturn(storage, false, __VA_ARGS__) #define WITH(...) test_common::GetWith(storage, false, __VA_ARGS__) #define RETURN_DISTINCT(...) test_common::GetReturn(storage, true, __VA_ARGS__) #define WITH_DISTINCT(...) test_common::GetWith(storage, true, __VA_ARGS__) #define UNWIND(...) test_common::GetUnwind(storage, __VA_ARGS__) #define ORDER_BY(...) test_common::GetOrderBy(__VA_ARGS__) #define SKIP(expr) \ test_common::Skip { (expr) } #define LIMIT(expr) \ test_common::Limit { (expr) } #define DELETE(...) test_common::GetDelete(storage, {__VA_ARGS__}) #define DETACH_DELETE(...) test_common::GetDelete(storage, {__VA_ARGS__}, true) #define SET(...) test_common::GetSet(storage, __VA_ARGS__) #define REMOVE(...) test_common::GetRemove(storage, __VA_ARGS__) #define MERGE(...) test_common::GetMerge(storage, __VA_ARGS__) #define ON_MATCH(...) \ test_common::OnMatch { \ std::vector { __VA_ARGS__ } \ } #define ON_CREATE(...) \ test_common::OnCreate { \ std::vector { __VA_ARGS__ } \ } #define CREATE_INDEX_ON(label, property) \ storage.Create(IndexQuery::Action::CREATE, (label), std::vector{(property)}) #define QUERY(...) test_common::GetQuery(storage, __VA_ARGS__) #define SINGLE_QUERY(...) test_common::GetSingleQuery(storage.Create(), __VA_ARGS__) #define UNION(...) test_common::GetCypherUnion(storage.Create(true), __VA_ARGS__) #define UNION_ALL(...) test_common::GetCypherUnion(storage.Create(false), __VA_ARGS__) #define FOREACH(...) test_common::GetForeach(storage, __VA_ARGS__) // Various operators #define NOT(expr) storage.Create((expr)) #define UPLUS(expr) storage.Create((expr)) #define UMINUS(expr) storage.Create((expr)) #define IS_NULL(expr) storage.Create((expr)) #define ADD(expr1, expr2) storage.Create((expr1), (expr2)) #define LESS(expr1, expr2) storage.Create((expr1), (expr2)) #define LESS_EQ(expr1, expr2) storage.Create((expr1), (expr2)) #define GREATER(expr1, expr2) storage.Create((expr1), (expr2)) #define GREATER_EQ(expr1, expr2) storage.Create((expr1), (expr2)) #define SUM(expr) storage.Create((expr), nullptr, Aggregation::Op::SUM) #define COUNT(expr) storage.Create((expr), nullptr, Aggregation::Op::COUNT) #define AVG(expr) storage.Create((expr), nullptr, Aggregation::Op::AVG) #define COLLECT_LIST(expr) storage.Create((expr), nullptr, Aggregation::Op::COLLECT_LIST) #define EQ(expr1, expr2) storage.Create((expr1), (expr2)) #define NEQ(expr1, expr2) storage.Create((expr1), (expr2)) #define AND(expr1, expr2) storage.Create((expr1), (expr2)) #define OR(expr1, expr2) storage.Create((expr1), (expr2)) #define IN_LIST(expr1, expr2) storage.Create((expr1), (expr2)) #define IF(cond, then, else) storage.Create((cond), (then), (else)) // Function call #define FN(function_name, ...) \ storage.Create(memgraph::utils::ToUpperCase(function_name), std::vector{__VA_ARGS__}) // List slicing #define SLICE(list, lower_bound, upper_bound) storage.Create(list, lower_bound, upper_bound) // all(variable IN list WHERE predicate) #define ALL(variable, list, where) storage.Create(storage.Create(variable), list, where) #define SINGLE(variable, list, where) storage.Create(storage.Create(variable), list, where) #define ANY(variable, list, where) storage.Create(storage.Create(variable), list, where) #define NONE(variable, list, where) storage.Create(storage.Create(variable), list, where) #define REDUCE(accumulator, initializer, variable, list, expr) \ storage.Create(storage.Create(accumulator), initializer, storage.Create(variable), \ list, expr) #define COALESCE(...) storage.Create(std::vector{__VA_ARGS__}) #define EXTRACT(variable, list, expr) storage.Create(storage.Create(variable), list, expr) #define AUTH_QUERY(action, user, role, user_or_role, password, privileges) \ storage.Create((action), (user), (role), (user_or_role), password, (privileges)) #define DROP_USER(usernames) storage.Create((usernames)) #define CALL_PROCEDURE(...) test_common::GetCallProcedure(storage, __VA_ARGS__) namespace memgraph::storage::v3::test { class ExpressionEvaluatorTest : public ::testing::Test { protected: LabelId primary_label{LabelId::FromInt(0)}; PropertyId primary_property{PropertyId::FromInt(1)}; PrimaryKey min_pk{PropertyValue(0)}; Shard db{primary_label, min_pk, std::nullopt}; Shard::Accessor storage_dba{db.Access()}; DbAccessor dba{&storage_dba}; AstStorage storage; memgraph::utils::MonotonicBufferResource mem{1024}; EvaluationContext ctx{.memory = &mem}; SymbolTable symbol_table; Frame frame{128}; ExpressionEvaluator eval{&frame, symbol_table, ctx, &dba, View::OLD}; void SetUp() override { db.StoreMapping({{1, "label"}, {2, "property"}}); ASSERT_TRUE( db.CreateSchema(primary_label, {storage::v3::SchemaProperty{primary_property, common::SchemaType::INT}})); } std::vector NamesToProperties(const std::vector &property_names) { std::vector properties; properties.reserve(property_names.size()); for (const auto &name : property_names) { properties.push_back(dba.NameToProperty(name)); } return properties; } std::vector NamesToLabels(const std::vector &label_names) { std::vector labels; labels.reserve(label_names.size()); for (const auto &name : label_names) { labels.push_back(dba.NameToLabel(name)); } return labels; } Identifier *CreateIdentifierWithValue(std::string name, const TypedValue &value) { auto *id = storage.Create(name, true); auto symbol = symbol_table.CreateSymbol(name, true); id->MapTo(symbol); frame[symbol] = value; return id; } template auto Eval(TExpression *expr) { ctx.properties = NamesToProperties(storage.properties_); ctx.labels = NamesToLabels(storage.labels_); auto value = expr->Accept(eval); EXPECT_EQ(value.GetMemoryResource(), &mem) << "ExpressionEvaluator must use the MemoryResource from " "EvaluationContext for allocations!"; return value; } }; TEST_F(ExpressionEvaluatorTest, OrOperator) { auto *op = storage.Create(storage.Create(true), storage.Create(false)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(true), storage.Create(true)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, XorOperator) { auto *op = storage.Create(storage.Create(true), storage.Create(false)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(true), storage.Create(true)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), false); } TEST_F(ExpressionEvaluatorTest, AndOperator) { auto *op = storage.Create(storage.Create(true), storage.Create(true)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(false), storage.Create(true)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), false); } TEST_F(ExpressionEvaluatorTest, AndOperatorShortCircuit) { { auto *op = storage.Create(storage.Create(false), storage.Create(5)); auto value = Eval(op); EXPECT_EQ(value.ValueBool(), 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(Eval(op), expr::ExpressionRuntimeException); } } TEST_F(ExpressionEvaluatorTest, AndOperatorNull) { storage.Create(memgraph::storage::v3::PropertyValue(1)); { // Null doesn't short circuit auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(5)); EXPECT_THROW(Eval(op), expr::ExpressionRuntimeException); } { auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(true)); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(false)); auto value = Eval(op); ASSERT_TRUE(value.IsBool()); EXPECT_EQ(value.ValueBool(), false); } } TEST_F(ExpressionEvaluatorTest, AdditionOperator) { auto *op = storage.Create(storage.Create(2), storage.Create(3)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 5); } TEST_F(ExpressionEvaluatorTest, SubtractionOperator) { auto *op = storage.Create(storage.Create(2), storage.Create(3)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), -1); } TEST_F(ExpressionEvaluatorTest, MultiplicationOperator) { auto *op = storage.Create(storage.Create(2), storage.Create(3)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 6); } TEST_F(ExpressionEvaluatorTest, DivisionOperator) { auto *op = storage.Create(storage.Create(50), storage.Create(10)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 5); } TEST_F(ExpressionEvaluatorTest, ModOperator) { auto *op = storage.Create(storage.Create(65), storage.Create(10)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 5); } TEST_F(ExpressionEvaluatorTest, EqualOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), false); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), true); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), false); } TEST_F(ExpressionEvaluatorTest, NotEqualOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, LessOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), false); } TEST_F(ExpressionEvaluatorTest, GreaterOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), false); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), false); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, LessEqualOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), true); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), true); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), false); } TEST_F(ExpressionEvaluatorTest, GreaterEqualOperator) { auto *op = storage.Create(storage.Create(10), storage.Create(15)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), false); op = storage.Create(storage.Create(15), storage.Create(15)); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), true); op = storage.Create(storage.Create(20), storage.Create(15)); auto val3 = Eval(op); ASSERT_EQ(val3.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, InListOperator) { 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 = Eval(op); EXPECT_EQ(value.ValueBool(), true); } { // Element doesn't exist in list. auto *op = storage.Create(storage.Create("x"), list_literal); auto value = Eval(op); EXPECT_EQ(value.ValueBool(), false); } { const auto p1 = PropertyToTypedValue(memgraph::storage::v3::PropertyValue()); storage.Create(memgraph::storage::v3::PropertyValue()); auto *list_literal = storage.Create( std::vector{storage.Create(memgraph::storage::v3::PropertyValue()), 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 = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Null list. auto *op = storage.Create(storage.Create("x"), storage.Create(memgraph::storage::v3::PropertyValue())); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Null literal. auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), list_literal); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Null literal, empty list. auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(std::vector())); auto value = Eval(op); EXPECT_FALSE(value.ValueBool()); } } TEST_F(ExpressionEvaluatorTest, ListIndexing) { 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 = Eval(op); EXPECT_EQ(value.ValueInt(), 3); } { // Out of bounds indexing. auto *op = storage.Create(list_literal, storage.Create(4)); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Out of bounds indexing with negative bound. auto *op = storage.Create(list_literal, storage.Create(-100)); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Legal indexing with negative index. auto *op = storage.Create(list_literal, storage.Create(-2)); auto value = Eval(op); EXPECT_EQ(value.ValueInt(), 3); } { // Indexing with one operator being null. auto *op = storage.Create( storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(-2)); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Indexing with incompatible type. auto *op = storage.Create(list_literal, storage.Create("bla")); EXPECT_THROW(Eval(op), expr::ExpressionRuntimeException); } } TEST_F(ExpressionEvaluatorTest, MapIndexing) { auto *map_literal = storage.Create( std::unordered_map{{storage.GetPropertyIx("a"), storage.Create(1)}, {storage.GetPropertyIx("b"), storage.Create(2)}, {storage.GetPropertyIx("c"), storage.Create(3)}}); { // Legal indexing. auto *op = storage.Create(map_literal, storage.Create("b")); auto value = Eval(op); EXPECT_EQ(value.ValueInt(), 2); } { // Legal indexing, non-existing key. auto *op = storage.Create(map_literal, storage.Create("z")); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Wrong key type. auto *op = storage.Create(map_literal, storage.Create(42)); EXPECT_THROW(Eval(op), expr::ExpressionRuntimeException); } { // Indexing with Null. auto *op = storage.Create( map_literal, storage.Create(memgraph::storage::v3::PropertyValue())); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } } TEST_F(ExpressionEvaluatorTest, VertexAndEdgeIndexing) { db.StoreMapping({{1, "label"}, {2, "property"}, {3, "edge_type"}, {4, "prop"}}); auto edge_type = dba.NameToEdgeType("edge_type"); auto prop = dba.NameToProperty("prop"); auto v1 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(0)}}); auto e11 = dba.InsertEdge(&v1, &v1, edge_type); ASSERT_TRUE(e11.HasValue()); ASSERT_TRUE(v1.SetPropertyAndValidate(prop, memgraph::storage::v3::PropertyValue(42)).HasValue()); ASSERT_TRUE(e11->SetProperty(prop, memgraph::storage::v3::PropertyValue(43)).HasValue()); dba.AdvanceCommand(); auto *vertex_id = CreateIdentifierWithValue("v1", TypedValue(v1)); auto *edge_id = CreateIdentifierWithValue("e11", TypedValue(*e11)); { // Legal indexing. auto *op1 = storage.Create(vertex_id, storage.Create("prop")); auto value1 = Eval(op1); EXPECT_EQ(value1.ValueInt(), 42); auto *op2 = storage.Create(edge_id, storage.Create("prop")); auto value2 = Eval(op2); EXPECT_EQ(value2.ValueInt(), 43); } { // Legal indexing, non-existing key. auto *op1 = storage.Create(vertex_id, storage.Create("blah")); auto value1 = Eval(op1); EXPECT_TRUE(value1.IsNull()); auto *op2 = storage.Create(edge_id, storage.Create("blah")); auto value2 = Eval(op2); EXPECT_TRUE(value2.IsNull()); } { // Wrong key type. auto *op1 = storage.Create(vertex_id, storage.Create(1)); EXPECT_THROW(Eval(op1), expr::ExpressionRuntimeException); auto *op2 = storage.Create(edge_id, storage.Create(1)); EXPECT_THROW(Eval(op2), expr::ExpressionRuntimeException); } { // Indexing with Null. auto *op1 = storage.Create( vertex_id, storage.Create(memgraph::storage::v3::PropertyValue())); auto value1 = Eval(op1); EXPECT_TRUE(value1.IsNull()); auto *op2 = storage.Create( edge_id, storage.Create(memgraph::storage::v3::PropertyValue())); auto value2 = Eval(op2); EXPECT_TRUE(value2.IsNull()); } } TEST_F(ExpressionEvaluatorTest, ListSlicingOperator) { 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.ValueList()) { int_list.push_back(x.ValueInt()); } return int_list; }; { // Legal slicing with both bounds defined. auto *op = storage.Create(list_literal, storage.Create(2), storage.Create(4)); auto value = Eval(op); 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 = Eval(op); 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 = Eval(op); EXPECT_THAT(extract_ints(value), ElementsAre()); } { // Bounds ouf or range. auto *op = storage.Create(list_literal, storage.Create(-100), storage.Create(10)); auto value = Eval(op); 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 = Eval(op); EXPECT_THAT(extract_ints(value), ElementsAre(1, 2, 3)); } { // Upper bound undefined. auto *op = storage.Create(list_literal, storage.Create(-2), nullptr); auto value = Eval(op); EXPECT_THAT(extract_ints(value), ElementsAre(3, 4)); } { // Bound of illegal type and null value bound. auto *op = storage.Create( list_literal, storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create("mirko")); EXPECT_THROW(Eval(op), expr::ExpressionRuntimeException); } { // List of illegal type. auto *op = storage.Create(storage.Create("a"), storage.Create(-2), nullptr); EXPECT_THROW(Eval(op), expr::ExpressionRuntimeException); } { // Null value list with undefined upper bound. auto *op = storage.Create(storage.Create(memgraph::storage::v3::PropertyValue()), storage.Create(-2), nullptr); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } { // Null value index. auto *op = storage.Create(list_literal, storage.Create(-2), storage.Create(memgraph::storage::v3::PropertyValue())); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } } TEST_F(ExpressionEvaluatorTest, IfOperator) { 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 = Eval(op); ASSERT_EQ(value.ValueInt(), 10); } { auto *condition_false = storage.Create(storage.Create(2), storage.Create(3)); auto *op = storage.Create(condition_false, then_expression, else_expression); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 20); } { auto *condition_exception = storage.Create(storage.Create(2), storage.Create(3)); auto *op = storage.Create(condition_exception, then_expression, else_expression); ASSERT_THROW(Eval(op), expr::ExpressionRuntimeException); } } TEST_F(ExpressionEvaluatorTest, NotOperator) { auto *op = storage.Create(storage.Create(false)); auto value = Eval(op); ASSERT_EQ(value.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, UnaryPlusOperator) { auto *op = storage.Create(storage.Create(5)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), 5); } TEST_F(ExpressionEvaluatorTest, UnaryMinusOperator) { auto *op = storage.Create(storage.Create(5)); auto value = Eval(op); ASSERT_EQ(value.ValueInt(), -5); } TEST_F(ExpressionEvaluatorTest, IsNullOperator) { auto *op = storage.Create(storage.Create(1)); auto val1 = Eval(op); ASSERT_EQ(val1.ValueBool(), false); op = storage.Create(storage.Create(PropertyValue())); auto val2 = Eval(op); ASSERT_EQ(val2.ValueBool(), true); } TEST_F(ExpressionEvaluatorTest, LabelsTest) { db.StoreMapping({{1, "label"}, {2, "property"}, {3, "ANIMAL"}, {4, "DOG"}, {5, "NICE_DOG"}, {6, "BAD_DOG"}}); auto v1 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(1)}}); ASSERT_TRUE(v1.AddLabelAndValidate(dba.NameToLabel("ANIMAL")).HasValue()); ASSERT_TRUE(v1.AddLabelAndValidate(dba.NameToLabel("DOG")).HasValue()); ASSERT_TRUE(v1.AddLabelAndValidate(dba.NameToLabel("NICE_DOG")).HasValue()); dba.AdvanceCommand(); auto *identifier = storage.Create("n"); auto node_symbol = symbol_table.CreateSymbol("n", true); identifier->MapTo(node_symbol); frame[node_symbol] = TypedValue(v1); { auto *op = storage.Create( identifier, std::vector{storage.GetLabelIx("DOG"), storage.GetLabelIx("ANIMAL")}); auto value = Eval(op); EXPECT_EQ(value.ValueBool(), true); } { auto *op = storage.Create( identifier, std::vector{storage.GetLabelIx("DOG"), storage.GetLabelIx("BAD_DOG"), storage.GetLabelIx("ANIMAL")}); auto value = Eval(op); EXPECT_EQ(value.ValueBool(), false); } { frame[node_symbol] = TypedValue(); auto *op = storage.Create( identifier, std::vector{storage.GetLabelIx("DOG"), storage.GetLabelIx("BAD_DOG"), storage.GetLabelIx("ANIMAL")}); auto value = Eval(op); EXPECT_TRUE(value.IsNull()); } } TEST_F(ExpressionEvaluatorTest, Aggregation) { auto aggr = storage.Create(storage.Create(42), nullptr, Aggregation::Op::COUNT); auto aggr_sym = symbol_table.CreateSymbol("aggr", true); aggr->MapTo(aggr_sym); frame[aggr_sym] = TypedValue(1); auto value = Eval(aggr); EXPECT_EQ(value.ValueInt(), 1); } TEST_F(ExpressionEvaluatorTest, ListLiteral) { auto *list_literal = storage.Create(std::vector{storage.Create(1), storage.Create("bla"), storage.Create(true)}); TypedValue result = Eval(list_literal); ASSERT_TRUE(result.IsList()); auto &result_elems = result.ValueList(); ASSERT_EQ(3, result_elems.size()); EXPECT_TRUE(result_elems[0].IsInt()); EXPECT_TRUE(result_elems[1].IsString()); EXPECT_TRUE(result_elems[2].IsBool()); } TEST_F(ExpressionEvaluatorTest, ParameterLookup) { ctx.parameters.Add(0, PropertyValue(42)); auto *param_lookup = storage.Create(0); auto value = Eval(param_lookup); ASSERT_TRUE(value.IsInt()); EXPECT_EQ(value.ValueInt(), 42); } TEST_F(ExpressionEvaluatorTest, FunctionAll1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *all = ALL("x", LIST(LITERAL(1), LITERAL(1)), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(all); ASSERT_TRUE(value.IsBool()); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAll2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *all = ALL("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(all); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAllNullList) { AstStorage storage; auto *all = ALL("x", LITERAL(PropertyValue()), WHERE(LITERAL(true))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); auto value = Eval(all); EXPECT_TRUE(value.IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionAllNullElementInList1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *all = ALL("x", LIST(LITERAL(1), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(all); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAllNullElementInList2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *all = ALL("x", LIST(LITERAL(2), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(all); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAllWhereWrongType) { AstStorage storage; auto *all = ALL("x", LIST(LITERAL(1)), WHERE(LITERAL(2))); const auto x_sym = symbol_table.CreateSymbol("x", true); all->identifier_->MapTo(x_sym); EXPECT_THROW(Eval(all), expr::ExpressionRuntimeException); } TEST_F(ExpressionEvaluatorTest, FunctionSingle1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); single->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(single); ASSERT_TRUE(value.IsBool()); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionSingle2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(GREATER(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); single->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(single); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionSingleNullList) { AstStorage storage; auto *single = SINGLE("x", LITERAL(PropertyValue()), WHERE(LITERAL(true))); const auto x_sym = symbol_table.CreateSymbol("x", true); single->identifier_->MapTo(x_sym); auto value = Eval(single); EXPECT_TRUE(value.IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionSingleNullElementInList1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(1), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); single->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(single); ASSERT_TRUE(value.IsBool()); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionSingleNullElementInList2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *single = SINGLE("x", LIST(LITERAL(2), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); single->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(single); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAny1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *any = ANY("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(any); ASSERT_TRUE(value.IsBool()); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAny2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *any = ANY("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(any); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAnyNullList) { AstStorage storage; auto *any = ANY("x", LITERAL(PropertyValue()), WHERE(LITERAL(true))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); auto value = Eval(any); EXPECT_TRUE(value.IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionAnyNullElementInList1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *any = ANY("x", LIST(LITERAL(0), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(any); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAnyNullElementInList2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *any = ANY("x", LIST(LITERAL(1), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(any); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionAnyWhereWrongType) { AstStorage storage; auto *any = ANY("x", LIST(LITERAL(1)), WHERE(LITERAL(2))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); EXPECT_THROW(Eval(any), expr::ExpressionRuntimeException); } TEST_F(ExpressionEvaluatorTest, FunctionNone1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *none = NONE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); none->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(none); ASSERT_TRUE(value.IsBool()); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionNone2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *none = NONE("x", LIST(LITERAL(1), LITERAL(2)), WHERE(EQ(ident_x, LITERAL(1)))); const auto x_sym = symbol_table.CreateSymbol("x", true); none->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(none); ASSERT_TRUE(value.IsBool()); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionNoneNullList) { AstStorage storage; auto *none = NONE("x", LITERAL(PropertyValue()), WHERE(LITERAL(true))); const auto x_sym = symbol_table.CreateSymbol("x", true); none->identifier_->MapTo(x_sym); auto value = Eval(none); EXPECT_TRUE(value.IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionNoneNullElementInList1) { AstStorage storage; auto *ident_x = IDENT("x"); auto *any = NONE("x", LIST(LITERAL(1), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); any->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(any); EXPECT_TRUE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionNoneNullElementInList2) { AstStorage storage; auto *ident_x = IDENT("x"); auto *none = NONE("x", LIST(LITERAL(0), LITERAL(PropertyValue())), WHERE(EQ(ident_x, LITERAL(0)))); const auto x_sym = symbol_table.CreateSymbol("x", true); none->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(none); EXPECT_FALSE(value.ValueBool()); } TEST_F(ExpressionEvaluatorTest, FunctionNoneWhereWrongType) { AstStorage storage; auto *none = NONE("x", LIST(LITERAL(1)), WHERE(LITERAL(2))); const auto x_sym = symbol_table.CreateSymbol("x", true); none->identifier_->MapTo(x_sym); EXPECT_THROW(Eval(none), expr::ExpressionRuntimeException); } TEST_F(ExpressionEvaluatorTest, 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)); const auto sum_sym = symbol_table.CreateSymbol("sum", true); reduce->accumulator_->MapTo(sum_sym); ident_sum->MapTo(sum_sym); const auto x_sym = symbol_table.CreateSymbol("x", true); reduce->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(reduce); ASSERT_TRUE(value.IsInt()); EXPECT_EQ(value.ValueInt(), 3); } TEST_F(ExpressionEvaluatorTest, FunctionExtract) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LIST(LITERAL(1), LITERAL(2), LITERAL(PropertyValue())), ADD(ident_x, LITERAL(1))); const auto x_sym = symbol_table.CreateSymbol("x", true); extract->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(extract); EXPECT_TRUE(value.IsList()); ; auto result = value.ValueList(); EXPECT_EQ(result[0].ValueInt(), 2); EXPECT_EQ(result[1].ValueInt(), 3); EXPECT_TRUE(result[2].IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionExtractNull) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LITERAL(PropertyValue()), ADD(ident_x, LITERAL(1))); const auto x_sym = symbol_table.CreateSymbol("x", true); extract->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); auto value = Eval(extract); EXPECT_TRUE(value.IsNull()); } TEST_F(ExpressionEvaluatorTest, FunctionExtractExceptions) { AstStorage storage; auto *ident_x = IDENT("x"); auto *extract = EXTRACT("x", LITERAL("bla"), ADD(ident_x, LITERAL(1))); const auto x_sym = symbol_table.CreateSymbol("x", true); extract->identifier_->MapTo(x_sym); ident_x->MapTo(x_sym); EXPECT_THROW(Eval(extract), expr::ExpressionRuntimeException); } // TEST_F(ExpressionEvaluatorTest, Coalesce) { // // coalesce() // EXPECT_THROW(Eval(COALESCE()), expr::ExpressionRuntimeException); // // coalesce(null, null) // EXPECT_TRUE(Eval(COALESCE(LITERAL(TypedValue()), LITERAL(TypedValue()))).IsNull()); // // coalesce(null, 2, 3) // EXPECT_EQ(Eval(COALESCE(LITERAL(TypedValue()), LITERAL(2), LITERAL(3))).ValueInt(), 2); // // coalesce(null, 2, assert(false), 3) // EXPECT_EQ(Eval(COALESCE(LITERAL(TypedValue()), LITERAL(2), FN("ASSERT", LITERAL(false)), LITERAL(3))).ValueInt(), // 2); // // (null, assert(false)) // EXPECT_THROW(Eval(COALESCE(LITERAL(TypedValue()), FN("ASSERT", LITERAL(false)))), // expr::ExpressionRuntimeException); // // coalesce([null, null]) // EXPECT_FALSE(Eval(COALESCE(LITERAL(TypedValue(std::vector{TypedValue(), TypedValue()})))).IsNull()); // } TEST_F(ExpressionEvaluatorTest, RegexMatchInvalidArguments) { EXPECT_TRUE(Eval(storage.Create(LITERAL(TypedValue()), LITERAL("regex"))).IsNull()); EXPECT_TRUE(Eval(storage.Create(LITERAL(3), LITERAL("regex"))).IsNull()); EXPECT_TRUE(Eval(storage.Create(LIST(LITERAL("string")), LITERAL("regex"))).IsNull()); EXPECT_TRUE(Eval(storage.Create(LITERAL("string"), LITERAL(TypedValue()))).IsNull()); EXPECT_THROW(Eval(storage.Create(LITERAL("string"), LITERAL(42))), expr::ExpressionRuntimeException); EXPECT_THROW(Eval(storage.Create(LITERAL("string"), LIST(LITERAL("regex")))), expr::ExpressionRuntimeException); } TEST_F(ExpressionEvaluatorTest, RegexMatchInvalidRegex) { EXPECT_THROW(Eval(storage.Create(LITERAL("text"), LITERAL("*ext"))), expr::ExpressionRuntimeException); EXPECT_THROW(Eval(storage.Create(LITERAL("text"), LITERAL("[ext"))), expr::ExpressionRuntimeException); } TEST_F(ExpressionEvaluatorTest, RegexMatch) { EXPECT_FALSE(Eval(storage.Create(LITERAL("text"), LITERAL(".*ex"))).ValueBool()); EXPECT_TRUE(Eval(storage.Create(LITERAL("text"), LITERAL(".*ext"))).ValueBool()); EXPECT_FALSE(Eval(storage.Create(LITERAL("text"), LITERAL("[ext]"))).ValueBool()); EXPECT_TRUE(Eval(storage.Create(LITERAL("text"), LITERAL(".+[ext]"))).ValueBool()); } class ExpressionEvaluatorPropertyLookup : public ExpressionEvaluatorTest { protected: std::pair prop_age = std::make_pair("age", PropertyId::FromUint(3)); std::pair prop_height = std::make_pair("height", PropertyId::FromUint(4)); Identifier *identifier = storage.Create("element", true); Symbol symbol = symbol_table.CreateSymbol("element", true); void SetUp() override { identifier->MapTo(symbol); db.StoreMapping({{1, "label"}, {2, "property"}, {3, "age"}, {4, "height"}}); ASSERT_TRUE( db.CreateSchema(primary_label, {storage::v3::SchemaProperty{primary_property, common::SchemaType::INT}})); } auto Value(std::pair property) { auto *op = storage.Create(identifier, storage.GetPropertyIx(property.first)); return Eval(op); } }; TEST_F(ExpressionEvaluatorPropertyLookup, Vertex) { auto v1 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(3)}}); ASSERT_TRUE(v1.SetPropertyAndValidate(prop_age.second, PropertyValue(10)).HasValue()); dba.AdvanceCommand(); frame[symbol] = TypedValue(v1); EXPECT_EQ(Value(prop_age).ValueInt(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, Duration) { db.StoreMapping({{1, "label"}, {2, "property"}, {3, "day"}, {4, "hour"}, {5, "minute"}, {6, "second"}, {7, "millisecond"}, {8, "microsecond"}, {9, "nanosecond"}}); const memgraph::utils::Duration dur({10, 1, 30, 2, 22, 45}); frame[symbol] = TypedValue(dur); const std::pair day = std::make_pair("day", dba.NameToProperty("day")); const auto total_days = Value(day); EXPECT_TRUE(total_days.IsInt()); EXPECT_EQ(total_days.ValueInt(), 10); const std::pair hour = std::make_pair("hour", dba.NameToProperty("hour")); const auto total_hours = Value(hour); EXPECT_TRUE(total_hours.IsInt()); EXPECT_EQ(total_hours.ValueInt(), 1); const std::pair minute = std::make_pair("minute", dba.NameToProperty("minute")); const auto total_mins = Value(minute); EXPECT_TRUE(total_mins.IsInt()); EXPECT_EQ(total_mins.ValueInt(), 1 * 60 + 30); const std::pair sec = std::make_pair("second", dba.NameToProperty("second")); const auto total_secs = Value(sec); EXPECT_TRUE(total_secs.IsInt()); const auto expected_secs = total_mins.ValueInt() * 60 + 2; EXPECT_EQ(total_secs.ValueInt(), expected_secs); const std::pair milli = std::make_pair("millisecond", dba.NameToProperty("millisecond")); const auto total_milli = Value(milli); EXPECT_TRUE(total_milli.IsInt()); const auto expected_milli = total_secs.ValueInt() * 1000 + 22; EXPECT_EQ(total_milli.ValueInt(), expected_milli); const std::pair micro = std::make_pair("microsecond", dba.NameToProperty("microsecond")); const auto total_micros = Value(micro); EXPECT_TRUE(total_micros.IsInt()); const auto expected_micros = expected_milli * 1000 + 45; EXPECT_EQ(total_micros.ValueInt(), expected_micros); const std::pair nano = std::make_pair("nanosecond", dba.NameToProperty("nanosecond")); const auto total_nano = Value(nano); EXPECT_TRUE(total_nano.IsInt()); const auto expected_nano = expected_micros * 1000; EXPECT_EQ(total_nano.ValueInt(), expected_nano); } TEST_F(ExpressionEvaluatorPropertyLookup, Date) { db.StoreMapping({{1, "label"}, {2, "property"}, {3, "year"}, {4, "month"}, {5, "day"}}); const memgraph::utils::Date date({1996, 11, 22}); frame[symbol] = TypedValue(date); const std::pair year = std::make_pair("year", dba.NameToProperty("year")); const auto y = Value(year); EXPECT_TRUE(y.IsInt()); EXPECT_EQ(y.ValueInt(), 1996); const std::pair month = std::make_pair("month", dba.NameToProperty("month")); const auto m = Value(month); EXPECT_TRUE(m.IsInt()); EXPECT_EQ(m.ValueInt(), 11); const std::pair day = std::make_pair("day", dba.NameToProperty("day")); const auto d = Value(day); EXPECT_TRUE(d.IsInt()); EXPECT_EQ(d.ValueInt(), 22); } TEST_F(ExpressionEvaluatorPropertyLookup, LocalTime) { const memgraph::utils::LocalTime lt({1, 2, 3, 11, 22}); frame[symbol] = TypedValue(lt); const std::pair hour = std::make_pair("hour", dba.NameToProperty("hour")); const auto h = Value(hour); EXPECT_TRUE(h.IsInt()); EXPECT_EQ(h.ValueInt(), 1); const std::pair minute = std::make_pair("minute", dba.NameToProperty("minute")); const auto min = Value(minute); EXPECT_TRUE(min.IsInt()); EXPECT_EQ(min.ValueInt(), 2); const std::pair second = std::make_pair("second", dba.NameToProperty("second")); const auto sec = Value(second); EXPECT_TRUE(sec.IsInt()); EXPECT_EQ(sec.ValueInt(), 3); const std::pair millis = std::make_pair("millisecond", dba.NameToProperty("millisecond")); const auto mil = Value(millis); EXPECT_TRUE(mil.IsInt()); EXPECT_EQ(mil.ValueInt(), 11); const std::pair micros = std::make_pair("microsecond", dba.NameToProperty("microsecond")); const auto mic = Value(micros); EXPECT_TRUE(mic.IsInt()); EXPECT_EQ(mic.ValueInt(), 22); } TEST_F(ExpressionEvaluatorPropertyLookup, LocalDateTime) { const memgraph::utils::LocalDateTime ldt({1993, 8, 6}, {2, 3, 4, 55, 40}); frame[symbol] = TypedValue(ldt); const std::pair year = std::make_pair("year", dba.NameToProperty("year")); const auto y = Value(year); EXPECT_TRUE(y.IsInt()); EXPECT_EQ(y.ValueInt(), 1993); const std::pair month = std::make_pair("month", dba.NameToProperty("month")); const auto m = Value(month); EXPECT_TRUE(m.IsInt()); EXPECT_EQ(m.ValueInt(), 8); const std::pair day = std::make_pair("day", dba.NameToProperty("day")); const auto d = Value(day); EXPECT_TRUE(d.IsInt()); EXPECT_EQ(d.ValueInt(), 6); const std::pair hour = std::make_pair("hour", dba.NameToProperty("hour")); const auto h = Value(hour); EXPECT_TRUE(h.IsInt()); EXPECT_EQ(h.ValueInt(), 2); const std::pair minute = std::make_pair("minute", dba.NameToProperty("minute")); const auto min = Value(minute); EXPECT_TRUE(min.IsInt()); EXPECT_EQ(min.ValueInt(), 3); const std::pair second = std::make_pair("second", dba.NameToProperty("second")); const auto sec = Value(second); EXPECT_TRUE(sec.IsInt()); EXPECT_EQ(sec.ValueInt(), 4); const std::pair millis = std::make_pair("millisecond", dba.NameToProperty("millisecond")); const auto mil = Value(millis); EXPECT_TRUE(mil.IsInt()); EXPECT_EQ(mil.ValueInt(), 55); const std::pair micros = std::make_pair("microsecond", dba.NameToProperty("microsecond")); const auto mic = Value(micros); EXPECT_TRUE(mic.IsInt()); EXPECT_EQ(mic.ValueInt(), 40); } TEST_F(ExpressionEvaluatorPropertyLookup, Edge) { db.StoreMapping({{1, "label"}, {2, "property"}, {3, "age"}, {4, "height"}, {5, "edge_type"}}); auto v1 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(1)}}); auto v2 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(2)}}); auto e12 = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("edge_type")); ASSERT_TRUE(e12.HasValue()); ASSERT_TRUE(e12->SetProperty(prop_age.second, PropertyValue(10)).HasValue()); dba.AdvanceCommand(); frame[symbol] = TypedValue(*e12); EXPECT_EQ(Value(prop_age).ValueInt(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, Null) { frame[symbol] = TypedValue(); EXPECT_TRUE(Value(prop_age).IsNull()); } TEST_F(ExpressionEvaluatorPropertyLookup, MapLiteral) { frame[symbol] = TypedValue(std::map{{prop_age.first, TypedValue(10)}}); EXPECT_EQ(Value(prop_age).ValueInt(), 10); EXPECT_TRUE(Value(prop_height).IsNull()); } // TODO(jbajic) Fix Function calls after enabling memgraph functions // class FunctionTest : public ExpressionEvaluatorTest { // protected: // void SetUp() override { // db.StoreMapping({{0, "label"}, {1, "property"}}); // ASSERT_TRUE( // db.CreateSchema(primary_label, {storage::v3::SchemaProperty{primary_property, common::SchemaType::INT}})); // } // std::vector ExpressionsFromTypedValues(const std::vector &tvs) { // std::vector expressions; // expressions.reserve(tvs.size()); // for (size_t i = 0; i < tvs.size(); ++i) { // auto *ident = storage.Create("arg_" + std::to_string(i), true); // auto sym = symbol_table.CreateSymbol("arg_" + std::to_string(i), true); // ident->MapTo(sym); // frame[sym] = tvs[i]; // expressions.push_back(ident); // } // return expressions; // } // TypedValue EvaluateFunctionWithExprs(const std::string &function_name, const std::vector // &expressions) { // auto *op = storage.Create(function_name, expressions); // return Eval(op); // } // template // TypedValue EvaluateFunction(const std::string &function_name, std::tuple args) { // std::vector tv_args; // tv_args.reserve(args.size()); // for (auto &arg : args) tv_args.emplace_back(std::move(arg)); // return EvaluateFunctionWithExprs(function_name, ExpressionsFromTypedValues(tv_args)); // } // template // TypedValue EvaluateFunction(const std::string &function_name, TArgs &&...args) { // return EvaluateFunctionWithExprs(function_name, // ExpressionsFromTypedValues(std::vector{TypedValue(args)...})); // } // }; // template // static TypedValue MakeTypedValueList(TArgs &&...args) { // return TypedValue(std::vector{TypedValue(args)...}); // } // TEST_F(FunctionTest, EndNode) { // db.StoreMapping({{0, "label"}, {1, "property"}, {2, "label1"}, {3, "label2"}}); // ASSERT_THROW(EvaluateFunction("ENDNODE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("ENDNODE", TypedValue()).IsNull()); // auto v1 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(0)}}); // ASSERT_TRUE(v1.AddLabelAndValidate(dba.NameToLabel("label1")).HasValue()); // auto v2 = *dba.InsertVertexAndValidate(primary_label, {}, {{primary_property, PropertyValue(1)}}); // ASSERT_TRUE(v2.AddLabelAndValidate(dba.NameToLabel("label2")).HasValue()); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("t")); // ASSERT_TRUE(e.HasValue()); // ASSERT_TRUE(*EvaluateFunction("ENDNODE", *e).ValueVertex().HasLabel(View::NEW, dba.NameToLabel("label2"))); // ASSERT_THROW(EvaluateFunction("ENDNODE", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Head) { // ASSERT_THROW(EvaluateFunction("HEAD"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("HEAD", TypedValue()).IsNull()); // auto argument = MakeTypedValueList(3, 4, 5); // ASSERT_EQ(EvaluateFunction("HEAD", argument).ValueInt(), 3); // argument.ValueList().clear(); // ASSERT_TRUE(EvaluateFunction("HEAD", argument).IsNull()); // ASSERT_THROW(EvaluateFunction("HEAD", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Properties) { // ASSERT_THROW(EvaluateFunction("PROPERTIES"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("PROPERTIES", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // ASSERT_TRUE(v1.SetProperty(dba.NameToProperty("height"), PropertyValue(5)).HasValue()); // ASSERT_TRUE(v1.SetProperty(dba.NameToProperty("age"), PropertyValue(10)).HasValue()); // auto v2 = dba.InsertVertex(); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("type1")); // ASSERT_TRUE(e.HasValue()); // ASSERT_TRUE(e->SetProperty(dba.NameToProperty("height"), PropertyValue(3)).HasValue()); // ASSERT_TRUE(e->SetProperty(dba.NameToProperty("age"), PropertyValue(15)).HasValue()); // dba.AdvanceCommand(); // auto prop_values_to_int = [](TypedValue t) { // std::unordered_map properties; // for (auto property : t.ValueMap()) { // properties[std::string(property.first)] = property.second.ValueInt(); // } // return properties; // }; // ASSERT_THAT(prop_values_to_int(EvaluateFunction("PROPERTIES", v1)), // UnorderedElementsAre(testing::Pair("height", 5), testing::Pair("age", 10))); // ASSERT_THAT(prop_values_to_int(EvaluateFunction("PROPERTIES", *e)), // UnorderedElementsAre(testing::Pair("height", 3), testing::Pair("age", 15))); // ASSERT_THROW(EvaluateFunction("PROPERTIES", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Last) { // ASSERT_THROW(EvaluateFunction("LAST"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("LAST", TypedValue()).IsNull()); // auto argument = MakeTypedValueList(3, 4, 5); // ASSERT_EQ(EvaluateFunction("LAST", argument).ValueInt(), 5); // argument.ValueList().clear(); // ASSERT_TRUE(EvaluateFunction("LAST", argument).IsNull()); // ASSERT_THROW(EvaluateFunction("LAST", 5), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Size) { // ASSERT_THROW(EvaluateFunction("SIZE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("SIZE", TypedValue()).IsNull()); // auto argument = MakeTypedValueList(3, 4, 5); // ASSERT_EQ(EvaluateFunction("SIZE", argument).ValueInt(), 3); // ASSERT_EQ(EvaluateFunction("SIZE", "john").ValueInt(), 4); // ASSERT_EQ(EvaluateFunction("SIZE", // std::map{ // {"a", TypedValue(5)}, {"b", TypedValue(true)}, {"c", TypedValue("123")}}) // .ValueInt(), // 3); // ASSERT_THROW(EvaluateFunction("SIZE", 5), expr::ExpressionRuntimeException); // auto v0 = dba.InsertVertex(); // memgraph::query::Path path(v0); // EXPECT_EQ(EvaluateFunction("SIZE", path).ValueInt(), 0); // auto v1 = dba.InsertVertex(); // auto edge = dba.InsertEdge(&v0, &v1, dba.NameToEdgeType("type")); // ASSERT_TRUE(edge.HasValue()); // path.Expand(*edge); // path.Expand(v1); // EXPECT_EQ(EvaluateFunction("SIZE", path).ValueInt(), 1); // } // TEST_F(FunctionTest, StartNode) { // ASSERT_THROW(EvaluateFunction("STARTNODE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("STARTNODE", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // ASSERT_TRUE(v1.AddLabel(dba.NameToLabel("label1")).HasValue()); // auto v2 = dba.InsertVertex(); // ASSERT_TRUE(v2.AddLabel(dba.NameToLabel("label2")).HasValue()); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("t")); // ASSERT_TRUE(e.HasValue()); // ASSERT_TRUE(*EvaluateFunction("STARTNODE", *e) // .ValueVertex() // .HasLabel(memgraph::storage::View::NEW, dba.NameToLabel("label1"))); // ASSERT_THROW(EvaluateFunction("STARTNODE", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Degree) { // ASSERT_THROW(EvaluateFunction("DEGREE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("DEGREE", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // auto v2 = dba.InsertVertex(); // auto v3 = dba.InsertVertex(); // auto e12 = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("t")); // ASSERT_TRUE(e12.HasValue()); // ASSERT_TRUE(dba.InsertEdge(&v3, &v2, dba.NameToEdgeType("t")).HasValue()); // dba.AdvanceCommand(); // ASSERT_EQ(EvaluateFunction("DEGREE", v1).ValueInt(), 1); // ASSERT_EQ(EvaluateFunction("DEGREE", v2).ValueInt(), 2); // ASSERT_EQ(EvaluateFunction("DEGREE", v3).ValueInt(), 1); // ASSERT_THROW(EvaluateFunction("DEGREE", 2), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("DEGREE", *e12), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, InDegree) { // ASSERT_THROW(EvaluateFunction("INDEGREE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("INDEGREE", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // auto v2 = dba.InsertVertex(); // auto v3 = dba.InsertVertex(); // auto e12 = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("t")); // ASSERT_TRUE(e12.HasValue()); // ASSERT_TRUE(dba.InsertEdge(&v3, &v2, dba.NameToEdgeType("t")).HasValue()); // dba.AdvanceCommand(); // ASSERT_EQ(EvaluateFunction("INDEGREE", v1).ValueInt(), 0); // ASSERT_EQ(EvaluateFunction("INDEGREE", v2).ValueInt(), 2); // ASSERT_EQ(EvaluateFunction("INDEGREE", v3).ValueInt(), 0); // ASSERT_THROW(EvaluateFunction("INDEGREE", 2), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("INDEGREE", *e12), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, OutDegree) { // ASSERT_THROW(EvaluateFunction("OUTDEGREE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("OUTDEGREE", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // auto v2 = dba.InsertVertex(); // auto v3 = dba.InsertVertex(); // auto e12 = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("t")); // ASSERT_TRUE(e12.HasValue()); // ASSERT_TRUE(dba.InsertEdge(&v3, &v2, dba.NameToEdgeType("t")).HasValue()); // dba.AdvanceCommand(); // ASSERT_EQ(EvaluateFunction("OUTDEGREE", v1).ValueInt(), 1); // ASSERT_EQ(EvaluateFunction("OUTDEGREE", v2).ValueInt(), 0); // ASSERT_EQ(EvaluateFunction("OUTDEGREE", v3).ValueInt(), 1); // ASSERT_THROW(EvaluateFunction("OUTDEGREE", 2), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("OUTDEGREE", *e12), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, ToBoolean) { // ASSERT_THROW(EvaluateFunction("TOBOOLEAN"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("TOBOOLEAN", TypedValue()).IsNull()); // 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").ValueBool(), true); // ASSERT_EQ(EvaluateFunction("TOBOOLEAN", "\n\tFalsE").ValueBool(), false); // ASSERT_TRUE(EvaluateFunction("TOBOOLEAN", "\n\tFALSEA ").IsNull()); // ASSERT_EQ(EvaluateFunction("TOBOOLEAN", true).ValueBool(), true); // ASSERT_EQ(EvaluateFunction("TOBOOLEAN", false).ValueBool(), false); // } // TEST_F(FunctionTest, ToFloat) { // ASSERT_THROW(EvaluateFunction("TOFLOAT"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("TOFLOAT", TypedValue()).IsNull()); // ASSERT_EQ(EvaluateFunction("TOFLOAT", " -3.5 \n\t").ValueDouble(), -3.5); // ASSERT_EQ(EvaluateFunction("TOFLOAT", "\n\t0.5e-1").ValueDouble(), 0.05); // ASSERT_TRUE(EvaluateFunction("TOFLOAT", "\n\t3.4e-3X ").IsNull()); // ASSERT_EQ(EvaluateFunction("TOFLOAT", -3.5).ValueDouble(), -3.5); // ASSERT_EQ(EvaluateFunction("TOFLOAT", -3).ValueDouble(), -3.0); // ASSERT_THROW(EvaluateFunction("TOFLOAT", true), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, ToInteger) { // ASSERT_THROW(EvaluateFunction("TOINTEGER"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("TOINTEGER", TypedValue()).IsNull()); // ASSERT_EQ(EvaluateFunction("TOINTEGER", false).ValueInt(), 0); // ASSERT_EQ(EvaluateFunction("TOINTEGER", true).ValueInt(), 1); // ASSERT_EQ(EvaluateFunction("TOINTEGER", "\n\t3").ValueInt(), 3); // ASSERT_EQ(EvaluateFunction("TOINTEGER", " -3.5 \n\t").ValueInt(), -3); // ASSERT_TRUE(EvaluateFunction("TOINTEGER", "\n\t3X ").IsNull()); // ASSERT_EQ(EvaluateFunction("TOINTEGER", -3.5).ValueInt(), -3); // ASSERT_EQ(EvaluateFunction("TOINTEGER", 3.5).ValueInt(), 3); // } // TEST_F(FunctionTest, Type) { // ASSERT_THROW(EvaluateFunction("TYPE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("TYPE", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // ASSERT_TRUE(v1.AddLabel(dba.NameToLabel("label1")).HasValue()); // auto v2 = dba.InsertVertex(); // ASSERT_TRUE(v2.AddLabel(dba.NameToLabel("label2")).HasValue()); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("type1")); // ASSERT_TRUE(e.HasValue()); // ASSERT_EQ(EvaluateFunction("TYPE", *e).ValueString(), "type1"); // ASSERT_THROW(EvaluateFunction("TYPE", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, ValueType) { // ASSERT_THROW(EvaluateFunction("VALUETYPE"), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("VALUETYPE", TypedValue(), TypedValue()), expr::ExpressionRuntimeException); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue()).ValueString(), "NULL"); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue(true)).ValueString(), "BOOLEAN"); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue(1)).ValueString(), "INTEGER"); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue(1.1)).ValueString(), "FLOAT"); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue("test")).ValueString(), "STRING"); // ASSERT_EQ( // EvaluateFunction("VALUETYPE", TypedValue(std::vector{TypedValue(1), TypedValue(2)})).ValueString(), // "LIST"); // ASSERT_EQ(EvaluateFunction("VALUETYPE", TypedValue(std::map{{"test", TypedValue(1)}})) // .ValueString(), // "MAP"); // auto v1 = dba.InsertVertex(); // auto v2 = dba.InsertVertex(); // ASSERT_EQ(EvaluateFunction("VALUETYPE", v1).ValueString(), "NODE"); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("type1")); // ASSERT_TRUE(e.HasValue()); // ASSERT_EQ(EvaluateFunction("VALUETYPE", *e).ValueString(), "RELATIONSHIP"); // Path p(v1, *e, v2); // ASSERT_EQ(EvaluateFunction("VALUETYPE", p).ValueString(), "PATH"); // } // TEST_F(FunctionTest, Labels) { // ASSERT_THROW(EvaluateFunction("LABELS"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("LABELS", TypedValue()).IsNull()); // auto v = dba.InsertVertex(); // ASSERT_TRUE(v.AddLabel(dba.NameToLabel("label1")).HasValue()); // ASSERT_TRUE(v.AddLabel(dba.NameToLabel("label2")).HasValue()); // dba.AdvanceCommand(); // std::vector labels; // auto _labels = EvaluateFunction("LABELS", v).ValueList(); // labels.reserve(_labels.size()); // for (auto label : _labels) { // labels.emplace_back(label.ValueString()); // } // ASSERT_THAT(labels, UnorderedElementsAre("label1", "label2")); // ASSERT_THROW(EvaluateFunction("LABELS", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, NodesRelationships) { // EXPECT_THROW(EvaluateFunction("NODES"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RELATIONSHIPS"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("NODES", TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("RELATIONSHIPS", TypedValue()).IsNull()); // { // auto v1 = dba.InsertVertex(); // auto v2 = dba.InsertVertex(); // auto v3 = dba.InsertVertex(); // auto e1 = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("Type")); // ASSERT_TRUE(e1.HasValue()); // auto e2 = dba.InsertEdge(&v2, &v3, dba.NameToEdgeType("Type")); // ASSERT_TRUE(e2.HasValue()); // memgraph::query::Path path{v1, *e1, v2, *e2, v3}; // dba.AdvanceCommand(); // 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), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RELATIONSHIPS", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Range) { // EXPECT_THROW(EvaluateFunction("RANGE"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("RANGE", 1, 2, TypedValue()).IsNull()); // EXPECT_THROW(EvaluateFunction("RANGE", 1, TypedValue(), 1.3), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RANGE", 1, 2, 0), expr::ExpressionRuntimeException); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 1, 3)), ElementsAre(1, 2, 3)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", -1, 5, 2)), ElementsAre(-1, 1, 3, 5)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 2, 10, 3)), ElementsAre(2, 5, 8)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 2, 2, 2)), ElementsAre(2)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 3, 0, 5)), ElementsAre()); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 5, 1, -2)), ElementsAre(5, 3, 1)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 6, 1, -2)), ElementsAre(6, 4, 2)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", 2, 2, -3)), ElementsAre(2)); // EXPECT_THAT(ToIntList(EvaluateFunction("RANGE", -2, 4, -1)), ElementsAre()); // } // TEST_F(FunctionTest, Keys) { // ASSERT_THROW(EvaluateFunction("KEYS"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("KEYS", TypedValue()).IsNull()); // auto v1 = dba.InsertVertex(); // ASSERT_TRUE(v1.SetProperty(dba.NameToProperty("height"), PropertyValue(5)).HasValue()); // ASSERT_TRUE(v1.SetProperty(dba.NameToProperty("age"), PropertyValue(10)).HasValue()); // auto v2 = dba.InsertVertex(); // auto e = dba.InsertEdge(&v1, &v2, dba.NameToEdgeType("type1")); // ASSERT_TRUE(e.HasValue()); // ASSERT_TRUE(e->SetProperty(dba.NameToProperty("width"), PropertyValue(3)).HasValue()); // ASSERT_TRUE(e->SetProperty(dba.NameToProperty("age"), PropertyValue(15)).HasValue()); // dba.AdvanceCommand(); // auto prop_keys_to_string = [](TypedValue t) { // std::vector keys; // for (auto property : t.ValueList()) { // keys.emplace_back(property.ValueString()); // } // return keys; // }; // ASSERT_THAT(prop_keys_to_string(EvaluateFunction("KEYS", v1)), UnorderedElementsAre("height", "age")); // ASSERT_THAT(prop_keys_to_string(EvaluateFunction("KEYS", *e)), UnorderedElementsAre("width", "age")); // ASSERT_THROW(EvaluateFunction("KEYS", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Tail) { // ASSERT_THROW(EvaluateFunction("TAIL"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("TAIL", TypedValue()).IsNull()); // auto argument = MakeTypedValueList(); // ASSERT_EQ(EvaluateFunction("TAIL", argument).ValueList().size(), 0U); // argument = MakeTypedValueList(3, 4, true, "john"); // auto list = EvaluateFunction("TAIL", argument).ValueList(); // ASSERT_EQ(list.size(), 3U); // ASSERT_EQ(list[0].ValueInt(), 4); // ASSERT_EQ(list[1].ValueBool(), true); // ASSERT_EQ(list[2].ValueString(), "john"); // ASSERT_THROW(EvaluateFunction("TAIL", 2), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, UniformSample) { // ASSERT_THROW(EvaluateFunction("UNIFORMSAMPLE"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("UNIFORMSAMPLE", TypedValue(), TypedValue()).IsNull()); // ASSERT_TRUE(EvaluateFunction("UNIFORMSAMPLE", TypedValue(), 1).IsNull()); // ASSERT_TRUE(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(), TypedValue()).IsNull()); // ASSERT_TRUE(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(), 1).IsNull()); // ASSERT_THROW(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(1, 2, 3), -1), expr::ExpressionRuntimeException); // ASSERT_EQ(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(1, 2, 3), 0).ValueList().size(), 0); // ASSERT_EQ(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(1, 2, 3), 2).ValueList().size(), 2); // ASSERT_EQ(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(1, 2, 3), 3).ValueList().size(), 3); // ASSERT_EQ(EvaluateFunction("UNIFORMSAMPLE", MakeTypedValueList(1, 2, 3), 5).ValueList().size(), 5); // } // TEST_F(FunctionTest, Abs) { // ASSERT_THROW(EvaluateFunction("ABS"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("ABS", TypedValue()).IsNull()); // ASSERT_EQ(EvaluateFunction("ABS", -2).ValueInt(), 2); // ASSERT_EQ(EvaluateFunction("ABS", -2.5).ValueDouble(), 2.5); // ASSERT_THROW(EvaluateFunction("ABS", true), expr::ExpressionRuntimeException); // } // // 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_F(FunctionTest, Log) { // ASSERT_THROW(EvaluateFunction("LOG"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("LOG", TypedValue()).IsNull()); // ASSERT_DOUBLE_EQ(EvaluateFunction("LOG", 2).ValueDouble(), log(2)); // ASSERT_DOUBLE_EQ(EvaluateFunction("LOG", 1.5).ValueDouble(), log(1.5)); // // Not portable, but should work on most platforms. // ASSERT_TRUE(std::isnan(EvaluateFunction("LOG", -1.5).ValueDouble())); // ASSERT_THROW(EvaluateFunction("LOG", true), expr::ExpressionRuntimeException); // } // // Function Round wraps round from cmath and will work if FunctionTest.Log test // // passes. This test is used to show behavior of round since it differs from // // neo4j's round. // TEST_F(FunctionTest, Round) { // ASSERT_THROW(EvaluateFunction("ROUND"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("ROUND", TypedValue()).IsNull()); // ASSERT_EQ(EvaluateFunction("ROUND", -2).ValueDouble(), -2); // ASSERT_EQ(EvaluateFunction("ROUND", -2.4).ValueDouble(), -2); // ASSERT_EQ(EvaluateFunction("ROUND", -2.5).ValueDouble(), -3); // ASSERT_EQ(EvaluateFunction("ROUND", -2.6).ValueDouble(), -3); // ASSERT_EQ(EvaluateFunction("ROUND", 2.4).ValueDouble(), 2); // ASSERT_EQ(EvaluateFunction("ROUND", 2.5).ValueDouble(), 3); // ASSERT_EQ(EvaluateFunction("ROUND", 2.6).ValueDouble(), 3); // ASSERT_THROW(EvaluateFunction("ROUND", true), expr::ExpressionRuntimeException); // } // // Check if wrapped functions are callable (check if everything was spelled // // correctly...). Wrapper correctnes is checked in FunctionTest.Log function // // test. // TEST_F(FunctionTest, WrappedMathFunctions) { // for (auto function_name : // {"FLOOR", "CEIL", "ROUND", "EXP", "LOG", "LOG10", "SQRT", "ACOS", "ASIN", "ATAN", "COS", "SIN", "TAN"}) { // EvaluateFunction(function_name, 0.5); // } // } // TEST_F(FunctionTest, Atan2) { // ASSERT_THROW(EvaluateFunction("ATAN2"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("ATAN2", TypedValue(), 1).IsNull()); // ASSERT_TRUE(EvaluateFunction("ATAN2", 1, TypedValue()).IsNull()); // ASSERT_DOUBLE_EQ(EvaluateFunction("ATAN2", 2, -1.0).ValueDouble(), atan2(2, -1)); // ASSERT_THROW(EvaluateFunction("ATAN2", 3.0, true), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Sign) { // ASSERT_THROW(EvaluateFunction("SIGN"), expr::ExpressionRuntimeException); // ASSERT_TRUE(EvaluateFunction("SIGN", TypedValue()).IsNull()); // ASSERT_EQ(EvaluateFunction("SIGN", -2).ValueInt(), -1); // ASSERT_EQ(EvaluateFunction("SIGN", -0.2).ValueInt(), -1); // ASSERT_EQ(EvaluateFunction("SIGN", 0.0).ValueInt(), 0); // ASSERT_EQ(EvaluateFunction("SIGN", 2.5).ValueInt(), 1); // ASSERT_THROW(EvaluateFunction("SIGN", true), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, E) { // ASSERT_THROW(EvaluateFunction("E", 1), expr::ExpressionRuntimeException); // ASSERT_DOUBLE_EQ(EvaluateFunction("E").ValueDouble(), M_E); // } // TEST_F(FunctionTest, Pi) { // ASSERT_THROW(EvaluateFunction("PI", 1), expr::ExpressionRuntimeException); // ASSERT_DOUBLE_EQ(EvaluateFunction("PI").ValueDouble(), M_PI); // } // TEST_F(FunctionTest, Rand) { // ASSERT_THROW(EvaluateFunction("RAND", 1), expr::ExpressionRuntimeException); // ASSERT_GE(EvaluateFunction("RAND").ValueDouble(), 0.0); // ASSERT_LT(EvaluateFunction("RAND").ValueDouble(), 1.0); // } // TEST_F(FunctionTest, StartsWith) { // EXPECT_THROW(EvaluateFunction(kStartsWith), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kStartsWith, "a", TypedValue()).IsNull()); // EXPECT_THROW(EvaluateFunction(kStartsWith, TypedValue(), 1.3), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kStartsWith, "abc", "abc").ValueBool()); // EXPECT_TRUE(EvaluateFunction(kStartsWith, "abcdef", "abc").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kStartsWith, "abcdef", "aBc").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kStartsWith, "abc", "abcd").ValueBool()); // } // TEST_F(FunctionTest, EndsWith) { // EXPECT_THROW(EvaluateFunction(kEndsWith), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kEndsWith, "a", TypedValue()).IsNull()); // EXPECT_THROW(EvaluateFunction(kEndsWith, TypedValue(), 1.3), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kEndsWith, "abc", "abc").ValueBool()); // EXPECT_TRUE(EvaluateFunction(kEndsWith, "abcdef", "def").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kEndsWith, "abcdef", "dEf").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kEndsWith, "bcd", "abcd").ValueBool()); // } // TEST_F(FunctionTest, Contains) { // EXPECT_THROW(EvaluateFunction(kContains), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kContains, "a", TypedValue()).IsNull()); // EXPECT_THROW(EvaluateFunction(kContains, TypedValue(), 1.3), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction(kContains, "abc", "abc").ValueBool()); // EXPECT_TRUE(EvaluateFunction(kContains, "abcde", "bcd").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kContains, "cde", "abcdef").ValueBool()); // EXPECT_FALSE(EvaluateFunction(kContains, "abcdef", "dEf").ValueBool()); // } // TEST_F(FunctionTest, Assert) { // // Invalid calls. // ASSERT_THROW(EvaluateFunction("ASSERT"), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("ASSERT", false, false), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("ASSERT", "string", false), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("ASSERT", false, "reason", true), expr::ExpressionRuntimeException); // // Valid calls, assertion fails. // ASSERT_THROW(EvaluateFunction("ASSERT", false), expr::ExpressionRuntimeException); // ASSERT_THROW(EvaluateFunction("ASSERT", false, "message"), expr::ExpressionRuntimeException); // try { // EvaluateFunction("ASSERT", false, "bbgba"); // } catch (expr::ExpressionRuntimeException &e) { // ASSERT_TRUE(std::string(e.what()).find("bbgba") != std::string::npos); // } // // Valid calls, assertion passes. // ASSERT_TRUE(EvaluateFunction("ASSERT", true).ValueBool()); // ASSERT_TRUE(EvaluateFunction("ASSERT", true, "message").ValueBool()); // } // TEST_F(FunctionTest, Counter) { // EXPECT_THROW(EvaluateFunction("COUNTER"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("COUNTER", "a"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("COUNTER", "a", "b"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("COUNTER", "a", "b", "c"), expr::ExpressionRuntimeException); // EXPECT_EQ(EvaluateFunction("COUNTER", "c1", 0).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("COUNTER", "c1", 0).ValueInt(), 1); // EXPECT_EQ(EvaluateFunction("COUNTER", "c2", 0).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("COUNTER", "c1", 0).ValueInt(), 2); // EXPECT_EQ(EvaluateFunction("COUNTER", "c2", 0).ValueInt(), 1); // EXPECT_EQ(EvaluateFunction("COUNTER", "c3", -1).ValueInt(), -1); // EXPECT_EQ(EvaluateFunction("COUNTER", "c3", -1).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("COUNTER", "c3", -1).ValueInt(), 1); // EXPECT_EQ(EvaluateFunction("COUNTER", "c4", 0, 5).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("COUNTER", "c4", 0, 5).ValueInt(), 5); // EXPECT_EQ(EvaluateFunction("COUNTER", "c4", 0, 5).ValueInt(), 10); // EXPECT_EQ(EvaluateFunction("COUNTER", "c5", 0, -5).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("COUNTER", "c5", 0, -5).ValueInt(), -5); // EXPECT_EQ(EvaluateFunction("COUNTER", "c5", 0, -5).ValueInt(), -10); // EXPECT_THROW(EvaluateFunction("COUNTER", "c6", 0, 0), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Id) { // auto va = dba.InsertVertex(); // auto ea = dba.InsertEdge(&va, &va, dba.NameToEdgeType("edge")); // ASSERT_TRUE(ea.HasValue()); // auto vb = dba.InsertVertex(); // dba.AdvanceCommand(); // EXPECT_TRUE(EvaluateFunction("ID", TypedValue()).IsNull()); // EXPECT_EQ(EvaluateFunction("ID", va).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("ID", *ea).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("ID", vb).ValueInt(), 1); // EXPECT_THROW(EvaluateFunction("ID"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("ID", 0), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("ID", va, *ea), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, ToStringNull) { EXPECT_TRUE(EvaluateFunction("TOSTRING", TypedValue()).IsNull()); } // TEST_F(FunctionTest, ToStringString) { // EXPECT_EQ(EvaluateFunction("TOSTRING", "").ValueString(), ""); // EXPECT_EQ(EvaluateFunction("TOSTRING", "this is a string").ValueString(), "this is a string"); // } // TEST_F(FunctionTest, ToStringInteger) { // EXPECT_EQ(EvaluateFunction("TOSTRING", -23321312).ValueString(), "-23321312"); // EXPECT_EQ(EvaluateFunction("TOSTRING", 0).ValueString(), "0"); // EXPECT_EQ(EvaluateFunction("TOSTRING", 42).ValueString(), "42"); // } // TEST_F(FunctionTest, ToStringDouble) { // 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_F(FunctionTest, ToStringBool) { // EXPECT_EQ(EvaluateFunction("TOSTRING", true).ValueString(), "true"); // EXPECT_EQ(EvaluateFunction("TOSTRING", false).ValueString(), "false"); // } // TEST_F(FunctionTest, ToStringDate) { // const auto date = memgraph::utils::Date({1970, 1, 2}); // EXPECT_EQ(EvaluateFunction("TOSTRING", date).ValueString(), "1970-01-02"); // } // TEST_F(FunctionTest, ToStringLocalTime) { // const auto lt = memgraph::utils::LocalTime({13, 2, 40, 100, 50}); // EXPECT_EQ(EvaluateFunction("TOSTRING", lt).ValueString(), "13:02:40.100050"); // } // TEST_F(FunctionTest, ToStringLocalDateTime) { // const auto ldt = memgraph::utils::LocalDateTime({1970, 1, 2}, {23, 02, 59}); // EXPECT_EQ(EvaluateFunction("TOSTRING", ldt).ValueString(), "1970-01-02T23:02:59.000000"); // } // TEST_F(FunctionTest, ToStringDuration) { // memgraph::utils::Duration duration{{.minute = 2, .second = 2, .microsecond = 33}}; // EXPECT_EQ(EvaluateFunction("TOSTRING", duration).ValueString(), "P0DT0H2M2.000033S"); // } // TEST_F(FunctionTest, ToStringExceptions) { // EXPECT_THROW(EvaluateFunction("TOSTRING", 1, 2, 3), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, TimestampVoid) { // ctx.timestamp = 42; // EXPECT_EQ(EvaluateFunction("TIMESTAMP").ValueInt(), 42); // } // TEST_F(FunctionTest, TimestampDate) { // ctx.timestamp = 42; // EXPECT_EQ(EvaluateFunction("TIMESTAMP", memgraph::utils::Date({1970, 1, 1})).ValueInt(), 0); // EXPECT_EQ(EvaluateFunction("TIMESTAMP", memgraph::utils::Date({1971, 1, 1})).ValueInt(), 31536000000000); // } // TEST_F(FunctionTest, TimestampLocalTime) { // ctx.timestamp = 42; // const memgraph::utils::LocalTime time(10000); // EXPECT_EQ(EvaluateFunction("TIMESTAMP", time).ValueInt(), 10000); // } // TEST_F(FunctionTest, TimestampLocalDateTime) { // ctx.timestamp = 42; // const memgraph::utils::LocalDateTime time(20000); // EXPECT_EQ(EvaluateFunction("TIMESTAMP", time).ValueInt(), 20000); // } // TEST_F(FunctionTest, TimestampDuration) { // ctx.timestamp = 42; // const memgraph::utils::Duration time(20000); // EXPECT_EQ(EvaluateFunction("TIMESTAMP", time).ValueInt(), 20000); // } // TEST_F(FunctionTest, TimestampExceptions) { // ctx.timestamp = 42; // EXPECT_THROW(EvaluateFunction("TIMESTAMP", 1).ValueInt(), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Left) { // EXPECT_THROW(EvaluateFunction("LEFT"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("LEFT", TypedValue(), TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("LEFT", TypedValue(), 10).IsNull()); // EXPECT_THROW(EvaluateFunction("LEFT", TypedValue(), -10), expr::ExpressionRuntimeException); // 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), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("LEFT", "memgraph", "graph"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("LEFT", 132, 10), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Right) { // EXPECT_THROW(EvaluateFunction("RIGHT"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("RIGHT", TypedValue(), TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("RIGHT", TypedValue(), 10).IsNull()); // EXPECT_THROW(EvaluateFunction("RIGHT", TypedValue(), -10), expr::ExpressionRuntimeException); // 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), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RIGHT", "memgraph", "graph"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RIGHT", 132, 10), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Trimming) { // EXPECT_TRUE(EvaluateFunction("LTRIM", TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("RTRIM", TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("TRIM", TypedValue()).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"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("RTRIM", "x", "y"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TRIM", "x", "y"), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Reverse) { // EXPECT_TRUE(EvaluateFunction("REVERSE", TypedValue()).IsNull()); // EXPECT_EQ(EvaluateFunction("REVERSE", "abc").ValueString(), "cba"); // EXPECT_THROW(EvaluateFunction("REVERSE", "x", "y"), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Replace) { // EXPECT_THROW(EvaluateFunction("REPLACE"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("REPLACE", TypedValue(), "l", "w").IsNull()); // EXPECT_TRUE(EvaluateFunction("REPLACE", "hello", TypedValue(), "w").IsNull()); // EXPECT_TRUE(EvaluateFunction("REPLACE", "hello", "l", TypedValue()).IsNull()); // EXPECT_EQ(EvaluateFunction("REPLACE", "hello", "l", "w").ValueString(), "hewwo"); // EXPECT_THROW(EvaluateFunction("REPLACE", 1, "l", "w"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("REPLACE", "hello", 1, "w"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("REPLACE", "hello", "l", 1), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Split) { // EXPECT_THROW(EvaluateFunction("SPLIT"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("SPLIT", "one,two", 1), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("SPLIT", 1, "one,two"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("SPLIT", TypedValue(), TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("SPLIT", "one,two", TypedValue()).IsNull()); // EXPECT_TRUE(EvaluateFunction("SPLIT", TypedValue(), ",").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_F(FunctionTest, Substring) { // EXPECT_THROW(EvaluateFunction("SUBSTRING"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("SUBSTRING", TypedValue(), 0, 10).IsNull()); // EXPECT_THROW(EvaluateFunction("SUBSTRING", TypedValue(), TypedValue()), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("SUBSTRING", TypedValue(), -10), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("SUBSTRING", TypedValue(), 0, TypedValue()), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("SUBSTRING", TypedValue(), 0, -10), expr::ExpressionRuntimeException); // 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_F(FunctionTest, ToLower) { // EXPECT_THROW(EvaluateFunction("TOLOWER"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("TOLOWER", TypedValue()).IsNull()); // EXPECT_EQ(EvaluateFunction("TOLOWER", "Ab__C").ValueString(), "ab__c"); // } // TEST_F(FunctionTest, ToUpper) { // EXPECT_THROW(EvaluateFunction("TOUPPER"), expr::ExpressionRuntimeException); // EXPECT_TRUE(EvaluateFunction("TOUPPER", TypedValue()).IsNull()); // EXPECT_EQ(EvaluateFunction("TOUPPER", "Ab__C").ValueString(), "AB__C"); // } // TEST_F(FunctionTest, ToByteString) { // EXPECT_THROW(EvaluateFunction("TOBYTESTRING"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", 42), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", TypedValue()), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", "", 42), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", "ff"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", "00"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("TOBYTESTRING", "0xG"), expr::ExpressionRuntimeException); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "").ValueString(), ""); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "0x").ValueString(), ""); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "0X").ValueString(), ""); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "0x0123456789aAbBcCdDeEfF").ValueString(), // "\x01\x23\x45\x67\x89\xAA\xBB\xCC\xDD\xEE\xFF"); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "0x042").ValueString().size(), 2); // EXPECT_EQ(EvaluateFunction("TOBYTESTRING", "0x042").ValueString(), // memgraph::utils::pmr::string("\x00\x42", 2, memgraph::utils::NewDeleteResource())); // } // TEST_F(FunctionTest, FromByteString) { // EXPECT_THROW(EvaluateFunction("FROMBYTESTRING"), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("FROMBYTESTRING", 42), expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("FROMBYTESTRING", TypedValue()), expr::ExpressionRuntimeException); // EXPECT_EQ(EvaluateFunction("FROMBYTESTRING", "").ValueString(), ""); // auto bytestring = EvaluateFunction("TOBYTESTRING", "0x123456789aAbBcCdDeEfF"); // EXPECT_EQ(EvaluateFunction("FROMBYTESTRING", bytestring).ValueString(), "0x0123456789aabbccddeeff"); // EXPECT_EQ(EvaluateFunction("FROMBYTESTRING", std::string("\x00\x42", 2)).ValueString(), "0x0042"); // } // TEST_F(FunctionTest, Date) { // const auto unix_epoch = memgraph::utils::Date({1970, 1, 1}); // EXPECT_EQ(EvaluateFunction("DATE", "1970-01-01").ValueDate(), unix_epoch); // const auto map_param = TypedValue( // std::map{{"year", TypedValue(1970)}, {"month", TypedValue(1)}, {"day", // TypedValue(1)}}); // EXPECT_EQ(EvaluateFunction("DATE", map_param).ValueDate(), unix_epoch); // const auto today = memgraph::utils::CurrentDate(); // EXPECT_EQ(EvaluateFunction("DATE").ValueDate(), today); // EXPECT_THROW(EvaluateFunction("DATE", "{}"), memgraph::utils::BasicException); // EXPECT_THROW(EvaluateFunction("DATE", std::map{{"years", TypedValue(1970)}}), // expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("DATE", std::map{{"mnths", TypedValue(1970)}}), // expr::ExpressionRuntimeException); // EXPECT_THROW(EvaluateFunction("DATE", std::map{{"dayz", TypedValue(1970)}}), // expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, LocalTime) { // const auto local_time = memgraph::utils::LocalTime({13, 3, 2, 0, 0}); // EXPECT_EQ(EvaluateFunction("LOCALTIME", "130302").ValueLocalTime(), local_time); // const auto one_sec_in_microseconds = 1000000; // const auto map_param = TypedValue(std::map{{"hour", TypedValue(1)}, // {"minute", TypedValue(2)}, // {"second", TypedValue(3)}, // {"millisecond", TypedValue(4)}, // {"microsecond", TypedValue(5)}}); // EXPECT_EQ(EvaluateFunction("LOCALTIME", map_param).ValueLocalTime(), memgraph::utils::LocalTime({1, 2, 3, 4, 5})); // const auto today = memgraph::utils::CurrentLocalTime(); // EXPECT_NEAR(EvaluateFunction("LOCALTIME").ValueLocalTime().MicrosecondsSinceEpoch(), // today.MicrosecondsSinceEpoch(), // one_sec_in_microseconds); // EXPECT_THROW(EvaluateFunction("LOCALTIME", "{}"), memgraph::utils::BasicException); // EXPECT_THROW(EvaluateFunction("LOCALTIME", TypedValue(std::map{{"hous", // TypedValue(1970)}})), // expr::ExpressionRuntimeException); // EXPECT_THROW( // EvaluateFunction("LOCALTIME", TypedValue(std::map{{"minut", TypedValue(1970)}})), // expr::ExpressionRuntimeException); // EXPECT_THROW( // EvaluateFunction("LOCALTIME", TypedValue(std::map{{"seconds", TypedValue(1970)}})), // expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, LocalDateTime) { // const auto local_date_time = memgraph::utils::LocalDateTime({1970, 1, 1}, {13, 3, 2, 0, 0}); // EXPECT_EQ(EvaluateFunction("LOCALDATETIME", "1970-01-01T13:03:02").ValueLocalDateTime(), local_date_time); // const auto today = memgraph::utils::CurrentLocalDateTime(); // const auto one_sec_in_microseconds = 1000000; // const auto map_param = TypedValue(std::map{{"year", TypedValue(1972)}, // {"month", TypedValue(2)}, // {"day", TypedValue(3)}, // {"hour", TypedValue(4)}, // {"minute", TypedValue(5)}, // {"second", TypedValue(6)}, // {"millisecond", TypedValue(7)}, // {"microsecond", TypedValue(8)}}); // EXPECT_EQ(EvaluateFunction("LOCALDATETIME", map_param).ValueLocalDateTime(), // memgraph::utils::LocalDateTime({1972, 2, 3}, {4, 5, 6, 7, 8})); // EXPECT_NEAR(EvaluateFunction("LOCALDATETIME").ValueLocalDateTime().MicrosecondsSinceEpoch(), // today.MicrosecondsSinceEpoch(), one_sec_in_microseconds); // EXPECT_THROW(EvaluateFunction("LOCALDATETIME", "{}"), memgraph::utils::BasicException); // EXPECT_THROW( // EvaluateFunction("LOCALDATETIME", TypedValue(std::map{{"hours", TypedValue(1970)}})), // expr::ExpressionRuntimeException); // EXPECT_THROW( // EvaluateFunction("LOCALDATETIME", TypedValue(std::map{{"seconds", // TypedValue(1970)}})), expr::ExpressionRuntimeException); // } // TEST_F(FunctionTest, Duration) { // const auto map_param = TypedValue(std::map{{"day", TypedValue(3)}, // {"hour", TypedValue(4)}, // {"minute", TypedValue(5)}, // {"second", TypedValue(6)}, // {"millisecond", TypedValue(7)}, // {"microsecond", TypedValue(8)}}); // EXPECT_EQ(EvaluateFunction("DURATION", map_param).ValueDuration(), memgraph::utils::Duration({3, 4, 5, 6, 7, 8})); // EXPECT_THROW(EvaluateFunction("DURATION", "{}"), memgraph::utils::BasicException); // EXPECT_THROW(EvaluateFunction("DURATION", TypedValue(std::map{{"hours", // TypedValue(1970)}})), // expr::ExpressionRuntimeException); // EXPECT_THROW( // EvaluateFunction("DURATION", TypedValue(std::map{{"seconds", TypedValue(1970)}})), // expr::ExpressionRuntimeException); // const auto map_param_negative = TypedValue(std::map{{"day", TypedValue(-3)}, // {"hour", TypedValue(-4)}, // {"minute", TypedValue(-5)}, // {"second", TypedValue(-6)}, // {"millisecond", TypedValue(-7)}, // {"microsecond", TypedValue(-8)}}); // EXPECT_EQ(EvaluateFunction("DURATION", map_param_negative).ValueDuration(), // memgraph::utils::Duration({-3, -4, -5, -6, -7, -8})); // EXPECT_EQ(EvaluateFunction("DURATION", "P4DT4H5M6.2S").ValueDuration(), // memgraph::utils::Duration({4, 4, 5, 6, 0, 200000})); // EXPECT_EQ(EvaluateFunction("DURATION", "P3DT4H5M6.100S").ValueDuration(), // memgraph::utils::Duration({3, 4, 5, 6, 0, 100000})); // EXPECT_EQ(EvaluateFunction("DURATION", "P3DT4H5M6.100110S").ValueDuration(), // memgraph::utils::Duration({3, 4, 5, 6, 100, 110})); // } } // namespace memgraph::storage::v3::test