Skip to content

Commit 0b58d04

Browse files
authored
--[BUGFIX] Make sure empty Markerset subconfigs are removed when appropriate (#2434)
* --make sure empty subconfigs are removed to minimize clutter in saved json. * --cleanup configuration; handle get attempt from non-existing subconfig Instead of asserting, return a warning along with an empty vector <T> if the named subconfig is not found. This is consistent with the behavior of other indirect subconfig queries. * --handle if the 'markers' subconfig does not exist in a MarkerSet It is possible to have a JSON marker set object associated with a link and task that lists no points. Due to the markersets being managed generically as nested configurations, with access accomplished via static casting, it is then possible to access a MarkerSet object that may not have a "markers" subconfig. This is a reasonable state, and should result in an assertion when attempting to query the MarkerSet.
1 parent 9d1fba2 commit 0b58d04

File tree

2 files changed

+83
-37
lines changed

2 files changed

+83
-37
lines changed

src/esp/core/Configuration.h

+34-28
Original file line numberDiff line numberDiff line change
@@ -1220,21 +1220,21 @@ class Configuration {
12201220
/**
12211221
* @brief Templated subconfig copy getter. Retrieves a shared pointer to a
12221222
* copy of the subConfig @ref esp::core::config::Configuration that has the
1223-
* passed @p name .
1223+
* passed @p cfgName .
12241224
*
12251225
* @tparam Type to return. Must inherit from @ref
12261226
* esp::core::config::Configuration
1227-
* @param name The name of the Configuration to retrieve.
1227+
* @param cfgName The name of the Configuration to retrieve.
12281228
* @return A pointer to a copy of the Configuration having the requested
12291229
* name, cast to the appropriate type, or nullptr if not found.
12301230
*/
12311231

12321232
template <typename T>
1233-
std::shared_ptr<T> getSubconfigCopy(const std::string& name) const {
1233+
std::shared_ptr<T> getSubconfigCopy(const std::string& cfgName) const {
12341234
static_assert(std::is_base_of<Configuration, T>::value,
12351235
"Configuration : Desired subconfig must be derived from "
12361236
"core::config::Configuration");
1237-
auto configIter = configMap_.find(name);
1237+
auto configIter = configMap_.find(cfgName);
12381238
if (configIter != configMap_.end()) {
12391239
// if exists return copy, so that consumers can modify it freely
12401240
return std::make_shared<T>(
@@ -1244,91 +1244,92 @@ class Configuration {
12441244
}
12451245

12461246
/**
1247-
* @brief return pointer to read-only sub-Configuration of given @p name.
1247+
* @brief return pointer to read-only sub-Configuration of given @p cfgName.
12481248
* Will fail if Configuration with given name dne.
1249-
* @param name The name of the desired Configuration.
1249+
* @param cfgName The name of the desired Configuration.
12501250
*/
12511251
std::shared_ptr<const Configuration> getSubconfigView(
1252-
const std::string& name) const {
1253-
auto configIter = configMap_.find(name);
1254-
CORRADE_ASSERT(
1255-
configIter != configMap_.end(),
1256-
"SubConfiguration with name " << name << " not found in Configuration.",
1257-
nullptr);
1252+
const std::string& cfgName) const {
1253+
auto configIter = configMap_.find(cfgName);
1254+
CORRADE_ASSERT(configIter != configMap_.end(),
1255+
"SubConfiguration with name "
1256+
<< cfgName << " not found in Configuration.",
1257+
nullptr);
12581258
// if exists return actual object
12591259
return configIter->second;
12601260
}
12611261

12621262
/**
12631263
* @brief Templated Version. Retrieves the stored shared pointer to the
1264-
* subConfig @ref esp::core::config::Configuration that has the passed @p name
1264+
* subConfig @ref esp::core::config::Configuration that has the passed @p cfgName
12651265
* , cast to the specified type. This will create a shared pointer to a new
12661266
* sub-Configuration if none exists and return it, cast to specified type.
12671267
*
12681268
* Use this function when you wish to modify this Configuration's
12691269
* subgroup, possibly creating it in the process.
12701270
* @tparam The type to cast the @ref esp::core::config::Configuration to. Type
12711271
* is checked to verify that it inherits from Configuration.
1272-
* @param name The name of the Configuration to edit.
1272+
* @param cfgName The name of the Configuration to edit.
12731273
* @return The actual pointer to the Configuration having the requested
12741274
* name, cast to the specified type.
12751275
*/
12761276
template <typename T>
1277-
std::shared_ptr<T> editSubconfig(const std::string& name) {
1277+
std::shared_ptr<T> editSubconfig(const std::string& cfgName) {
12781278
static_assert(std::is_base_of<Configuration, T>::value,
12791279
"Configuration : Desired subconfig must be derived from "
12801280
"core::config::Configuration");
12811281
// retrieve existing (or create new) subgroup, with passed name
12821282
return std::static_pointer_cast<T>(
1283-
addOrEditSubgroup<T>(name).first->second);
1283+
addOrEditSubgroup<T>(cfgName).first->second);
12841284
}
12851285

12861286
/**
12871287
* @brief move specified subgroup config into configMap at desired name.
12881288
* Will replace any subConfiguration at given name without warning if
12891289
* present.
1290-
* @param name The name of the subConfiguration to add
1290+
* @param cfgName The name of the subConfiguration to add
12911291
* @param configPtr A pointer to a subConfiguration to add.
12921292
*/
12931293
template <typename T>
1294-
void setSubconfigPtr(const std::string& name, std::shared_ptr<T>& configPtr) {
1294+
void setSubconfigPtr(const std::string& cfgName,
1295+
std::shared_ptr<T>& configPtr) {
12951296
static_assert(std::is_base_of<Configuration, T>::value,
12961297
"Configuration : Desired subconfig must be derived from "
12971298
"core::config::Configuration");
12981299

1299-
configMap_[name] = std::move(configPtr);
1300+
configMap_[cfgName] = std::move(configPtr);
13001301
} // setSubconfigPtr
13011302

13021303
/**
13031304
* @brief Removes and returns the named subconfig. If not found, returns an
13041305
* empty subconfig with a warning.
1305-
* @param name The name of the subConfiguration to delete
1306+
* @param cfgName The name of the subConfiguration to delete
13061307
* @return a shared pointer to the removed subConfiguration.
13071308
*/
1308-
std::shared_ptr<Configuration> removeSubconfig(const std::string& name) {
1309-
ConfigMapType::const_iterator mapIter = configMap_.find(name);
1309+
std::shared_ptr<Configuration> removeSubconfig(const std::string& cfgName) {
1310+
ConfigMapType::const_iterator mapIter = configMap_.find(cfgName);
13101311
if (mapIter != configMap_.end()) {
13111312
configMap_.erase(mapIter);
13121313
return mapIter->second;
13131314
}
1314-
ESP_WARNING() << "Name :" << name
1315+
ESP_WARNING() << "Name :" << cfgName
13151316
<< "not present in map of subConfigurations.";
13161317
return {};
13171318
}
13181319

13191320
/**
13201321
* @brief Retrieve the number of entries held by the subconfig with the
13211322
* given name
1322-
* @param name The name of the subconfig to query. If not found, returns 0
1323+
* @param cfgName The name of the subconfig to query. If not found, returns 0
13231324
* with a warning.
13241325
* @return The number of entries in the named subconfig
13251326
*/
1326-
int getSubconfigNumEntries(const std::string& name) const {
1327-
auto configIter = configMap_.find(name);
1327+
int getSubconfigNumEntries(const std::string& cfgName) const {
1328+
auto configIter = configMap_.find(cfgName);
13281329
if (configIter != configMap_.end()) {
13291330
return configIter->second->getNumEntries();
13301331
}
1331-
ESP_WARNING() << "No Subconfig found named :" << name;
1332+
ESP_WARNING() << "No Subconfig found named :" << cfgName;
13321333
return 0;
13331334
}
13341335

@@ -1403,7 +1404,12 @@ class Configuration {
14031404
std::vector<T> getSubconfigValsOfTypeInVector(
14041405
const std::string& subCfgName) const {
14051406
const ConfigValType desiredType = configValTypeFor<T>();
1406-
const auto subCfg = getSubconfigView(subCfgName);
1407+
auto configIter = configMap_.find(subCfgName);
1408+
if (configIter == configMap_.end()) {
1409+
ESP_WARNING() << "No Subconfig found named :" << subCfgName;
1410+
return {};
1411+
}
1412+
const auto subCfg = configIter->second;
14071413
const auto& subCfgTags = subCfg->getKeysByType(desiredType, true);
14081414
std::vector<T> res;
14091415
res.reserve(subCfgTags.size());

src/esp/metadata/attributes/MarkerSets.h

+49-9
Original file line numberDiff line numberDiff line change
@@ -26,14 +26,24 @@ class MarkerSet : public esp::core::config::Configuration {
2626
/**
2727
* @brief Returns the number of existing marker points in this MarkerSet.
2828
*/
29-
int getNumPoints() const {
29+
int getNumPoints() {
30+
// the 'markers' subconfig may not exist due to how the MarkerSet hierarchy
31+
// is loaded from JSON.
32+
if (!hasSubconfig("markers")) {
33+
return 0;
34+
}
3035
return getSubconfigView("markers")->getNumValues();
3136
}
3237

3338
/**
3439
* @brief Returns a list of all the marker points in this MarkerSet
3540
*/
3641
std::vector<Mn::Vector3> getAllPoints() const {
42+
// the 'markers' subconfig may not exist due to how the MarkerSet hierarchy
43+
// is loaded from JSON.
44+
if (!hasSubconfig("markers")) {
45+
return {};
46+
}
3747
return getSubconfigValsOfTypeInVector<Mn::Vector3>("markers");
3848
}
3949

@@ -51,6 +61,7 @@ class MarkerSet : public esp::core::config::Configuration {
5161
*/
5262
int rekeyAllMarkers() { return rekeySubconfigValues("markers"); }
5363

64+
public:
5465
ESP_SMART_POINTERS(MarkerSet)
5566
}; // class MarkerSet
5667

@@ -134,7 +145,11 @@ class LinkSet : public esp::core::config::Configuration {
134145
*/
135146
void setMarkerSetPoints(const std::string& markerSetName,
136147
const std::vector<Mn::Vector3>& markerList) {
137-
editMarkerSet(markerSetName)->setAllPoints(markerList);
148+
if (markerList.size() == 0) {
149+
removeMarkerSet(markerSetName);
150+
} else {
151+
editMarkerSet(markerSetName)->setAllPoints(markerList);
152+
}
138153
}
139154

140155
/**
@@ -310,7 +325,12 @@ class TaskSet : public esp::core::config::Configuration {
310325
void setLinkMarkerSetPoints(const std::string& linkSetName,
311326
const std::string& markerSetName,
312327
const std::vector<Mn::Vector3>& markerList) {
313-
editLinkSet(linkSetName)->setMarkerSetPoints(markerSetName, markerList);
328+
auto linkSet = editLinkSet(linkSetName);
329+
linkSet->setMarkerSetPoints(markerSetName, markerList);
330+
// The above may result in the link set being empty. If so remove it.
331+
if (linkSet->getNumMarkerSets() == 0) {
332+
removeLinkSet(linkSetName);
333+
}
314334
}
315335

316336
/**
@@ -325,7 +345,12 @@ class TaskSet : public esp::core::config::Configuration {
325345
const std::string& linkSetName,
326346
const std::unordered_map<std::string, std::vector<Mn::Vector3>>&
327347
markers) {
328-
editLinkSet(linkSetName)->setAllMarkerPoints(markers);
348+
auto linkSet = editLinkSet(linkSetName);
349+
linkSet->setAllMarkerPoints(markers);
350+
// The above may result in the link set being empty. If so remove it.
351+
if (linkSet->getNumMarkerSets() == 0) {
352+
removeLinkSet(linkSetName);
353+
}
329354
}
330355

331356
/**
@@ -542,7 +567,6 @@ class MarkerSets : public esp::core::config::Configuration {
542567
* @param linkSetName the name of the LinkSet within @p taskSetName
543568
* @param markerSetName the name of the MarkerSet within @p linkSetName
544569
*/
545-
546570
void initTaskLinkMarkerSet(const std::string& taskSetName,
547571
const std::string& linkSetName,
548572
const std::string& markerSetName) {
@@ -561,8 +585,12 @@ class MarkerSets : public esp::core::config::Configuration {
561585
const std::string& linkSetName,
562586
const std::string& markerSetName,
563587
const std::vector<Mn::Vector3>& markerList) {
564-
editTaskSet(taskSetName)
565-
->setLinkMarkerSetPoints(linkSetName, markerSetName, markerList);
588+
auto taskSetPtr = editTaskSet(taskSetName);
589+
taskSetPtr->setLinkMarkerSetPoints(linkSetName, markerSetName, markerList);
590+
// After this process, the taskset might be empty, if so, delete it.
591+
if (taskSetPtr->getNumLinkSets() == 0) {
592+
removeTaskSet(taskSetName);
593+
}
566594
}
567595

568596
/**
@@ -579,7 +607,12 @@ class MarkerSets : public esp::core::config::Configuration {
579607
const std::string& linkSetName,
580608
const std::unordered_map<std::string, std::vector<Mn::Vector3>>&
581609
markerMap) {
582-
editTaskSet(taskSetName)->setLinkSetPoints(linkSetName, markerMap);
610+
auto taskSetPtr = editTaskSet(taskSetName);
611+
taskSetPtr->setLinkSetPoints(linkSetName, markerMap);
612+
// After this process, the taskset might be empty, if so, delete it.
613+
if (taskSetPtr->getNumLinkSets() == 0) {
614+
removeTaskSet(taskSetName);
615+
}
583616
}
584617

585618
/**
@@ -595,7 +628,14 @@ class MarkerSets : public esp::core::config::Configuration {
595628
std::string,
596629
std::unordered_map<std::string, std::vector<Mn::Vector3>>>&
597630
markerMap) {
598-
editTaskSet(taskSetName)->setAllMarkerPoints(markerMap);
631+
auto taskSetPtr = editTaskSet(taskSetName);
632+
for (const auto& markers : markerMap) {
633+
taskSetPtr->setLinkSetPoints(markers.first, markers.second);
634+
}
635+
// After this process, the taskset might be empty, if so, delete it.
636+
if (taskSetPtr->getNumLinkSets() == 0) {
637+
removeTaskSet(taskSetName);
638+
}
599639
}
600640

601641
/**

0 commit comments

Comments
 (0)