Implemented ServiceGroup::GetMembers() and HostGroup::GetMembers().

This commit is contained in:
Gunnar Beutner 2013-01-24 13:21:35 +01:00
parent 8819b4c843
commit ad5256588d
9 changed files with 179 additions and 65 deletions

View File

@ -237,11 +237,11 @@ void CompatComponent::DumpHostObject(ofstream& fp, const Host::Ptr& host)
<< "\t" << "active_checks_enabled" << "\t" << 1 << "\n" << "\t" << "active_checks_enabled" << "\t" << 1 << "\n"
<< "\t" << "passive_checks_enabled" << "\t" << 1 << "\n"; << "\t" << "passive_checks_enabled" << "\t" << 1 << "\n";
set<String> parents = host->GetParents(); set<Host::Ptr> parents = host->GetParents();
if (!parents.empty()) { if (!parents.empty()) {
fp << "\t" << "parents" << "\t"; fp << "\t" << "parents" << "\t";
DumpStringList(fp, parents); DumpNameList(fp, parents);
fp << "\n"; fp << "\n";
} }
@ -370,90 +370,49 @@ void CompatComponent::StatusTimerHandler(void)
<< "# This file is auto-generated. Do not modify this file." << "\n" << "# This file is auto-generated. Do not modify this file." << "\n"
<< "\n"; << "\n";
map<String, vector<String> > hostgroups;
DynamicObject::Ptr object; DynamicObject::Ptr object;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Host")->GetObjects()) { BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Host")->GetObjects()) {
const Host::Ptr& host = static_pointer_cast<Host>(object); const Host::Ptr& host = static_pointer_cast<Host>(object);
Dictionary::Ptr dict;
dict = host->GetGroups();
if (dict) {
Value hostgroup;
BOOST_FOREACH(tie(tuples::ignore, hostgroup), dict) {
hostgroups[hostgroup].push_back(host->GetName());
}
}
DumpHostStatus(statusfp, host); DumpHostStatus(statusfp, host);
DumpHostObject(objectfp, host); DumpHostObject(objectfp, host);
} }
pair<String, vector<String > > hgt; BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("HostGroup")->GetObjects()) {
BOOST_FOREACH(hgt, hostgroups) { const HostGroup::Ptr& hg = static_pointer_cast<HostGroup>(object);
const String& name = hgt.first;
const vector<String>& hosts = hgt.second;
objectfp << "define hostgroup {" << "\n" objectfp << "define hostgroup {" << "\n"
<< "\t" << "hostgroup_name" << "\t" << name << "\n"; << "\t" << "hostgroup_name" << "\t" << hg->GetName() << "\n"
<< "\t" << "alias" << "\t" << hg->GetAlias() << "\n"
if (HostGroup::Exists(name)) { << "\t" << "notes_url" << "\t" << hg->GetNotesUrl() << "\n"
HostGroup::Ptr hg = HostGroup::GetByName(name); << "\t" << "action_url" << "\t" << hg->GetActionUrl() << "\n";
objectfp << "\t" << "alias" << "\t" << hg->GetAlias() << "\n"
<< "\t" << "notes_url" << "\t" << hg->GetNotesUrl() << "\n"
<< "\t" << "action_url" << "\t" << hg->GetActionUrl() << "\n";
}
objectfp << "\t" << "members" << "\t"; objectfp << "\t" << "members" << "\t";
DumpNameList(objectfp, hg->GetMembers());
DumpStringList(objectfp, hosts);
objectfp << "\n" objectfp << "\n"
<< "}" << "\n"; << "}" << "\n";
} }
map<String, vector<Service::Ptr> > servicegroups;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Service")->GetObjects()) { BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Service")->GetObjects()) {
Service::Ptr service = static_pointer_cast<Service>(object); const Service::Ptr& service = static_pointer_cast<Service>(object);
Dictionary::Ptr dict;
dict = service->GetGroups();
if (dict) {
Value servicegroup;
BOOST_FOREACH(tie(tuples::ignore, servicegroup), dict) {
servicegroups[servicegroup].push_back(service);
}
}
DumpServiceStatus(statusfp, service); DumpServiceStatus(statusfp, service);
DumpServiceObject(objectfp, service); DumpServiceObject(objectfp, service);
} }
pair<String, vector<Service::Ptr> > sgt; BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("ServiceGroup")->GetObjects()) {
BOOST_FOREACH(sgt, servicegroups) { const ServiceGroup::Ptr& sg = static_pointer_cast<ServiceGroup>(object);
const String& name = sgt.first;
const vector<Service::Ptr>& services = sgt.second;
objectfp << "define servicegroup {" << "\n" objectfp << "define servicegroup {" << "\n"
<< "\t" << "servicegroup_name" << "\t" << name << "\n"; << "\t" << "servicegroup_name" << "\t" << sg->GetName() << "\n"
<< "\t" << "alias" << "\t" << sg->GetAlias() << "\n"
if (ServiceGroup::Exists(name)) { << "\t" << "notes_url" << "\t" << sg->GetNotesUrl() << "\n"
ServiceGroup::Ptr sg = ServiceGroup::GetByName(name); << "\t" << "action_url" << "\t" << sg->GetActionUrl() << "\n";
objectfp << "\t" << "alias" << "\t" << sg->GetAlias() << "\n"
<< "\t" << "notes_url" << "\t" << sg->GetNotesUrl() << "\n"
<< "\t" << "action_url" << "\t" << sg->GetActionUrl() << "\n";
}
objectfp << "\t" << "members" << "\t"; objectfp << "\t" << "members" << "\t";
vector<String> sglist; vector<String> sglist;
vector<Service::Ptr>::iterator vt; BOOST_FOREACH(const Service::Ptr& service, sg->GetMembers()) {
BOOST_FOREACH(const Service::Ptr& service, services) {
sglist.push_back(service->GetHost()->GetName()); sglist.push_back(service->GetHost()->GetName());
sglist.push_back(service->GetAlias()); sglist.push_back(service->GetAlias());
} }

View File

@ -49,6 +49,21 @@ private:
void DumpHostStatus(ofstream& fp, const Host::Ptr& host); void DumpHostStatus(ofstream& fp, const Host::Ptr& host);
void DumpHostObject(ofstream& fp, const Host::Ptr& host); void DumpHostObject(ofstream& fp, const Host::Ptr& host);
template<typename T>
void DumpNameList(ofstream& fp, const T& list)
{
typename T::const_iterator it;
bool first = true;
for (it = list.begin(); it != list.end(); it++) {
if (!first)
fp << ",";
else
first = false;
fp << (*it)->GetName();
}
}
template<typename T> template<typename T>
void DumpStringList(ofstream& fp, const T& list) void DumpStringList(ofstream& fp, const T& list)
{ {
@ -64,7 +79,6 @@ private:
} }
} }
void DumpServiceStatus(ofstream& fp, const Service::Ptr& service); void DumpServiceStatus(ofstream& fp, const Service::Ptr& service);
void DumpServiceObject(ofstream& fp, const Service::Ptr& service); void DumpServiceObject(ofstream& fp, const Service::Ptr& service);

View File

@ -39,6 +39,8 @@ Host::Host(const Dictionary::Ptr& properties)
m_InitializerDone = true; m_InitializerDone = true;
} }
HostGroup::InvalidateMembersCache();
} }
String Host::GetAlias(void) const String Host::GetAlias(void) const
@ -70,9 +72,9 @@ Dictionary::Ptr Host::GetGroups(void) const
return Get("hostgroups"); return Get("hostgroups");
} }
set<String> Host::GetParents(void) set<Host::Ptr> Host::GetParents(void)
{ {
set<String> parents; set<Host::Ptr> parents;
Dictionary::Ptr dependencies = Get("dependencies"); Dictionary::Ptr dependencies = Get("dependencies");
if (dependencies) { if (dependencies) {
@ -82,10 +84,10 @@ set<String> Host::GetParents(void)
BOOST_FOREACH(tie(tuples::ignore, dependency), dependencies) { BOOST_FOREACH(tie(tuples::ignore, dependency), dependencies) {
Service::Ptr service = Service::GetByName(dependency); Service::Ptr service = Service::GetByName(dependency);
String parent = service->GetHost()->GetName(); Host::Ptr parent = service->GetHost();
/* ignore ourselves */ /* ignore ourselves */
if (parent == GetName()) if (parent->GetName() == GetName())
continue; continue;
parents.insert(parent); parents.insert(parent);
@ -273,3 +275,9 @@ void Host::ObjectRemovedHandler(const ConfigItem::Ptr& item)
} }
} }
void Host::OnAttributeChanged(const String& name, const Value& oldValue)
{
if (name == "hostgroups")
HostGroup::InvalidateMembersCache();
}

View File

@ -41,12 +41,15 @@ public:
String GetAlias(void) const; String GetAlias(void) const;
Dictionary::Ptr GetGroups(void) const; Dictionary::Ptr GetGroups(void) const;
set<String> GetParents(void); set<Host::Ptr> GetParents(void);
Dictionary::Ptr GetMacros(void) const; Dictionary::Ptr GetMacros(void) const;
bool IsReachable(void); bool IsReachable(void);
bool IsUp(void); bool IsUp(void);
protected:
void OnAttributeChanged(const String& name, const Value& oldValue);
private: private:
static bool m_InitializerDone; static bool m_InitializerDone;

View File

@ -21,6 +21,9 @@
using namespace icinga; using namespace icinga;
map<String, vector<String> > HostGroup::m_MembersCache;
bool HostGroup::m_MembersCacheValid = true;
static AttributeDescription hostGroupAttributes[] = { static AttributeDescription hostGroupAttributes[] = {
{ "alias", Attribute_Config }, { "alias", Attribute_Config },
{ "notes_url", Attribute_Config }, { "notes_url", Attribute_Config },
@ -64,3 +67,54 @@ HostGroup::Ptr HostGroup::GetByName(const String& name)
return dynamic_pointer_cast<HostGroup>(configObject); return dynamic_pointer_cast<HostGroup>(configObject);
} }
set<Host::Ptr> HostGroup::GetMembers(void) const
{
set<Host::Ptr> hosts;
ValidateMembersCache();
BOOST_FOREACH(const String& hst, m_MembersCache[GetName()]) {
if (!Host::Exists(hst))
continue;
Host::Ptr host = Host::GetByName(hst);
hosts.insert(host);
}
return hosts;
}
void HostGroup::InvalidateMembersCache(void)
{
m_MembersCacheValid = false;
m_MembersCache.clear();
}
void HostGroup::ValidateMembersCache(void)
{
if (m_MembersCacheValid)
return;
m_MembersCache.clear();
DynamicObject::Ptr object;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Host")->GetObjects()) {
const Host::Ptr& host = static_pointer_cast<Host>(object);
Dictionary::Ptr dict;
dict = host->GetGroups();
if (dict) {
Value hostgroup;
BOOST_FOREACH(tie(tuples::ignore, hostgroup), dict) {
if (!HostGroup::Exists(hostgroup))
Logger::Write(LogWarning, "icinga", "Host group '" + hostgroup + "' used but not defined.");
m_MembersCache[hostgroup].push_back(host->GetName());
}
}
}
m_MembersCacheValid = true;
}

View File

@ -44,6 +44,15 @@ public:
String GetAlias(void) const; String GetAlias(void) const;
String GetNotesUrl(void) const; String GetNotesUrl(void) const;
String GetActionUrl(void) const; String GetActionUrl(void) const;
set<Host::Ptr> GetMembers(void) const;
static void InvalidateMembersCache(void);
private:
static map<String, vector<String> > m_MembersCache;
static bool m_MembersCacheValid;
static void ValidateMembersCache(void);
}; };
} }

View File

@ -62,7 +62,9 @@ boost::signal<void (const Service::Ptr&, const Value&)> Service::OnNextCheckChan
Service::Service(const Dictionary::Ptr& serializedObject) Service::Service(const Dictionary::Ptr& serializedObject)
: DynamicObject(serializedObject) : DynamicObject(serializedObject)
{ } {
ServiceGroup::InvalidateMembersCache();
}
String Service::GetAlias(void) const String Service::GetAlias(void) const
{ {
@ -583,6 +585,8 @@ void Service::OnAttributeChanged(const String& name, const Value& oldValue)
OnCheckerChanged(GetSelf(), oldValue); OnCheckerChanged(GetSelf(), oldValue);
else if (name == "next_check") else if (name == "next_check")
OnNextCheckChanged(GetSelf(), oldValue); OnNextCheckChanged(GetSelf(), oldValue);
else if (name == "servicegroups")
ServiceGroup::InvalidateMembersCache();
} }
void Service::BeginExecuteCheck(const function<void (void)>& callback) void Service::BeginExecuteCheck(const function<void (void)>& callback)

View File

@ -21,6 +21,9 @@
using namespace icinga; using namespace icinga;
map<String, vector<String> > ServiceGroup::m_MembersCache;
bool ServiceGroup::m_MembersCacheValid;
static AttributeDescription serviceGroupAttributes[] = { static AttributeDescription serviceGroupAttributes[] = {
{ "alias", Attribute_Config }, { "alias", Attribute_Config },
{ "notes_url", Attribute_Config }, { "notes_url", Attribute_Config },
@ -64,3 +67,54 @@ ServiceGroup::Ptr ServiceGroup::GetByName(const String& name)
return dynamic_pointer_cast<ServiceGroup>(configObject); return dynamic_pointer_cast<ServiceGroup>(configObject);
} }
set<Service::Ptr> ServiceGroup::GetMembers(void) const
{
set<Service::Ptr> services;
ValidateMembersCache();
BOOST_FOREACH(const String& svc, m_MembersCache[GetName()]) {
if (!Service::Exists(svc))
continue;
Service::Ptr service = Service::GetByName(svc);
services.insert(service);
}
return services;
}
void ServiceGroup::InvalidateMembersCache(void)
{
m_MembersCacheValid = false;
m_MembersCache.clear();
}
void ServiceGroup::ValidateMembersCache(void)
{
if (m_MembersCacheValid)
return;
m_MembersCache.clear();
DynamicObject::Ptr object;
BOOST_FOREACH(tie(tuples::ignore, object), DynamicType::GetByName("Service")->GetObjects()) {
const Service::Ptr& service = static_pointer_cast<Service>(object);
Dictionary::Ptr dict;
dict = service->GetGroups();
if (dict) {
Value servicegroup;
BOOST_FOREACH(tie(tuples::ignore, servicegroup), dict) {
if (!ServiceGroup::Exists(servicegroup))
Logger::Write(LogWarning, "icinga", "Service group '" + servicegroup + "' used but not defined.");
m_MembersCache[servicegroup].push_back(service->GetName());
}
}
}
m_MembersCacheValid = true;
}

View File

@ -44,6 +44,15 @@ public:
String GetAlias(void) const; String GetAlias(void) const;
String GetNotesUrl(void) const; String GetNotesUrl(void) const;
String GetActionUrl(void) const; String GetActionUrl(void) const;
set<Service::Ptr> GetMembers(void) const;
static void InvalidateMembersCache(void);
private:
static map<String, vector<String> > m_MembersCache;
static bool m_MembersCacheValid;
static void ValidateMembersCache(void);
}; };
} }