2023-03-27 21:46:00 +08:00
|
|
|
// 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 <chrono>
|
|
|
|
#include <stop_token>
|
|
|
|
#include <string>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "disk_test_utils.hpp"
|
2023-03-27 21:46:00 +08:00
|
|
|
#include "interpreter_faker.hpp"
|
2023-06-29 17:44:55 +08:00
|
|
|
#include "storage/v2/inmemory/storage.hpp"
|
2023-03-27 21:46:00 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
Tests rely on the fact that interpreters are sequentially added to runninng_interpreters to get transaction_id of its
|
|
|
|
corresponding interpreter/.
|
|
|
|
*/
|
2023-06-29 17:44:55 +08:00
|
|
|
template <typename StorageType>
|
2023-03-27 21:46:00 +08:00
|
|
|
class TransactionQueueSimpleTest : public ::testing::Test {
|
|
|
|
protected:
|
2023-06-29 17:44:55 +08:00
|
|
|
const std::string testSuite = "transactin_queue";
|
2023-03-27 21:46:00 +08:00
|
|
|
std::filesystem::path data_directory{std::filesystem::temp_directory_path() / "MG_tests_unit_transaction_queue_intr"};
|
2023-06-29 17:44:55 +08:00
|
|
|
memgraph::query::InterpreterContext interpreter_context{
|
|
|
|
std::make_unique<StorageType>(disk_test_utils::GenerateOnDiskConfig(testSuite)), {}, data_directory};
|
2023-03-27 21:46:00 +08:00
|
|
|
InterpreterFaker running_interpreter{&interpreter_context}, main_interpreter{&interpreter_context};
|
2023-06-29 17:44:55 +08:00
|
|
|
|
|
|
|
void TearDown() override { disk_test_utils::RemoveRocksDbDirs(testSuite); }
|
2023-03-27 21:46:00 +08:00
|
|
|
};
|
|
|
|
|
2023-06-29 17:44:55 +08:00
|
|
|
using StorageTypes = ::testing::Types<memgraph::storage::InMemoryStorage, memgraph::storage::DiskStorage>;
|
|
|
|
TYPED_TEST_CASE(TransactionQueueSimpleTest, StorageTypes);
|
|
|
|
|
|
|
|
TYPED_TEST(TransactionQueueSimpleTest, TwoInterpretersInterleaving) {
|
2023-03-27 21:46:00 +08:00
|
|
|
bool started = false;
|
|
|
|
std::jthread running_thread = std::jthread(
|
|
|
|
[this, &started](std::stop_token st, int thread_index) {
|
2023-06-29 17:44:55 +08:00
|
|
|
this->running_interpreter.Interpret("BEGIN");
|
2023-03-27 21:46:00 +08:00
|
|
|
started = true;
|
|
|
|
},
|
|
|
|
0);
|
|
|
|
|
|
|
|
{
|
|
|
|
while (!started) {
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(20));
|
|
|
|
}
|
2023-06-29 17:44:55 +08:00
|
|
|
this->main_interpreter.Interpret("CREATE (:Person {prop: 1})");
|
|
|
|
auto show_stream = this->main_interpreter.Interpret("SHOW TRANSACTIONS");
|
2023-03-27 21:46:00 +08:00
|
|
|
ASSERT_EQ(show_stream.GetResults().size(), 2U);
|
|
|
|
// superadmin executing the transaction
|
|
|
|
EXPECT_EQ(show_stream.GetResults()[0][0].ValueString(), "");
|
|
|
|
ASSERT_TRUE(show_stream.GetResults()[0][1].IsString());
|
|
|
|
EXPECT_EQ(show_stream.GetResults()[0][2].ValueList().at(0).ValueString(), "SHOW TRANSACTIONS");
|
|
|
|
// Also anonymous user executing
|
|
|
|
EXPECT_EQ(show_stream.GetResults()[1][0].ValueString(), "");
|
|
|
|
ASSERT_TRUE(show_stream.GetResults()[1][1].IsString());
|
|
|
|
// Kill the other transaction
|
|
|
|
std::string run_trans_id = show_stream.GetResults()[1][1].ValueString();
|
|
|
|
std::string esc_run_trans_id = "'" + run_trans_id + "'";
|
2023-06-29 17:44:55 +08:00
|
|
|
auto terminate_stream = this->main_interpreter.Interpret("TERMINATE TRANSACTIONS " + esc_run_trans_id);
|
2023-03-27 21:46:00 +08:00
|
|
|
// check result of killing
|
|
|
|
ASSERT_EQ(terminate_stream.GetResults().size(), 1U);
|
|
|
|
EXPECT_EQ(terminate_stream.GetResults()[0][0].ValueString(), run_trans_id);
|
|
|
|
ASSERT_TRUE(terminate_stream.GetResults()[0][1].ValueBool()); // that the transaction is actually killed
|
|
|
|
// check the number of transactions now
|
2023-06-29 17:44:55 +08:00
|
|
|
auto show_stream_after_killing = this->main_interpreter.Interpret("SHOW TRANSACTIONS");
|
2023-03-27 21:46:00 +08:00
|
|
|
ASSERT_EQ(show_stream_after_killing.GetResults().size(), 1U);
|
|
|
|
// test the state of the database
|
2023-06-29 17:44:55 +08:00
|
|
|
auto results_stream = this->main_interpreter.Interpret("MATCH (n) RETURN n");
|
2023-03-27 21:46:00 +08:00
|
|
|
ASSERT_EQ(results_stream.GetResults().size(), 1U); // from the main interpreter
|
2023-06-29 17:44:55 +08:00
|
|
|
this->main_interpreter.Interpret("MATCH (n) DETACH DELETE n");
|
2023-03-27 21:46:00 +08:00
|
|
|
// finish thread
|
|
|
|
running_thread.request_stop();
|
|
|
|
}
|
|
|
|
}
|