diff --git a/CONTRIBUTORS.txt b/CONTRIBUTORS.txt index 6fd1fb414..6069c8725 100644 --- a/CONTRIBUTORS.txt +++ b/CONTRIBUTORS.txt @@ -22,6 +22,7 @@ Authors from Codership Oy: * Daniele Sciascia , Codership Oy * Philip Stoev , Codership Oy * Mario Karuza , Codership Oy + * Jan Lindström , Codership Oy [Codership employees, add name and email/username above this line, but leave this line intact] Other contributors: diff --git a/galera/src/galera-sym.map b/galera/src/galera-sym.map index 12c349857..87e406dbc 100644 --- a/galera/src/galera-sym.map +++ b/galera/src/galera-sym.map @@ -18,6 +18,8 @@ wsrep_init_config_service_v1; wsrep_deinit_config_service_v1; wsrep_node_isolation_mode_set_v1; + wsrep_init_connection_monitor_service_v1; + wsrep_deinit_connection_monitor_service_v1; local: *; }; diff --git a/galera/src/wsrep_provider.cpp b/galera/src/wsrep_provider.cpp index 1d1c325c9..898e17001 100644 --- a/galera/src/wsrep_provider.cpp +++ b/galera/src/wsrep_provider.cpp @@ -1,5 +1,5 @@ // -// Copyright (C) 2010-2021 Codership Oy +// Copyright (C) 2010-2024 Codership Oy // #include "wsrep_api.h" @@ -23,6 +23,7 @@ #include "gu_event_service.hpp" #include "wsrep_config_service.h" #include "wsrep_node_isolation.h" +#include "wsrep_connection_monitor_service.h" #include @@ -1941,4 +1942,14 @@ wsrep_node_isolation_mode_set_v1(enum wsrep_node_isolation_mode mode) return WSREP_NODE_ISOLATION_SUCCESS; } +extern "C" +int wsrep_init_connection_monitor_service_v1(wsrep_connection_monitor_service_v1_t *connection_monitor_service) +{ + return gu::init_connection_monitor_service_v1(connection_monitor_service); +} + +extern "C" void wsrep_deinit_connection_monitor_service_v1() +{ + gu::deinit_connection_monitor_service_v1(); +} diff --git a/galerautils/src/gu_asio.cpp b/galerautils/src/gu_asio.cpp index d40ecddbe..4a1910a66 100644 --- a/galerautils/src/gu_asio.cpp +++ b/galerautils/src/gu_asio.cpp @@ -1,5 +1,5 @@ // -// Copyright (C) 2014-2020 Codership Oy +// Copyright (C) 2014-2024 Codership Oy // #include "gu_config.hpp" @@ -53,6 +53,8 @@ static wsrep_tls_service_v1_t* gu_tls_service(0); static wsrep_allowlist_service_v1_t* gu_allowlist_service(0); +static wsrep_connection_monitor_service_v1_t* gu_connection_monitor_service(0); + // // AsioIpAddress wrapper // @@ -955,3 +957,100 @@ void gu::deinit_allowlist_service_v1() std::atomic gu::gu_asio_node_isolation_mode{ WSREP_NODE_ISOLATION_NOT_ISOLATED }; + +// +// ConnectionMonitor +// + +static std::mutex gu_connection_monitor_service_init_mutex; +static size_t gu_connection_monitor_service_usage; + +int gu::init_connection_monitor_service_v1(wsrep_connection_monitor_service_v1_t* connection_monitor) +{ + log_info << "init_connection_monitor_service_v1"; + std::lock_guard lock(gu_connection_monitor_service_init_mutex); + ++gu_connection_monitor_service_usage; + if (gu_connection_monitor_service) + { + assert(gu_connection_monitor_service == connection_monitor); + return 0; + } + gu_connection_monitor_service = connection_monitor; + return 0; +} + +void gu::deinit_connection_monitor_service_v1() +{ + log_info << "deinit_connection_monitor_service_v1"; + std::lock_guard lock(gu_connection_monitor_service_init_mutex); + assert(gu_connection_monitor_service_usage > 0); + --gu_connection_monitor_service_usage; + if (gu_connection_monitor_service_usage == 0) + gu_connection_monitor_service = 0; +} + +void gu::connection_monitor_connect(wsrep_connection_key_t id, + const std::string& scheme, + const std::string& local_addr, + const std::string& remote_uuid, + const std::string& remote_addr) +{ + if (gu_connection_monitor_service == nullptr) + { + return; // No action + } + + // tcp://127.0.0.1:19006 --> 127.0.0.1:19006 + std::string ra = remote_addr.substr(6, remote_addr.length()); + std::string la = local_addr.substr(6, local_addr.length()); + + wsrep_buf_t const remote = {remote_uuid.c_str(), remote_uuid.length() }; + wsrep_buf_t const lscheme = {scheme.c_str(), scheme.length() }; + wsrep_buf_t const raddr = {ra.c_str(), ra.length() }; + wsrep_buf_t const laddr = {la.c_str(), la.length() }; + + gu_connection_monitor_service->connection_monitor_connect_cb( + gu_connection_monitor_service->context, + id, + &lscheme, + &laddr, + &remote, + &raddr); +} + +void gu::connection_monitor_disconnect(wsrep_connection_key_t id) +{ + if (gu_connection_monitor_service == nullptr) + { + return; // No action + } + + gu_connection_monitor_service->connection_monitor_disconnect_cb( + gu_connection_monitor_service->context, + id); +} + +void gu::connection_monitor_ssl_info(wsrep_connection_key_t id, + const std::string& chipher, + const std::string& issuer, + const std::string& subject, + const std::string& version) +{ + if (gu_connection_monitor_service == nullptr) + { + return; // No action + } + + wsrep_buf_t const ch = {chipher.c_str(), chipher.length() }; + wsrep_buf_t const iss = {issuer.c_str(), issuer.length() }; + wsrep_buf_t const sub = {subject.c_str(), subject.length() }; + wsrep_buf_t const vers = {version.c_str(), version.length() }; + + gu_connection_monitor_service->connection_monitor_ssl_info_cb( + gu_connection_monitor_service->context, + id, + &ch, + &iss, + &sub, + &vers); +} diff --git a/galerautils/src/gu_asio.hpp b/galerautils/src/gu_asio.hpp index c1254d04b..9d9eb2a27 100644 --- a/galerautils/src/gu_asio.hpp +++ b/galerautils/src/gu_asio.hpp @@ -17,6 +17,7 @@ #include "wsrep_tls_service.h" #include "wsrep_allowlist_service.h" #include "wsrep_node_isolation.h" +#include "wsrep_connection_monitor_service.h" #include // tcp_info @@ -808,6 +809,23 @@ namespace gu extern std::atomic gu_asio_node_isolation_mode; + /* Init/deinit global connection monitoring service hooks */ + int init_connection_monitor_service_v1(wsrep_connection_monitor_service_v1_t*); + void deinit_connection_monitor_service_v1(); + /* Connection monitor connect callback */ + void connection_monitor_connect(wsrep_connection_key_t id, + const std::string& scheme, + const std::string& local_addr, + const std::string& remote_uuid, + const std::string& remote_addr); + /* Connection monitor disconnect callback */ + void connection_monitor_disconnect(wsrep_connection_key_t id); + /* Connection monitor ssl info callback */ + void connection_monitor_ssl_info(wsrep_connection_key_t id, + const std::string& chipher, + const std::string& issuer, + const std::string& subject, + const std::string& version); } #endif // GU_ASIO_HPP diff --git a/galerautils/src/gu_asio_stream_engine.cpp b/galerautils/src/gu_asio_stream_engine.cpp index f5e047cea..1dd69ddbe 100644 --- a/galerautils/src/gu_asio_stream_engine.cpp +++ b/galerautils/src/gu_asio_stream_engine.cpp @@ -90,6 +90,10 @@ class AsioTcpStreamEngine : public gu::AsioStreamEngine { return gu::AsioErrorCode(last_error_, gu_asio_system_category); } + + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) GALERA_OVERRIDE {} + private: void clear_error() { last_error_ = 0; } int fd_; @@ -99,7 +103,7 @@ class AsioTcpStreamEngine : public gu::AsioStreamEngine #ifdef GALERA_HAVE_SSL #include - +#include #if OPENSSL_VERSION_NUMBER >= 0x1010100fL #define HAVE_READ_EX #define HAVE_WRITE_EX @@ -187,6 +191,21 @@ class AsioSslStreamEngine : public gu::AsioStreamEngine last_verify_error_); } + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) GALERA_OVERRIDE + { + clear_error(); + chipher = SSL_get_cipher(ssl_); + X509 *ssl_cert = SSL_get_peer_certificate(ssl_); + if (ssl_cert != nullptr) + { + subject = X509_NAME_oneline(X509_get_subject_name(ssl_cert), 0, 0); + issuer = X509_NAME_oneline(X509_get_issuer_name(ssl_cert), 0, 0); + X509_free(ssl_cert); + } + version = SSL_get_version(ssl_); + } + private: void clear_error() { @@ -560,6 +579,9 @@ class AsioDynamicStreamEngine : public gu::AsioStreamEngine return engine_->last_error(); } + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) GALERA_OVERRIDE {} + private: bool socket_poll(long msec) { @@ -687,6 +709,9 @@ class AsioWsrepStreamEngine : public gu::AsioStreamEngine return gu::AsioErrorCode(last_error_value_, last_error_category_, &stream_); } + + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) GALERA_OVERRIDE {} private: enum op_status map_status(enum wsrep_tls_result status) diff --git a/galerautils/src/gu_asio_stream_engine.hpp b/galerautils/src/gu_asio_stream_engine.hpp index 16fc88880..918185775 100644 --- a/galerautils/src/gu_asio_stream_engine.hpp +++ b/galerautils/src/gu_asio_stream_engine.hpp @@ -117,6 +117,12 @@ namespace gu AsioIoService&, const std::string& scheme, int fd, bool non_blocking); + /** + * Fetch SSL/TLS information + */ + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) = 0; + protected: AsioStreamEngine() { } }; diff --git a/galerautils/src/gu_asio_stream_react.cpp b/galerautils/src/gu_asio_stream_react.cpp index df09c28ab..74807c641 100644 --- a/galerautils/src/gu_asio_stream_react.cpp +++ b/galerautils/src/gu_asio_stream_react.cpp @@ -91,6 +91,8 @@ void gu::AsioStreamReact::close() try { GU_ASIO_DEBUG(debug_print() << "Socket not open on close"); } + + gu::connection_monitor_disconnect((wsrep_connection_key_t)this); socket_.close(); } // Catch all the possible exceptions here, not only asio ones. @@ -198,11 +200,32 @@ void gu::AsioStreamReact::connect(const gu::URI& uri) try socket_.connect(resolve_result->endpoint()); connected_ = true; prepare_engine(false); + assign_addresses(); + auto result(engine_->client_handshake()); switch (result) { case AsioStreamEngine::success: + { + gu::connection_monitor_connect((wsrep_connection_key_t)this, + scheme_, + local_addr_, + "", // remote uuid + remote_addr_); +#ifdef GALERA_HAVE_SSL + if (engine_->scheme() == gu::scheme::ssl) + { + std::string chipher, issuer, subject, version; + engine_->get_SSL_info(chipher, issuer, subject, version); + gu::connection_monitor_ssl_info((wsrep_connection_key_t)this, + chipher, + issuer, + subject, + version); + } +#endif return; + } case AsioStreamEngine::want_read: case AsioStreamEngine::want_write: case AsioStreamEngine::eof: @@ -389,6 +412,13 @@ void gu::AsioStreamReact::connect_handler( set_socket_options(socket_); prepare_engine(true); assign_addresses(); + + gu::connection_monitor_connect((wsrep_connection_key_t)this, + scheme_, + local_addr_, + "", // remote uuid + remote_addr_); + GU_ASIO_DEBUG(debug_print() << " AsioStreamReact::connect_handler: init handshake"); auto result(engine_->client_handshake()); diff --git a/galerautils/tests/gu_asio_test.cpp b/galerautils/tests/gu_asio_test.cpp index 6cf254b52..a69af9a41 100644 --- a/galerautils/tests/gu_asio_test.cpp +++ b/galerautils/tests/gu_asio_test.cpp @@ -98,6 +98,9 @@ class MockStreamEngine : public gu::AsioStreamEngine } } + virtual void get_SSL_info(std::string &chipher, std::string &subject, + std::string &issuer, std::string &version) GALERA_OVERRIDE {} + enum op_status next_result; int next_error; size_t count_client_handshake_called; diff --git a/gcomm/src/asio_tcp.hpp b/gcomm/src/asio_tcp.hpp index 1af8da345..d6baf26bf 100644 --- a/gcomm/src/asio_tcp.hpp +++ b/gcomm/src/asio_tcp.hpp @@ -55,7 +55,7 @@ class gcomm::AsioTcpSocket : virtual std::string local_addr() const GALERA_OVERRIDE; virtual std::string remote_addr() const GALERA_OVERRIDE; virtual State state() const GALERA_OVERRIDE { return state_; } - virtual SocketId id() const GALERA_OVERRIDE { return &socket_; } + virtual SocketId id() const GALERA_OVERRIDE { return socket_.get(); } virtual SocketStats stats() const GALERA_OVERRIDE; private: // AsioSocketHandler interface diff --git a/gcomm/src/gmcast.cpp b/gcomm/src/gmcast.cpp index dda401503..23dabaded 100644 --- a/gcomm/src/gmcast.cpp +++ b/gcomm/src/gmcast.cpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2009-2019 Codership Oy + * Copyright (C) 2009-2024 Codership Oy */ #include "gmcast.hpp" @@ -584,6 +584,14 @@ void gcomm::GMCast::gmcast_connect(const std::string& remote_addr) segment_, group_name_); + std::ostringstream os; + os << peer->remote_uuid().full_str(); + gu::connection_monitor_connect((wsrep_connection_key_t)tp->id(), + get_scheme(pnet_, use_ssl_, dynamic_socket_), + peer->local_addr(), + os.str(), + remote_addr); + std::pair ret = proto_map_->insert(std::make_pair(tp->id(), peer)); @@ -676,6 +684,14 @@ void gcomm::GMCast::handle_established(Proto* est) // UUID checks are handled during protocol handshake assert(est->remote_uuid() != uuid()); + std::ostringstream os; + os << est->remote_uuid().full_str(); + gu::connection_monitor_connect((wsrep_connection_key_t)est->socket()->id(), + get_scheme(pnet_, use_ssl_, dynamic_socket_), + est->local_addr(), + os.str(), + est->remote_addr()); + if (is_evicted(est->remote_uuid())) { log_warn << "Closing connection to evicted node " << est->remote_uuid(); diff --git a/gcomm/src/gmcast_proto.hpp b/gcomm/src/gmcast_proto.hpp index aedf60deb..9054ffe30 100644 --- a/gcomm/src/gmcast_proto.hpp +++ b/gcomm/src/gmcast_proto.hpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2009-2019 Codership Oy + * Copyright (C) 2009-2024 Codership Oy */ #ifndef GCOMM_GMCAST_PROTO_HPP @@ -150,6 +150,7 @@ class gcomm::gmcast::Proto SocketPtr socket() const { return tp_; } const std::string& remote_addr() const { return remote_addr_; } + const std::string& local_addr() const { return local_addr_; } const std::string& mcast_addr() const { return mcast_addr_; } const LinkMap& link_map() const { return link_map_; } diff --git a/wsrep/src b/wsrep/src index b03acea36..ba314b9e7 160000 --- a/wsrep/src +++ b/wsrep/src @@ -1 +1 @@ -Subproject commit b03acea3661835a48f52ed44dc0f6e7f037390a0 +Subproject commit ba314b9e709c91252667ce5e67cef8053a436bf5