added multi threaded pool test

This commit is contained in:
Sascha Kühl 2024-04-12 16:34:30 +02:00
parent 25bcc362f2
commit b4765cac88
6 changed files with 127 additions and 74 deletions

View File

@ -12,7 +12,7 @@ list(APPEND CMAKE_MODULE_PATH ${catch2_SOURCE_DIR}/extras)
include(CTest)
include(Catch)
set(POSTGRES_CONNECTION_STRING "postgres://test:test123@127.0.0.1:5432/matador_test")
set(POSTGRES_CONNECTION_STRING "postgres://test:test123@127.0.0.1:15432/test")
configure_file(Connection.hpp.in ${PROJECT_BINARY_DIR}/backends/postgres/test/connection.hpp @ONLY IMMEDIATE)

View File

@ -86,19 +86,21 @@ public:
}
connection_pointer acquire() {
std::unique_lock<std::mutex> lock(mutex_);
while (idle_connections_.empty()) {
cv.wait(lock);
}
return get_next_connection();
}
connection_pointer try_acquire() {
std::unique_lock<std::mutex> lock(mutex_);
if (idle_connections_.empty()) {
return {nullptr, this};
}
pointer next_connection{nullptr};
for (auto &item : idle_connections_) {
next_connection = item.second;
auto node = idle_connections_.extract(item.first);
inuse_connections_.insert(std::move(node));
break;
}
return {next_connection, this};
return get_next_connection();
}
connection_pointer acquire(size_t id) {
@ -151,8 +153,21 @@ public:
const connection_info &info() const {
return info_;
}
private:
connection_pointer get_next_connection() {
pointer next_connection{nullptr};
for (auto &item : idle_connections_) {
next_connection = item.second;
auto node = idle_connections_.extract(item.first);
inuse_connections_.insert(std::move(node));
break;
}
return {next_connection, this};
}
private:
mutable std::mutex mutex_;
std::condition_variable cv;
std::vector<IdConnection<Connection>> connection_repo_;
using pointer = IdConnection<Connection>*;
using connection_map = std::unordered_map<size_t, pointer>;

View File

@ -34,8 +34,6 @@ add_executable(tests
models/location.hpp
models/optional.hpp
ConvertTest.cpp
DummyConnection.hpp
DummyConnection.cpp
EntityQueryBuilderTest.cpp
models/author.hpp
models/book.hpp

View File

@ -1,15 +1,42 @@
#include <catch2/catch_test_macros.hpp>
#include <matador/sql/connection_pool.hpp>
#include <matador/sql/noop_connection.hpp>
#include "DummyConnection.hpp"
class AutoResetEvent
{
public:
AutoResetEvent() : state(false) {}
AutoResetEvent(const AutoResetEvent& other) = delete;
void WaitOne() {
std::unique_lock<std::mutex> lock(sync);
underlying.wait(lock, [this](){return state.load();});
}
void Set() {
std::unique_lock<std::mutex> lock(sync);
state = true;
underlying.notify_all();
}
void Reset() {
std::unique_lock<std::mutex> lock(sync);
state = false;
}
private:
std::condition_variable underlying;
std::mutex sync;
std::atomic<bool> state;
};
using namespace matador::sql;
TEST_CASE("Create connection pool", "[connection pool]") {
using pool_t = connection_pool<matador::test::DummyConnection>;
using pool_t = connection_pool<noop_connection>;
pool_t pool("sqlite://sqlite.db", 4);
pool_t pool("noop://noop.db", 4);
REQUIRE(pool.size() == 4);
REQUIRE(pool.idle() == 4);
@ -51,9 +78,9 @@ TEST_CASE("Create connection pool", "[connection pool]") {
}
TEST_CASE("Acquire connection by id", "[connection pool]") {
using pool_t = connection_pool<matador::test::DummyConnection>;
using pool_t = connection_pool<noop_connection>;
pool_t pool("sqlite://sqlite.db", 4);
pool_t pool("noop://noop.db", 4);
REQUIRE(pool.size() == 4);
REQUIRE(pool.idle() == 4);
@ -79,3 +106,74 @@ TEST_CASE("Acquire connection by id", "[connection pool]") {
REQUIRE(same_ptr.valid());
REQUIRE(same_ptr.id() == connection_id);
}
TEST_CASE("Try acquire connection", "[connection pool][try acquire]") {
using pool_t = connection_pool<noop_connection>;
pool_t pool("noop://noop.db", 1);
REQUIRE(pool.size() == 1);
REQUIRE(pool.idle() == 1);
REQUIRE(pool.inuse() == 0);
auto ptr = pool.try_acquire();
REQUIRE(ptr.valid());
REQUIRE(ptr.id());
REQUIRE(ptr.id().value() > 0);
REQUIRE(ptr->is_open());
REQUIRE(pool.size() == 1);
REQUIRE(pool.idle() == 0);
REQUIRE(pool.inuse() == 1);
auto ptr2 = pool.try_acquire();
REQUIRE(!ptr2.valid());
pool.release(ptr);
REQUIRE(!ptr.valid());
REQUIRE(pool.size() == 1);
REQUIRE(pool.idle() == 1);
REQUIRE(pool.inuse() == 0);
ptr2 = pool.try_acquire();
REQUIRE(ptr2.valid());
REQUIRE(ptr2.id());
REQUIRE(ptr2.id().value() > 0);
REQUIRE(ptr2->is_open());
REQUIRE(pool.size() == 1);
REQUIRE(pool.idle() == 0);
REQUIRE(pool.inuse() == 1);
pool.release(ptr2);
AutoResetEvent reset_event1;
AutoResetEvent reset_event2;
AutoResetEvent reset_event3;
std::thread t([&reset_event1, &reset_event2, &reset_event3, &pool]() {
auto c1 = pool.acquire();
REQUIRE(c1.valid());
REQUIRE(c1.id());
REQUIRE(c1.id().value() > 0);
reset_event1.Set();
reset_event2.WaitOne();
pool.release(c1);
REQUIRE(!c1.valid());
reset_event3.Set();
});
reset_event1.WaitOne();
ptr2 = pool.try_acquire();
REQUIRE(!ptr2.valid());
reset_event2.Set();
reset_event3.WaitOne();
ptr2 = pool.try_acquire();
REQUIRE(ptr2.valid());
REQUIRE(ptr2.id());
REQUIRE(ptr2.id().value() > 0);
t.join();
}

View File

@ -1,28 +0,0 @@
#include "DummyConnection.hpp"
namespace matador::test {
DummyConnection::DummyConnection(sql::connection_info info)
: connection_info_(std::move(info))
{}
DummyConnection::DummyConnection(const std::string &dns)
: DummyConnection(sql::connection_info::parse(dns))
{}
void DummyConnection::open()
{
is_open_ = true;
}
void DummyConnection::close()
{
is_open_ = false;
}
bool DummyConnection::is_open() const
{
return is_open_;
}
}

View File

@ -1,30 +0,0 @@
#ifndef QUERY_DUMMYCONNECTION_HPP
#define QUERY_DUMMYCONNECTION_HPP
#include "matador/sql/connection_info.hpp"
namespace matador::test {
class DummyConnection
{
public:
explicit DummyConnection(sql::connection_info info);
explicit DummyConnection(const std::string& dns);
DummyConnection(const DummyConnection &x) = default;
DummyConnection& operator=(const DummyConnection &x) = default;
DummyConnection(DummyConnection &&x) noexcept = default;
DummyConnection& operator=(DummyConnection &&x) noexcept = default;
~DummyConnection() = default;
void open();
void close();
[[nodiscard]] bool is_open() const;
private:
sql::connection_info connection_info_;
bool is_open_{};
};
}
#endif //QUERY_DUMMYCONNECTION_HPP