2018-10-04 21:23:07 +08:00
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
2018-08-16 16:13:04 +08:00
|
|
|
|
|
|
|
#include "query/frontend/semantic/required_privileges.hpp"
|
2018-11-07 01:15:55 +08:00
|
|
|
#include "storage/common/types/types.hpp"
|
2018-08-16 16:13:04 +08:00
|
|
|
|
|
|
|
#include "query_common.hpp"
|
|
|
|
|
|
|
|
using namespace query;
|
|
|
|
|
|
|
|
class FakeDbAccessor {};
|
|
|
|
|
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
|
|
|
const std::string EDGE_TYPE = "0";
|
|
|
|
const std::string LABEL_0 = "label0";
|
|
|
|
const std::string LABEL_1 = "label1";
|
|
|
|
const std::string PROP_0 = "prop0";
|
2018-08-16 16:13:04 +08:00
|
|
|
|
|
|
|
using ::testing::UnorderedElementsAre;
|
|
|
|
|
|
|
|
class TestPrivilegeExtractor : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
AstStorage storage;
|
|
|
|
FakeDbAccessor dba;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, CreateNode) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(CREATE(PATTERN(NODE("n")))));
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::CREATE));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeDelete) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), DELETE(IDENT("n"))));
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::DELETE));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeReturn) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), RETURN("n")));
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchCreateExpand) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(SINGLE_QUERY(
|
2018-08-16 16:13:04 +08:00
|
|
|
MATCH(PATTERN(NODE("n"))),
|
|
|
|
CREATE(PATTERN(NODE("n"),
|
|
|
|
EDGE("r", EdgeAtom::Direction::OUT, {EDGE_TYPE}),
|
|
|
|
NODE("m")))));
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::CREATE));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeSetLabels) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(
|
|
|
|
SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), SET("n", {LABEL_0, LABEL_1})));
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::SET));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeSetProperty) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(
|
|
|
|
SINGLE_QUERY(MATCH(PATTERN(NODE("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
|
|
|
SET(PROPERTY_LOOKUP(storage.Create<Identifier>("n"), PROP_0),
|
|
|
|
LITERAL(42))));
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::SET));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeSetProperties) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query =
|
|
|
|
QUERY(SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), SET("n", LIST())));
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::SET));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeRemoveLabels) {
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = QUERY(
|
2018-08-16 16:13:04 +08:00
|
|
|
SINGLE_QUERY(MATCH(PATTERN(NODE("n"))), REMOVE("n", {LABEL_0, LABEL_1})));
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::REMOVE));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, MatchNodeRemoveProperty) {
|
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 *query = QUERY(SINGLE_QUERY(
|
|
|
|
MATCH(PATTERN(NODE("n"))),
|
|
|
|
REMOVE(PROPERTY_LOOKUP(storage.Create<Identifier>("n"), PROP_0))));
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::MATCH,
|
|
|
|
AuthQuery::Privilege::REMOVE));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, CreateIndex) {
|
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 *query = CREATE_INDEX_ON(storage.GetLabelIx(LABEL_0),
|
|
|
|
storage.GetPropertyIx(PROP_0));
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::INDEX));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, AuthQuery) {
|
2018-10-19 22:18:44 +08:00
|
|
|
auto *query = AUTH_QUERY(AuthQuery::Action::CREATE_ROLE, "", "role", "",
|
|
|
|
nullptr, std::vector<AuthQuery::Privilege>{});
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::AUTH));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, StreamQuery) {
|
|
|
|
std::string stream_name("kafka");
|
|
|
|
std::string stream_uri("localhost:1234");
|
|
|
|
std::string stream_topic("tropik");
|
|
|
|
std::string transform_uri("localhost:1234/file.py");
|
|
|
|
|
2018-10-19 22:18:44 +08:00
|
|
|
std::vector<StreamQuery *> stream_queries = {
|
2018-08-16 16:13:04 +08:00
|
|
|
CREATE_STREAM(stream_name, stream_uri, stream_topic, transform_uri,
|
|
|
|
nullptr, nullptr),
|
|
|
|
DROP_STREAM(stream_name),
|
|
|
|
SHOW_STREAMS,
|
|
|
|
START_STREAM(stream_name, nullptr),
|
|
|
|
STOP_STREAM(stream_name),
|
|
|
|
START_ALL_STREAMS,
|
|
|
|
STOP_ALL_STREAMS};
|
|
|
|
|
2018-10-19 22:18:44 +08:00
|
|
|
for (auto *query : stream_queries) {
|
2018-10-10 21:19:34 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
2018-08-16 16:13:04 +08:00
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::STREAM));
|
|
|
|
}
|
|
|
|
}
|
2019-02-19 20:31:46 +08:00
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, ShowIndexInfo) {
|
|
|
|
auto *query = storage.Create<InfoQuery>();
|
|
|
|
query->info_type_ = InfoQuery::InfoType::INDEX;
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::INDEX));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, ShowStatsInfo) {
|
|
|
|
auto *query = storage.Create<InfoQuery>();
|
|
|
|
query->info_type_ = InfoQuery::InfoType::STORAGE;
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::STATS));
|
|
|
|
}
|
|
|
|
|
2019-02-22 22:02:31 +08:00
|
|
|
TEST_F(TestPrivilegeExtractor, ShowConstraintInfo) {
|
|
|
|
auto *query = storage.Create<InfoQuery>();
|
|
|
|
query->info_type_ = InfoQuery::InfoType::CONSTRAINT;
|
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::CONSTRAINT));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, CreateConstraint) {
|
|
|
|
auto *query = storage.Create<ConstraintQuery>();
|
|
|
|
query->action_type_ = ConstraintQuery::ActionType::CREATE;
|
2019-03-28 19:49:23 +08:00
|
|
|
query->constraint_.label = storage.GetLabelIx("label");
|
|
|
|
query->constraint_.properties.push_back(storage.GetPropertyIx("prop0"));
|
|
|
|
query->constraint_.properties.push_back(storage.GetPropertyIx("prop1"));
|
2019-02-22 22:02:31 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::CONSTRAINT));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TestPrivilegeExtractor, DropConstraint) {
|
|
|
|
auto *query = storage.Create<ConstraintQuery>();
|
|
|
|
query->action_type_ = ConstraintQuery::ActionType::DROP;
|
2019-03-28 19:49:23 +08:00
|
|
|
query->constraint_.label = storage.GetLabelIx("label");
|
|
|
|
query->constraint_.properties.push_back(storage.GetPropertyIx("prop0"));
|
|
|
|
query->constraint_.properties.push_back(storage.GetPropertyIx("prop1"));
|
2019-02-22 22:02:31 +08:00
|
|
|
EXPECT_THAT(GetRequiredPrivileges(query),
|
|
|
|
UnorderedElementsAre(AuthQuery::Privilege::CONSTRAINT));
|
|
|
|
}
|