AdapterLoaders.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /*
  2. * AdapterLoaders.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "AdapterLoaders.h"
  12. #include "Filesystem.h"
  13. #include "../json/JsonNode.h"
  14. VCMI_LIB_NAMESPACE_BEGIN
  15. CMappedFileLoader::CMappedFileLoader(const std::string & mountPoint, const JsonNode &config)
  16. {
  17. for(auto entry : config.Struct())
  18. {
  19. //fileList[ResourcePath(mountPoint + entry.first)] = ResourcePath(mountPoint + entry.second.String());
  20. fileList.emplace(ResourcePath(mountPoint + entry.first), ResourcePath(mountPoint + entry.second.String()));
  21. }
  22. }
  23. std::unique_ptr<CInputStream> CMappedFileLoader::load(const ResourcePath & resourceName) const
  24. {
  25. return CResourceHandler::get()->load(fileList.at(resourceName));
  26. }
  27. bool CMappedFileLoader::existsResource(const ResourcePath & resourceName) const
  28. {
  29. return fileList.count(resourceName) != 0;
  30. }
  31. std::string CMappedFileLoader::getMountPoint() const
  32. {
  33. return ""; // does not have any meaning with this type of data source
  34. }
  35. std::optional<boost::filesystem::path> CMappedFileLoader::getResourceName(const ResourcePath & resourceName) const
  36. {
  37. return CResourceHandler::get()->getResourceName(fileList.at(resourceName));
  38. }
  39. std::unordered_set<ResourcePath> CMappedFileLoader::getFilteredFiles(std::function<bool(const ResourcePath &)> filter) const
  40. {
  41. std::unordered_set<ResourcePath> foundID;
  42. for(const auto & file : fileList)
  43. {
  44. if (filter(file.first))
  45. foundID.insert(file.first);
  46. }
  47. return foundID;
  48. }
  49. std::string CMappedFileLoader::getFullFileURI(const ResourcePath& resourceName) const
  50. {
  51. return CResourceHandler::get()->getFullFileURI(fileList.at(resourceName));
  52. }
  53. std::time_t CMappedFileLoader::getLastWriteTime(const ResourcePath& resourceName) const
  54. {
  55. return CResourceHandler::get()->getLastWriteTime(fileList.at(resourceName));
  56. }
  57. CFilesystemList::CFilesystemList()
  58. {
  59. }
  60. CFilesystemList::~CFilesystemList()
  61. {
  62. }
  63. std::unique_ptr<CInputStream> CFilesystemList::load(const ResourcePath & resourceName) const
  64. {
  65. // load resource from last loader that have it (last overridden version)
  66. for(const auto & loader : boost::adaptors::reverse(loaders))
  67. if (loader->existsResource(resourceName))
  68. return loader->load(resourceName);
  69. throw std::runtime_error("Resource with name " + resourceName.getName() + " and type "
  70. + EResTypeHelper::getEResTypeAsString(resourceName.getType()) + " wasn't found.");
  71. }
  72. bool CFilesystemList::existsResource(const ResourcePath & resourceName) const
  73. {
  74. for(const auto & loader : loaders)
  75. if (loader->existsResource(resourceName))
  76. return true;
  77. return false;
  78. }
  79. std::string CFilesystemList::getMountPoint() const
  80. {
  81. return "";
  82. }
  83. std::optional<boost::filesystem::path> CFilesystemList::getResourceName(const ResourcePath & resourceName) const
  84. {
  85. if (existsResource(resourceName))
  86. return getResourcesWithName(resourceName).back()->getResourceName(resourceName);
  87. return std::optional<boost::filesystem::path>();
  88. }
  89. std::set<boost::filesystem::path> CFilesystemList::getResourceNames(const ResourcePath & resourceName) const
  90. {
  91. std::set<boost::filesystem::path> paths;
  92. for(auto& loader : getResourcesWithName(resourceName))
  93. {
  94. auto rn = loader->getResourceName(resourceName);
  95. if(rn)
  96. {
  97. paths.insert(rn->string());
  98. }
  99. }
  100. return paths;
  101. }
  102. void CFilesystemList::updateFilteredFiles(std::function<bool(const std::string &)> filter) const
  103. {
  104. for(const auto & loader : loaders)
  105. loader->updateFilteredFiles(filter);
  106. }
  107. std::unordered_set<ResourcePath> CFilesystemList::getFilteredFiles(std::function<bool(const ResourcePath &)> filter) const
  108. {
  109. std::unordered_set<ResourcePath> ret;
  110. for(const auto & loader : loaders)
  111. for(const auto & entry : loader->getFilteredFiles(filter))
  112. ret.insert(entry);
  113. return ret;
  114. }
  115. bool CFilesystemList::createResource(const std::string & filename, bool update)
  116. {
  117. logGlobal->trace("Creating %s", filename);
  118. for (auto & loader : boost::adaptors::reverse(loaders))
  119. {
  120. if (writeableLoaders.count(loader.get()) != 0 // writeable,
  121. && loader->createResource(filename, update)) // successfully created
  122. {
  123. // Check if resource was created successfully. Possible reasons for this to fail
  124. // a) loader failed to create resource (e.g. read-only FS)
  125. // b) in update mode, call with filename that does not exists
  126. assert(load(ResourcePath(filename)));
  127. logGlobal->trace("Resource created successfully");
  128. return true;
  129. }
  130. }
  131. logGlobal->trace("Failed to create resource");
  132. return false;
  133. }
  134. std::vector<const ISimpleResourceLoader *> CFilesystemList::getResourcesWithName(const ResourcePath & resourceName) const
  135. {
  136. std::vector<const ISimpleResourceLoader *> ret;
  137. for(const auto & loader : loaders)
  138. boost::range::copy(loader->getResourcesWithName(resourceName), std::back_inserter(ret));
  139. return ret;
  140. }
  141. void CFilesystemList::addLoader(std::unique_ptr<ISimpleResourceLoader> loader, bool writeable)
  142. {
  143. if (writeable)
  144. writeableLoaders.insert(loader.get());
  145. loaders.push_back(std::move(loader));
  146. }
  147. bool CFilesystemList::removeLoader(ISimpleResourceLoader * loader)
  148. {
  149. for(auto loaderIterator = loaders.begin(); loaderIterator != loaders.end(); ++loaderIterator)
  150. {
  151. if(loaderIterator->get() == loader)
  152. {
  153. loaders.erase(loaderIterator);
  154. writeableLoaders.erase(loader);
  155. return true;
  156. }
  157. }
  158. return false;
  159. }
  160. std::string CFilesystemList::getFullFileURI(const ResourcePath& resourceName) const
  161. {
  162. for (const auto& loader : boost::adaptors::reverse(loaders))
  163. if (loader->existsResource(resourceName))
  164. return loader->getFullFileURI(resourceName);
  165. throw std::runtime_error("Resource with name " + resourceName.getName() + " and type "
  166. + EResTypeHelper::getEResTypeAsString(resourceName.getType()) + " wasn't found.");
  167. }
  168. std::time_t CFilesystemList::getLastWriteTime(const ResourcePath& resourceName) const
  169. {
  170. for (const auto& loader : boost::adaptors::reverse(loaders))
  171. if (loader->existsResource(resourceName))
  172. return loader->getLastWriteTime(resourceName);
  173. throw std::runtime_error("Resource with name " + resourceName.getName() + " and type "
  174. + EResTypeHelper::getEResTypeAsString(resourceName.getType()) + " wasn't found.");
  175. }
  176. VCMI_LIB_NAMESPACE_END