/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */ #include "config/configitem.hpp" #include "config/configcompilercontext.hpp" #include "config/applyrule.hpp" #include "config/objectrule.hpp" #include "config/configcompiler.hpp" #include "base/application.hpp" #include "base/configtype.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/exception.hpp" #include "base/function.hpp" #include "base/utility.hpp" #include #include #include #include #include #include #include using namespace icinga; std::mutex ConfigItem::m_Mutex; ConfigItem::TypeMap ConfigItem::m_Items; ConfigItem::TypeMap ConfigItem::m_DefaultTemplates; ConfigItem::ItemList ConfigItem::m_UnnamedItems; ConfigItem::IgnoredItemList ConfigItem::m_IgnoredItems; REGISTER_FUNCTION(Internal, run_with_activation_context, &ConfigItem::RunWithActivationContext, "func"); /** * 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(Type::Ptr type, String name, bool abstract, Expression::Ptr exprl, Expression::Ptr filter, bool defaultTmpl, bool ignoreOnError, DebugInfo debuginfo, Dictionary::Ptr scope, String zone, String package) : m_Type(std::move(type)), m_Name(std::move(name)), m_Abstract(abstract), m_Expression(std::move(exprl)), m_Filter(std::move(filter)), m_DefaultTmpl(defaultTmpl), m_IgnoreOnError(ignoreOnError), m_DebugInfo(std::move(debuginfo)), m_Scope(std::move(scope)), m_Zone(std::move(zone)), m_Package(std::move(package)) { } /** * Retrieves the type of the configuration item. * * @returns The type. */ Type::Ptr ConfigItem::GetType() const { return m_Type; } /** * Retrieves the name of the configuration item. * * @returns The name. */ String ConfigItem::GetName() const { return m_Name; } /** * Checks whether the item is abstract. * * @returns true if the item is abstract, false otherwise. */ bool ConfigItem::IsAbstract() const { return m_Abstract; } bool ConfigItem::IsDefaultTemplate() const { return m_DefaultTmpl; } bool ConfigItem::IsIgnoreOnError() const { return m_IgnoreOnError; } /** * Retrieves the debug information for the configuration item. * * @returns The debug information. */ DebugInfo ConfigItem::GetDebugInfo() const { return m_DebugInfo; } Dictionary::Ptr ConfigItem::GetScope() const { return m_Scope; } ConfigObject::Ptr ConfigItem::GetObject() const { return m_Object; } /** * Retrieves the expression list for the configuration item. * * @returns The expression list. */ Expression::Ptr ConfigItem::GetExpression() const { return m_Expression; } /** * Retrieves the object filter for the configuration item. * * @returns The filter expression. */ Expression::Ptr ConfigItem::GetFilter() const { return m_Filter; } class DefaultValidationUtils final : public ValidationUtils { public: bool ValidateName(const String& type, const String& name) const override { ConfigItem::Ptr item = ConfigItem::GetByTypeAndName(Type::GetByName(type), name); if (!item || item->IsAbstract()) return false; return true; } }; /** * Commits the configuration item by creating a ConfigObject * object. * * @returns The ConfigObject that was created/updated. */ ConfigObject::Ptr ConfigItem::Commit(bool discard) { Type::Ptr type = GetType(); #ifdef I2_DEBUG Log(LogDebug, "ConfigItem") << "Commit called for ConfigItem Type=" << type->GetName() << ", Name=" << GetName(); #endif /* I2_DEBUG */ /* Make sure the type is valid. */ if (!type || !ConfigObject::TypeInstance->IsAssignableFrom(type)) BOOST_THROW_EXCEPTION(ScriptError("Type '" + type->GetName() + "' does not exist.", m_DebugInfo)); if (IsAbstract()) return nullptr; ConfigObject::Ptr dobj = static_pointer_cast(type->Instantiate(std::vector())); dobj->SetDebugInfo(m_DebugInfo); dobj->SetZoneName(m_Zone); dobj->SetPackage(m_Package); dobj->SetName(m_Name); DebugHint debugHints; ScriptFrame frame(true, dobj); if (m_Scope) m_Scope->CopyTo(frame.Locals); try { m_Expression->Evaluate(frame, &debugHints); } catch (const std::exception& ex) { if (m_IgnoreOnError) { Log(LogNotice, "ConfigObject") << "Ignoring config object '" << m_Name << "' of type '" << type->GetName() << "' due to errors: " << DiagnosticInformation(ex); { std::unique_lock lock(m_Mutex); m_IgnoredItems.push_back(m_DebugInfo.Path); } return nullptr; } throw; } if (discard) m_Expression.reset(); String item_name; String short_name = dobj->GetShortName(); if (!short_name.IsEmpty()) { item_name = short_name; dobj->SetName(short_name); } else item_name = m_Name; String name = item_name; auto *nc = dynamic_cast(type.get()); if (nc) { if (name.IsEmpty()) BOOST_THROW_EXCEPTION(ScriptError("Object name must not be empty.", m_DebugInfo)); name = nc->MakeName(name, dobj); if (name.IsEmpty()) BOOST_THROW_EXCEPTION(std::runtime_error("Could not determine name for object")); } if (name != item_name) dobj->SetShortName(item_name); dobj->SetName(name); Dictionary::Ptr dhint = debugHints.ToDictionary(); try { DefaultValidationUtils utils; dobj->Validate(FAConfig, utils); } catch (ValidationError& ex) { if (m_IgnoreOnError) { Log(LogNotice, "ConfigObject") << "Ignoring config object '" << m_Name << "' of type '" << type->GetName() << "' due to errors: " << DiagnosticInformation(ex); { std::unique_lock lock(m_Mutex); m_IgnoredItems.push_back(m_DebugInfo.Path); } return nullptr; } ex.SetDebugHint(dhint); throw; } try { dobj->OnConfigLoaded(); } catch (const std::exception& ex) { if (m_IgnoreOnError) { Log(LogNotice, "ConfigObject") << "Ignoring config object '" << m_Name << "' of type '" << m_Type->GetName() << "' due to errors: " << DiagnosticInformation(ex); { std::unique_lock lock(m_Mutex); m_IgnoredItems.push_back(m_DebugInfo.Path); } return nullptr; } throw; } Value serializedObject; try { if (ConfigCompilerContext::GetInstance()->IsOpen()) { serializedObject = Serialize(dobj, FAConfig); } else { AssertNoCircularReferences(dobj); } } catch (const CircularReferenceError& ex) { BOOST_THROW_EXCEPTION(ValidationError(dobj, ex.GetPath(), "Circular references are not allowed")); } if (ConfigCompilerContext::GetInstance()->IsOpen()) { Dictionary::Ptr persistentItem = new Dictionary({ { "type", type->GetName() }, { "name", GetName() }, { "properties", serializedObject }, { "debug_hints", dhint }, { "debug_info", new Array({ m_DebugInfo.Path, m_DebugInfo.FirstLine, m_DebugInfo.FirstColumn, m_DebugInfo.LastLine, m_DebugInfo.LastColumn, }) } }); ConfigCompilerContext::GetInstance()->WriteObject(persistentItem); } dhint.reset(); dobj->Register(); m_Object = dobj; return dobj; } /** * Registers the configuration item. */ void ConfigItem::Register() { m_ActivationContext = ActivationContext::GetCurrentContext(); std::unique_lock lock(m_Mutex); /* 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(m_Type.get())) m_UnnamedItems.emplace_back(this); else { auto& items = m_Items[m_Type]; auto it = items.find(m_Name); if (it != items.end()) { std::ostringstream msgbuf; msgbuf << "A configuration item of type '" << m_Type->GetName() << "' and name '" << GetName() << "' already exists (" << it->second->GetDebugInfo() << "), new declaration: " << GetDebugInfo(); BOOST_THROW_EXCEPTION(ScriptError(msgbuf.str())); } m_Items[m_Type][m_Name] = this; if (m_DefaultTmpl) m_DefaultTemplates[m_Type][m_Name] = this; } } /** * Unregisters the configuration item. */ void ConfigItem::Unregister() { if (m_Object) { m_Object->Unregister(); m_Object.reset(); } std::unique_lock lock(m_Mutex); m_UnnamedItems.erase(std::remove(m_UnnamedItems.begin(), m_UnnamedItems.end(), this), m_UnnamedItems.end()); m_Items[m_Type].erase(m_Name); m_DefaultTemplates[m_Type].erase(m_Name); } /** * 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::GetByTypeAndName(const Type::Ptr& type, const String& name) { std::unique_lock lock(m_Mutex); auto it = m_Items.find(type); if (it == m_Items.end()) return nullptr; auto it2 = it->second.find(name); if (it2 == it->second.end()) return nullptr; return it2->second; } bool ConfigItem::CommitNewItems(const ActivationContext::Ptr& context, WorkQueue& upq, std::vector& newItems) { typedef std::pair ItemPair; std::unordered_map> itemsByType; std::vector::size_type total = 0; { std::unique_lock lock(m_Mutex); for (const TypeMap::value_type& kv : m_Items) { std::vector items; for (const ItemMap::value_type& kv2 : kv.second) { if (kv2.second->m_Abstract || kv2.second->m_Object) continue; if (kv2.second->m_ActivationContext != context) continue; items.emplace_back(kv2.second, false); } if (!items.empty()) { total += items.size(); itemsByType.emplace(kv.first.get(), std::move(items)); } } ItemList newUnnamedItems; for (const ConfigItem::Ptr& item : m_UnnamedItems) { if (item->m_ActivationContext != context) { newUnnamedItems.push_back(item); continue; } if (item->m_Abstract || item->m_Object) continue; itemsByType[item->m_Type.get()].emplace_back(item, true); ++total; } m_UnnamedItems.swap(newUnnamedItems); } if (!total) return true; #ifdef I2_DEBUG Log(LogDebug, "configitem") << "Committing " << total << " new items."; #endif /* I2_DEBUG */ std::set types; std::set completed_types; int itemsCount {0}; for (const Type::Ptr& type : Type::GetAllTypes()) { if (ConfigObject::TypeInstance->IsAssignableFrom(type)) types.insert(type); } while (types.size() != completed_types.size()) { for (const Type::Ptr& type : types) { if (completed_types.find(type) != completed_types.end()) continue; bool unresolved_dep = false; /* skip this type (for now) if there are unresolved load dependencies */ for (auto pLoadDep : type->GetLoadDependencies()) { if (types.find(pLoadDep) != types.end() && completed_types.find(pLoadDep) == completed_types.end()) { unresolved_dep = true; break; } } if (unresolved_dep) continue; std::atomic committed_items(0); { auto items (itemsByType.find(type.get())); if (items != itemsByType.end()) { for (const ItemPair& pair: items->second) { newItems.emplace_back(pair.first); } upq.ParallelFor(items->second, [&committed_items](const ItemPair& ip) { const ConfigItem::Ptr& item = ip.first; if (!item->Commit(ip.second)) { if (item->IsIgnoreOnError()) { item->Unregister(); } return; } committed_items++; }); upq.Join(); } } itemsCount += committed_items; completed_types.insert(type); #ifdef I2_DEBUG if (committed_items > 0) Log(LogDebug, "configitem") << "Committed " << committed_items << " items of type '" << type->GetName() << "'."; #endif /* I2_DEBUG */ if (upq.HasExceptions()) return false; } } #ifdef I2_DEBUG Log(LogDebug, "configitem") << "Committed " << itemsCount << " items."; #endif /* I2_DEBUG */ completed_types.clear(); while (types.size() != completed_types.size()) { for (const Type::Ptr& type : types) { if (completed_types.find(type) != completed_types.end()) continue; bool unresolved_dep = false; /* skip this type (for now) if there are unresolved load dependencies */ for (auto pLoadDep : type->GetLoadDependencies()) { if (types.find(pLoadDep) != types.end() && completed_types.find(pLoadDep) == completed_types.end()) { unresolved_dep = true; break; } } if (unresolved_dep) continue; std::atomic notified_items(0); { auto items (itemsByType.find(type.get())); if (items != itemsByType.end()) { upq.ParallelFor(items->second, [¬ified_items](const ItemPair& ip) { const ConfigItem::Ptr& item = ip.first; if (!item->m_Object) return; try { item->m_Object->OnAllConfigLoaded(); notified_items++; } catch (const std::exception& ex) { if (!item->m_IgnoreOnError) throw; Log(LogNotice, "ConfigObject") << "Ignoring config object '" << item->m_Name << "' of type '" << item->m_Type->GetName() << "' due to errors: " << DiagnosticInformation(ex); item->Unregister(); { std::unique_lock lock(item->m_Mutex); item->m_IgnoredItems.push_back(item->m_DebugInfo.Path); } } }); upq.Join(); } } completed_types.insert(type); #ifdef I2_DEBUG if (notified_items > 0) Log(LogDebug, "configitem") << "Sent OnAllConfigLoaded to " << notified_items << " items of type '" << type->GetName() << "'."; #endif /* I2_DEBUG */ if (upq.HasExceptions()) return false; notified_items = 0; for (auto loadDep : type->GetLoadDependencies()) { auto items (itemsByType.find(loadDep)); if (items != itemsByType.end()) { upq.ParallelFor(items->second, [&type, ¬ified_items](const ItemPair& ip) { const ConfigItem::Ptr& item = ip.first; if (!item->m_Object) return; ActivationScope ascope(item->m_ActivationContext); item->m_Object->CreateChildObjects(type); notified_items++; }); } } upq.Join(); #ifdef I2_DEBUG if (notified_items > 0) Log(LogDebug, "configitem") << "Sent CreateChildObjects to " << notified_items << " items of type '" << type->GetName() << "'."; #endif /* I2_DEBUG */ if (upq.HasExceptions()) return false; // Make sure to activate any additionally generated items if (!CommitNewItems(context, upq, newItems)) return false; } } return true; } bool ConfigItem::CommitItems(const ActivationContext::Ptr& context, WorkQueue& upq, std::vector& newItems, bool silent) { if (!silent) Log(LogInformation, "ConfigItem", "Committing config item(s)."); if (!CommitNewItems(context, upq, newItems)) { upq.ReportExceptions("config"); for (const ConfigItem::Ptr& item : newItems) { item->Unregister(); } return false; } ApplyRule::CheckMatches(silent); if (!silent) { /* log stats for external parsers */ typedef std::map ItemCountMap; ItemCountMap itemCounts; for (const ConfigItem::Ptr& item : newItems) { if (!item->m_Object) continue; itemCounts[item->m_Object->GetReflectionType()]++; } for (const ItemCountMap::value_type& kv : itemCounts) { Log(LogInformation, "ConfigItem") << "Instantiated " << kv.second << " " << (kv.second != 1 ? kv.first->GetPluralName() : kv.first->GetName()) << "."; } } return true; } /** * ActivateItems activates new config items. * * @param newItems Vector of items to be activated * @param runtimeCreated Whether the objects were created by a runtime object * @param mainConfigActivation Whether this is the call for activating the main configuration during startup * @param withModAttrs Whether this call shall read the modified attributes file * @param cookie Cookie for preventing message loops * @return Whether the config activation was successful (in case of errors, exceptions are thrown) */ bool ConfigItem::ActivateItems(const std::vector& newItems, bool runtimeCreated, bool mainConfigActivation, bool withModAttrs, const Value& cookie) { static std::mutex mtx; std::unique_lock lock(mtx); if (withModAttrs) { /* restore modified attributes */ if (Utility::PathExists(Configuration::ModAttrPath)) { std::unique_ptr expression = ConfigCompiler::CompileFile(Configuration::ModAttrPath); if (expression) { try { ScriptFrame frame(true); expression->Evaluate(frame); } catch (const std::exception& ex) { Log(LogCritical, "config", DiagnosticInformation(ex)); } } } } for (const ConfigItem::Ptr& item : newItems) { if (!item->m_Object) continue; ConfigObject::Ptr object = item->m_Object; if (object->IsActive()) continue; #ifdef I2_DEBUG Log(LogDebug, "ConfigItem") << "Setting 'active' to true for object '" << object->GetName() << "' of type '" << object->GetReflectionType()->GetName() << "'"; #endif /* I2_DEBUG */ object->PreActivate(); } if (mainConfigActivation) Log(LogInformation, "ConfigItem", "Triggering Start signal for config items"); /* Activate objects in priority order. */ std::vector types = Type::GetAllTypes(); std::sort(types.begin(), types.end(), [](const Type::Ptr& a, const Type::Ptr& b) { if (a->GetActivationPriority() < b->GetActivationPriority()) return true; return false; }); /* Find the last logger type to be activated. */ Type::Ptr lastLoggerType = nullptr; for (const Type::Ptr& type : types) { if (Logger::TypeInstance->IsAssignableFrom(type)) { lastLoggerType = type; } } for (const Type::Ptr& type : types) { for (const ConfigItem::Ptr& item : newItems) { if (!item->m_Object) continue; ConfigObject::Ptr object = item->m_Object; Type::Ptr objectType = object->GetReflectionType(); if (objectType != type) continue; #ifdef I2_DEBUG Log(LogDebug, "ConfigItem") << "Activating object '" << object->GetName() << "' of type '" << objectType->GetName() << "' with priority " << objectType->GetActivationPriority(); #endif /* I2_DEBUG */ object->Activate(runtimeCreated, cookie); } if (mainConfigActivation && type == lastLoggerType) { /* Disable early logging configuration once the last logger type was activated. */ Logger::DisableEarlyLogging(); } } if (mainConfigActivation) Log(LogInformation, "ConfigItem", "Activated all objects."); return true; } bool ConfigItem::RunWithActivationContext(const Function::Ptr& function) { ActivationScope scope; if (!function) BOOST_THROW_EXCEPTION(ScriptError("'function' argument must not be null.")); function->Invoke(); WorkQueue upq(25000, Configuration::Concurrency); upq.SetName("ConfigItem::RunWithActivationContext"); std::vector newItems; if (!CommitItems(scope.GetContext(), upq, newItems, true)) return false; if (!ActivateItems(newItems, false, false)) return false; return true; } std::vector ConfigItem::GetItems(const Type::Ptr& type) { std::vector items; std::unique_lock lock(m_Mutex); auto it = m_Items.find(type); if (it == m_Items.end()) return items; items.reserve(it->second.size()); for (const ItemMap::value_type& kv : it->second) { items.push_back(kv.second); } return items; } std::vector ConfigItem::GetDefaultTemplates(const Type::Ptr& type) { std::vector items; std::unique_lock lock(m_Mutex); auto it = m_DefaultTemplates.find(type); if (it == m_DefaultTemplates.end()) return items; items.reserve(it->second.size()); for (const ItemMap::value_type& kv : it->second) { items.push_back(kv.second); } return items; } void ConfigItem::RemoveIgnoredItems(const String& allowedConfigPath) { std::unique_lock lock(m_Mutex); for (const String& path : m_IgnoredItems) { if (path.Find(allowedConfigPath) == String::NPos) continue; Log(LogNotice, "ConfigItem") << "Removing ignored item path '" << path << "'."; (void) unlink(path.CStr()); } m_IgnoredItems.clear(); }