DatPatcher.cpp 7.6 KB

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