textspatch.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. #include "textspatch.h"
  2. #include "LotroDat/LotroDat.h"
  3. #include "LotroDat/Database.h"
  4. #include <QUrlQuery>
  5. #include <QSet>
  6. #include <QThread>
  7. TextsPatch::TextsPatch(LotroDatManager *mgr, QObject *parent) : Patch("TextsPatch", mgr, parent)
  8. {
  9. connect(lotro_mgr_, &LotroDatManager::operationStarted, this, &TextsPatch::onLotroDatManagerOperationStarted);
  10. connect(lotro_mgr_, &LotroDatManager::operationFinished, this, &TextsPatch::onLotroDatManagerOperationFinished);
  11. connect(lotro_mgr_, &LotroDatManager::statusChanged, this, &TextsPatch::onLotroDatManagerStatusChanged);
  12. }
  13. void TextsPatch::checkForUpdates()
  14. {
  15. QUrlQuery query; // query for building GET-request aka patch-version
  16. foreach (QString db_name, databases_names) {
  17. query.addQueryItem(db_name, "100");
  18. }
  19. QUrl target_url;
  20. target_url.setUrl(Settings::getValue("Network/patch_updates_url").toString());
  21. target_url.setQuery(query);
  22. QByteArray target_array;
  23. Downloader downloader;
  24. downloader.setUrl(target_url);
  25. downloader.targetBytearray = &target_array;
  26. downloader.start();
  27. downloader.waitForDownloaded();
  28. if (target_array.isEmpty()) {
  29. qWarning() << *this << "Cannot check for updates, target_array is empty!";
  30. emit errorOccured(E_CHECKFORUPDATES, this, "");
  31. emit operationFinished(E_CHECKFORUPDATES, this);
  32. return;
  33. }
  34. QStringList patch_info = QString(target_array).split('|');
  35. if (patch_info.size() != databases_names.size()) {
  36. qCritical() << __FUNCTION__ << "Incorrect patches number! Data: " << patch_info;
  37. emit errorOccured(E_CHECKFORUPDATES, this, "");
  38. emit operationFinished(E_CHECKFORUPDATES, this);
  39. return;
  40. }
  41. for (int i = 0; i < databases_names.size(); ++i) {
  42. QStringList patch_data = patch_info[i].split(":::");
  43. if (patch_data.size() != 3) {
  44. qCritical() << __FUNCTION__ << "Incorrect patch entry size! Entry: " << patch_data;
  45. emit errorOccured(E_CHECKFORUPDATES, this, "");
  46. emit operationFinished(E_CHECKFORUPDATES, this);
  47. return;
  48. }
  49. QString patch_filename = Settings::getValue("General/PatchDownloadDir").toString() + "/" + QUrl(patch_data[0]).fileName();
  50. Settings::setValue("PatchDatabases/" + databases_names[i] + "/url", patch_data[0]);
  51. Settings::setValue("PatchDatabases/" + databases_names[i] + "/hashsum", patch_data[1]);
  52. Settings::setValue("PatchDatabases/" + databases_names[i] + "/datetime", patch_data[2]);
  53. Settings::setValue("PatchDatabases/" + databases_names[i] + "/path", patch_filename);
  54. }
  55. emit operationFinished(E_CHECKFORUPDATES, this);
  56. }
  57. void TextsPatch::download()
  58. {
  59. foreach (QString db_name, databases_names) {
  60. QString settings_prefix = "PatchDatabases/" + db_name;
  61. QString target_filename = QApplication::applicationDirPath() + "/" + Settings::getValue(settings_prefix + "/path").toString();
  62. qDebug() << patch_name_ << ": Checking if file " << target_filename << " matches its hashsum";
  63. if (FileSystem::fileHash(target_filename) == Settings::getValue(settings_prefix + "/hashsum").toString()) {
  64. qInfo() << *this << ": file " << target_filename << " is up-to-date, no need to download";
  65. continue;
  66. }
  67. FileSystem::createFilePath(target_filename);
  68. QFile* target_file = new QFile(target_filename);
  69. if (!target_file->open(QIODevice::ReadWrite | QIODevice::Truncate)) {
  70. qWarning() << *this << "Cannot open file " << target_filename;
  71. continue;
  72. }
  73. qInfo() << *this << ": beginning download of file " << target_filename;
  74. Settings::setValue("DatabaseNeedInstall/" + db_name, true);
  75. Downloader* downloader = new Downloader();
  76. downloader->setUrl(Settings::getValue(settings_prefix + "/url").toUrl());
  77. downloader->targetFile = target_file;
  78. connect(downloader, &Downloader::progressChanged, this, &TextsPatch::onDownloaderProgressChanged);
  79. connect(downloader, &Downloader::downloadFinished, this, &TextsPatch::onDownloaderFinished);
  80. downloaders_.insert(downloader);
  81. downloader->start();
  82. }
  83. if (downloaders_.empty()) {
  84. emit operationFinished(E_DOWNLOAD, this);
  85. }
  86. }
  87. void TextsPatch::install()
  88. {
  89. foreach (QString db_name, databases_names) {
  90. if (!Settings::getValue("DatabaseNeedInstall/" + db_name).toBool()) {
  91. continue;
  92. }
  93. ++elapsed_patches_to_install_;
  94. QMetaObject::invokeMethod(lotro_mgr_, "installPatch", Qt::QueuedConnection,
  95. Q_ARG(QString, getPatchName() + "_" + db_name),
  96. Q_ARG(QString, Settings::getValue("PatchDatabases/" + db_name + "/path").toString())
  97. );
  98. }
  99. if (elapsed_patches_to_install_ == 0) {
  100. emit operationFinished(E_INSTALL, this);
  101. }
  102. }
  103. void TextsPatch::activate()
  104. {
  105. emit operationFinished(E_ACTIVATE, this);
  106. return;
  107. QString operation_name = Settings::getValue("Components/fonts").toBool() ? "enableCategory" : "disableCategory";
  108. ++elapsed_patches_to_install_;
  109. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  110. Q_ARG(QString, getPatchName() + "_fonts"),
  111. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_FONTS_COMMON)
  112. );
  113. operation_name = Settings::getValue("Components/texts_main").toBool() ? "enableCategory" : "disableCategory";
  114. ++elapsed_patches_to_install_;
  115. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  116. Q_ARG(QString, getPatchName() + "_texts_main"),
  117. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_TEXTS_COMMON)
  118. );
  119. operation_name = Settings::getValue("Components/texts_items").toBool() ? "enableCategory" : "disableCategory";
  120. ++elapsed_patches_to_install_;
  121. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  122. Q_ARG(QString, getPatchName() + "_texts_items"),
  123. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_TEXTS_ITEMS)
  124. );
  125. operation_name = Settings::getValue("Components/texts_emotes").toBool() ? "enableCategory" : "disableCategory";
  126. ++elapsed_patches_to_install_;
  127. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  128. Q_ARG(QString, getPatchName() + "_texts_emotes"),
  129. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_TEXTS_EMOTES)
  130. );
  131. operation_name = Settings::getValue("Components/videos").toBool() ? "enableCategory" : "disableCategory";
  132. ++elapsed_patches_to_install_;
  133. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  134. Q_ARG(QString, getPatchName() + "_texts_videos_refs"),
  135. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_TEXTS_VIDEOS_REFS)
  136. );
  137. }
  138. void TextsPatch::onDownloaderProgressChanged(Downloader *, Downloader::Status)
  139. {
  140. Downloader::Status all_downloads_status;
  141. foreach (Downloader* downloader, downloaders_) {
  142. all_downloads_status = all_downloads_status + downloader->getDownloadStatus();
  143. }
  144. emit progressChanged(OperationProgress(all_downloads_status), this);
  145. }
  146. void TextsPatch::onDownloaderFinished(Downloader *ptr)
  147. {
  148. ptr->targetFile->close();
  149. ptr->targetFile->deleteLater();
  150. Downloader::Status all_downloads_status;
  151. for (const Downloader* downloader : downloaders_) {
  152. all_downloads_status = all_downloads_status + downloader->getDownloadStatus();
  153. }
  154. if (!all_downloads_status.running) {
  155. for (Downloader* downloader : downloaders_) {
  156. downloader->deleteLater();
  157. }
  158. downloaders_.clear();
  159. emit operationFinished(E_DOWNLOAD, this);
  160. }
  161. }
  162. void TextsPatch::onLotroDatManagerOperationFinished(QString operation_name, QVector<QVariant> args, bool successful)
  163. {
  164. if (args.size() == 0 || !args[0].toString().contains(getPatchName())) {
  165. return; // This means, that message from LotroManager is addressed to another patchset
  166. }
  167. if (operation_name.contains("installPatch")) {
  168. if (!successful) {
  169. qCritical() << *this << "Database " + args[0].toString() + " (" + args[1].toString() + ") was not installed due to Legacy core error.";
  170. emit errorOccured(E_INSTALL, this, "Database " + args[0].toString() + " (" + args[1].toString() + ") was not installed due to Legacy core error.");
  171. }
  172. QString db_name = args[0].toString().split('_').at(1);
  173. Settings::setValue("DatabaseNeedInstall/" + db_name, false);
  174. --elapsed_patches_to_install_;
  175. if (!elapsed_patches_to_install_) {
  176. is_being_patched_by_lotro_dat_manager_ = false;
  177. emit operationFinished(E_INSTALL, this);
  178. }
  179. }
  180. if (operation_name.contains("enableCategory") || operation_name.contains("disableCategory")) {
  181. if (!successful) {
  182. qCritical() << *this << "Error in patch " + args[0].toString() + ": activating category " + QString::number(args[1].toInt()) + " failed.";
  183. emit errorOccured(E_ACTIVATE, this, "Error in patch " + args[0].toString() + ": activating category " + QString::number(args[1].toInt()) + " failed.");
  184. }
  185. --elapsed_patches_to_install_;
  186. if (!elapsed_patches_to_install_) {
  187. is_being_patched_by_lotro_dat_manager_ = false;
  188. emit operationFinished(E_ACTIVATE, this);
  189. }
  190. }
  191. }
  192. void TextsPatch::onLotroDatManagerOperationStarted(QString, QVector<QVariant> args)
  193. {
  194. if (args.size() == 0 || !args[0].toString().contains(getPatchName())) {
  195. return;
  196. }
  197. is_being_patched_by_lotro_dat_manager_ = true;
  198. }
  199. void TextsPatch::onLotroDatManagerStatusChanged(LotroDatManager::Status status)
  200. {
  201. if (!is_being_patched_by_lotro_dat_manager_) {
  202. return;
  203. }
  204. emit progressChanged(OperationProgress(status), this);
  205. }