void ServiceDbObject::OnConfigUpdate(void) { Service::Ptr service = static_pointer_cast<Service>(GetObject()); /* service dependencies */ Log(LogDebug, "db_ido", "service dependencies for '" + service->GetName() + "'"); BOOST_FOREACH(const Dependency::Ptr& dep, service->GetDependencies()) { Checkable::Ptr parent = dep->GetParent(); if (!parent) continue; Log(LogDebug, "db_ido", "service parents: " + parent->GetName()); int state_filter = dep->GetStateFilter(); /* service dependencies */ Dictionary::Ptr fields1 = make_shared<Dictionary>(); fields1->Set("service_object_id", parent); fields1->Set("dependent_service_object_id", service); fields1->Set("inherits_parent", 1); fields1->Set("timeperiod_object_id", dep->GetPeriod()); fields1->Set("fail_on_ok", (state_filter & StateFilterOK) ? 1 : 0); fields1->Set("fail_on_warning", (state_filter & StateFilterWarning) ? 1 : 0); fields1->Set("fail_on_critical", (state_filter & StateFilterCritical) ? 1 : 0); fields1->Set("fail_on_unknown", (state_filter & StateFilterUnknown) ? 1 : 0); fields1->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query1; query1.Table = GetType()->GetTable() + "dependencies"; query1.Type = DbQueryInsert; query1.Category = DbCatConfig; query1.Fields = fields1; OnQuery(query1); } /* service contacts, contactgroups */ Log(LogDebug, "db_ido", "service contacts: " + service->GetName()); BOOST_FOREACH(const User::Ptr& user, CompatUtility::GetCheckableNotificationUsers(service)) { Log(LogDebug, "db_ido", "service contacts: " + user->GetName()); Dictionary::Ptr fields_contact = make_shared<Dictionary>(); fields_contact->Set("service_id", DbValue::FromObjectInsertID(service)); fields_contact->Set("contact_object_id", user); fields_contact->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query_contact; query_contact.Table = GetType()->GetTable() + "_contacts"; query_contact.Type = DbQueryInsert; query_contact.Category = DbCatConfig; query_contact.Fields = fields_contact; OnQuery(query_contact); }
bool ServiceGroup::ResolveGroupMembership(const Service::Ptr& service, bool add, int rstack) { if (add && rstack > 20) { Log(LogWarning, "ServiceGroup") << "Too many nested groups for group '" << GetName() << "': Service '" << service->GetName() << "' membership assignment failed."; return false; } Array::Ptr groups = GetGroups(); if (groups && groups->GetLength() > 0) { ObjectLock olock(groups); for (const String& name : groups) { ServiceGroup::Ptr group = ServiceGroup::GetByName(name); if (group && !group->ResolveGroupMembership(service, add, rstack + 1)) return false; } } if (add) AddMember(service); else RemoveMember(service); return true; }
bool ServiceGroup::EvaluateObjectRule(const Service::Ptr& service, const ConfigItem::Ptr& group) { String group_name = group->GetName(); CONTEXT("Evaluating rule for group '" + group_name + "'"); Host::Ptr host = service->GetHost(); ScriptFrame frame; if (group->GetScope()) group->GetScope()->CopyTo(frame.Locals); frame.Locals->Set("host", host); frame.Locals->Set("service", service); if (!group->GetFilter()->Evaluate(frame).GetValue().ToBool()) return false; Log(LogDebug, "ServiceGroup") << "Assigning membership for group '" << group_name << "' to service '" << service->GetName() << "'"; Array::Ptr groups = service->GetGroups(); groups->Add(group_name); return true; }
void ScheduledDowntime::EvaluateApplyRules(const Service::Ptr& service) { CONTEXT("Evaluating 'apply' rules for service '" + service->GetName() + "'"); for (ApplyRule& rule : ApplyRule::GetRules("ScheduledDowntime")) { if (rule.GetTargetType() != "Service") continue; if (EvaluateApplyRule(service, rule)) rule.AddMatch(); } }
void ServiceGroup::EvaluateObjectRules(const Service::Ptr& service) { CONTEXT("Evaluating group membership for service '" + service->GetName() + "'"); for (const ConfigItem::Ptr& group : ConfigItem::GetItems("ServiceGroup")) { if (!group->GetFilter()) continue; EvaluateObjectRule(service, group); } }
bool ServiceGroup::EvaluateObjectRule(const Service::Ptr service, const ObjectRule& rule) { DebugInfo di = rule.GetDebugInfo(); std::ostringstream msgbuf; msgbuf << "Evaluating 'object' rule (" << di << ")"; CONTEXT(msgbuf.str()); Host::Ptr host = service->GetHost(); Dictionary::Ptr locals = make_shared<Dictionary>(); locals->Set("host", host); locals->Set("service", service); if (!rule.EvaluateFilter(locals)) return false; std::ostringstream msgbuf2; msgbuf2 << "Assigning membership for group '" << rule.GetName() << "' to service '" << service->GetName() << "' for rule " << di; Log(LogDebug, "icinga", msgbuf2.str()); String group_name = rule.GetName(); ServiceGroup::Ptr group = ServiceGroup::GetByName(group_name); if (!group) { Log(LogCritical, "icinga", "Invalid membership assignment. Group '" + group_name + "' does not exist."); return false; } /* assign service group membership */ group->ResolveGroupMembership(service, true); return true; }
void ServiceDbObject::OnConfigUpdate(void) { Service::Ptr service = static_pointer_cast<Service>(GetObject()); /* service dependencies */ Log(LogDebug, "ido", "service dependencies for '" + service->GetName() + "'"); DbQuery query_del1; query_del1.Table = GetType()->GetTable() + "dependencies"; query_del1.Type = DbQueryDelete; query_del1.WhereCriteria = boost::make_shared<Dictionary>(); query_del1.WhereCriteria->Set("dependent_service_object_id", service); OnQuery(query_del1); BOOST_FOREACH(const Service::Ptr& parent, service->GetParentServices()) { Log(LogDebug, "ido", "service parents: " + parent->GetName()); /* service dependencies */ Dictionary::Ptr fields1 = boost::make_shared<Dictionary>(); fields1->Set("service_object_id", parent); fields1->Set("dependent_service_object_id", service); fields1->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query1; query1.Table = GetType()->GetTable() + "dependencies"; query1.Type = DbQueryInsert; query1.Fields = fields1; OnQuery(query1); } /* custom variables */ Log(LogDebug, "ido", "service customvars for '" + service->GetName() + "'"); DbQuery query_del2; query_del2.Table = "customvariables"; query_del2.Type = DbQueryDelete; query_del2.WhereCriteria = boost::make_shared<Dictionary>(); query_del2.WhereCriteria->Set("object_id", service); OnQuery(query_del2); Dictionary::Ptr customvars; { ObjectLock olock(service); customvars = CompatUtility::GetCustomVariableConfig(service); } if (customvars) { ObjectLock olock (customvars); String key; Value value; BOOST_FOREACH(boost::tie(key, value), customvars) { Log(LogDebug, "ido", "service customvar key: '" + key + "' value: '" + Convert::ToString(value) + "'"); Dictionary::Ptr fields2 = boost::make_shared<Dictionary>(); fields2->Set("varname", Convert::ToString(key)); fields2->Set("varvalue", Convert::ToString(value)); fields2->Set("config_type", 1); fields2->Set("has_been_modified", 0); fields2->Set("object_id", service); fields2->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query2; query2.Table = "customvariables"; query2.Type = DbQueryInsert; query2.Fields = fields2; OnQuery(query2); } }
void CheckerComponent::CheckThreadProc() { Utility::SetThreadName("Check Scheduler"); IcingaApplication::Ptr icingaApp = IcingaApplication::GetInstance(); boost::mutex::scoped_lock lock(m_Mutex); for (;;) { typedef boost::multi_index::nth_index<CheckableSet, 1>::type CheckTimeView; CheckTimeView& idx = boost::get<1>(m_IdleCheckables); while (idx.begin() == idx.end() && !m_Stopped) m_CV.wait(lock); if (m_Stopped) break; auto it = idx.begin(); CheckableScheduleInfo csi = *it; double wait = csi.NextCheck - Utility::GetTime(); //#ifdef I2_DEBUG // Log(LogDebug, "CheckerComponent") // << "Pending checks " << Checkable::GetPendingChecks() // << " vs. max concurrent checks " << icingaApp->GetMaxConcurrentChecks() << "."; //#endif /* I2_DEBUG */ if (Checkable::GetPendingChecks() >= icingaApp->GetMaxConcurrentChecks()) wait = 0.5; if (wait > 0) { /* Wait for the next check. */ m_CV.timed_wait(lock, boost::posix_time::milliseconds(long(wait * 1000))); continue; } Checkable::Ptr checkable = csi.Object; m_IdleCheckables.erase(checkable); bool forced = checkable->GetForceNextCheck(); bool check = true; if (!forced) { if (!checkable->IsReachable(DependencyCheckExecution)) { Log(LogNotice, "CheckerComponent") << "Skipping check for object '" << checkable->GetName() << "': Dependency failed."; check = false; } Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(checkable); if (host && !service && (!checkable->GetEnableActiveChecks() || !icingaApp->GetEnableHostChecks())) { Log(LogNotice, "CheckerComponent") << "Skipping check for host '" << host->GetName() << "': active host checks are disabled"; check = false; } if (host && service && (!checkable->GetEnableActiveChecks() || !icingaApp->GetEnableServiceChecks())) { Log(LogNotice, "CheckerComponent") << "Skipping check for service '" << service->GetName() << "': active service checks are disabled"; check = false; } TimePeriod::Ptr tp = checkable->GetCheckPeriod(); if (tp && !tp->IsInside(Utility::GetTime())) { Log(LogNotice, "CheckerComponent") << "Skipping check for object '" << checkable->GetName() << "': not in check period '" << tp->GetName() << "'"; check = false; } } /* reschedule the checkable if checks are disabled */ if (!check) { m_IdleCheckables.insert(GetCheckableScheduleInfo(checkable)); lock.unlock(); Log(LogDebug, "CheckerComponent") << "Checks for checkable '" << checkable->GetName() << "' are disabled. Rescheduling check."; checkable->UpdateNextCheck(); lock.lock(); continue; } csi = GetCheckableScheduleInfo(checkable); Log(LogDebug, "CheckerComponent") << "Scheduling info for checkable '" << checkable->GetName() << "' (" << Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", checkable->GetNextCheck()) << "): Object '" << csi.Object->GetName() << "', Next Check: " << Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", csi.NextCheck) << "(" << csi.NextCheck << ")."; m_PendingCheckables.insert(csi); lock.unlock(); if (forced) { ObjectLock olock(checkable); checkable->SetForceNextCheck(false); } Log(LogDebug, "CheckerComponent") << "Executing check for '" << checkable->GetName() << "'"; Checkable::IncreasePendingChecks(); Utility::QueueAsyncCallback(std::bind(&CheckerComponent::ExecuteCheckHelper, CheckerComponent::Ptr(this), checkable)); lock.lock(); } }
void ServiceDbObject::OnConfigUpdateHeavy(void) { Service::Ptr service = static_pointer_cast<Service>(GetObject()); /* groups */ Array::Ptr groups = service->GetGroups(); std::vector<DbQuery> queries; DbQuery query1; query1.Table = DbType::GetByName("ServiceGroup")->GetTable() + "_members"; query1.Type = DbQueryDelete; query1.Category = DbCatConfig; query1.WhereCriteria = new Dictionary(); query1.WhereCriteria->Set("service_object_id", service); queries.push_back(query1); if (groups) { ObjectLock olock(groups); for (const String& groupName : groups) { ServiceGroup::Ptr group = ServiceGroup::GetByName(groupName); DbQuery query2; query2.Table = DbType::GetByName("ServiceGroup")->GetTable() + "_members"; query2.Type = DbQueryInsert; query2.Category = DbCatConfig; query2.Fields = new Dictionary(); query2.Fields->Set("instance_id", 0); /* DbConnection class fills in real ID */ query2.Fields->Set("servicegroup_id", DbValue::FromObjectInsertID(group)); query2.Fields->Set("service_object_id", service); query2.WhereCriteria = new Dictionary(); query2.WhereCriteria->Set("instance_id", 0); /* DbConnection class fills in real ID */ query2.WhereCriteria->Set("servicegroup_id", DbValue::FromObjectInsertID(group)); query2.WhereCriteria->Set("service_object_id", service); queries.push_back(query2); } } DbObject::OnMultipleQueries(queries); /* service dependencies */ Log(LogDebug, "ServiceDbObject") << "service dependencies for '" << service->GetName() << "'"; queries.clear(); DbQuery query2; query2.Table = GetType()->GetTable() + "dependencies"; query2.Type = DbQueryDelete; query2.Category = DbCatConfig; query2.WhereCriteria = new Dictionary(); query2.WhereCriteria->Set("dependent_service_object_id", service); queries.push_back(query2); for (const Dependency::Ptr& dep : service->GetDependencies()) { Checkable::Ptr parent = dep->GetParent(); if (!parent) { Log(LogDebug, "ServiceDbObject") << "Missing parent for dependency '" << dep->GetName() << "'."; continue; } Log(LogDebug, "ServiceDbObject") << "service parents: " << parent->GetName(); int state_filter = dep->GetStateFilter(); /* service dependencies */ Dictionary::Ptr fields1 = new Dictionary(); fields1->Set("service_object_id", parent); fields1->Set("dependent_service_object_id", service); fields1->Set("inherits_parent", 1); fields1->Set("timeperiod_object_id", dep->GetPeriod()); fields1->Set("fail_on_ok", (state_filter & StateFilterOK) ? 1 : 0); fields1->Set("fail_on_warning", (state_filter & StateFilterWarning) ? 1 : 0); fields1->Set("fail_on_critical", (state_filter & StateFilterCritical) ? 1 : 0); fields1->Set("fail_on_unknown", (state_filter & StateFilterUnknown) ? 1 : 0); fields1->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query1; query1.Table = GetType()->GetTable() + "dependencies"; query1.Type = DbQueryInsert; query1.Category = DbCatConfig; query1.Fields = fields1; queries.push_back(query1); } DbObject::OnMultipleQueries(queries); /* service contacts, contactgroups */ Log(LogDebug, "ServiceDbObject") << "service contacts: " << service->GetName(); queries.clear(); DbQuery query3; query3.Table = GetType()->GetTable() + "_contacts"; query3.Type = DbQueryDelete; query3.Category = DbCatConfig; query3.WhereCriteria = new Dictionary(); query3.WhereCriteria->Set("service_id", DbValue::FromObjectInsertID(service)); queries.push_back(query3); for (const User::Ptr& user : CompatUtility::GetCheckableNotificationUsers(service)) { Log(LogDebug, "ServiceDbObject") << "service contacts: " << user->GetName(); Dictionary::Ptr fields_contact = new Dictionary(); fields_contact->Set("service_id", DbValue::FromObjectInsertID(service)); fields_contact->Set("contact_object_id", user); fields_contact->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query_contact; query_contact.Table = GetType()->GetTable() + "_contacts"; query_contact.Type = DbQueryInsert; query_contact.Category = DbCatConfig; query_contact.Fields = fields_contact; queries.push_back(query_contact); } DbObject::OnMultipleQueries(queries); Log(LogDebug, "ServiceDbObject") << "service contactgroups: " << service->GetName(); queries.clear(); DbQuery query4; query4.Table = GetType()->GetTable() + "_contactgroups"; query4.Type = DbQueryDelete; query4.Category = DbCatConfig; query4.WhereCriteria = new Dictionary(); query4.WhereCriteria->Set("service_id", DbValue::FromObjectInsertID(service)); queries.push_back(query4); for (const UserGroup::Ptr& usergroup : CompatUtility::GetCheckableNotificationUserGroups(service)) { Log(LogDebug, "ServiceDbObject") << "service contactgroups: " << usergroup->GetName(); Dictionary::Ptr fields_contact = new Dictionary(); fields_contact->Set("service_id", DbValue::FromObjectInsertID(service)); fields_contact->Set("contactgroup_object_id", usergroup); fields_contact->Set("instance_id", 0); /* DbConnection class fills in real ID */ DbQuery query_contact; query_contact.Table = GetType()->GetTable() + "_contactgroups"; query_contact.Type = DbQueryInsert; query_contact.Category = DbCatConfig; query_contact.Fields = fields_contact; queries.push_back(query_contact); } DbObject::OnMultipleQueries(queries); DoCommonConfigUpdate(); }