2015-07-30 17:50:17 +02:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2018-01-02 12:06:00 +01:00
|
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) *
|
2015-07-30 17:50:17 +02:00
|
|
|
* *
|
|
|
|
* This program is free software; you can redistribute it and/or *
|
|
|
|
* modify it under the terms of the GNU General Public License *
|
|
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
|
|
* of the License, or (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
* This program is distributed in the hope that it will be useful, *
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
|
|
* GNU General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU General Public License *
|
|
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
#include "remote/actionshandler.hpp"
|
|
|
|
#include "remote/httputility.hpp"
|
|
|
|
#include "remote/filterutility.hpp"
|
|
|
|
#include "remote/apiaction.hpp"
|
|
|
|
#include "base/exception.hpp"
|
2015-08-21 15:50:40 +02:00
|
|
|
#include "base/logger.hpp"
|
2015-07-30 17:50:17 +02:00
|
|
|
#include <set>
|
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
|
|
|
REGISTER_URLHANDLER("/v1/actions", ActionsHandler);
|
|
|
|
|
2016-05-10 15:16:35 +02:00
|
|
|
bool ActionsHandler::HandleRequest(const ApiUser::Ptr& user, HttpRequest& request, HttpResponse& response, const Dictionary::Ptr& params)
|
2015-07-30 17:50:17 +02:00
|
|
|
{
|
2015-09-28 16:08:14 +02:00
|
|
|
if (request.RequestUrl->GetPath().size() != 3)
|
|
|
|
return false;
|
2015-09-22 17:58:12 +02:00
|
|
|
|
2015-09-28 16:08:14 +02:00
|
|
|
if (request.RequestMethod != "POST")
|
|
|
|
return false;
|
2015-07-30 17:50:17 +02:00
|
|
|
|
|
|
|
String actionName = request.RequestUrl->GetPath()[2];
|
|
|
|
|
|
|
|
ApiAction::Ptr action = ApiAction::GetByName(actionName);
|
|
|
|
|
2015-09-22 17:58:12 +02:00
|
|
|
if (!action) {
|
2017-12-20 15:31:05 +01:00
|
|
|
HttpUtility::SendJsonError(response, params, 404, "Action '" + actionName + "' does not exist.");
|
2015-09-22 17:58:12 +02:00
|
|
|
return true;
|
|
|
|
}
|
2015-07-30 17:50:17 +02:00
|
|
|
|
|
|
|
QueryDescription qd;
|
|
|
|
|
2015-08-24 08:01:33 +02:00
|
|
|
const std::vector<String>& types = action->GetTypes();
|
2015-08-26 10:58:59 +02:00
|
|
|
std::vector<Value> objs;
|
2015-07-30 17:50:17 +02:00
|
|
|
|
2015-09-28 08:57:25 +02:00
|
|
|
String permission = "actions/" + actionName;
|
|
|
|
|
2015-08-24 08:01:33 +02:00
|
|
|
if (!types.empty()) {
|
2015-08-26 10:58:59 +02:00
|
|
|
qd.Types = std::set<String>(types.begin(), types.end());
|
2015-09-28 08:57:25 +02:00
|
|
|
qd.Permission = permission;
|
2015-08-24 08:01:33 +02:00
|
|
|
|
2015-09-22 17:58:12 +02:00
|
|
|
try {
|
2015-09-28 08:57:25 +02:00
|
|
|
objs = FilterUtility::GetFilterTargets(qd, params, user);
|
2015-09-22 17:58:12 +02:00
|
|
|
} catch (const std::exception& ex) {
|
2017-12-20 15:31:05 +01:00
|
|
|
HttpUtility::SendJsonError(response, params, 404,
|
2017-12-19 15:50:05 +01:00
|
|
|
"No objects found.",
|
|
|
|
HttpUtility::GetLastParameter(params, "verboseErrors") ? DiagnosticInformation(ex) : "");
|
2015-09-22 17:58:12 +02:00
|
|
|
return true;
|
|
|
|
}
|
2015-09-28 08:57:25 +02:00
|
|
|
} else {
|
|
|
|
FilterUtility::CheckPermission(user, permission);
|
2018-01-04 09:14:55 +01:00
|
|
|
objs.emplace_back(nullptr);
|
2015-09-28 08:57:25 +02:00
|
|
|
}
|
2015-07-30 17:50:17 +02:00
|
|
|
|
2018-01-11 11:17:38 +01:00
|
|
|
ArrayData results;
|
2015-07-30 17:50:17 +02:00
|
|
|
|
2015-08-21 15:50:40 +02:00
|
|
|
Log(LogNotice, "ApiActionHandler")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Running action " << actionName;
|
2015-08-21 15:50:40 +02:00
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const ConfigObject::Ptr& obj : objs) {
|
2015-07-30 17:50:17 +02:00
|
|
|
try {
|
2018-01-11 11:17:38 +01:00
|
|
|
results.emplace_back(action->Invoke(obj, params));
|
2015-07-30 17:50:17 +02:00
|
|
|
} catch (const std::exception& ex) {
|
2018-01-11 11:17:38 +01:00
|
|
|
Dictionary::Ptr fail = new Dictionary({
|
|
|
|
{ "code", 500 },
|
|
|
|
{ "status", "Action execution failed: '" + DiagnosticInformation(ex, false) + "'." }
|
|
|
|
});
|
|
|
|
|
2015-10-27 15:26:19 +01:00
|
|
|
if (HttpUtility::GetLastParameter(params, "verboseErrors"))
|
2015-09-22 17:58:12 +02:00
|
|
|
fail->Set("diagnostic information", DiagnosticInformation(ex));
|
2018-01-11 11:17:38 +01:00
|
|
|
|
|
|
|
results.emplace_back(std::move(fail));
|
2015-07-30 17:50:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-06 10:43:33 +02:00
|
|
|
int statusCode = 500;
|
2018-04-06 10:03:09 +02:00
|
|
|
String statusMessage = "No action executed successfully";
|
2018-04-05 10:47:17 +02:00
|
|
|
|
|
|
|
for (const Dictionary::Ptr& res : results) {
|
2018-04-06 10:03:09 +02:00
|
|
|
if (res->Contains("code") && res->Get("code") == 200) {
|
|
|
|
statusCode = 200;
|
|
|
|
statusMessage = "OK";
|
2018-04-05 10:47:17 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-06 10:03:09 +02:00
|
|
|
response.SetStatus(statusCode, statusMessage);
|
|
|
|
|
2018-01-11 11:17:38 +01:00
|
|
|
Dictionary::Ptr result = new Dictionary({
|
|
|
|
{ "results", new Array(std::move(results)) }
|
|
|
|
});
|
2015-07-30 17:50:17 +02:00
|
|
|
|
2017-12-20 15:31:05 +01:00
|
|
|
HttpUtility::SendJsonBody(response, params, result);
|
2015-07-30 17:50:17 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|