From c7220cb51ad5dc1aee133afd2b8cec80bb11ae3e Mon Sep 17 00:00:00 2001 From: John Turner <7strbass@gmail.com> Date: Fri, 13 Sep 2024 11:07:25 -0400 Subject: [PATCH] --[BE] - Add ability to filter Configuration subconfigs (#2471) * --add subconfig filtering process. This function will remove any values and subconfigs from a Configuration that match those found within a passed Configuration. * --minor naming clarification * --expand Configuration tests to test filtering; rename test Test only tests Configurations, so name appropriately --- src/esp/bindings/ConfigBindings.cpp | 2 +- src/esp/core/Configuration.cpp | 46 +++ src/esp/core/Configuration.h | 37 +- src/tests/CMakeLists.txt | 2 +- src/tests/ConfigurationTest.cpp | 556 ++++++++++++++++++++++++++++ src/tests/CoreTest.cpp | 334 ----------------- 6 files changed, 621 insertions(+), 356 deletions(-) create mode 100644 src/tests/ConfigurationTest.cpp delete mode 100644 src/tests/CoreTest.cpp diff --git a/src/esp/bindings/ConfigBindings.cpp b/src/esp/bindings/ConfigBindings.cpp index 78425448e8..c2a4db78f0 100644 --- a/src/esp/bindings/ConfigBindings.cpp +++ b/src/esp/bindings/ConfigBindings.cpp @@ -152,7 +152,7 @@ void initConfigBindings(py::module& m) { R"(Returns whether or not this Configuration has the passed key. Does not check subconfigurations.)", "key"_a) .def( - "has_key_to_type", &Configuration::hasKeyOfType, + "has_key_to_type", &Configuration::hasKeyToValOfType, R"(Returns whether passed key points to a value of specified ConfigValType)", "key"_a, "value_type"_a) .def( diff --git a/src/esp/core/Configuration.cpp b/src/esp/core/Configuration.cpp index b95b1f735d..a6d8030d67 100644 --- a/src/esp/core/Configuration.cpp +++ b/src/esp/core/Configuration.cpp @@ -783,6 +783,52 @@ std::vector Configuration::findValue( return breadcrumbs; } +void Configuration::overwriteWithConfig( + const std::shared_ptr& src) { + if (src->getNumEntries() == 0) { + return; + } + // copy every element over from src + for (const auto& elem : src->valueMap_) { + valueMap_[elem.first] = elem.second; + } + // merge subconfigs + for (const auto& subConfig : src->configMap_) { + const auto name = subConfig.first; + // make if DNE and merge src subconfig + addOrEditSubgroup(name).first->second->overwriteWithConfig( + subConfig.second); + } +} // Configuration::overwriteWithConfig + +void Configuration::filterFromConfig( + const std::shared_ptr& src) { + if (src->getNumEntries() == 0) { + return; + } + // filter out every element that is present with the same value in both src + // and this. + for (const auto& elem : src->valueMap_) { + ValueMapType::const_iterator mapIter = valueMap_.find(elem.first); + // if present and has the same data, erase this configuration's data + if ((mapIter != valueMap_.end()) && (mapIter->second == elem.second)) { + valueMap_.erase(mapIter); + } + } + // repeat process on all subconfigs of src that are present in this. + for (const auto& subConfig : src->configMap_) { + // find if this has subconfig of same name + ConfigMapType::iterator mapIter = configMap_.find(subConfig.first); + if (mapIter != configMap_.end()) { + mapIter->second->filterFromConfig(subConfig.second); + // remove the subconfig if it has no entries after filtering + if (mapIter->second->getNumEntries() == 0) { + configMap_.erase(mapIter); + } + } + } +} // Configuration::filterFromConfig + Configuration& Configuration::operator=(const Configuration& otr) { if (this != &otr) { configMap_.clear(); diff --git a/src/esp/core/Configuration.h b/src/esp/core/Configuration.h index e5234109a2..3aa1c61478 100644 --- a/src/esp/core/Configuration.h +++ b/src/esp/core/Configuration.h @@ -697,7 +697,7 @@ class Configuration { : configMap_(std::move(otr.configMap_)), valueMap_(std::move(otr.valueMap_)) {} // move ctor - // virtual destructor set to that pybind11 recognizes attributes inheritance + // virtual destructor set so that pybind11 recognizes attributes inheritance // from Configuration to be polymorphic virtual ~Configuration() = default; @@ -1163,7 +1163,8 @@ class Configuration { * @param desiredType the @ref ConfigValType to compare the value's type to * @return Whether @p key references a value that is of @p desiredType. */ - bool hasKeyOfType(const std::string& key, ConfigValType desiredType) { + bool hasKeyToValOfType(const std::string& key, + ConfigValType desiredType) const { ValueMapType::const_iterator mapIter = valueMap_.find(key); return (mapIter != valueMap_.end() && (mapIter->second.getType() == desiredType)); @@ -1247,7 +1248,6 @@ class Configuration { * @return A pointer to a copy of the Configuration having the requested * name, cast to the appropriate type, or nullptr if not found. */ - template std::shared_ptr getSubconfigCopy(const std::string& cfgName) const { static_assert(std::is_base_of::value, @@ -1371,27 +1371,24 @@ class Configuration { /** * @brief Merges Configuration pointed to by @p src into this - * Configuration, including all subconfigs. Passed config overwrites + * Configuration, including all subconfigs. Passed config overwrites * existing data in this config. * @param src The source of Configuration data we wish to merge into this * Configuration. */ - void overwriteWithConfig(const std::shared_ptr& src) { - if (src->getNumEntries() == 0) { - return; - } - // copy every element over from src - for (const auto& elem : src->valueMap_) { - valueMap_[elem.first] = elem.second; - } - // merge subconfigs - for (const auto& subConfig : src->configMap_) { - const auto name = subConfig.first; - // make if DNE and merge src subconfig - addOrEditSubgroup(name).first->second->overwriteWithConfig( - subConfig.second); - } - } + void overwriteWithConfig(const std::shared_ptr& src); + + /** + * @brief Performs the opposite operation to @ref Configuration::overwriteWithConfig. + * All values and subconfigs in the passed Configuration will be removed from + * this config unless the data they hold is different. Any empty subconfigs + * will be removed as well. + * + * @param src The source of Configuration data we wish to prune from this + * Configuration. + */ + + void filterFromConfig(const std::shared_ptr& src); /** * @brief Returns a const iterator across the map of values. diff --git a/src/tests/CMakeLists.txt b/src/tests/CMakeLists.txt index d276dee3af..7d7c7ae766 100644 --- a/src/tests/CMakeLists.txt +++ b/src/tests/CMakeLists.txt @@ -74,7 +74,7 @@ if(MagnumPlugins_KtxImageConverter_FOUND) target_link_libraries(GfxBatchRendererTest PRIVATE MagnumPlugins::KtxImageConverter) endif() -corrade_add_test(CoreTest CoreTest.cpp LIBRARIES core io) +corrade_add_test(ConfigurationTest ConfigurationTest.cpp LIBRARIES core io) corrade_add_test(CullingTest CullingTest.cpp LIBRARIES gfx) target_include_directories(CullingTest PRIVATE ${CMAKE_CURRENT_BINARY_DIR}) diff --git a/src/tests/ConfigurationTest.cpp b/src/tests/ConfigurationTest.cpp new file mode 100644 index 0000000000..35b66d7f6b --- /dev/null +++ b/src/tests/ConfigurationTest.cpp @@ -0,0 +1,556 @@ +// Copyright (c) Meta Platforms, Inc. and its affiliates. +// This source code is licensed under the MIT license found in the +// LICENSE file in the root directory of this source tree. + +#include +#include +#include "esp/core/Configuration.h" +#include "esp/core/Esp.h" + +using namespace esp::core::config; +namespace Cr = Corrade; + +namespace { + +struct ConfigurationTest : Cr::TestSuite::Tester { + explicit ConfigurationTest(); + + /** + * @brief A Configuration is a hierarchical storage structure. This + * recursively builds a subconfiguration hierarchy within the passed @p config + * using the given values. + * @param countPerDepth Number of subconfigs to build for each depth layer. + * @param curDepth Current recursion depth. + * @param totalDepth Total recursion depth +1 == total subconfig hierarchy + * depth of given @p config counting base layer (add 1 because head + * recursion). + * @param config The Configuration in which to build the subconfig hierarchy. + */ + void buildSubconfigsInConfig(int countPerDepth, + int curDepth, + int totalDepth, + Configuration::ptr& config); + + /** + * @brief A Configuration is a hierarchical storage structure. This + * recursively verifies the passed @p config has the subconfig hierarchy + * structure as described by the given values. + * @param countPerDepth Number of subconfigs that should be preseent in each + * depth layer. + * @param curDepth Current recursion depth. + * @param totalDepth Total recursion depth +1 == total subconfig hierarchy + * depth of given @p config counting base layer (add 1 because head + * recursion). + * @param config The Configuration holding the subconfig hierarchy we are + * trying to verify. + */ + void verifySubconfigTree(int countPerDepth, + int curDepth, + int totalDepth, + Configuration::cptr& config) const; + + /** + * @brief Recursively add or modify the string value of the given key in every + * subconfig of @p config with the given value + */ + void addOrModSubconfigTreeVals(const std::string& key, + const std::string& val, + Configuration::ptr& config); + + /** + * @brief Recursively verify that the given key exists in every subconfig of + * @p config with the appropriately modified version of the given value. + */ + void verifySubconfigTreeVals(const std::string& key, + const std::string& val, + Configuration::cptr& config) const; + + /** + * @brief Verifies that the two passed Configurations hold identical + * subconfig hierarchies for both values and structure. + */ + void compareSubconfigs(Configuration::cptr& src, Configuration::cptr& target); + + /** + * @brief Basic test of Configuration functionality + */ + void TestConfiguration(); + + /** + * @brief Test Configuration find capability. Find returns a list of + * subconfiguration keys required to find a particular key + */ + void TestSubconfigFind(); + + /** + * @brief Test Configuration find, merge and edit capability. Builds + * subconfig hierarchy, finds a desired subconfig, duplicates and merges it + * with another and verifies results. + */ + void TestSubconfigFindAndMerge(); + + /** + * @brief Test Configuration subconfig filtering. Builds two identical + * subconfigs, modifies one such that it has changed values and also added + * values, then filters it using original copy and verifies that only + * new/different values remains. + */ + void TestSubconfigFilter(); + + esp::logging::LoggingContext loggingContext_; +}; // struct ConfigurationTest + +ConfigurationTest::ConfigurationTest() { + addTests({ + &ConfigurationTest::TestConfiguration, + &ConfigurationTest::TestSubconfigFind, + &ConfigurationTest::TestSubconfigFindAndMerge, + &ConfigurationTest::TestSubconfigFilter, + }); +} + +void ConfigurationTest::buildSubconfigsInConfig(int countPerDepth, + int curDepth, + int totalDepth, + Configuration::ptr& config) { + if (curDepth == totalDepth) { + return; + } + std::string breadcrumb = config->get("breadcrumb"); + for (int i = 0; i < countPerDepth; ++i) { + const std::string subconfigKey = Cr::Utility::formatString( + "breadcrumb_{}_depth_{}_subconfig_{}", breadcrumb, curDepth, i); + Configuration::ptr newCfg = + config->editSubconfig(subconfigKey); + // set the values within the new subconfig that describe the subconfig's + // status + newCfg->set("depth", curDepth); + newCfg->set("iter", i); + newCfg->set("breadcrumb", Cr::Utility::formatString("{}{}", breadcrumb, i)); + newCfg->set("key", subconfigKey); + + buildSubconfigsInConfig(countPerDepth, curDepth + 1, totalDepth, newCfg); + } +} // ConfigurationTest::buildSubconfigsInConfig + +void ConfigurationTest::verifySubconfigTree(int countPerDepth, + int curDepth, + int totalDepth, + Configuration::cptr& config) const { + if (curDepth == totalDepth) { + return; + } + std::string breadcrumb = config->get("breadcrumb"); + for (int i = 0; i < countPerDepth; ++i) { + const std::string subconfigKey = Cr::Utility::formatString( + "breadcrumb_{}_depth_{}_subconfig_{}", breadcrumb, curDepth, i); + // Verify subconfig with key exists + CORRADE_VERIFY(config->hasSubconfig(subconfigKey)); + Configuration::cptr newCfg = config->getSubconfigView(subconfigKey); + CORRADE_VERIFY(newCfg); + // Verify subconfig depth value is as expected + CORRADE_COMPARE(newCfg->get("depth"), curDepth); + // Verify iteration and parent iteration + CORRADE_COMPARE(newCfg->get("iter"), i); + CORRADE_COMPARE(newCfg->get("breadcrumb"), + Cr::Utility::formatString("{}{}", breadcrumb, i)); + CORRADE_COMPARE(newCfg->get("key"), subconfigKey); + // Check into tree + verifySubconfigTree(countPerDepth, curDepth + 1, totalDepth, newCfg); + } + CORRADE_COMPARE(config->getNumSubconfigs(), countPerDepth); + +} // ConfigurationTest::verifySubconfigTree + +void ConfigurationTest::compareSubconfigs(Configuration::cptr& src, + Configuration::cptr& target) { + // verify target has at least as many subconfigs that src has + CORRADE_VERIFY(src->getNumSubconfigs() <= target->getNumSubconfigs()); + // verify that target has all the values that src has, and they are equal. + auto srcIterValPair = src->getValuesIterator(); + for (auto& cfgIter = srcIterValPair.first; cfgIter != srcIterValPair.second; + ++cfgIter) { + CORRADE_VERIFY(cfgIter->second == target->get(cfgIter->first)); + } + + // Verify all subconfigs have the same keys - get begin/end iterators for src + // subconfigs + auto srcIterConfigPair = src->getSubconfigIterator(); + for (auto& cfgIter = srcIterConfigPair.first; + cfgIter != srcIterConfigPair.second; ++cfgIter) { + CORRADE_VERIFY(target->hasSubconfig(cfgIter->first)); + Configuration::cptr srcSubConfig = cfgIter->second; + Configuration::cptr tarSubConfig = target->getSubconfigView(cfgIter->first); + compareSubconfigs(srcSubConfig, tarSubConfig); + } + +} // ConfigurationTest::compareSubconfigs + +void ConfigurationTest::addOrModSubconfigTreeVals(const std::string& key, + const std::string& val, + Configuration::ptr& config) { + std::string newVal = val; + if (config->hasKeyToValOfType(key, + esp::core::config::ConfigValType::String)) { + newVal = + Cr::Utility::formatString("{}_{}", config->get(key), val); + } + config->set(key, newVal); + auto cfgIterPair = config->getSubconfigIterator(); + // Get subconfig keys + const auto& subsetKeys = config->getSubconfigKeys(); + for (const auto& subKey : subsetKeys) { + auto subconfig = config->editSubconfig(subKey); + addOrModSubconfigTreeVals(key, val, subconfig); + } +} // ConfigurationTest::addOrModSubconfigTreeVals + +void ConfigurationTest::verifySubconfigTreeVals( + const std::string& key, + const std::string& val, + Configuration::cptr& config) const { + CORRADE_VERIFY( + config->hasKeyToValOfType(key, esp::core::config::ConfigValType::String)); + std::string testVal = config->get(key); + std::size_t foundLoc = testVal.find(val); + CORRADE_VERIFY(foundLoc != std::string::npos); + auto srcIterConfigPair = config->getSubconfigIterator(); + for (auto& cfgIter = srcIterConfigPair.first; + cfgIter != srcIterConfigPair.second; ++cfgIter) { + Configuration::cptr subCfg = cfgIter->second; + verifySubconfigTreeVals(key, val, subCfg); + } +} // ConfigurationTest::verifySubconfigTreeVals + +void ConfigurationTest::TestConfiguration() { + Configuration cfg; + cfg.set("myBool", true); + cfg.set("myInt", 10); + cfg.set("myFloatToDouble", 1.2f); + cfg.set("myVec2", Mn::Vector2{1.0, 2.0}); + cfg.set("myVec3", Mn::Vector3{1.0, 2.0, 3.0}); + cfg.set("myVec4", Mn::Vector4{1.0, 2.0, 3.0, 4.0}); + cfg.set("myQuat", Mn::Quaternion{{1.0, 2.0, 3.0}, 0.1}); + cfg.set("myMat3", Mn::Matrix3(Mn::Math::IdentityInit)); + cfg.set("myMat4", Mn::Matrix4(Mn::Math::IdentityInit)); + cfg.set("myRad", Mn::Rad{1.23}); + cfg.set("myString", "test"); + + CORRADE_VERIFY(cfg.hasValue("myBool")); + CORRADE_VERIFY(cfg.hasValue("myInt")); + CORRADE_VERIFY(cfg.hasValue("myFloatToDouble")); + CORRADE_VERIFY(cfg.hasValue("myVec2")); + CORRADE_VERIFY(cfg.hasValue("myVec3")); + CORRADE_VERIFY(cfg.hasValue("myMat3")); + CORRADE_VERIFY(cfg.hasValue("myVec4")); + CORRADE_VERIFY(cfg.hasValue("myQuat")); + CORRADE_VERIFY(cfg.hasValue("myMat3")); + CORRADE_VERIFY(cfg.hasValue("myRad")); + CORRADE_VERIFY(cfg.hasValue("myString")); + + CORRADE_COMPARE(cfg.get("myBool"), true); + CORRADE_COMPARE(cfg.get("myInt"), 10); + CORRADE_COMPARE(cfg.get("myFloatToDouble"), 1.2f); + CORRADE_COMPARE(cfg.get("myVec2"), Mn::Vector2(1.0, 2.0)); + CORRADE_COMPARE(cfg.get("myVec3"), Mn::Vector3(1.0, 2.0, 3.0)); + CORRADE_COMPARE(cfg.get("myVec4"), + Mn::Vector4(1.0, 2.0, 3.0, 4.0)); + CORRADE_COMPARE(cfg.get("myQuat"), + Mn::Quaternion({1.0, 2.0, 3.0}, 0.1)); + CORRADE_COMPARE(cfg.get("myRad"), Mn::Rad(1.23)); + + for (int i = 0; i < 3; ++i) { + CORRADE_COMPARE(cfg.get("myMat3").row(i)[i], 1); + } + for (int i = 0; i < 4; ++i) { + CORRADE_COMPARE(cfg.get("myMat4").row(i)[i], 1); + } + + CORRADE_COMPARE(cfg.get("myString"), "test"); +} // ConfigurationTest::TestConfiguration test + +// Test configuration find functionality +void ConfigurationTest::TestSubconfigFind() { + Configuration::ptr cfg = Configuration::create(); + Configuration::ptr baseCfg = cfg; + // Build layers of subconfigs + for (int i = 0; i < 10; ++i) { + Configuration::ptr newCfg = cfg->editSubconfig( + Cr::Utility::formatString("depth_{}_subconfig_{}", i + 1, i)); + newCfg->set("depth", i + 1); + cfg = newCfg; + } + // Set deepest layer config to have treasure + cfg->set("treasure", "this is the treasure!"); + // Find the treasure! + std::vector keyList = baseCfg->findValue("treasure"); + + // Display breadcrumbs + std::string resStr = Cr::Utility::formatString( + "Vector of 'treasure' keys is size : {}", keyList.size()); + for (const auto& key : keyList) { + Cr::Utility::formatInto(resStr, resStr.size(), "\n\t{}", key); + } + ESP_DEBUG() << resStr; + // Verify the found treasure + CORRADE_COMPARE(keyList.size(), 11); + + Configuration::cptr viewConfig = baseCfg; + for (int i = 0; i < 10; ++i) { + const std::string subconfigKey = + Cr::Utility::formatString("depth_{}_subconfig_{}", i + 1, i); + CORRADE_COMPARE(keyList[i], subconfigKey); + // Verify that subconfig with given name exists + CORRADE_VERIFY(viewConfig->hasSubconfig(subconfigKey)); + // retrieve actual subconfig + Configuration::cptr newCfg = viewConfig->getSubconfigView(subconfigKey); + // verity subconfig has correct depth value + CORRADE_COMPARE(newCfg->get("depth"), i + 1); + viewConfig = newCfg; + } + CORRADE_VERIFY(viewConfig->hasValue("treasure")); + CORRADE_COMPARE(viewConfig->get("treasure"), + "this is the treasure!"); + + // Verify pirate isn't found + std::vector notFoundKeyList = baseCfg->findValue("pirate"); + CORRADE_COMPARE(notFoundKeyList.size(), 0); +} + +// Test subconfig edit/merge and save +void ConfigurationTest::TestSubconfigFindAndMerge() { + Configuration::ptr cfg = Configuration::create(); + CORRADE_VERIFY(cfg); + // Set base value for parent iteration + cfg->set("breadcrumb", ""); + int depthPlus1 = 5; + int count = 3; + // Build subconfig tree 4 levels deep, 3 subconfigs per config + buildSubconfigsInConfig(count, 0, depthPlus1, cfg); + Configuration::cptr const_cfg = + std::const_pointer_cast(cfg); + // Verify subconfig tree structure using const views + verifySubconfigTree(count, 0, depthPlus1, const_cfg); + + // grab a subconfig, edit it and save it with a different key + // use find to get key path + const std::string keyToFind = "breadcrumb_212_depth_3_subconfig_2"; + // Find breadcrumb path of keys into subconfigs that lead to keyToFind + std::vector subconfigKeyPath = cfg->findValue(keyToFind); + // Display breadcrumbs + std::string resStr = Cr::Utility::formatString( + "Vector of desired subconfig keys to get to '{}' " + "subconfig is size : {}", + keyToFind, subconfigKeyPath.size()); + for (const auto& key : subconfigKeyPath) { + Cr::Utility::formatInto(resStr, resStr.size(), "\n\t{}", key); + } + ESP_DEBUG() << resStr; + // Verify there are 4 layers of subconfigs to keyToFind + CORRADE_COMPARE(subconfigKeyPath.size(), 4); + // Verify the last entry holds keyToFind + CORRADE_COMPARE(subconfigKeyPath[3], keyToFind); + // Procede through the subconfig layers and find the desired target config + Configuration::cptr viewConfig = cfg; + Configuration::cptr lastConfig = cfg; + for (const std::string& key : subconfigKeyPath) { + Configuration::cptr tempConfig = viewConfig->getSubconfigView(key); + CORRADE_COMPARE(tempConfig->get("key"), key); + lastConfig = viewConfig; + viewConfig = tempConfig; + } + + // By here lastConfig is the parent of viewConfig, the config we want at + // keyToFind + // We make a copy of viewConfig and verify the copy has the same values as the + // original + Configuration::ptr cfgToEdit = + lastConfig->getSubconfigCopy(keyToFind); + CORRADE_VERIFY(*cfgToEdit == *viewConfig); + // Now modify new subconfig copy and verify the original viewConfig is not + // also modified + const std::string newKey = "edited_subconfig"; + cfgToEdit->set("key", newKey); + cfgToEdit->set("depth", 0); + cfgToEdit->set("breadcrumb", ""); + cfgToEdit->set("test_string", "this is an added test string"); + + // Added edited subconfig to base config (top of hierarchy) + cfg->setSubconfigPtr(newKey, cfgToEdit); + CORRADE_VERIFY(cfg->hasSubconfig(newKey)); + // Get an immutable view of this subconfig and verify the data it holds + Configuration::cptr cfgToVerify = cfg->getSubconfigView(newKey); + CORRADE_COMPARE(cfgToVerify->get("breadcrumb"), ""); + CORRADE_COMPARE(cfgToVerify->get("test_string"), + "this is an added test string"); + CORRADE_COMPARE(cfgToVerify->get("key"), newKey); + CORRADE_COMPARE(cfgToVerify->get("depth"), 0); + // Make sure original was not modified + CORRADE_VERIFY(viewConfig->get("breadcrumb") != ""); + CORRADE_VERIFY(viewConfig->get("key") != newKey); + CORRADE_VERIFY(viewConfig->get("depth") != 0); + CORRADE_VERIFY(!viewConfig->hasValue("test_string")); + + // Now Test merge/Overwriting + const std::string mergedKey = "merged_subconfig"; + Configuration::ptr cfgToOverwrite = + cfg->editSubconfig(mergedKey); + + // first set some test values that will be clobbered + cfgToOverwrite->set("key", mergedKey); + cfgToOverwrite->set("depth", 11); + cfgToOverwrite->set("breadcrumb", "1123"); + cfgToOverwrite->set("test_string", "this string will be clobbered"); + // Now add some values that won't be clobbered + cfgToOverwrite->set("myBool", true); + cfgToOverwrite->set("myInt", 10); + cfgToOverwrite->set("myFloatToDouble", 1.2f); + cfgToOverwrite->set("myVec2", Mn::Vector2{1.0, 2.0}); + cfgToOverwrite->set("myVec3", Mn::Vector3{1.0, 2.0, 3.0}); + cfgToOverwrite->set("myVec4", Mn::Vector4{1.0, 2.0, 3.0, 4.0}); + cfgToOverwrite->set("myQuat", Mn::Quaternion{{1.0, 2.0, 3.0}, 0.1}); + cfgToOverwrite->set("myRad", Mn::Rad{1.23}); + cfgToOverwrite->set("myString", "test"); + + // Now overwrite the values from cfgToVerify + cfgToOverwrite->overwriteWithConfig(cfgToVerify); + + CORRADE_VERIFY(cfg->hasSubconfig(mergedKey)); + // Verify all the overwritten values are correct + Configuration::cptr cfgToVerifyOverwrite = cfg->getSubconfigView(mergedKey); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("breadcrumb"), ""); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("test_string"), + "this is an added test string"); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("key"), newKey); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("depth"), 0); + // Verify original non-overwritten values are still present + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myBool"), true); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myInt"), 10); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myFloatToDouble"), 1.2f); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec2"), + Mn::Vector2(1.0, 2.0)); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec3"), + Mn::Vector3(1.0, 2.0, 3.0)); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec4"), + Mn::Vector4(1.0, 2.0, 3.0, 4.0)); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myQuat"), + Mn::Quaternion({1.0, 2.0, 3.0}, 0.1)); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myRad"), Mn::Rad(1.23)); + CORRADE_COMPARE(cfgToVerifyOverwrite->get("myString"), "test"); + // Verify overwrite performed properly + compareSubconfigs(cfgToVerify, cfgToVerifyOverwrite); +} + +void ConfigurationTest::TestSubconfigFilter() { + Configuration::ptr cfg = Configuration::create(); + CORRADE_VERIFY(cfg); + // Build and verify hierarchy + int depthPlus1 = 5; + int count = 3; + // build two identical subconfigs and place within base Configuration + const std::string filterKey = "cfgToFilterBy"; + // Get actual subconfig + Configuration::ptr cfgToFilterBy = + cfg->editSubconfig(filterKey); + // add some root values will be present in 'modified' subconfig too + cfgToFilterBy->set("baseStrKey", "filterKey"); + cfgToFilterBy->set("baseIntKey", 1); + cfgToFilterBy->set("baseBoolKey", true); + cfgToFilterBy->set("baseDoubleKey", 3.14); + cfgToFilterBy->set("baseStrKeyToMod", "modFilterKey"); + cfgToFilterBy->set("baseIntKeyToMod", 11); + cfgToFilterBy->set("baseBoolKeyToMod", true); + cfgToFilterBy->set("baseDoubleKeyToMod", 2.718); + // Build subconfig tree 4 levels deep, 3 subconfigs per config + buildSubconfigsInConfig(count, 0, depthPlus1, cfgToFilterBy); + + const std::string modKey = "modifiedSubconfig"; + // get copy of original filter config + Configuration::ptr cfgToModify = + cfg->getSubconfigCopy(filterKey); + // move copy into cfg + cfg->setSubconfigPtr(modKey, cfgToModify); + // retrieve pointer to new copy + cfgToModify = cfg->editSubconfig(modKey); + + // Build subconfig tree 4 levels deep, 3 subconfigs per config + buildSubconfigsInConfig(count, 0, depthPlus1, cfgToModify); + + // compare both configs and verify they are equal + { + Configuration::cptr baseCompareCfg = + std::static_pointer_cast(cfgToFilterBy); + Configuration::cptr modCompareCfg = + std::static_pointer_cast(cfgToModify); + // verify both subconfigs are the same + compareSubconfigs(baseCompareCfg, modCompareCfg); + } + + // modify and save appropriate subconfig, filter using base subconfig, and + // then verify expected results + // Modify values that are also present in cfgToFilterBy + cfgToModify->set("baseStrKeyToMod", "_modified"); + cfgToModify->set("baseIntKeyToMod", 111); + cfgToModify->set("baseBoolKeyToMod", false); + cfgToModify->set("baseDoubleKeyToMod", 1234.5); + // Add 'mod' string to end of each config's breadcrumb string + addOrModSubconfigTreeVals("breadcrumb", "mod", cfgToModify); + // Add new string to each config + addOrModSubconfigTreeVals("newStrValue", "new string", cfgToModify); + // Add values that are not present in cfgToFilterBy + cfgToModify->set("myBool", true); + cfgToModify->set("myInt", 10); + cfgToModify->set("myFloatToDouble", 1.2f); + cfgToModify->set("myVec2", Mn::Vector2{1.0, 2.0}); + cfgToModify->set("myVec3", Mn::Vector3{1.0, 2.0, 3.0}); + cfgToModify->set("myVec4", Mn::Vector4{1.0, 2.0, 3.0, 4.0}); + cfgToModify->set("myQuat", Mn::Quaternion{{1.0, 2.0, 3.0}, 0.1}); + cfgToModify->set("myRad", Mn::Rad{1.23}); + cfgToModify->set("myString", "test"); + + // Now filter the config to not have any data shared ith cfgToFilterBy + cfgToModify->filterFromConfig(cfgToFilterBy); + // Verify old shared values are gone + CORRADE_VERIFY(!cfgToModify->hasValue("baseStrKey")); + CORRADE_VERIFY(!cfgToModify->hasValue("baseIntKey")); + CORRADE_VERIFY(!cfgToModify->hasValue("baseBoolKey")); + CORRADE_VERIFY(!cfgToModify->hasValue("baseDoubleKey")); + + // Verify modified shared values are present + CORRADE_VERIFY(cfgToModify->hasValue("baseStrKeyToMod")); + CORRADE_VERIFY(cfgToModify->hasValue("baseIntKeyToMod")); + CORRADE_VERIFY(cfgToModify->hasValue("baseBoolKeyToMod")); + CORRADE_VERIFY(cfgToModify->hasValue("baseDoubleKeyToMod")); + // ...and have expected values + CORRADE_COMPARE(cfgToModify->get("baseStrKeyToMod"), + "_modified"); + CORRADE_COMPARE(cfgToModify->get("baseIntKeyToMod"), 111); + CORRADE_COMPARE(cfgToModify->get("baseBoolKeyToMod"), false); + CORRADE_COMPARE(cfgToModify->get("baseDoubleKeyToMod"), 1234.5); + + // Verify modified values still present + CORRADE_COMPARE(cfgToModify->get("myBool"), true); + CORRADE_COMPARE(cfgToModify->get("myInt"), 10); + CORRADE_COMPARE(cfgToModify->get("myFloatToDouble"), 1.2f); + CORRADE_COMPARE(cfgToModify->get("myVec2"), + Mn::Vector2(1.0, 2.0)); + CORRADE_COMPARE(cfgToModify->get("myVec3"), + Mn::Vector3(1.0, 2.0, 3.0)); + CORRADE_COMPARE(cfgToModify->get("myVec4"), + Mn::Vector4(1.0, 2.0, 3.0, 4.0)); + CORRADE_COMPARE(cfgToModify->get("myQuat"), + Mn::Quaternion({1.0, 2.0, 3.0}, 0.1)); + CORRADE_COMPARE(cfgToModify->get("myRad"), Mn::Rad(1.23)); + CORRADE_COMPARE(cfgToModify->get("myString"), "test"); + + Configuration::cptr constModCfg = + std::const_pointer_cast(cfgToModify); + // verify breadcrumb mod is present + verifySubconfigTreeVals("breadcrumb", "mod", constModCfg); + // verify newStrValue is present + verifySubconfigTreeVals("newStrValue", "new string", constModCfg); +} // ConfigurationTest::TestSubconfigFilter + +} // namespace + +CORRADE_TEST_MAIN(ConfigurationTest) diff --git a/src/tests/CoreTest.cpp b/src/tests/CoreTest.cpp deleted file mode 100644 index 2bef3e8a99..0000000000 --- a/src/tests/CoreTest.cpp +++ /dev/null @@ -1,334 +0,0 @@ -// Copyright (c) Meta Platforms, Inc. and its affiliates. -// This source code is licensed under the MIT license found in the -// LICENSE file in the root directory of this source tree. - -#include -#include -#include "esp/core/Configuration.h" -#include "esp/core/Esp.h" - -using namespace esp::core::config; -namespace Cr = Corrade; - -namespace { - -struct CoreTest : Cr::TestSuite::Tester { - explicit CoreTest(); - - void buildSubconfigsInConfig(int countPerDepth, - int curDepth, - int totalDepth, - Configuration::ptr& config); - - void verifySubconfigTree(int countPerDepth, - int curDepth, - int totalDepth, - Configuration::cptr& config); - - void compareSubconfigs(Configuration::cptr& src, Configuration::cptr& target); - - void TestConfiguration(); - - /** - * @brief Test Configuration find and subconfig edit capability. Find returns - * a list of subconfiguration keys required to find a particular key - */ - void TestConfigurationSubconfigFind(); - - esp::logging::LoggingContext loggingContext_; -}; // struct CoreTest - -CoreTest::CoreTest() { - addTests({ - &CoreTest::TestConfiguration, - &CoreTest::TestConfigurationSubconfigFind, - }); -} - -void CoreTest::buildSubconfigsInConfig(int countPerDepth, - int curDepth, - int totalDepth, - Configuration::ptr& config) { - if (curDepth == totalDepth) { - return; - } - std::string breadcrumb = config->get("breakcrumb"); - for (int i = 0; i < countPerDepth; ++i) { - const std::string subconfigKey = Cr::Utility::formatString( - "breadcrumb_{}_depth_{}_subconfig_{}", breadcrumb, curDepth, i); - Configuration::ptr newCfg = - config->editSubconfig(subconfigKey); - newCfg->set("depth", curDepth); - newCfg->set("iter", i); - newCfg->set("breakcrumb", Cr::Utility::formatString("{}{}", breadcrumb, i)); - newCfg->set("key", subconfigKey); - - buildSubconfigsInConfig(countPerDepth, curDepth + 1, totalDepth, newCfg); - } -} // CoreTest::buildSubconfigsInConfig - -void CoreTest::verifySubconfigTree(int countPerDepth, - int curDepth, - int totalDepth, - Configuration::cptr& config) { - if (curDepth == totalDepth) { - return; - } - std::string breadcrumb = config->get("breakcrumb"); - for (int i = 0; i < countPerDepth; ++i) { - const std::string subconfigKey = Cr::Utility::formatString( - "breadcrumb_{}_depth_{}_subconfig_{}", breadcrumb, curDepth, i); - // Verify subconfig with key exists - CORRADE_VERIFY(config->hasSubconfig(subconfigKey)); - Configuration::cptr newCfg = config->getSubconfigView(subconfigKey); - CORRADE_VERIFY(newCfg); - // Verify subconfig depth value is as expected - CORRADE_COMPARE(newCfg->get("depth"), curDepth); - // Verify iteration and parent iteration - CORRADE_COMPARE(newCfg->get("iter"), i); - CORRADE_COMPARE(newCfg->get("breakcrumb"), - Cr::Utility::formatString("{}{}", breadcrumb, i)); - CORRADE_COMPARE(newCfg->get("key"), subconfigKey); - // Check into tree - verifySubconfigTree(countPerDepth, curDepth + 1, totalDepth, newCfg); - } - CORRADE_COMPARE(config->getNumSubconfigs(), countPerDepth); - -} // CoreTest::verifySubconfigTree - -void CoreTest::compareSubconfigs(Configuration::cptr& src, - Configuration::cptr& target) { - // verify target has at least all the number of subconfigs that src has - CORRADE_VERIFY(src->getNumSubconfigs() <= target->getNumSubconfigs()); - // verify that target has all the values that src has, and they are equal. - auto srcIterValPair = src->getValuesIterator(); - for (auto& cfgIter = srcIterValPair.first; cfgIter != srcIterValPair.second; - ++cfgIter) { - CORRADE_VERIFY(cfgIter->second == target->get(cfgIter->first)); - } - - // Verify all subconfigs have the same keys - get begin/end iterators for src - // subconfigs - auto srcIterConfigPair = src->getSubconfigIterator(); - for (auto& cfgIter = srcIterConfigPair.first; - cfgIter != srcIterConfigPair.second; ++cfgIter) { - CORRADE_VERIFY(target->hasSubconfig(cfgIter->first)); - Configuration::cptr srcSubConfig = cfgIter->second; - Configuration::cptr tarSubConfig = target->getSubconfigView(cfgIter->first); - compareSubconfigs(srcSubConfig, tarSubConfig); - } - -} // CoreTest::compareSubconfigs - -void CoreTest::TestConfiguration() { - Configuration cfg; - cfg.set("myBool", true); - cfg.set("myInt", 10); - cfg.set("myFloatToDouble", 1.2f); - cfg.set("myVec2", Mn::Vector2{1.0, 2.0}); - cfg.set("myVec3", Mn::Vector3{1.0, 2.0, 3.0}); - cfg.set("myVec4", Mn::Vector4{1.0, 2.0, 3.0, 4.0}); - cfg.set("myQuat", Mn::Quaternion{{1.0, 2.0, 3.0}, 0.1}); - cfg.set("myMat3", Mn::Matrix3(Mn::Math::IdentityInit)); - cfg.set("myMat4", Mn::Matrix4(Mn::Math::IdentityInit)); - cfg.set("myRad", Mn::Rad{1.23}); - cfg.set("myString", "test"); - - CORRADE_VERIFY(cfg.hasValue("myBool")); - CORRADE_VERIFY(cfg.hasValue("myInt")); - CORRADE_VERIFY(cfg.hasValue("myFloatToDouble")); - CORRADE_VERIFY(cfg.hasValue("myVec2")); - CORRADE_VERIFY(cfg.hasValue("myVec3")); - CORRADE_VERIFY(cfg.hasValue("myMat3")); - CORRADE_VERIFY(cfg.hasValue("myVec4")); - CORRADE_VERIFY(cfg.hasValue("myQuat")); - CORRADE_VERIFY(cfg.hasValue("myMat3")); - CORRADE_VERIFY(cfg.hasValue("myRad")); - CORRADE_VERIFY(cfg.hasValue("myString")); - - CORRADE_COMPARE(cfg.get("myBool"), true); - CORRADE_COMPARE(cfg.get("myInt"), 10); - CORRADE_COMPARE(cfg.get("myFloatToDouble"), 1.2f); - CORRADE_COMPARE(cfg.get("myVec2"), Mn::Vector2(1.0, 2.0)); - CORRADE_COMPARE(cfg.get("myVec3"), Mn::Vector3(1.0, 2.0, 3.0)); - CORRADE_COMPARE(cfg.get("myVec4"), - Mn::Vector4(1.0, 2.0, 3.0, 4.0)); - CORRADE_COMPARE(cfg.get("myQuat"), - Mn::Quaternion({1.0, 2.0, 3.0}, 0.1)); - CORRADE_COMPARE(cfg.get("myRad"), Mn::Rad(1.23)); - - for (int i = 0; i < 3; ++i) { - CORRADE_COMPARE(cfg.get("myMat3").row(i)[i], 1); - } - for (int i = 0; i < 4; ++i) { - CORRADE_COMPARE(cfg.get("myMat4").row(i)[i], 1); - } - - CORRADE_COMPARE(cfg.get("myString"), "test"); -} // CoreTest::TestConfiguration test - -// Test configuration find functionality -void CoreTest::TestConfigurationSubconfigFind() { - { - Configuration::ptr cfg = Configuration::create(); - Configuration::ptr baseCfg = cfg; - // Build layers of subconfigs - for (int i = 0; i < 10; ++i) { - Configuration::ptr newCfg = cfg->editSubconfig( - Cr::Utility::formatString("depth_{}_subconfig_{}", i + 1, i)); - newCfg->set("depth", i + 1); - cfg = newCfg; - } - // Set deepest layer config to have treasure - cfg->set("treasure", "this is the treasure!"); - // Find the treasure! - std::vector keyList = baseCfg->findValue("treasure"); - - // Display breadcrumbs - std::string resStr = Cr::Utility::formatString( - "Vector of 'treasure' keys is size : {}", keyList.size()); - for (const auto& key : keyList) { - Cr::Utility::formatInto(resStr, resStr.size(), "\n\t{}", key); - } - ESP_DEBUG() << resStr; - // Verify the found treasure - CORRADE_COMPARE(keyList.size(), 11); - - Configuration::cptr viewConfig = baseCfg; - for (int i = 0; i < 10; ++i) { - const std::string subconfigKey = - Cr::Utility::formatString("depth_{}_subconfig_{}", i + 1, i); - CORRADE_COMPARE(keyList[i], subconfigKey); - // Verify that subconfig with given name exists - CORRADE_VERIFY(viewConfig->hasSubconfig(subconfigKey)); - // retrieve actual subconfig - Configuration::cptr newCfg = viewConfig->getSubconfigView(subconfigKey); - // verity subconfig has correct depth value - CORRADE_COMPARE(newCfg->get("depth"), i + 1); - viewConfig = newCfg; - } - CORRADE_VERIFY(viewConfig->hasValue("treasure")); - CORRADE_COMPARE(viewConfig->get("treasure"), - "this is the treasure!"); - - // Verify pirate isn't found - std::vector notFoundKeyList = baseCfg->findValue("pirate"); - CORRADE_COMPARE(notFoundKeyList.size(), 0); - } - // Test subconfig edit/merge and save - { - Configuration::ptr cfg = Configuration::create(); - CORRADE_VERIFY(cfg); - // Set base value for parent iteration - cfg->set("breakcrumb", ""); - int depth = 5; - int count = 3; - // Build subconfig tree 4 levels deep, 3 subconfigs per config - buildSubconfigsInConfig(count, 0, depth, cfg); - Configuration::cptr const_cfg = - std::const_pointer_cast(cfg); - // Verify subconfig tree structure using const views - verifySubconfigTree(count, 0, depth, const_cfg); - - // grab a subconfig, edit it and save it with a different key - // use find to get key path - const std::string keyToFind = "breadcrumb_212_depth_3_subconfig_2"; - std::vector subconfigPath = cfg->findValue(keyToFind); - // Display breadcrumbs - std::string resStr = Cr::Utility::formatString( - "Vector of desired subconfig keys to get to '{}' " - "subconfig is size : {}", - keyToFind, subconfigPath.size()); - for (const auto& key : subconfigPath) { - Cr::Utility::formatInto(resStr, resStr.size(), "\n\t{}", key); - } - ESP_DEBUG() << resStr; - CORRADE_COMPARE(subconfigPath.size(), 4); - CORRADE_COMPARE(subconfigPath[3], keyToFind); - Configuration::cptr viewConfig = cfg; - Configuration::cptr lastConfig = cfg; - for (const std::string& key : subconfigPath) { - Configuration::cptr tempConfig = viewConfig->getSubconfigView(key); - CORRADE_COMPARE(tempConfig->get("key"), key); - lastConfig = viewConfig; - viewConfig = tempConfig; - } - // By here lastConfig is the parent of the config we want, and viewConfig is - // the config we are getting a copy of. - Configuration::ptr cfgToEdit = - lastConfig->getSubconfigCopy(keyToFind); - const std::string newKey = "edited_subconfig"; - cfgToEdit->set("key", newKey); - cfgToEdit->set("depth", 0); - cfgToEdit->set("breakcrumb", ""); - cfgToEdit->set("test_string", "this is an added test string"); - // Added edited subconfig to base config - cfg->setSubconfigPtr(newKey, cfgToEdit); - CORRADE_VERIFY(cfg->hasSubconfig(newKey)); - Configuration::cptr cfgToVerify = cfg->getSubconfigView(newKey); - CORRADE_COMPARE(cfgToVerify->get("breakcrumb"), ""); - CORRADE_COMPARE(cfgToVerify->get("test_string"), - "this is an added test string"); - CORRADE_COMPARE(cfgToVerify->get("key"), newKey); - CORRADE_COMPARE(cfgToVerify->get("depth"), 0); - // Make sure original was not modified - CORRADE_VERIFY(viewConfig->get("breakcrumb") != ""); - CORRADE_VERIFY(viewConfig->get("key") != newKey); - CORRADE_VERIFY(viewConfig->get("depth") != 0); - CORRADE_VERIFY(!viewConfig->hasValue("test_string")); - - // Now Test merge/Overwriting - const std::string mergedKey = "merged_subconfig"; - Configuration::ptr cfgToOverwrite = - cfg->editSubconfig(mergedKey); - - // first set some test values that will be clobbered - cfgToOverwrite->set("key", mergedKey); - cfgToOverwrite->set("depth", 11); - cfgToOverwrite->set("breakcrumb", "1123"); - cfgToOverwrite->set("test_string", "this string will be clobbered"); - // Now add some values that won't be clobbered - cfgToOverwrite->set("myBool", true); - cfgToOverwrite->set("myInt", 10); - cfgToOverwrite->set("myFloatToDouble", 1.2f); - cfgToOverwrite->set("myVec2", Mn::Vector2{1.0, 2.0}); - cfgToOverwrite->set("myVec3", Mn::Vector3{1.0, 2.0, 3.0}); - cfgToOverwrite->set("myVec4", Mn::Vector4{1.0, 2.0, 3.0, 4.0}); - cfgToOverwrite->set("myQuat", Mn::Quaternion{{1.0, 2.0, 3.0}, 0.1}); - cfgToOverwrite->set("myRad", Mn::Rad{1.23}); - cfgToOverwrite->set("myString", "test"); - - // Now overwrite the values from cfgToVerify - cfgToOverwrite->overwriteWithConfig(cfgToVerify); - - CORRADE_VERIFY(cfg->hasSubconfig(mergedKey)); - // Verify all the overwritten values are correct - Configuration::cptr cfgToVerifyOverwrite = cfg->getSubconfigView(mergedKey); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("breakcrumb"), ""); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("test_string"), - "this is an added test string"); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("key"), newKey); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("depth"), 0); - // Verify original non-overwritten values are still present - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myBool"), true); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myInt"), 10); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myFloatToDouble"), 1.2f); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec2"), - Mn::Vector2(1.0, 2.0)); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec3"), - Mn::Vector3(1.0, 2.0, 3.0)); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myVec4"), - Mn::Vector4(1.0, 2.0, 3.0, 4.0)); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myQuat"), - Mn::Quaternion({1.0, 2.0, 3.0}, 0.1)); - CORRADE_COMPARE(cfgToVerifyOverwrite->get("myRad"), Mn::Rad(1.23)); - - // Verify overwrite performed properly - compareSubconfigs(cfgToVerify, cfgToVerifyOverwrite); - } - -} // CoreTest::TestConfigurationSubconfigFind test - -} // namespace - -CORRADE_TEST_MAIN(CoreTest)