mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
90d1f4be02
The qTox Project is not associated with the Tox Project in any way, with the exception of qTox using the Tox Projet's toxcore collection of libraries. In particular, the Tox Projet does not own copyright over the qTox Project's qTox collection of software, source code, and assets. The qTox Project's assets are under the sole copyright of the qTox contributors, and no partiular rights are granted to the Tox Project.
189 lines
5.3 KiB
C++
189 lines
5.3 KiB
C++
/*
|
|
Copyright © 2014 by The qTox Project
|
|
|
|
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 <http://www.gnu.org/licenses/>
|
|
*/
|
|
|
|
|
|
#include "widget.h"
|
|
#include "ui_widget.h"
|
|
|
|
#include <QDir>
|
|
#include <QFile>
|
|
#include <QProcess>
|
|
#include <QMessageBox>
|
|
#include <QMetaObject>
|
|
|
|
#include "settingsDir.h"
|
|
#include "update.h"
|
|
|
|
#ifdef Q_OS_WIN
|
|
const bool supported = true;
|
|
const QString QTOX_PATH = "qtox.exe";
|
|
#else
|
|
const bool supported = false;
|
|
const QString QTOX_PATH;
|
|
#endif
|
|
|
|
Widget::Widget(QWidget *parent) :
|
|
QWidget(parent),
|
|
ui(new Ui::Widget)
|
|
{
|
|
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)
|
|
{
|
|
QMessageBox::critical(this,tr("Error"), message+'\n'+tr("qTox will restart now."));
|
|
deleteUpdate();
|
|
restoreBackups();
|
|
startQToxAndExit();
|
|
}
|
|
|
|
void Widget::deleteUpdate()
|
|
{
|
|
QDir updateDir(getSettingsDirPath()+"/update/");
|
|
updateDir.removeRecursively();
|
|
}
|
|
|
|
void Widget::startQToxAndExit()
|
|
{
|
|
QProcess::startDetached(QTOX_PATH);
|
|
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 = getSettingsDirPath()+"/update/";
|
|
QDir updateDir(updateDirStr);
|
|
if (!updateDir.exists())
|
|
fatalError(tr("No update found."));
|
|
|
|
// 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();
|
|
|
|
setProgress(1);
|
|
|
|
QList<UpdateFileMeta> updateFlist = parseFlist(updateFlistData);
|
|
setProgress(2);
|
|
QList<UpdateFileMeta> diff = genUpdateDiff(updateFlist);
|
|
setProgress(4);
|
|
for (UpdateFileMeta fileMeta : diff)
|
|
if (!QFile::exists(updateDirStr+fileMeta.installpath))
|
|
fatalError(tr("The update is incomplete."));
|
|
|
|
if (diff.size() == 0)
|
|
fatalError(tr("The diff list is empty."));
|
|
|
|
setProgress(5);
|
|
|
|
/// 2. Check the update (5-50%)
|
|
float checkProgressStep = 45.0/(float)diff.size();
|
|
float checkProgress = 5;
|
|
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(50);
|
|
|
|
/// 3. Install the update (50-95%)
|
|
float installProgressStep = 45.0/(float)diff.size();
|
|
float installProgress = 50;
|
|
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
|
|
startQToxAndExit();
|
|
}
|