2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2015-06-22 11:11:21 +02:00
|
|
|
|
2020-07-06 14:28:24 +02:00
|
|
|
#include "base/logger.hpp"
|
2015-06-22 11:11:21 +02:00
|
|
|
#include "remote/httphandler.hpp"
|
2015-09-22 17:58:12 +02:00
|
|
|
#include "remote/httputility.hpp"
|
2015-06-22 11:11:21 +02:00
|
|
|
#include "base/singleton.hpp"
|
2016-05-10 15:16:35 +02:00
|
|
|
#include "base/exception.hpp"
|
2015-09-22 17:58:12 +02:00
|
|
|
#include <boost/algorithm/string/join.hpp>
|
2019-02-14 17:27:17 +01:00
|
|
|
#include <boost/beast/http.hpp>
|
2015-06-22 11:11:21 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
|
|
|
Dictionary::Ptr HttpHandler::m_UrlTree;
|
|
|
|
|
|
|
|
void HttpHandler::Register(const Url::Ptr& url, const HttpHandler::Ptr& handler)
|
|
|
|
{
|
|
|
|
if (!m_UrlTree)
|
|
|
|
m_UrlTree = new Dictionary();
|
|
|
|
|
|
|
|
Dictionary::Ptr node = m_UrlTree;
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const String& elem : url->GetPath()) {
|
2015-06-22 11:11:21 +02:00
|
|
|
Dictionary::Ptr children = node->Get("children");
|
|
|
|
|
|
|
|
if (!children) {
|
|
|
|
children = new Dictionary();
|
|
|
|
node->Set("children", children);
|
|
|
|
}
|
|
|
|
|
2015-07-22 12:23:56 +02:00
|
|
|
Dictionary::Ptr sub_node = children->Get(elem);
|
|
|
|
if (!sub_node) {
|
|
|
|
sub_node = new Dictionary();
|
|
|
|
children->Set(elem, sub_node);
|
|
|
|
}
|
2015-06-22 11:11:21 +02:00
|
|
|
|
|
|
|
node = sub_node;
|
|
|
|
}
|
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
Array::Ptr handlers = node->Get("handlers");
|
2015-06-22 11:11:21 +02:00
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
if (!handlers) {
|
|
|
|
handlers = new Array();
|
|
|
|
node->Set("handlers", handlers);
|
|
|
|
}
|
|
|
|
|
|
|
|
handlers->Add(handler);
|
2015-06-22 11:11:21 +02:00
|
|
|
}
|
|
|
|
|
2019-02-14 17:27:17 +01:00
|
|
|
void HttpHandler::ProcessRequest(
|
2019-02-15 11:51:12 +01:00
|
|
|
AsioTlsStream& stream,
|
2019-02-14 17:27:17 +01:00
|
|
|
const ApiUser::Ptr& user,
|
|
|
|
boost::beast::http::request<boost::beast::http::string_body>& request,
|
2019-02-15 11:51:12 +01:00
|
|
|
boost::beast::http::response<boost::beast::http::string_body>& response,
|
|
|
|
boost::asio::yield_context& yc,
|
2019-04-02 17:37:29 +02:00
|
|
|
HttpServerConnection& server
|
2019-02-14 17:27:17 +01:00
|
|
|
)
|
2015-06-22 11:11:21 +02:00
|
|
|
{
|
|
|
|
Dictionary::Ptr node = m_UrlTree;
|
2015-07-28 13:57:59 +02:00
|
|
|
std::vector<HttpHandler::Ptr> handlers;
|
2019-02-14 17:27:17 +01:00
|
|
|
|
2023-01-04 17:34:49 +01:00
|
|
|
Url::Ptr url = new Url(std::string(request.target()));
|
2019-02-14 17:27:17 +01:00
|
|
|
auto& path (url->GetPath());
|
2015-06-22 11:11:21 +02:00
|
|
|
|
2016-08-24 19:59:13 +02:00
|
|
|
for (std::vector<String>::size_type i = 0; i <= path.size(); i++) {
|
2015-07-28 13:57:59 +02:00
|
|
|
Array::Ptr current_handlers = node->Get("handlers");
|
|
|
|
|
|
|
|
if (current_handlers) {
|
|
|
|
ObjectLock olock(current_handlers);
|
2017-11-30 08:19:58 +01:00
|
|
|
for (const HttpHandler::Ptr& current_handler : current_handlers) {
|
2015-07-28 13:57:59 +02:00
|
|
|
handlers.push_back(current_handler);
|
|
|
|
}
|
|
|
|
}
|
2015-06-22 11:11:21 +02:00
|
|
|
|
|
|
|
Dictionary::Ptr children = node->Get("children");
|
|
|
|
|
|
|
|
if (!children) {
|
|
|
|
node.reset();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-08-11 09:31:28 +02:00
|
|
|
if (i == path.size())
|
|
|
|
break;
|
|
|
|
|
|
|
|
node = children->Get(path[i]);
|
2015-06-22 11:11:21 +02:00
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
if (!node)
|
2015-06-22 11:11:21 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
std::reverse(handlers.begin(), handlers.end());
|
2015-06-22 11:11:21 +02:00
|
|
|
|
2016-05-10 15:16:35 +02:00
|
|
|
Dictionary::Ptr params;
|
|
|
|
|
|
|
|
try {
|
2019-02-14 17:27:17 +01:00
|
|
|
params = HttpUtility::FetchRequestParameters(url, request.body());
|
2016-05-10 15:16:35 +02:00
|
|
|
} catch (const std::exception& ex) {
|
2017-12-20 15:31:05 +01:00
|
|
|
HttpUtility::SendJsonError(response, params, 400, "Invalid request body: " + DiagnosticInformation(ex, false));
|
2016-05-10 15:16:35 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
bool processed = false;
|
2019-09-23 09:48:50 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* HandleRequest may throw a permission exception.
|
|
|
|
* DO NOT return a specific permission error. This
|
|
|
|
* allows attackers to guess from words which objects
|
|
|
|
* do exist.
|
|
|
|
*/
|
|
|
|
try {
|
|
|
|
for (const HttpHandler::Ptr& handler : handlers) {
|
|
|
|
if (handler->HandleRequest(stream, user, request, url, response, params, yc, server)) {
|
|
|
|
processed = true;
|
|
|
|
break;
|
|
|
|
}
|
2015-07-28 13:57:59 +02:00
|
|
|
}
|
2020-07-06 14:28:24 +02:00
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogWarning, "HttpServerConnection")
|
|
|
|
<< "Error while processing HTTP request: " << ex.what();
|
|
|
|
|
2019-09-23 09:48:50 +02:00
|
|
|
processed = false;
|
2015-07-28 13:57:59 +02:00
|
|
|
}
|
2016-05-10 15:16:35 +02:00
|
|
|
|
2015-07-28 13:57:59 +02:00
|
|
|
if (!processed) {
|
2019-02-14 17:27:17 +01:00
|
|
|
HttpUtility::SendJsonError(response, params, 404, "The requested path '" + boost::algorithm::join(path, "/") +
|
|
|
|
"' could not be found or the request method is not valid for this path.");
|
2015-06-22 11:11:21 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2015-09-22 17:58:12 +02:00
|
|
|
|