2018-12-12 00:13:28 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <climits>
|
|
|
|
#include <limits>
|
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <antlr4-runtime.h>
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2019-03-15 22:58:16 +08:00
|
|
|
#include "query/distributed/frontend/ast/ast_serialization.hpp"
|
2019-05-02 20:24:11 +08:00
|
|
|
#include "query/frontend/ast/ast.hpp"
|
2018-12-12 00:13:28 +08:00
|
|
|
#include "query/frontend/ast/cypher_main_visitor.hpp"
|
|
|
|
#include "query/frontend/opencypher/parser.hpp"
|
|
|
|
#include "query/frontend/stripped.hpp"
|
|
|
|
#include "query/typed_value.hpp"
|
2019-05-02 20:24:11 +08:00
|
|
|
#include "slk/serialization.hpp"
|
|
|
|
|
|
|
|
#include "slk_common.hpp"
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using namespace query;
|
|
|
|
using namespace query::frontend;
|
|
|
|
using query::TypedValue;
|
|
|
|
using testing::ElementsAre;
|
|
|
|
using testing::Pair;
|
|
|
|
using testing::UnorderedElementsAre;
|
|
|
|
|
|
|
|
// Base class for all test types
|
|
|
|
class Base {
|
|
|
|
public:
|
|
|
|
ParsingContext context_;
|
|
|
|
Parameters parameters_;
|
2019-02-25 21:16:32 +08:00
|
|
|
|
|
|
|
virtual ~Base() {}
|
|
|
|
|
|
|
|
virtual Query *ParseQuery(const std::string &query_string) = 0;
|
2018-12-12 00:13:28 +08:00
|
|
|
|
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
|
|
|
virtual PropertyIx Prop(const std::string &prop_name) = 0;
|
2018-12-12 00:13:28 +08:00
|
|
|
|
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
|
|
|
virtual LabelIx Label(const std::string &label_name) = 0;
|
|
|
|
|
|
|
|
virtual EdgeTypeIx EdgeType(const std::string &edge_type_name) = 0;
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
TypedValue LiteralValue(Expression *expression) {
|
|
|
|
if (context_.is_query_cached) {
|
|
|
|
auto *param_lookup = dynamic_cast<ParameterLookup *>(expression);
|
|
|
|
return parameters_.AtTokenPosition(param_lookup->token_position_);
|
|
|
|
} else {
|
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(expression);
|
|
|
|
return literal->value_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckLiteral(Expression *expression, const TypedValue &expected,
|
2019-04-23 17:00:49 +08:00
|
|
|
const std::optional<int> &token_position = std::nullopt) {
|
2018-12-12 00:13:28 +08:00
|
|
|
TypedValue value;
|
|
|
|
if (!expected.IsNull() && context_.is_query_cached) {
|
|
|
|
auto *param_lookup = dynamic_cast<ParameterLookup *>(expression);
|
|
|
|
ASSERT_TRUE(param_lookup);
|
|
|
|
if (token_position)
|
|
|
|
EXPECT_EQ(param_lookup->token_position_, *token_position);
|
|
|
|
value = parameters_.AtTokenPosition(param_lookup->token_position_);
|
|
|
|
} else {
|
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(expression);
|
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
if (token_position) ASSERT_EQ(literal->token_position_, *token_position);
|
|
|
|
value = literal->value_;
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(TypedValue::BoolEqual{}(value, expected));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class SlkAstGenerator : public Base {
|
|
|
|
public:
|
2019-02-25 21:16:32 +08:00
|
|
|
Query *ParseQuery(const std::string &query_string) override {
|
|
|
|
::frontend::opencypher::Parser parser(query_string);
|
2018-12-12 00:13:28 +08:00
|
|
|
AstStorage tmp_storage;
|
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
|
|
|
CypherMainVisitor visitor(context_, &tmp_storage);
|
2018-12-12 00:13:28 +08:00
|
|
|
visitor.visit(parser.tree());
|
|
|
|
|
2019-05-02 20:24:11 +08:00
|
|
|
slk::Loopback loopback;
|
|
|
|
auto builder = loopback.GetBuilder();
|
|
|
|
{ SaveAstPointer(visitor.query(), builder); }
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
{
|
2019-05-02 20:24:11 +08:00
|
|
|
auto reader = loopback.GetReader();
|
|
|
|
return LoadAstPointer<Query>(&storage_, reader);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
PropertyIx Prop(const std::string &prop_name) override {
|
|
|
|
return storage_.GetPropertyIx(prop_name);
|
|
|
|
}
|
2018-12-12 00:13:28 +08:00
|
|
|
|
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
|
|
|
LabelIx Label(const std::string &name) override {
|
|
|
|
return storage_.GetLabelIx(name);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
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
|
|
|
EdgeTypeIx EdgeType(const std::string &name) override {
|
|
|
|
return storage_.GetEdgeTypeIx(name);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
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
|
|
|
|
|
|
|
AstStorage storage_;
|
2018-12-12 00:13:28 +08:00
|
|
|
};
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
class CypherMainVisitorTest
|
|
|
|
: public ::testing::TestWithParam<std::shared_ptr<Base>> {};
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
std::shared_ptr<Base> gAstGeneratorTypes[] = {
|
|
|
|
std::make_shared<SlkAstGenerator>(),
|
|
|
|
};
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(AstGeneratorTypes, CypherMainVisitorTest,
|
|
|
|
::testing::ValuesIn(gAstGeneratorTypes));
|
|
|
|
|
|
|
|
// NOTE: The above used to use *Typed Tests* functionality of gtest library.
|
|
|
|
// Unfortunately, the compilation time of this test increased to about 1 minute!
|
|
|
|
// Although using Typed Tests is the recommended way to achieve what we want, we
|
|
|
|
// are (ab)using *Value-Parameterized Tests* functionality instead. This cuts
|
|
|
|
// down the compilation time to about 20 seconds. The original code is here for
|
|
|
|
// future reference in case someone gets the idea to change to *appropriate*
|
|
|
|
// Typed Tests mechanism and ruin the compilation times.
|
|
|
|
//
|
2019-05-03 02:53:54 +08:00
|
|
|
// typedef ::testing::Types<SlkAstGenerator>
|
2019-02-25 21:16:32 +08:00
|
|
|
// AstGeneratorTypes;
|
|
|
|
//
|
|
|
|
// TYPED_TEST_CASE(CypherMainVisitorTest, AstGeneratorTypes);
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, SyntaxException) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE ()-[*1....2]-()"),
|
|
|
|
SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, SyntaxExceptionOnTrailingText) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 2 + 2 mirko"), SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, PropertyLookup) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN n.x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *property_lookup = dynamic_cast<PropertyLookup *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(property_lookup->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(property_lookup->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
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
|
|
|
ASSERT_EQ(property_lookup->property_, ast_generator.Prop("x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, LabelsTest) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN n:x:y"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *labels_test = dynamic_cast<LabelsTest *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(labels_test->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(labels_test->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
ASSERT_THAT(labels_test->labels_,
|
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
|
|
|
ElementsAre(ast_generator.Label("x"), ast_generator.Label("y")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, EscapedLabel) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN n:`l-$\"'ab``e````l`"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *labels_test = dynamic_cast<LabelsTest *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(labels_test->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
ASSERT_THAT(labels_test->labels_,
|
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
|
|
|
ElementsAre(ast_generator.Label("l-$\"'ab`e``l")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, KeywordLabel) {
|
2018-12-12 00:13:28 +08:00
|
|
|
for (const auto &label : {"DeLete", "UsER"}) {
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery(fmt::format("RETURN n:{}", label)));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *labels_test = dynamic_cast<LabelsTest *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(labels_test->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
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
|
|
|
ASSERT_THAT(labels_test->labels_, ElementsAre(ast_generator.Label(label)));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, HexLetterLabel) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN n:a"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *labels_test = dynamic_cast<LabelsTest *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(labels_test->expression_);
|
|
|
|
EXPECT_EQ(identifier->name_, "n");
|
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
|
|
|
ASSERT_THAT(labels_test->labels_, ElementsAre(ast_generator.Label("a")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnNoDistinctNoBagSemantics) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnDistinct) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN DISTINCT x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.distinct);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnLimit) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN x LIMIT 5"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.limit);
|
|
|
|
ast_generator.CheckLiteral(return_clause->body_.limit, 5);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnSkip) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN x SKIP 5"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.skip);
|
|
|
|
ast_generator.CheckLiteral(return_clause->body_.skip, 5);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnOrderBy) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN x, y, z ORDER BY z ASC, x, y DESC"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 3U);
|
|
|
|
std::vector<std::pair<Ordering, std::string>> ordering;
|
|
|
|
for (const auto &sort_item : return_clause->body_.order_by) {
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(sort_item.expression);
|
|
|
|
ordering.emplace_back(sort_item.ordering, identifier->name_);
|
|
|
|
}
|
|
|
|
ASSERT_THAT(ordering, UnorderedElementsAre(Pair(Ordering::ASC, "z"),
|
|
|
|
Pair(Ordering::ASC, "x"),
|
|
|
|
Pair(Ordering::DESC, "y")));
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnNamedIdentifier) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN var AS var5"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
auto *named_expr = return_clause->body_.named_expressions[0];
|
|
|
|
ASSERT_EQ(named_expr->name_, "var5");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "var");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnAsterisk) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN *"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 0U);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, IntegerLiteral) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 42"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, 42, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, IntegerLiteralTooLarge) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 10000000000000000000000000"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, BooleanLiteralTrue) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN TrUe"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, true, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, BooleanLiteralFalse) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN faLSE"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, false, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, NullLiteral) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN nULl"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, TypedValue::Null,
|
|
|
|
1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ParenthesizedExpression) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN (2)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, OrOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN true Or false oR n"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *or_operator2 = dynamic_cast<OrOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(or_operator2);
|
|
|
|
auto *or_operator1 = dynamic_cast<OrOperator *>(or_operator2->expression1_);
|
|
|
|
ASSERT_TRUE(or_operator1);
|
|
|
|
ast_generator.CheckLiteral(or_operator1->expression1_, true);
|
|
|
|
ast_generator.CheckLiteral(or_operator1->expression2_, false);
|
|
|
|
auto *operand3 = dynamic_cast<Identifier *>(or_operator2->expression2_);
|
|
|
|
ASSERT_TRUE(operand3);
|
|
|
|
ASSERT_EQ(operand3->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, XorOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN true xOr false"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *xor_operator = dynamic_cast<XorOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(xor_operator->expression1_, true);
|
|
|
|
ast_generator.CheckLiteral(xor_operator->expression2_, false);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, AndOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN true and false"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *and_operator = dynamic_cast<AndOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(and_operator->expression1_, true);
|
|
|
|
ast_generator.CheckLiteral(and_operator->expression2_, false);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, AdditionSubtractionOperators) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 1 - 2 + 3"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *addition_operator = dynamic_cast<AdditionOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(addition_operator);
|
|
|
|
auto *subtraction_operator =
|
|
|
|
dynamic_cast<SubtractionOperator *>(addition_operator->expression1_);
|
|
|
|
ASSERT_TRUE(subtraction_operator);
|
|
|
|
ast_generator.CheckLiteral(subtraction_operator->expression1_, 1);
|
|
|
|
ast_generator.CheckLiteral(subtraction_operator->expression2_, 2);
|
|
|
|
ast_generator.CheckLiteral(addition_operator->expression2_, 3);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MulitplicationOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 2 * 3"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *mult_operator = dynamic_cast<MultiplicationOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(mult_operator->expression1_, 2);
|
|
|
|
ast_generator.CheckLiteral(mult_operator->expression2_, 3);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DivisionOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 2 / 3"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *div_operator = dynamic_cast<DivisionOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(div_operator->expression1_, 2);
|
|
|
|
ast_generator.CheckLiteral(div_operator->expression2_, 3);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ModOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 2 % 3"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *mod_operator = dynamic_cast<ModOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(mod_operator->expression1_, 2);
|
|
|
|
ast_generator.CheckLiteral(mod_operator->expression2_, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CHECK_COMPARISON(TYPE, VALUE1, VALUE2) \
|
|
|
|
do { \
|
|
|
|
auto *and_operator = dynamic_cast<AndOperator *>(_operator); \
|
|
|
|
ASSERT_TRUE(and_operator); \
|
|
|
|
_operator = and_operator->expression1_; \
|
|
|
|
auto *cmp_operator = dynamic_cast<TYPE *>(and_operator->expression2_); \
|
|
|
|
ASSERT_TRUE(cmp_operator); \
|
|
|
|
ast_generator.CheckLiteral(cmp_operator->expression1_, VALUE1); \
|
|
|
|
ast_generator.CheckLiteral(cmp_operator->expression2_, VALUE2); \
|
|
|
|
} while (0)
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ComparisonOperators) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN 2 = 3 != 4 <> 5 < 6 > 7 <= 8 >= 9"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
Expression *_operator =
|
|
|
|
return_clause->body_.named_expressions[0]->expression_;
|
|
|
|
CHECK_COMPARISON(GreaterEqualOperator, 8, 9);
|
|
|
|
CHECK_COMPARISON(LessEqualOperator, 7, 8);
|
|
|
|
CHECK_COMPARISON(GreaterOperator, 6, 7);
|
|
|
|
CHECK_COMPARISON(LessOperator, 5, 6);
|
|
|
|
CHECK_COMPARISON(NotEqualOperator, 4, 5);
|
|
|
|
CHECK_COMPARISON(NotEqualOperator, 3, 4);
|
|
|
|
auto *cmp_operator = dynamic_cast<EqualOperator *>(_operator);
|
|
|
|
ASSERT_TRUE(cmp_operator);
|
|
|
|
ast_generator.CheckLiteral(cmp_operator->expression1_, 2);
|
|
|
|
ast_generator.CheckLiteral(cmp_operator->expression2_, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef CHECK_COMPARISON
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ListIndexing) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN [1,2,3] [ 2 ]"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *list_index_op = dynamic_cast<SubscriptOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(list_index_op);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(list_index_op->expression1_);
|
|
|
|
EXPECT_TRUE(list);
|
|
|
|
ast_generator.CheckLiteral(list_index_op->expression2_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ListSlicingOperatorNoBounds) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN [1,2,3] [ .. ]"),
|
|
|
|
SemanticException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ListSlicingOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN [1,2,3] [ .. 2 ]"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *list_slicing_op = dynamic_cast<ListSlicingOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(list_slicing_op);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(list_slicing_op->list_);
|
|
|
|
EXPECT_TRUE(list);
|
|
|
|
EXPECT_FALSE(list_slicing_op->lower_bound_);
|
|
|
|
ast_generator.CheckLiteral(list_slicing_op->upper_bound_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, InListOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN 5 IN [1,2]"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *in_list_operator = dynamic_cast<InListOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(in_list_operator);
|
|
|
|
ast_generator.CheckLiteral(in_list_operator->expression1_, 5);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(in_list_operator->expression2_);
|
|
|
|
ASSERT_TRUE(list);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, InWithListIndexing) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN 1 IN [[1,2]][0]"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *in_list_operator = dynamic_cast<InListOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(in_list_operator);
|
|
|
|
ast_generator.CheckLiteral(in_list_operator->expression1_, 1);
|
|
|
|
auto *list_indexing =
|
|
|
|
dynamic_cast<SubscriptOperator *>(in_list_operator->expression2_);
|
|
|
|
ASSERT_TRUE(list_indexing);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(list_indexing->expression1_);
|
|
|
|
EXPECT_TRUE(list);
|
|
|
|
ast_generator.CheckLiteral(list_indexing->expression2_, 0);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, CaseGenericForm) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN CASE WHEN n < 10 THEN 1 WHEN n > 10 THEN 2 END"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *if_operator = dynamic_cast<IfOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(if_operator);
|
|
|
|
auto *condition = dynamic_cast<LessOperator *>(if_operator->condition_);
|
|
|
|
ASSERT_TRUE(condition);
|
|
|
|
ast_generator.CheckLiteral(if_operator->then_expression_, 1);
|
|
|
|
|
|
|
|
auto *if_operator2 =
|
|
|
|
dynamic_cast<IfOperator *>(if_operator->else_expression_);
|
|
|
|
ASSERT_TRUE(if_operator2);
|
|
|
|
auto *condition2 = dynamic_cast<GreaterOperator *>(if_operator2->condition_);
|
|
|
|
ASSERT_TRUE(condition2);
|
|
|
|
ast_generator.CheckLiteral(if_operator2->then_expression_, 2);
|
|
|
|
ast_generator.CheckLiteral(if_operator2->else_expression_, TypedValue::Null);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, CaseGenericFormElse) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN CASE WHEN n < 10 THEN 1 ELSE 2 END"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *if_operator = dynamic_cast<IfOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto *condition = dynamic_cast<LessOperator *>(if_operator->condition_);
|
|
|
|
ASSERT_TRUE(condition);
|
|
|
|
ast_generator.CheckLiteral(if_operator->then_expression_, 1);
|
|
|
|
ast_generator.CheckLiteral(if_operator->else_expression_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, CaseSimpleForm) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN CASE 5 WHEN 10 THEN 1 END"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *if_operator = dynamic_cast<IfOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto *condition = dynamic_cast<EqualOperator *>(if_operator->condition_);
|
|
|
|
ASSERT_TRUE(condition);
|
|
|
|
ast_generator.CheckLiteral(condition->expression1_, 5);
|
|
|
|
ast_generator.CheckLiteral(condition->expression2_, 10);
|
|
|
|
ast_generator.CheckLiteral(if_operator->then_expression_, 1);
|
|
|
|
ast_generator.CheckLiteral(if_operator->else_expression_, TypedValue::Null);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, IsNull) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 2 iS NulL"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *is_type_operator = dynamic_cast<IsNullOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(is_type_operator->expression_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, IsNotNull) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN 2 iS nOT NulL"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *not_operator = dynamic_cast<NotOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
auto *is_type_operator =
|
|
|
|
dynamic_cast<IsNullOperator *>(not_operator->expression_);
|
|
|
|
ast_generator.CheckLiteral(is_type_operator->expression_, 2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, NotOperator) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN not true"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *not_operator = dynamic_cast<NotOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ast_generator.CheckLiteral(not_operator->expression_, true);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, UnaryMinusPlusOperators) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN -+5"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *unary_minus_operator = dynamic_cast<UnaryMinusOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(unary_minus_operator);
|
|
|
|
auto *unary_plus_operator =
|
|
|
|
dynamic_cast<UnaryPlusOperator *>(unary_minus_operator->expression_);
|
|
|
|
ASSERT_TRUE(unary_plus_operator);
|
|
|
|
ast_generator.CheckLiteral(unary_plus_operator->expression_, 5);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Aggregation) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN COUNT(a), MIN(b), MAX(c), SUM(d), AVG(e), COLLECT(f), COUNT(*)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 7U);
|
|
|
|
Aggregation::Op ops[] = {
|
|
|
|
Aggregation::Op::COUNT, Aggregation::Op::MIN,
|
|
|
|
Aggregation::Op::MAX, Aggregation::Op::SUM,
|
|
|
|
Aggregation::Op::AVG, Aggregation::Op::COLLECT_LIST};
|
|
|
|
std::string ids[] = {"a", "b", "c", "d", "e", "f"};
|
|
|
|
for (int i = 0; i < 6; ++i) {
|
|
|
|
auto *aggregation = dynamic_cast<Aggregation *>(
|
|
|
|
return_clause->body_.named_expressions[i]->expression_);
|
|
|
|
ASSERT_TRUE(aggregation);
|
|
|
|
ASSERT_EQ(aggregation->op_, ops[i]);
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(aggregation->expression1_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, ids[i]);
|
|
|
|
}
|
|
|
|
auto *aggregation = dynamic_cast<Aggregation *>(
|
|
|
|
return_clause->body_.named_expressions[6]->expression_);
|
|
|
|
ASSERT_TRUE(aggregation);
|
|
|
|
ASSERT_EQ(aggregation->op_, Aggregation::Op::COUNT);
|
|
|
|
ASSERT_FALSE(aggregation->expression1_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, UndefinedFunction) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery(
|
|
|
|
"RETURN "
|
|
|
|
"IHopeWeWillNeverHaveAwesomeMemgraphProcedureWithS"
|
|
|
|
"uchALongAndAwesomeNameSinceThisTestWouldFail(1)"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Function) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN abs(n, 2)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1);
|
|
|
|
auto *function = dynamic_cast<Function *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(function);
|
|
|
|
ASSERT_TRUE(function->function_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralDoubleQuotes) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN \"mi'rko\""));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, "mi'rko", 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralSingleQuotes) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 'mi\"rko'"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, "mi\"rko", 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralEscapedChars) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN '\\\\\\'\\\"\\b\\B\\f\\F\\n\\N\\r\\R\\t\\T'"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_,
|
|
|
|
"\\'\"\b\b\f\f\n\n\r\r\t\t", 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralEscapedUtf16) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN '\\u221daaa\\u221daaa'"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_,
|
|
|
|
u8"\u221daaa\u221daaa", 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralEscapedUtf16Error) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN '\\U221daaa'"),
|
|
|
|
SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StringLiteralEscapedUtf32) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN '\\U0001F600aaaa\\U0001F600aaaaaaaa'"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_,
|
|
|
|
u8"\U0001F600aaaa\U0001F600aaaaaaaa", 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DoubleLiteral) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 3.5"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, 3.5, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DoubleLiteralExponent) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN 5e-1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ast_generator.CheckLiteral(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_, 0.5, 1);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ListLiteral) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN [3, [], 'johhny']"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *list_literal = dynamic_cast<ListLiteral *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(list_literal);
|
|
|
|
ASSERT_EQ(3, list_literal->elements_.size());
|
|
|
|
ast_generator.CheckLiteral(list_literal->elements_[0], 3);
|
|
|
|
auto *elem_1 = dynamic_cast<ListLiteral *>(list_literal->elements_[1]);
|
|
|
|
ASSERT_TRUE(elem_1);
|
|
|
|
EXPECT_EQ(0, elem_1->elements_.size());
|
|
|
|
ast_generator.CheckLiteral(list_literal->elements_[2], "johhny");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MapLiteral) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN {a: 1, b: 'bla', c: [1, {a: 42}]}"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
auto *map_literal = dynamic_cast<MapLiteral *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(map_literal);
|
|
|
|
ASSERT_EQ(3, map_literal->elements_.size());
|
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
|
|
|
ast_generator.CheckLiteral(map_literal->elements_[ast_generator.Prop("a")],
|
|
|
|
1);
|
|
|
|
ast_generator.CheckLiteral(map_literal->elements_[ast_generator.Prop("b")],
|
|
|
|
"bla");
|
2018-12-12 00:13:28 +08:00
|
|
|
auto *elem_2 = dynamic_cast<ListLiteral *>(
|
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
|
|
|
map_literal->elements_[ast_generator.Prop("c")]);
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(elem_2);
|
|
|
|
EXPECT_EQ(2, elem_2->elements_.size());
|
|
|
|
auto *elem_2_1 = dynamic_cast<MapLiteral *>(elem_2->elements_[1]);
|
|
|
|
ASSERT_TRUE(elem_2_1);
|
|
|
|
EXPECT_EQ(1, elem_2_1->elements_.size());
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, NodePattern) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"MATCH (:label1:label2:label3 {a : 5, b : 10}) RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_TRUE(match->patterns_[0]);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 1U);
|
|
|
|
auto node = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[0]);
|
|
|
|
ASSERT_TRUE(node);
|
|
|
|
ASSERT_TRUE(node->identifier_);
|
|
|
|
EXPECT_EQ(node->identifier_->name_,
|
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(1));
|
|
|
|
EXPECT_FALSE(node->identifier_->user_declared_);
|
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_THAT(node->labels_,
|
|
|
|
UnorderedElementsAre(ast_generator.Label("label1"),
|
|
|
|
ast_generator.Label("label2"),
|
|
|
|
ast_generator.Label("label3")));
|
|
|
|
std::unordered_map<PropertyIx, int64_t> properties;
|
2018-12-12 00:13:28 +08:00
|
|
|
for (auto x : node->properties_) {
|
|
|
|
TypedValue value = ast_generator.LiteralValue(x.second);
|
|
|
|
ASSERT_TRUE(value.type() == TypedValue::Type::Int);
|
|
|
|
properties[x.first] = value.Value<int64_t>();
|
|
|
|
}
|
|
|
|
EXPECT_THAT(properties,
|
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
|
|
|
UnorderedElementsAre(Pair(ast_generator.Prop("a"), 5),
|
|
|
|
Pair(ast_generator.Prop("b"), 10)));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, PropertyMapSameKeyAppearsTwice) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("MATCH ({a : 1, a : 2})"),
|
|
|
|
SemanticException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, NodePatternIdentifier) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH (var) RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
auto node = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[0]);
|
|
|
|
ASSERT_TRUE(node);
|
|
|
|
ASSERT_TRUE(node->identifier_);
|
|
|
|
EXPECT_EQ(node->identifier_->name_, "var");
|
|
|
|
EXPECT_TRUE(node->identifier_->user_declared_);
|
|
|
|
EXPECT_THAT(node->labels_, UnorderedElementsAre());
|
|
|
|
EXPECT_THAT(node->properties_, UnorderedElementsAre());
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternNoDetails) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()--() RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_TRUE(match->patterns_[0]);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *node1 = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[0]);
|
|
|
|
ASSERT_TRUE(node1);
|
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(edge);
|
|
|
|
auto *node2 = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[2]);
|
|
|
|
ASSERT_TRUE(node2);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::BOTH);
|
|
|
|
ASSERT_TRUE(edge->identifier_);
|
|
|
|
EXPECT_THAT(edge->identifier_->name_,
|
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(2));
|
|
|
|
EXPECT_FALSE(edge->identifier_->user_declared_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// PatternPart in braces.
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, PatternPartBraces) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ((()--())) RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_TRUE(match->patterns_[0]);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *node1 = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[0]);
|
|
|
|
ASSERT_TRUE(node1);
|
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(edge);
|
|
|
|
auto *node2 = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[2]);
|
|
|
|
ASSERT_TRUE(node2);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::BOTH);
|
|
|
|
ASSERT_TRUE(edge->identifier_);
|
|
|
|
EXPECT_THAT(edge->identifier_->name_,
|
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(2));
|
|
|
|
EXPECT_FALSE(edge->identifier_->user_declared_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternDetails) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"MATCH ()<-[:type1|type2 {a : 5, b : 10}]-() RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::IN);
|
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_THAT(edge->edge_types_,
|
|
|
|
UnorderedElementsAre(ast_generator.EdgeType("type1"),
|
|
|
|
ast_generator.EdgeType("type2")));
|
|
|
|
std::unordered_map<PropertyIx, int64_t> properties;
|
2018-12-12 00:13:28 +08:00
|
|
|
for (auto x : edge->properties_) {
|
|
|
|
TypedValue value = ast_generator.LiteralValue(x.second);
|
|
|
|
ASSERT_TRUE(value.type() == TypedValue::Type::Int);
|
|
|
|
properties[x.first] = value.Value<int64_t>();
|
|
|
|
}
|
|
|
|
EXPECT_THAT(properties,
|
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
|
|
|
UnorderedElementsAre(Pair(ast_generator.Prop("a"), 5),
|
|
|
|
Pair(ast_generator.Prop("b"), 10)));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternVariable) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[var]->() RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
ASSERT_TRUE(edge->identifier_);
|
|
|
|
EXPECT_THAT(edge->identifier_->name_, "var");
|
|
|
|
EXPECT_TRUE(edge->identifier_->user_declared_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert that match has a single pattern with a single edge atom and store it
|
|
|
|
// in edge parameter.
|
|
|
|
void AssertMatchSingleEdgeAtom(Match *match, EdgeAtom *&edge) {
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(edge);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternUnbounded) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternLowerBounded) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*42..]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
ast_generator.CheckLiteral(edge->lower_bound_, 42);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternUpperBounded) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*..42]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
ast_generator.CheckLiteral(edge->upper_bound_, 42);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternLowerUpperBounded) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*24..42]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
ast_generator.CheckLiteral(edge->lower_bound_, 24);
|
|
|
|
ast_generator.CheckLiteral(edge->upper_bound_, 42);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternFixedRange) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*42]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
ast_generator.CheckLiteral(edge->lower_bound_, 42);
|
|
|
|
ast_generator.CheckLiteral(edge->upper_bound_, 42);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternFloatingUpperBound) {
|
2018-12-12 00:13:28 +08:00
|
|
|
// [r*1...2] should be parsed as [r*1..0.2]
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*1...2]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
ast_generator.CheckLiteral(edge->lower_bound_, 1);
|
|
|
|
ast_generator.CheckLiteral(edge->upper_bound_, 0.2);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternUnboundedWithProperty) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r* {prop: 42}]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
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
|
|
|
ast_generator.CheckLiteral(edge->properties_[ast_generator.Prop("prop")], 42);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest,
|
|
|
|
RelationshipPatternDotsUnboundedWithEdgeTypeProperty) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"MATCH ()-[r:edge_type*..{prop: 42}]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
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
|
|
|
ast_generator.CheckLiteral(edge->properties_[ast_generator.Prop("prop")], 42);
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_EQ(edge->edge_types_.size(), 1U);
|
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
|
|
|
auto edge_type = ast_generator.EdgeType("edge_type");
|
2018-12-12 00:13:28 +08:00
|
|
|
EXPECT_EQ(edge->edge_types_[0], edge_type);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RelationshipPatternUpperBoundedWithProperty) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH ()-[r*..2{prop: 42}]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(edge->type_, EdgeAtom::Type::DEPTH_FIRST);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
ast_generator.CheckLiteral(edge->upper_bound_, 2);
|
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
|
|
|
ast_generator.CheckLiteral(edge->properties_[ast_generator.Prop("prop")], 42);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnUnanemdIdentifier) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("RETURN var"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = return_clause->body_.named_expressions[0];
|
|
|
|
ASSERT_TRUE(named_expr);
|
|
|
|
ASSERT_EQ(named_expr->name_, "var");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "var");
|
|
|
|
ASSERT_TRUE(identifier->user_declared_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Create) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("CREATE (n)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *create = dynamic_cast<Create *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(create);
|
|
|
|
ASSERT_EQ(create->patterns_.size(), 1U);
|
|
|
|
ASSERT_TRUE(create->patterns_[0]);
|
|
|
|
ASSERT_EQ(create->patterns_[0]->atoms_.size(), 1U);
|
|
|
|
auto node = dynamic_cast<NodeAtom *>(create->patterns_[0]->atoms_[0]);
|
|
|
|
ASSERT_TRUE(node);
|
|
|
|
ASSERT_TRUE(node->identifier_);
|
|
|
|
ASSERT_EQ(node->identifier_->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Delete) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("DELETE n, m"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *del = dynamic_cast<Delete *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(del);
|
|
|
|
ASSERT_FALSE(del->detach_);
|
|
|
|
ASSERT_EQ(del->expressions_.size(), 2U);
|
|
|
|
auto *identifier1 = dynamic_cast<Identifier *>(del->expressions_[0]);
|
|
|
|
ASSERT_TRUE(identifier1);
|
|
|
|
ASSERT_EQ(identifier1->name_, "n");
|
|
|
|
auto *identifier2 = dynamic_cast<Identifier *>(del->expressions_[1]);
|
|
|
|
ASSERT_TRUE(identifier2);
|
|
|
|
ASSERT_EQ(identifier2->name_, "m");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DeleteDetach) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("DETACH DELETE n"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *del = dynamic_cast<Delete *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(del);
|
|
|
|
ASSERT_TRUE(del->detach_);
|
|
|
|
ASSERT_EQ(del->expressions_.size(), 1U);
|
|
|
|
auto *identifier1 = dynamic_cast<Identifier *>(del->expressions_[0]);
|
|
|
|
ASSERT_TRUE(identifier1);
|
|
|
|
ASSERT_EQ(identifier1->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, OptionalMatchWhere) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("OPTIONAL MATCH (n) WHERE m RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_TRUE(match->optional_);
|
|
|
|
ASSERT_TRUE(match->where_);
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(match->where_->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "m");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Set) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("SET a.x = b, c = d, e += f, g : h : i "));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 4U);
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_property = dynamic_cast<SetProperty *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(set_property);
|
|
|
|
ASSERT_TRUE(set_property->property_lookup_);
|
|
|
|
auto *identifier1 =
|
|
|
|
dynamic_cast<Identifier *>(set_property->property_lookup_->expression_);
|
|
|
|
ASSERT_TRUE(identifier1);
|
|
|
|
ASSERT_EQ(identifier1->name_, "a");
|
|
|
|
ASSERT_EQ(set_property->property_lookup_->property_,
|
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
|
|
|
ast_generator.Prop("x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
auto *identifier2 = dynamic_cast<Identifier *>(set_property->expression_);
|
|
|
|
ASSERT_EQ(identifier2->name_, "b");
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_properties_assignment =
|
|
|
|
dynamic_cast<SetProperties *>(single_query->clauses_[1]);
|
|
|
|
ASSERT_TRUE(set_properties_assignment);
|
|
|
|
ASSERT_FALSE(set_properties_assignment->update_);
|
|
|
|
ASSERT_TRUE(set_properties_assignment->identifier_);
|
|
|
|
ASSERT_EQ(set_properties_assignment->identifier_->name_, "c");
|
|
|
|
auto *identifier =
|
|
|
|
dynamic_cast<Identifier *>(set_properties_assignment->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "d");
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_properties_update =
|
|
|
|
dynamic_cast<SetProperties *>(single_query->clauses_[2]);
|
|
|
|
ASSERT_TRUE(set_properties_update);
|
|
|
|
ASSERT_TRUE(set_properties_update->update_);
|
|
|
|
ASSERT_TRUE(set_properties_update->identifier_);
|
|
|
|
ASSERT_EQ(set_properties_update->identifier_->name_, "e");
|
|
|
|
auto *identifier =
|
|
|
|
dynamic_cast<Identifier *>(set_properties_update->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "f");
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_labels = dynamic_cast<SetLabels *>(single_query->clauses_[3]);
|
|
|
|
ASSERT_TRUE(set_labels);
|
|
|
|
ASSERT_TRUE(set_labels->identifier_);
|
|
|
|
ASSERT_EQ(set_labels->identifier_->name_, "g");
|
|
|
|
ASSERT_THAT(set_labels->labels_,
|
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
|
|
|
UnorderedElementsAre(ast_generator.Label("h"),
|
|
|
|
ast_generator.Label("i")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Remove) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("REMOVE a.x, g : h : i"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
{
|
|
|
|
auto *remove_property =
|
|
|
|
dynamic_cast<RemoveProperty *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(remove_property);
|
|
|
|
ASSERT_TRUE(remove_property->property_lookup_);
|
|
|
|
auto *identifier1 = dynamic_cast<Identifier *>(
|
|
|
|
remove_property->property_lookup_->expression_);
|
|
|
|
ASSERT_TRUE(identifier1);
|
|
|
|
ASSERT_EQ(identifier1->name_, "a");
|
|
|
|
ASSERT_EQ(remove_property->property_lookup_->property_,
|
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
|
|
|
ast_generator.Prop("x"));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
{
|
|
|
|
auto *remove_labels =
|
|
|
|
dynamic_cast<RemoveLabels *>(single_query->clauses_[1]);
|
|
|
|
ASSERT_TRUE(remove_labels);
|
|
|
|
ASSERT_TRUE(remove_labels->identifier_);
|
|
|
|
ASSERT_EQ(remove_labels->identifier_->name_, "g");
|
|
|
|
ASSERT_THAT(remove_labels->labels_,
|
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
|
|
|
UnorderedElementsAre(ast_generator.Label("h"),
|
|
|
|
ast_generator.Label("i")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, With) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("WITH n AS m RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(with);
|
|
|
|
ASSERT_FALSE(with->body_.distinct);
|
|
|
|
ASSERT_FALSE(with->body_.limit);
|
|
|
|
ASSERT_FALSE(with->body_.skip);
|
|
|
|
ASSERT_EQ(with->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_FALSE(with->where_);
|
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
|
|
|
ASSERT_EQ(named_expr->name_, "m");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithNonAliasedExpression) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("WITH n.x RETURN 1"),
|
|
|
|
SemanticException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithNonAliasedVariable) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query =
|
|
|
|
dynamic_cast<CypherQuery *>(ast_generator.ParseQuery("WITH n RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(with);
|
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
|
|
|
ASSERT_EQ(named_expr->name_, "n");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithDistinct) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("WITH DISTINCT n AS m RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(with->body_.distinct);
|
|
|
|
ASSERT_FALSE(with->where_);
|
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
|
|
|
ASSERT_EQ(named_expr->name_, "m");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithBag) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"WITH n as m ORDER BY m SKIP 1 LIMIT 2 RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(with->body_.distinct);
|
|
|
|
ASSERT_FALSE(with->where_);
|
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
// No need to check contents of body. That is checked in RETURN clause tests.
|
|
|
|
ASSERT_EQ(with->body_.order_by.size(), 1U);
|
|
|
|
ASSERT_TRUE(with->body_.limit);
|
|
|
|
ASSERT_TRUE(with->body_.skip);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithWhere) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("WITH n AS m WHERE k RETURN 1"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(with);
|
|
|
|
ASSERT_TRUE(with->where_);
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(with->where_->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "k");
|
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
|
|
|
ASSERT_EQ(named_expr->name_, "m");
|
|
|
|
auto *identifier2 = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier2->name_, "n");
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, WithAnonymousVariableCapture) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("WITH 5 as anon1 MATCH () return *"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 3U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[1]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
auto *pattern = match->patterns_[0];
|
|
|
|
ASSERT_TRUE(pattern);
|
|
|
|
ASSERT_EQ(pattern->atoms_.size(), 1U);
|
|
|
|
auto *atom = dynamic_cast<NodeAtom *>(pattern->atoms_[0]);
|
|
|
|
ASSERT_TRUE(atom);
|
|
|
|
ASSERT_NE("anon1", atom->identifier_->name_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ClausesOrdering) {
|
2018-12-12 00:13:28 +08:00
|
|
|
// Obviously some of the ridiculous combinations don't fail here, but they
|
|
|
|
// will fail in semantic analysis or they make perfect sense as a part of
|
|
|
|
// bigger query.
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ast_generator.ParseQuery("RETURN 1");
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 RETURN 1"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 MATCH (n) RETURN n"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 DELETE n"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 MERGE (n)"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 WITH n AS m RETURN 1"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("RETURN 1 AS n UNWIND n AS x RETURN x"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
ASSERT_THROW(
|
|
|
|
ast_generator.ParseQuery("OPTIONAL MATCH (n) MATCH (m) RETURN n, m"),
|
|
|
|
SemanticException);
|
|
|
|
ast_generator.ParseQuery("OPTIONAL MATCH (n) WITH n MATCH (m) RETURN n, m");
|
|
|
|
ast_generator.ParseQuery("OPTIONAL MATCH (n) OPTIONAL MATCH (m) RETURN n, m");
|
|
|
|
ast_generator.ParseQuery("MATCH (n) OPTIONAL MATCH (m) RETURN n, m");
|
|
|
|
|
|
|
|
ast_generator.ParseQuery("CREATE (n)");
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET n:x MATCH (n) RETURN n"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
2019-02-25 21:16:32 +08:00
|
|
|
ast_generator.ParseQuery("REMOVE n.x SET n.x = 1");
|
|
|
|
ast_generator.ParseQuery("REMOVE n:L RETURN n");
|
|
|
|
ast_generator.ParseQuery("SET n.x = 1 WITH n AS m RETURN m");
|
|
|
|
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("MATCH (n)"), SemanticException);
|
|
|
|
ast_generator.ParseQuery("MATCH (n) MATCH (n) RETURN n");
|
|
|
|
ast_generator.ParseQuery("MATCH (n) SET n = m");
|
|
|
|
ast_generator.ParseQuery("MATCH (n) RETURN n");
|
|
|
|
ast_generator.ParseQuery("MATCH (n) WITH n AS m RETURN m");
|
|
|
|
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("WITH 1 AS n"), SemanticException);
|
|
|
|
ast_generator.ParseQuery("WITH 1 AS n WITH n AS m RETURN m");
|
|
|
|
ast_generator.ParseQuery("WITH 1 AS n RETURN n");
|
|
|
|
ast_generator.ParseQuery("WITH 1 AS n SET n += m");
|
|
|
|
ast_generator.ParseQuery("WITH 1 AS n MATCH (n) RETURN n");
|
|
|
|
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("UNWIND [1,2,3] AS x"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
2019-02-25 21:16:32 +08:00
|
|
|
ASSERT_THROW(
|
|
|
|
ast_generator.ParseQuery("CREATE (n) UNWIND [1,2,3] AS x RETURN x"),
|
|
|
|
SemanticException);
|
|
|
|
ast_generator.ParseQuery("UNWIND [1,2,3] AS x CREATE (n) RETURN x");
|
|
|
|
ast_generator.ParseQuery("CREATE (n) WITH n UNWIND [1,2,3] AS x RETURN x");
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Merge) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MERGE (a) -[:r]- (b) ON MATCH SET a.x = b.x "
|
|
|
|
"ON CREATE SET b :label ON MATCH SET b = a"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *merge = dynamic_cast<Merge *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(merge);
|
|
|
|
EXPECT_TRUE(dynamic_cast<Pattern *>(merge->pattern_));
|
|
|
|
ASSERT_EQ(merge->on_match_.size(), 2U);
|
|
|
|
EXPECT_TRUE(dynamic_cast<SetProperty *>(merge->on_match_[0]));
|
|
|
|
EXPECT_TRUE(dynamic_cast<SetProperties *>(merge->on_match_[1]));
|
|
|
|
ASSERT_EQ(merge->on_create_.size(), 1U);
|
|
|
|
EXPECT_TRUE(dynamic_cast<SetLabels *>(merge->on_create_[0]));
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Unwind) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("UNWIND [1,2,3] AS elem RETURN elem"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *unwind = dynamic_cast<Unwind *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(unwind);
|
|
|
|
auto *ret = dynamic_cast<Return *>(single_query->clauses_[1]);
|
|
|
|
EXPECT_TRUE(ret);
|
|
|
|
ASSERT_TRUE(unwind->named_expression_);
|
|
|
|
EXPECT_EQ(unwind->named_expression_->name_, "elem");
|
|
|
|
auto *expr = unwind->named_expression_->expression_;
|
|
|
|
ASSERT_TRUE(expr);
|
|
|
|
ASSERT_TRUE(dynamic_cast<ListLiteral *>(expr));
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, UnwindWithoutAsError) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("UNWIND [1,2,3] RETURN 42"),
|
|
|
|
SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, CreateIndex) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *index_query = dynamic_cast<IndexQuery *>(
|
|
|
|
ast_generator.ParseQuery("Create InDeX oN :mirko(slavko)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(index_query);
|
|
|
|
EXPECT_EQ(index_query->action_, IndexQuery::Action::CREATE);
|
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(index_query->label_, ast_generator.Label("mirko"));
|
|
|
|
std::vector<PropertyIx> expected_properties{ast_generator.Prop("slavko")};
|
2018-12-12 00:13:28 +08:00
|
|
|
EXPECT_EQ(index_query->properties_, expected_properties);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DropIndex) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *index_query = dynamic_cast<IndexQuery *>(
|
|
|
|
ast_generator.ParseQuery("dRoP InDeX oN :mirko(slavko)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(index_query);
|
|
|
|
EXPECT_EQ(index_query->action_, IndexQuery::Action::DROP);
|
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(index_query->label_, ast_generator.Label("mirko"));
|
|
|
|
std::vector<PropertyIx> expected_properties{ast_generator.Prop("slavko")};
|
2018-12-12 00:13:28 +08:00
|
|
|
EXPECT_EQ(index_query->properties_, expected_properties);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DropIndexWithoutProperties) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("dRoP InDeX oN :mirko()"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DropIndexWithMultipleProperties) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("dRoP InDeX oN :mirko(slavko, pero)"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnAll) {
|
|
|
|
{
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("RETURN all(x in [1,2,3])"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
2018-12-12 00:13:28 +08:00
|
|
|
{
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN all(x IN [1,2,3] WHERE x = 2)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *ret = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(ret);
|
|
|
|
ASSERT_EQ(ret->body_.named_expressions.size(), 1U);
|
|
|
|
auto *all =
|
|
|
|
dynamic_cast<All *>(ret->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(all);
|
|
|
|
EXPECT_EQ(all->identifier_->name_, "x");
|
|
|
|
auto *list_literal = dynamic_cast<ListLiteral *>(all->list_expression_);
|
|
|
|
EXPECT_TRUE(list_literal);
|
|
|
|
auto *eq = dynamic_cast<EqualOperator *>(all->where_->expression_);
|
|
|
|
EXPECT_TRUE(eq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnSingle) {
|
|
|
|
{
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("RETURN single(x in [1,2,3])"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN single(x IN [1,2,3] WHERE x = 2)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *ret = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(ret);
|
|
|
|
ASSERT_EQ(ret->body_.named_expressions.size(), 1U);
|
|
|
|
auto *single =
|
|
|
|
dynamic_cast<Single *>(ret->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(single);
|
|
|
|
EXPECT_EQ(single->identifier_->name_, "x");
|
|
|
|
auto *list_literal = dynamic_cast<ListLiteral *>(single->list_expression_);
|
|
|
|
EXPECT_TRUE(list_literal);
|
|
|
|
auto *eq = dynamic_cast<EqualOperator *>(single->where_->expression_);
|
|
|
|
EXPECT_TRUE(eq);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnReduce) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN reduce(sum = 0, x IN [1,2,3] | sum + x)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *ret = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(ret);
|
|
|
|
ASSERT_EQ(ret->body_.named_expressions.size(), 1U);
|
|
|
|
auto *reduce =
|
|
|
|
dynamic_cast<Reduce *>(ret->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(reduce);
|
|
|
|
EXPECT_EQ(reduce->accumulator_->name_, "sum");
|
|
|
|
ast_generator.CheckLiteral(reduce->initializer_, 0);
|
|
|
|
EXPECT_EQ(reduce->identifier_->name_, "x");
|
|
|
|
auto *list_literal = dynamic_cast<ListLiteral *>(reduce->list_);
|
|
|
|
EXPECT_TRUE(list_literal);
|
|
|
|
auto *add = dynamic_cast<AdditionOperator *>(reduce->expression_);
|
|
|
|
EXPECT_TRUE(add);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ReturnExtract) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN extract(x IN [1,2,3] | sum + x)"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *ret = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(ret);
|
|
|
|
ASSERT_EQ(ret->body_.named_expressions.size(), 1U);
|
|
|
|
auto *extract =
|
|
|
|
dynamic_cast<Extract *>(ret->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(extract);
|
|
|
|
EXPECT_EQ(extract->identifier_->name_, "x");
|
|
|
|
auto *list_literal = dynamic_cast<ListLiteral *>(extract->list_);
|
|
|
|
EXPECT_TRUE(list_literal);
|
|
|
|
auto *add = dynamic_cast<AdditionOperator *>(extract->expression_);
|
|
|
|
EXPECT_TRUE(add);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MatchBfsReturn) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"MATCH (n) -[r:type1|type2 *bfs..10 (e, n|e.prop = 42)]-> (m) RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *bfs = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(bfs);
|
|
|
|
EXPECT_TRUE(bfs->IsVariable());
|
|
|
|
EXPECT_EQ(bfs->direction_, EdgeAtom::Direction::OUT);
|
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_THAT(bfs->edge_types_,
|
|
|
|
UnorderedElementsAre(ast_generator.EdgeType("type1"),
|
|
|
|
ast_generator.EdgeType("type2")));
|
2018-12-12 00:13:28 +08:00
|
|
|
EXPECT_EQ(bfs->identifier_->name_, "r");
|
|
|
|
EXPECT_EQ(bfs->filter_lambda_.inner_edge->name_, "e");
|
|
|
|
EXPECT_TRUE(bfs->filter_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_EQ(bfs->filter_lambda_.inner_node->name_, "n");
|
|
|
|
EXPECT_TRUE(bfs->filter_lambda_.inner_node->user_declared_);
|
|
|
|
ast_generator.CheckLiteral(bfs->upper_bound_, 10);
|
|
|
|
auto *eq = dynamic_cast<EqualOperator *>(bfs->filter_lambda_.expression);
|
|
|
|
ASSERT_TRUE(eq);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MatchVariableLambdaSymbols) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("MATCH () -[*]- () RETURN *"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *var_expand = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(var_expand);
|
|
|
|
ASSERT_TRUE(var_expand->IsVariable());
|
|
|
|
EXPECT_FALSE(var_expand->filter_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_FALSE(var_expand->filter_lambda_.inner_node->user_declared_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MatchWShortestReturn) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
2018-12-12 00:13:28 +08:00
|
|
|
"MATCH ()-[r:type1|type2 *wShortest 10 (we, wn | 42) total_weight "
|
2019-02-25 21:16:32 +08:00
|
|
|
"(e, n | true)]->() RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *shortest = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(shortest);
|
|
|
|
EXPECT_TRUE(shortest->IsVariable());
|
|
|
|
EXPECT_EQ(shortest->type_, EdgeAtom::Type::WEIGHTED_SHORTEST_PATH);
|
|
|
|
EXPECT_EQ(shortest->direction_, EdgeAtom::Direction::OUT);
|
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_THAT(shortest->edge_types_,
|
|
|
|
UnorderedElementsAre(ast_generator.EdgeType("type1"),
|
|
|
|
ast_generator.EdgeType("type2")));
|
2018-12-12 00:13:28 +08:00
|
|
|
ast_generator.CheckLiteral(shortest->upper_bound_, 10);
|
|
|
|
EXPECT_FALSE(shortest->lower_bound_);
|
|
|
|
EXPECT_EQ(shortest->identifier_->name_, "r");
|
|
|
|
EXPECT_EQ(shortest->filter_lambda_.inner_edge->name_, "e");
|
|
|
|
EXPECT_TRUE(shortest->filter_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_EQ(shortest->filter_lambda_.inner_node->name_, "n");
|
|
|
|
EXPECT_TRUE(shortest->filter_lambda_.inner_node->user_declared_);
|
|
|
|
ast_generator.CheckLiteral(shortest->filter_lambda_.expression, true);
|
|
|
|
EXPECT_EQ(shortest->weight_lambda_.inner_edge->name_, "we");
|
|
|
|
EXPECT_TRUE(shortest->weight_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_EQ(shortest->weight_lambda_.inner_node->name_, "wn");
|
|
|
|
EXPECT_TRUE(shortest->weight_lambda_.inner_node->user_declared_);
|
|
|
|
ast_generator.CheckLiteral(shortest->weight_lambda_.expression, 42);
|
|
|
|
ASSERT_TRUE(shortest->total_weight_);
|
|
|
|
EXPECT_EQ(shortest->total_weight_->name_, "total_weight");
|
|
|
|
EXPECT_TRUE(shortest->total_weight_->user_declared_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, MatchWShortestNoFilterReturn) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
2018-12-12 00:13:28 +08:00
|
|
|
"MATCH ()-[r:type1|type2 *wShortest 10 (we, wn | 42)]->() "
|
2019-02-25 21:16:32 +08:00
|
|
|
"RETURN r"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *shortest = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(shortest);
|
|
|
|
EXPECT_TRUE(shortest->IsVariable());
|
|
|
|
EXPECT_EQ(shortest->type_, EdgeAtom::Type::WEIGHTED_SHORTEST_PATH);
|
|
|
|
EXPECT_EQ(shortest->direction_, EdgeAtom::Direction::OUT);
|
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_THAT(shortest->edge_types_,
|
|
|
|
UnorderedElementsAre(ast_generator.EdgeType("type1"),
|
|
|
|
ast_generator.EdgeType("type2")));
|
2018-12-12 00:13:28 +08:00
|
|
|
ast_generator.CheckLiteral(shortest->upper_bound_, 10);
|
|
|
|
EXPECT_FALSE(shortest->lower_bound_);
|
|
|
|
EXPECT_EQ(shortest->identifier_->name_, "r");
|
|
|
|
EXPECT_FALSE(shortest->filter_lambda_.expression);
|
|
|
|
EXPECT_FALSE(shortest->filter_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_FALSE(shortest->filter_lambda_.inner_node->user_declared_);
|
|
|
|
EXPECT_EQ(shortest->weight_lambda_.inner_edge->name_, "we");
|
|
|
|
EXPECT_TRUE(shortest->weight_lambda_.inner_edge->user_declared_);
|
|
|
|
EXPECT_EQ(shortest->weight_lambda_.inner_node->name_, "wn");
|
|
|
|
EXPECT_TRUE(shortest->weight_lambda_.inner_node->user_declared_);
|
|
|
|
ast_generator.CheckLiteral(shortest->weight_lambda_.expression, 42);
|
|
|
|
ASSERT_TRUE(shortest->total_weight_);
|
|
|
|
EXPECT_FALSE(shortest->total_weight_->user_declared_);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, SemanticExceptionOnWShortestLowerBound) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery(
|
|
|
|
"MATCH ()-[r *wShortest 10.. (e, n | 42)]-() RETURN r"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery(
|
|
|
|
"MATCH ()-[r *wShortest 10..20 (e, n | 42)]-() RETURN r"),
|
|
|
|
SemanticException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, SemanticExceptionOnWShortestWithoutLambda) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("MATCH ()-[r *wShortest]-() RETURN r"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SemanticException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, SemanticExceptionOnUnionTypeMix) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery(
|
|
|
|
"RETURN 5 as X UNION ALL RETURN 6 AS X UNION RETURN 7 AS X"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery(
|
|
|
|
"RETURN 5 as X UNION RETURN 6 AS X UNION ALL RETURN 7 AS X"),
|
|
|
|
SemanticException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, Union) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN 5 AS X, 6 AS Y UNION RETURN 6 AS X, 5 AS Y"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 2U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
|
|
|
|
ASSERT_EQ(query->cypher_unions_.size(), 1);
|
|
|
|
auto *cypher_union = query->cypher_unions_.at(0);
|
|
|
|
ASSERT_TRUE(cypher_union);
|
|
|
|
ASSERT_TRUE(cypher_union->distinct_);
|
|
|
|
ASSERT_TRUE(single_query = cypher_union->single_query_);
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 2U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, UnionAll) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"RETURN 5 AS X UNION ALL RETURN 6 AS X UNION ALL RETURN 7 AS X"));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
|
|
|
|
ASSERT_EQ(query->cypher_unions_.size(), 2);
|
|
|
|
|
|
|
|
auto *cypher_union = query->cypher_unions_.at(0);
|
|
|
|
ASSERT_TRUE(cypher_union);
|
|
|
|
ASSERT_FALSE(cypher_union->distinct_);
|
|
|
|
ASSERT_TRUE(single_query = cypher_union->single_query_);
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
|
|
|
|
cypher_union = query->cypher_unions_.at(1);
|
|
|
|
ASSERT_TRUE(cypher_union);
|
|
|
|
ASSERT_FALSE(cypher_union->distinct_);
|
|
|
|
ASSERT_TRUE(single_query = cypher_union->single_query_);
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.order_by.size(), 0U);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
ASSERT_FALSE(return_clause->body_.limit);
|
|
|
|
ASSERT_FALSE(return_clause->body_.skip);
|
|
|
|
ASSERT_FALSE(return_clause->body_.distinct);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
void check_auth_query(Base *ast_generator, std::string input,
|
|
|
|
AuthQuery::Action action, std::string user,
|
|
|
|
std::string role, std::string user_or_role,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::optional<TypedValue> password,
|
2018-12-12 00:13:28 +08:00
|
|
|
std::vector<AuthQuery::Privilege> privileges) {
|
2019-02-25 21:16:32 +08:00
|
|
|
auto *auth_query =
|
|
|
|
dynamic_cast<AuthQuery *>(ast_generator->ParseQuery(input));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(auth_query);
|
|
|
|
EXPECT_EQ(auth_query->action_, action);
|
|
|
|
EXPECT_EQ(auth_query->user_, user);
|
|
|
|
EXPECT_EQ(auth_query->role_, role);
|
|
|
|
EXPECT_EQ(auth_query->user_or_role_, user_or_role);
|
|
|
|
ASSERT_EQ(static_cast<bool>(auth_query->password_),
|
|
|
|
static_cast<bool>(password));
|
|
|
|
if (password) {
|
2019-02-25 21:16:32 +08:00
|
|
|
ast_generator->CheckLiteral(auth_query->password_, *password);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
EXPECT_EQ(auth_query->privileges_, privileges);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, UserOrRoleName) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE ROLE `us|er`"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE ROLE `us er`"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "CREATE ROLE `user`",
|
|
|
|
AuthQuery::Action::CREATE_ROLE, "", "user", "", {}, {});
|
|
|
|
check_auth_query(&ast_generator, "CREATE ROLE us___er",
|
|
|
|
AuthQuery::Action::CREATE_ROLE, "", "us___er", "", {}, {});
|
|
|
|
check_auth_query(&ast_generator, "CREATE ROLE `us+er`",
|
|
|
|
AuthQuery::Action::CREATE_ROLE, "", "us+er", "", {}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, CreateRole) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE ROLE"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "CREATE ROLE rola",
|
|
|
|
AuthQuery::Action::CREATE_ROLE, "", "rola", "", {}, {});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE ROLE lagano rolamo"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, DropRole) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DROP ROLE"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "DROP ROLE rola",
|
|
|
|
AuthQuery::Action::DROP_ROLE, "", "rola", "", {}, {});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DROP ROLE lagano rolamo"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
2019-02-25 21:16:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ShowRoles) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW ROLES ROLES"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SHOW ROLES", AuthQuery::Action::SHOW_ROLES,
|
|
|
|
"", "", "", {}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, CreateUser) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE USER"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE USER 123"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "CREATE USER user",
|
|
|
|
AuthQuery::Action::CREATE_USER, "user", "", "", {}, {});
|
|
|
|
check_auth_query(&ast_generator, "CREATE USER user IDENTIFIED BY 'password'",
|
|
|
|
AuthQuery::Action::CREATE_USER, "user", "", "", "password",
|
|
|
|
{});
|
|
|
|
check_auth_query(&ast_generator, "CREATE USER user IDENTIFIED BY ''",
|
|
|
|
AuthQuery::Action::CREATE_USER, "user", "", "", "", {});
|
|
|
|
check_auth_query(&ast_generator, "CREATE USER user IDENTIFIED BY null",
|
|
|
|
AuthQuery::Action::CREATE_USER, "user", "", "",
|
|
|
|
TypedValue::Null, {});
|
|
|
|
ASSERT_THROW(
|
|
|
|
ast_generator.ParseQuery("CRATE USER user IDENTIFIED BY password"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE USER user IDENTIFIED BY 5"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CREATE USER user IDENTIFIED BY "),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, SetPassword) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET PASSWORD FOR"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET PASSWORD FOR user "),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SET PASSWORD FOR user TO null",
|
|
|
|
AuthQuery::Action::SET_PASSWORD, "user", "", "",
|
|
|
|
TypedValue::Null, {});
|
|
|
|
check_auth_query(&ast_generator, "SET PASSWORD FOR user TO 'password'",
|
|
|
|
AuthQuery::Action::SET_PASSWORD, "user", "", "", "password",
|
|
|
|
{});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET PASSWORD FOR user To 5"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, DropUser) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DROP USER"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "DROP USER user",
|
|
|
|
AuthQuery::Action::DROP_USER, "user", "", "", {}, {});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DROP USER lagano rolamo"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ShowUsers) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW USERS ROLES"), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SHOW USERS", AuthQuery::Action::SHOW_USERS,
|
|
|
|
"", "", "", {}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, SetRole) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET ROLE"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET ROLE user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET ROLE FOR user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SET ROLE FOR user TO"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SET ROLE FOR user TO role",
|
|
|
|
AuthQuery::Action::SET_ROLE, "user", "role", "", {}, {});
|
|
|
|
check_auth_query(&ast_generator, "SET ROLE FOR user TO null",
|
|
|
|
AuthQuery::Action::SET_ROLE, "user", "null", "", {}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ClearRole) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CLEAR ROLE"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CLEAR ROLE user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("CLEAR ROLE FOR user TO"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "CLEAR ROLE FOR user",
|
|
|
|
AuthQuery::Action::CLEAR_ROLE, "user", "", "", {}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, GrantPrivilege) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("GRANT"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("GRANT TO user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("GRANT BLABLA TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("GRANT MATCH, TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("GRANT MATCH, BLABLA TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "GRANT MATCH TO user",
|
|
|
|
AuthQuery::Action::GRANT_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH});
|
|
|
|
check_auth_query(&ast_generator, "GRANT MATCH, AUTH TO user",
|
|
|
|
AuthQuery::Action::GRANT_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH, AuthQuery::Privilege::AUTH});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, DenyPrivilege) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DENY"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DENY TO user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DENY BLABLA TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DENY MATCH, TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("DENY MATCH, BLABLA TO user"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "DENY MATCH TO user",
|
|
|
|
AuthQuery::Action::DENY_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH});
|
|
|
|
check_auth_query(&ast_generator, "DENY MATCH, AUTH TO user",
|
|
|
|
AuthQuery::Action::DENY_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH, AuthQuery::Privilege::AUTH});
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, RevokePrivilege) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("REVOKE"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("REVOKE FROM user"), SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("REVOKE BLABLA FROM user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("REVOKE MATCH, FROM user"),
|
|
|
|
SyntaxException);
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("REVOKE MATCH, BLABLA FROM user"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "REVOKE MATCH FROM user",
|
|
|
|
AuthQuery::Action::REVOKE_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH});
|
|
|
|
check_auth_query(&ast_generator, "REVOKE MATCH, AUTH FROM user",
|
|
|
|
AuthQuery::Action::REVOKE_PRIVILEGE, "", "", "user", {},
|
|
|
|
{AuthQuery::Privilege::MATCH, AuthQuery::Privilege::AUTH});
|
|
|
|
check_auth_query(&ast_generator, "REVOKE ALL PRIVILEGES FROM user",
|
|
|
|
AuthQuery::Action::REVOKE_PRIVILEGE, "", "", "user", {},
|
|
|
|
kPrivilegesAll);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ShowPrivileges) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW PRIVILEGES FOR"),
|
|
|
|
SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SHOW PRIVILEGES FOR user",
|
|
|
|
AuthQuery::Action::SHOW_PRIVILEGES, "", "", "user", {}, {});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW PRIVILEGES FOR user1, user2"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ShowRoleForUser) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW ROLE FOR "), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SHOW ROLE FOR user",
|
|
|
|
AuthQuery::Action::SHOW_ROLE_FOR_USER, "user", "", "", {},
|
|
|
|
{});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW ROLE FOR user1, user2"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, ShowUsersForRole) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW USERS FOR "), SyntaxException);
|
|
|
|
check_auth_query(&ast_generator, "SHOW USERS FOR role",
|
|
|
|
AuthQuery::Action::SHOW_USERS_FOR_ROLE, "", "role", "", {},
|
|
|
|
{});
|
|
|
|
ASSERT_THROW(ast_generator.ParseQuery("SHOW USERS FOR role1, role2"),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CypherMainVisitorTest, CreateStream) {
|
2018-12-12 00:13:28 +08:00
|
|
|
auto check_create_stream =
|
2019-02-25 21:16:32 +08:00
|
|
|
[this](std::string input, const std::string &stream_name,
|
|
|
|
const std::string &stream_uri, const std::string &stream_topic,
|
|
|
|
const std::string &transform_uri,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::optional<int64_t> batch_interval_in_ms,
|
|
|
|
std::optional<int64_t> batch_size) {
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(stream_query);
|
|
|
|
EXPECT_EQ(stream_query->action_, StreamQuery::Action::CREATE_STREAM);
|
|
|
|
EXPECT_EQ(stream_query->stream_name_, stream_name);
|
|
|
|
ASSERT_TRUE(stream_query->stream_uri_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->stream_uri_,
|
|
|
|
TypedValue(stream_uri));
|
|
|
|
ASSERT_TRUE(stream_query->stream_topic_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->stream_topic_,
|
|
|
|
TypedValue(stream_topic));
|
|
|
|
ASSERT_TRUE(stream_query->transform_uri_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->transform_uri_,
|
|
|
|
TypedValue(transform_uri));
|
|
|
|
if (batch_interval_in_ms) {
|
|
|
|
ASSERT_TRUE(stream_query->batch_interval_in_ms_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->batch_interval_in_ms_,
|
|
|
|
TypedValue(*batch_interval_in_ms));
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(stream_query->batch_interval_in_ms_, nullptr);
|
|
|
|
}
|
|
|
|
if (batch_size) {
|
|
|
|
ASSERT_TRUE(stream_query->batch_size_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->batch_size_,
|
|
|
|
TypedValue(*batch_size));
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(stream_query->batch_size_, nullptr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
check_create_stream(
|
|
|
|
"CREATE STREAM stream AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TOPIC 'tropika' "
|
|
|
|
"WITH TRANSFORM 'localhost/test.py'",
|
2019-04-23 17:00:49 +08:00
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", std::nullopt,
|
|
|
|
std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
check_create_stream(
|
|
|
|
"CreaTE StreaM stream AS LOad daTA KAFKA 'localhost' "
|
|
|
|
"WitH TopIC 'tropika' "
|
|
|
|
"WITH TRAnsFORM 'localhost/test.py' bAtCH inTErvAL 168",
|
2019-04-23 17:00:49 +08:00
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168, std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
check_create_stream(
|
|
|
|
"CreaTE StreaM stream AS LOad daTA KAFKA 'localhost' "
|
|
|
|
"WITH TopIC 'tropika' "
|
|
|
|
"WITH TRAnsFORM 'localhost/test.py' bAtCH SizE 17",
|
2019-04-23 17:00:49 +08:00
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", std::nullopt, 17);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
check_create_stream(
|
|
|
|
"CreaTE StreaM stream AS LOad daTA KAFKA 'localhost' "
|
|
|
|
"WitH TOPic 'tropika' "
|
|
|
|
"WITH TRAnsFORM 'localhost/test.py' bAtCH inTErvAL 168 Batch SIze 17",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168, 17);
|
|
|
|
|
|
|
|
EXPECT_THROW(check_create_stream(
|
|
|
|
"CREATE STREAM stream AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TRANSFORM 'localhost/test.py' BATCH INTERVAL 'jedan' ",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_create_stream(
|
|
|
|
"CREATE STREAM stream AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TOPIC 'tropika' "
|
|
|
|
"WITH TRANSFORM 'localhost/test.py' BATCH SIZE 'jedan' ",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py",
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt, 17),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_create_stream(
|
|
|
|
"CREATE STREAM 123 AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TOPIC 'tropika' "
|
|
|
|
"WITH TRANSFORM 'localhost/test.py' BATCH INTERVAL 168 ",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
2019-04-23 17:00:49 +08:00
|
|
|
EXPECT_THROW(
|
|
|
|
check_create_stream("CREATE STREAM stream AS LOAD DATA KAFKA localhost "
|
|
|
|
"WITH TOPIC 'tropika' "
|
|
|
|
"WITH TRANSFORM 'localhost/test.py'",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py",
|
|
|
|
std::nullopt, std::nullopt),
|
|
|
|
SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
EXPECT_THROW(check_create_stream(
|
|
|
|
"CREATE STREAM stream AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TOPIC 2"
|
|
|
|
"WITH TRANSFORM localhost/test.py BATCH INTERVAL 168 ",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_create_stream(
|
|
|
|
"CREATE STREAM stream AS LOAD DATA KAFKA 'localhost' "
|
|
|
|
"WITH TOPIC 'tropika'"
|
|
|
|
"WITH TRANSFORM localhost/test.py BATCH INTERVAL 168 ",
|
|
|
|
"stream", "localhost", "tropika", "localhost/test.py", 168,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, DropStream) {
|
|
|
|
auto check_drop_stream = [this](std::string input,
|
|
|
|
const std::string &stream_name) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(stream_query);
|
|
|
|
EXPECT_EQ(stream_query->action_, StreamQuery::Action::DROP_STREAM);
|
|
|
|
EXPECT_EQ(stream_query->stream_name_, stream_name);
|
|
|
|
};
|
|
|
|
|
|
|
|
check_drop_stream("DRop stREAm stream", "stream");
|
|
|
|
check_drop_stream("DRop stREAm strim", "strim");
|
|
|
|
|
|
|
|
EXPECT_THROW(check_drop_stream("DROp sTREAM", ""), SyntaxException);
|
|
|
|
|
|
|
|
EXPECT_THROW(check_drop_stream("DROP STreAM 123", "123"), SyntaxException);
|
|
|
|
|
|
|
|
EXPECT_THROW(check_drop_stream("DroP STREAM '123'", "123"), SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, ShowStreams) {
|
|
|
|
auto check_show_streams = [this](std::string input) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(stream_query);
|
|
|
|
EXPECT_EQ(stream_query->action_, StreamQuery::Action::SHOW_STREAMS);
|
|
|
|
};
|
|
|
|
|
|
|
|
check_show_streams("SHOW STREAMS");
|
|
|
|
|
|
|
|
EXPECT_THROW(check_show_streams("SHOW STREAMS lololo"), SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StartStopStream) {
|
2019-04-23 17:00:49 +08:00
|
|
|
auto check_start_stop_stream = [this](std::string input,
|
|
|
|
const std::string &stream_name,
|
|
|
|
bool is_start,
|
|
|
|
std::optional<int64_t> limit_batches) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
|
|
|
ASSERT_TRUE(stream_query);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
EXPECT_EQ(stream_query->stream_name_, stream_name);
|
|
|
|
EXPECT_EQ(stream_query->action_, is_start
|
|
|
|
? StreamQuery::Action::START_STREAM
|
|
|
|
: StreamQuery::Action::STOP_STREAM);
|
|
|
|
|
|
|
|
if (limit_batches) {
|
|
|
|
ASSERT_TRUE(is_start);
|
|
|
|
ASSERT_TRUE(stream_query->limit_batches_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->limit_batches_,
|
|
|
|
TypedValue(*limit_batches));
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(stream_query->limit_batches_, nullptr);
|
|
|
|
}
|
|
|
|
};
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
check_start_stop_stream("stARt STreaM STREAM", "STREAM", true, std::nullopt);
|
|
|
|
check_start_stop_stream("stARt STreaM strim", "strim", true, std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
check_start_stop_stream("StARt STreAM strim LimIT 10 BATchES", "strim", true,
|
|
|
|
10);
|
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
check_start_stop_stream("StoP StrEAM strim", "strim", false, std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
|
|
|
EXPECT_THROW(check_start_stop_stream("staRT STReaM 'strim'", "strim", true,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_start_stop_stream("sTART STReaM strim LImiT 'dva' BATCheS",
|
|
|
|
"strim", true, 2),
|
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_start_stop_stream("StoP STreAM 'strim'", "strim", false,
|
2019-04-23 17:00:49 +08:00
|
|
|
std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(check_start_stop_stream("STOp sTREAM strim LIMit 2 baTCHES",
|
|
|
|
"strim", false, 2),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, StartStopAllStreams) {
|
|
|
|
auto check_start_stop_all_streams = [this](std::string input, bool is_start) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
2018-12-12 00:13:28 +08:00
|
|
|
ASSERT_TRUE(stream_query);
|
|
|
|
EXPECT_EQ(stream_query->action_,
|
|
|
|
is_start ? StreamQuery::Action::START_ALL_STREAMS
|
|
|
|
: StreamQuery::Action::STOP_ALL_STREAMS);
|
|
|
|
};
|
|
|
|
|
|
|
|
check_start_stop_all_streams("STarT AlL StreAMs", true);
|
|
|
|
|
|
|
|
check_start_stop_all_streams("StoP aLL STrEAMs", false);
|
|
|
|
|
|
|
|
EXPECT_THROW(check_start_stop_all_streams("StaRT aLL STreAM", true),
|
|
|
|
SyntaxException);
|
|
|
|
|
|
|
|
EXPECT_THROW(check_start_stop_all_streams("SToP AlL STREaM", false),
|
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestStream) {
|
2019-04-23 17:00:49 +08:00
|
|
|
auto check_test_stream = [this](std::string input,
|
|
|
|
const std::string &stream_name,
|
|
|
|
std::optional<int64_t> limit_batches) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *stream_query =
|
|
|
|
dynamic_cast<StreamQuery *>(ast_generator.ParseQuery(input));
|
|
|
|
ASSERT_TRUE(stream_query);
|
|
|
|
EXPECT_EQ(stream_query->stream_name_, stream_name);
|
|
|
|
EXPECT_EQ(stream_query->action_, StreamQuery::Action::TEST_STREAM);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
if (limit_batches) {
|
|
|
|
ASSERT_TRUE(stream_query->limit_batches_);
|
|
|
|
ast_generator.CheckLiteral(stream_query->limit_batches_,
|
|
|
|
TypedValue(*limit_batches));
|
|
|
|
} else {
|
|
|
|
EXPECT_EQ(stream_query->limit_batches_, nullptr);
|
|
|
|
}
|
|
|
|
};
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
check_test_stream("TesT STreaM strim", "strim", std::nullopt);
|
|
|
|
check_test_stream("TesT STreaM STREAM", "STREAM", std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
check_test_stream("tESt STreAM STREAM LimIT 10 BATchES", "STREAM", 10);
|
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
check_test_stream("Test StrEAM STREAM", "STREAM", std::nullopt);
|
2018-12-12 00:13:28 +08:00
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
EXPECT_THROW(check_test_stream("tEST STReaM 'strim'", "strim", std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
EXPECT_THROW(
|
|
|
|
check_test_stream("test STReaM strim LImiT 'dva' BATCheS", "strim", 2),
|
|
|
|
SyntaxException);
|
2019-04-23 17:00:49 +08:00
|
|
|
EXPECT_THROW(check_test_stream("test STreAM 'strim'", "strim", std::nullopt),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestExplainRegularQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_TRUE(dynamic_cast<ExplainQuery *>(
|
|
|
|
ast_generator.ParseQuery("EXPLAIN RETURN n")));
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestExplainExplainQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("EXPLAIN EXPLAIN RETURN n"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestExplainAuthQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("EXPLAIN SHOW ROLES"), SyntaxException);
|
2018-12-12 00:13:28 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestExplainStreamQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("EXPLAIN SHOW STREAMS"),
|
2018-12-12 00:13:28 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestProfileRegularQuery) {
|
2019-01-15 18:11:06 +08:00
|
|
|
{
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_TRUE(dynamic_cast<ProfileQuery *>(
|
|
|
|
ast_generator.ParseQuery("PROFILE RETURN n")));
|
2019-01-15 18:11:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestProfileComplicatedQuery) {
|
2019-01-15 18:11:06 +08:00
|
|
|
{
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_TRUE(dynamic_cast<ProfileQuery *>(ast_generator.ParseQuery(
|
2019-01-15 18:11:06 +08:00
|
|
|
"profile optional match (n) where n.hello = 5 "
|
|
|
|
"return n union optional match (n) where n.there = 10 "
|
2019-02-25 21:16:32 +08:00
|
|
|
"return n")));
|
2019-01-15 18:11:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestProfileProfileQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("PROFILE PROFILE RETURN n"),
|
2019-01-15 18:11:06 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestProfileAuthQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("PROFILE SHOW ROLES"), SyntaxException);
|
2019-01-15 18:11:06 +08:00
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, TestProfileStreamQuery) {
|
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
EXPECT_THROW(ast_generator.ParseQuery("PROFILE SHOW STREAMS"),
|
2019-01-15 18:11:06 +08:00
|
|
|
SyntaxException);
|
|
|
|
}
|
|
|
|
|
2019-02-25 21:16:32 +08:00
|
|
|
TEST_P(CypherMainVisitorTest, RegexMatch) {
|
2019-02-27 18:12:24 +08:00
|
|
|
{
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(ast_generator.ParseQuery(
|
|
|
|
"MATCH (n) WHERE n.name =~ \".*bla.*\" RETURN n.name"));
|
2019-02-27 18:12:24 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 2U);
|
|
|
|
auto *match_clause = dynamic_cast<Match *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match_clause);
|
|
|
|
auto *regex_match =
|
|
|
|
dynamic_cast<RegexMatch *>(match_clause->where_->expression_);
|
|
|
|
ASSERT_TRUE(regex_match);
|
|
|
|
ASSERT_TRUE(dynamic_cast<PropertyLookup *>(regex_match->string_expr_));
|
|
|
|
ast_generator.CheckLiteral(regex_match->regex_, ".*bla.*");
|
|
|
|
}
|
|
|
|
{
|
2019-02-25 21:16:32 +08:00
|
|
|
auto &ast_generator = *GetParam();
|
|
|
|
auto *query = dynamic_cast<CypherQuery *>(
|
|
|
|
ast_generator.ParseQuery("RETURN \"text\" =~ \".*bla.*\""));
|
2019-02-27 18:12:24 +08:00
|
|
|
ASSERT_TRUE(query);
|
|
|
|
ASSERT_TRUE(query->single_query_);
|
|
|
|
auto *single_query = query->single_query_;
|
|
|
|
ASSERT_EQ(single_query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(single_query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expression = return_clause->body_.named_expressions[0];
|
|
|
|
auto *regex_match =
|
|
|
|
dynamic_cast<RegexMatch *>(named_expression->expression_);
|
|
|
|
ASSERT_TRUE(regex_match);
|
|
|
|
ast_generator.CheckLiteral(regex_match->string_expr_, "text");
|
|
|
|
ast_generator.CheckLiteral(regex_match->regex_, ".*bla.*");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-12 00:13:28 +08:00
|
|
|
} // namespace
|