diff --git a/BaseLib/ConfigTreeNew-impl.h b/BaseLib/ConfigTreeNew-impl.h index 41ea3f9dc7c2509fa7fb2036f97a63aebe3b302d..efc51517a300fa8608d42888f0d6b970f189480c 100644 --- a/BaseLib/ConfigTreeNew-impl.h +++ b/BaseLib/ConfigTreeNew-impl.h @@ -71,7 +71,7 @@ ConfigTreeNew:: getConfParamList(std::string const& param) const { checkUnique(param); - markVisited<T>(param, true); + markVisited<T>(param, false, true); auto p = _tree->equal_range(param); return Range<ValueIterator<T> >( @@ -144,7 +144,7 @@ ConfigTreeNew:: getAttribute(std::string const& attr) const { checkUniqueAttr(attr); - markVisited(attr, true); + markVisited<T>(attr, true); if (auto attrs = _tree->get_child_optional("<xmlattr>")) { if (auto a = attrs->get_child_optional(attr)) { @@ -166,11 +166,13 @@ getAttribute(std::string const& attr) const template<typename T> ConfigTreeNew::CountType& ConfigTreeNew:: -markVisited(std::string const& key, bool const peek_only) const +markVisited(std::string const& key, bool const is_attr, + bool const peek_only) const { auto const type = std::type_index(typeid(T)); - auto p = _visited_params.emplace(key, CountType{peek_only ? 0 : 1, type}); + auto p = _visited_params.emplace(std::make_pair(is_attr, key), + CountType{peek_only ? 0 : 1, type}); if (!p.second) { // no insertion happened auto& v = p.first->second; diff --git a/BaseLib/ConfigTreeNew.cpp b/BaseLib/ConfigTreeNew.cpp index c5da6e1606677d6a8e7f97f661f97297de6facef..15fbbf0ab724e958d30548eb8598d75086789175 100644 --- a/BaseLib/ConfigTreeNew.cpp +++ b/BaseLib/ConfigTreeNew.cpp @@ -211,7 +211,7 @@ void ConfigTreeNew::checkUnique(const std::string &key) const { checkKeyname(key); - if (_visited_params.find(key) != _visited_params.end()) { + if (_visited_params.find({false, key}) != _visited_params.end()) { error("Key <" + key + "> has already been processed."); } } @@ -220,7 +220,7 @@ void ConfigTreeNew::checkUniqueAttr(const std::string &attr) const { checkKeyname(attr); - if (_visited_params.find("<xmlattr>." + attr) != _visited_params.end()) { + if (_visited_params.find({true, attr}) != _visited_params.end()) { error("Attribute \"" + attr + "\" has already been processed."); } } @@ -229,16 +229,17 @@ ConfigTreeNew::CountType& ConfigTreeNew:: markVisited(std::string const& key, bool const peek_only) const { - return markVisited<ConfigTreeNew>(key, peek_only); + return markVisited<ConfigTreeNew>(key, false, peek_only); } void ConfigTreeNew:: -markVisitedDecrement(std::string const& key) const +markVisitedDecrement(bool const is_attr, std::string const& key) const { auto const type = std::type_index(typeid(nullptr)); - auto p = _visited_params.emplace(key, CountType{-1, type}); + auto p = _visited_params.emplace(std::make_pair(is_attr, key), + CountType{-1, type}); if (!p.second) { // no insertion happened auto& v = p.first->second; @@ -276,24 +277,40 @@ ConfigTreeNew::checkAndInvalidate() for (auto const& p : *_tree) { DBUG("-- %s <%s> ", _path.c_str(), p.first.c_str()); if (p.first != "<xmlattr>") // attributes are handled below - markVisitedDecrement(p.first); + markVisitedDecrement(false, p.first); + DBUG("tag %s", p.first.c_str()); } // iterate over attributes if (auto attrs = _tree->get_child_optional("<xmlattr>")) { + DBUG("has attributes"); for (auto const& p : *attrs) { - markVisitedDecrement("<xmlattr>." + p.first); - // markVisitedDecrement("<xmlattr>"); + markVisitedDecrement(true, p.first); + DBUG("attr %s", p.first.c_str()); } } - for (auto const& p : _visited_params) { - if (p.second.count > 0) { - warning("Key <" + p.first + "> has been read " + std::to_string(p.second.count) - + " time(s) more than it was present in the configuration tree."); - } else if (p.second.count < 0) { - warning("Key <" + p.first + "> has been read " + std::to_string(-p.second.count) - + " time(s) less than it was present in the configuration tree."); + for (auto const& p : _visited_params) + { + auto const& tag = std::get<1>(p.first); + auto const& count = p.second.count; + + if (std::get<0>(p.first)) { // tag + if (count > 0) { + warning("XML attribute \"" + tag + "\" has been read " + std::to_string(count) + + " time(s) more than it was present in the configuration tree."); + } else if (count < 0) { + warning("XML attribute \"" + tag + "\" has been read " + std::to_string(-count) + + " time(s) less than it was present in the configuration tree."); + } + } else { // XML attribute + if (count > 0) { + warning("Key <" + tag + "> has been read " + std::to_string(count) + + " time(s) more than it was present in the configuration tree."); + } else if (count < 0) { + warning("Key <" + tag + "> has been read " + std::to_string(-count) + + " time(s) less than it was present in the configuration tree."); + } } } diff --git a/BaseLib/ConfigTreeNew.h b/BaseLib/ConfigTreeNew.h index 1cc7a6ae835ebf8b0b87b12a33dcadfc12737f55..8f9879ebca05d4a314c6c8074599764175b2b78e 100644 --- a/BaseLib/ConfigTreeNew.h +++ b/BaseLib/ConfigTreeNew.h @@ -396,7 +396,8 @@ private: * \c param peek_only if true, do not change the read-count of the given key. */ template<typename T> - CountType& markVisited(std::string const& key, bool peek_only = false) const; + CountType& markVisited(std::string const& key, bool const is_attr, + bool peek_only = false) const; /*! Keeps track of the key \c key and its value type ConfigTree. * @@ -408,7 +409,7 @@ private: //! Used in the destructor to compute the difference between number of reads of a parameter //! and the number of times it exists in the ConfigTree - void markVisitedDecrement(std::string const& key) const; + void markVisitedDecrement(bool const is_attr, std::string const& key) const; //! TODO doc bool hasChildren(ConfigTreeNew const& ct) const; @@ -431,6 +432,8 @@ private: //! \todo add file name + using KeyType = std::pair<bool, std::string>; + //! A map key -> (count, type) keeping track which parameters have been read how often //! and which datatype they have. //! @@ -438,7 +441,7 @@ private: //! Therefore it has to be mutable in order to be able to read from //! constant instances, e.g., those passed as constant references to //! temporaries. - mutable std::map<std::string, CountType> _visited_params; + mutable std::map<KeyType, CountType> _visited_params; //! \todo doc mutable bool _have_read_data = false;