patchdownloader.cpp 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include "patchdownloader.h"
  2. #include "models/filesystem.h"
  3. #include <QApplication>
  4. #include <QDir>
  5. #include <QStringList>
  6. #include <QUrlQuery>
  7. #include <QVariant>
  8. #include <QDebug>
  9. PatchDownloader::PatchDownloader(QSettings *settings, QObject *parent) : QObject(parent), patch_download_dir(QApplication::applicationDirPath() + "/data")
  10. {
  11. app_settings = settings;
  12. active_downloads_number = 0;
  13. connect(&update_check_timer, &QTimer::timeout, this, &PatchDownloader::checkForUpdates);
  14. update_check_timer.setInterval(1000 * 60 * 5); // 5 minutes
  15. update_check_timer.start();
  16. }
  17. PatchDownloader::~PatchDownloader()
  18. {
  19. foreach (Downloader* downloader, downloads_list) {
  20. downloader->waitForDownloaded();
  21. downloader->deleteLater();
  22. }
  23. }
  24. QString PatchDownloader::getPredictedDownloadSizeFormatted()
  25. {
  26. double mbytes = 0;
  27. if (app_settings->value("patch_database/sound", "Disabled").toString() == "Enabled") {
  28. mbytes += 650;
  29. }
  30. if (app_settings->value("patch_database/text", "Disabled").toString() == "Enabled") {
  31. mbytes += 80;
  32. }
  33. if (app_settings->value("patch_database/image", "Disabled").toString() == "Enabled") {
  34. mbytes += 120;
  35. }
  36. if (app_settings->value("patch_database/loadscreen", "Disabled").toString() == "Enabled") {
  37. mbytes += 3;
  38. }
  39. if (app_settings->value("patch_database/texture", "Disabled").toString() == "Enabled") {
  40. mbytes += 4;
  41. }
  42. if (app_settings->value("patch_database/font", "Disabled").toString() == "Enabled") {
  43. mbytes += 1;
  44. }
  45. if (app_settings->value("patch_database/video", "Disabled").toString() == "Enabled") {
  46. mbytes += 2100;
  47. }
  48. QString unit = "Мб";
  49. if (mbytes > 1024) {
  50. mbytes /= 1024;
  51. unit = "Гб";
  52. }
  53. return QString::number(mbytes, 'f', 1) + " " + unit;
  54. }
  55. QString PatchDownloader::getDatabasePathByPatchName(QString name)
  56. {
  57. return patch_download_dir.absolutePath() + "/" + patch_data[name].url.fileName();
  58. }
  59. void PatchDownloader::checkForUpdates()
  60. {
  61. foreach (QString patch_name, all_patch_names) {
  62. if (downloads_list.contains(patch_name))
  63. continue;
  64. downloads_list[patch_name] = new Downloader();
  65. connect(downloads_list[patch_name], &Downloader::progressChanged, this, &PatchDownloader::onDownloaderProgressChanged);
  66. connect(downloads_list[patch_name], &Downloader::downloadFinished, this, &PatchDownloader::onDownloaderCompleted);
  67. }
  68. if (active_downloads_number > 0) {
  69. qDebug() << "PatchDownloader: downloads are not ready yet, passing checkForUpdates";
  70. return;
  71. }
  72. emit checkForUpdatesStarted();
  73. if (!updatePatchList()) {
  74. emit checkForUpdatesFinished();
  75. return;
  76. }
  77. removeOldPatchesFromDirecrory();
  78. addMissingPatchesToDownloadList();
  79. emit checkForUpdatesFinished();
  80. }
  81. void PatchDownloader::onDownloaderCompleted(Downloader *downloader_ptr)
  82. {
  83. if (downloader_ptr->targetFile) {
  84. downloader_ptr->targetFile->close();
  85. downloader_ptr->targetFile->deleteLater();
  86. }
  87. active_downloads_number--;
  88. if (active_downloads_number == 0) {
  89. emit downloadCompleted();
  90. }
  91. }
  92. void PatchDownloader::onDownloaderProgressChanged(Downloader*)
  93. {
  94. quint64 totalSize = 0;
  95. quint64 downloadedSize = 0;
  96. quint64 summary_speed = 0;
  97. quint64 time_elapsed = 0;
  98. foreach (Downloader* downloader, downloads_list) {
  99. totalSize += downloader->getBytesTotal();
  100. downloadedSize += downloader->getBytesDownloaded();
  101. if (downloader->getBytesTotal() != downloader->getBytesDownloaded()) {
  102. summary_speed += downloader->getSpeed();
  103. }
  104. }
  105. time_elapsed = (totalSize - downloadedSize) / qMax(quint64(1), summary_speed);
  106. emit progressChanged(downloadedSize, totalSize,
  107. Downloader::getSpeedFormatted(summary_speed),
  108. Downloader::getElapsedTimeFormatted(time_elapsed));
  109. }
  110. int PatchDownloader::versionFromPatchFilename(QString filename)
  111. {
  112. int version = 0;
  113. for (int i = filename.indexOf("_v") + 2; i < filename.indexOf("_v") + 7; i += 2) {
  114. version = version * 10 + (filename.at(i).toLatin1() - '0');
  115. }
  116. return version;
  117. }
  118. bool PatchDownloader::updatePatchList()
  119. {
  120. QUrlQuery query; // query for building GET-request aka patch-version
  121. foreach (QString patch_name, all_patch_names) {
  122. query.addQueryItem(patch_name, "100");
  123. }
  124. QUrl target_url;
  125. target_url.setUrl("http://translate.lotros.ru/groupware/check_updates");
  126. target_url.setQuery(query);
  127. QByteArray target_array;
  128. Downloader downloader;
  129. downloader.setUrl(target_url);
  130. downloader.targetBytearray = &target_array;
  131. downloader.start();
  132. downloader.waitForDownloaded();
  133. if (target_array.isEmpty()) {
  134. qDebug() << __FUNCTION__ << "Cannot download, target_array is empty!";
  135. emit getPatchListError();
  136. return false;
  137. }
  138. QStringList entry_list = QString(target_array).split('|');
  139. if (entry_list.size() != all_patch_names.size()) {
  140. qDebug() << __FUNCTION__ << "Entry list size is not equal to patch names size!" << QString(target_array);
  141. emit getPatchListError();
  142. return false;
  143. }
  144. for (int i = 0; i < entry_list.size(); ++i) {
  145. QStringList current_patch_data = entry_list[i].split(":::");
  146. if (current_patch_data.size() != 3) {
  147. qDebug() << __FUNCTION__ << "Incorrect patch entry size! Entry: " << entry_list[i];
  148. emit getPatchListError();
  149. return false;
  150. }
  151. patch_data[all_patch_names[i]] = {current_patch_data[0], current_patch_data[1], current_patch_data[2], all_patch_names[i]};
  152. }
  153. return true;
  154. }
  155. bool PatchDownloader::removeOldPatchesFromDirecrory()
  156. {
  157. QStringList actual_hash_list;
  158. foreach (Patch patch, patch_data) {
  159. actual_hash_list.append(patch.md5_hash);
  160. }
  161. qDebug() << "Removing old patches. Current hash list " << actual_hash_list;
  162. QStringList paths = patch_download_dir.entryList(QDir::Files);
  163. foreach (QString filename, paths) {
  164. QString hash = FileSystem::fileHash(patch_download_dir.absolutePath() + "/" + filename, QCryptographicHash::Md5);
  165. if (!actual_hash_list.contains(hash)) {
  166. qDebug() << "File " << filename << " with hash " << hash << "seems outdated, deleting!";
  167. if (!QFile::remove(patch_download_dir.absolutePath() + "/" +filename)) {
  168. qDebug() << __FUNCTION__ << "Unable to remove file " << filename;
  169. emit removeFileError(patch_download_dir.absolutePath() + "/" + filename);
  170. }
  171. }
  172. }
  173. return true;
  174. }
  175. bool PatchDownloader::addMissingPatchesToDownloadList()
  176. {
  177. QDir dir(patch_download_dir);
  178. if (!dir.exists())
  179. QDir().mkdir(patch_download_dir.absolutePath());
  180. bool download_started = false;
  181. foreach (Patch patch, patch_data) {
  182. QString patch_filepath = patch_download_dir.absolutePath() + "/" + patch.url.fileName();
  183. qDebug() << "Patch" << patch.name << "is marked as" << app_settings->value("patch_databases/" + patch.name, "Disabled");
  184. if (app_settings->value("patch_databases/" + patch.name, "Disabled").toString() != "Enabled")
  185. continue;
  186. if (FileSystem::fileExists(patch_filepath)) {
  187. if (FileSystem::fileHash(patch_filepath, QCryptographicHash::Md5) == patch.md5_hash)
  188. continue;
  189. if (!QFile::remove(patch_filepath)) {
  190. qDebug() << __FUNCTION__ << "Unable to remove file " << patch_filepath;
  191. emit removeFileError(patch_filepath);
  192. continue;
  193. }
  194. }
  195. if (!download_started) {
  196. download_started = true;
  197. qDebug() << "Started downloads of PatchDownloader!";
  198. emit downloadStarted();
  199. }
  200. qDebug() << "Starting download of file " << patch_filepath << " from url " << patch.url;
  201. downloads_list[patch.name]->setUrl(patch.url);
  202. downloads_list[patch.name]->targetFile = new QFile(patch_filepath, downloads_list[patch.name]);
  203. downloads_list[patch.name]->targetFile->open(QIODevice::ReadWrite);
  204. downloads_list[patch.name]->start();
  205. active_downloads_number++;
  206. }
  207. return true;
  208. }