Skip to content

Commit

Permalink
--[BE] - ManagedContainer updates and cleanups (#2475)
Browse files Browse the repository at this point in the history
* --clean up handle query
* --move object library map settings to base class.
* --make managed container map member variables private.
* --verify templated accessor results are cast to legitimate type.
* --simplify caller-message arguments
* --clarify naming of existing dirty flag, in preparation of file-save-based flag.
* --move filepath check flag to base attributes to make available
* --treat pbr/ibl helper key as hidden field
* --provide rudiments of file-status dirty flags.
The purpose of this flag is to specify that a particular Managed File-based object has been registered in its Manager in a state that is different than that of its disk-based counterpart.
* --rename constructor-building method for clarity
  • Loading branch information
jturner65 authored Sep 30, 2024
1 parent c0b2110 commit d924c81
Show file tree
Hide file tree
Showing 38 changed files with 346 additions and 161 deletions.
6 changes: 5 additions & 1 deletion examples/tutorials/nb_python/ECCV_2020_Advanced_Features.py
Original file line number Diff line number Diff line change
Expand Up @@ -171,7 +171,11 @@ def build_dict_of_PhyObj_attrs(phys_obj_template):
False,
"boolean",
)
res_dict["is_dirty"] = (phys_obj_template.is_dirty, False, "boolean")
res_dict["filenames_are_dirty"] = (
phys_obj_template.filenames_are_dirty,
False,
"boolean",
)
return res_dict


Expand Down
6 changes: 5 additions & 1 deletion examples/tutorials/nb_python/asset_viewer.py
Original file line number Diff line number Diff line change
Expand Up @@ -331,7 +331,11 @@ def build_dict_of_PhyObj_attrs(phys_obj_template):
"boolean",
)
res_dict["is_collidable"] = (phys_obj_template.is_collidable, True, "boolean")
res_dict["is_dirty"] = (phys_obj_template.is_dirty, False, "boolean")
res_dict["filenames_are_dirty"] = (
phys_obj_template.filenames_are_dirty,
False,
"boolean",
)
return res_dict


Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -163,7 +163,11 @@
" False,\n",
" \"boolean\",\n",
" )\n",
" res_dict[\"is_dirty\"] = (phys_obj_template.is_dirty, False, \"boolean\")\n",
" res_dict[\"filenames_are_dirty\"] = (\n",
" phys_obj_template.filenames_are_dirty,\n",
" False,\n",
" \"boolean\",\n",
" )\n",
" return res_dict\n",
"\n",
"\n",
Expand Down
6 changes: 5 additions & 1 deletion examples/tutorials/notebooks/asset_viewer.ipynb
Original file line number Diff line number Diff line change
Expand Up @@ -329,7 +329,11 @@
" \"boolean\",\n",
" )\n",
" res_dict[\"is_collidable\"] = (phys_obj_template.is_collidable, True, \"boolean\")\n",
" res_dict[\"is_dirty\"] = (phys_obj_template.is_dirty, False, \"boolean\")\n",
" res_dict[\"filenames_are_dirty\"] = (\n",
" phys_obj_template.filenames_are_dirty,\n",
" False,\n",
" \"boolean\",\n",
" )\n",
" return res_dict\n",
"\n",
"\n",
Expand Down
2 changes: 1 addition & 1 deletion src/esp/assets/ResourceManager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2885,7 +2885,7 @@ bool ResourceManager::instantiateAssetsOnDemand(
// object has acquired a copy of its parent attributes. No object should
// ever have a copy of attributes with isDirty == true - any editing of
// attributes for objects requires object rebuilding.
if (objectAttributes->getIsDirty()) {
if (objectAttributes->getFilePathsAreDirty()) {
CORRADE_ASSERT(
(getObjectAttributesManager()->registerObject(
objectAttributes, objectTemplateHandle) != ID_UNDEFINED),
Expand Down
12 changes: 6 additions & 6 deletions src/esp/bindings/AttributesBindings.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -453,7 +453,11 @@ void initAttributesBindings(py::module& m) {
R"(Class name of Attributes template.)")
.def_property_readonly(
"csv_info", &AbstractAttributes::getObjectInfo,
R"(Comma-separated informational string describing this Attributes template)");
R"(Comma-separated informational string describing this Attributes template)")
.def_property_readonly(
"filenames_are_dirty", &AbstractAttributes::getFilePathsAreDirty,
R"(Whether filenames or paths in this attributes have been changed requiring
re-registration before they can be used to create an object. )");

// Attributes should only use named properties or subconfigurations to set
// specific values, to guarantee essential value type integrity. This will
Expand Down Expand Up @@ -680,11 +684,7 @@ void initAttributesBindings(py::module& m) {
.def_property(
"is_collidable", &ObjectAttributes::getIsCollidable,
&ObjectAttributes::setIsCollidable,
R"(Whether constructions built from this template are collidable upon initialization.)")
.def_property_readonly(
"is_dirty", &AbstractObjectAttributes::getIsDirty,
R"(Whether values in this attributes have been changed requiring
re-registration before they can be used to create an object. )");
R"(Whether constructions built from this template are collidable upon initialization.)");

// ==== ObjectAttributes ====
py::class_<ObjectAttributes, AbstractObjectAttributes, ObjectAttributes::ptr>(
Expand Down
27 changes: 27 additions & 0 deletions src/esp/core/managedContainers/AbstractFileBasedManagedObject.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,26 @@ class AbstractFileBasedManagedObject : public AbstractManagedObject {
*/
virtual std::string getActualFilename() const = 0;

/**
* @brief Get whether this ManagedObject has been saved to disk in its current
* state. Only applicable to registered ManagedObjects
*/
virtual bool isAttrSaved() const = 0;

/**
* @brief Set that this ManagedObject has values that are different than its
* most recently saved-to-disk version. This is called when the ManagedObject
* is registered.
*/

void setAttrIsNotSaved() { setFileSaveStatus(false); }

/**
* @brief Set that this ManagedObject is the same as its saved-to-disk
* version. This is called when the ManagedObject is saved to disk.
*/
void setAttrIsSaved() { setFileSaveStatus(true); }

/**
* @brief This will return a simplified version of the
* AbstractFileBasedManagedObject handle, removing extensions and any parent
Expand All @@ -66,6 +86,13 @@ class AbstractFileBasedManagedObject : public AbstractManagedObject {
virtual io::JsonGenericValue writeToJsonObject(
io::JsonAllocator& allocator) const = 0;

protected:
/**
* @brief Set this ManagedObject's save status (i.e. whether it matches its
* version on disk or not)
*/
virtual void setFileSaveStatus(bool _isSaved) = 0;

public:
ESP_SMART_POINTERS(AbstractFileBasedManagedObject)
}; // class AbstractFileBasedManagedObject
Expand Down
97 changes: 53 additions & 44 deletions src/esp/core/managedContainers/ManagedContainer.h
Original file line number Diff line number Diff line change
Expand Up @@ -163,18 +163,18 @@ class ManagedContainer : public ManagedContainerBase {
"registration, so registration aborted.";
return ID_UNDEFINED;
}
if ("" != objectHandle) {
return this->registerObjectInternal(std::move(managedObject),
objectHandle, forceRegistration);
}
std::string handleToSet = managedObject->getHandle();
// If no handle give, query object for handle
std::string handleToSet =
("" == objectHandle) ? managedObject->getHandle() : objectHandle;
// if still no handle, fail registration
if ("" == handleToSet) {
ESP_ERROR(Magnum::Debug::Flag::NoSpace)
<< "<" << this->objectType_
<< "> : No valid handle specified to register this managed object, "
"so registration aborted.";
return ID_UNDEFINED;
}
// Perform actual registration
return this->registerObjectInternal(std::move(managedObject), handleToSet,
forceRegistration);
} // ManagedContainer::registerObject
Expand All @@ -193,8 +193,7 @@ class ManagedContainer : public ManagedContainerBase {
*/
ManagedPtr getObjectByID(int managedObjectID) const {
std::string objectHandle = getObjectHandleByID(managedObjectID);
if (!checkExistsWithMessage(objectHandle,
"<" + this->objectType_ + ">::getObjectByID")) {
if (!checkExistsWithMessage(objectHandle, "getObjectByID")) {
return nullptr;
}
return getObjectInternal<T>(objectHandle);
Expand All @@ -212,8 +211,7 @@ class ManagedContainer : public ManagedContainerBase {
* exist
*/
ManagedPtr getObjectByHandle(const std::string& objectHandle) const {
if (!checkExistsWithMessage(
objectHandle, "<" + this->objectType_ + ">::getObjectByHandle")) {
if (!checkExistsWithMessage(objectHandle, "getObjectByHandle")) {
return nullptr;
}
return getObjectInternal<T>(objectHandle);
Expand Down Expand Up @@ -250,7 +248,7 @@ class ManagedContainer : public ManagedContainerBase {
/**
* @brief Retrieve a map of key= std::string handle; value = copy of
* ManagedPtr object where the handles match the passed @p . See @ref
* ManagedContainerBase::getObjectHandlesBySubStringPerType.
* ManagedContainerBase::getAllObjectHandlesBySubStringPerType.
* @param subStr substring key to search for within existing managed objects.
* @param contains whether to search for keys containing, or excluding,
* passed @p subStr
Expand All @@ -260,8 +258,8 @@ class ManagedContainer : public ManagedContainerBase {
std::unordered_map<std::string, ManagedPtr> getObjectsByHandleSubstring(
const std::string& subStr = "",
bool contains = true) {
std::vector<std::string> keys = this->getObjectHandlesBySubStringPerType(
objectLibKeyByID_, subStr, contains, false);
std::vector<std::string> keys =
this->getAllObjectHandlesBySubStringPerType(subStr, contains, false);

std::unordered_map<std::string, ManagedPtr> res;
res.reserve(keys.size());
Expand All @@ -280,7 +278,7 @@ class ManagedContainer : public ManagedContainerBase {
/**
* @brief Templated version. Retrieve a map of key= std::string handle; value
* = copy of ManagedPtr object where the handles match the passed @p . See
* @ref ManagedContainerBase::getObjectHandlesBySubStringPerType.
* @ref ManagedContainerBase::getAllObjectHandlesBySubStringPerType.
*
* @tparam Desired downcast class that inerheits from this ManagedContainer's
* ManagedObject type.
Expand All @@ -294,8 +292,11 @@ class ManagedContainer : public ManagedContainerBase {
std::unordered_map<std::string, std::shared_ptr<U>>
getObjectsByHandleSubstring(const std::string& subStr = "",
bool contains = true) {
std::vector<std::string> keys = this->getObjectHandlesBySubStringPerType(
objectLibKeyByID_, subStr, contains, false);
static_assert(std::is_base_of<T, U>::value,
"ManagedContainer :: Desired type must be derived from "
"Managed object type");
std::vector<std::string> keys =
this->getAllObjectHandlesBySubStringPerType(subStr, contains, false);

std::unordered_map<std::string, std::shared_ptr<U>> res;
res.reserve(keys.size());
Expand All @@ -321,13 +322,10 @@ class ManagedContainer : public ManagedContainerBase {
*/
ManagedPtr removeObjectByID(int objectID) {
std::string objectHandle = getObjectHandleByID(objectID);
if (!checkExistsWithMessage(
objectHandle, "<" + this->objectType_ + ">::removeObjectByID")) {
if (!checkExistsWithMessage(objectHandle, "removeObjectByID")) {
return nullptr;
}
return removeObjectInternal(
objectID, objectHandle,
"<" + this->objectType_ + ">::removeObjectByID");
return removeObjectInternal(objectID, objectHandle, "removeObjectByID");
}

/**
Expand All @@ -339,17 +337,14 @@ class ManagedContainer : public ManagedContainerBase {
* exist
*/
ManagedPtr removeObjectByHandle(const std::string& objectHandle) {
if (!checkExistsWithMessage(objectHandle, "<" + this->objectType_ +
">::removeObjectByHandle")) {
if (!checkExistsWithMessage(objectHandle, "removeObjectByHandle")) {
return nullptr;
}
int objectID = this->getObjectIDByHandle(objectHandle);
if (objectID == ID_UNDEFINED) {
return nullptr;
}
return removeObjectInternal(
objectID, objectHandle,
"<" + this->objectType_ + ">::removeObjectByHandle");
return removeObjectInternal(objectID, objectHandle, "removeObjectByHandle");
}

/**
Expand Down Expand Up @@ -461,6 +456,9 @@ class ManagedContainer : public ManagedContainerBase {
*/
template <class U>
std::shared_ptr<U> getObjectOrCopyByHandle(const std::string& objectHandle) {
static_assert(std::is_base_of<T, U>::value,
"ManagedContainer :: Desired type must be derived from "
"Managed object type");
// call non-template version
auto res = getObjectOrCopyByHandle(objectHandle);
if (nullptr == res) {
Expand All @@ -480,8 +478,7 @@ class ManagedContainer : public ManagedContainerBase {
*/
ManagedPtr getObjectCopyByID(int managedObjectID) {
std::string objectHandle = getObjectHandleByID(managedObjectID);
if (!checkExistsWithMessage(
objectHandle, "<" + this->objectType_ + ">::getObjectCopyByID")) {
if (!checkExistsWithMessage(objectHandle, "getObjectCopyByID")) {
return nullptr;
}
auto orig = getObjectInternal<T>(objectHandle);
Expand All @@ -496,8 +493,7 @@ class ManagedContainer : public ManagedContainerBase {
* does not exist
*/
ManagedPtr getObjectCopyByHandle(const std::string& objectHandle) {
if (!checkExistsWithMessage(objectHandle, "<" + this->objectType_ +
">::getObjectCopyByHandle")) {
if (!checkExistsWithMessage(objectHandle, "getObjectCopyByHandle")) {
return nullptr;
}
auto orig = getObjectInternal<T>(objectHandle);
Expand Down Expand Up @@ -544,6 +540,9 @@ class ManagedContainer : public ManagedContainerBase {
*/
template <class U>
std::shared_ptr<U> getObjectCopyByID(int managedObjectID) {
static_assert(std::is_base_of<T, U>::value,
"ManagedContainer :: Desired type must be derived from "
"Managed object type");
// call non-template version
auto res = getObjectCopyByID(managedObjectID);
if (nullptr == res) {
Expand All @@ -563,6 +562,9 @@ class ManagedContainer : public ManagedContainerBase {
*/
template <class U>
std::shared_ptr<U> getObjectCopyByHandle(const std::string& objectHandle) {
static_assert(std::is_base_of<T, U>::value,
"ManagedContainer :: Desired type must be derived from "
"Managed object type");
// call non-template version
auto res = getObjectCopyByHandle(objectHandle);
if (nullptr == res) {
Expand Down Expand Up @@ -635,18 +637,22 @@ class ManagedContainer : public ManagedContainerBase {
const std::string& src);

/**
* @brief Build a shared pointer to a copy of a the passed managed object,
* of appropriate managed object type for passed object type. This is the
* function called by the copy constructor map.
* @brief This is the function called by the copy constructor map. Build a
* shared pointer to a copy of a the passed managed object, of appropriate
* managed object type for passed object type.
*
* @tparam U Type of managed object being created - must be a derived class
* of ManagedPtr
* @param orig original object of type ManagedPtr being copied
*/
template <typename U>
ManagedPtr createObjectCopy(ManagedPtr& orig) {
template <class U>
ManagedPtr createObjCopyCtorMapEntry(ManagedPtr& orig) {
static_assert(std::is_base_of<T, U>::value,
"ManagedContainer :: Desired type must be derived from "
"Managed object type");
// don't call init on copy - assume copy is already properly initialized.
return U::create(*(static_cast<U*>(orig.get())));
} // ManagedContainer::
} // ManagedContainer::createObjCopyCtorMapEntry

/**
* @brief Build an @ref esp::core::managedContainers::AbstractManagedObject
Expand Down Expand Up @@ -770,8 +776,7 @@ class ManagedContainer : public ManagedContainerBase {
// original
ManagedPtr managedObjectCopy = copyObject(object);
// add to libraries
setObjectInternal(managedObjectCopy, objectHandle);
objectLibKeyByID_.emplace(objectID, objectHandle);
setObjectInternal(managedObjectCopy, objectID, objectHandle);
return objectID;
} // ManagedContainer::addObjectToLibrary

Expand Down Expand Up @@ -817,8 +822,8 @@ auto ManagedContainer<T, Access>::removeObjectsBySubstring(
getObjectHandlesBySubstring(subStr, contains);
for (const std::string& objectHandle : handles) {
int objID = this->getObjectIDByHandle(objectHandle);
ManagedPtr ptr = removeObjectInternal(objID, objectHandle,
"<" + this->objectType_ + ">");
ManagedPtr ptr =
removeObjectInternal(objID, objectHandle, "removeObjectsBySubstring");
if (nullptr != ptr) {
res.push_back(ptr);
}
Expand All @@ -832,19 +837,23 @@ auto ManagedContainer<T, Access>::removeObjectInternal(
const std::string& objectHandle,
const std::string& sourceStr) -> ManagedPtr {
if (!checkExistsWithMessage(objectHandle, sourceStr)) {
ESP_DEBUG() << sourceStr << ": Unable to remove" << objectType_
<< "managed object" << objectHandle << ": Does not exist.";
ESP_DEBUG(Magnum::Debug::Flag::NoSpace)
<< "<" + this->objectType_ + ">::" << sourceStr
<< " : Unable to remove requested managed object `" << objectHandle
<< "` : Does not exist.";
return nullptr;
}
std::string msg;
if (this->getIsUndeletable(objectHandle)) {
msg = "Required Undeletable Managed Object";
} else if (this->getIsUserLocked(objectHandle)) {
msg = "User-locked Object. To delete managed object, unlock it";
msg = "User-locked Object. To delete managed object, unlock it";
}
if (msg.length() != 0) {
ESP_DEBUG() << sourceStr << ": Unable to remove" << objectType_
<< "managed object" << objectHandle << ":" << msg << ".";
ESP_DEBUG(Magnum::Debug::Flag::NoSpace)
<< "<" + this->objectType_ + ">::" << sourceStr
<< " : Unable to remove requested managed object `" << objectHandle
<< "` : Object is a " << msg << ".";
return nullptr;
}
ManagedPtr managedObject = getObjectInternal<T>(objectHandle);
Expand Down
3 changes: 1 addition & 2 deletions src/esp/core/managedContainers/ManagedContainerBase.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,7 @@ namespace managedContainers {
bool ManagedContainerBase::setLock(const std::string& objectHandle, bool lock) {
// if managed object does not currently exist then do not attempt to modify
// its lock state
if (!checkExistsWithMessage(objectHandle,
"<" + this->objectType_ + ">::setLock")) {
if (!checkExistsWithMessage(objectHandle, "setLock")) {
return false;
}
// if setting lock else clearing lock
Expand Down
Loading

0 comments on commit d924c81

Please sign in to comment.