...
This commit is contained in:
parent
346a5e6965
commit
a482a8dbc8
@ -28,8 +28,22 @@ set(ADC_NETWORK_HEADERS
|
|||||||
net/adc_netproto.h
|
net/adc_netproto.h
|
||||||
net/adc_netservice.h
|
net/adc_netservice.h
|
||||||
net/adc_endpoint.h
|
net/adc_endpoint.h
|
||||||
|
net/adc_netserver.h
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
option(SPDLOG_LIBRARY "Use SPDLOG library for logging" ON)
|
||||||
|
|
||||||
|
if (SPDLOG_LIBRARY)
|
||||||
|
find_package(spdlog REQUIRED)
|
||||||
|
|
||||||
|
set(ADC_NETWORK_HEADERS ${ADC_NETWORK_HEADERS}
|
||||||
|
net/adc_netserver_spdlog.h
|
||||||
|
)
|
||||||
|
|
||||||
|
add_compile_definitions(PUBLIC USE_SPDLOG_LIBRARY)
|
||||||
|
endif()
|
||||||
|
|
||||||
option(BUILD_TESTS "Build tests" ON)
|
option(BUILD_TESTS "Build tests" ON)
|
||||||
|
|
||||||
if (BUILD_TESTS)
|
if (BUILD_TESTS)
|
||||||
|
|||||||
88
common/adc_spdlog.h
Normal file
88
common/adc_spdlog.h
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef USE_SPDLOG_LIBRARY
|
||||||
|
|
||||||
|
#include <spdlog/logger.h>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
|
|
||||||
|
#include "adc_traits.h"
|
||||||
|
|
||||||
|
namespace adc
|
||||||
|
{
|
||||||
|
|
||||||
|
template <typename BaseT>
|
||||||
|
class AdcSpdlogGenericDecorator : public BaseT
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
std::shared_ptr<spdlog::logger> _logger;
|
||||||
|
|
||||||
|
void* _thisAddress;
|
||||||
|
|
||||||
|
public:
|
||||||
|
constexpr static char loggerDefaultFormat[] = "[%Y-%m-%d %T.%e] [%l]: [%v]";
|
||||||
|
|
||||||
|
template <typename... BaseCtorArgTs>
|
||||||
|
AdcSpdlogGenericDecorator(std::shared_ptr<spdlog::logger> logger, BaseCtorArgTs&&... ctor_args)
|
||||||
|
: BaseT(std::forward<BaseCtorArgTs>(ctor_args)...), _logger(logger), _thisAddress((void*)std::addressof(*this))
|
||||||
|
{
|
||||||
|
_logger->set_pattern(loggerDefaultFormat);
|
||||||
|
|
||||||
|
logMethodCalling("AdcSpdlogGenericDecorator"); // in debug
|
||||||
|
logMethodCalling(); // in trace
|
||||||
|
|
||||||
|
_logger->debug("set logger with {}", loggerInfo());
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~AdcSpdlogGenericDecorator()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
logMethodCalling("AdcSpdlogGenericDecorator"); // in debug
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected:
|
||||||
|
// format string must contain two formating specifications:
|
||||||
|
// 1 - logger name (string)
|
||||||
|
// 2 - logging level (string)
|
||||||
|
template <traits::adc_output_char_range R>
|
||||||
|
R loggerInfo(std::string_view fmt) const
|
||||||
|
{
|
||||||
|
R info;
|
||||||
|
|
||||||
|
std::format_to(std::back_inserter(info), fmt, _logger->name(), spdlog::level::to_string_view(_logger->level()));
|
||||||
|
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string loggerInfo() const
|
||||||
|
{
|
||||||
|
//
|
||||||
|
return loggerInfo<std::string>("name [{}] and log-level [{}]");
|
||||||
|
}
|
||||||
|
|
||||||
|
void logMethodCalling(spdlog::level::level_enum level = spdlog::level::trace) const
|
||||||
|
{
|
||||||
|
_logger->log(level, "call {} (this: {}, thread ID: {})", __PRETTY_FUNCTION__, _thisAddress,
|
||||||
|
std::this_thread::get_id());
|
||||||
|
}
|
||||||
|
|
||||||
|
void logMethodCalling(std::string_view class_name,
|
||||||
|
spdlog::level::level_enum level = spdlog::level::debug,
|
||||||
|
std::string_view method_name = "") const
|
||||||
|
{
|
||||||
|
if (method_name.empty()) {
|
||||||
|
method_name = __FUNCTION__;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger->log(level, "call {}::{} (this: {}, thread ID: {})", class_name, method_name, _thisAddress,
|
||||||
|
std::this_thread::get_id());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace adc
|
||||||
|
|
||||||
|
#endif
|
||||||
@ -21,17 +21,40 @@ namespace adc
|
|||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
/* Server session and its abstract decorator */
|
namespace traits
|
||||||
|
{
|
||||||
|
|
||||||
template <typename ImplT>
|
// network server session implementation concept
|
||||||
|
template <typename T>
|
||||||
|
concept adc_netserver_session_impl_c = requires(T t, const T t_const) {
|
||||||
|
typename T::session_ident_t;
|
||||||
|
|
||||||
|
{ t_const.sessionIdent() } -> std::same_as<typename T::session_ident_t>;
|
||||||
|
{ t.start() } -> std::same_as<void>;
|
||||||
|
{ t.stop() } -> std::same_as<void>;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace traits
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Server session */
|
||||||
|
|
||||||
|
template <traits::adc_netserver_session_impl_c ImplT>
|
||||||
class AdcNetServerSession : std::enable_shared_from_this<AdcNetServerSession<ImplT>>
|
class AdcNetServerSession : std::enable_shared_from_this<AdcNetServerSession<ImplT>>
|
||||||
{
|
{
|
||||||
|
protected:
|
||||||
|
ImplT _impl;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using shared_ptr_t = std::shared_ptr<AdcNetServerSession>;
|
typedef ImplT session_impl_t;
|
||||||
using weak_ptr_t = std::weak_ptr<AdcNetServerSession>;
|
typedef std::shared_ptr<AdcNetServerSession> shared_ptr_t;
|
||||||
|
typedef std::weak_ptr<AdcNetServerSession> weak_ptr_t;
|
||||||
|
|
||||||
|
using typename ImplT::session_ident_t;
|
||||||
|
|
||||||
template <typename... ImplCtorArgTs>
|
template <typename... ImplCtorArgTs>
|
||||||
AdcNetServerSession(ImplCtorArgTs&&... ctor_args) : _implUptr(std::forward<ImplCtorArgTs>(ctor_args)...)
|
AdcNetServerSession(ImplCtorArgTs&&... ctor_args) : _impl(std::forward<ImplCtorArgTs>(ctor_args)...)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -39,63 +62,72 @@ public:
|
|||||||
virtual ~AdcNetServerSession() = default;
|
virtual ~AdcNetServerSession() = default;
|
||||||
|
|
||||||
|
|
||||||
auto sessionIdent() const
|
virtual session_ident_t sessionIdent() const
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
return _implUptr->_sessionIdent();
|
return _impl._sessionIdent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
virtual void start()
|
virtual void start()
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
_implUptr->start();
|
_impl.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void stop()
|
virtual void stop()
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
_implUptr->stop();
|
_impl.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
|
||||||
std::unique_ptr<ImplT> _implUptr;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
template <typename ImplT, std::derived_from<AdcNetServerSession<ImplT>> SessionT>
|
|
||||||
class AdcNetServerSessionDecorator : public SessionT
|
namespace traits
|
||||||
{
|
{
|
||||||
public:
|
|
||||||
using SessionT::sessionIdent;
|
|
||||||
|
|
||||||
template <typename... ImplCtorArgTs>
|
// network server session concept
|
||||||
AdcNetServerSessionDecorator(ImplCtorArgTs&&... ctor_args) : SessionT(std::forward<ImplCtorArgTs>(ctor_args)...)
|
template <typename T>
|
||||||
{
|
concept adc_netserver_session_c = requires {
|
||||||
}
|
typename T::session_impl_t;
|
||||||
|
|
||||||
virtual ~AdcNetServerSessionDecorator() = default;
|
std::derived_from<T, AdcNetServerSession<typename T::session_impl_t>>;
|
||||||
|
|
||||||
virtual std::invoke_result_t<decltype(sessionIdent)> sessionIdent() const = 0;
|
|
||||||
|
|
||||||
virtual void start() = 0;
|
|
||||||
|
|
||||||
virtual void stop() = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
} // namespace traits
|
||||||
|
|
||||||
|
|
||||||
/* Server */
|
/* network server */
|
||||||
|
|
||||||
template <typename ImplT>
|
namespace traits
|
||||||
|
{
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
concept adc_netserver_impl_c = requires(T t, const T t_const) {
|
||||||
|
typename T::server_ident_t;
|
||||||
|
|
||||||
|
{ t_const.serverIdent() } -> std::same_as<typename T::server_ident_t>;
|
||||||
|
{ t.start() } -> std::same_as<void>;
|
||||||
|
{ t.stop() } -> std::same_as<void>;
|
||||||
|
{ t.daemonizePrepare() } -> std::same_as<void>;
|
||||||
|
{ t.daemonizeFinalize() } -> std::same_as<void>;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace traits
|
||||||
|
|
||||||
|
template <traits::adc_netserver_impl_c ImplT>
|
||||||
class AdcNetServer
|
class AdcNetServer
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
std::unique_ptr<ImplT> _implUptr;
|
ImplT _impl;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
typedef ImplT server_impl_t;
|
||||||
|
using typename ImplT::server_ident_t;
|
||||||
|
|
||||||
template <typename... ImplCtorArgTs>
|
template <typename... ImplCtorArgTs>
|
||||||
AdcNetServer(ImplCtorArgTs&&... ctor_args) : _implUptr(std::make_unique<ImplT>(ctor_args)...)
|
AdcNetServer(ImplCtorArgTs&&... ctor_args) : _impl(std::make_unique<ImplT>(ctor_args)...)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -103,18 +135,27 @@ public:
|
|||||||
virtual ~AdcNetServer() = default;
|
virtual ~AdcNetServer() = default;
|
||||||
|
|
||||||
|
|
||||||
virtual void start() = 0;
|
virtual server_ident_t serverIdent() const
|
||||||
|
{
|
||||||
|
//
|
||||||
|
return _impl.serverIdent();
|
||||||
|
}
|
||||||
|
|
||||||
virtual void stop() = 0;
|
virtual void start()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
_impl.start();
|
||||||
|
};
|
||||||
|
|
||||||
template <typename SessionT>
|
virtual void stop()
|
||||||
void start() {};
|
{
|
||||||
|
//
|
||||||
|
_impl.stop();
|
||||||
|
};
|
||||||
|
|
||||||
template <typename SessionT>
|
|
||||||
void stop() {};
|
|
||||||
|
|
||||||
// run server as daemon (still only on POSIX OSes)
|
// run server as daemon (still only on POSIX OSes)
|
||||||
void daemonize()
|
virtual void daemonize()
|
||||||
{
|
{
|
||||||
daemonizePrepare();
|
daemonizePrepare();
|
||||||
|
|
||||||
@ -161,9 +202,34 @@ public:
|
|||||||
daemonizeFinalize();
|
daemonizeFinalize();
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void daemonizePrepare() = 0;
|
|
||||||
|
|
||||||
virtual void daemonizeFinalize() = 0;
|
protected:
|
||||||
|
virtual void daemonizePrepare()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
_impl.daemonizePrepare();
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void daemonizeFinalize()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
_impl.daemonizeFinalize();
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
namespace traits
|
||||||
|
{
|
||||||
|
|
||||||
|
// network server concept
|
||||||
|
template <typename T>
|
||||||
|
concept adc_netserver_c = requires {
|
||||||
|
typename T::server_impl_t;
|
||||||
|
std::derived_from<T, AdcNetServer<typename T::server_impl_t>>;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace traits
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
} // namespace adc
|
} // namespace adc
|
||||||
|
|||||||
151
net/adc_netserver_spdlog.h
Normal file
151
net/adc_netserver_spdlog.h
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
ABSTRACT DEVICE COMPONENTS LIBRARY
|
||||||
|
|
||||||
|
spdlog-library extensions
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef USE_SPDLOG_LIBRARY
|
||||||
|
|
||||||
|
#include <spdlog/logger.h>
|
||||||
|
|
||||||
|
#include "../common/adc_traits.h"
|
||||||
|
#include "adc_netserver.h"
|
||||||
|
|
||||||
|
namespace adc
|
||||||
|
{
|
||||||
|
|
||||||
|
template <traits::adc_netserver_session_c SessionT>
|
||||||
|
class AdcNetServerSessionSpdlogDecorator : public SessionT
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
std::shared_ptr<spdlog::logger> _logger;
|
||||||
|
|
||||||
|
void* _thisAddress;
|
||||||
|
std::string _sessionID;
|
||||||
|
|
||||||
|
public:
|
||||||
|
// using SessionT::sessionIdent;
|
||||||
|
using typename SessionT::session_ident_t;
|
||||||
|
|
||||||
|
static_assert(traits::formattable<session_ident_t>, "NETWORK SESSION IDENT TYPE MUST BE A FORMATTABLE ONE!!!");
|
||||||
|
|
||||||
|
template <typename... ImplCtorArgTs>
|
||||||
|
AdcNetServerSessionSpdlogDecorator(std::shared_ptr<spdlog::logger> logger, ImplCtorArgTs&&... ctor_args)
|
||||||
|
: SessionT(std::forward<ImplCtorArgTs>(ctor_args)...),
|
||||||
|
_logger(logger),
|
||||||
|
_thisAddress((void*)std::addressof(this))
|
||||||
|
{
|
||||||
|
fmt::format_to(std::back_inserter(_sessionID), "{}", SessionT::sessionIdent());
|
||||||
|
|
||||||
|
_logger->trace("Call AdcNetServerSessionSpdlogDecorator::AdcNetServerSessionSpdlogDecorator (this: {})",
|
||||||
|
_thisAddress);
|
||||||
|
|
||||||
|
_logger->debug("Creating network server session with ID = [{}] (ADDR = {})", _sessionID, _thisAddress);
|
||||||
|
_logger->debug("use logger with name [{}] and level [{}]", _logger->name(),
|
||||||
|
spdlog::level::to_string_view(_logger->level()));
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~AdcNetServerSessionSpdlogDecorator()
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSessionSpdlogDecorator::~AdcNetServerSessionSpdlogDecorator (this: {})",
|
||||||
|
_thisAddress);
|
||||||
|
|
||||||
|
_logger->debug("Deleting network server session with ID = [{}] (ADDR = {})", _sessionID, _thisAddress);
|
||||||
|
|
||||||
|
_logger->flush();
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual session_ident_t sessionIdent() const override
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSessionSpdlogDecorator::sessionIdent (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
return SessionT::sessionIdent();
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void start() override
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSessionSpdlogDecorator::start (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
_logger->info("Starting network server session with ID = [{}] (ADDR = {})", _sessionID, _thisAddress);
|
||||||
|
|
||||||
|
SessionT::start();
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void stop() override
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSessionSpdlogDecorator::stop (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
_logger->info("Stopping network server session with ID = [{}] (ADDR = {})", _sessionID, _thisAddress);
|
||||||
|
|
||||||
|
SessionT::stop();
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
template <traits::adc_netserver_c ServerT>
|
||||||
|
class AdcNetServerSpdlogDecorator : public ServerT
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
std::shared_ptr<spdlog::logger> _logger;
|
||||||
|
|
||||||
|
void* _thisAddress;
|
||||||
|
std::string _serverID;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using typename ServerT::server_ident_t;
|
||||||
|
|
||||||
|
template <typename... ImplCtorArgTs>
|
||||||
|
AdcNetServerSpdlogDecorator(std::shared_ptr<spdlog::logger> logger, ImplCtorArgTs&&... ctor_args)
|
||||||
|
: ServerT(std::forward<ImplCtorArgTs>(ctor_args)...), _logger(logger), _thisAddress((void*)std::addressof(this))
|
||||||
|
{
|
||||||
|
fmt::format_to(std::back_inserter(_serverID), "{}", ServerT::serverIdent());
|
||||||
|
|
||||||
|
_logger->trace("Call AdcNetServerSpdlogDecorator::AdcNetServerSpdlogDecorator (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
_logger->debug("Creating network server session with ID = [{}] (ADDR = {})", _serverID, _thisAddress);
|
||||||
|
_logger->debug("use logger with name [{}] and level [{}]", _logger->name(),
|
||||||
|
spdlog::level::to_string_view(_logger->level()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
virtual ~AdcNetServerSpdlogDecorator()
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSpdlogDecorator::~AdcNetServerSpdlogDecorator (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
_logger->debug("Deleting network server with ID = [{}] (ADDR = {})", _serverID, _thisAddress);
|
||||||
|
|
||||||
|
_logger->flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
virtual server_ident_t serverIdent() const
|
||||||
|
{
|
||||||
|
//
|
||||||
|
return ServerT::serverIdent();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void start()
|
||||||
|
{
|
||||||
|
_logger->trace("Call AdcNetServerSpdlogDecorator::start (this: {})", _thisAddress);
|
||||||
|
|
||||||
|
_logger->info("Starting network server with ID = [{}] (ADDR = {})", _serverID, _thisAddress);
|
||||||
|
|
||||||
|
ServerT::start();
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void stop()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
ServerT::stop();
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace adc
|
||||||
|
|
||||||
|
#endif
|
||||||
Loading…
x
Reference in New Issue
Block a user