patchdownloader.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. #include "patchdownloader.h"
  2. #include "models/filesystem.h"
  3. #include "models/settings.h"
  4. #include <QApplication>
  5. #include <QDir>
  6. #include <QStringList>
  7. #include <QUrlQuery>
  8. #include <QVariant>
  9. #include <QDebug>
  10. Q_DECLARE_METATYPE(QList<PatchDownloadData>)
  11. PatchDownloader::PatchDownloader(QObject *parent) : QObject(parent), patch_download_dir(QApplication::applicationDirPath() + "/data")
  12. {
  13. qRegisterMetaType<QList<PatchDownloadData>>();
  14. active_downloads_number = 0;
  15. connect(&update_check_timer, &QTimer::timeout, this, &PatchDownloader::checkForUpdates);
  16. update_check_timer.setInterval(1000 * 60 * 5); // 5 minutes
  17. update_check_timer.start();
  18. }
  19. PatchDownloader::~PatchDownloader()
  20. {
  21. foreach (Downloader* downloader, downloads_list) {
  22. downloader->waitForDownloaded();
  23. downloader->deleteLater();
  24. }
  25. }
  26. QString PatchDownloader::getPredictedDownloadSizeFormatted()
  27. {
  28. double mbytes = 0;
  29. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  30. mbytes += 650;
  31. }
  32. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  33. mbytes += 80;
  34. }
  35. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  36. mbytes += 120;
  37. }
  38. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  39. mbytes += 3;
  40. }
  41. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  42. mbytes += 4;
  43. }
  44. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  45. mbytes += 1;
  46. }
  47. if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") {
  48. mbytes += 2100;
  49. }
  50. QString unit = "Мб";
  51. if (mbytes > 1024) {
  52. mbytes /= 1024;
  53. unit = "Гб";
  54. }
  55. return QString::number(mbytes, 'f', 1) + " " + unit;
  56. }
  57. QString PatchDownloader::getDatabasePathByPatchName(QString name)
  58. {
  59. return patch_download_dir.absolutePath() + "/" + patch_data[name].url.fileName();
  60. }
  61. void PatchDownloader::checkForUpdates()
  62. {
  63. foreach (QString patch_name, all_patch_names) {
  64. if (downloads_list.contains(patch_name))
  65. continue;
  66. downloads_list[patch_name] = new Downloader();
  67. connect(downloads_list[patch_name], &Downloader::progressChanged, this, &PatchDownloader::onDownloaderProgressChanged);
  68. connect(downloads_list[patch_name], &Downloader::downloadFinished, this, &PatchDownloader::onDownloaderCompleted);
  69. }
  70. if (active_downloads_number > 0) {
  71. qDebug() << "PatchDownloader: downloads are not ready yet, passing checkForUpdates";
  72. return;
  73. }
  74. emit checkForUpdatesStarted();
  75. if (!updatePatchList()) {
  76. emit checkForUpdatesFinished();
  77. return;
  78. }
  79. removeOldPatchesFromDirecrory();
  80. DownloadMissingPatches();
  81. qDebug() << "Finished checking for updates!";
  82. emit checkForUpdatesFinished();
  83. }
  84. void PatchDownloader::onDownloaderCompleted(Downloader *downloader_ptr)
  85. {
  86. if (downloader_ptr->targetFile) {
  87. downloader_ptr->targetFile->close();
  88. downloader_ptr->targetFile->deleteLater();
  89. }
  90. QString patch_finished_download_name = "none";
  91. foreach (QString patch_name, downloads_list.keys()) {
  92. if (downloads_list[patch_name] == downloader_ptr)
  93. patch_finished_download_name = patch_name;
  94. }
  95. if (patch_finished_download_name != "none") {
  96. QString patch_filepath = patch_download_dir.absolutePath() + "/" + patch_data[patch_finished_download_name].url.fileName();
  97. Settings::setValue("DatabaseDownloadDate/" + patch_data[patch_finished_download_name].name, QDate::currentDate().toString("dd.MM.yyyy"));
  98. Settings::setValue("DatabaseApplied/" + patch_data[patch_finished_download_name].name, "False");
  99. Settings::setValue("DatabasePath/" + patch_data[patch_finished_download_name].name, patch_filepath);
  100. emit changePatchStatus(patch_finished_download_name, "Загрузка завершена");
  101. }
  102. active_downloads_number--;
  103. if (active_downloads_number == 0) {
  104. emit downloadCompleted();
  105. }
  106. }
  107. void PatchDownloader::onDownloaderProgressChanged(Downloader*)
  108. {
  109. quint64 totalSize = 0;
  110. quint64 downloadedSize = 0;
  111. quint64 summary_speed = 0;
  112. quint64 time_elapsed = 0;
  113. foreach (Downloader* downloader, downloads_list) {
  114. totalSize += downloader->getBytesTotal();
  115. downloadedSize += downloader->getBytesDownloaded();
  116. if (downloader->getBytesTotal() != downloader->getBytesDownloaded()) {
  117. summary_speed += downloader->getSpeed();
  118. }
  119. }
  120. time_elapsed = (totalSize - downloadedSize) / qMax(quint64(1), summary_speed);
  121. QList<PatchDownloadData> download_data;
  122. download_data << PatchDownloadData({"general", downloadedSize, totalSize, Downloader::getSpeedFormatted(summary_speed), Downloader::getElapsedTimeFormatted(time_elapsed)});
  123. foreach (QString patch_name, downloads_list.keys()) {
  124. Downloader* patch_download = downloads_list[patch_name];
  125. if (patch_download->isStarted()) {
  126. download_data << PatchDownloadData({patch_name, patch_download->getBytesDownloaded(), patch_download->getBytesTotal(),
  127. Downloader::getSpeedFormatted(patch_download->getSpeed()),
  128. Downloader::getElapsedTimeFormatted(patch_download->getElapsedTime())});
  129. }
  130. }
  131. emit progressChanged(download_data);
  132. }
  133. int PatchDownloader::versionFromPatchFilename(QString filename)
  134. {
  135. int version = 0;
  136. for (int i = filename.indexOf("_v") + 2; i < filename.indexOf("_v") + 7; i += 2) {
  137. version = version * 10 + (filename.at(i).toLatin1() - '0');
  138. }
  139. return version;
  140. }
  141. bool PatchDownloader::updatePatchList()
  142. {
  143. QUrlQuery query; // query for building GET-request aka patch-version
  144. foreach (QString patch_name, all_patch_names) {
  145. query.addQueryItem(patch_name, "100");
  146. }
  147. QUrl target_url;
  148. target_url.setUrl("http://translate.lotros.ru/groupware/check_updates");
  149. target_url.setQuery(query);
  150. QByteArray target_array;
  151. Downloader downloader;
  152. downloader.setUrl(target_url);
  153. downloader.targetBytearray = &target_array;
  154. downloader.start();
  155. downloader.waitForDownloaded();
  156. if (target_array.isEmpty()) {
  157. qDebug() << __FUNCTION__ << "Cannot download, target_array is empty!";
  158. emit getPatchListError();
  159. return false;
  160. }
  161. QStringList entry_list = QString(target_array).split('|');
  162. if (entry_list.size() != all_patch_names.size()) {
  163. qDebug() << __FUNCTION__ << "Entry list size is not equal to patch names size!" << QString(target_array);
  164. emit getPatchListError();
  165. return false;
  166. }
  167. for (int i = 0; i < entry_list.size(); ++i) {
  168. QStringList current_patch_data = entry_list[i].split(":::");
  169. if (current_patch_data.size() != 3) {
  170. qDebug() << __FUNCTION__ << "Incorrect patch entry size! Entry: " << entry_list[i];
  171. emit getPatchListError();
  172. return false;
  173. }
  174. patch_data[all_patch_names[i]] = {current_patch_data[0], current_patch_data[1], current_patch_data[2], all_patch_names[i]};
  175. }
  176. return true;
  177. }
  178. bool PatchDownloader::removeOldPatchesFromDirecrory()
  179. {
  180. QStringList actual_hash_list;
  181. foreach (Patch patch, patch_data) {
  182. actual_hash_list.append(patch.md5_hash);
  183. }
  184. qDebug() << "Removing old patches. Current hash list " << actual_hash_list;
  185. QStringList paths = patch_download_dir.entryList(QDir::Files);
  186. foreach (QString filename, paths) {
  187. QString hash = FileSystem::fileHash(patch_download_dir.absolutePath() + "/" + filename, QCryptographicHash::Md5);
  188. if (!actual_hash_list.contains(hash)) {
  189. qDebug() << "File " << filename << " with hash " << hash << "seems outdated, deleting!";
  190. if (!QFile::remove(patch_download_dir.absolutePath() + "/" +filename)) {
  191. qDebug() << __FUNCTION__ << "Unable to remove file " << filename;
  192. emit removeFileError(patch_download_dir.absolutePath() + "/" + filename);
  193. }
  194. }
  195. }
  196. return true;
  197. }
  198. bool PatchDownloader::DownloadMissingPatches()
  199. {
  200. QDir dir(patch_download_dir);
  201. if (!dir.exists())
  202. QDir().mkdir(patch_download_dir.absolutePath());
  203. bool download_started = false;
  204. foreach (Patch patch, patch_data) {
  205. QString patch_filepath = patch_download_dir.absolutePath() + "/" + patch.url.fileName();
  206. qDebug() << "Patch" << patch.name << "is marked as" << Settings::getValue("DatabaseDownload/" + patch.name).toString();
  207. if (Settings::getValue("DatabaseDownload/" + patch.name).toString() != "Enabled") {
  208. emit changePatchStatus(patch.name, "Не выбраны для установки");
  209. continue;
  210. }
  211. if (FileSystem::fileExists(patch_filepath)) {
  212. if (FileSystem::fileHash(patch_filepath, QCryptographicHash::Md5) == patch.md5_hash) {
  213. Settings::setValue("DatabasePath/" + patch.name, patch_filepath);
  214. QString download_date = Settings::getValue("DatabaseDownloadDate/" + patch.name).toString();
  215. emit changePatchStatus(patch.name, "Актуальная версия (" + download_date + ")");
  216. continue;
  217. }
  218. Settings::setValue("DatabasePath/" + patch.name, "none");
  219. if (!QFile::remove(patch_filepath)) {
  220. qDebug() << __FUNCTION__ << "Unable to remove file " << patch_filepath;
  221. emit changePatchStatus(patch.name, "Ошибка обновления 0x1");
  222. emit removeFileError(patch_filepath);
  223. continue;
  224. }
  225. }
  226. emit changePatchStatus(patch.name, "Ожидает скачивания");
  227. if (!download_started) {
  228. download_started = true;
  229. qDebug() << "Started downloads of PatchDownloader!";
  230. emit downloadStarted();
  231. }
  232. qDebug() << "Starting download of file " << patch_filepath << " from url " << patch.url;
  233. downloads_list[patch.name]->setUrl(patch.url);
  234. downloads_list[patch.name]->targetFile = new QFile(patch_filepath, downloads_list[patch.name]);
  235. downloads_list[patch.name]->targetFile->open(QIODevice::ReadWrite);
  236. downloads_list[patch.name]->start();
  237. active_downloads_number++;
  238. }
  239. foreach (Patch patch, patch_data) {
  240. downloads_list[patch.name]->waitForDownloaded();
  241. }
  242. qDebug() << "Finished downloading patches!";
  243. return true;
  244. }