8cd9f696cf
This PR introduces READ COMMITTED and READ UNCOMMITTED isolation levels. The isolation level can be set with a config or with a query for different scopes.
242 lines
9.2 KiB
C++
242 lines
9.2 KiB
C++
#include <gflags/gflags.h>
|
|
#include <mgclient.hpp>
|
|
|
|
#include "utils/logging.hpp"
|
|
#include "utils/timer.hpp"
|
|
|
|
DEFINE_uint64(bolt_port, 7687, "Bolt port");
|
|
DEFINE_uint64(timeout, 120, "Timeout seconds");
|
|
|
|
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;
|
|
}
|
|
|
|
auto GetVertexCount(std::unique_ptr<mg::Client> &client) {
|
|
MG_ASSERT(client->Execute("MATCH (n) RETURN count(n)"));
|
|
auto maybe_row = client->FetchOne();
|
|
MG_ASSERT(maybe_row, "Failed to fetch vertex count");
|
|
|
|
const auto &row = *maybe_row;
|
|
MG_ASSERT(row.size() == 1, "Got invalid result for vertex count");
|
|
|
|
client->FetchOne();
|
|
return row[0].ValueInt();
|
|
}
|
|
|
|
void CleanDatabase() {
|
|
auto client = GetClient();
|
|
MG_ASSERT(client->Execute("MATCH (n) DETACH DELETE n;"));
|
|
client->DiscardAll();
|
|
}
|
|
|
|
void TestSnapshotIsolation(std::unique_ptr<mg::Client> &client) {
|
|
spdlog::info("Verifying SNAPSHOT ISOLATION");
|
|
|
|
auto creator = GetClient();
|
|
|
|
MG_ASSERT(client->BeginTransaction());
|
|
MG_ASSERT(creator->BeginTransaction());
|
|
|
|
constexpr auto vertex_count = 10;
|
|
for (size_t i = 0; i < vertex_count; ++i) {
|
|
MG_ASSERT(creator->Execute("CREATE ()"));
|
|
creator->DiscardAll();
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == 0,
|
|
"Invalid number of vertices found for SNAPSHOT ISOLATION (found {}, expected {}). Read vertices from a "
|
|
"transaction which started "
|
|
"at a later point.",
|
|
current_vertex_count, 0);
|
|
}
|
|
|
|
MG_ASSERT(creator->CommitTransaction());
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == 0,
|
|
"Invalid number of vertices found for SNAPSHOT ISOLATION (found {}, expected {}). Read vertices from a "
|
|
"transaction which started "
|
|
"at a later point.",
|
|
current_vertex_count, 0);
|
|
MG_ASSERT(client->CommitTransaction());
|
|
CleanDatabase();
|
|
}
|
|
|
|
void TestReadCommitted(std::unique_ptr<mg::Client> &client) {
|
|
spdlog::info("Verifying READ COMMITTED");
|
|
|
|
auto creator = GetClient();
|
|
|
|
MG_ASSERT(client->BeginTransaction());
|
|
MG_ASSERT(creator->BeginTransaction());
|
|
|
|
constexpr auto vertex_count = 10;
|
|
for (size_t i = 0; i < vertex_count; ++i) {
|
|
MG_ASSERT(creator->Execute("CREATE ()"));
|
|
creator->DiscardAll();
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == 0,
|
|
"Invalid number of vertices found for READ COMMITTED (found {}, expected {}. Read vertices from a "
|
|
"transaction which is not "
|
|
"committed.",
|
|
current_vertex_count, 0);
|
|
}
|
|
|
|
MG_ASSERT(creator->CommitTransaction());
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == vertex_count,
|
|
"Invalid number of vertices found for READ COMMITTED (found {}, expected {}). Failed to read vertices "
|
|
"from a committed transaction",
|
|
current_vertex_count, vertex_count);
|
|
MG_ASSERT(client->CommitTransaction());
|
|
CleanDatabase();
|
|
}
|
|
|
|
void TestReadUncommitted(std::unique_ptr<mg::Client> &client) {
|
|
spdlog::info("Verifying READ UNCOMMITTED");
|
|
|
|
auto creator = GetClient();
|
|
|
|
MG_ASSERT(client->BeginTransaction());
|
|
MG_ASSERT(creator->BeginTransaction());
|
|
|
|
constexpr auto vertex_count = 10;
|
|
for (size_t i = 1; i <= vertex_count; ++i) {
|
|
MG_ASSERT(creator->Execute("CREATE ()"));
|
|
creator->DiscardAll();
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == i,
|
|
"Invalid number of vertices found for READ UNCOMMITTED (found {}, expected {}). Failed to read vertices "
|
|
"from a different transaction.",
|
|
current_vertex_count, i);
|
|
}
|
|
|
|
MG_ASSERT(creator->CommitTransaction());
|
|
|
|
auto current_vertex_count = GetVertexCount(client);
|
|
MG_ASSERT(current_vertex_count == vertex_count,
|
|
"Invalid number of vertices found for READ UNCOMMITTED (found {}, expected {}). Failed to read vertices "
|
|
"from a different transaction",
|
|
current_vertex_count, vertex_count);
|
|
MG_ASSERT(client->CommitTransaction());
|
|
CleanDatabase();
|
|
}
|
|
|
|
constexpr std::array isolation_levels{std::pair{"SNAPSHOT ISOLATION", &TestSnapshotIsolation},
|
|
std::pair{"READ COMMITTED", &TestReadCommitted},
|
|
std::pair{"READ UNCOMMITTED", &TestReadUncommitted}};
|
|
|
|
void TestGlobalIsolationLevel() {
|
|
spdlog::info("\n\n----Test global isolation levels----\n");
|
|
auto first_client = GetClient();
|
|
auto second_client = GetClient();
|
|
|
|
for (const auto &[isolation_level, verification_function] : isolation_levels) {
|
|
spdlog::info("--------------------------");
|
|
spdlog::info("Setting global isolation level to {}", isolation_level);
|
|
MG_ASSERT(first_client->Execute(fmt::format("SET GLOBAL TRANSACTION ISOLATION LEVEL {}", isolation_level)));
|
|
first_client->DiscardAll();
|
|
|
|
verification_function(first_client);
|
|
verification_function(second_client);
|
|
spdlog::info("--------------------------\n");
|
|
}
|
|
}
|
|
|
|
void TestSessionIsolationLevel() {
|
|
spdlog::info("\n\n----Test session isolation levels----\n");
|
|
|
|
auto global_client = GetClient();
|
|
auto session_client = GetClient();
|
|
for (const auto &[global_isolation_level, global_verification_function] : isolation_levels) {
|
|
spdlog::info("Setting global isolation level to {}", global_isolation_level);
|
|
MG_ASSERT(global_client->Execute(fmt::format("SET GLOBAL TRANSACTION ISOLATION LEVEL {}", global_isolation_level)));
|
|
global_client->DiscardAll();
|
|
|
|
for (const auto &[session_isolation_level, session_verification_function] : isolation_levels) {
|
|
spdlog::info("--------------------------");
|
|
spdlog::info("Setting session isolation level to {}", session_isolation_level);
|
|
MG_ASSERT(
|
|
session_client->Execute(fmt::format("SET SESSION TRANSACTION ISOLATION LEVEL {}", session_isolation_level)));
|
|
session_client->DiscardAll();
|
|
|
|
spdlog::info("Verifying client which is using global isolation level");
|
|
global_verification_function(global_client);
|
|
spdlog::info("Verifying client which is using session isolation level");
|
|
session_verification_function(session_client);
|
|
spdlog::info("--------------------------\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
// Priority of applying the isolation level from highest priority NEXT -> SESSION -> GLOBAL
|
|
void TestNextIsolationLevel() {
|
|
spdlog::info("\n\n----Test next isolation levels----\n");
|
|
|
|
auto global_client = GetClient();
|
|
auto session_client = GetClient();
|
|
for (const auto &[global_isolation_level, global_verification_function] : isolation_levels) {
|
|
spdlog::info("Setting global isolation level to {}", global_isolation_level);
|
|
MG_ASSERT(global_client->Execute(fmt::format("SET GLOBAL TRANSACTION ISOLATION LEVEL {}", global_isolation_level)));
|
|
global_client->DiscardAll();
|
|
|
|
for (const auto &[session_isolation_level, session_verification_function] : isolation_levels) {
|
|
spdlog::info("Setting session isolation level to {}", session_isolation_level);
|
|
MG_ASSERT(
|
|
session_client->Execute(fmt::format("SET SESSION TRANSACTION ISOLATION LEVEL {}", session_isolation_level)));
|
|
session_client->DiscardAll();
|
|
|
|
for (const auto &[next_isolation_level, next_verification_function] : isolation_levels) {
|
|
spdlog::info("--------------------------");
|
|
spdlog::info("Verifying client which is using global isolation level");
|
|
global_verification_function(global_client);
|
|
spdlog::info("Verifying client which is using session isolation level");
|
|
session_verification_function(session_client);
|
|
|
|
spdlog::info("Setting isolation level of the next transaction to {}", next_isolation_level);
|
|
MG_ASSERT(global_client->Execute(fmt::format("SET NEXT TRANSACTION ISOLATION LEVEL {}", next_isolation_level)));
|
|
global_client->DiscardAll();
|
|
MG_ASSERT(
|
|
session_client->Execute(fmt::format("SET NEXT TRANSACTION ISOLATION LEVEL {}", next_isolation_level)));
|
|
session_client->DiscardAll();
|
|
|
|
spdlog::info("Verifying client which is using global isolation level while next isolation level is set");
|
|
next_verification_function(global_client);
|
|
spdlog::info("Verifying client which is using session isolation level while next isolation level is set");
|
|
next_verification_function(session_client);
|
|
|
|
spdlog::info("Verifying client which is using global isolation level after the next isolation level was used");
|
|
global_verification_function(global_client);
|
|
spdlog::info("Verifying client which is using session isolation level after the next isolation level was used");
|
|
session_verification_function(session_client);
|
|
spdlog::info("--------------------------\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int main(int argc, char **argv) {
|
|
google::SetUsageMessage("Memgraph E2E Isolation Levels");
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
|
logging::RedirectToStderr();
|
|
|
|
mg::Client::Init();
|
|
|
|
TestGlobalIsolationLevel();
|
|
TestSessionIsolationLevel();
|
|
TestNextIsolationLevel();
|
|
|
|
return 0;
|
|
}
|