icinga2/lib/config/configitem.cpp

227 lines
6.1 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;
ConfigItem::ItemMap ConfigItem::m_Items;
boost::signal<void (const ConfigItem::Ptr&)> ConfigItem::OnCommitted;
boost::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 ExpressionList::Ptr& exprl, const vector<String>& parents,
2012-07-06 14:33:10 +02:00
const DebugInfo& debuginfo)
: m_Type(type), m_Name(name), m_ExpressionList(exprl),
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;
}
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;
}
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::CalculateProperties(const Dictionary::Ptr& dictionary) const
{
BOOST_FOREACH(const String& name, m_Parents) {
2012-07-16 22:00:50 +02:00
ConfigItem::Ptr parent = ConfigItem::GetObject(GetType(), name);
if (!parent) {
stringstream message;
2012-09-19 12:32:39 +02:00
message << "Parent object '" << name << "' does not"
" exist (" << m_DebugInfo << ")";
2012-07-17 20:41:06 +02:00
throw_exception(domain_error(message.str()));
}
parent->CalculateProperties(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.
*/
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr ConfigItem::Commit(void)
{
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr dobj = m_DynamicObject.lock();
2012-06-15 19:32:41 +02:00
Dictionary::Ptr properties = boost::make_shared<Dictionary>();
CalculateProperties(properties);
/* Create a fake update in the format that
* DynamicObject::ApplyUpdate expects. */
Dictionary::Ptr attrs = boost::make_shared<Dictionary>();
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", DynamicObject::GetCurrentTx());
attrs->Set(key, attr);
}
Dictionary::Ptr update = boost::make_shared<Dictionary>();
update->Set("attrs", attrs);
update->Set("configTx", DynamicObject::GetCurrentTx());
DynamicType::Ptr dtype = DynamicType::GetByName(GetType());
2013-01-23 10:41:38 +01:00
if (!dtype)
throw_exception(runtime_error("Type '" + GetType() + "' does not exist."));
if (!dobj)
dobj = dtype->GetObject(GetName());
if (!dobj)
dobj = dtype->CreateObject(update);
else
dobj->ApplyUpdate(update, Attribute_Config);
2012-07-30 10:17:29 +02:00
m_DynamicObject = dobj;
if (dobj->IsAbstract())
dobj->Unregister();
else
dobj->Register();
/* TODO: Figure out whether there are any child objects which inherit
* from this config item and Commit() them as well */
m_Items[make_pair(GetType(), GetName())] = GetSelf();
OnCommitted(GetSelf());
return dobj;
}
2012-09-19 12:32:39 +02:00
/**
* Unregisters the configuration item.
*/
void ConfigItem::Unregister(void)
{
2012-07-30 10:17:29 +02:00
DynamicObject::Ptr dobj = m_DynamicObject.lock();
2012-06-27 18:43:34 +02:00
if (dobj)
dobj->Unregister();
ConfigItem::ItemMap::iterator it;
it = m_Items.find(make_pair(GetType(), GetName()));
if (it != m_Items.end())
m_Items.erase(it);
OnRemoved(GetSelf());
}
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
{
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.
*/
ConfigItem::Ptr ConfigItem::GetObject(const String& type, const String& name)
{
ConfigItem::ItemMap::iterator it;
it = m_Items.find(make_pair(type, name));
if (it == m_Items.end())
return ConfigItem::Ptr();
return it->second;
}