DatExporter.cpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. #include <DatFile.h>
  2. #include <DatSubsystems/DatExporter.h>
  3. #include <DatSubsystems/DatFileSystem.h>
  4. #include <BinaryData.h>
  5. #include <SubfileData.h>
  6. #include <SubFile.h>
  7. #include <string>
  8. #include <fstream>
  9. #include <algorithm>
  10. namespace LOTRO_DAT {
  11. DatExporter::DatExporter(DatFile *datFilePtr) : dat(datFilePtr) {
  12. }
  13. /*!
  14. * \author Gi1dor
  15. * \date 05.07.2018
  16. * Извлечение в папку всех файлов, имеющих заданный тип
  17. * \param[in] type Тип файлов
  18. * \param[in] output_directory_path путь к папке, куда будут записаны файлы. Имя экспортируемого файла - file_id.ext, где file_id - id файла, ext - его расширение
  19. *
  20. * \returns DatOperationResult.value - количество извлечённых файлов
  21. */
  22. DatOperationResult<int> DatExporter::ExtractAllFilesByType(const FILE_TYPE &type, std::string output_directory_path) {
  23. int success_exported = 0;
  24. int iterated_files = 0;
  25. dat->GetStatusModule().SetStatus(DatStatus::E_EXTRACTING);
  26. dat->GetStatusModule().SetPercentage(0);
  27. dat->GetStatusModule().SetDebugMessage("Starting extracting files of type " + std::to_string(type));
  28. LOG(INFO) << "Extracting files by type " + std::to_string(type) + "to database...";
  29. auto operation = dat->GetFileSystem().PerformOperationOnAllFiles([&iterated_files, &success_exported, this, type, &output_directory_path](std::shared_ptr<SubFile>& file) -> void {
  30. iterated_files++;
  31. dat->GetStatusModule().SetPercentage(iterated_files * 100u / dat->GetFileSystem().GetInitialisedFilesNumber());
  32. if ((file->FileType() & type) == NO_TYPE)
  33. return;
  34. SubfileData export_file = file->PrepareForExport(dat->GetFileSystem().GetFileData(*file, 8).value);
  35. dat->GetStatusModule().SetDebugMessage("Extracting file " + std::to_string(file->file_id()) + " (Files by type -> directory)");
  36. if (export_file.Empty())
  37. return;
  38. bool result;
  39. if (type == TEXT)
  40. result = WriteStringToFile(export_file.text_data, output_directory_path + std::to_string(file->file_id()) + file->Extension()).result;
  41. else
  42. result = export_file.binary_data.WriteToFile(output_directory_path + std::to_string(file->file_id()) + file->Extension());
  43. if (result)
  44. success_exported++;
  45. });
  46. LOG(INFO) << "Extracting files: successfully exported " << success_exported << " files";
  47. dat->GetStatusModule().ClearAll();
  48. return DatOperationResult<int>(success_exported, SUCCESS);
  49. }
  50. /*!
  51. * \author Gi1dor
  52. * \date 05.07.2018
  53. * Извлечение файла по его id
  54. * \param[in] file_id id файла
  55. * \param[in] output_filename - имя файла, куда будет записаны данные.
  56. */
  57. DatOperationResult<> DatExporter::ExtractFileById(long long file_id, std::string output_filename) {
  58. dat->GetStatusModule().SetStatus(DatStatus::E_EXTRACTING);
  59. dat->GetStatusModule().SetPercentage(0);
  60. dat->GetStatusModule().SetDebugMessage("Extracting file " + std::to_string(file_id) + " to file " + output_filename);
  61. auto operation_GetFile = dat->GetFileSystem().GetFile(file_id);
  62. if (operation_GetFile.result == ERROR) {
  63. dat->GetStatusModule().ClearAll();
  64. return DatOperationResult<>(ERROR, "EXTRACTFILEBYID: File not found! id = " + std::to_string(file_id));
  65. }
  66. std::shared_ptr<SubFile> file = operation_GetFile.value;
  67. auto operation_GetFileData = dat->GetFileSystem().GetFileData(*file, 8);
  68. if (operation_GetFileData.result == ERROR) {
  69. dat->GetStatusModule().ClearAll();
  70. return DatOperationResult<>(ERROR, "EXTRACTFILEBYID: Can't get data for id = " + std::to_string(file_id));
  71. }
  72. SubfileData export_data = file->PrepareForExport(operation_GetFileData.value);
  73. bool result;
  74. if (file->FileType() == TEXT)
  75. result = WriteStringToFile(export_data.text_data, output_filename + file->Extension()).result;
  76. else
  77. result = export_data.binary_data.WriteToFile(output_filename + file->Extension());
  78. dat->GetStatusModule().ClearAll();
  79. if (!result)
  80. return DatOperationResult<>(ERROR, "EXTRACTFILEBYID: Cannot write to file" + output_filename + file->Extension());
  81. return DatOperationResult<>(SUCCESS);
  82. }
  83. /*!
  84. * \author Gi1dor
  85. * \date 05.07.2018
  86. * Извлечение в базу данных всех файлов, имеющих заданный тип
  87. * \param[in] type Тип файлов
  88. * \param[in] db Указатель на объект базы данных
  89. *
  90. * \returns DatOperationResult.value - количество извлечённых файлов
  91. */
  92. DatOperationResult<int> DatExporter::ExtractAllFilesByType(const FILE_TYPE &type, Database *db) {
  93. if (!db)
  94. return DatOperationResult<int>(0, ERROR, "EXTRACTALLBYTYPETODB: database is nullptr");
  95. dat->GetStatusModule().SetStatus(DatStatus::E_EXTRACTING);
  96. dat->GetStatusModule().SetPercentage(0);
  97. dat->GetStatusModule().SetDebugMessage("Initialising extraction of files by type " + std::to_string(type) + "to database...");
  98. LOG(INFO) << "Extracting files by type " + std::to_string(type) + "to database...";
  99. int success_exported = 0;
  100. int iterated_files = 0;
  101. auto operation = dat->GetFileSystem().PerformOperationOnAllFiles([&success_exported, &iterated_files, this, type, db](std::shared_ptr<SubFile>& file) {
  102. iterated_files++;
  103. dat->GetStatusModule().SetPercentage(iterated_files * 100u / dat->GetFileSystem().GetInitialisedFilesNumber());
  104. if ((file->FileType() & type) == NO_TYPE)
  105. return;
  106. dat->GetStatusModule().SetDebugMessage("Extracting file " + std::to_string(file->file_id()) + " (Files by type -> database)");
  107. SubfileData export_file = file->PrepareForExport(dat->GetFileSystem().GetFileData(*file, 8).value);
  108. bool result = db->PushFile(export_file);
  109. if (result)
  110. success_exported++;
  111. });
  112. LOG(INFO) << "Extracting files: successfully exported " << success_exported << " files";
  113. dat->GetStatusModule().ClearAll();
  114. return DatOperationResult<int>(success_exported, SUCCESS);
  115. }
  116. /*!
  117. * \author Gi1dor
  118. * \date 05.07.2018
  119. * Запись строки в формате UTF-16 в файл
  120. * \param[in] str Строка для записи в файл
  121. * \param[in] path Путь к файлу, в который строка будет записана
  122. *
  123. * \warning Путь к файлу должен быть корректным (папки должны существовать). Если и сам файл существует, то он будет перезаписан
  124. */
  125. DatOperationResult<> DatExporter::WriteStringToFile(const std::u16string &str, const std::string &path) {
  126. std::basic_ofstream<char16_t> output_stream(path, std::ios::out);
  127. if (!output_stream.is_open())
  128. return DatOperationResult<>(ERROR, "WRITESTRINGTOFILE: cant open file " + path);
  129. output_stream << str;
  130. output_stream.close();
  131. return DatOperationResult<>();
  132. }
  133. };