2024-01-06 00:42:54 +08:00
|
|
|
// Copyright 2024 Memgraph Ltd.
|
2021-10-26 14:53:56 +08:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2021-06-14 21:47:57 +08:00
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "disk_test_utils.hpp"
|
|
|
|
#include "storage/v2/disk/storage.hpp"
|
|
|
|
#include "storage/v2/inmemory/storage.hpp"
|
2021-06-14 21:47:57 +08:00
|
|
|
#include "storage/v2/isolation_level.hpp"
|
2023-12-13 21:19:01 +08:00
|
|
|
#include "utils/on_scope_exit.hpp"
|
2024-01-06 00:42:54 +08:00
|
|
|
using memgraph::replication::ReplicationRole;
|
2021-06-14 21:47:57 +08:00
|
|
|
|
|
|
|
namespace {
|
2023-06-29 17:44:55 +08:00
|
|
|
int64_t VerticesCount(memgraph::storage::Storage::Accessor *accessor) {
|
2021-06-14 21:47:57 +08:00
|
|
|
int64_t count{0};
|
2023-06-29 17:44:55 +08:00
|
|
|
for ([[maybe_unused]] const auto &vertex : accessor->Vertices(memgraph::storage::View::NEW)) {
|
2021-06-14 21:47:57 +08:00
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2022-03-31 19:52:43 +08:00
|
|
|
inline constexpr std::array isolation_levels{memgraph::storage::IsolationLevel::SNAPSHOT_ISOLATION,
|
|
|
|
memgraph::storage::IsolationLevel::READ_COMMITTED,
|
|
|
|
memgraph::storage::IsolationLevel::READ_UNCOMMITTED};
|
2021-06-14 21:47:57 +08:00
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2022-02-22 20:33:45 +08:00
|
|
|
class StorageIsolationLevelTest : public ::testing::TestWithParam<memgraph::storage::IsolationLevel> {
|
2021-06-14 21:47:57 +08:00
|
|
|
public:
|
|
|
|
struct PrintToStringParamName {
|
2022-02-22 20:33:45 +08:00
|
|
|
std::string operator()(const testing::TestParamInfo<memgraph::storage::IsolationLevel> &info) {
|
|
|
|
return std::string(IsolationLevelToString(static_cast<memgraph::storage::IsolationLevel>(info.param)));
|
2021-06-14 21:47:57 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
void TestVisibility(std::unique_ptr<memgraph::storage::Storage> &storage,
|
|
|
|
const memgraph::storage::IsolationLevel &default_isolation_level,
|
|
|
|
const memgraph::storage::IsolationLevel &override_isolation_level) {
|
2024-01-06 00:42:54 +08:00
|
|
|
auto creator = storage->Access(ReplicationRole::MAIN);
|
|
|
|
auto default_isolation_level_reader = storage->Access(ReplicationRole::MAIN);
|
|
|
|
auto override_isolation_level_reader = storage->Access(ReplicationRole::MAIN, override_isolation_level);
|
2021-06-14 21:47:57 +08:00
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
ASSERT_EQ(VerticesCount(default_isolation_level_reader.get()), 0);
|
|
|
|
ASSERT_EQ(VerticesCount(override_isolation_level_reader.get()), 0);
|
2021-06-14 21:47:57 +08:00
|
|
|
|
2022-03-31 19:52:43 +08:00
|
|
|
static constexpr auto iteration_count = 10;
|
2021-06-14 21:47:57 +08:00
|
|
|
{
|
|
|
|
SCOPED_TRACE(fmt::format(
|
|
|
|
"Visibility while the creator transaction is active "
|
|
|
|
"(default isolation level = {}, override isolation level = {})",
|
|
|
|
IsolationLevelToString(default_isolation_level), IsolationLevelToString(override_isolation_level)));
|
|
|
|
for (size_t i = 1; i <= iteration_count; ++i) {
|
2023-06-29 17:44:55 +08:00
|
|
|
creator->CreateVertex();
|
2021-06-14 21:47:57 +08:00
|
|
|
|
|
|
|
const auto check_vertices_count = [i](auto &accessor, const auto isolation_level) {
|
2022-02-22 20:33:45 +08:00
|
|
|
const auto expected_count = isolation_level == memgraph::storage::IsolationLevel::READ_UNCOMMITTED ? i : 0;
|
2023-06-29 17:44:55 +08:00
|
|
|
EXPECT_EQ(VerticesCount(accessor.get()), expected_count);
|
2021-06-14 21:47:57 +08:00
|
|
|
};
|
|
|
|
check_vertices_count(default_isolation_level_reader, default_isolation_level);
|
|
|
|
check_vertices_count(override_isolation_level_reader, override_isolation_level);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
ASSERT_FALSE(creator->Commit().HasError());
|
2021-06-14 21:47:57 +08:00
|
|
|
{
|
|
|
|
SCOPED_TRACE(fmt::format(
|
|
|
|
"Visibility after the creator transaction is committed "
|
|
|
|
"(default isolation level = {}, override isolation level = {})",
|
|
|
|
IsolationLevelToString(default_isolation_level), IsolationLevelToString(override_isolation_level)));
|
2022-03-31 19:52:43 +08:00
|
|
|
const auto check_vertices_count = [](auto &accessor, const auto isolation_level) {
|
2021-06-14 21:47:57 +08:00
|
|
|
const auto expected_count =
|
2022-02-22 20:33:45 +08:00
|
|
|
isolation_level == memgraph::storage::IsolationLevel::SNAPSHOT_ISOLATION ? 0 : iteration_count;
|
2023-06-29 17:44:55 +08:00
|
|
|
ASSERT_EQ(VerticesCount(accessor.get()), expected_count);
|
2021-06-14 21:47:57 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
check_vertices_count(default_isolation_level_reader, default_isolation_level);
|
|
|
|
check_vertices_count(override_isolation_level_reader, override_isolation_level);
|
|
|
|
}
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
ASSERT_FALSE(default_isolation_level_reader->Commit().HasError());
|
|
|
|
ASSERT_FALSE(override_isolation_level_reader->Commit().HasError());
|
2021-06-14 21:47:57 +08:00
|
|
|
|
|
|
|
SCOPED_TRACE("Visibility after a new transaction is started");
|
2024-01-06 00:42:54 +08:00
|
|
|
auto verifier = storage->Access(ReplicationRole::MAIN);
|
2023-06-29 17:44:55 +08:00
|
|
|
ASSERT_EQ(VerticesCount(verifier.get()), iteration_count);
|
|
|
|
ASSERT_FALSE(verifier->Commit().HasError());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(StorageIsolationLevelTest, VisibilityInMemoryStorage) {
|
|
|
|
const auto default_isolation_level = GetParam();
|
|
|
|
|
|
|
|
for (const auto override_isolation_level : isolation_levels) {
|
|
|
|
std::unique_ptr<memgraph::storage::Storage> storage(new memgraph::storage::InMemoryStorage(
|
|
|
|
{memgraph::storage::Config{.transaction = {.isolation_level = default_isolation_level}}}));
|
|
|
|
this->TestVisibility(storage, default_isolation_level, override_isolation_level);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(StorageIsolationLevelTest, VisibilityOnDiskStorage) {
|
|
|
|
const auto default_isolation_level = GetParam();
|
|
|
|
|
|
|
|
const std::string testSuite = "storage_v2_isolation_level";
|
|
|
|
auto config = disk_test_utils::GenerateOnDiskConfig(testSuite);
|
|
|
|
config.transaction.isolation_level = default_isolation_level;
|
|
|
|
|
|
|
|
for (const auto override_isolation_level : isolation_levels) {
|
|
|
|
std::unique_ptr<memgraph::storage::Storage> storage(new memgraph::storage::DiskStorage(config));
|
2023-12-13 21:19:01 +08:00
|
|
|
auto on_exit = memgraph::utils::OnScopeExit{[&]() { disk_test_utils::RemoveRocksDbDirs(testSuite); }};
|
2023-06-29 17:44:55 +08:00
|
|
|
try {
|
|
|
|
this->TestVisibility(storage, default_isolation_level, override_isolation_level);
|
|
|
|
} catch (memgraph::utils::NotYetImplemented &) {
|
|
|
|
if (default_isolation_level != memgraph::storage::IsolationLevel::SNAPSHOT_ISOLATION ||
|
|
|
|
override_isolation_level != memgraph::storage::IsolationLevel::SNAPSHOT_ISOLATION) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
throw;
|
|
|
|
}
|
2021-06-14 21:47:57 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedStorageIsolationLevelTests, StorageIsolationLevelTest,
|
|
|
|
::testing::ValuesIn(isolation_levels), StorageIsolationLevelTest::PrintToStringParamName());
|