icinga2/lib/config/configitem.cpp

432 lines
10 KiB
C++
Raw Normal View History

/******************************************************************************
* 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-config.h"
using namespace icinga;
recursive_mutex ConfigItem::m_Mutex;
ConfigItem::ItemMap ConfigItem::m_Items;
2013-02-17 19:14:34 +01:00
signals2::signal<void (const ConfigItem::Ptr&)> ConfigItem::OnCommitted;
signals2::signal<void (const ConfigItem::Ptr&)> ConfigItem::OnRemoved;
2012-09-19 12:32:39 +02:00
/**
* Constructor for the ConfigItem class.
*
* @param type The object type.
* @param name The name of the item.
* @param exprl Expression list for the item.
* @param parents Parent objects for the item.
* @param debuginfo Debug information.
*/
ConfigItem::ConfigItem(const String& type, const String& name,
const String& unit, const ExpressionList::Ptr& exprl,
const vector<String>& parents, const DebugInfo& debuginfo)
: m_Type(type), m_Name(name), m_Unit(unit), m_ExpressionList(exprl),
2012-07-06 14:33:10 +02:00
m_Parents(parents), m_DebugInfo(debuginfo)
{
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the type of the configuration item.
*
* @returns The type.
*/
String ConfigItem::GetType(void) const
{
return m_Type;
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the name of the configuration item.
*
* @returns The name.
*/
String ConfigItem::GetName(void) const
{
return m_Name;
}
/**
* Retrieves the name of the compilation unit this item belongs to.
*
* @returns The unit name.
*/
String ConfigItem::GetUnit(void) const
{
return m_Unit;
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the debug information for the configuration item.
*
* @returns The debug information.
*/
DebugInfo ConfigItem::GetDebugInfo(void) const
{
return m_DebugInfo;
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the expression list for the configuration item.
*
* @returns The expression list.
*/
ExpressionList::Ptr ConfigItem::GetExpressionList(void) const
{
return m_ExpressionList;
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the list of parents for the configuration item.
*
* @returns The list of parents.
*/
vector<String> ConfigItem::GetParents(void) const
{
return m_Parents;
}
Dictionary::Ptr ConfigItem::Link(void) const
{
Dictionary::Ptr attrs = boost::make_shared<Dictionary>();
InternalLink(attrs);
return attrs;
}
2012-09-19 12:32:39 +02:00
/**
* Calculates the object's properties based on parent objects and the object's
* expression list.
*
* @param dictionary The dictionary that should be used to store the
* properties.
*/
void ConfigItem::InternalLink(const Dictionary::Ptr& dictionary) const
{
2013-03-02 09:07:47 +01:00
ObjectLock olock(this);
BOOST_FOREACH(const String& name, m_Parents) {
2013-02-18 14:40:24 +01:00
ConfigItem::Ptr parent;
ConfigCompilerContext *context = ConfigCompilerContext::GetContext();
if (context)
2013-03-04 15:52:42 +01:00
parent = context->GetItem(m_Type, name);
2013-02-18 14:40:24 +01:00
/* ignore already active objects while we're in the compiler
* context and linking to existing items is disabled. */
if (!parent && (!context || (context->GetFlags() & CompilerLinkExisting)))
2013-03-04 15:52:42 +01:00
parent = ConfigItem::GetObject(m_Type, name);
if (!parent) {
stringstream message;
2012-09-19 12:32:39 +02:00
message << "Parent object '" << name << "' does not"
" exist (" << m_DebugInfo << ")";
BOOST_THROW_EXCEPTION(domain_error(message.str()));
}
parent->InternalLink(dictionary);
}
m_ExpressionList->Execute(dictionary);
}
2012-09-19 12:32:39 +02:00
/**
* Commits the configuration item by creating or updating a DynamicObject
* object.
*
* @returns The DynamicObject that was created/updated.
*/
2013-03-02 09:07:47 +01:00
DynamicObject::Ptr ConfigItem::Commit(void)
{
2013-03-02 09:07:47 +01:00
assert(!OwnsLock());
2013-02-20 19:52:25 +01:00
2013-03-02 09:07:47 +01:00
String type, name;
2013-02-20 19:52:25 +01:00
2013-03-02 09:07:47 +01:00
Logger::Write(LogDebug, "base", "Commit called for ConfigItem Type=" + GetType() + ", Name=" + GetName());
/* Make sure the type is valid. */
2013-03-02 09:07:47 +01:00
DynamicType::Ptr dtype = DynamicType::GetByName(GetType());
if (!dtype)
2013-03-02 09:07:47 +01:00
BOOST_THROW_EXCEPTION(runtime_error("Type '" + GetType() + "' does not exist."));
/* Try to find an existing item with the same type and name. */
pair<String, String> ikey = make_pair(GetType(), GetName());
ConfigItem::Ptr oldItem;
2013-02-20 19:52:25 +01:00
{
recursive_mutex::scoped_lock lock(m_Mutex);
2013-02-20 19:52:25 +01:00
ItemMap::iterator it = m_Items.find(ikey);
2013-03-02 09:07:47 +01:00
if (it != m_Items.end())
oldItem = it->second;
}
2013-03-02 09:07:47 +01:00
set<ConfigItem::WeakPtr> children;
2013-03-02 09:07:47 +01:00
if (oldItem) {
ObjectLock olock(oldItem);
2013-02-20 19:52:25 +01:00
2013-03-02 09:07:47 +01:00
/* Unregister the old item from its parents. */
oldItem->UnregisterFromParents();
/* Steal the old item's children. */
2013-03-04 15:52:42 +01:00
children = oldItem->m_ChildObjects;
2013-03-02 09:07:47 +01:00
}
{
ObjectLock olock(this);
m_ChildObjects = children;
}
ConfigItem::Ptr self = GetSelf();
2013-03-02 09:07:47 +01:00
{
recursive_mutex::scoped_lock lock(m_Mutex);
2013-02-20 19:52:25 +01:00
/* Register this item. */
m_Items[ikey] = self;
2013-02-20 19:52:25 +01:00
}
2013-03-04 15:52:42 +01:00
DynamicObject::Ptr dobj = m_DynamicObject.lock();
2013-03-02 09:07:47 +01:00
2013-03-01 12:07:52 +01:00
if (!dobj)
2013-03-04 15:52:42 +01:00
dobj = dtype->GetObject(m_Name);
/* Register this item with its parents. */
2013-03-04 15:52:42 +01:00
BOOST_FOREACH(const String& parentName, m_Parents) {
ConfigItem::Ptr parent = GetObject(m_Type, parentName);
parent->m_ChildObjects.insert(self);
}
/* Create a fake update in the format that
* DynamicObject::ApplyUpdate expects. */
Dictionary::Ptr attrs = boost::make_shared<Dictionary>();
2013-03-01 12:07:52 +01:00
double tx = DynamicObject::GetCurrentTx();
2013-03-02 09:07:47 +01:00
Dictionary::Ptr properties = Link();
2013-03-01 12:07:52 +01:00
{
ObjectLock olock(properties);
String key;
Value data;
BOOST_FOREACH(tie(key, data), properties) {
Dictionary::Ptr attr = boost::make_shared<Dictionary>();
attr->Set("data", data);
attr->Set("type", Attribute_Config);
attr->Set("tx", tx);
attr->Seal();
attrs->Set(key, attr);
}
}
2013-03-01 12:07:52 +01:00
attrs->Seal();
Dictionary::Ptr update = boost::make_shared<Dictionary>();
update->Set("attrs", attrs);
update->Set("configTx", DynamicObject::GetCurrentTx());
2013-03-01 12:07:52 +01:00
update->Seal();
/* Update or create the object and apply the configuration settings. */
2013-02-18 14:40:24 +01:00
bool was_null = false;
2013-02-18 14:40:24 +01:00
if (!dobj) {
2013-03-04 15:52:42 +01:00
dobj = dtype->CreateObject(update);
2013-02-18 14:40:24 +01:00
was_null = true;
}
2013-03-02 09:07:47 +01:00
if (!was_null)
dobj->ApplyUpdate(update, Attribute_Config);
2013-02-18 14:40:24 +01:00
2013-03-02 09:07:47 +01:00
{
ObjectLock olock(this);
2013-02-18 14:40:24 +01:00
2013-03-02 09:07:47 +01:00
m_DynamicObject = dobj;
2013-02-18 14:40:24 +01:00
}
2013-03-02 09:07:47 +01:00
if (dobj->IsAbstract())
dobj->Unregister();
else
dobj->Register();
/* notify our children of the update */
BOOST_FOREACH(const ConfigItem::WeakPtr wchild, children) {
const ConfigItem::Ptr& child = wchild.lock();
if (!child)
continue;
child->OnParentCommitted();
}
2013-03-04 15:52:42 +01:00
OnCommitted(self);
return dobj;
}
2012-09-19 12:32:39 +02:00
/**
* Unregisters the configuration item.
*/
void ConfigItem::Unregister(void)
{
2013-03-02 09:07:47 +01:00
assert(!OwnsLock());
2012-06-27 18:43:34 +02:00
2013-03-04 15:52:42 +01:00
DynamicObject::Ptr dobj = m_DynamicObject.lock();
2013-03-02 09:07:47 +01:00
if (dobj)
2012-06-27 18:43:34 +02:00
dobj->Unregister();
2013-03-02 09:07:47 +01:00
{
ObjectLock olock(this);
2013-03-02 09:07:47 +01:00
ConfigItem::ItemMap::iterator it;
2013-03-04 15:52:42 +01:00
it = m_Items.find(make_pair(m_Type, m_Name));
2013-03-02 09:07:47 +01:00
if (it != m_Items.end())
m_Items.erase(it);
2013-03-04 15:52:42 +01:00
UnregisterFromParents();
}
OnRemoved(GetSelf());
}
void ConfigItem::UnregisterFromParents(void)
{
2013-03-04 15:52:42 +01:00
assert(OwnsLock());
2013-03-02 09:07:47 +01:00
BOOST_FOREACH(const String& parentName, m_Parents) {
ConfigItem::Ptr parent = GetObject(GetType(), parentName);
2013-03-02 09:07:47 +01:00
if (parent)
2013-03-04 15:52:42 +01:00
parent->m_ChildObjects.erase(GetSelf());
}
}
/*
* Notifies an item that one of its parents has been committed.
*/
void ConfigItem::OnParentCommitted(void)
{
2013-03-02 09:07:47 +01:00
assert(!OwnsLock());
2013-03-04 15:52:42 +01:00
ConfigItem::Ptr self = GetSelf();
2013-02-20 19:52:25 +01:00
2013-03-04 15:52:42 +01:00
if (GetObject(m_Type, m_Name) != self)
return;
2013-02-20 19:52:25 +01:00
2013-03-02 09:07:47 +01:00
Commit();
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the DynamicObject that belongs to the configuration item.
*
* @returns The DynamicObject.
*/
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr ConfigItem::GetDynamicObject(void) const
{
2013-03-02 09:07:47 +01:00
ObjectLock olock(this);
2012-07-30 10:17:29 +02:00
return m_DynamicObject.lock();
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves a configuration item by type and name.
*
* @param type The type of the ConfigItem that is to be looked up.
* @param name The name of the ConfigItem that is to be looked up.
* @returns The configuration item.
2013-02-17 19:14:34 +01:00
* @threadsafety Always.
2012-09-19 12:32:39 +02:00
*/
ConfigItem::Ptr ConfigItem::GetObject(const String& type, const String& name)
{
recursive_mutex::scoped_lock lock(m_Mutex);
ConfigItem::ItemMap::iterator it;
2013-02-17 19:14:34 +01:00
it = m_Items.find(make_pair(type, name));
2013-02-17 19:14:34 +01:00
if (it != m_Items.end())
return it->second;
return ConfigItem::Ptr();
}
/**
* Dumps the config item to the specified stream using Icinga's config item
* syntax.
*
* @param fp The stream.
*/
void ConfigItem::Dump(ostream& fp) const
{
2013-03-02 09:07:47 +01:00
ObjectLock olock(this);
fp << "object \"" << m_Type << "\" \"" << m_Name << "\"";
if (m_Parents.size() > 0) {
fp << " inherits";
bool first = true;
BOOST_FOREACH(const String& name, m_Parents) {
if (!first)
fp << ",";
else
first = false;
fp << " \"" << name << "\"";
}
}
fp << " {" << "\n";
m_ExpressionList->Dump(fp, 1);
fp << "}" << "\n";
}
2013-02-17 19:14:34 +01:00
/**
2013-02-18 14:40:24 +01:00
* @threadsafety Always.
2013-02-17 19:14:34 +01:00
*/
void ConfigItem::UnloadUnit(const String& unit)
{
recursive_mutex::scoped_lock lock(m_Mutex);
2013-02-17 19:14:34 +01:00
Logger::Write(LogInformation, "config", "Unloading config items from compilation unit '" + unit + "'");
vector<ConfigItem::Ptr> obsoleteItems;
2013-02-19 23:02:08 +01:00
ConfigItem::Ptr item;
BOOST_FOREACH(tie(tuples::ignore, item), m_Items) {
ObjectLock olock(item);
2013-02-18 14:40:24 +01:00
2013-03-04 15:52:42 +01:00
if (item->m_Unit != unit)
2013-02-19 23:02:08 +01:00
continue;
2013-02-18 14:40:24 +01:00
2013-02-19 23:02:08 +01:00
obsoleteItems.push_back(item);
}
2013-02-18 14:40:24 +01:00
BOOST_FOREACH(const ConfigItem::Ptr& item, obsoleteItems) {
item->Unregister();
}
}