/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) * * * * 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 "icinga/host.hpp" #include "icinga/host-ti.cpp" #include "icinga/service.hpp" #include "icinga/hostgroup.hpp" #include "icinga/pluginutility.hpp" #include "icinga/scheduleddowntime.hpp" #include "base/objectlock.hpp" #include "base/convert.hpp" #include "base/utility.hpp" #include "base/debug.hpp" #include "base/json.hpp" using namespace icinga; REGISTER_TYPE(Host); void Host::OnAllConfigLoaded() { ObjectImpl::OnAllConfigLoaded(); String zoneName = GetZoneName(); if (!zoneName.IsEmpty()) { Zone::Ptr zone = Zone::GetByName(zoneName); if (zone && zone->IsGlobal()) BOOST_THROW_EXCEPTION(std::invalid_argument("Host '" + GetName() + "' cannot be put into global zone '" + zone->GetName() + "'.")); } HostGroup::EvaluateObjectRules(this); Array::Ptr groups = GetGroups(); if (groups) { groups = groups->ShallowClone(); ObjectLock olock(groups); for (const String& name : groups) { HostGroup::Ptr hg = HostGroup::GetByName(name); if (hg) hg->ResolveGroupMembership(this, true); } } } void Host::CreateChildObjects(const Type::Ptr& childType) { if (childType == ScheduledDowntime::TypeInstance) ScheduledDowntime::EvaluateApplyRules(this); if (childType == Notification::TypeInstance) Notification::EvaluateApplyRules(this); if (childType == Dependency::TypeInstance) Dependency::EvaluateApplyRules(this); if (childType == Service::TypeInstance) Service::EvaluateApplyRules(this); } void Host::Stop(bool runtimeRemoved) { ObjectImpl::Stop(runtimeRemoved); Array::Ptr groups = GetGroups(); if (groups) { ObjectLock olock(groups); for (const String& name : groups) { HostGroup::Ptr hg = HostGroup::GetByName(name); if (hg) hg->ResolveGroupMembership(this, false); } } // TODO: unregister slave services/notifications? } std::vector Host::GetServices() const { boost::mutex::scoped_lock lock(m_ServicesMutex); std::vector services; services.reserve(m_Services.size()); typedef std::pair ServicePair; for (const ServicePair& kv : m_Services) { services.push_back(kv.second); } return services; } void Host::AddService(const Service::Ptr& service) { boost::mutex::scoped_lock lock(m_ServicesMutex); m_Services[service->GetShortName()] = service; } void Host::RemoveService(const Service::Ptr& service) { boost::mutex::scoped_lock lock(m_ServicesMutex); m_Services.erase(service->GetShortName()); } int Host::GetTotalServices() const { return GetServices().size(); } Service::Ptr Host::GetServiceByShortName(const Value& name) { if (name.IsScalar()) { { boost::mutex::scoped_lock lock(m_ServicesMutex); auto it = m_Services.find(name); if (it != m_Services.end()) return it->second; } return nullptr; } else if (name.IsObjectType()) { Dictionary::Ptr dict = name; String short_name; return Service::GetByNamePair(dict->Get("host"), dict->Get("service")); } else { BOOST_THROW_EXCEPTION(std::invalid_argument("Host/Service name pair is invalid: " + JsonEncode(name))); } } HostState Host::CalculateState(ServiceState state) { switch (state) { case ServiceOK: case ServiceWarning: return HostUp; default: return HostDown; } } HostState Host::GetState() const { return CalculateState(GetStateRaw()); } HostState Host::GetLastState() const { return CalculateState(GetLastStateRaw()); } HostState Host::GetLastHardState() const { return CalculateState(GetLastHardStateRaw()); } /* keep in sync with Service::GetSeverity() */ int Host::GetSeverity() const { int severity = 0; ObjectLock olock(this); ServiceState state = GetStateRaw(); /* OK/Warning = Up, Critical/Unknownb = Down */ if (!HasBeenChecked()) severity |= SeverityFlagPending; else if (state == ServiceUnknown) severity |= SeverityFlagCritical; else if (state == ServiceCritical) severity |= SeverityFlagCritical; if (IsInDowntime()) severity |= SeverityFlagDowntime; else if (IsAcknowledged()) severity |= SeverityFlagAcknowledgement; else severity |= SeverityFlagUnhandled; olock.Unlock(); return severity; } bool Host::IsStateOK(ServiceState state) { return Host::CalculateState(state) == HostUp; } void Host::SaveLastState(ServiceState state, double timestamp) { if (state == ServiceOK || state == ServiceWarning) SetLastStateUp(timestamp); else if (state == ServiceCritical) SetLastStateDown(timestamp); } HostState Host::StateFromString(const String& state) { if (state == "UP") return HostUp; else return HostDown; } String Host::StateToString(HostState state) { switch (state) { case HostUp: return "UP"; case HostDown: return "DOWN"; default: return "INVALID"; } } StateType Host::StateTypeFromString(const String& type) { if (type == "SOFT") return StateTypeSoft; else return StateTypeHard; } String Host::StateTypeToString(StateType type) { if (type == StateTypeSoft) return "SOFT"; else return "HARD"; } bool Host::ResolveMacro(const String& macro, const CheckResult::Ptr&, Value *result) const { if (macro == "state") { *result = StateToString(GetState()); return true; } else if (macro == "state_id") { *result = GetState(); return true; } else if (macro == "state_type") { *result = StateTypeToString(GetStateType()); return true; } else if (macro == "last_state") { *result = StateToString(GetLastState()); return true; } else if (macro == "last_state_id") { *result = GetLastState(); return true; } else if (macro == "last_state_type") { *result = StateTypeToString(GetLastStateType()); return true; } else if (macro == "last_state_change") { *result = static_cast(GetLastStateChange()); return true; } else if (macro == "downtime_depth") { *result = GetDowntimeDepth(); return true; } else if (macro == "duration_sec") { *result = Utility::GetTime() - GetLastStateChange(); return true; } else if (macro == "num_services" || macro == "num_services_ok" || macro == "num_services_warning" || macro == "num_services_unknown" || macro == "num_services_critical") { int filter = -1; int count = 0; if (macro == "num_services_ok") filter = ServiceOK; else if (macro == "num_services_warning") filter = ServiceWarning; else if (macro == "num_services_unknown") filter = ServiceUnknown; else if (macro == "num_services_critical") filter = ServiceCritical; for (const Service::Ptr& service : GetServices()) { if (filter != -1 && service->GetState() != filter) continue; count++; } *result = count; return true; } CheckResult::Ptr cr = GetLastCheckResult(); if (cr) { if (macro == "latency") { *result = cr->CalculateLatency(); return true; } else if (macro == "execution_time") { *result = cr->CalculateExecutionTime(); return true; } else if (macro == "output") { *result = cr->GetOutput(); return true; } else if (macro == "perfdata") { *result = PluginUtility::FormatPerfdata(cr->GetPerformanceData()); return true; } else if (macro == "check_source") { *result = cr->GetCheckSource(); return true; } } return false; }