2024-01-06 00:42:54 +08:00
|
|
|
// Copyright 2024 Memgraph Ltd.
|
2021-10-26 14:53:56 +08:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2023-05-17 02:05:35 +08:00
|
|
|
#include <cstddef>
|
2018-11-07 17:54:35 +08:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "disk_test_utils.hpp"
|
2018-11-07 17:54:35 +08:00
|
|
|
#include "query/frontend/ast/ast.hpp"
|
|
|
|
#include "query/frontend/ast/pretty_print.hpp"
|
|
|
|
#include "query_common.hpp"
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "storage/v2/disk/storage.hpp"
|
|
|
|
#include "storage/v2/inmemory/storage.hpp"
|
2018-11-07 17:54:35 +08:00
|
|
|
#include "utils/string.hpp"
|
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
using namespace memgraph::query;
|
|
|
|
using memgraph::query::test_common::ToString;
|
2018-11-07 17:54:35 +08:00
|
|
|
using testing::ElementsAre;
|
|
|
|
using testing::UnorderedElementsAre;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
template <typename StorageType>
|
|
|
|
class ExpressionPrettyPrinterTest : public ::testing::Test {
|
|
|
|
public:
|
|
|
|
const std::string testSuite = "query_pretty_print";
|
|
|
|
memgraph::storage::Config config = disk_test_utils::GenerateOnDiskConfig(testSuite);
|
|
|
|
std::unique_ptr<memgraph::storage::Storage> db{new StorageType(config)};
|
2024-01-06 00:42:54 +08:00
|
|
|
std::unique_ptr<memgraph::storage::Storage::Accessor> storage_dba{
|
|
|
|
db->Access(memgraph::replication::ReplicationRole::MAIN)};
|
2023-06-29 17:44:55 +08:00
|
|
|
memgraph::query::DbAccessor dba{storage_dba.get()};
|
2018-11-07 17:54:35 +08:00
|
|
|
AstStorage storage;
|
2023-06-29 17:44:55 +08:00
|
|
|
|
|
|
|
void TearDown() override {
|
|
|
|
if (std::is_same<StorageType, memgraph::storage::DiskStorage>::value) {
|
|
|
|
disk_test_utils::RemoveRocksDbDirs(testSuite);
|
|
|
|
}
|
|
|
|
}
|
2018-11-07 17:54:35 +08:00
|
|
|
};
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
using StorageTypes = ::testing::Types<memgraph::storage::InMemoryStorage, memgraph::storage::DiskStorage>;
|
|
|
|
TYPED_TEST_CASE(ExpressionPrettyPrinterTest, StorageTypes);
|
|
|
|
|
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, Literals) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// 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-06-04 22:11:31 +08:00
|
|
|
EXPECT_EQ(ToString(LITERAL(TypedValue())), "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"]
|
2022-02-22 20:33:45 +08:00
|
|
|
std::vector<memgraph::storage::PropertyValue> values{memgraph::storage::PropertyValue(1),
|
|
|
|
memgraph::storage::PropertyValue(),
|
|
|
|
memgraph::storage::PropertyValue("hello")};
|
|
|
|
EXPECT_EQ(ToString(LITERAL(memgraph::storage::PropertyValue(values))), "[1, null, \"hello\"]");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// {hello: 1, there: 2}
|
2022-02-22 20:33:45 +08:00
|
|
|
std::map<std::string, memgraph::storage::PropertyValue> map{{"hello", memgraph::storage::PropertyValue(1)},
|
|
|
|
{"there", memgraph::storage::PropertyValue(2)}};
|
|
|
|
EXPECT_EQ(ToString(LITERAL(memgraph::storage::PropertyValue(map))), "{\"hello\": 1, \"there\": 2}");
|
|
|
|
|
|
|
|
std::vector<memgraph::storage::PropertyValue> tt_vec{
|
|
|
|
memgraph::storage::PropertyValue(memgraph::storage::TemporalData(memgraph::storage::TemporalType::Duration, 1)),
|
|
|
|
memgraph::storage::PropertyValue(memgraph::storage::TemporalData(memgraph::storage::TemporalType::Duration, -2)),
|
|
|
|
memgraph::storage::PropertyValue(memgraph::storage::TemporalData(memgraph::storage::TemporalType::LocalTime, 2)),
|
|
|
|
memgraph::storage::PropertyValue(
|
|
|
|
memgraph::storage::TemporalData(memgraph::storage::TemporalType::LocalDateTime, 3)),
|
|
|
|
memgraph::storage::PropertyValue(memgraph::storage::TemporalData(memgraph::storage::TemporalType::Date, 4))};
|
|
|
|
EXPECT_EQ(ToString(LITERAL(memgraph::storage::PropertyValue(tt_vec))),
|
2021-09-20 22:53:42 +08:00
|
|
|
"[DURATION(\"P0DT0H0M0.000001S\"), DURATION(\"P0DT0H0M-0.000002S\"), LOCALTIME(\"00:00:00.000002\"), "
|
2021-09-16 20:45:35 +08:00
|
|
|
"LOCALDATETIME(\"1970-01-01T00:00:00.000003\"), DATE(\"1970-01-01\")]");
|
2023-05-17 02:05:35 +08:00
|
|
|
|
|
|
|
// map {literalEntry: 10, variableSelector: a, .map, .*}
|
|
|
|
auto elements = std::unordered_map<memgraph::query::PropertyIx, memgraph::query::Expression *>{
|
2023-06-29 17:44:55 +08:00
|
|
|
{this->storage.GetPropertyIx("literalEntry"), LITERAL(10)},
|
|
|
|
{this->storage.GetPropertyIx("variableSelector"), IDENT("a")},
|
|
|
|
{this->storage.GetPropertyIx("propertySelector"),
|
|
|
|
PROPERTY_LOOKUP(this->dba, "map", PROPERTY_PAIR(this->dba, "hello"))},
|
|
|
|
{this->storage.GetPropertyIx("allPropertiesSelector"), ALL_PROPERTIES_LOOKUP("map")}};
|
2023-05-17 02:05:35 +08:00
|
|
|
EXPECT_EQ(ToString(MAP_PROJECTION(IDENT("map"), elements)),
|
|
|
|
"(Identifier \"map\"){\"allPropertiesSelector\": .*, \"literalEntry\": 10, \"propertySelector\": "
|
|
|
|
"(PropertyLookup (Identifier \"map\") \"hello\"), \"variableSelector\": (Identifier \"a\")}");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, Identifiers) {
|
2019-01-16 17:43:32 +08:00
|
|
|
// x
|
|
|
|
EXPECT_EQ(ToString(IDENT("x")), "(Identifier \"x\")");
|
|
|
|
|
|
|
|
// hello_there
|
|
|
|
EXPECT_EQ(ToString(IDENT("hello_there")), "(Identifier \"hello_there\")");
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, Reducing) {
|
2019-01-16 17:43:32 +08:00
|
|
|
// all(x in list where x.prop = 42)
|
2023-06-29 17:44:55 +08:00
|
|
|
auto prop = this->dba.NameToProperty("prop");
|
2022-02-22 20:33:45 +08:00
|
|
|
EXPECT_EQ(ToString(ALL("x", LITERAL(std::vector<memgraph::storage::PropertyValue>{}),
|
2023-06-29 17:44:55 +08:00
|
|
|
WHERE(EQ(PROPERTY_LOOKUP(this->dba, "x", prop), LITERAL(42))))),
|
2019-01-16 17:43:32 +08:00
|
|
|
"(All (Identifier \"x\") [] (== (PropertyLookup "
|
|
|
|
"(Identifier \"x\") \"prop\") 42))");
|
|
|
|
|
|
|
|
// reduce(accumulator = initial_value, variable IN list | expression)
|
2021-02-18 22:32:43 +08:00
|
|
|
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\"))");
|
2019-01-16 17:43:32 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, UnaryOperators) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// 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-06-04 22:11:31 +08:00
|
|
|
EXPECT_EQ(ToString(IS_NULL(LITERAL(TypedValue()))), "(IsNull null)");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, BinaryOperators) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// and(null, 5)
|
2021-01-21 22:47:56 +08:00
|
|
|
EXPECT_EQ(ToString(AND(LITERAL(TypedValue()), LITERAL(5))), "(And null 5)");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// or(5, {hello: "there"}["hello"])
|
2023-06-29 17:44:55 +08:00
|
|
|
EXPECT_EQ(ToString(OR(
|
|
|
|
LITERAL(5),
|
|
|
|
PROPERTY_LOOKUP(this->dba, MAP(std::make_pair(this->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-06-04 22:11:31 +08:00
|
|
|
EXPECT_EQ(ToString(AND(COALESCE(LITERAL(TypedValue()), LITERAL(1)),
|
2023-06-29 17:44:55 +08:00
|
|
|
MAP(std::make_pair(this->storage.GetPropertyIx("hello"), LITERAL("there"))))),
|
2019-01-16 17:43:32 +08:00
|
|
|
"(And (Coalesce [null, 1]) {\"hello\": \"there\"})");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, Coalesce) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// 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)
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue()), LITERAL(TypedValue()))), "(Coalesce [null, null])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, 2, 3)
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue()), LITERAL(2), LITERAL(3))), "(Coalesce [null, 2, 3])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, 2, assert(false), 3)
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue()), LITERAL(2), FN("ASSERT", LITERAL(false)), LITERAL(3))),
|
2019-01-16 17:43:32 +08:00
|
|
|
"(Coalesce [null, 2, (Function \"ASSERT\" [false]), 3])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce(null, assert(false))
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue()), FN("ASSERT", LITERAL(false)))),
|
|
|
|
"(Coalesce [null, (Function \"ASSERT\" [false])])");
|
2018-11-07 17:54:35 +08:00
|
|
|
|
|
|
|
// coalesce([null, null])
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(COALESCE(LITERAL(TypedValue(std::vector<TypedValue>{TypedValue(), TypedValue()})))),
|
2019-01-16 17:43:32 +08:00
|
|
|
"(Coalesce [[null, null]])");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, ParameterLookup) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// and($hello, $there)
|
2021-02-18 22:32:43 +08:00
|
|
|
EXPECT_EQ(ToString(AND(PARAMETER_LOOKUP(1), PARAMETER_LOOKUP(2))), "(And (ParameterLookup 1) (ParameterLookup 2))");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, PropertyLookup) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// {hello: "there"}["hello"]
|
2023-06-29 17:44:55 +08:00
|
|
|
EXPECT_EQ(ToString(PROPERTY_LOOKUP(
|
|
|
|
this->dba, MAP(std::make_pair(this->storage.GetPropertyIx("hello"), LITERAL("there"))), "hello")),
|
2021-02-18 22:32:43 +08:00
|
|
|
"(PropertyLookup {\"hello\": \"there\"} \"hello\")");
|
2018-11-07 17:54:35 +08:00
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
TYPED_TEST(ExpressionPrettyPrinterTest, NamedExpression) {
|
2018-11-07 17:54:35 +08:00
|
|
|
// 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
|