2
0

AdapterLoaders.cpp 5.1 KB

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