/****************************************************************************** * Icinga 2 * * Copyright (C) 2012 Icinga Development Team (http://www.icinga.org/) * * * * 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 "i2-icinga.h" using namespace icinga; bool I2_EXPORT ExternalCommand::m_Initialized; map I2_EXPORT ExternalCommand::m_Commands; void ExternalCommand::Execute(double time, const String& command, const vector& arguments) { if (!m_Initialized) { RegisterCommand("HELLO_WORLD", &ExternalCommand::HelloWorld); RegisterCommand("PROCESS_SERVICE_CHECK_RESULT", &ExternalCommand::ProcessServiceCheckResult); RegisterCommand("SCHEDULE_SVC_CHECK", &ExternalCommand::ScheduleSvcCheck); RegisterCommand("SCHEDULE_FORCED_SVC_CHECK", &ExternalCommand::ScheduleForcedSvcCheck); m_Initialized = true; } map::iterator it; it = m_Commands.find(command); if (it == m_Commands.end()) throw_exception(invalid_argument("The external command '" + command + "' does not exist.")); it->second(time, arguments); } void ExternalCommand::RegisterCommand(const String& command, const ExternalCommand::Callback& callback) { m_Commands[command] = callback; } void ExternalCommand::HelloWorld(double time, const vector& arguments) { Logger::Write(LogInformation, "icinga", "HelloWorld external command called."); } void ExternalCommand::ProcessServiceCheckResult(double time, const vector& arguments) { if (arguments.size() < 4) throw_exception(invalid_argument("Expected 4 arguments.")); if (!Service::Exists(arguments[1])) throw_exception(invalid_argument("The service '" + arguments[1] + "' does not exist.")); Service::Ptr service = Service::GetByName(arguments[1]); int exitStatus = arguments[2].ToDouble(); Dictionary::Ptr result = PluginCheckTask::ParseCheckOutput(arguments[3]); result->Set("state", PluginCheckTask::ExitStatusToState(exitStatus)); double now = Utility::GetTime(); result->Set("schedule_start", time); result->Set("schedule_end", time); result->Set("execution_start", time); result->Set("execution_end", time); Logger::Write(LogInformation, "icinga", "Processing passive check result for service '" + arguments[1] + "'"); service->ProcessCheckResult(result); } void ExternalCommand::ScheduleSvcCheck(double time, const vector& arguments) { if (arguments.size() < 3) throw_exception(invalid_argument("Expected 3 arguments.")); if (!Service::Exists(arguments[1])) throw_exception(invalid_argument("The service '" + arguments[1] + "' does not exist.")); Service::Ptr service = Service::GetByName(arguments[1]); double planned_check = arguments[2].ToDouble(); if (planned_check > service->GetNextCheck()) { Logger::Write(LogInformation, "icinga", "Ignoring reschedule request for service '" + arguments[1] + "' (next check is already sooner than requested check time)"); return; } Logger::Write(LogInformation, "icinga", "Rescheduling next check for service '" + arguments[1] + "'"); service->SetNextCheck(planned_check); } void ExternalCommand::ScheduleForcedSvcCheck(double time, const vector& arguments) { if (arguments.size() < 3) throw_exception(invalid_argument("Expected 3 arguments.")); if (!Service::Exists(arguments[1])) throw_exception(invalid_argument("The service '" + arguments[1] + "' does not exist.")); Service::Ptr service = Service::GetByName(arguments[1]); // TODO: force checks (once we have time periods) Logger::Write(LogInformation, "icinga", "Rescheduling next check for service '" + arguments[1] + "'"); service->SetNextCheck(arguments[2].ToDouble()); }