diff --git a/lib/remote/apilistener.cpp b/lib/remote/apilistener.cpp index d680cc76a..578e1a682 100644 --- a/lib/remote/apilistener.cpp +++ b/lib/remote/apilistener.cpp @@ -23,9 +23,13 @@ #include #include #include +#include +#include #include #include #include +#include +#include using namespace icinga; @@ -373,15 +377,36 @@ bool ApiListener::AddListener(const String& node, const String& service) Log(LogInformation, "ApiListener") << "Started new listener on '[" << localEndpoint.address() << "]:" << localEndpoint.port() << "'"; - asio::spawn(io, [acceptor](asio::yield_context yc) { - // TODO - }); + asio::spawn(io, [this, acceptor, sslContext](asio::yield_context yc) { ListenerCoroutineProc(yc, acceptor, sslContext); }); UpdateStatusFile(localEndpoint); return true; } +void ApiListener::ListenerCoroutineProc(boost::asio::yield_context yc, const std::shared_ptr& server, const std::shared_ptr& sslContext) +{ + namespace asio = boost::asio; + namespace ssl = asio::ssl; + using asio::ip::tcp; + + auto& io (server->get_io_service()); + auto sslConn (std::make_shared>(io, *sslContext)); + + for (;;) { + try { + server->async_accept(sslConn->lowest_layer(), yc); + } catch (const std::exception& ex) { + Log(LogCritical, "ApiListener") << "Cannot accept new connection: " << DiagnosticInformation(ex, false); + continue; + } + + asio::spawn(io, [this, sslConn](asio::yield_context yc) { NewClientHandler(yc, sslConn, String(), RoleServer); }); + + sslConn = std::make_shared>(io, *sslContext); + } +} + /** * Creates a new JSON-RPC client and connects to the specified endpoint. * @@ -601,6 +626,70 @@ void ApiListener::NewClientHandlerInternal(const Socket::Ptr& client, const Stri } } +void ApiListener::NewClientHandler(boost::asio::yield_context yc, const std::shared_ptr>& client, const String& hostname, ConnectionRole role) +{ + try { + NewClientHandlerInternal(yc, client, hostname, role); + } catch (const std::exception& ex) { + Log(LogCritical, "ApiListener") + << "Exception while handling new API client connection: " << DiagnosticInformation(ex, false); + + Log(LogDebug, "ApiListener") + << "Exception while handling new API client connection: " << DiagnosticInformation(ex); + } +} + +/** + * Processes a new client connection. + * + * @param client The new client. + */ +void ApiListener::NewClientHandlerInternal(boost::asio::yield_context yc, const std::shared_ptr>& client, const String& hostname, ConnectionRole role) +{ + namespace ssl = boost::asio::ssl; + + String conninfo; + + { + std::ostringstream conninfo_; + + if (role == RoleClient) { + conninfo_ << "to"; + } else { + conninfo_ << "from"; + } + + auto endpoint (client->lowest_layer().remote_endpoint()); + + conninfo_ << " [" << endpoint.address() << "]:" << endpoint.port(); + + conninfo = conninfo_.str(); + } + + client->set_verify_mode(ssl::verify_peer | ssl::verify_client_once); + + if (role == RoleClient) { + String environmentName = Application::GetAppEnvironment(); + String serverName = hostname; + + if (!environmentName.IsEmpty()) + serverName += ":" + environmentName; + +#ifdef SSL_CTRL_SET_TLSEXT_HOSTNAME + if (!hostname.IsEmpty()) { + SSL_set_tlsext_host_name(client->native_handle(), serverName.CStr()); + } +#endif /* SSL_CTRL_SET_TLSEXT_HOSTNAME */ + } + + try { + client->async_handshake(role == RoleClient ? client->client : client->server, yc); + } catch (const std::exception& ex) { + Log(LogCritical, "ApiListener") + << "Client TLS handshake failed (" << conninfo << "): " << DiagnosticInformation(ex, false); + } +} + void ApiListener::SyncClient(const JsonRpcConnection::Ptr& aclient, const Endpoint::Ptr& endpoint, bool needSync) { Zone::Ptr eZone = endpoint->GetZone(); diff --git a/lib/remote/apilistener.hpp b/lib/remote/apilistener.hpp index 1de66ed4f..e8b578aba 100644 --- a/lib/remote/apilistener.hpp +++ b/lib/remote/apilistener.hpp @@ -15,7 +15,9 @@ #include "base/tlsstream.hpp" #include "base/threadpool.hpp" #include +#include #include +#include #include namespace icinga @@ -132,6 +134,10 @@ private: void NewClientHandler(const Socket::Ptr& client, const String& hostname, ConnectionRole role); void NewClientHandlerInternal(const Socket::Ptr& client, const String& hostname, ConnectionRole role); + void NewClientHandler(boost::asio::yield_context yc, const std::shared_ptr>& client, const String& hostname, ConnectionRole role); + void NewClientHandlerInternal(boost::asio::yield_context yc, const std::shared_ptr>& client, const String& hostname, ConnectionRole role); + void ListenerCoroutineProc(boost::asio::yield_context yc, const std::shared_ptr& server, const std::shared_ptr& sslContext); + static ThreadPool& GetTP(); static void EnqueueAsyncCallback(const std::function& callback, SchedulerPolicy policy = DefaultScheduler);