From deefd078bbcd8b68f3c595c372d6f8ca896b0d47 Mon Sep 17 00:00:00 2001 From: Nathan Hughes Date: Wed, 13 Nov 2024 21:32:53 +0000 Subject: [PATCH 1/4] avoid log errors when type is missing but config is initialized --- config_utilities/include/config_utilities/virtual_config.h | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/config_utilities/include/config_utilities/virtual_config.h b/config_utilities/include/config_utilities/virtual_config.h index 86c0153..e24e869 100644 --- a/config_utilities/include/config_utilities/virtual_config.h +++ b/config_utilities/include/config_utilities/virtual_config.h @@ -206,13 +206,14 @@ void declare_config(VirtualConfig& config) { auto data = internal::Visitor::visitVirtualConfig(config.isSet(), config.optional_, config.getType()); // If setting values create the wrapped config using the string identifier. + const bool type_optional = config.optional_ || config.config_; if (data) { std::string type; - const bool success = config.optional_ ? internal::getTypeImpl(*data, type, Settings().factory_type_param_name) - : internal::getType(*data, type); + const bool success = type_optional ? internal::getTypeImpl(*data, type, Settings().factory_type_param_name) + : internal::getType(*data, type); if (success) { config.config_ = internal::ConfigFactory::create(type); - } else if (!config.optional_) { + } else if (!type_optional) { std::stringstream ss; ss << "Could not get type for '" << internal::ModuleInfo::fromTypes().typeInfo() << "'"; internal::Logger::logError(ss.str()); From 6bc3e7e628de0e30a38fba3aec45b96b3cdccf0a Mon Sep 17 00:00:00 2001 From: Nathan Hughes Date: Thu, 14 Nov 2024 14:51:34 +0000 Subject: [PATCH 2/4] simplify logic and add comments --- .../include/config_utilities/factory.h | 13 ++++++++---- .../include/config_utilities/virtual_config.h | 11 +++++----- config_utilities/src/factory.cpp | 20 +++++++++++-------- 3 files changed, 27 insertions(+), 17 deletions(-) diff --git a/config_utilities/include/config_utilities/factory.h b/config_utilities/include/config_utilities/factory.h index 0974256..bcb47bf 100644 --- a/config_utilities/include/config_utilities/factory.h +++ b/config_utilities/include/config_utilities/factory.h @@ -61,11 +61,16 @@ std::vector convertArguments() { }; } -//! @brief Helper function to read the type param from a node. -bool getTypeImpl(const YAML::Node& data, std::string& type, const std::string& param_name); +/** @brief Helper function to read the type param from a node. + * @param data YAML node to read type from + * @param type Type value to filll + * @param required Whether or not the type field is required + * @param param_name Field in YAML node to read (empty string defaults to Settings().factory_type_param_name) + */ +bool getType(const YAML::Node& data, std::string& type, bool required = true, const std::string& param_name = ""); -//! @brief Get type from YAML node directly -bool getType(const YAML::Node& data, std::string& type); +/*//! @brief Get type from YAML node directly*/ +/*bool getType(const YAML::Node& data, std::string& type);*/ //! @brief Struct recording typenames for a module (i.e., the constructor signature). Can be used as a map key struct ModuleInfo { diff --git a/config_utilities/include/config_utilities/virtual_config.h b/config_utilities/include/config_utilities/virtual_config.h index e24e869..d27e4ef 100644 --- a/config_utilities/include/config_utilities/virtual_config.h +++ b/config_utilities/include/config_utilities/virtual_config.h @@ -205,15 +205,16 @@ template void declare_config(VirtualConfig& config) { auto data = internal::Visitor::visitVirtualConfig(config.isSet(), config.optional_, config.getType()); + // underlying derived type is not required if the config is optional, or if the config has been + // initialized to a derived type already (i.e., config_ is already populated) + const bool type_required = !config.optional_ && !config.config_; + // If setting values create the wrapped config using the string identifier. - const bool type_optional = config.optional_ || config.config_; if (data) { std::string type; - const bool success = type_optional ? internal::getTypeImpl(*data, type, Settings().factory_type_param_name) - : internal::getType(*data, type); - if (success) { + if (internal::getType(*data, type, type_required)) { config.config_ = internal::ConfigFactory::create(type); - } else if (!type_optional) { + } else if (type_required) { std::stringstream ss; ss << "Could not get type for '" << internal::ModuleInfo::fromTypes().typeInfo() << "'"; internal::Logger::logError(ss.str()); diff --git a/config_utilities/src/factory.cpp b/config_utilities/src/factory.cpp index f1e247e..5a32beb 100644 --- a/config_utilities/src/factory.cpp +++ b/config_utilities/src/factory.cpp @@ -213,28 +213,32 @@ ModuleRegistry& ModuleRegistry::instance() { } // Helper function to read the type param from a node. -bool getTypeImpl(const YAML::Node& data, std::string& type, const std::string& param_name) { +bool getTypeImpl(const YAML::Node& data, std::string& type, bool required, const std::string& key) { if (!data.IsMap()) { return false; } - if (!data[param_name]) { + + // Get the type or print an error. + if (!data[key]) { return false; } + try { - type = data[param_name].as(); + type = data[key].as(); } catch (const YAML::Exception& e) { return false; } + return true; } -bool getType(const YAML::Node& data, std::string& type) { - // Get the type or print an error. - const std::string param_name = Settings::instance().factory_type_param_name; - if (!getTypeImpl(data, type, param_name)) { - Logger::logError("Could not read the param '" + param_name + "' to deduce the type of the module to create."); +bool getType(const YAML::Node& data, std::string& type, bool required, const std::string& param_name) { + const std::string key = param_name.empty() ? Settings::instance().factory_type_param_name : param_name; + if (!getTypeImpl(data, type, required, key)) { + Logger::logError("Could not read the param '" + key + "' to deduce the type of the module to create."); return false; } + return true; } From 65bc4e3250a3161810c14ff15a9d82e9242f173f Mon Sep 17 00:00:00 2001 From: Nathan Hughes Date: Thu, 14 Nov 2024 16:06:25 +0000 Subject: [PATCH 3/4] drop commented code --- config_utilities/include/config_utilities/factory.h | 3 --- 1 file changed, 3 deletions(-) diff --git a/config_utilities/include/config_utilities/factory.h b/config_utilities/include/config_utilities/factory.h index bcb47bf..7c03122 100644 --- a/config_utilities/include/config_utilities/factory.h +++ b/config_utilities/include/config_utilities/factory.h @@ -69,9 +69,6 @@ std::vector convertArguments() { */ bool getType(const YAML::Node& data, std::string& type, bool required = true, const std::string& param_name = ""); -/*//! @brief Get type from YAML node directly*/ -/*bool getType(const YAML::Node& data, std::string& type);*/ - //! @brief Struct recording typenames for a module (i.e., the constructor signature). Can be used as a map key struct ModuleInfo { template From 9fb700f92a00961cd127aa2bfa0fa8f87375d9b0 Mon Sep 17 00:00:00 2001 From: Nathan Hughes Date: Thu, 14 Nov 2024 17:55:57 +0000 Subject: [PATCH 4/4] get required logic correct --- config_utilities/src/factory.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/config_utilities/src/factory.cpp b/config_utilities/src/factory.cpp index 5a32beb..44a97d2 100644 --- a/config_utilities/src/factory.cpp +++ b/config_utilities/src/factory.cpp @@ -213,7 +213,7 @@ ModuleRegistry& ModuleRegistry::instance() { } // Helper function to read the type param from a node. -bool getTypeImpl(const YAML::Node& data, std::string& type, bool required, const std::string& key) { +bool getTypeImpl(const YAML::Node& data, std::string& type, const std::string& key) { if (!data.IsMap()) { return false; } @@ -234,12 +234,12 @@ bool getTypeImpl(const YAML::Node& data, std::string& type, bool required, const bool getType(const YAML::Node& data, std::string& type, bool required, const std::string& param_name) { const std::string key = param_name.empty() ? Settings::instance().factory_type_param_name : param_name; - if (!getTypeImpl(data, type, required, key)) { + const auto success = getTypeImpl(data, type, key); + if (!success && required) { Logger::logError("Could not read the param '" + key + "' to deduce the type of the module to create."); - return false; } - return true; + return success; } } // namespace config::internal