2017-03-12 04:19:57 +08:00
|
|
|
#include <algorithm>
|
2017-02-24 00:15:31 +08:00
|
|
|
#include <climits>
|
2017-07-19 21:48:43 +08:00
|
|
|
#include <limits>
|
2017-02-24 00:15:31 +08:00
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
2017-03-12 04:19:57 +08:00
|
|
|
#include <vector>
|
|
|
|
|
2017-02-24 00:15:31 +08:00
|
|
|
#include "antlr4-runtime.h"
|
2017-07-12 18:17:15 +08:00
|
|
|
#include "database/dbms.hpp"
|
2017-02-24 00:15:31 +08:00
|
|
|
#include "gmock/gmock.h"
|
2017-03-17 23:29:27 +08:00
|
|
|
#include "gtest/gtest.h"
|
2017-03-12 04:19:57 +08:00
|
|
|
#include "query/context.hpp"
|
2017-03-23 23:20:06 +08:00
|
|
|
#include "query/frontend/ast/ast.hpp"
|
2017-03-12 04:19:57 +08:00
|
|
|
#include "query/frontend/ast/cypher_main_visitor.hpp"
|
2017-02-24 00:15:31 +08:00
|
|
|
#include "query/frontend/opencypher/parser.hpp"
|
2017-06-21 22:10:52 +08:00
|
|
|
#include "query/frontend/stripped.hpp"
|
2017-04-10 18:22:48 +08:00
|
|
|
#include "query/typed_value.hpp"
|
2017-02-24 00:15:31 +08:00
|
|
|
|
2017-03-17 17:43:18 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
using namespace query;
|
|
|
|
using namespace query::frontend;
|
2017-04-10 18:22:48 +08:00
|
|
|
using query::TypedValue;
|
2017-03-21 22:10:59 +08:00
|
|
|
using testing::Pair;
|
2017-05-05 00:07:54 +08:00
|
|
|
using testing::ElementsAre;
|
|
|
|
using testing::UnorderedElementsAre;
|
2017-03-17 17:43:18 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
// This generator uses ast constructed by parsing the query.
|
2017-03-17 17:43:18 +08:00
|
|
|
class AstGenerator {
|
2017-03-17 23:29:27 +08:00
|
|
|
public:
|
2017-03-17 17:43:18 +08:00
|
|
|
AstGenerator(const std::string &query)
|
2017-03-17 23:29:27 +08:00
|
|
|
: dbms_(),
|
|
|
|
db_accessor_(dbms_.active()),
|
|
|
|
context_(Config{}, *db_accessor_),
|
|
|
|
query_string_(query),
|
|
|
|
parser_(query),
|
|
|
|
visitor_(context_),
|
|
|
|
query_([&]() {
|
2017-03-17 17:43:18 +08:00
|
|
|
visitor_.visit(parser_.tree());
|
|
|
|
return visitor_.query();
|
|
|
|
}()) {}
|
|
|
|
|
|
|
|
Dbms dbms_;
|
|
|
|
std::unique_ptr<GraphDbAccessor> db_accessor_;
|
|
|
|
Context context_;
|
|
|
|
std::string query_string_;
|
|
|
|
::frontend::opencypher::Parser parser_;
|
|
|
|
CypherMainVisitor visitor_;
|
|
|
|
Query *query_;
|
|
|
|
};
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
// This clones ast, but uses original one. This done just to ensure that cloning
|
|
|
|
// doesn't change original.
|
|
|
|
class OriginalAfterCloningAstGenerator : public AstGenerator {
|
|
|
|
public:
|
|
|
|
OriginalAfterCloningAstGenerator(const std::string &query)
|
|
|
|
: AstGenerator(query) {
|
|
|
|
AstTreeStorage storage;
|
|
|
|
visitor_.query()->Clone(storage);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// This generator clones parsed ast and uses that one.
|
|
|
|
// Original ast is cleared after cloning to ensure that cloned ast doesn't reuse
|
|
|
|
// any data from original ast.
|
|
|
|
class ClonedAstGenerator {
|
|
|
|
public:
|
|
|
|
ClonedAstGenerator(const std::string &query)
|
|
|
|
: dbms_(),
|
|
|
|
db_accessor_(dbms_.active()),
|
|
|
|
context_(Config{}, *db_accessor_),
|
|
|
|
query_string_(query),
|
|
|
|
query_([&]() {
|
|
|
|
::frontend::opencypher::Parser parser(query);
|
|
|
|
CypherMainVisitor visitor(context_);
|
|
|
|
visitor.visit(parser.tree());
|
|
|
|
return visitor.query()->Clone(storage);
|
|
|
|
}()) {}
|
|
|
|
|
|
|
|
Dbms dbms_;
|
|
|
|
std::unique_ptr<GraphDbAccessor> db_accessor_;
|
|
|
|
Context context_;
|
|
|
|
std::string query_string_;
|
|
|
|
AstTreeStorage storage;
|
|
|
|
Query *query_;
|
|
|
|
};
|
|
|
|
|
2017-06-21 22:10:52 +08:00
|
|
|
// This generator strips ast, clones it and then plugs stripped out literals in
|
|
|
|
// the same way it is done in ast cacheing in interpreter.
|
|
|
|
class CachedAstGenerator {
|
|
|
|
public:
|
|
|
|
CachedAstGenerator(const std::string &query)
|
|
|
|
: dbms_(),
|
|
|
|
db_accessor_(dbms_.active()),
|
|
|
|
context_(Config{}, *db_accessor_),
|
|
|
|
query_string_(query),
|
|
|
|
storage_([&]() {
|
|
|
|
StrippedQuery stripped(query_string_);
|
|
|
|
::frontend::opencypher::Parser parser(stripped.query());
|
|
|
|
CypherMainVisitor visitor(context_);
|
|
|
|
visitor.visit(parser.tree());
|
|
|
|
CachedAst cached(std::move(visitor.storage()));
|
2017-06-26 21:42:13 +08:00
|
|
|
return cached.Plug(stripped.literals(), stripped.named_expressions());
|
2017-06-21 22:10:52 +08:00
|
|
|
}()),
|
|
|
|
query_(storage_.query()) {}
|
|
|
|
|
|
|
|
Dbms dbms_;
|
|
|
|
std::unique_ptr<GraphDbAccessor> db_accessor_;
|
|
|
|
Context context_;
|
|
|
|
std::string query_string_;
|
|
|
|
AstTreeStorage storage_;
|
|
|
|
Query *query_;
|
|
|
|
};
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
template <typename T>
|
|
|
|
class CypherMainVisitorTest : public ::testing::Test {};
|
|
|
|
|
|
|
|
typedef ::testing::Types<AstGenerator, OriginalAfterCloningAstGenerator,
|
2017-06-21 22:10:52 +08:00
|
|
|
ClonedAstGenerator, CachedAstGenerator>
|
2017-06-13 17:17:12 +08:00
|
|
|
AstGeneratorTypes;
|
|
|
|
TYPED_TEST_CASE(CypherMainVisitorTest, AstGeneratorTypes);
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, SyntaxException) {
|
2017-07-19 21:48:43 +08:00
|
|
|
ASSERT_THROW(TypeParam("CREATE ()-[*1....2]-()"), SyntaxException);
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, SyntaxExceptionOnTrailingText) {
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 2 + 2 mirko"), SyntaxException);
|
2017-04-05 00:30:10 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, PropertyLookup) {
|
|
|
|
TypeParam ast_generator("RETURN n.x");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *property_lookup = dynamic_cast<PropertyLookup *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(property_lookup->expression_);
|
|
|
|
auto identifier = dynamic_cast<Identifier *>(property_lookup->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
ASSERT_EQ(property_lookup->property_,
|
|
|
|
ast_generator.db_accessor_->property("x"));
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, LabelsTest) {
|
|
|
|
TypeParam ast_generator("RETURN n:x:y");
|
2017-05-05 00:07:54 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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_,
|
|
|
|
ElementsAre(ast_generator.db_accessor_->label("x"),
|
|
|
|
ast_generator.db_accessor_->label("y")));
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, EscapedLabel) {
|
|
|
|
TypeParam ast_generator("RETURN n:`l-$\"'ab``e````l`");
|
2017-05-06 21:14:46 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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_,
|
|
|
|
ElementsAre(ast_generator.db_accessor_->label("l-$\"'ab`e``l")));
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:55:24 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, KeywordLabel) {
|
|
|
|
ASSERT_THROW(TypeParam("RETURN n:DEletE"), SemanticException);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, HexLetterLabel) {
|
|
|
|
TypeParam ast_generator("RETURN n:a");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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");
|
|
|
|
ASSERT_THAT(labels_test->labels_,
|
|
|
|
ElementsAre(ast_generator.db_accessor_->label("a")));
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnNoDistinctNoBagSemantics) {
|
|
|
|
TypeParam ast_generator("RETURN x");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-05-06 00:12:16 +08:00
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
2017-04-21 22:09:49 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnDistinct) {
|
|
|
|
TypeParam ast_generator("RETURN DISTINCT x");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.distinct);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnLimit) {
|
|
|
|
TypeParam ast_generator("RETURN x LIMIT 5");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.limit);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(return_clause->body_.limit);
|
2017-04-21 22:09:49 +08:00
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_EQ(literal->value_.Value<int64_t>(), 5);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnSkip) {
|
|
|
|
TypeParam ast_generator("RETURN x SKIP 5");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.skip);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(return_clause->body_.skip);
|
2017-04-21 22:09:49 +08:00
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_EQ(literal->value_.Value<int64_t>(), 5);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnOrderBy) {
|
|
|
|
TypeParam ast_generator("RETURN x, y, z ORDER BY z ASC, x, y DESC");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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.second);
|
|
|
|
ordering.emplace_back(sort_item.first, identifier->name_);
|
|
|
|
}
|
|
|
|
ASSERT_THAT(ordering, UnorderedElementsAre(Pair(Ordering::ASC, "z"),
|
|
|
|
Pair(Ordering::ASC, "x"),
|
|
|
|
Pair(Ordering::DESC, "y")));
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnNamedIdentifier) {
|
|
|
|
TypeParam ast_generator("RETURN var AS var5");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-05-06 00:12:16 +08:00
|
|
|
ASSERT_FALSE(return_clause->body_.all_identifiers);
|
2017-04-20 19:16:14 +08:00
|
|
|
auto *named_expr = return_clause->body_.named_expressions[0];
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_EQ(named_expr->name_, "var5");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "var");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnAsterisk) {
|
|
|
|
TypeParam ast_generator("RETURN *");
|
2017-05-06 00:12:16 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause->body_.all_identifiers);
|
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 0U);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, IntegerLiteral) {
|
|
|
|
TypeParam ast_generator("RETURN 42");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<int64_t>(), 42);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, IntegerLiteralTooLarge) {
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 10000000000000000000000000"),
|
2017-03-22 18:15:24 +08:00
|
|
|
SemanticException);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, BooleanLiteralTrue) {
|
|
|
|
TypeParam ast_generator("RETURN TrUe");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<bool>(), true);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, BooleanLiteralFalse) {
|
|
|
|
TypeParam ast_generator("RETURN faLSE");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<bool>(), false);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, NullLiteral) {
|
|
|
|
TypeParam ast_generator("RETURN nULl");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.type(), TypedValue::Type::Null);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ParenthesizedExpression) {
|
|
|
|
TypeParam ast_generator("RETURN (2)");
|
2017-04-08 00:32:40 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-08 00:32:40 +08:00
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_EQ(literal->value_.Value<int64_t>(), 2);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, OrOperator) {
|
|
|
|
TypeParam ast_generator("RETURN true Or false oR n");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *or_operator2 = dynamic_cast<OrOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(or_operator2);
|
|
|
|
auto *or_operator1 = dynamic_cast<OrOperator *>(or_operator2->expression1_);
|
|
|
|
ASSERT_TRUE(or_operator1);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(or_operator1->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<bool>(), true);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(or_operator1->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<bool>(), false);
|
|
|
|
auto *operand3 = dynamic_cast<Identifier *>(or_operator2->expression2_);
|
|
|
|
ASSERT_TRUE(operand3);
|
|
|
|
ASSERT_EQ(operand3->name_, "n");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, XorOperator) {
|
|
|
|
TypeParam ast_generator("RETURN true xOr false");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *xor_operator = dynamic_cast<XorOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(xor_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<bool>(), true);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(xor_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<bool>(), false);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, AndOperator) {
|
|
|
|
TypeParam ast_generator("RETURN true and false");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *and_operator = dynamic_cast<AndOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(and_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<bool>(), true);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(and_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<bool>(), false);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, AdditionSubtractionOperators) {
|
|
|
|
TypeParam ast_generator("RETURN 1 - 2 + 3");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *addition_operator = dynamic_cast<AdditionOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(addition_operator);
|
|
|
|
auto *subtraction_operator =
|
|
|
|
dynamic_cast<SubtractionOperator *>(addition_operator->expression1_);
|
|
|
|
ASSERT_TRUE(subtraction_operator);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(subtraction_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 1);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(subtraction_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), 2);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand3 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(addition_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand3);
|
|
|
|
ASSERT_EQ(operand3->value_.Value<int64_t>(), 3);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, MulitplicationOperator) {
|
|
|
|
TypeParam ast_generator("RETURN 2 * 3");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *mult_operator = dynamic_cast<MultiplicationOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(mult_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(mult_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), 3);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, DivisionOperator) {
|
|
|
|
TypeParam ast_generator("RETURN 2 / 3");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *div_operator = dynamic_cast<DivisionOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(div_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(div_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), 3);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ModOperator) {
|
|
|
|
TypeParam ast_generator("RETURN 2 % 3");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *mod_operator = dynamic_cast<ModOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(mod_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(mod_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand2);
|
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), 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); \
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = \
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(cmp_operator->expression1_); \
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), VALUE1); \
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = \
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(cmp_operator->expression2_); \
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), VALUE2); \
|
|
|
|
} while (0)
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ComparisonOperators) {
|
|
|
|
TypeParam ast_generator("RETURN 2 = 3 != 4 <> 5 < 6 > 7 <= 8 >= 9");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-21 22:09:49 +08:00
|
|
|
Expression *_operator =
|
|
|
|
return_clause->body_.named_expressions[0]->expression_;
|
2017-03-23 23:20:06 +08:00
|
|
|
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);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 = dynamic_cast<PrimitiveLiteral *>(cmp_operator->expression1_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand2 = dynamic_cast<PrimitiveLiteral *>(cmp_operator->expression2_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_EQ(operand2->value_.Value<int64_t>(), 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef CHECK_COMPARISON
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ListIndexingOperator) {
|
|
|
|
TypeParam ast_generator("RETURN [1,2,3] [ 2 ]");
|
2017-05-04 00:12:20 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *list_index_op = dynamic_cast<ListIndexingOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(list_index_op);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(list_index_op->expression1_);
|
|
|
|
EXPECT_TRUE(list);
|
|
|
|
auto *index = dynamic_cast<PrimitiveLiteral *>(list_index_op->expression2_);
|
|
|
|
ASSERT_EQ(index->value_.Value<int64_t>(), 2);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ListSlicingOperatorNoBounds) {
|
|
|
|
ASSERT_THROW(TypeParam("RETURN [1,2,3] [ .. ]"), SemanticException);
|
2017-05-04 00:12:20 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ListSlicingOperator) {
|
|
|
|
TypeParam ast_generator("RETURN [1,2,3] [ .. 2 ]");
|
2017-05-04 00:12:20 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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_);
|
|
|
|
auto *upper_bound =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(list_slicing_op->upper_bound_);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<int64_t>(), 2);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, InListOperator) {
|
|
|
|
TypeParam ast_generator("RETURN 5 IN [1,2]");
|
2017-05-07 19:48:34 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *in_list_operator = dynamic_cast<InListOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(in_list_operator);
|
|
|
|
auto *literal =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(in_list_operator->expression1_);
|
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_EQ(literal->value_.Value<int64_t>(), 5);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(in_list_operator->expression2_);
|
|
|
|
ASSERT_TRUE(list);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, InWithListIndexing) {
|
|
|
|
TypeParam ast_generator("RETURN 1 IN [[1,2]][0]");
|
2017-05-18 22:17:19 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *in_list_operator = dynamic_cast<InListOperator *>(
|
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
|
|
|
ASSERT_TRUE(in_list_operator);
|
|
|
|
auto *literal =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(in_list_operator->expression1_);
|
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
EXPECT_EQ(literal->value_.Value<int64_t>(), 1);
|
2017-06-13 17:17:12 +08:00
|
|
|
auto *list_indexing =
|
|
|
|
dynamic_cast<ListIndexingOperator *>(in_list_operator->expression2_);
|
2017-05-18 22:17:19 +08:00
|
|
|
ASSERT_TRUE(list_indexing);
|
|
|
|
auto *list = dynamic_cast<ListLiteral *>(list_indexing->expression1_);
|
|
|
|
EXPECT_TRUE(list);
|
2017-06-13 17:17:12 +08:00
|
|
|
auto *list_index =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(list_indexing->expression2_);
|
2017-05-18 22:17:19 +08:00
|
|
|
EXPECT_TRUE(list_index);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, IsNull) {
|
|
|
|
TypeParam ast_generator("RETURN 2 iS NulL");
|
2017-04-10 22:33:53 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *is_type_operator = dynamic_cast<IsNullOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(is_type_operator->expression_);
|
2017-04-10 22:33:53 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, IsNotNull) {
|
|
|
|
TypeParam ast_generator("RETURN 2 iS nOT NulL");
|
2017-04-10 22:33:53 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *not_operator = dynamic_cast<NotOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-10 22:33:53 +08:00
|
|
|
auto *is_type_operator =
|
|
|
|
dynamic_cast<IsNullOperator *>(not_operator->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand1 =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(is_type_operator->expression_);
|
2017-04-10 22:33:53 +08:00
|
|
|
ASSERT_TRUE(operand1);
|
|
|
|
ASSERT_EQ(operand1->value_.Value<int64_t>(), 2);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, NotOperator) {
|
|
|
|
TypeParam ast_generator("RETURN not true");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *not_operator = dynamic_cast<NotOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand = dynamic_cast<PrimitiveLiteral *>(not_operator->expression_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand);
|
|
|
|
ASSERT_EQ(operand->value_.Value<bool>(), true);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, UnaryMinusPlusOperators) {
|
|
|
|
TypeParam ast_generator("RETURN -+5");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
auto *unary_minus_operator = dynamic_cast<UnaryMinusOperator *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(unary_minus_operator);
|
|
|
|
auto *unary_plus_operator =
|
|
|
|
dynamic_cast<UnaryPlusOperator *>(unary_minus_operator->expression_);
|
|
|
|
ASSERT_TRUE(unary_plus_operator);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *operand =
|
|
|
|
dynamic_cast<PrimitiveLiteral *>(unary_plus_operator->expression_);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(operand);
|
|
|
|
ASSERT_EQ(operand->value_.Value<int64_t>(), 5);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Aggregation) {
|
|
|
|
TypeParam ast_generator(
|
2017-05-19 21:49:25 +08:00
|
|
|
"RETURN COUNT(a), MIN(b), MAX(c), SUM(d), AVG(e), COLLECT(f), COUNT(*)");
|
2017-04-11 20:39:34 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-05-19 21:49:25 +08:00
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 7U);
|
2017-04-11 20:39:34 +08:00
|
|
|
Aggregation::Op ops[] = {Aggregation::Op::COUNT, Aggregation::Op::MIN,
|
2017-05-19 21:49:25 +08:00
|
|
|
Aggregation::Op::MAX, Aggregation::Op::SUM,
|
|
|
|
Aggregation::Op::AVG, Aggregation::Op::COLLECT};
|
|
|
|
std::string ids[] = {"a", "b", "c", "d", "e", "f"};
|
|
|
|
for (int i = 0; i < 6; ++i) {
|
2017-04-11 20:39:34 +08:00
|
|
|
auto *aggregation = dynamic_cast<Aggregation *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[i]->expression_);
|
2017-04-11 20:39:34 +08:00
|
|
|
ASSERT_TRUE(aggregation);
|
|
|
|
ASSERT_EQ(aggregation->op_, ops[i]);
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(aggregation->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
|
|
|
ASSERT_EQ(identifier->name_, ids[i]);
|
|
|
|
}
|
2017-05-06 00:12:16 +08:00
|
|
|
auto *aggregation = dynamic_cast<Aggregation *>(
|
2017-05-19 21:49:25 +08:00
|
|
|
return_clause->body_.named_expressions[6]->expression_);
|
2017-05-06 00:12:16 +08:00
|
|
|
ASSERT_TRUE(aggregation);
|
|
|
|
ASSERT_EQ(aggregation->op_, Aggregation::Op::COUNT);
|
|
|
|
ASSERT_FALSE(aggregation->expression_);
|
2017-04-11 20:39:34 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, UndefinedFunction) {
|
|
|
|
ASSERT_THROW(TypeParam("RETURN "
|
|
|
|
"IHopeWeWillNeverHaveAwesomeMemgraphProcedureWithS"
|
|
|
|
"uchALongAndAwesomeNameSinceThisTestWouldFail(1)"),
|
2017-04-12 01:12:52 +08:00
|
|
|
SemanticException);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Function) {
|
|
|
|
TypeParam ast_generator("RETURN abs(n, 2)");
|
2017-04-12 01:12:52 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-20 19:16:14 +08:00
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1);
|
2017-04-12 01:12:52 +08:00
|
|
|
auto *function = dynamic_cast<Function *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-04-12 01:12:52 +08:00
|
|
|
ASSERT_TRUE(function);
|
|
|
|
ASSERT_TRUE(function->function_);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, StringLiteralDoubleQuotes) {
|
|
|
|
TypeParam ast_generator("RETURN \"mi'rko\"");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<std::string>(), "mi'rko");
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, StringLiteralSingleQuotes) {
|
|
|
|
TypeParam ast_generator("RETURN 'mi\"rko'");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<std::string>(), "mi\"rko");
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, StringLiteralEscapedChars) {
|
|
|
|
TypeParam ast_generator("RETURN '\\\\\\'\\\"\\b\\B\\f\\F\\n\\N\\r\\R\\t\\T'");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<std::string>(), "\\'\"\b\b\f\f\n\n\r\r\t\t");
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, StringLiteralEscapedUtf16) {
|
|
|
|
TypeParam ast_generator("RETURN '\\u221daaa\\U221daaa'");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<std::string>(), u8"\u221daaa\u221daaa");
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, StringLiteralEscapedUtf32) {
|
|
|
|
TypeParam ast_generator("RETURN '\\u0001F600aaaa\\U0001F600aaaaaaaa'");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<std::string>(),
|
2017-03-21 22:10:59 +08:00
|
|
|
u8"\U0001F600aaaa\U0001F600aaaaaaaa");
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, DoubleLiteral) {
|
|
|
|
TypeParam ast_generator("RETURN 3.5");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<double>(), 3.5);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, DoubleLiteralExponent) {
|
|
|
|
TypeParam ast_generator("RETURN 5e-1");
|
2017-03-21 22:10:59 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(
|
2017-04-20 19:16:14 +08:00
|
|
|
return_clause->body_.named_expressions[0]->expression_);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
2017-06-13 22:25:08 +08:00
|
|
|
EXPECT_EQ(literal->value_.Value<double>(), 0.5);
|
|
|
|
EXPECT_EQ(literal->token_position_, 2);
|
2017-03-21 22:10:59 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ListLiteral) {
|
|
|
|
TypeParam ast_generator("RETURN [3, [], 'johhny']");
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(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());
|
|
|
|
auto *elem_0 = dynamic_cast<PrimitiveLiteral *>(list_literal->elements_[0]);
|
|
|
|
ASSERT_TRUE(elem_0);
|
|
|
|
EXPECT_EQ(elem_0->value_.type(), TypedValue::Type::Int);
|
|
|
|
auto *elem_1 = dynamic_cast<ListLiteral *>(list_literal->elements_[1]);
|
|
|
|
ASSERT_TRUE(elem_1);
|
|
|
|
EXPECT_EQ(0, elem_1->elements_.size());
|
|
|
|
auto *elem_2 = dynamic_cast<PrimitiveLiteral *>(list_literal->elements_[2]);
|
|
|
|
ASSERT_TRUE(elem_2);
|
|
|
|
EXPECT_EQ(elem_2->value_.type(), TypedValue::Type::String);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, NodePattern) {
|
|
|
|
TypeParam ast_generator(
|
2017-04-11 18:28:46 +08:00
|
|
|
"MATCH (:label1:label2:label3 {a : 5, b : 10}) RETURN 1");
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
2017-04-11 18:28:46 +08:00
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-17 17:43:18 +08:00
|
|
|
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_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_EQ(node->identifier_->name_,
|
2017-03-17 17:43:18 +08:00
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(1));
|
2017-05-06 00:12:16 +08:00
|
|
|
EXPECT_FALSE(node->identifier_->user_declared_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(node->labels_, UnorderedElementsAre(
|
2017-03-17 17:43:18 +08:00
|
|
|
ast_generator.db_accessor_->label("label1"),
|
|
|
|
ast_generator.db_accessor_->label("label2"),
|
|
|
|
ast_generator.db_accessor_->label("label3")));
|
2017-03-29 18:37:58 +08:00
|
|
|
std::unordered_map<GraphDbTypes::Property, int64_t> properties;
|
2017-03-21 22:10:59 +08:00
|
|
|
for (auto x : node->properties_) {
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(x.second);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_TRUE(literal->value_.type() == TypedValue::Type::Int);
|
|
|
|
properties[x.first] = literal->value_.Value<int64_t>();
|
|
|
|
}
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(properties,
|
2017-03-21 22:10:59 +08:00
|
|
|
UnorderedElementsAre(
|
|
|
|
Pair(ast_generator.db_accessor_->property("a"), 5),
|
|
|
|
Pair(ast_generator.db_accessor_->property("b"), 10)));
|
2017-05-19 22:15:30 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, PropertyMapSameKeyAppearsTwice) {
|
|
|
|
EXPECT_THROW(TypeParam("MATCH ({a : 1, a : 2})"), SemanticException);
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, NodePatternIdentifier) {
|
|
|
|
TypeParam ast_generator("MATCH (var) RETURN 1");
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-17 17:43:18 +08:00
|
|
|
auto node = dynamic_cast<NodeAtom *>(match->patterns_[0]->atoms_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(node);
|
2017-03-17 17:43:18 +08:00
|
|
|
ASSERT_TRUE(node->identifier_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_EQ(node->identifier_->name_, "var");
|
2017-05-06 00:12:16 +08:00
|
|
|
EXPECT_TRUE(node->identifier_->user_declared_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(node->labels_, UnorderedElementsAre());
|
|
|
|
EXPECT_THAT(node->properties_, UnorderedElementsAre());
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternNoDetails) {
|
|
|
|
TypeParam ast_generator("MATCH ()--() RETURN 1");
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-17 17:43:18 +08:00
|
|
|
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);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::BOTH);
|
2017-03-17 17:43:18 +08:00
|
|
|
ASSERT_TRUE(edge->identifier_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(edge->identifier_->name_,
|
2017-03-17 17:43:18 +08:00
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(2));
|
2017-05-06 00:12:16 +08:00
|
|
|
EXPECT_FALSE(edge->identifier_->user_declared_);
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
|
|
|
|
2017-03-23 23:20:06 +08:00
|
|
|
// PatternPart in braces.
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, PatternPartBraces) {
|
|
|
|
TypeParam ast_generator("MATCH ((()--())) RETURN 1");
|
2017-03-23 23:20:06 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-23 23:20:06 +08:00
|
|
|
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);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::BOTH);
|
2017-03-23 23:20:06 +08:00
|
|
|
ASSERT_TRUE(edge->identifier_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(edge->identifier_->name_,
|
2017-03-23 23:20:06 +08:00
|
|
|
CypherMainVisitor::kAnonPrefix + std::to_string(2));
|
2017-05-06 00:12:16 +08:00
|
|
|
EXPECT_FALSE(edge->identifier_->user_declared_);
|
2017-03-23 23:20:06 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternDetails) {
|
|
|
|
TypeParam ast_generator(
|
2017-04-11 18:28:46 +08:00
|
|
|
"MATCH ()<-[:type1|type2 {a : 5, b : 10}]-() RETURN 1");
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(edge);
|
2017-05-30 15:53:27 +08:00
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::IN);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(
|
2017-03-21 18:25:44 +08:00
|
|
|
edge->edge_types_,
|
2017-03-17 17:43:18 +08:00
|
|
|
UnorderedElementsAre(ast_generator.db_accessor_->edge_type("type1"),
|
|
|
|
ast_generator.db_accessor_->edge_type("type2")));
|
2017-03-29 18:37:58 +08:00
|
|
|
std::unordered_map<GraphDbTypes::Property, int64_t> properties;
|
2017-03-21 22:10:59 +08:00
|
|
|
for (auto x : edge->properties_) {
|
2017-04-28 18:06:18 +08:00
|
|
|
auto *literal = dynamic_cast<PrimitiveLiteral *>(x.second);
|
2017-03-21 22:10:59 +08:00
|
|
|
ASSERT_TRUE(literal);
|
|
|
|
ASSERT_TRUE(literal->value_.type() == TypedValue::Type::Int);
|
|
|
|
properties[x.first] = literal->value_.Value<int64_t>();
|
|
|
|
}
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(properties,
|
2017-03-21 22:10:59 +08:00
|
|
|
UnorderedElementsAre(
|
|
|
|
Pair(ast_generator.db_accessor_->property("a"), 5),
|
|
|
|
Pair(ast_generator.db_accessor_->property("b"), 10)));
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternVariable) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[var]->() RETURN 1");
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(match);
|
|
|
|
EXPECT_FALSE(match->optional_);
|
|
|
|
EXPECT_FALSE(match->where_);
|
2017-03-17 17:43:18 +08:00
|
|
|
auto *edge = dynamic_cast<EdgeAtom *>(match->patterns_[0]->atoms_[1]);
|
2017-04-27 16:42:53 +08:00
|
|
|
ASSERT_TRUE(edge);
|
2017-05-30 15:53:27 +08:00
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
2017-03-17 17:43:18 +08:00
|
|
|
ASSERT_TRUE(edge->identifier_);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_THAT(edge->identifier_->name_, "var");
|
2017-05-06 00:12:16 +08:00
|
|
|
EXPECT_TRUE(edge->identifier_->user_declared_);
|
2017-03-17 17:43:18 +08:00
|
|
|
}
|
2017-02-24 00:15:31 +08:00
|
|
|
|
2017-07-19 21:48:43 +08:00
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternUnbounded) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternLowerBounded) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*42..]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
auto *lower_bound = dynamic_cast<PrimitiveLiteral *>(edge->lower_bound_);
|
|
|
|
ASSERT_TRUE(lower_bound);
|
|
|
|
EXPECT_TRUE(lower_bound->value_.Value<int64_t>() == 42);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternUpperBounded) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*..42]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
auto upper_bound = dynamic_cast<PrimitiveLiteral *>(edge->upper_bound_);
|
|
|
|
ASSERT_TRUE(upper_bound);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternLowerUpperBounded) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*24..42]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
auto lower_bound = dynamic_cast<PrimitiveLiteral *>(edge->lower_bound_);
|
|
|
|
ASSERT_TRUE(lower_bound);
|
|
|
|
EXPECT_EQ(lower_bound->value_.Value<int64_t>(), 24);
|
|
|
|
auto upper_bound = dynamic_cast<PrimitiveLiteral *>(edge->upper_bound_);
|
|
|
|
ASSERT_TRUE(upper_bound);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternFixedRange) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*42]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
auto lower_bound = dynamic_cast<PrimitiveLiteral *>(edge->lower_bound_);
|
|
|
|
ASSERT_TRUE(lower_bound);
|
|
|
|
EXPECT_EQ(lower_bound->value_.Value<int64_t>(), 42);
|
|
|
|
auto upper_bound = dynamic_cast<PrimitiveLiteral *>(edge->upper_bound_);
|
|
|
|
ASSERT_TRUE(upper_bound);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternFloatingUpperBound) {
|
|
|
|
// [r*1...2] should be parsed as [r*1..0.2]
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*1...2]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
auto lower_bound = dynamic_cast<PrimitiveLiteral *>(edge->lower_bound_);
|
|
|
|
ASSERT_TRUE(lower_bound);
|
|
|
|
EXPECT_EQ(lower_bound->value_.Value<int64_t>(), 1);
|
|
|
|
auto upper_bound = dynamic_cast<PrimitiveLiteral *>(edge->upper_bound_);
|
|
|
|
ASSERT_TRUE(upper_bound);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<double>(), 0.2);
|
|
|
|
}
|
|
|
|
|
2017-07-20 21:31:20 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternUnboundedWithProperty) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r* {prop: 42}]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
auto prop = ast_generator.db_accessor_->property("prop");
|
|
|
|
auto prop_literal = dynamic_cast<PrimitiveLiteral *>(edge->properties_[prop]);
|
|
|
|
EXPECT_EQ(prop_literal->value_.Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest,
|
|
|
|
RelationshipPatternDotsUnboundedWithEdgeTypeProperty) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r:edge_type*..{prop: 42}]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
EXPECT_EQ(edge->upper_bound_, nullptr);
|
|
|
|
auto prop = ast_generator.db_accessor_->property("prop");
|
|
|
|
auto prop_literal = dynamic_cast<PrimitiveLiteral *>(edge->properties_[prop]);
|
|
|
|
EXPECT_EQ(prop_literal->value_.Value<int64_t>(), 42);
|
|
|
|
ASSERT_EQ(edge->edge_types_.size(), 1U);
|
|
|
|
auto edge_type = ast_generator.db_accessor_->edge_type("edge_type");
|
|
|
|
EXPECT_EQ(edge->edge_types_[0], edge_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, RelationshipPatternUpperBoundedWithProperty) {
|
|
|
|
TypeParam ast_generator("MATCH ()-[r*..2{prop: 42}]->() RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
EdgeAtom *edge = nullptr;
|
|
|
|
AssertMatchSingleEdgeAtom(match, edge);
|
|
|
|
EXPECT_EQ(edge->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_TRUE(edge->has_range_);
|
|
|
|
EXPECT_EQ(edge->lower_bound_, nullptr);
|
|
|
|
auto upper_bound = dynamic_cast<PrimitiveLiteral *>(edge->upper_bound_);
|
|
|
|
ASSERT_TRUE(upper_bound);
|
|
|
|
EXPECT_EQ(upper_bound->value_.Value<int64_t>(), 2);
|
|
|
|
auto prop = ast_generator.db_accessor_->property("prop");
|
|
|
|
auto prop_literal = dynamic_cast<PrimitiveLiteral *>(edge->properties_[prop]);
|
|
|
|
EXPECT_EQ(prop_literal->value_.Value<int64_t>(), 42);
|
|
|
|
}
|
|
|
|
|
2017-03-12 07:04:10 +08:00
|
|
|
// // PatternPart with variable.
|
2017-06-13 17:17:12 +08:00
|
|
|
// TYPED_TEST(CypherMainVisitorTest, PatternPartVariable) {
|
2017-03-12 07:04:10 +08:00
|
|
|
// ParserTables parser("CREATE var=()--()");
|
|
|
|
// ASSERT_EQ(parser.identifiers_map_.size(), 1U);
|
|
|
|
// ASSERT_EQ(parser.pattern_parts_.size(), 1U);
|
|
|
|
// ASSERT_EQ(parser.relationships_.size(), 1U);
|
|
|
|
// ASSERT_EQ(parser.nodes_.size(), 2U);
|
|
|
|
// ASSERT_EQ(parser.pattern_parts_.begin()->second.nodes.size(), 2U);
|
|
|
|
// ASSERT_EQ(parser.pattern_parts_.begin()->second.relationships.size(), 1U);
|
2017-03-17 17:43:18 +08:00
|
|
|
// ASSERT_NE(parser.identifiers_map_.find("var"),
|
|
|
|
// parser.identifiers_map_.end());
|
2017-03-12 07:04:10 +08:00
|
|
|
// auto output_identifier = parser.identifiers_map_["var"];
|
|
|
|
// ASSERT_NE(parser.pattern_parts_.find(output_identifier),
|
|
|
|
// parser.pattern_parts_.end());
|
|
|
|
// }
|
2017-03-17 23:29:27 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnUnanemdIdentifier) {
|
|
|
|
TypeParam ast_generator("RETURN var");
|
2017-03-17 23:29:27 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *return_clause = dynamic_cast<Return *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(return_clause);
|
2017-04-20 19:16:14 +08:00
|
|
|
ASSERT_EQ(return_clause->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = return_clause->body_.named_expressions[0];
|
2017-03-17 23:29:27 +08:00
|
|
|
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");
|
2017-05-06 00:12:16 +08:00
|
|
|
ASSERT_TRUE(identifier->user_declared_);
|
2017-03-17 23:29:27 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Create) {
|
|
|
|
TypeParam ast_generator("CREATE (n)");
|
2017-03-17 23:29:27 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *create = dynamic_cast<Create *>(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");
|
|
|
|
}
|
2017-03-27 16:04:37 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Delete) {
|
|
|
|
TypeParam ast_generator("DELETE n, m");
|
2017-03-27 16:04:37 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *del = dynamic_cast<Delete *>(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");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, DeleteDetach) {
|
|
|
|
TypeParam ast_generator("DETACH DELETE n");
|
2017-03-27 16:04:37 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *del = dynamic_cast<Delete *>(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");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, OptionalMatchWhere) {
|
|
|
|
TypeParam ast_generator("OPTIONAL MATCH (n) WHERE m RETURN 1");
|
2017-03-27 16:04:37 +08:00
|
|
|
auto *query = ast_generator.query_;
|
2017-04-11 18:28:46 +08:00
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
2017-03-27 16:04:37 +08:00
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
2017-04-27 16:42:53 +08:00
|
|
|
EXPECT_TRUE(match->optional_);
|
2017-03-27 16:04:37 +08:00
|
|
|
ASSERT_TRUE(match->where_);
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(match->where_->expression_);
|
|
|
|
ASSERT_TRUE(identifier);
|
2017-04-11 18:28:46 +08:00
|
|
|
ASSERT_EQ(identifier->name_, "m");
|
2017-03-27 16:04:37 +08:00
|
|
|
}
|
2017-03-28 16:20:20 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Set) {
|
|
|
|
TypeParam ast_generator("SET a.x = b, c = d, e += f, g : h : i ");
|
2017-03-28 16:20:20 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 4U);
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_property = dynamic_cast<SetProperty *>(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_,
|
|
|
|
ast_generator.db_accessor_->property("x"));
|
|
|
|
auto *identifier2 = dynamic_cast<Identifier *>(set_property->expression_);
|
|
|
|
ASSERT_EQ(identifier2->name_, "b");
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *set_properties_assignment =
|
|
|
|
dynamic_cast<SetProperties *>(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 *>(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 *>(query->clauses_[3]);
|
|
|
|
ASSERT_TRUE(set_labels);
|
|
|
|
ASSERT_TRUE(set_labels->identifier_);
|
|
|
|
ASSERT_EQ(set_labels->identifier_->name_, "g");
|
|
|
|
ASSERT_THAT(set_labels->labels_,
|
|
|
|
UnorderedElementsAre(ast_generator.db_accessor_->label("h"),
|
|
|
|
ast_generator.db_accessor_->label("i")));
|
|
|
|
}
|
|
|
|
}
|
2017-03-30 14:38:48 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Remove) {
|
|
|
|
TypeParam ast_generator("REMOVE a.x, g : h : i");
|
2017-03-30 14:38:48 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *remove_property = dynamic_cast<RemoveProperty *>(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_,
|
|
|
|
ast_generator.db_accessor_->property("x"));
|
|
|
|
}
|
|
|
|
{
|
|
|
|
auto *remove_labels = dynamic_cast<RemoveLabels *>(query->clauses_[1]);
|
|
|
|
ASSERT_TRUE(remove_labels);
|
|
|
|
ASSERT_TRUE(remove_labels->identifier_);
|
|
|
|
ASSERT_EQ(remove_labels->identifier_->name_, "g");
|
|
|
|
ASSERT_THAT(remove_labels->labels_,
|
|
|
|
UnorderedElementsAre(ast_generator.db_accessor_->label("h"),
|
|
|
|
ast_generator.db_accessor_->label("i")));
|
|
|
|
}
|
|
|
|
}
|
2017-04-05 21:26:27 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, With) {
|
|
|
|
TypeParam ast_generator("WITH n AS m RETURN 1");
|
2017-04-05 21:26:27 +08:00
|
|
|
auto *query = ast_generator.query_;
|
2017-04-11 18:28:46 +08:00
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
2017-04-05 21:26:27 +08:00
|
|
|
auto *with = dynamic_cast<With *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(with);
|
2017-04-21 22:09:49 +08:00
|
|
|
ASSERT_FALSE(with->body_.distinct);
|
|
|
|
ASSERT_FALSE(with->body_.limit);
|
|
|
|
ASSERT_FALSE(with->body_.skip);
|
|
|
|
ASSERT_EQ(with->body_.order_by.size(), 0U);
|
2017-04-05 21:26:27 +08:00
|
|
|
ASSERT_FALSE(with->where_);
|
2017-04-20 19:16:14 +08:00
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
2017-04-05 21:26:27 +08:00
|
|
|
ASSERT_EQ(named_expr->name_, "m");
|
|
|
|
auto *identifier = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier->name_, "n");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, WithNonAliasedExpression) {
|
|
|
|
ASSERT_THROW(TypeParam("WITH n.x RETURN 1"), SemanticException);
|
2017-04-21 23:06:55 +08:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, WithNonAliasedVariable) {
|
|
|
|
TypeParam ast_generator("WITH n RETURN 1");
|
2017-04-21 23:06:55 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(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");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, WithDistinct) {
|
|
|
|
TypeParam ast_generator("WITH DISTINCT n AS m RETURN 1");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(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");
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, WithBag) {
|
|
|
|
TypeParam ast_generator("WITH n as m ORDER BY m SKIP 1 LIMIT 2 RETURN 1");
|
2017-04-21 22:09:49 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
auto *with = dynamic_cast<With *>(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);
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, WithWhere) {
|
|
|
|
TypeParam ast_generator("WITH n AS m WHERE k RETURN 1");
|
2017-04-05 21:26:27 +08:00
|
|
|
auto *query = ast_generator.query_;
|
2017-04-11 18:28:46 +08:00
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
2017-04-05 21:26:27 +08:00
|
|
|
auto *with = dynamic_cast<With *>(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");
|
2017-04-20 19:16:14 +08:00
|
|
|
ASSERT_EQ(with->body_.named_expressions.size(), 1U);
|
|
|
|
auto *named_expr = with->body_.named_expressions[0];
|
2017-04-05 21:26:27 +08:00
|
|
|
ASSERT_EQ(named_expr->name_, "m");
|
|
|
|
auto *identifier2 = dynamic_cast<Identifier *>(named_expr->expression_);
|
|
|
|
ASSERT_EQ(identifier2->name_, "n");
|
|
|
|
}
|
2017-04-11 18:28:46 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, ClausesOrdering) {
|
2017-04-11 18:28:46 +08:00
|
|
|
// Obviously some of the ridiculous combinations don't fail here, but they
|
2017-04-24 22:01:00 +08:00
|
|
|
// will fail in semantic analysis or they make perfect sense as a part of
|
2017-04-11 18:28:46 +08:00
|
|
|
// bigger query.
|
2017-06-13 17:17:12 +08:00
|
|
|
TypeParam("RETURN 1");
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 RETURN 1"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 MATCH (n) RETURN n"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 DELETE n"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 MERGE (n)"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 WITH n AS m RETURN 1"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("RETURN 1 AS n UNWIND n AS x RETURN x"),
|
2017-05-03 17:07:07 +08:00
|
|
|
SemanticException);
|
2017-04-11 18:28:46 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
ASSERT_THROW(TypeParam("OPTIONAL MATCH (n) MATCH (m) RETURN n, m"),
|
2017-05-19 22:01:01 +08:00
|
|
|
SemanticException);
|
2017-06-13 17:17:12 +08:00
|
|
|
TypeParam("OPTIONAL MATCH (n) WITH n MATCH (m) RETURN n, m");
|
|
|
|
TypeParam("OPTIONAL MATCH (n) OPTIONAL MATCH (m) RETURN n, m");
|
|
|
|
TypeParam("MATCH (n) OPTIONAL MATCH (m) RETURN n, m");
|
|
|
|
|
|
|
|
TypeParam("CREATE (n)");
|
|
|
|
ASSERT_THROW(TypeParam("SET n:x MATCH (n) RETURN n"), SemanticException);
|
|
|
|
TypeParam("REMOVE n.x SET n.x = 1");
|
|
|
|
TypeParam("REMOVE n:L RETURN n");
|
|
|
|
TypeParam("SET n.x = 1 WITH n AS m RETURN m");
|
|
|
|
|
|
|
|
ASSERT_THROW(TypeParam("MATCH (n)"), SemanticException);
|
|
|
|
TypeParam("MATCH (n) MATCH (n) RETURN n");
|
|
|
|
TypeParam("MATCH (n) SET n = m");
|
|
|
|
TypeParam("MATCH (n) RETURN n");
|
|
|
|
TypeParam("MATCH (n) WITH n AS m RETURN m");
|
|
|
|
|
|
|
|
ASSERT_THROW(TypeParam("WITH 1 AS n"), SemanticException);
|
|
|
|
TypeParam("WITH 1 AS n WITH n AS m RETURN m");
|
|
|
|
TypeParam("WITH 1 AS n RETURN n");
|
|
|
|
TypeParam("WITH 1 AS n SET n += m");
|
|
|
|
TypeParam("WITH 1 AS n MATCH (n) RETURN n");
|
|
|
|
|
|
|
|
ASSERT_THROW(TypeParam("UNWIND [1,2,3] AS x"), SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("CREATE (n) UNWIND [1,2,3] AS x RETURN x"),
|
2017-05-03 17:07:07 +08:00
|
|
|
SemanticException);
|
2017-06-13 17:17:12 +08:00
|
|
|
TypeParam("UNWIND [1,2,3] AS x CREATE (n) RETURN x");
|
|
|
|
TypeParam("CREATE (n) WITH n UNWIND [1,2,3] AS x RETURN x");
|
2017-07-06 19:03:14 +08:00
|
|
|
|
|
|
|
TypeParam("CREATE INDEX ON :a(b)");
|
|
|
|
ASSERT_THROW(TypeParam("CREATE INDEX ON :a(n) CREATE INDEX ON :b(c)"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("CREATE (n) CREATE INDEX ON :a(n)"),
|
|
|
|
SemanticException);
|
|
|
|
ASSERT_THROW(TypeParam("CREATE INDEX ON :a(n) RETURN 2 + 2"),
|
|
|
|
SemanticException);
|
2017-04-11 18:28:46 +08:00
|
|
|
}
|
2017-04-25 19:44:57 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Merge) {
|
|
|
|
TypeParam ast_generator(
|
2017-04-25 19:44:57 +08:00
|
|
|
"MERGE (a) -[:r]- (b) ON MATCH SET a.x = b.x "
|
|
|
|
"ON CREATE SET b :label ON MATCH SET b = a");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *merge = dynamic_cast<Merge *>(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]));
|
|
|
|
}
|
2017-05-03 17:07:07 +08:00
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, Unwind) {
|
|
|
|
TypeParam ast_generator("UNWIND [1,2,3] AS elem RETURN elem");
|
2017-05-03 17:07:07 +08:00
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
auto *unwind = dynamic_cast<Unwind *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(unwind);
|
|
|
|
auto *ret = dynamic_cast<Return *>(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));
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:17:12 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, UnwindWithoutAsError) {
|
|
|
|
EXPECT_THROW(TypeParam("UNWIND [1,2,3] RETURN 42"), SyntaxException);
|
2017-05-03 17:07:07 +08:00
|
|
|
}
|
2017-07-06 19:03:14 +08:00
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, CreateIndex) {
|
|
|
|
TypeParam ast_generator("Create InDeX oN :mirko(slavko)");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *create_index = dynamic_cast<CreateIndex *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(create_index);
|
|
|
|
ASSERT_EQ(create_index->label_, ast_generator.db_accessor_->label("mirko"));
|
|
|
|
ASSERT_EQ(create_index->property_,
|
|
|
|
ast_generator.db_accessor_->property("slavko"));
|
|
|
|
}
|
2017-07-25 19:01:08 +08:00
|
|
|
|
|
|
|
TYPED_TEST(CypherMainVisitorTest, ReturnAll) {
|
|
|
|
TypeParam ast_generator("RETURN all(x IN [1,2,3] WHERE x = 2)");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 1U);
|
|
|
|
auto *ret = dynamic_cast<Return *>(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);
|
|
|
|
}
|
|
|
|
|
2017-07-30 05:37:46 +08:00
|
|
|
TYPED_TEST(CypherMainVisitorTest, MatchBfsReturn) {
|
|
|
|
TypeParam ast_generator(
|
|
|
|
"MATCH (n) -bfs[r](e, n|e.prop = 42, 10)-> (m) RETURN r");
|
|
|
|
auto *query = ast_generator.query_;
|
|
|
|
ASSERT_EQ(query->clauses_.size(), 2U);
|
|
|
|
auto *match = dynamic_cast<Match *>(query->clauses_[0]);
|
|
|
|
ASSERT_TRUE(match);
|
|
|
|
ASSERT_EQ(match->patterns_.size(), 1U);
|
|
|
|
ASSERT_EQ(match->patterns_[0]->atoms_.size(), 3U);
|
|
|
|
auto *bfs = dynamic_cast<BreadthFirstAtom *>(match->patterns_[0]->atoms_[1]);
|
|
|
|
ASSERT_TRUE(bfs);
|
|
|
|
EXPECT_EQ(bfs->direction_, EdgeAtom::Direction::OUT);
|
|
|
|
EXPECT_EQ(bfs->identifier_->name_, "r");
|
|
|
|
EXPECT_EQ(bfs->traversed_edge_identifier_->name_, "e");
|
|
|
|
EXPECT_EQ(bfs->next_node_identifier_->name_, "n");
|
|
|
|
auto *max_depth = dynamic_cast<PrimitiveLiteral *>(bfs->max_depth_);
|
|
|
|
ASSERT_TRUE(max_depth);
|
|
|
|
EXPECT_EQ(max_depth->value_.Value<int64_t>(), 10U);
|
|
|
|
auto *eq = dynamic_cast<EqualOperator *>(bfs->filter_expression_);
|
|
|
|
ASSERT_TRUE(eq);
|
|
|
|
}
|
|
|
|
|
2017-02-24 00:15:31 +08:00
|
|
|
}
|