mirror of
https://github.com/Icinga/icinga2.git
synced 2025-08-14 06:08:15 +02:00
The wait group gets passed to HttpServerConnection, then down to the HttpHandlers. For those handlers that modify the program state, the wait group is locked so ApiListener will wait on Stop() for the request to complete. If the request iterates over config objects, a further check on the state of the wait group is added to abort early and not delay program shutdown. In that case, 503 responses will be sent to the client. Additionally, in HttpServerConnection, no further requests than the one already started will be allowed once the wait group is joining.
96 lines
2.0 KiB
C++
96 lines
2.0 KiB
C++
/* Icinga 2 | (c) 2024 Icinga GmbH | GPLv2+ */
|
|
|
|
#include "base/defer.hpp"
|
|
#include "remote/filterutility.hpp"
|
|
#include "remote/httputility.hpp"
|
|
#include "remote/mallocinfohandler.hpp"
|
|
#include <cstddef>
|
|
#include <cstdio>
|
|
#include <string>
|
|
|
|
#ifdef HAVE_MALLOC_INFO
|
|
# include <errno.h>
|
|
# include <malloc.h>
|
|
#endif /* HAVE_MALLOC_INFO */
|
|
|
|
using namespace icinga;
|
|
|
|
REGISTER_URLHANDLER("/v1/debug/malloc_info", MallocInfoHandler);
|
|
|
|
bool MallocInfoHandler::HandleRequest(
|
|
const WaitGroup::Ptr&,
|
|
AsioTlsStream&,
|
|
const ApiUser::Ptr& user,
|
|
boost::beast::http::request<boost::beast::http::string_body>& request,
|
|
const Url::Ptr& url,
|
|
boost::beast::http::response<boost::beast::http::string_body>& response,
|
|
const Dictionary::Ptr& params,
|
|
boost::asio::yield_context&,
|
|
HttpServerConnection&
|
|
)
|
|
{
|
|
namespace http = boost::beast::http;
|
|
|
|
if (url->GetPath().size() != 3) {
|
|
return false;
|
|
}
|
|
|
|
if (request.method() != http::verb::get) {
|
|
return false;
|
|
}
|
|
|
|
FilterUtility::CheckPermission(user, "debug");
|
|
|
|
#ifndef HAVE_MALLOC_INFO
|
|
HttpUtility::SendJsonError(response, params, 501, "malloc_info(3) not available.");
|
|
#else /* HAVE_MALLOC_INFO */
|
|
char* buf = nullptr;
|
|
size_t bufSize = 0;
|
|
FILE* f = nullptr;
|
|
|
|
Defer release ([&f, &buf]() {
|
|
if (f) {
|
|
(void)fclose(f);
|
|
}
|
|
|
|
free(buf);
|
|
});
|
|
|
|
f = open_memstream(&buf, &bufSize);
|
|
|
|
if (!f) {
|
|
auto error (errno);
|
|
|
|
BOOST_THROW_EXCEPTION(posix_error()
|
|
<< boost::errinfo_api_function("open_memstream")
|
|
<< boost::errinfo_errno(error));
|
|
}
|
|
|
|
if (malloc_info(0, f)) {
|
|
auto error (errno);
|
|
|
|
BOOST_THROW_EXCEPTION(posix_error()
|
|
<< boost::errinfo_api_function("malloc_info")
|
|
<< boost::errinfo_errno(error));
|
|
}
|
|
|
|
auto closeErr (fclose(f));
|
|
f = nullptr;
|
|
|
|
if (closeErr) {
|
|
auto error (errno);
|
|
|
|
BOOST_THROW_EXCEPTION(posix_error()
|
|
<< boost::errinfo_api_function("fclose")
|
|
<< boost::errinfo_errno(error));
|
|
}
|
|
|
|
response.result(200);
|
|
response.set(http::field::content_type, "application/xml");
|
|
response.body() = std::string(buf, bufSize);
|
|
response.content_length(response.body().size());
|
|
#endif /* HAVE_MALLOC_INFO */
|
|
|
|
return true;
|
|
}
|