DatPatcher.cpp 8.6 KB

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