/****************************************************************************** * 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_exception(invalid_argument("Service '" + name + "' does not exist.")); return configObject; } Host Service::GetHost(void) const { string hostname; if (!GetProperty("host_name", &hostname)) throw_exception(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::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; Variant dependency; BOOST_FOREACH(tie(tuples::ignore, dependency), dependencies) { if (result->Contains(dependency)) continue; result->Set(dependency, dependency); Service service = Service::GetByName(dependency); 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); Variant dependency; BOOST_FOREACH(tie(tuples::ignore, dependency), dependencies) { Service service = Service::GetByName(dependency); /* 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, previous, next, interval; time(&now); previous = GetNextCheck(); if (GetStateType() == StateTypeSoft) interval = GetRetryInterval(); else interval = GetCheckInterval(); next = now + ((now - previous) / interval + 1) * interval; SetNextCheck(next); } 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_exception(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; Variant pattern; BOOST_FOREACH(tie(tuples::ignore, pattern), checkers) { 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(); Variant dependency; BOOST_FOREACH(tie(tuples::ignore, dependency), dependencies) { string name; if (services && services->Contains(dependency)) name = host.GetName() + "-" + static_cast(dependency); else name = static_cast(dependency); result->Set(name, name); } return result; }