#include "patchdownloader.h" #include "models/filesystem.h" #include "models/settings.h" #include #include #include #include #include #include Q_DECLARE_METATYPE(QList) PatchDownloader::PatchDownloader(QObject *parent) : QObject(parent), patch_download_dir(QApplication::applicationDirPath() + "/data") { qRegisterMetaType>(); active_downloads_number = 0; connect(&update_check_timer, &QTimer::timeout, this, &PatchDownloader::checkForUpdates); update_check_timer.setInterval(1000 * 60 * 5); // 5 minutes update_check_timer.start(); } PatchDownloader::~PatchDownloader() { foreach (Downloader* downloader, downloads_list) { downloader->waitForDownloaded(); downloader->deleteLater(); } } QString PatchDownloader::getPredictedDownloadSizeFormatted() { double mbytes = 0; if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 650; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 80; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 120; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 3; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 4; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 1; } if (Settings::getValue("DatabaseDownload/sound").toString() == "Enabled") { mbytes += 2100; } QString unit = "Мб"; if (mbytes > 1024) { mbytes /= 1024; unit = "Гб"; } return QString::number(mbytes, 'f', 1) + " " + unit; } QString PatchDownloader::getDatabasePathByPatchName(QString name) { return patch_download_dir.absolutePath() + "/" + patch_data[name].url.fileName(); } void PatchDownloader::checkForUpdates() { foreach (QString patch_name, all_patch_names) { if (downloads_list.contains(patch_name)) continue; downloads_list[patch_name] = new Downloader(); connect(downloads_list[patch_name], &Downloader::progressChanged, this, &PatchDownloader::onDownloaderProgressChanged); connect(downloads_list[patch_name], &Downloader::downloadFinished, this, &PatchDownloader::onDownloaderCompleted); } if (active_downloads_number > 0) { qDebug() << "PatchDownloader: downloads are not ready yet, passing checkForUpdates"; return; } emit checkForUpdatesStarted(); if (!updatePatchList()) { emit checkForUpdatesFinished(); return; } removeOldPatchesFromDirecrory(); DownloadMissingPatches(); qDebug() << "Finished checking for updates!"; emit checkForUpdatesFinished(); } void PatchDownloader::onDownloaderCompleted(Downloader *downloader_ptr) { if (downloader_ptr->targetFile) { downloader_ptr->targetFile->close(); downloader_ptr->targetFile->deleteLater(); } QString patch_finished_download_name = "none"; foreach (QString patch_name, downloads_list.keys()) { if (downloads_list[patch_name] == downloader_ptr) patch_finished_download_name = patch_name; } if (patch_finished_download_name != "none") { QString patch_filepath = patch_download_dir.absolutePath() + "/" + patch_data[patch_finished_download_name].url.fileName(); Settings::setValue("DatabaseDownloadDate/" + patch_data[patch_finished_download_name].name, QDate::currentDate().toString("dd.MM.yyyy")); Settings::setValue("DatabaseApplied/" + patch_data[patch_finished_download_name].name, "False"); Settings::setValue("DatabasePath/" + patch_data[patch_finished_download_name].name, patch_filepath); emit changePatchStatus(patch_finished_download_name, "Загрузка завершена"); } active_downloads_number--; if (active_downloads_number == 0) { emit downloadCompleted(); } } void PatchDownloader::onDownloaderProgressChanged(Downloader*) { quint64 totalSize = 0; quint64 downloadedSize = 0; quint64 summary_speed = 0; quint64 time_elapsed = 0; foreach (Downloader* downloader, downloads_list) { totalSize += downloader->getBytesTotal(); downloadedSize += downloader->getBytesDownloaded(); if (downloader->getBytesTotal() != downloader->getBytesDownloaded()) { summary_speed += downloader->getSpeed(); } } time_elapsed = (totalSize - downloadedSize) / qMax(quint64(1), summary_speed); QList download_data; download_data << PatchDownloadData({"general", downloadedSize, totalSize, Downloader::getSpeedFormatted(summary_speed), Downloader::getElapsedTimeFormatted(time_elapsed)}); foreach (QString patch_name, downloads_list.keys()) { Downloader* patch_download = downloads_list[patch_name]; if (patch_download->isStarted()) { download_data << PatchDownloadData({patch_name, patch_download->getBytesDownloaded(), patch_download->getBytesTotal(), Downloader::getSpeedFormatted(patch_download->getSpeed()), Downloader::getElapsedTimeFormatted(patch_download->getElapsedTime())}); } } emit progressChanged(download_data); } int PatchDownloader::versionFromPatchFilename(QString filename) { int version = 0; for (int i = filename.indexOf("_v") + 2; i < filename.indexOf("_v") + 7; i += 2) { version = version * 10 + (filename.at(i).toLatin1() - '0'); } return version; } bool PatchDownloader::updatePatchList() { QUrlQuery query; // query for building GET-request aka patch-version foreach (QString patch_name, all_patch_names) { query.addQueryItem(patch_name, "100"); } QUrl target_url; target_url.setUrl("http://translate.lotros.ru/groupware/check_updates"); target_url.setQuery(query); QByteArray target_array; Downloader downloader; downloader.setUrl(target_url); downloader.targetBytearray = &target_array; downloader.start(); downloader.waitForDownloaded(); if (target_array.isEmpty()) { qDebug() << __FUNCTION__ << "Cannot download, target_array is empty!"; emit getPatchListError(); return false; } QStringList entry_list = QString(target_array).split('|'); if (entry_list.size() != all_patch_names.size()) { qDebug() << __FUNCTION__ << "Entry list size is not equal to patch names size!" << QString(target_array); emit getPatchListError(); return false; } for (int i = 0; i < entry_list.size(); ++i) { QStringList current_patch_data = entry_list[i].split(":::"); if (current_patch_data.size() != 3) { qDebug() << __FUNCTION__ << "Incorrect patch entry size! Entry: " << entry_list[i]; emit getPatchListError(); return false; } patch_data[all_patch_names[i]] = {current_patch_data[0], current_patch_data[1], current_patch_data[2], all_patch_names[i]}; } return true; } bool PatchDownloader::removeOldPatchesFromDirecrory() { QStringList actual_hash_list; foreach (Patch patch, patch_data) { actual_hash_list.append(patch.md5_hash); } qDebug() << "Removing old patches. Current hash list " << actual_hash_list; QStringList paths = patch_download_dir.entryList(QDir::Files); foreach (QString filename, paths) { QString hash = FileSystem::fileHash(patch_download_dir.absolutePath() + "/" + filename, QCryptographicHash::Md5); if (!actual_hash_list.contains(hash)) { qDebug() << "File " << filename << " with hash " << hash << "seems outdated, deleting!"; if (!QFile::remove(patch_download_dir.absolutePath() + "/" +filename)) { qDebug() << __FUNCTION__ << "Unable to remove file " << filename; emit removeFileError(patch_download_dir.absolutePath() + "/" + filename); } } } return true; } bool PatchDownloader::DownloadMissingPatches() { QDir dir(patch_download_dir); if (!dir.exists()) QDir().mkdir(patch_download_dir.absolutePath()); bool download_started = false; foreach (Patch patch, patch_data) { QString patch_filepath = patch_download_dir.absolutePath() + "/" + patch.url.fileName(); qDebug() << "Patch" << patch.name << "is marked as" << Settings::getValue("DatabaseDownload/" + patch.name).toString(); if (Settings::getValue("DatabaseDownload/" + patch.name).toString() != "Enabled") { emit changePatchStatus(patch.name, "Не выбраны для установки"); continue; } if (FileSystem::fileExists(patch_filepath)) { if (FileSystem::fileHash(patch_filepath, QCryptographicHash::Md5) == patch.md5_hash) { Settings::setValue("DatabasePath/" + patch.name, patch_filepath); QString download_date = Settings::getValue("DatabaseDownloadDate/" + patch.name).toString(); emit changePatchStatus(patch.name, "Актуальная версия (" + download_date + ")"); continue; } Settings::setValue("DatabasePath/" + patch.name, "none"); if (!QFile::remove(patch_filepath)) { qDebug() << __FUNCTION__ << "Unable to remove file " << patch_filepath; emit changePatchStatus(patch.name, "Ошибка обновления 0x1"); emit removeFileError(patch_filepath); continue; } } emit changePatchStatus(patch.name, "Ожидает скачивания"); if (!download_started) { download_started = true; qDebug() << "Started downloads of PatchDownloader!"; emit downloadStarted(); } qDebug() << "Starting download of file " << patch_filepath << " from url " << patch.url; downloads_list[patch.name]->setUrl(patch.url); downloads_list[patch.name]->targetFile = new QFile(patch_filepath, downloads_list[patch.name]); downloads_list[patch.name]->targetFile->open(QIODevice::ReadWrite); downloads_list[patch.name]->start(); active_downloads_number++; } foreach (Patch patch, patch_data) { downloads_list[patch.name]->waitForDownloaded(); } qDebug() << "Finished downloading patches!"; return true; }