/* Copyright © 2014 by The qTox Project Contributors This file is part of qTox, a Qt-based graphical interface for Tox. qTox is libre software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. qTox is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with qTox. If not, see */ #include "widget.h" #include "ui_widget.h" #include #include #include #include #include #include #include #include #include "update.h" #ifdef Q_OS_WIN #ifdef _WIN32_WINNT #undef _WIN32_WINNT #endif #define _WIN32_WINNT 0x0600 // Vista for SHGetKnownFolderPath #include #include #include #include const bool supported = true; const QString QTOX_PATH = "qtox.exe"; #else const bool supported = false; const QString QTOX_PATH; #endif const QString SETTINGS_FILE = "settings.ini"; Widget::Widget(const Settings& s) : QWidget(nullptr) , ui(new Ui::Widget) , settings{s} { ui->setupUi(this); // Updates only for supported platforms if (!supported) fatalError(tr("The qTox updater is not supported on this platform.")); QMetaObject::invokeMethod(this, "update", Qt::QueuedConnection); } Widget::~Widget() { delete ui; } void Widget::setProgress(int value) { ui->progress->setValue(value); ui->progress->repaint(); qApp->processEvents(); } void Widget::fatalError(QString message) { qCritical() << "Update aborted with error:" << message; QMessageBox::critical(this, tr("Error"), message + '\n' + tr("qTox will restart now.")); deleteUpdate(); restoreBackups(); startQToxAndExit(); } void Widget::deleteUpdate() { QDir updateDir(settings.getSettingsDirPath() + "/update/"); updateDir.removeRecursively(); } void Widget::startQToxAndExit() { #ifdef Q_OS_WIN // Try to restart qTox as the actual user with our unelevated token STARTUPINFOW si; PROCESS_INFORMATION pi; SecureZeroMemory(&si, sizeof(si)); SecureZeroMemory(&pi, sizeof(pi)); si.cb = sizeof(si); bool unelevateOk = true; auto advapi32H = LoadLibrary(TEXT("advapi32.dll")); if ((unelevateOk = (advapi32H != nullptr))) { auto CreateProcessWithTokenWH = (decltype(&CreateProcessWithTokenW))GetProcAddress(advapi32H, "CreateProcessWithTokenW"); if ((unelevateOk = (CreateProcessWithTokenWH != nullptr))) { if (!CreateProcessWithTokenWH(settings.getPrimaryToken(), 0, QTOX_PATH.toStdWString().c_str(), 0, 0, 0, QApplication::applicationDirPath().toStdWString().c_str(), &si, &pi)) unelevateOk = false; } } CloseHandle(pi.hProcess); CloseHandle(pi.hThread); if (!unelevateOk) { qWarning() << "Failed to start unelevated qTox"; QProcess::startDetached(QTOX_PATH); } #else QProcess::startDetached(QTOX_PATH); #endif exit(0); } void Widget::deleteBackups() { for (QString file : backups) QFile(file + ".bak").remove(); } void Widget::restoreBackups() { for (QString file : backups) QFile(file + ".bak").rename(file); } void Widget::update() { /// 1. Find and parse the update (0-5%) // Check that the dir exists QString updateDirStr = settings.getSettingsDirPath() + "/update/"; QDir updateDir(updateDirStr); if (!updateDir.exists()) fatalError(tr("No update found.")); setProgress(2); // Check that we have a flist and that every file on the diff exists QFile updateFlistFile(updateDirStr + "flist"); if (!updateFlistFile.open(QIODevice::ReadOnly)) fatalError(tr("The update is incomplete!")); QByteArray updateFlistData = updateFlistFile.readAll(); updateFlistFile.close(); QList updateFlist = parseFlist(updateFlistData); setProgress(5); /// 2. Generate a diff (5-50%) QList diff = genUpdateDiff(updateFlist, this); for (UpdateFileMeta fileMeta : diff) if (!QFile::exists(updateDirStr + fileMeta.installpath)) fatalError(tr("The update is incomplete.")); if (diff.size() == 0) fatalError(tr("The update is empty!")); setProgress(50); qDebug() << "Diff generated," << diff.size() << "files to update"; /// 2. Check the update (50-75%) float checkProgressStep = 25.0 / (float)diff.size(); float checkProgress = 50; for (UpdateFileMeta fileMeta : diff) { UpdateFile file; file.metadata = fileMeta; QFile fileFile(updateDirStr + fileMeta.installpath); if (!fileFile.open(QIODevice::ReadOnly)) fatalError(tr("Update files are unreadable.")); file.data = fileFile.readAll(); fileFile.close(); if (file.data.size() != (int)fileMeta.size) fatalError(tr("Update files are corrupted.")); if (crypto_sign_verify_detached(file.metadata.sig, (unsigned char*)file.data.data(), file.data.size(), key) != 0) fatalError(tr("Update files are corrupted.")); checkProgress += checkProgressStep; setProgress(checkProgress); } setProgress(75); qDebug() << "Update files signature verified, installing"; /// 3. Install the update (75-95%) float installProgressStep = 20.0 / (float)diff.size(); float installProgress = 75; for (UpdateFileMeta fileMeta : diff) { // Backup old files if (QFile(fileMeta.installpath).exists()) { QFile(fileMeta.installpath + ".bak").remove(); QFile(fileMeta.installpath).rename(fileMeta.installpath + ".bak"); backups.append(fileMeta.installpath); } // Install new ones QDir().mkpath(QFileInfo(fileMeta.installpath).absolutePath()); QFile fileFile(updateDirStr + fileMeta.installpath); if (!fileFile.copy(fileMeta.installpath)) fatalError(tr("Unable to copy the update's files from ") + (updateDirStr + fileMeta.installpath) + " to " + fileMeta.installpath); installProgress += installProgressStep; setProgress(installProgress); } setProgress(95); /// 4. Delete the update and backups (95-100%) deleteUpdate(); setProgress(97); deleteBackups(); setProgress(100); /// 5. Start qTox and exit qDebug() << "Update applied, restarting qTox!"; startQToxAndExit(); }