diff --git a/CMakeLists.txt b/CMakeLists.txt index 19818aa..000625a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -38,6 +38,8 @@ else() add_definitions(-D_WIN32_WINNT=0x0A00) endif(NOT WIN32) +find_package(ASIO REQUIRED MODULE) + add_subdirectory(src) option(BUILD_EVERYTHING "Build UnitTests & Console Logger" ON) diff --git a/cmake/FindASIO.cmake b/cmake/FindASIO.cmake index b6d0559..4ee4b5b 100644 --- a/cmake/FindASIO.cmake +++ b/cmake/FindASIO.cmake @@ -18,3 +18,14 @@ mark_as_advanced( ASIO_ROOT_DIR ASIO_INCLUDE_DIR ) + +if(ASIO_FOUND) + set(ASIO_INCLUDE_DIRS ${ASIO_INCLUDE_DIR}) +endif() + +if(ASIO_FOUND AND NOT TARGET ASIO::ASIO) + add_library(ASIO::ASIO INTERFACE IMPORTED) + set_target_properties(ASIO::ASIO PROPERTIES + INTERFACE_INCLUDE_DIRECTORIES "${ASIO_INCLUDE_DIR}" + ) +endif() diff --git a/cmake/GraylogLoggerConfig.cmake.in b/cmake/GraylogLoggerConfig.cmake.in index 5f98ef3..16807cd 100644 --- a/cmake/GraylogLoggerConfig.cmake.in +++ b/cmake/GraylogLoggerConfig.cmake.in @@ -1,8 +1,12 @@ -get_filename_component(GraylogLogger_CMAKE_DIR "${CMAKE_CURRENT_LIST_FILE}" PATH) +get_filename_component(GraylogLogger_CMAKE_DIR "${CMAKE_CURRENT_LIST_FILE}" DIRECTORY) include(CMakeFindDependencyMacro) +set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_CURRENT_LIST_DIR}/../cmake/") + +find_dependency(Threads REQUIRED) + if(NOT TARGET GraylogLogger::graylog_logger) include("${GraylogLogger_CMAKE_DIR}/GraylogLoggerTargets.cmake") endif() -set(GraylogLogger_LIBRARIES GraylogLogger::graylog_logger) \ No newline at end of file +set(GraylogLogger_LIBRARIES GraylogLogger::graylog_logger) diff --git a/conanfile.txt b/conanfile.txt index 332ac05..08de515 100644 --- a/conanfile.txt +++ b/conanfile.txt @@ -4,7 +4,7 @@ asio/1.13.0@bincrafters/stable jsonformoderncpp/3.6.1@vthiery/stable [options] -gtest:shared=True +gtest:shared=False [generators] cmake diff --git a/include/graylog_logger/ConnectionStatus.hpp b/include/graylog_logger/ConnectionStatus.hpp new file mode 100644 index 0000000..b1c4f30 --- /dev/null +++ b/include/graylog_logger/ConnectionStatus.hpp @@ -0,0 +1,18 @@ +/* Copyright (C) 2019 European Spallation Source, ERIC. See LICENSE file */ +//===----------------------------------------------------------------------===// +/// +/// \file +/// +/// \brief Graylog connection status. +/// +//===----------------------------------------------------------------------===// +#pragma once + +namespace Log { +enum class Status { + ADDR_LOOKUP, + ADDR_RETRY_WAIT, + CONNECT, + SEND_LOOP, +}; +} // namespace Log diff --git a/include/graylog_logger/GraylogInterface.hpp b/include/graylog_logger/GraylogInterface.hpp index d96a8a3..7c30ae7 100644 --- a/include/graylog_logger/GraylogInterface.hpp +++ b/include/graylog_logger/GraylogInterface.hpp @@ -9,12 +9,25 @@ #pragma once -#include "graylog_logger/GraylogConnection.hpp" +#include "graylog_logger/ConnectionStatus.hpp" #include "graylog_logger/LogUtil.hpp" namespace Log { +class GraylogConnection { +public: + using Status = Log::Status; + GraylogConnection(std::string Host, int Port); + virtual ~GraylogConnection(); + virtual void sendMessage(std::string Msg); + virtual Status getConnectionStatus() const; + virtual bool messageQueueEmpty(); + virtual size_t messageQueueSize(); -class GraylogInterface : public BaseLogHandler, private GraylogConnection { +private: + class Impl; + std::unique_ptr Pimpl; +}; +class GraylogInterface : public BaseLogHandler, public GraylogConnection { public: GraylogInterface(const std::string &Host, int Port, size_t MaxQueueLength = 100); @@ -22,8 +35,6 @@ class GraylogInterface : public BaseLogHandler, private GraylogConnection { void addMessage(const LogMessage &Message) override; bool emptyQueue() override; size_t queueSize() override; - using GraylogConnection::Status; - using GraylogConnection::getConnectionStatus; protected: std::string logMsgToJSON(const LogMessage &Message); diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index d5112ef..22c1428 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1,4 +1,5 @@ find_package(Threads REQUIRED) +find_package(nlohmann_json REQUIRED) set(Graylog_SRC ConsoleInterface.cpp @@ -15,19 +16,21 @@ set(Graylog_INC ../include/graylog_logger/ConcurrentQueue.hpp ../include/graylog_logger/ConsoleInterface.hpp ../include/graylog_logger/FileInterface.hpp - ../include/graylog_logger/GraylogConnection.hpp + GraylogConnection.hpp ../include/graylog_logger/GraylogInterface.hpp ../include/graylog_logger/Log.hpp ../include/graylog_logger/Logger.hpp ../include/graylog_logger/LoggingBase.hpp ../include/graylog_logger/LogUtil.hpp ../include/graylog_logger/ThreadedExecutor.hpp -) - + ../include/graylog_logger/ConnectionStatus.hpp) + set(common_libs - Threads::Threads - CONAN_PKG::asio - CONAN_PKG::jsonformoderncpp) + PUBLIC + Threads::Threads + ) + +get_target_property(JSON_INCLUDE_DIR nlohmann_json::nlohmann_json INTERFACE_INCLUDE_DIRECTORIES) add_library(graylog_logger SHARED ${Graylog_SRC} ${Graylog_INC}) add_library(GraylogLogger::graylog_logger ALIAS graylog_logger) @@ -38,6 +41,8 @@ target_include_directories(graylog_logger $ PRIVATE "." + ${ASIO_INCLUDE_DIR} + ${JSON_INCLUDE_DIR} ) add_library(graylog_logger_static STATIC ${Graylog_SRC} ${Graylog_INC} ${Graylog_private_INC}) @@ -49,6 +54,8 @@ target_include_directories(graylog_logger_static $ PRIVATE "." + ${ASIO_INCLUDE_DIR} + ${JSON_INCLUDE_DIR} ) set_target_properties(graylog_logger PROPERTIES VERSION ${PROJECT_VERSION}) diff --git a/src/GraylogConnection.cpp b/src/GraylogConnection.cpp index 1294a53..52a5c21 100644 --- a/src/GraylogConnection.cpp +++ b/src/GraylogConnection.cpp @@ -8,14 +8,14 @@ /// //===----------------------------------------------------------------------===// -#include "graylog_logger/GraylogConnection.hpp" +#include "GraylogConnection.hpp" #include #include #include namespace { #define UNUSED_ARG(x) (void)x; -} +} // namespace namespace Log { @@ -46,7 +46,7 @@ struct QueryResult { int NextEndpoint{0}; }; -void GraylogConnection::tryConnect(QueryResult AllEndpoints) { +void GraylogConnection::Impl::tryConnect(QueryResult AllEndpoints) { asio::ip::tcp::endpoint CurrentEndpoint = AllEndpoints.getNextEndpoint(); auto HandlerGlue = [this, AllEndpoints](auto &Err) { this->connectHandler(Err, AllEndpoints); @@ -55,15 +55,15 @@ void GraylogConnection::tryConnect(QueryResult AllEndpoints) { setState(Status::CONNECT); } -GraylogConnection::GraylogConnection(std::string Host, int Port) +GraylogConnection::Impl::Impl(std::string Host, int Port) : HostAddress(std::move(Host)), HostPort(std::to_string(Port)), Service(), Work(std::make_unique(Service)), Socket(Service), Resolver(Service), ReconnectTimeout(Service, 10s) { doAddressQuery(); - AsioThread = std::thread(&GraylogConnection::threadFunction, this); + AsioThread = std::thread(&GraylogConnection::Impl::threadFunction, this); } -void GraylogConnection::resolverHandler( +void GraylogConnection::Impl::resolverHandler( const asio::error_code &Error, asio::ip::tcp::resolver::iterator EndpointIter) { if (Error) { @@ -75,8 +75,8 @@ void GraylogConnection::resolverHandler( tryConnect(AllEndpoints); } -void GraylogConnection::connectHandler(const asio::error_code &Error, - const QueryResult &AllEndpoints) { +void GraylogConnection::Impl::connectHandler(const asio::error_code &Error, + const QueryResult &AllEndpoints) { if (!Error) { setState(Status::SEND_LOOP); auto HandlerGlue = [this](auto &Error, auto Size) { @@ -94,7 +94,7 @@ void GraylogConnection::connectHandler(const asio::error_code &Error, tryConnect(AllEndpoints); } -void GraylogConnection::reConnect(ReconnectDelay Delay) { +void GraylogConnection::Impl::reConnect(ReconnectDelay Delay) { auto HandlerGlue = [this](auto &Err) { this->doAddressQuery(); }; switch (Delay) { case ReconnectDelay::SHORT: @@ -109,8 +109,8 @@ void GraylogConnection::reConnect(ReconnectDelay Delay) { setState(Status::ADDR_RETRY_WAIT); } -void GraylogConnection::receiveHandler(const asio::error_code &Error, - std::size_t BytesReceived) { +void GraylogConnection::Impl::receiveHandler(const asio::error_code &Error, + std::size_t BytesReceived) { UNUSED_ARG(BytesReceived); if (Error) { Socket.close(); @@ -123,7 +123,7 @@ void GraylogConnection::receiveHandler(const asio::error_code &Error, Socket.async_receive(asio::buffer(InputBuffer), HandlerGlue); } -void GraylogConnection::trySendMessage() { +void GraylogConnection::Impl::trySendMessage() { if (not Socket.is_open()) { return; } @@ -149,8 +149,8 @@ void GraylogConnection::trySendMessage() { } } -void GraylogConnection::sentMessageHandler(const asio::error_code &Error, - std::size_t BytesSent) { +void GraylogConnection::Impl::sentMessageHandler(const asio::error_code &Error, + std::size_t BytesSent) { if (BytesSent == MessageBuffer.size()) { MessageBuffer.clear(); } else if (BytesSent > 0) { @@ -166,7 +166,7 @@ void GraylogConnection::sentMessageHandler(const asio::error_code &Error, trySendMessage(); } -void GraylogConnection::waitForMessage() { +void GraylogConnection::Impl::waitForMessage() { if (not Socket.is_open()) { return; } @@ -179,7 +179,7 @@ void GraylogConnection::waitForMessage() { Service.post([this]() { this->waitForMessage(); }); } -void GraylogConnection::doAddressQuery() { +void GraylogConnection::Impl::doAddressQuery() { setState(Status::ADDR_LOOKUP); asio::ip::tcp::resolver::query Query(HostAddress, HostPort); auto HandlerGlue = [this](auto &Error, auto EndpointIter) { @@ -188,7 +188,7 @@ void GraylogConnection::doAddressQuery() { Resolver.async_resolve(Query, HandlerGlue); } -GraylogConnection::~GraylogConnection() { +GraylogConnection::Impl::~Impl() { Service.stop(); AsioThread.join(); try { @@ -198,13 +198,15 @@ GraylogConnection::~GraylogConnection() { } } -GraylogConnection::Status GraylogConnection::getConnectionStatus() const { +GraylogConnection::Impl::Status +GraylogConnection::Impl::getConnectionStatus() const { return ConnectionState; } -void GraylogConnection::threadFunction() { Service.run(); } +void GraylogConnection::Impl::threadFunction() { Service.run(); } -void GraylogConnection::setState(GraylogConnection::Status NewState) { +void GraylogConnection::Impl::setState( + GraylogConnection::Impl::Status NewState) { ConnectionState = NewState; } diff --git a/include/graylog_logger/GraylogConnection.hpp b/src/GraylogConnection.hpp similarity index 83% rename from include/graylog_logger/GraylogConnection.hpp rename to src/GraylogConnection.hpp index cb169ab..ec0541f 100644 --- a/include/graylog_logger/GraylogConnection.hpp +++ b/src/GraylogConnection.hpp @@ -10,6 +10,8 @@ #pragma once #include "graylog_logger/ConcurrentQueue.hpp" +#include "graylog_logger/ConnectionStatus.hpp" +#include "graylog_logger/GraylogInterface.hpp" #include #include #include @@ -24,18 +26,15 @@ struct QueryResult; /// \todo Implement timeouts in the ASIO code in case we ever have problems with /// bad connections. -class GraylogConnection { +class GraylogConnection::Impl { public: - GraylogConnection(std::string Host, int Port); - virtual ~GraylogConnection(); - virtual void sendMessage(std::string msg) { LogMessages.push(msg); }; - enum class Status { - ADDR_LOOKUP, - ADDR_RETRY_WAIT, - CONNECT, - SEND_LOOP, - }; + using Status = Log::Status; + Impl(std::string Host, int Port); + virtual ~Impl(); + virtual void sendMessage(std::string Msg) { LogMessages.push(Msg); }; Status getConnectionStatus() const; + bool messageQueueEmpty() { return LogMessages.empty(); } + size_t messageQueueSize() { return LogMessages.size(); } protected: enum class ReconnectDelay { LONG, SHORT }; diff --git a/src/GraylogInterface.cpp b/src/GraylogInterface.cpp index b667a69..174c0b0 100644 --- a/src/GraylogInterface.cpp +++ b/src/GraylogInterface.cpp @@ -9,26 +9,44 @@ //===----------------------------------------------------------------------===// #include "graylog_logger/GraylogInterface.hpp" +#include "GraylogConnection.hpp" #include #include #include namespace Log { +GraylogConnection::GraylogConnection(std::string Host, int Port) + : Pimpl(std::make_unique(std::move(Host), Port)) {} + +void GraylogConnection::sendMessage(std::string Msg) { + Pimpl->sendMessage(Msg); +} + +Status GraylogConnection::getConnectionStatus() const { + return Pimpl->getConnectionStatus(); +} + +bool GraylogConnection::messageQueueEmpty() { + return Pimpl->messageQueueEmpty(); +} + +size_t GraylogConnection::messageQueueSize() { + return Pimpl->messageQueueSize(); +} + +GraylogConnection::~GraylogConnection() {} + GraylogInterface::GraylogInterface(const std::string &Host, const int Port, const size_t MaxQueueLength) : BaseLogHandler(MaxQueueLength), GraylogConnection(Host, Port) {} -bool GraylogInterface::emptyQueue() { - return GraylogConnection::LogMessages.empty(); -} +bool GraylogInterface::emptyQueue() { return messageQueueEmpty(); } -size_t GraylogInterface::queueSize() { - return GraylogConnection::LogMessages.size(); -} +size_t GraylogInterface::queueSize() { return messageQueueSize(); } void GraylogInterface::addMessage(const LogMessage &Message) { - if (GraylogConnection::LogMessages.size() < BaseLogHandler::QueueLength) { + if (messageQueueSize() < BaseLogHandler::QueueLength) { sendMessage(logMsgToJSON(Message)); } } diff --git a/unit_tests/CMakeLists.txt b/unit_tests/CMakeLists.txt index 875bd8d..6a1d32e 100644 --- a/unit_tests/CMakeLists.txt +++ b/unit_tests/CMakeLists.txt @@ -29,6 +29,8 @@ add_executable(unit_tests ${UnitTest_SRC} ${UnitTest_INC}) target_link_libraries(unit_tests GraylogLogger::graylog_logger_static - CONAN_PKG::gtest) + CONAN_PKG::jsonformoderncpp + CONAN_PKG::gtest + ASIO::ASIO) add_test(TestAll unit_tests) diff --git a/unit_tests/QueueLengthTest.cpp b/unit_tests/QueueLengthTest.cpp index 4355de9..1771369 100644 --- a/unit_tests/QueueLengthTest.cpp +++ b/unit_tests/QueueLengthTest.cpp @@ -9,6 +9,7 @@ #include "graylog_logger/ConsoleInterface.hpp" #include "graylog_logger/FileInterface.hpp" #include "graylog_logger/GraylogInterface.hpp" +#include #include #include #include