/****************************************************************************** * 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 "icinga/hostgroup.h" #include "base/dynamictype.h" #include "base/logger_fwd.h" #include "base/objectlock.h" #include "base/timer.h" #include #include using namespace icinga; static boost::mutex l_Mutex; static std::map > l_MembersCache; static bool l_MembersCacheNeedsUpdate = false; static Timer::Ptr l_MembersCacheTimer; REGISTER_TYPE(HostGroup); HostGroup::HostGroup(const Dictionary::Ptr& serializedUpdate) : DynamicObject(serializedUpdate) { RegisterAttribute("display_name", Attribute_Config, &m_DisplayName); RegisterAttribute("notes_url", Attribute_Config, &m_NotesUrl); RegisterAttribute("action_url", Attribute_Config, &m_ActionUrl); } HostGroup::~HostGroup(void) { InvalidateMembersCache(); } void HostGroup::OnRegistrationCompleted(void) { ASSERT(!OwnsLock()); InvalidateMembersCache(); } String HostGroup::GetDisplayName(void) const { if (!m_DisplayName.IsEmpty()) return m_DisplayName; else return GetName(); } String HostGroup::GetNotesUrl(void) const { return m_NotesUrl; } String HostGroup::GetActionUrl(void) const { return m_ActionUrl; } HostGroup::Ptr HostGroup::GetByName(const String& name) { DynamicObject::Ptr configObject = DynamicObject::GetObject("HostGroup", name); if (!configObject) BOOST_THROW_EXCEPTION(std::invalid_argument("HostGroup '" + name + "' does not exist.")); return dynamic_pointer_cast(configObject); } std::set HostGroup::GetMembers(void) const { std::set hosts; { boost::mutex::scoped_lock lock(l_Mutex); BOOST_FOREACH(const Host::WeakPtr& whost, l_MembersCache[GetName()]) { Host::Ptr host = whost.lock(); if (!host) continue; hosts.insert(host); } } return hosts; } void HostGroup::InvalidateMembersCache(void) { boost::mutex::scoped_lock lock(l_Mutex); if (l_MembersCacheNeedsUpdate) return; /* Someone else has already requested a refresh. */ if (!l_MembersCacheTimer) { l_MembersCacheTimer = boost::make_shared(); l_MembersCacheTimer->SetInterval(0.5); l_MembersCacheTimer->OnTimerExpired.connect(boost::bind(&HostGroup::RefreshMembersCache)); l_MembersCacheTimer->Start(); } l_MembersCacheNeedsUpdate = true; } void HostGroup::RefreshMembersCache(void) { { boost::mutex::scoped_lock lock(l_Mutex); if (!l_MembersCacheNeedsUpdate) return; l_MembersCacheNeedsUpdate = false; } Log(LogDebug, "icinga", "Updating HostGroup members cache."); std::map > newMembersCache; BOOST_FOREACH(const DynamicObject::Ptr& object, DynamicType::GetObjects("Host")) { const Host::Ptr& host = static_pointer_cast(object); Array::Ptr groups; groups = host->GetGroups(); if (groups) { ObjectLock mlock(groups); BOOST_FOREACH(const Value& group, groups) { newMembersCache[group].push_back(host); } } } boost::mutex::scoped_lock lock(l_Mutex); l_MembersCache.swap(newMembersCache); }