/****************************************************************************** * 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-cib.h" using namespace icinga; Service::Service(const ConfigObject::Ptr& configObject) : ConfigObjectAdapter(configObject) { assert(GetType() == "service"); } string Service::GetAlias(void) const { string value; if (GetProperty("alias", &value)) return value; return GetName(); } bool Service::Exists(const string& name) { return (ConfigObject::GetObject("service", name)); } Service Service::GetByName(const string& name) { ConfigObject::Ptr configObject = ConfigObject::GetObject("service", name); if (!configObject) throw invalid_argument("Service '" + name + "' does not exist."); return configObject; } Host Service::GetHost(void) const { string hostname; if (!GetProperty("host_name", &hostname)) throw runtime_error("Service object is missing the 'host_name' property."); return Host::GetByName(hostname); } Dictionary::Ptr Service::GetMacros(void) const { Dictionary::Ptr macros; GetProperty("macros", ¯os); return macros; } string Service::GetCheckType(void) const { string value = "nagios"; GetProperty("check_type", &value); return value; } string Service::GetCheckCommand(void) const { string value; GetProperty("check_command", &value); return value; } long Service::GetMaxCheckAttempts(void) const { long value = 3; GetProperty("max_check_attempts", &value); return value; } long Service::GetCheckInterval(void) const { long value = 300; GetProperty("check_interval", &value); if (value < 15) value = 15; return value; } long Service::GetRetryInterval(void) const { long value; if (!GetProperty("retry_interval", &value)) value = GetCheckInterval() / 5; return value; } Dictionary::Ptr Service::GetDependencies(void) const { Dictionary::Ptr value; GetProperty("dependencies", &value); return value; } void Service::GetDependenciesRecursive(const Dictionary::Ptr& result) const { assert(result); Dictionary::Ptr dependencies = GetDependencies(); if (!dependencies) return; Dictionary::Iterator it; for (it = dependencies->Begin(); it != dependencies->End(); it++) { if (result->Contains(it->second)) continue; result->Set(it->second, it->second); Service service = Service::GetByName(it->second); service.GetDependenciesRecursive(result); } } Dictionary::Ptr Service::GetGroups(void) const { Dictionary::Ptr value; GetProperty("servicegroups", &value); return value; } Dictionary::Ptr Service::GetCheckers(void) const { Dictionary::Ptr value; GetProperty("checkers", &value); return value; } bool Service::IsReachable(void) const { Dictionary::Ptr dependencies = boost::make_shared(); GetDependenciesRecursive(dependencies); Dictionary::Iterator it; for (it = dependencies->Begin(); it != dependencies->End(); it++) { Service service = Service::GetByName(it->second); /* ignore ourselves */ if (service.GetName() == GetName()) continue; /* ignore pending services */ if (!service.HasLastCheckResult()) continue; /* ignore soft states */ if (service.GetStateType() == StateTypeSoft) continue; /* ignore services states OK and Warning */ if (service.GetState() == StateOK || service.GetState() == StateWarning) continue; return false; } return true; } void Service::SetNextCheck(time_t nextCheck) { SetTag("next_check", (long)nextCheck); } time_t Service::GetNextCheck(void) { long value; if (!GetTag("next_check", &value)) { value = time(NULL) + rand() % GetCheckInterval(); SetNextCheck(value); } return value; } void Service::UpdateNextCheck(void) { time_t now; time(&now); if (GetStateType() == StateTypeSoft) SetNextCheck(now + GetRetryInterval()); else SetNextCheck(now + GetCheckInterval()); } void Service::SetChecker(const string& checker) { SetTag("checker", checker); } string Service::GetChecker(void) const { string value; GetTag("checker", &value); return value; } void Service::SetCurrentCheckAttempt(long attempt) { SetTag("check_attempt", attempt); } long Service::GetCurrentCheckAttempt(void) const { long value = 1; GetTag("check_attempt", &value); return value; } void Service::SetState(ServiceState state) { SetTag("state", static_cast(state)); } ServiceState Service::GetState(void) const { long value = StateUnknown; GetTag("state", &value); return static_cast(value); } void Service::SetStateType(ServiceStateType type) { SetTag("state_type", static_cast(type)); } ServiceStateType Service::GetStateType(void) const { long value = StateTypeHard; GetTag("state_type", &value); return static_cast(value); } void Service::SetLastCheckResult(const CheckResult& result) { SetTag("last_result", result.GetDictionary()); } bool Service::HasLastCheckResult(void) const { Dictionary::Ptr value; return GetTag("last_result", &value) && value; } CheckResult Service::GetLastCheckResult(void) const { Dictionary::Ptr value; if (!GetTag("last_result", &value)) throw invalid_argument("Service has no last check result."); return CheckResult(value); } void Service::SetLastStateChange(time_t ts) { SetTag("last_state_change", static_cast(ts)); } time_t Service::GetLastStateChange(void) const { long value; if (!GetTag("last_state_change", &value)) value = IcingaApplication::GetInstance()->GetStartTime(); return value; } void Service::SetLastHardStateChange(time_t ts) { SetTag("last_hard_state_change", static_cast(ts)); } time_t Service::GetLastHardStateChange(void) const { long value; if (!GetTag("last_hard_state_change", &value)) value = IcingaApplication::GetInstance()->GetStartTime(); return value; } void Service::ApplyCheckResult(const CheckResult& cr) { long attempt = GetCurrentCheckAttempt(); if (cr.GetState() == StateOK) { if (GetState() == StateOK) SetStateType(StateTypeHard); attempt = 1; } else { if (attempt >= GetMaxCheckAttempts()) { SetStateType(StateTypeHard); attempt = 1; } else if (GetStateType() == StateTypeSoft || GetState() == StateOK) { SetStateType(StateTypeSoft); attempt++; } } SetCurrentCheckAttempt(attempt); SetState(cr.GetState()); } ServiceState Service::StateFromString(const string& state) { /* TODO: make this thread-safe */ static map stateLookup; if (stateLookup.empty()) { stateLookup["ok"] = StateOK; stateLookup["warning"] = StateWarning; stateLookup["critical"] = StateCritical; stateLookup["uncheckable"] = StateUncheckable; stateLookup["unknown"] = StateUnknown; } map::iterator it; it = stateLookup.find(state); if (it == stateLookup.end()) return StateUnknown; else return it->second; } string Service::StateToString(ServiceState state) { switch (state) { case StateOK: return "ok"; case StateWarning: return "warning"; case StateCritical: return "critical"; case StateUncheckable: return "uncheckable"; case StateUnknown: default: return "unknown"; } } ServiceStateType Service::StateTypeFromString(const string& type) { if (type == "soft") return StateTypeSoft; else return StateTypeHard; } string Service::StateTypeToString(ServiceStateType type) { if (type == StateTypeSoft) return "soft"; else return "hard"; } bool Service::IsAllowedChecker(const string& checker) const { Dictionary::Ptr checkers = GetCheckers(); if (!checkers) return true; Dictionary::Iterator it; for (it = checkers->Begin(); it != checkers->End(); it++) { string pattern = it->second; if (Utility::Match(pattern, checker)) return true; } return false; } Dictionary::Ptr Service::ResolveDependencies(Host host, const Dictionary::Ptr& dependencies) { Dictionary::Ptr services; host.GetProperty("services", &services); Dictionary::Ptr result = boost::make_shared(); Dictionary::Iterator it; for (it = dependencies->Begin(); it != dependencies->End(); it++) { string name; if (services && services->Contains(it->first)) name = host.GetName() + "-" + it->first; else name = it->first; result->Set(name, name); } return result; }