From 6aa23554270b8f49f542a4a5d9dd1842e411b660 Mon Sep 17 00:00:00 2001 From: "Alexander A. Klimov" Date: Wed, 4 Dec 2024 10:57:30 +0100 Subject: [PATCH 1/5] DependencyGraph: switch "parent" and "child" terminology The .ti files call `DependencyGraph::AddDependency(this, service.get())`. Obviously, `service.get()` is the parent and `this` (Downtime, Notification, ...) is the child. The DependencyGraph terminology should reflect this not to confuse its future users. --- lib/base/dependencygraph.cpp | 16 ++++++++-------- lib/base/dependencygraph.hpp | 6 +++--- lib/base/scriptutils.cpp | 2 +- lib/remote/configobjectutility.cpp | 2 +- lib/remote/objectqueryhandler.cpp | 2 +- 5 files changed, 14 insertions(+), 14 deletions(-) diff --git a/lib/base/dependencygraph.cpp b/lib/base/dependencygraph.cpp index 025eb3ea35a..88cb18194a2 100644 --- a/lib/base/dependencygraph.cpp +++ b/lib/base/dependencygraph.cpp @@ -7,18 +7,18 @@ using namespace icinga; std::mutex DependencyGraph::m_Mutex; std::map > DependencyGraph::m_Dependencies; -void DependencyGraph::AddDependency(Object *parent, Object *child) +void DependencyGraph::AddDependency(Object* child, Object* parent) { std::unique_lock lock(m_Mutex); - m_Dependencies[child][parent]++; + m_Dependencies[parent][child]++; } -void DependencyGraph::RemoveDependency(Object *parent, Object *child) +void DependencyGraph::RemoveDependency(Object* child, Object* parent) { std::unique_lock lock(m_Mutex); - auto& refs = m_Dependencies[child]; - auto it = refs.find(parent); + auto& refs = m_Dependencies[parent]; + auto it = refs.find(child); if (it == refs.end()) return; @@ -29,15 +29,15 @@ void DependencyGraph::RemoveDependency(Object *parent, Object *child) refs.erase(it); if (refs.empty()) - m_Dependencies.erase(child); + m_Dependencies.erase(parent); } -std::vector DependencyGraph::GetParents(const Object::Ptr& child) +std::vector DependencyGraph::GetChildren(const Object::Ptr& parent) { std::vector objects; std::unique_lock lock(m_Mutex); - auto it = m_Dependencies.find(child.get()); + auto it = m_Dependencies.find(parent.get()); if (it != m_Dependencies.end()) { typedef std::pair kv_pair; diff --git a/lib/base/dependencygraph.hpp b/lib/base/dependencygraph.hpp index 51aa90ecad5..43dd97c08f2 100644 --- a/lib/base/dependencygraph.hpp +++ b/lib/base/dependencygraph.hpp @@ -18,9 +18,9 @@ namespace icinga { class DependencyGraph { public: - static void AddDependency(Object *parent, Object *child); - static void RemoveDependency(Object *parent, Object *child); - static std::vector GetParents(const Object::Ptr& child); + static void AddDependency(Object* child, Object* parent); + static void RemoveDependency(Object* child, Object* parent); + static std::vector GetChildren(const Object::Ptr& parent); private: DependencyGraph(); diff --git a/lib/base/scriptutils.cpp b/lib/base/scriptutils.cpp index 7fe856de355..5bf2164a7b8 100644 --- a/lib/base/scriptutils.cpp +++ b/lib/base/scriptutils.cpp @@ -520,7 +520,7 @@ String ScriptUtils::MsiGetComponentPathShim(const String& component) Array::Ptr ScriptUtils::TrackParents(const Object::Ptr& child) { - return Array::FromVector(DependencyGraph::GetParents(child)); + return Array::FromVector(DependencyGraph::GetChildren(child)); } double ScriptUtils::Ptr(const Object::Ptr& object) diff --git a/lib/remote/configobjectutility.cpp b/lib/remote/configobjectutility.cpp index 60268f6e195..75dcfab93ca 100644 --- a/lib/remote/configobjectutility.cpp +++ b/lib/remote/configobjectutility.cpp @@ -303,7 +303,7 @@ bool ConfigObjectUtility::CreateObject(const Type::Ptr& type, const String& full bool ConfigObjectUtility::DeleteObjectHelper(const ConfigObject::Ptr& object, bool cascade, const Array::Ptr& errors, const Array::Ptr& diagnosticInformation, const Value& cookie) { - std::vector parents = DependencyGraph::GetParents(object); + auto parents (DependencyGraph::GetChildren(object)); Type::Ptr type = object->GetReflectionType(); diff --git a/lib/remote/objectqueryhandler.cpp b/lib/remote/objectqueryhandler.cpp index ad73030dafb..4c40ef3ed40 100644 --- a/lib/remote/objectqueryhandler.cpp +++ b/lib/remote/objectqueryhandler.cpp @@ -208,7 +208,7 @@ bool ObjectQueryHandler::HandleRequest( Array::Ptr used_by = new Array(); metaAttrs.emplace_back("used_by", used_by); - for (const Object::Ptr& pobj : DependencyGraph::GetParents((obj))) + for (auto& pobj : DependencyGraph::GetChildren(obj)) { ConfigObject::Ptr configObj = dynamic_pointer_cast(pobj); From c64ff492ec76617f21a5bbc396fa1654366974c2 Mon Sep 17 00:00:00 2001 From: "Alexander A. Klimov" Date: Wed, 4 Dec 2024 11:19:37 +0100 Subject: [PATCH 2/5] DependencyGraph: use ConfigObject*, not Object* This saves dynamic_cast + if() on every item of GetChildren(). --- lib/base/dependencygraph.cpp | 13 ++++++------- lib/base/dependencygraph.hpp | 10 +++++----- lib/base/scriptutils.cpp | 2 +- lib/remote/configobjectutility.cpp | 7 +------ lib/remote/objectqueryhandler.cpp | 8 +------- tools/mkclass/classcompiler.cpp | 24 ++++++++++++------------ 6 files changed, 26 insertions(+), 38 deletions(-) diff --git a/lib/base/dependencygraph.cpp b/lib/base/dependencygraph.cpp index 88cb18194a2..58f685ce013 100644 --- a/lib/base/dependencygraph.cpp +++ b/lib/base/dependencygraph.cpp @@ -5,15 +5,15 @@ using namespace icinga; std::mutex DependencyGraph::m_Mutex; -std::map > DependencyGraph::m_Dependencies; +std::map> DependencyGraph::m_Dependencies; -void DependencyGraph::AddDependency(Object* child, Object* parent) +void DependencyGraph::AddDependency(ConfigObject* child, ConfigObject* parent) { std::unique_lock lock(m_Mutex); m_Dependencies[parent][child]++; } -void DependencyGraph::RemoveDependency(Object* child, Object* parent) +void DependencyGraph::RemoveDependency(ConfigObject* child, ConfigObject* parent) { std::unique_lock lock(m_Mutex); @@ -32,16 +32,15 @@ void DependencyGraph::RemoveDependency(Object* child, Object* parent) m_Dependencies.erase(parent); } -std::vector DependencyGraph::GetChildren(const Object::Ptr& parent) +std::vector DependencyGraph::GetChildren(const ConfigObject::Ptr& parent) { - std::vector objects; + std::vector objects; std::unique_lock lock(m_Mutex); auto it = m_Dependencies.find(parent.get()); if (it != m_Dependencies.end()) { - typedef std::pair kv_pair; - for (const kv_pair& kv : it->second) { + for (auto& kv : it->second) { objects.emplace_back(kv.first); } } diff --git a/lib/base/dependencygraph.hpp b/lib/base/dependencygraph.hpp index 43dd97c08f2..6afb08fc598 100644 --- a/lib/base/dependencygraph.hpp +++ b/lib/base/dependencygraph.hpp @@ -4,7 +4,7 @@ #define DEPENDENCYGRAPH_H #include "base/i2-base.hpp" -#include "base/object.hpp" +#include "base/configobject.hpp" #include #include @@ -18,15 +18,15 @@ namespace icinga { class DependencyGraph { public: - static void AddDependency(Object* child, Object* parent); - static void RemoveDependency(Object* child, Object* parent); - static std::vector GetChildren(const Object::Ptr& parent); + static void AddDependency(ConfigObject* child, ConfigObject* parent); + static void RemoveDependency(ConfigObject* child, ConfigObject* parent); + static std::vector GetChildren(const ConfigObject::Ptr& parent); private: DependencyGraph(); static std::mutex m_Mutex; - static std::map > m_Dependencies; + static std::map> m_Dependencies; }; } diff --git a/lib/base/scriptutils.cpp b/lib/base/scriptutils.cpp index 5bf2164a7b8..3611799ffc9 100644 --- a/lib/base/scriptutils.cpp +++ b/lib/base/scriptutils.cpp @@ -520,7 +520,7 @@ String ScriptUtils::MsiGetComponentPathShim(const String& component) Array::Ptr ScriptUtils::TrackParents(const Object::Ptr& child) { - return Array::FromVector(DependencyGraph::GetChildren(child)); + return Array::FromVector(DependencyGraph::GetChildren(dynamic_pointer_cast(child))); } double ScriptUtils::Ptr(const Object::Ptr& object) diff --git a/lib/remote/configobjectutility.cpp b/lib/remote/configobjectutility.cpp index 75dcfab93ca..9d195805026 100644 --- a/lib/remote/configobjectutility.cpp +++ b/lib/remote/configobjectutility.cpp @@ -319,12 +319,7 @@ bool ConfigObjectUtility::DeleteObjectHelper(const ConfigObject::Ptr& object, bo return false; } - for (const Object::Ptr& pobj : parents) { - ConfigObject::Ptr parentObj = dynamic_pointer_cast(pobj); - - if (!parentObj) - continue; - + for (auto& parentObj : parents) { DeleteObjectHelper(parentObj, cascade, errors, diagnosticInformation, cookie); } diff --git a/lib/remote/objectqueryhandler.cpp b/lib/remote/objectqueryhandler.cpp index 4c40ef3ed40..8b731378933 100644 --- a/lib/remote/objectqueryhandler.cpp +++ b/lib/remote/objectqueryhandler.cpp @@ -208,13 +208,7 @@ bool ObjectQueryHandler::HandleRequest( Array::Ptr used_by = new Array(); metaAttrs.emplace_back("used_by", used_by); - for (auto& pobj : DependencyGraph::GetChildren(obj)) - { - ConfigObject::Ptr configObj = dynamic_pointer_cast(pobj); - - if (!configObj) - continue; - + for (auto& configObj : DependencyGraph::GetChildren(obj)) { used_by->Add(new Dictionary({ { "type", configObj->GetReflectionType()->GetName() }, { "name", configObj->GetName() } diff --git a/tools/mkclass/classcompiler.cpp b/tools/mkclass/classcompiler.cpp index 3a49576b87d..cabb3c6fe9c 100644 --- a/tools/mkclass/classcompiler.cpp +++ b/tools/mkclass/classcompiler.cpp @@ -894,13 +894,13 @@ void ClassCompiler::HandleClass(const Klass& klass, const ClassDebugInfo&) m_Impl << "\t" << "if (oldValue) {" << std::endl << "\t\t" << "ObjectLock olock(oldValue);" << std::endl << "\t\t" << "for (const String& ref : oldValue) {" << std::endl - << "\t\t\t" << "DependencyGraph::RemoveDependency(this, ConfigObject::GetObject"; + << "\t\t\tDependencyGraph::RemoveDependency("; /* Ew */ if (field.Type.TypeName == "Zone" && m_Library == "base") - m_Impl << "(\"Zone\", "; + m_Impl << "dynamic_cast(this), ConfigObject::GetObject(\"Zone\", "; else - m_Impl << "<" << field.Type.TypeName << ">("; + m_Impl << "this, ConfigObject::GetObject<" << field.Type.TypeName << ">("; m_Impl << "ref).get());" << std::endl << "\t\t" << "}" << std::endl @@ -908,36 +908,36 @@ void ClassCompiler::HandleClass(const Klass& klass, const ClassDebugInfo&) << "\t" << "if (newValue) {" << std::endl << "\t\t" << "ObjectLock olock(newValue);" << std::endl << "\t\t" << "for (const String& ref : newValue) {" << std::endl - << "\t\t\t" << "DependencyGraph::AddDependency(this, ConfigObject::GetObject"; + << "\t\t\tDependencyGraph::AddDependency("; /* Ew */ if (field.Type.TypeName == "Zone" && m_Library == "base") - m_Impl << "(\"Zone\", "; + m_Impl << "dynamic_cast(this), ConfigObject::GetObject(\"Zone\", "; else - m_Impl << "<" << field.Type.TypeName << ">("; + m_Impl << "this, ConfigObject::GetObject<" << field.Type.TypeName << ">("; m_Impl << "ref).get());" << std::endl << "\t\t" << "}" << std::endl << "\t" << "}" << std::endl; } else { m_Impl << "\t" << "if (!oldValue.IsEmpty())" << std::endl - << "\t\t" << "DependencyGraph::RemoveDependency(this, ConfigObject::GetObject"; + << "\t\tDependencyGraph::RemoveDependency("; /* Ew */ if (field.Type.TypeName == "Zone" && m_Library == "base") - m_Impl << "(\"Zone\", "; + m_Impl << "dynamic_cast(this), ConfigObject::GetObject(\"Zone\", "; else - m_Impl << "<" << field.Type.TypeName << ">("; + m_Impl << "this, ConfigObject::GetObject<" << field.Type.TypeName << ">("; m_Impl << "oldValue).get());" << std::endl << "\t" << "if (!newValue.IsEmpty())" << std::endl - << "\t\t" << "DependencyGraph::AddDependency(this, ConfigObject::GetObject"; + << "\t\tDependencyGraph::AddDependency("; /* Ew */ if (field.Type.TypeName == "Zone" && m_Library == "base") - m_Impl << "(\"Zone\", "; + m_Impl << "dynamic_cast(this), ConfigObject::GetObject(\"Zone\", "; else - m_Impl << "<" << field.Type.TypeName << ">("; + m_Impl << "this, ConfigObject::GetObject<" << field.Type.TypeName << ">("; m_Impl << "newValue).get());" << std::endl; } From 5c0ce6350cc159fba4ddd96cd362bff6587f207d Mon Sep 17 00:00:00 2001 From: Yonas Habteab Date: Wed, 4 Dec 2024 08:03:01 +0100 Subject: [PATCH 3/5] DependencyGraph: Allow lookups by parent & child dependencies --- lib/base/dependencygraph.cpp | 62 ++++++++++++++++++++--------- lib/base/dependencygraph.hpp | 75 +++++++++++++++++++++++++++++++++++- 2 files changed, 116 insertions(+), 21 deletions(-) diff --git a/lib/base/dependencygraph.cpp b/lib/base/dependencygraph.cpp index 58f685ce013..a9d0dbb95cd 100644 --- a/lib/base/dependencygraph.cpp +++ b/lib/base/dependencygraph.cpp @@ -5,45 +5,69 @@ using namespace icinga; std::mutex DependencyGraph::m_Mutex; -std::map> DependencyGraph::m_Dependencies; +DependencyGraph::DependencyMap DependencyGraph::m_Dependencies; void DependencyGraph::AddDependency(ConfigObject* child, ConfigObject* parent) { std::unique_lock lock(m_Mutex); - m_Dependencies[parent][child]++; + if (auto [it, inserted] = m_Dependencies.insert(Edge(parent, child)); !inserted) { + m_Dependencies.modify(it, [](Edge& e) { e.count++; }); + } } void DependencyGraph::RemoveDependency(ConfigObject* child, ConfigObject* parent) { std::unique_lock lock(m_Mutex); - auto& refs = m_Dependencies[parent]; - auto it = refs.find(child); + if (auto it(m_Dependencies.find(Edge(parent, child))); it != m_Dependencies.end()) { + // A number <= 1 means, this isn't referenced by anyone and should be erased from the container. + if (it->count == 1) { + m_Dependencies.erase(it); + return; + } - if (it == refs.end()) - return; + // Otherwise, each remove operation will should decrement this by 1 till it reaches <= 1 + // and causes the edge to completely be erased from the container. + m_Dependencies.modify(it, [](Edge& e) { e.count--; }); + } +} - it->second--; +/** + * Returns all the parent objects of the given child object. + * + * @param child The child object. + * + * @returns A list of the parent objects. + */ +std::vector DependencyGraph::GetParents(const ConfigObject::Ptr& child) +{ + std::vector objects; - if (it->second == 0) - refs.erase(it); + std::unique_lock lock(m_Mutex); + auto [begin, end] = m_Dependencies.get<2>().equal_range(child.get()); + std::transform(begin, end, std::back_inserter(objects), [](const Edge& edge) { + return edge.parent; + }); - if (refs.empty()) - m_Dependencies.erase(parent); + return objects; } +/** + * Returns all the dependent objects of the given parent object. + * + * @param parent The parent object. + * + * @returns A list of the dependent objects. + */ std::vector DependencyGraph::GetChildren(const ConfigObject::Ptr& parent) { std::vector objects; - std::unique_lock lock(m_Mutex); - auto it = m_Dependencies.find(parent.get()); - - if (it != m_Dependencies.end()) { - for (auto& kv : it->second) { - objects.emplace_back(kv.first); - } - } + std::unique_lock lock(m_Mutex); + auto [begin, end] = m_Dependencies.get<1>().equal_range(parent.get()); + std::transform(begin, end, std::back_inserter(objects), [](const Edge& edge) { + return edge.child; + }); return objects; } diff --git a/lib/base/dependencygraph.hpp b/lib/base/dependencygraph.hpp index 6afb08fc598..ef1a2a4a61c 100644 --- a/lib/base/dependencygraph.hpp +++ b/lib/base/dependencygraph.hpp @@ -5,7 +5,9 @@ #include "base/i2-base.hpp" #include "base/configobject.hpp" -#include +#include +#include +#include #include namespace icinga { @@ -20,13 +22,82 @@ class DependencyGraph public: static void AddDependency(ConfigObject* child, ConfigObject* parent); static void RemoveDependency(ConfigObject* child, ConfigObject* parent); + static std::vector GetParents(const ConfigObject::Ptr& child); static std::vector GetChildren(const ConfigObject::Ptr& parent); private: DependencyGraph(); + /** + * Represents an undirected dependency edge between two objects. + * + * It allows to traverse the graph in both directions, i.e. from parent to child and vice versa. + */ + struct Edge + { + ConfigObject* parent; // The parent object of the child one. + ConfigObject* child; // The dependent object of the parent. + // Counter for the number of parent <-> child edges to allow duplicates. + int count; + + Edge(ConfigObject* parent, ConfigObject* child, int count = 1): parent(parent), child(child), count(count) + { + } + + struct Hash + { + /** + * Generates a unique hash of the given Edge object. + * + * Note, the hash value is generated only by combining the hash values of the parent and child pointers. + * + * @param edge The Edge object to be hashed. + * + * @return size_t The resulting hash value of the given object. + */ + size_t operator()(const Edge& edge) const + { + size_t seed = 0; + boost::hash_combine(seed, edge.parent); + boost::hash_combine(seed, edge.child); + + return seed; + } + }; + + struct Equal + { + /** + * Compares whether the two Edge objects contain the same parent and child pointers. + * + * Note, the member property count is not taken into account for equality checks. + * + * @param a The first Edge object to compare. + * @param b The second Edge object to compare. + * + * @return bool Returns true if the two objects are equal, false otherwise. + */ + bool operator()(const Edge& a, const Edge& b) const + { + return a.parent == b.parent && a.child == b.child; + } + }; + }; + + using DependencyMap = boost::multi_index_container< + Edge, // The value type we want to sore in the container. + boost::multi_index::indexed_by< + // The first indexer is used for lookups by the Edge from child to parent, thus it + // needs its own hash function and comparison predicate. + boost::multi_index::hashed_unique, Edge::Hash, Edge::Equal>, + // These two indexers are used for lookups by the parent and child pointers. + boost::multi_index::hashed_non_unique>, + boost::multi_index::hashed_non_unique> + > + >; + static std::mutex m_Mutex; - static std::map> m_Dependencies; + static DependencyMap m_Dependencies; }; } From 7501525550aa709789ce3aee01aed88e979b02ea Mon Sep 17 00:00:00 2001 From: Yonas Habteab Date: Mon, 16 Sep 2024 09:20:43 +0200 Subject: [PATCH 4/5] ApiListener: Sync runtime configs in order --- lib/remote/apilistener-configsync.cpp | 60 +++++++++++++++++++++------ lib/remote/apilistener.hpp | 2 + 2 files changed, 49 insertions(+), 13 deletions(-) diff --git a/lib/remote/apilistener-configsync.cpp b/lib/remote/apilistener-configsync.cpp index 04436ad8b99..6d97258ec58 100644 --- a/lib/remote/apilistener-configsync.cpp +++ b/lib/remote/apilistener-configsync.cpp @@ -5,11 +5,13 @@ #include "remote/configobjectutility.hpp" #include "remote/jsonrpc.hpp" #include "base/configtype.hpp" -#include "base/json.hpp" #include "base/convert.hpp" +#include "base/dependencygraph.hpp" +#include "base/json.hpp" #include "config/vmops.hpp" #include "remote/configobjectslock.hpp" #include +#include using namespace icinga; @@ -393,6 +395,40 @@ void ApiListener::UpdateConfigObject(const ConfigObject::Ptr& object, const Mess } } +/** + * Syncs the specified object and its direct and indirect parents to the provided client + * in topological order of their dependency graph recursively. + * + * Objects that the client does not have access to are skipped without going through their dependency graph. + * + * Please do not use this method to forward remote generated cluster updates; it should only be used to + * send local updates to that specific non-nullptr client. + * + * @param object The config object you want to sync. + * @param azone The zone of the client you want to send the update to. + * @param client The JsonRpc client you send the update to. + * @param syncedObjects Used to cache the already synced objects. + */ +void ApiListener::UpdateConfigObjectWithParents(const ConfigObject::Ptr& object, const Zone::Ptr& azone, + const JsonRpcConnection::Ptr& client, std::unordered_set& syncedObjects) +{ + if (syncedObjects.find(object.get()) != syncedObjects.end()) { + return; + } + + /* don't sync objects for non-matching parent-child zones */ + if (!azone->CanAccessObject(object)) { + return; + } + syncedObjects.emplace(object.get()); + + for (const auto& parent : DependencyGraph::GetParents(object)) { + UpdateConfigObjectWithParents(parent, azone, client, syncedObjects); + } + + /* send the config object to the connected client */ + UpdateConfigObject(object, nullptr, client); +} void ApiListener::DeleteConfigObject(const ConfigObject::Ptr& object, const MessageOrigin::Ptr& origin, const JsonRpcConnection::Ptr& client) @@ -454,19 +490,17 @@ void ApiListener::SendRuntimeConfigObjects(const JsonRpcConnection::Ptr& aclient Log(LogInformation, "ApiListener") << "Syncing runtime objects to endpoint '" << endpoint->GetName() << "'."; + std::unordered_set syncedObjects; for (const Type::Ptr& type : Type::GetAllTypes()) { - auto *dtype = dynamic_cast(type.get()); - - if (!dtype) - continue; - - for (const ConfigObject::Ptr& object : dtype->GetObjects()) { - /* don't sync objects for non-matching parent-child zones */ - if (!azone->CanAccessObject(object)) - continue; - - /* send the config object to the connected client */ - UpdateConfigObject(object, nullptr, aclient); + if (auto *ctype = dynamic_cast(type.get())) { + for (const auto& object : ctype->GetObjects()) { + // All objects must be synced sorted by their dependency graph. + // Otherwise, downtimes/comments etc. might get synced before their respective Checkables, which will + // result in comments and downtimes being ignored by the other endpoint since it does not yet know + // about their checkables. Given that the runtime config updates event does not trigger a reload on the + // remote endpoint, these objects won't be synced again until the next reload. + UpdateConfigObjectWithParents(object, azone, aclient, syncedObjects); + } } } diff --git a/lib/remote/apilistener.hpp b/lib/remote/apilistener.hpp index fced0a8afb1..eae1fa03e61 100644 --- a/lib/remote/apilistener.hpp +++ b/lib/remote/apilistener.hpp @@ -247,6 +247,8 @@ class ApiListener final : public ObjectImpl /* configsync */ void UpdateConfigObject(const ConfigObject::Ptr& object, const MessageOrigin::Ptr& origin, const JsonRpcConnection::Ptr& client = nullptr); + void UpdateConfigObjectWithParents(const ConfigObject::Ptr& object, const Zone::Ptr& azone, + const JsonRpcConnection::Ptr& client, std::unordered_set& syncedObjects); void DeleteConfigObject(const ConfigObject::Ptr& object, const MessageOrigin::Ptr& origin, const JsonRpcConnection::Ptr& client = nullptr); void SendRuntimeConfigObjects(const JsonRpcConnection::Ptr& aclient); From 156ba265e122624edbe3a176effc6f2038bd87f1 Mon Sep 17 00:00:00 2001 From: Julian Brost Date: Thu, 19 Dec 2024 12:24:26 +0100 Subject: [PATCH 5/5] Simplify `DependencyGraph:RemoveDependency()` method --- lib/base/dependencygraph.cpp | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/lib/base/dependencygraph.cpp b/lib/base/dependencygraph.cpp index a9d0dbb95cd..06446cfc281 100644 --- a/lib/base/dependencygraph.cpp +++ b/lib/base/dependencygraph.cpp @@ -20,15 +20,14 @@ void DependencyGraph::RemoveDependency(ConfigObject* child, ConfigObject* parent std::unique_lock lock(m_Mutex); if (auto it(m_Dependencies.find(Edge(parent, child))); it != m_Dependencies.end()) { - // A number <= 1 means, this isn't referenced by anyone and should be erased from the container. - if (it->count == 1) { + if (it->count > 1) { + // Remove a duplicate edge from child to node, i.e. decrement the corresponding counter. + m_Dependencies.modify(it, [](Edge& e) { e.count--; }); + } else { + // Remove the last edge from child to node (decrementing the counter would set it to 0), + // thus remove that connection from the data structure completely. m_Dependencies.erase(it); - return; } - - // Otherwise, each remove operation will should decrement this by 1 till it reaches <= 1 - // and causes the edge to completely be erased from the container. - m_Dependencies.modify(it, [](Edge& e) { e.count--; }); } }