memgraph/tests/e2e/module_file_manager/module_file_manager.cpp
andrejtonev e8850549d2
Add multi-tenancy v1 (#952)
* Decouple BoltSession and communication::bolt::Session
* Add CREATE/USE/DROP DATABASE
* Add SHOW DATABASES
* Cover WebSocket session
* Simple session safety implemented via RWLock
* Storage symlinks for backward. compatibility
* Extend the audit log with the DB info
* Add auth part
* Add tenant recovery
2023-08-01 18:49:11 +02:00

281 lines
12 KiB
C++

// Copyright 2023 Memgraph Ltd.
//
// Use of this software is governed by the Business Source License
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
// License, and you may not use this file except in compliance with the Business Source License.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.
#include <filesystem>
#include <fstream>
#include <gflags/gflags.h>
#include <mgclient.hpp>
#include "utils/file.hpp"
#include "utils/logging.hpp"
#include "utils/timer.hpp"
DEFINE_uint64(bolt_port, 7687, "Bolt port");
DEFINE_uint64(timeout, 120, "Timeout seconds");
DEFINE_bool(multi_db, false, "Run test in multi db environment");
namespace {
auto GetClient() {
auto client =
mg::Client::Connect({.host = "127.0.0.1", .port = static_cast<uint16_t>(FLAGS_bolt_port), .use_ssl = false});
MG_ASSERT(client, "Failed to connect!");
return client;
}
std::vector<std::filesystem::path> GetModuleFiles(auto &client) {
MG_ASSERT(client->Execute("CALL mg.get_module_files() YIELD path"));
const auto result_rows = client->FetchAll();
MG_ASSERT(result_rows, "Failed to get results");
std::vector<std::filesystem::path> result;
result.reserve(result_rows->size());
for (const auto &row : *result_rows) {
MG_ASSERT(row.size() == 1, "Invalid result received from mg.get_module_files");
result.emplace_back(row[0].ValueString());
}
return result;
}
bool ModuleFileExists(auto &client, const auto &path) {
const auto module_files = GetModuleFiles(client);
return std::any_of(module_files.begin(), module_files.end(),
[&](const auto &module_file) { return module_file == path; });
}
void AssertModuleFileExists(auto &client, const auto &path) {
MG_ASSERT(ModuleFileExists(client, path), "Module file {} is missing", path);
}
void AssertModuleFileNotExists(auto &client, const auto &path) {
MG_ASSERT(!ModuleFileExists(client, path), "Invalid module file {} is present", path);
}
bool ProcedureExists(auto &client, const std::string_view procedure_name,
std::optional<std::filesystem::path> path = std::nullopt) {
MG_ASSERT(client->Execute("CALL mg.procedures() YIELD name, path"));
const auto result_rows = client->FetchAll();
MG_ASSERT(result_rows, "Failed to get results for mg.procedures()");
return std::find_if(result_rows->begin(), result_rows->end(), [&, procedure_name](const auto &row) {
MG_ASSERT(row.size() == 2, "Invalid result received from mg.procedures()");
if (row[0].ValueString() == procedure_name) {
if (path) {
return row[1].ValueString() == std::filesystem::canonical(*path).generic_string();
}
return true;
}
return false;
}) != result_rows->end();
}
void AssertProcedureExists(auto &client, const std::string_view procedure_name,
std::optional<std::filesystem::path> path = std::nullopt) {
MG_ASSERT(ProcedureExists(client, procedure_name, path), "Procedure {} is missing", procedure_name);
}
void AssertProcedureNotExists(auto &client, const std::string_view procedure_name) {
MG_ASSERT(!ProcedureExists(client, procedure_name), "Invalid procedure ('{}') is present", procedure_name);
}
template <typename TException>
void AssertQueryFails(auto &client, const std::string &query, std::optional<std::string> expected_message) {
spdlog::info("Asserting query '{}' fails", query);
MG_ASSERT(client->Execute(query));
try {
client->FetchAll();
} catch (const TException &exception) {
if (expected_message) {
MG_ASSERT(*expected_message == exception.what(),
"Exception with a different message was thrown.\n\t\tExpected: {}\n\t\tActual: {}", *expected_message,
exception.what());
}
return;
}
LOG_FATAL("Didn't receive expected exception");
}
std::string CreateModuleFileQuery(const std::string_view filename, const std::string_view content) {
return fmt::format("CALL mg.create_module_file('{}', '{}') YIELD path", filename, content);
}
std::filesystem::path CreateModuleFile(auto &client, const std::string_view filename, const std::string_view content) {
spdlog::info("Creating module file '{}' with content:\n{}", filename, content);
MG_ASSERT(client->Execute(CreateModuleFileQuery(filename, content)));
const auto result_row = client->FetchOne();
MG_ASSERT(result_row && result_row->size() == 1, "Received invalid result from mg.create_module_file");
MG_ASSERT(!client->FetchOne().has_value(), "Too many results received from mg.create_module_file");
return result_row->at(0).ValueString();
}
std::string GetModuleFileQuery(const std::filesystem::path &path) {
return fmt::format("CALL mg.get_module_file({}) YIELD content", path);
}
std::string GetModuleFile(auto &client, const std::filesystem::path &path) {
spdlog::info("Getting content of module file '{}'", path);
MG_ASSERT(client->Execute(GetModuleFileQuery(path)));
const auto result_row = client->FetchOne();
MG_ASSERT(result_row && result_row->size() == 1, "Received invalid result from mg.get_module_file");
MG_ASSERT(!client->FetchOne().has_value(), "Too many results received from mg.get_module_file");
return std::string{result_row->at(0).ValueString()};
}
std::string UpdateModuleFileQuery(const std::filesystem::path &path, const std::string_view content) {
return fmt::format("CALL mg.update_module_file({}, '{}')", path, content);
}
void UpdateModuleFile(auto &client, const std::filesystem::path &path, const std::string_view content) {
spdlog::info("Updating module file {} with content:\n{}", path, content);
MG_ASSERT(client->Execute(UpdateModuleFileQuery(path, content)));
MG_ASSERT(client->FetchAll().has_value());
}
std::string DeleteModuleFileQuery(const std::filesystem::path &path) {
return fmt::format("CALL mg.delete_module_file({})", path);
}
void DeleteModuleFile(auto &client, const std::filesystem::path &path) {
spdlog::info("Deleting module file {}", path);
MG_ASSERT(client->Execute(DeleteModuleFileQuery(path)));
MG_ASSERT(client->FetchAll().has_value());
}
inline constexpr std::string_view module_content1 = R"(import mgp
@mgp.read_proc
def simple1(ctx: mgp.ProcCtx) -> mgp.Record(result=bool):
return mgp.Record(mutable=True))";
inline constexpr std::string_view module_content2 = R"(import mgp
@mgp.read_proc
def simple2(ctx: mgp.ProcCtx) -> mgp.Record(result=bool):
return mgp.Record(mutable=True))";
} // namespace
int main(int argc, char **argv) {
google::SetUsageMessage("Memgraph E2E Isolation Levels");
gflags::ParseCommandLineFlags(&argc, &argv, true);
memgraph::logging::RedirectToStderr();
mg::Client::Init();
auto client = GetClient();
if (FLAGS_multi_db) {
client->Execute("CREATE DATABASE clean;");
client->DiscardAll();
client->Execute("USE DATABASE clean;");
client->DiscardAll();
client->Execute("MATCH (n) DETACH DELETE n;");
client->DiscardAll();
}
AssertQueryFails<mg::ClientException>(client, CreateModuleFileQuery("some.cpp", "some content"),
"mg.create_module_file: The specified file isn't in the supported format.");
AssertQueryFails<mg::ClientException>(client, CreateModuleFileQuery("../some.cpp", "some content"),
"mg.create_module_file: Invalid relative path defined. The module file cannot "
"be define outside the internal modules directory.");
AssertProcedureNotExists(client, "some.simple1");
const auto module_path = CreateModuleFile(client, "some.py", module_content1);
AssertQueryFails<mg::ClientException>(client, CreateModuleFileQuery("some.py", "some content"),
"mg.create_module_file: File with the same name already exists!");
AssertProcedureExists(client, "some.simple1", module_path);
AssertModuleFileExists(client, module_path);
MG_ASSERT(GetModuleFile(client, module_path) == module_content1,
"Content received from mg.get_module_file is incorrect");
AssertQueryFails<mg::ClientException>(client, GetModuleFileQuery("some.py"),
"mg.get_module_file: The path should be an absolute path.");
AssertQueryFails<mg::ClientException>(client, GetModuleFileQuery(module_path.parent_path() / "some.cpp"),
"mg.get_module_file: The specified file isn't in the supported format.");
AssertQueryFails<mg::ClientException>(client, GetModuleFileQuery(module_path.parent_path() / "some2.py"),
"mg.get_module_file: The specified file doesn't exist.");
AssertQueryFails<mg::ClientException>(client, UpdateModuleFileQuery("some.py", "some content"),
"mg.update_module_file: The path should be an absolute path.");
AssertQueryFails<mg::ClientException>(client,
UpdateModuleFileQuery(module_path.parent_path() / "some.cpp", "some content"),
"mg.update_module_file: The specified file isn't in the supported format.");
AssertQueryFails<mg::ClientException>(client,
UpdateModuleFileQuery(module_path.parent_path() / "some2.py", "some content"),
"mg.update_module_file: The specified file doesn't exist.");
UpdateModuleFile(client, module_path, module_content2);
AssertProcedureNotExists(client, "some.simple1");
AssertProcedureExists(client, "some.simple2", module_path);
AssertModuleFileExists(client, module_path);
MG_ASSERT(GetModuleFile(client, module_path) == module_content2,
"Content received from mg.get_module_file is incorrect");
AssertQueryFails<mg::ClientException>(client, DeleteModuleFileQuery("some.py"),
"mg.delete_module_file: The path should be an absolute path.");
AssertQueryFails<mg::ClientException>(client, DeleteModuleFileQuery(module_path.parent_path() / "some.cpp"),
"mg.delete_module_file: The specified file isn't in the supported format.");
AssertQueryFails<mg::ClientException>(client, DeleteModuleFileQuery(module_path.parent_path() / "some2.py"),
"mg.delete_module_file: The specified file doesn't exist.");
DeleteModuleFile(client, module_path);
AssertProcedureNotExists(client, "some.simple1");
AssertProcedureNotExists(client, "some.simple2");
AssertModuleFileNotExists(client, module_path);
const auto non_module_directory =
std::filesystem::temp_directory_path() / "module_file_manager_e2e_non_module_directory";
memgraph::utils::EnsureDirOrDie(non_module_directory);
const auto non_module_file_path{non_module_directory / "something.py"};
{
std::ofstream non_module_file{non_module_file_path};
MG_ASSERT(non_module_file.is_open(), "Failed to open {} for writing", non_module_file_path);
static constexpr std::string_view content = "import mgp";
non_module_file.write(content.data(), content.size());
non_module_file.flush();
}
AssertQueryFails<mg::ClientException>(
client, GetModuleFileQuery(non_module_file_path),
"mg.get_module_file: The specified file isn't contained in any of the module directories.");
AssertQueryFails<mg::ClientException>(
client, UpdateModuleFileQuery(non_module_file_path, "some content"),
"mg.update_module_file: The specified file isn't contained in any of the module directories.");
AssertQueryFails<mg::ClientException>(
client, DeleteModuleFileQuery(non_module_file_path),
"mg.delete_module_file: The specified file isn't contained in any of the module directories.");
MG_ASSERT(std::filesystem::remove_all(non_module_directory), "Failed to cleanup directories");
return 0;
}