/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */ #include "remote/eventshandler.hpp" #include "remote/httputility.hpp" #include "remote/filterutility.hpp" #include "config/configcompiler.hpp" #include "config/expression.hpp" #include "base/defer.hpp" #include "base/io-engine.hpp" #include "base/objectlock.hpp" #include "base/json.hpp" #include #include #include #include #include using namespace icinga; REGISTER_URLHANDLER("/v1/events", EventsHandler); const std::map l_EventTypes ({ {"AcknowledgementCleared", EventType::AcknowledgementCleared}, {"AcknowledgementSet", EventType::AcknowledgementSet}, {"CheckResult", EventType::CheckResult}, {"CommentAdded", EventType::CommentAdded}, {"CommentRemoved", EventType::CommentRemoved}, {"DowntimeAdded", EventType::DowntimeAdded}, {"DowntimeRemoved", EventType::DowntimeRemoved}, {"DowntimeStarted", EventType::DowntimeStarted}, {"DowntimeTriggered", EventType::DowntimeTriggered}, {"Flapping", EventType::Flapping}, {"Notification", EventType::Notification}, {"StateChange", EventType::StateChange}, {"ObjectCreated", EventType::ObjectCreated}, {"ObjectDeleted", EventType::ObjectDeleted}, {"ObjectModified", EventType::ObjectModified} }); const String l_ApiQuery (""); bool EventsHandler::HandleRequest( const WaitGroup::Ptr&, HttpRequest& request, HttpResponse& response, boost::asio::yield_context& yc ) { namespace asio = boost::asio; namespace http = boost::beast::http; if (request.Url()->GetPath().size() != 2) return false; if (request.method() != http::verb::post) return false; if (request.version() == 10) { response.SendJsonError(request.Params(), 400, "HTTP/1.0 not supported for event streams."); return true; } Array::Ptr types = request.Params()->Get("types"); if (!types) { response.SendJsonError(request.Params(), 400, "'types' query parameter is required."); return true; } { ObjectLock olock(types); for (String type : types) { FilterUtility::CheckPermission(request.User(), "events/" + type); } } String queueName = request.GetLastParameter("queue"); if (queueName.IsEmpty()) { response.SendJsonError(request.Params(), 400, "'queue' query parameter is required."); return true; } std::set eventTypes; { ObjectLock olock(types); for (String type : types) { auto typeId (l_EventTypes.find(type)); if (typeId != l_EventTypes.end()) { eventTypes.emplace(typeId->second); } } } EventsSubscriber subscriber (std::move(eventTypes), request.GetLastParameter("filter"), l_ApiQuery); response.result(http::status::ok); response.set(http::field::content_type, "application/json"); response.StartStreaming(); IoBoundWorkSlot dontLockTheIoThread (yc); auto writer = std::make_shared>(response); JsonEncoder encoder(writer); for (;;) { auto event(subscriber.GetInbox()->Shift(yc)); if (event && response.IsWritable()) { encoder.Encode(event); response.body() << '\n'; response.Write(yc); } else { return true; } } }