/****************************************************************************** * 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-icinga.h" using namespace icinga; boost::mutex UserGroup::m_Mutex; map > UserGroup::m_MembersCache; bool UserGroup::m_MembersCacheNeedsUpdate = false; Timer::Ptr UserGroup::m_MembersCacheTimer; REGISTER_TYPE(UserGroup); UserGroup::UserGroup(const Dictionary::Ptr& properties) : DynamicObject(properties) { RegisterAttribute("display_name", Attribute_Config, &m_DisplayName); } UserGroup::~UserGroup(void) { InvalidateMembersCache(); } /** * @threadsafety Always. */ void UserGroup::OnRegistrationCompleted(void) { assert(!OwnsLock()); InvalidateMembersCache(); } /** * @threadsafety Always. */ String UserGroup::GetDisplayName(void) const { if (!m_DisplayName.IsEmpty()) return m_DisplayName; else return GetName(); } /** * @threadsafety Always. */ UserGroup::Ptr UserGroup::GetByName(const String& name) { DynamicObject::Ptr configObject = DynamicObject::GetObject("UserGroup", name); if (!configObject) BOOST_THROW_EXCEPTION(invalid_argument("UserGroup '" + name + "' does not exist.")); return dynamic_pointer_cast(configObject); } /** * @threadsafety Always. */ set UserGroup::GetMembers(void) const { set users; { boost::mutex::scoped_lock lock(m_Mutex); BOOST_FOREACH(const User::WeakPtr& wuser, m_MembersCache[GetName()]) { User::Ptr user = wuser.lock(); if (!user) continue; users.insert(user); } } return users; } /** * @threadsafety Always. */ void UserGroup::InvalidateMembersCache(void) { boost::mutex::scoped_lock lock(m_Mutex); if (m_MembersCacheNeedsUpdate) return; /* Someone else has already requested a refresh. */ if (!m_MembersCacheTimer) { m_MembersCacheTimer = boost::make_shared(); m_MembersCacheTimer->SetInterval(0.5); m_MembersCacheTimer->OnTimerExpired.connect(boost::bind(&UserGroup::RefreshMembersCache)); m_MembersCacheTimer->Start(); } m_MembersCacheNeedsUpdate = true; } /** * @threadsafety Always. */ void UserGroup::RefreshMembersCache(void) { { boost::mutex::scoped_lock lock(m_Mutex); if (!m_MembersCacheNeedsUpdate) return; m_MembersCacheNeedsUpdate = false; } Logger::Write(LogInformation, "icinga", "Updating UserGroup members cache."); map > newMembersCache; BOOST_FOREACH(const DynamicObject::Ptr& object, DynamicType::GetObjects("User")) { const User::Ptr& user = static_pointer_cast(object); Dictionary::Ptr dict; dict = user->GetGroups(); if (dict) { ObjectLock mlock(dict); Value UserGroup; BOOST_FOREACH(tie(tuples::ignore, UserGroup), dict) { newMembersCache[UserGroup].push_back(user); } } } boost::mutex::scoped_lock lock(m_Mutex); m_MembersCache.swap(newMembersCache); }