icinga2/lib/config/configitem.cpp

442 lines
12 KiB
C++
Raw Normal View History

/******************************************************************************
* Icinga 2 *
* Copyright (C) 2012-2014 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. *
******************************************************************************/
2014-05-25 16:23:35 +02:00
#include "config/configitem.hpp"
#include "config/configcompilercontext.hpp"
#include "config/applyrule.hpp"
#include "config/objectrule.hpp"
#include "config/configtype.hpp"
#include "base/application.hpp"
#include "base/dynamictype.hpp"
#include "base/objectlock.hpp"
#include "base/convert.hpp"
2014-10-19 14:21:12 +02:00
#include "base/logger.hpp"
2014-05-25 16:23:35 +02:00
#include "base/debug.hpp"
#include "base/workqueue.hpp"
#include "base/exception.hpp"
#include "base/stdiostream.hpp"
#include "base/netstring.hpp"
2014-11-06 19:35:47 +01:00
#include "base/serializer.hpp"
2014-10-26 19:59:49 +01:00
#include "base/json.hpp"
#include "base/configerror.hpp"
2013-03-16 21:18:53 +01:00
#include <sstream>
#include <fstream>
2013-03-16 21:18:53 +01:00
#include <boost/foreach.hpp>
using namespace icinga;
2013-03-15 18:21:29 +01:00
boost::mutex ConfigItem::m_Mutex;
ConfigItem::TypeMap ConfigItem::m_Items;
ConfigItem::ItemList ConfigItem::m_UnnamedItems;
ConfigItem::ItemList ConfigItem::m_CommittedItems;
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 unit The unit of the item.
* @param abstract Whether the item is a template.
2012-09-19 12:32:39 +02:00
* @param exprl Expression list for the item.
* @param debuginfo Debug information.
*/
ConfigItem::ConfigItem(const String& type, const String& name,
2014-11-09 19:48:28 +01:00
bool abstract, const boost::shared_ptr<Expression>& exprl,
const boost::shared_ptr<Expression>& filter,
2014-11-06 19:35:47 +01:00
const DebugInfo& debuginfo, const Object::Ptr& scope,
const String& zone)
2014-11-06 19:35:47 +01:00
: m_Type(type), m_Name(name), m_Abstract(abstract),
m_Expression(exprl), m_Filter(filter),
m_DebugInfo(debuginfo), m_Scope(scope), m_Zone(zone)
{
}
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;
}
/**
* Checks whether the item is abstract.
*
* @returns true if the item is abstract, false otherwise.
*/
bool ConfigItem::IsAbstract(void) const
{
return m_Abstract;
}
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;
}
2014-11-06 19:35:47 +01:00
Object::Ptr ConfigItem::GetScope(void) const
{
return m_Scope;
}
2012-09-19 12:32:39 +02:00
/**
* Retrieves the expression list for the configuration item.
*
* @returns The expression list.
*/
2014-11-09 19:48:28 +01:00
boost::shared_ptr<Expression> ConfigItem::GetExpression(void) const
{
2014-11-09 19:48:28 +01:00
return m_Expression;
}
/**
* Retrieves the object filter for the configuration item.
*
* @returns The filter expression.
*/
boost::shared_ptr<Expression> ConfigItem::GetFilter(void) const
{
return m_Filter;
}
2012-09-19 12:32:39 +02:00
/**
2013-12-03 09:59:21 +01:00
* Commits the configuration item by creating a DynamicObject
2012-09-19 12:32:39 +02:00
* object.
*
* @returns The DynamicObject that was created/updated.
*/
2014-11-06 19:35:47 +01:00
DynamicObject::Ptr ConfigItem::Commit(bool discard)
{
ASSERT(!OwnsLock());
2013-02-20 19:52:25 +01:00
#ifdef _DEBUG
2014-10-19 17:52:17 +02:00
Log(LogDebug, "ConfigItem")
<< "Commit called for ConfigItem Type=" << GetType() << ", Name=" << GetName();
#endif /* _DEBUG */
/* Make sure the type is valid. */
2014-11-06 19:35:47 +01:00
Type::Ptr type = Type::GetByName(GetType());
2014-11-11 23:06:47 +01:00
ASSERT(type && Type::GetByName("DynamicObject")->IsAssignableFrom(type));
2013-03-02 09:07:47 +01:00
if (IsAbstract())
return DynamicObject::Ptr();
2014-11-06 19:35:47 +01:00
DynamicObject::Ptr dobj = static_pointer_cast<DynamicObject>(type->Instantiate());
dobj->SetDebugInfo(m_DebugInfo);
2014-11-06 19:35:47 +01:00
dobj->SetTypeName(m_Type);
dobj->SetZone(m_Zone);
Dictionary::Ptr locals = new Dictionary();
2014-11-06 19:35:47 +01:00
locals->Set("__parent", m_Scope);
dobj->SetParentScope(locals);
2014-11-11 23:06:47 +01:00
locals.reset();
2014-11-06 19:35:47 +01:00
dobj->SetName(m_Name);
2014-11-06 19:35:47 +01:00
DebugHint debugHints;
try {
2014-11-09 19:48:28 +01:00
m_Expression->Evaluate(dobj, &debugHints);
2014-11-06 19:35:47 +01:00
} catch (const ConfigError& ex) {
const DebugInfo *di = boost::get_error_info<errinfo_debuginfo>(ex);
ConfigCompilerContext::GetInstance()->AddMessage(true, ex.what(), di ? *di : DebugInfo());
} catch (const std::exception& ex) {
ConfigCompilerContext::GetInstance()->AddMessage(true, DiagnosticInformation(ex));
}
if (discard)
2014-11-09 19:48:28 +01:00
m_Expression.reset();
2014-11-06 19:35:47 +01:00
2014-11-11 23:06:47 +01:00
dobj->SetParentScope(Object::Ptr());
2014-11-06 19:35:47 +01:00
String name = m_Name;
NameComposer *nc = dynamic_cast<NameComposer *>(type.get());
2014-11-06 19:35:47 +01:00
if (nc) {
name = nc->MakeName(m_Name, dobj);
if (name.IsEmpty())
BOOST_THROW_EXCEPTION(std::runtime_error("Could not determine name for object"));
}
if (name != m_Name)
dobj->SetShortName(m_Name);
dobj->SetName(name);
dobj->OnConfigLoaded();
{
boost::mutex::scoped_lock lock(m_Mutex);
m_CommittedItems.push_back(this);
}
2014-11-06 19:35:47 +01:00
Dictionary::Ptr attrs = Serialize(dobj, FAConfig);
Dictionary::Ptr persistentItem = new Dictionary();
2014-11-06 19:35:47 +01:00
persistentItem->Set("type", GetType());
persistentItem->Set("name", GetName());
persistentItem->Set("properties", attrs);
persistentItem->Set("debug_hints", debugHints.ToDictionary());
ConfigCompilerContext::GetInstance()->WriteObject(persistentItem);
2014-11-11 23:06:47 +01:00
persistentItem.reset();
2014-11-06 19:35:47 +01:00
ConfigType::Ptr ctype = ConfigType::GetByName(GetType());
if (!ctype)
ConfigCompilerContext::GetInstance()->AddMessage(false, "No validation type found for object '" + GetName() + "' of type '" + GetType() + "'");
else {
TypeRuleUtilities utils;
try {
ctype->ValidateItem(GetName(), dobj, GetDebugInfo(), &utils);
2014-11-06 19:35:47 +01:00
} catch (const ConfigError& ex) {
const DebugInfo *di = boost::get_error_info<errinfo_debuginfo>(ex);
ConfigCompilerContext::GetInstance()->AddMessage(true, ex.what(), di ? *di : DebugInfo());
} catch (const std::exception& ex) {
ConfigCompilerContext::GetInstance()->AddMessage(true, DiagnosticInformation(ex));
}
}
dobj->Register();
m_Object = dobj;
return dobj;
}
/**
* Registers the configuration item.
*/
void ConfigItem::Register(void)
{
Type::Ptr type = Type::GetByName(m_Type);
/* If this is a non-abstract object with a composite name
* we register it in m_UnnamedItems instead of m_Items. */
if (!m_Abstract && dynamic_cast<NameComposer *>(type.get())) {
boost::mutex::scoped_lock lock(m_Mutex);
m_UnnamedItems.push_back(this);
} else {
boost::mutex::scoped_lock lock(m_Mutex);
m_Items[m_Type][m_Name] = this;
}
}
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)
{
{
boost::mutex::scoped_lock lock(m_Mutex);
ConfigItem::TypeMap::const_iterator it = m_Items.find(type);
if (it == m_Items.end())
return ConfigItem::Ptr();
ConfigItem::ItemMap::const_iterator it2 = it->second.find(name);
2013-02-17 19:14:34 +01:00
if (it2 == it->second.end())
return ConfigItem::Ptr();
return it2->second;
}
}
bool ConfigItem::CommitNewItems(ParallelWorkQueue& upq)
{
typedef std::pair<ConfigItem::Ptr, bool> ItemPair;
std::vector<ItemPair> items;
2013-09-24 13:13:14 +02:00
do {
items.clear();
2013-02-17 19:14:34 +01:00
{
boost::mutex::scoped_lock lock(m_Mutex);
BOOST_FOREACH(const TypeMap::value_type& kv, m_Items) {
BOOST_FOREACH(const ItemMap::value_type& kv2, kv.second)
{
if (!kv2.second->m_Abstract && !kv2.second->m_Object)
items.push_back(std::make_pair(kv2.second, false));
}
}
BOOST_FOREACH(const ConfigItem::Ptr& item, m_UnnamedItems) {
if (!item->m_Abstract && !item->m_Object)
items.push_back(std::make_pair(item, true));
}
m_UnnamedItems.clear();
}
BOOST_FOREACH(const ItemPair& ip, items) {
upq.Enqueue(boost::bind(&ConfigItem::Commit, ip.first, ip.second));
}
upq.Join();
if (ConfigCompilerContext::GetInstance()->HasErrors())
return false;
std::vector<ConfigItem::Ptr> new_items;
{
boost::mutex::scoped_lock lock(m_Mutex);
new_items.swap(m_CommittedItems);
}
BOOST_FOREACH(const ConfigItem::Ptr& item, new_items) {
upq.Enqueue(boost::bind(&DynamicObject::OnAllConfigLoaded, item->m_Object));
}
upq.Join();
} while (!items.empty());
return true;
}
bool ConfigItem::ValidateItems(void)
{
ParallelWorkQueue upq;
if (ConfigCompilerContext::GetInstance()->HasErrors())
return false;
Log(LogInformation, "ConfigItem", "Committing config items");
if (!CommitNewItems(upq))
return false;
2013-08-29 19:05:06 +02:00
ApplyRule::CheckMatches();
ApplyRule::DiscardRules();
ConfigItem::DiscardItems();
ConfigType::DiscardTypes();
/* log stats for external parsers */
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
int count = std::distance(type->GetObjects().first, type->GetObjects().second);
if (count > 0)
2014-10-19 17:52:17 +02:00
Log(LogInformation, "ConfigItem")
<< "Checked " << count << " " << type->GetName() << "(s).";
}
return !ConfigCompilerContext::GetInstance()->HasErrors();
}
bool ConfigItem::ActivateItems(void)
{
2013-09-24 13:13:14 +02:00
if (ConfigCompilerContext::GetInstance()->HasErrors())
return false;
2013-08-29 10:40:43 +02:00
/* restore the previous program state */
try {
DynamicObject::RestoreObjects(Application::GetStatePath());
} catch (const std::exception& ex) {
2014-10-19 17:52:17 +02:00
Log(LogCritical, "ConfigItem")
<< "Failed to restore state file: " << DiagnosticInformation(ex);
}
2013-08-29 10:40:43 +02:00
Log(LogInformation, "ConfigItem", "Triggering Start signal for config items");
ParallelWorkQueue upq;
2013-08-29 19:25:34 +02:00
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) {
if (object->IsActive())
continue;
2013-08-29 19:05:06 +02:00
#ifdef _DEBUG
2014-10-19 17:52:17 +02:00
Log(LogDebug, "ConfigItem")
<< "Activating object '" << object->GetName() << "' of type '" << object->GetType()->GetName() << "'";
#endif /* _DEBUG */
upq.Enqueue(boost::bind(&DynamicObject::Activate, object));
}
}
upq.Join();
#ifdef _DEBUG
BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) {
BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) {
2013-08-29 19:25:34 +02:00
ASSERT(object->IsActive());
}
}
#endif /* _DEBUG */
Log(LogInformation, "ConfigItem", "Activated all objects.");
2013-09-24 13:13:14 +02:00
return true;
}
void ConfigItem::DiscardItems(void)
{
boost::mutex::scoped_lock lock(m_Mutex);
m_Items.clear();
}
std::vector<ConfigItem::Ptr> ConfigItem::GetItems(const String& type)
{
std::vector<ConfigItem::Ptr> items;
boost::mutex::scoped_lock lock(m_Mutex);
TypeMap::const_iterator it = m_Items.find(type);
if (it == m_Items.end())
return items;
BOOST_FOREACH(const ItemMap::value_type& kv, it->second)
{
items.push_back(kv.second);
}
return items;
}