2018-11-07 17:54:35 +08:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include "database/single_node/graph_db_accessor.hpp"
|
|
|
|
#include "query/frontend/ast/ast.hpp"
|
|
|
|
#include "query/frontend/ast/pretty_print.hpp"
|
|
|
|
#include "query_common.hpp"
|
|
|
|
#include "utils/string.hpp"
|
|
|
|
|
|
|
|
using namespace query;
|
|
|
|
using query::test_common::ToList;
|
|
|
|
using query::test_common::ToString;
|
|
|
|
using testing::ElementsAre;
|
|
|
|
using testing::UnorderedElementsAre;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
struct ExpressionPrettyPrinterTest : public ::testing::Test {
|
|
|
|
ExpressionPrettyPrinterTest() : pdba{db.Access()}, dba{*pdba} {}
|
|
|
|
|
|
|
|
database::GraphDb db;
|
|
|
|
std::unique_ptr<database::GraphDbAccessor> pdba;
|
|
|
|
database::GraphDbAccessor &dba;
|
|
|
|
AstStorage storage;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, Literals) {
|
|
|
|
// 1
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(1)), "1");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// "hello"
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL("hello")), "\"hello\"");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// null
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(TypedValue::Null)), "null");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// true
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(true)), "true");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// false
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(false)), "false");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// [1 null "hello"]
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(
|
|
|
|
(std::vector<PropertyValue>{1, PropertyValue::Null, "hello"}))),
|
2018-11-07 17:54:35 +08:00
|
|
|
"[1, null, \"hello\"]");
|
|
|
|
|
|
|
|
// {hello: 1, there: 2}
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL((std::map<std::string, PropertyValue>{
|
|
|
|
{"hello", 1}, {"there", 2}}))),
|
2018-11-07 17:54:35 +08:00
|
|
|
"{\"hello\": 1, \"there\": 2}");
|
|
|
|
}
|
|
|
|
|
2019-01-16 17:43:32 +08:00
|
|
|
TEST_F(ExpressionPrettyPrinterTest, Identifiers) {
|
|
|
|
// x
|
|
|
|
EXPECT_EQ(ToString(IDENT("x")), "(Identifier \"x\")");
|
|
|
|
|
|
|
|
// hello_there
|
|
|
|
EXPECT_EQ(ToString(IDENT("hello_there")), "(Identifier \"hello_there\")");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, Reducing) {
|
|
|
|
// all(x in list where x.prop = 42)
|
|
|
|
auto prop = dba.Property("prop");
|
|
|
|
EXPECT_EQ(ToString(ALL("x", LITERAL(std::vector<PropertyValue>{}),
|
|
|
|
WHERE(EQ(PROPERTY_LOOKUP("x", prop), LITERAL(42))))),
|
|
|
|
"(All (Identifier \"x\") [] (== (PropertyLookup "
|
|
|
|
"(Identifier \"x\") \"prop\") 42))");
|
|
|
|
|
|
|
|
// reduce(accumulator = initial_value, variable IN list | expression)
|
|
|
|
EXPECT_EQ(
|
|
|
|
ToString(REDUCE("accumulator", IDENT("initial_value"), "variable",
|
|
|
|
IDENT("list"), IDENT("expression"))),
|
|
|
|
"(Reduce (Identifier \"accumulator\") (Identifier \"initial_value\") "
|
|
|
|
"(Identifier \"variable\") (Identifier \"list\") (Identifier "
|
|
|
|
"\"expression\"))");
|
|
|
|
}
|
|
|
|
|
2018-11-07 17:54:35 +08:00
|
|
|
TEST_F(ExpressionPrettyPrinterTest, UnaryOperators) {
|
|
|
|
// not(false)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(NOT(LITERAL(false))), "(Not false)");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// +1
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(UPLUS(LITERAL(1))), "(+ 1)");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// -1
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(UMINUS(LITERAL(1))), "(- 1)");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// null IS NULL
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(IS_NULL(LITERAL(TypedValue::Null))), "(IsNull null)");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, BinaryOperators) {
|
|
|
|
// and(null, 5)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(AND(LITERAL(TypedValue::Null), LITERAL(5))),
|
2018-11-07 17:54:35 +08:00
|
|
|
"(And null 5)");
|
|
|
|
|
|
|
|
// or(5, {hello: "there"}["hello"])
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(OR(LITERAL(5),
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
PROPERTY_LOOKUP(
|
|
|
|
MAP(std::make_pair(storage.GetPropertyIx("hello"),
|
|
|
|
LITERAL("there"))),
|
|
|
|
"hello"))),
|
|
|
|
"(Or 5 (PropertyLookup {\"hello\": \"there\"} \"hello\"))");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// and(coalesce(null, 1), {hello: "there"})
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(AND(COALESCE(LITERAL(TypedValue::Null), LITERAL(1)),
|
|
|
|
MAP(std::make_pair(storage.GetPropertyIx("hello"),
|
|
|
|
LITERAL("there"))))),
|
|
|
|
"(And (Coalesce [null, 1]) {\"hello\": \"there\"})");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, Coalesce) {
|
|
|
|
// coalesce()
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE()), "(Coalesce [])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, null)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(
|
|
|
|
ToString(COALESCE(LITERAL(TypedValue::Null), LITERAL(TypedValue::Null))),
|
|
|
|
"(Coalesce [null, null])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, 2, 3)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(
|
|
|
|
ToString(COALESCE(LITERAL(TypedValue::Null), LITERAL(2), LITERAL(3))),
|
|
|
|
"(Coalesce [null, 2, 3])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, 2, assert(false), 3)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue::Null), LITERAL(2),
|
|
|
|
FN("ASSERT", LITERAL(false)), LITERAL(3))),
|
|
|
|
"(Coalesce [null, 2, (Function \"ASSERT\" [false]), 3])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, assert(false))
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue::Null),
|
|
|
|
FN("ASSERT", LITERAL(false)))),
|
2018-11-07 17:54:35 +08:00
|
|
|
"(Coalesce [null, (Function \"ASSERT\" [false])])");
|
|
|
|
|
|
|
|
// coalesce([null, null])
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue(
|
|
|
|
std::vector<TypedValue>{TypedValue::Null, TypedValue::Null})))),
|
|
|
|
"(Coalesce [[null, null]])");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, ParameterLookup) {
|
|
|
|
// and($hello, $there)
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(AND(PARAMETER_LOOKUP(1), PARAMETER_LOOKUP(2))),
|
2018-11-07 17:54:35 +08:00
|
|
|
"(And (ParameterLookup 1) (ParameterLookup 2))");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, PropertyLookup) {
|
|
|
|
// {hello: "there"}["hello"]
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
EXPECT_EQ(
|
2019-01-16 17:43:32 +08:00
|
|
|
ToString(PROPERTY_LOOKUP(
|
|
|
|
MAP(std::make_pair(storage.GetPropertyIx("hello"), LITERAL("there"))),
|
|
|
|
"hello")),
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
"(PropertyLookup {\"hello\": \"there\"} \"hello\")");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ExpressionPrettyPrinterTest, NamedExpression) {
|
|
|
|
// n AS 1
|
2019-01-16 17:43:32 +08:00
|
|
|
EXPECT_EQ(ToString(NEXPR("n", LITERAL(1))), "(NamedExpression \"n\" 1)");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|