/****************************************************************************** * 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. * ******************************************************************************/ #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" #include "base/logger.hpp" #include "base/debug.hpp" #include "base/workqueue.hpp" #include "base/exception.hpp" #include "base/stdiostream.hpp" #include "base/netstring.hpp" #include "base/serializer.hpp" #include "base/json.hpp" #include "base/configerror.hpp" #include #include #include using namespace icinga; boost::mutex ConfigItem::m_Mutex; ConfigItem::TypeMap ConfigItem::m_Items; ConfigItem::ItemList ConfigItem::m_UnnamedItems; ConfigItem::ItemList ConfigItem::m_CommittedItems; /** * 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. * @param exprl Expression list for the item. * @param debuginfo Debug information. */ ConfigItem::ConfigItem(const String& type, const String& name, bool abstract, const boost::shared_ptr& exprl, const boost::shared_ptr& filter, const DebugInfo& debuginfo, const Object::Ptr& scope, const String& zone) : m_Type(type), m_Name(name), m_Abstract(abstract), m_Expression(exprl), m_Filter(filter), m_DebugInfo(debuginfo), m_Scope(scope), m_Zone(zone) { } /** * Retrieves the type of the configuration item. * * @returns The type. */ String ConfigItem::GetType(void) const { return m_Type; } /** * 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; } /** * Retrieves the debug information for the configuration item. * * @returns The debug information. */ DebugInfo ConfigItem::GetDebugInfo(void) const { return m_DebugInfo; } Object::Ptr ConfigItem::GetScope(void) const { return m_Scope; } /** * Retrieves the expression list for the configuration item. * * @returns The expression list. */ boost::shared_ptr ConfigItem::GetExpression(void) const { return m_Expression; } /** * Retrieves the object filter for the configuration item. * * @returns The filter expression. */ boost::shared_ptr ConfigItem::GetFilter(void) const { return m_Filter; } /** * Commits the configuration item by creating a DynamicObject * object. * * @returns The DynamicObject that was created/updated. */ DynamicObject::Ptr ConfigItem::Commit(bool discard) { ASSERT(!OwnsLock()); #ifdef _DEBUG Log(LogDebug, "ConfigItem") << "Commit called for ConfigItem Type=" << GetType() << ", Name=" << GetName(); #endif /* _DEBUG */ /* Make sure the type is valid. */ Type::Ptr type = Type::GetByName(GetType()); ASSERT(type && Type::GetByName("DynamicObject")->IsAssignableFrom(type)); if (IsAbstract()) return DynamicObject::Ptr(); DynamicObject::Ptr dobj = static_pointer_cast(type->Instantiate()); dobj->SetDebugInfo(m_DebugInfo); dobj->SetTypeName(m_Type); dobj->SetZone(m_Zone); Dictionary::Ptr locals = new Dictionary(); locals->Set("__parent", m_Scope); dobj->SetParentScope(locals); locals.reset(); dobj->SetName(m_Name); DebugHint debugHints; try { m_Expression->Evaluate(dobj, &debugHints); } catch (const ConfigError& ex) { const DebugInfo *di = boost::get_error_info(ex); ConfigCompilerContext::GetInstance()->AddMessage(true, ex.what(), di ? *di : DebugInfo()); } catch (const std::exception& ex) { ConfigCompilerContext::GetInstance()->AddMessage(true, DiagnosticInformation(ex)); } if (discard) m_Expression.reset(); dobj->SetParentScope(Object::Ptr()); String name = m_Name; NameComposer *nc = dynamic_cast(type.get()); 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); } Dictionary::Ptr attrs = Serialize(dobj, FAConfig); Dictionary::Ptr persistentItem = new Dictionary(); persistentItem->Set("type", GetType()); persistentItem->Set("name", GetName()); persistentItem->Set("properties", attrs); persistentItem->Set("debug_hints", debugHints.ToDictionary()); ConfigCompilerContext::GetInstance()->WriteObject(persistentItem); persistentItem.reset(); 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); } catch (const ConfigError& ex) { const DebugInfo *di = boost::get_error_info(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(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; } } /** * 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); if (it2 == it->second.end()) return ConfigItem::Ptr(); return it2->second; } } bool ConfigItem::CommitNewItems(ParallelWorkQueue& upq) { typedef std::pair ItemPair; std::vector items; do { items.clear(); { 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 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; 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) Log(LogInformation, "ConfigItem") << "Checked " << count << " " << type->GetName() << "(s)."; } return !ConfigCompilerContext::GetInstance()->HasErrors(); } bool ConfigItem::ActivateItems(void) { if (ConfigCompilerContext::GetInstance()->HasErrors()) return false; /* restore the previous program state */ try { DynamicObject::RestoreObjects(Application::GetStatePath()); } catch (const std::exception& ex) { Log(LogCritical, "ConfigItem") << "Failed to restore state file: " << DiagnosticInformation(ex); } Log(LogInformation, "ConfigItem", "Triggering Start signal for config items"); ParallelWorkQueue upq; BOOST_FOREACH(const DynamicType::Ptr& type, DynamicType::GetTypes()) { BOOST_FOREACH(const DynamicObject::Ptr& object, type->GetObjects()) { if (object->IsActive()) continue; #ifdef _DEBUG 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()) { ASSERT(object->IsActive()); } } #endif /* _DEBUG */ Log(LogInformation, "ConfigItem", "Activated all objects."); return true; } void ConfigItem::DiscardItems(void) { boost::mutex::scoped_lock lock(m_Mutex); m_Items.clear(); } std::vector ConfigItem::GetItems(const String& type) { std::vector 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; }