Skip to content

Commit

Permalink
HttpServerConnection: Don't spawn useless coroutines
Browse files Browse the repository at this point in the history
Currently, for each `Disconnect()` call, we spawn a coroutine, but every
one of them is just usesless, except the first one. However, since all
`Disconnect()` usages share the same asio strand and cannot interfere
with each other, spawning another coroutine within `Disconnect()` isn't
even necessary. When a coroutine calls `Disconnect()` now, it will
immediately initiate an async shutdown of the socket, potentially causing
the coroutine to yield and allowing the others to resume. Therefore, the
`m_ShuttingDown` flag is still required by the coroutines to be checked
regularly.
  • Loading branch information
yhabteab committed Nov 14, 2024
1 parent d68ee3f commit 5c0f9bf
Show file tree
Hide file tree
Showing 2 changed files with 37 additions and 31 deletions.
65 changes: 35 additions & 30 deletions lib/remote/httpserverconnection.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -68,44 +68,49 @@ void HttpServerConnection::Start()
IoEngine::SpawnCoroutine(m_IoStrand, [this, keepAlive](asio::yield_context yc) { CheckLiveness(yc); });
}

void HttpServerConnection::Disconnect()
/**
* Tries to asynchronously shut down the SSL stream and underlying socket.
*
* It is important to note that this method should only be called from within a coroutine that uses `m_IoStrand`.
*
* @param yc boost::asio::yield_context The coroutine yield context which you are calling this method from.
*/
void HttpServerConnection::Disconnect(boost::asio::yield_context yc)
{
namespace asio = boost::asio;

HttpServerConnection::Ptr keepAlive (this);
if (m_ShuttingDown) {
return;
}

IoEngine::SpawnCoroutine(m_IoStrand, [this, keepAlive](asio::yield_context yc) {
if (!m_ShuttingDown) {
m_ShuttingDown = true;
m_ShuttingDown = true;

Log(LogInformation, "HttpServerConnection")
<< "HTTP client disconnected (from " << m_PeerAddress << ")";

/*
* Do not swallow exceptions in a coroutine.
* https://github.com/Icinga/icinga2/issues/7351
* We must not catch `detail::forced_unwind exception` as
* this is used for unwinding the stack.
*
* Just use the error_code dummy here.
*/
boost::system::error_code ec;
Log(LogInformation, "HttpServerConnection")
<< "HTTP client disconnected (from " << m_PeerAddress << ")";

/*
* Do not swallow exceptions in a coroutine.
* https://github.com/Icinga/icinga2/issues/7351
* We must not catch `detail::forced_unwind exception` as
* this is used for unwinding the stack.
*
* Just use the error_code dummy here.
*/
boost::system::error_code ec;

m_CheckLivenessTimer.cancel();
m_CheckLivenessTimer.cancel();

m_Stream->lowest_layer().cancel(ec);
m_Stream->lowest_layer().cancel(ec);

m_Stream->next_layer().async_shutdown(yc[ec]);
m_Stream->next_layer().async_shutdown(yc[ec]);

m_Stream->lowest_layer().shutdown(m_Stream->lowest_layer().shutdown_both, ec);
m_Stream->lowest_layer().shutdown(m_Stream->lowest_layer().shutdown_both, ec);

auto listener (ApiListener::GetInstance());
auto listener (ApiListener::GetInstance());

if (listener) {
listener->RemoveHttpClient(this);
}
}
});
if (listener) {
listener->RemoveHttpClient(this);
}
}

void HttpServerConnection::StartStreaming()
Expand All @@ -126,7 +131,7 @@ void HttpServerConnection::StartStreaming()
m_Stream->async_read_some(readBuf, yc[ec]);
} while (!ec);

Disconnect();
Disconnect(yc);
}
});
}
Expand Down Expand Up @@ -563,7 +568,7 @@ void HttpServerConnection::ProcessMessages(boost::asio::yield_context yc)
}
}

Disconnect();
Disconnect(yc);
}

void HttpServerConnection::CheckLiveness(boost::asio::yield_context yc)
Expand All @@ -582,7 +587,7 @@ void HttpServerConnection::CheckLiveness(boost::asio::yield_context yc)
Log(LogInformation, "HttpServerConnection")
<< "No messages for HTTP connection have been received in the last 10 seconds.";

Disconnect();
Disconnect(yc);
break;
}
}
Expand Down
3 changes: 2 additions & 1 deletion lib/remote/httpserverconnection.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,6 @@ class HttpServerConnection final : public Object
HttpServerConnection(const String& identity, bool authenticated, const Shared<AsioTlsStream>::Ptr& stream);

void Start();
void Disconnect();
void StartStreaming();

bool Disconnected();
Expand All @@ -45,6 +44,8 @@ class HttpServerConnection final : public Object

HttpServerConnection(const String& identity, bool authenticated, const Shared<AsioTlsStream>::Ptr& stream, boost::asio::io_context& io);

void Disconnect(boost::asio::yield_context yc);

void ProcessMessages(boost::asio::yield_context yc);
void CheckLiveness(boost::asio::yield_context yc);
};
Expand Down

0 comments on commit 5c0f9bf

Please sign in to comment.