DatPatcher.cpp 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. #include <DatSubsystems/DatPatcher.h>
  2. #include <SubfileData.h>
  3. #include <DatFile.h>
  4. #include <yaml-cpp/yaml.h>
  5. namespace LOTRO_DAT {
  6. DatPatcher::DatPatcher(DatFile *datFilePtr) : dat(datFilePtr) {
  7. }
  8. /*!
  9. * \Author Gi1dor
  10. * \date 07.07.2018
  11. * Обновление файла. Записывает данные файла в dat контейнер, используя информацию в SubfileData. Если ранее файл не существовал - он не будет создан (вернёт ошибку).
  12. * \warning В процессе применения локаль будет автоматически переключена в положение PATCHED
  13. * \param[in] data Новые данные файла
  14. * \param[in] single_file Флаг, который означает, что применяемый патч состоит из одного файла.
  15. * Если true, то функция будет управлять состоянием DatStatus и обновлять его.
  16. * Иначе эта обязанность делегируется вызывающей функции
  17. */
  18. DatOperationResult<> DatPatcher::PatchFile(const SubfileData &data, bool single_file) {
  19. auto file_id = data.options["fid"].as<long long>();
  20. if (single_file) {
  21. dat->GetStatusModule().SetStatus(DatStatus::E_PATCHING);
  22. dat->GetStatusModule().SetPercentage(0);
  23. }
  24. dat->GetStatusModule().SetDebugMessage("Patching file with id " + std::to_string(file_id));
  25. auto getfile_operation = dat->GetFileSystem().GetFile(file_id);
  26. if (getfile_operation.result == ERROR) {
  27. if (single_file)
  28. dat->GetStatusModule().ClearAll();
  29. return DatOperationResult<>(ERROR,
  30. "PATCHSUBFILEDATA: Unable to find file with id " + std::to_string(file_id));
  31. }
  32. auto &file = getfile_operation.value;
  33. // If file has inactive category, then we should set it to patched state in order to commit patch and
  34. // then in ApplyFilePatch() function, if new category is still inactive, return dictionary to its original state;
  35. if (dat->GetLocaleManager().CategoryIsInactive(file->category) != 0) {
  36. auto operation = dat->GetLocaleManager().GetLocaleFile(file->file_id(), DatLocaleManager::PATCHED);
  37. if (operation.result == SUCCESS)
  38. dat->GetFileSystem().UpdateFileInfo(operation.value);
  39. }
  40. if (data.options["cat"].IsDefined())
  41. file->category = data.options["cat"].as<long long>();
  42. else
  43. file->category = 1;
  44. auto getdata_operation = dat->GetFileSystem().GetFileData(*file, 0);
  45. if (getdata_operation.result == ERROR) {
  46. if (single_file)
  47. dat->GetStatusModule().ClearAll();
  48. return DatOperationResult<>(ERROR,
  49. "PATCHSUBFILEDATA: can't get file data for id = " + std::to_string(file_id));
  50. }
  51. auto &file_data = getdata_operation.value;
  52. BinaryData patch_data = file->MakeForImport(file_data, data);
  53. auto result = ApplyFilePatch(file, patch_data);
  54. if (single_file)
  55. dat->GetStatusModule().ClearAll();
  56. if (result.result == ERROR)
  57. return DatOperationResult<>(ERROR,
  58. "PATCHSUBFILEDATA: applyfilepatch failed for id = " + std::to_string(file_id));
  59. return DatOperationResult<>(SUCCESS);
  60. }
  61. /*!
  62. * \Author Gi1dor
  63. * \date 07.07.2018
  64. * Обновление всех файлов в Database.
  65. * \warning В процессе применения локаль будет автоматически переключена в положение PATCHED
  66. * \param[in] db Указатель на базу данных. База должна быть проинициализирована
  67. */
  68. DatOperationResult<int> DatPatcher::PatchAllDatabase(Database *db) {
  69. if (!db)
  70. return DatOperationResult<int>(0, ERROR, "PATCHALLDATABASE: db is nullptr");
  71. dat->GetStatusModule().SetStatus(DatStatus::E_PATCHING);
  72. dat->GetStatusModule().SetPercentage(0);
  73. dat->GetStatusModule().SetDebugMessage("Patching database...");
  74. SubfileData data;
  75. data = db->GetNextFile();
  76. int successfully_patched = 0;
  77. unsigned db_rows = db->CountRows();
  78. for (unsigned i = 0; i < db_rows; ++i) {
  79. auto operation = PatchFile(data, false);
  80. if (operation.result == SUCCESS)
  81. successfully_patched++;
  82. dat->GetStatusModule().SetPercentage(i * 100 / db_rows);
  83. data = db->GetNextFile();
  84. }
  85. LOG(INFO) << "Database import success: patched " + std::to_string(successfully_patched) + " out of " +
  86. std::to_string(db_rows) + " files";
  87. dat->GetStatusModule().ClearAll();
  88. return DatOperationResult<int>(successfully_patched, SUCCESS);
  89. }
  90. /*!
  91. * \Author Gi1dor
  92. * \date 07.07.2018
  93. * Функция, вызываемая функцией PatchFile, отвечающая за корректную запись файла в dat контейнер и обновление
  94. * информации о нём в файловой системе
  95. * \param[in] file Указатель на объект файла в файловой системе (полученный через DatFileSystem::GetFileInfo)
  96. * \param[in] data Бинарные данные собранного нового файла, готовые к записи в dat
  97. */
  98. DatOperationResult<> DatPatcher::ApplyFilePatch(std::shared_ptr<SubFile> file, BinaryData &data) {
  99. long long file_id = file->file_id();
  100. if (dat->GetLocaleManager().GetCurrentLocale() != DatLocaleManager::PATCHED) {
  101. LOG(INFO) << "Changing locale to PATCHED(RU) in order to patch file";
  102. dat->GetLocaleManager().SetLocale(DatLocaleManager::PATCHED);
  103. }
  104. //LOG(INFO) << "Patching file with id = " << file_id;
  105. if (dat->GetLocaleManager().GetLocaleFile(file_id, DatLocaleManager::ORIGINAL).result == ERROR)
  106. dat->GetLocaleManager().UpdateLocaleFile(DatLocaleManager::ORIGINAL, SubFile(*file));
  107. SubFile new_file = SubFile(*file);
  108. if (dat->GetLocaleManager().GetLocaleFile(file_id, DatLocaleManager::PATCHED).result == ERROR
  109. || data.size() > file->block_size()) {
  110. new_file.file_offset_ = dat->GetIO().file_size;
  111. new_file.block_size_ = std::max(data.size(), 256u);
  112. dat->GetIO().file_size += new_file.block_size_ + 8;
  113. }
  114. new_file.file_size_ = data.size() - 8;
  115. data.Append(BinaryData::FromNumber<4>(0), 0); // set additional fragments count to zero
  116. if (file_id != data.ToNumber<4>(8))
  117. LOG(WARNING) << "Created data's file_id " << file_id << "doesn't match to original: "
  118. << data.ToNumber<4>(8);
  119. auto operation = dat->GetIO().WriteData(data, data.size(), new_file.file_offset());
  120. if (operation.result == ERROR)
  121. return DatOperationResult<>(ERROR, "APPLYPATCHFILE: Unable to write data for file with id " +
  122. std::to_string(file_id));
  123. dat->GetFileSystem().UpdateFileInfo(new_file);
  124. dat->GetLocaleManager().UpdateLocaleFile(DatLocaleManager::PATCHED, new_file);
  125. // If file category is inactive, then return file header data in dictionary to original state
  126. if (dat->GetLocaleManager().CategoryIsInactive(new_file.category))
  127. dat->GetFileSystem().UpdateFileInfo(
  128. dat->GetLocaleManager().GetLocaleFile(file_id, DatLocaleManager::ORIGINAL).value);
  129. dat->GetLocaleManager().UpdateCategory(file_id, new_file.category);
  130. //LOG(INFO) << "Successfully patched file with id = " << file_id;
  131. return DatOperationResult<>(SUCCESS);
  132. }
  133. };