From 747fd335925df422aaedd7353c30cf89b03c0583 Mon Sep 17 00:00:00 2001 From: "Tux3 / Mlkj / !Lev.uXFMLA" Date: Sun, 9 Nov 2014 15:51:00 +0100 Subject: [PATCH 1/5] Create AutoUpdater, teach it to check new updates It can't read a diff or apply an update yet, just check. --- qtox.pro | 10 +++--- src/autoupdate.cpp | 76 ++++++++++++++++++++++++++++++++++++++++++++++ src/autoupdate.h | 47 ++++++++++++++++++++++++++++ src/main.cpp | 6 +++- 4 files changed, 134 insertions(+), 5 deletions(-) create mode 100644 src/autoupdate.cpp create mode 100644 src/autoupdate.h diff --git a/qtox.pro b/qtox.pro index 608595616..d14ba985d 100644 --- a/qtox.pro +++ b/qtox.pro @@ -57,7 +57,7 @@ contains(JENKINS,YES) { # Rules for Windows, Mac OSX, and Linux win32 { RC_FILE = windows/qtox.rc - LIBS += -liphlpapi -L$$PWD/libs/lib -ltoxav -ltoxcore -ltoxencryptsave -ltoxdns -lvpx -lpthread + LIBS += -liphlpapi -L$$PWD/libs/lib -ltoxav -ltoxcore -ltoxencryptsave -ltoxdns -lvpx -lpthread -lsodium LIBS += -L$$PWD/libs/lib -lopencv_core248 -lopencv_highgui248 -lopencv_imgproc248 -lOpenAL32 -lopus LIBS += -lz -lopengl32 -lole32 -loleaut32 -luuid -lvfw32 -ljpeg -ltiff -lpng -ljasper -lIlmImf -lHalf -lws2_32 } else { @@ -76,7 +76,7 @@ win32 { LIBS += -Wl,-Bdynamic -lv4l1 -lv4l2 -lavformat -lavcodec -lavutil -lswscale -lusb-1.0 } else { - LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lvpx -lopenal -lopencv_core -lopencv_highgui -lopencv_imgproc + LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lvpx -lsodium -lopenal -lopencv_core -lopencv_highgui -lopencv_imgproc } contains(JENKINS, YES) { @@ -146,7 +146,8 @@ HEADERS += src/widget/form/addfriendform.h \ src/ipc.h \ src/widget/toxuri.h \ src/toxdns.h \ - src/widget/toxsave.h + src/widget/toxsave.h \ + src/autoupdate.h SOURCES += \ src/widget/form/addfriendform.cpp \ @@ -209,4 +210,5 @@ SOURCES += \ src/widget/toxuri.cpp \ src/toxdns.cpp \ src/ipc.cpp \ - src/widget/toxsave.cpp + src/widget/toxsave.cpp \ + src/autoupdate.cpp diff --git a/src/autoupdate.cpp b/src/autoupdate.cpp new file mode 100644 index 000000000..244bd4876 --- /dev/null +++ b/src/autoupdate.cpp @@ -0,0 +1,76 @@ +#include "src/autoupdate.h" +#include +#include +#include + +#include + +#ifdef _WIN32 +const QString AutoUpdater::platform = "win32"; +#else +const QString AutoUpdater::platform = "win32"; ///TODO: FIXME: undefine, we want an empty qstring +#endif +const QString AutoUpdater::updateServer = "http://127.0.0.1"; +const QString AutoUpdater::checkURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/version"; +const QString AutoUpdater::flistURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/flist"; +const QString AutoUpdater::filesURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/files/"; +unsigned char AutoUpdater::key[crypto_sign_PUBLICKEYBYTES] = +{ + 0xa5, 0x80, 0xf3, 0xb7, 0xd0, 0x10, 0xc0, 0xf9, 0xd6, 0xcf, 0x48, 0x15, 0x99, 0x70, 0x92, 0x49, + 0xf6, 0xe8, 0xe5, 0xe2, 0x6c, 0x73, 0x8c, 0x48, 0x25, 0xed, 0x01, 0x72, 0xf7, 0x6c, 0x17, 0x28 +}; + +bool AutoUpdater::isUpdateAvailable() +{ + QString newVersion = getUpdateVersion(); + if (newVersion.isEmpty() || newVersion == GIT_VERSION) + return false; + else + return true; +} + +QString AutoUpdater::getUpdateVersion() +{ + QString version; + + // Updates only for supported platforms + if (platform.isEmpty()) + return version; + + QNetworkAccessManager *manager = new QNetworkAccessManager; + + QNetworkReply* reply = manager->get(QNetworkRequest(QUrl(checkURI))); + while (!reply->isFinished()) + qApp->processEvents(); + + if (reply->error() != QNetworkReply::NoError) + { + qWarning() << "AutoUpdater: getUpdateVersion: network error: "<errorString(); + return version; + } + + QByteArray data = reply->readAll(); + + // Check updater protocol version + if ((int)data[0] != '1') + { + qWarning() << "AutoUpdater: getUpdateVersion: Bad protocol version "<<(uint8_t)data[0]; + return version; + } + + // Check the signature + QByteArray sigData = data.mid(1, crypto_sign_BYTES); + unsigned char* sig = (unsigned char*)sigData.data(); + QByteArray msgData = data.mid(1+crypto_sign_BYTES); + unsigned char* msg = (unsigned char*)msgData.data(); + + if (crypto_sign_verify_detached(sig, msg, msgData.size(), key) != 0) + { + qCritical() << "AutoUpdater: getUpdateVersion: RECEIVED FORGED VERSION FILE FROM "< + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef AUTOUPDATE_H +#define AUTOUPDATE_H + +#include + +/// Handles checking and applying updates for qTox +class AutoUpdater +{ +public: + /// Connects to the qTox update server, returns true if an update is available for download + /// Will call getUpdateVersion, and as such may block and processEvents + static bool isUpdateAvailable(); + /// Fetch the version string of the last update available from the qTox update server + /// Will try to follow qTox's proxy settings, may block and processEvents + static QString getUpdateVersion(); + +private: + AutoUpdater() = delete; + +private: + // Constants + static const QString updateServer; ///< Hostname of the qTox update server + static const QString platform; ///< Name of platform we're trying to get updates for + static const QString checkURI; ///< URI of the file containing the latest version string + static const QString flistURI; ///< URI of the file containing info on each file (hash, signature, size, name, ..) + static const QString filesURI; ///< URI of the actual files of the latest version + static unsigned char key[]; +}; + +#endif // AUTOUPDATE_H diff --git a/src/main.cpp b/src/main.cpp index 19cc22f11..e014ea128 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -27,6 +27,8 @@ #include #include +#include + #ifdef LOG_TO_FILE static QtMessageHandler dflt; static QTextStream* logFile {nullptr}; @@ -55,6 +57,8 @@ int main(int argc, char *argv[]) a.setApplicationName("qTox"); a.setOrganizationName("Tox"); + sodium_init(); // For the auto-updater + #ifdef LOG_TO_FILE logFile = new QTextStream; dflt = qInstallMessageHandler(nullptr); @@ -127,7 +131,7 @@ int main(int argc, char *argv[]) } // Run - Widget* w = Widget::getInstance(); + Widget* w = Widget::getInstance(); int errorcode = a.exec(); delete w; From 965254d1516fb9cbc2071a499b8b96a205f137d8 Mon Sep 17 00:00:00 2001 From: "Tux3 / Mlkj / !Lev.uXFMLA" Date: Sun, 9 Nov 2014 17:55:38 +0100 Subject: [PATCH 2/5] Teach the AutoUpdater to read a flist We can now read the list of files in an update, along with the install path, file size, and signature Everything, metadata and files, is signed with ed25519, but not encrypted --- qtox.pro | 6 +- src/autoupdate.cpp | 135 +++++++++++++++++++++- src/autoupdate.h | 27 +++++ src/misc/serialize.cpp | 254 +++++++++++++++++++++++++++++++++++++++++ src/misc/serialize.h | 45 ++++++++ src/widget/widget.cpp | 15 ++- 6 files changed, 474 insertions(+), 8 deletions(-) create mode 100644 src/misc/serialize.cpp create mode 100644 src/misc/serialize.h diff --git a/qtox.pro b/qtox.pro index d14ba985d..61393fae3 100644 --- a/qtox.pro +++ b/qtox.pro @@ -147,7 +147,8 @@ HEADERS += src/widget/form/addfriendform.h \ src/widget/toxuri.h \ src/toxdns.h \ src/widget/toxsave.h \ - src/autoupdate.h + src/autoupdate.h \ + src/misc/serialize.h SOURCES += \ src/widget/form/addfriendform.cpp \ @@ -211,4 +212,5 @@ SOURCES += \ src/toxdns.cpp \ src/ipc.cpp \ src/widget/toxsave.cpp \ - src/autoupdate.cpp + src/autoupdate.cpp \ + src/misc/serialize.cpp diff --git a/src/autoupdate.cpp b/src/autoupdate.cpp index 244bd4876..6e932f86d 100644 --- a/src/autoupdate.cpp +++ b/src/autoupdate.cpp @@ -1,10 +1,26 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + #include "src/autoupdate.h" +#include "src/misc/serialize.h" #include #include #include -#include - #ifdef _WIN32 const QString AutoUpdater::platform = "win32"; #else @@ -38,7 +54,6 @@ QString AutoUpdater::getUpdateVersion() return version; QNetworkAccessManager *manager = new QNetworkAccessManager; - QNetworkReply* reply = manager->get(QNetworkRequest(QUrl(checkURI))); while (!reply->isFinished()) qApp->processEvents(); @@ -46,15 +61,21 @@ QString AutoUpdater::getUpdateVersion() if (reply->error() != QNetworkReply::NoError) { qWarning() << "AutoUpdater: getUpdateVersion: network error: "<errorString(); + reply->deleteLater(); + manager->deleteLater(); return version; } QByteArray data = reply->readAll(); + reply->deleteLater(); + manager->deleteLater(); + if (data.size() < (int)(1+crypto_sign_BYTES)) + return version; // Check updater protocol version if ((int)data[0] != '1') { - qWarning() << "AutoUpdater: getUpdateVersion: Bad protocol version "<<(uint8_t)data[0]; + qWarning() << "AutoUpdater: getUpdateVersion: Bad version "<<(uint8_t)data[0]; return version; } @@ -74,3 +95,109 @@ QString AutoUpdater::getUpdateVersion() return version; } + +QList AutoUpdater::genUpdateDiff() +{ + QList diff; + + // Updates only for supported platforms + if (platform.isEmpty()) + return diff; + + QList newFlist = getUpdateFlist(); + + return diff; +} + +QList AutoUpdater::parseflist(QByteArray flistData) +{ + QList flist; + + if (flistData.isEmpty()) + { + qWarning() << "AutoUpdater::parseflist: Empty data"; + return flist; + } + + // Check version + if (flistData[0] != '1') + { + qWarning() << "AutoUpdater: parseflist: Bad version "<<(uint8_t)flistData[0]; + return flist; + } + flistData = flistData.mid(1); + + // Check signature + if (flistData.size() < (int)(crypto_sign_BYTES)) + { + qWarning() << "AutoUpdater::parseflist: Truncated data"; + return flist; + } + else + { + QByteArray msgData = flistData.mid(crypto_sign_BYTES); + unsigned char* msg = (unsigned char*)msgData.data(); + if (crypto_sign_verify_detached((unsigned char*)flistData.data(), msg, msgData.size(), key) != 0) + { + qCritical() << "AutoUpdater: parseflist: FORGED FLIST FILE"; + return flist; + } + flistData = flistData.mid(crypto_sign_BYTES); + } + + // Parse. We assume no errors handling needed since the signature is valid. + while (!flistData.isEmpty()) + { + qDebug() << "Got "< AutoUpdater::getUpdateFlist() +{ + QList flist; + + QNetworkAccessManager *manager = new QNetworkAccessManager; + QNetworkReply* reply = manager->get(QNetworkRequest(QUrl(flistURI))); + while (!reply->isFinished()) + qApp->processEvents(); + + if (reply->error() != QNetworkReply::NoError) + { + qWarning() << "AutoUpdater: getUpdateFlist: network error: "<errorString(); + reply->deleteLater(); + manager->deleteLater(); + return flist; + } + + QByteArray data = reply->readAll(); + reply->deleteLater(); + manager->deleteLater(); + + flist = parseflist(data); + return flist; +} diff --git a/src/autoupdate.h b/src/autoupdate.h index 21ede32f3..30502c9bc 100644 --- a/src/autoupdate.h +++ b/src/autoupdate.h @@ -19,10 +19,27 @@ #define AUTOUPDATE_H #include +#include +#include /// Handles checking and applying updates for qTox class AutoUpdater { +public: + struct UpdateFileMeta + { + unsigned char sig[crypto_sign_BYTES]; ///< Signature of the file (ed25519) + QString id; ///< Unique id of the file + QString installpath; ///< Local path including the file name. May be relative to qtox-updater or absolute + uint64_t size; ///< Size in bytes of the file + }; + + struct UpdateFile + { + UpdateFileMeta metadata; + QByteArray data; + }; + public: /// Connects to the qTox update server, returns true if an update is available for download /// Will call getUpdateVersion, and as such may block and processEvents @@ -30,6 +47,16 @@ public: /// Fetch the version string of the last update available from the qTox update server /// Will try to follow qTox's proxy settings, may block and processEvents static QString getUpdateVersion(); + /// Generates a list of files we need to update + /// Will try to follow qTox's proxy settings, may block and processEvents + static QList genUpdateDiff(); + +protected: + /// Parses and validates a flist file. Returns an empty list on error + static QList parseflist(QByteArray flistData); + /// Get the update server's flist and parse it. Returns an empty list on error + /// Will try to follow qTox's proxy settings, may block and processEvents + static QList getUpdateFlist(); private: AutoUpdater() = delete; diff --git a/src/misc/serialize.cpp b/src/misc/serialize.cpp new file mode 100644 index 000000000..dcf352e85 --- /dev/null +++ b/src/misc/serialize.cpp @@ -0,0 +1,254 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "src/misc/serialize.h" + +QByteArray doubleToData(double num) +{ + union + { + char tab[8]; + double n; + } castUnion; + //char n[8]; + //*((double*) n) = num; + + castUnion.n=num; + return QByteArray(castUnion.tab,8); +} + +QByteArray floatToData(float num) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.n=num; + return QByteArray(castUnion.tab,4); +} + +float dataToFloat(QByteArray data) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.tab[0]=data.data()[0]; + castUnion.tab[1]=data.data()[1]; + castUnion.tab[2]=data.data()[2]; + castUnion.tab[3]=data.data()[3]; + return castUnion.n; +} + +// Converts a string into PNet string data +QByteArray stringToData(QString str) +{ + QByteArray data(4,0); + // Write the size in a Uint of variable lenght (8-32 bits) + int i=0; + uint num1 = (uint)str.toUtf8().size(); + while (num1 >= 0x80) + { + data[i] = (unsigned char)(num1 | 0x80); i++; + num1 = num1 >> 7; + } + data[i]=num1; + data.resize(i+1); + data+=str.toUtf8(); + return data; +} + +QString dataToString(QByteArray data) +{ + // Variable UInt32 + unsigned char num3; + int num = 0; + int num2 = 0; + int i=0; + do + { + num3 = data[i]; i++; + num |= (num3 & 0x7f) << num2; + num2 += 7; + } while ((num3 & 0x80) != 0); + unsigned int strlen = (uint) num; + + if (!strlen) + return QString(); + + data = data.right(data.size()-i); // Remove the strlen + data.truncate(strlen); + return QString(data); +} + +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data) +{ + uint endvalue=0; + uint value=0; + if (numberOfBits <= 8) + { + endvalue = (uchar)data[0]; + goto done; + } + value = (uchar)data[0]; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + endvalue = (value | ((uint) ((uchar)data[1]) << 8)); + goto done; + } + value |= (uint) (((uchar)data[1]) << 8); + numberOfBits -= 8; + if (numberOfBits <= 8) + { + uint num2 = (uint) (((uchar)data[2]) << 0x10); + endvalue = (value | num2); + goto done; + } + value |= (uint) (((uchar)data[2]) << 0x10); + numberOfBits -= 8; + endvalue = (value | ((uint) (((uchar)data[3]) << 0x18))); + goto done; + + done: + + float num = max - min; + int num2 = (((int) 1) << numberOfBits) - 1; + float num3 = endvalue; + float num4 = num3 / ((float) num2); + return (min + (num4 * num)); +} + +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits) +{ + QByteArray data; + float num = max - min; + float num2 = (value - min) / num; + int num3 = (((int) 1) << numberOfBits) - 1; + uint source = num3 * num2; + + if (numberOfBits <= 8) + { + data += (unsigned char)source; + return data; + } + data += (unsigned char)source; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>8; + return data; + } + data += (unsigned char)source>>8; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>16; + return data; + } + data += (unsigned char)source>>16; + data += (unsigned char)source>>24; + + return data; +} + +uint8_t dataToUint8(QByteArray data) +{ + return (uint8_t)data[0]; +} + +uint16_t dataToUint16(QByteArray data) +{ + return ((uint16_t)(uint8_t)data[0]) + +(((uint16_t)(uint8_t)data[1])<<8); +} + +uint32_t dataToUint32(QByteArray data) +{ + return ((uint32_t)(uint8_t)data[0]) + +(((uint32_t)(uint8_t)data[1])<<8) + +(((uint32_t)(uint8_t)data[2])<<16) + +(((uint32_t)(uint8_t)data[3])<<24); +} + +uint64_t dataToUint64(QByteArray data) +{ + return ((uint64_t)(uint8_t)data[0]) + +(((uint64_t)(uint8_t)data[1])<<8) + +(((uint64_t)(uint8_t)data[2])<<16) + +(((uint64_t)(uint8_t)data[3])<<24) + +(((uint64_t)(uint8_t)data[4])<<32) + +(((uint64_t)(uint8_t)data[5])<<40) + +(((uint64_t)(uint8_t)data[6])<<48) + +(((uint64_t)(uint8_t)data[7])<<56); +} + +unsigned getVUint32Size(QByteArray data) +{ + unsigned lensize=0; + { + unsigned char num3; + do { + num3 = data[lensize]; + lensize++; + } while ((num3 & 0x80) != 0); + } + return lensize; +} + +QByteArray uint8ToData(uint8_t num) +{ + QByteArray data(1,0); + data[0] = (uint8_t)num; + return data; +} + +QByteArray uint16ToData(uint16_t num) +{ + QByteArray data(2,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + return data; +} + +QByteArray uint32ToData(uint32_t num) +{ + QByteArray data(4,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + return data; +} + +QByteArray uint64ToData(uint64_t num) +{ + QByteArray data(8,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + data[4] = (uint8_t)((num>>32) & 0xFF); + data[5] = (uint8_t)((num>>40) & 0xFF); + data[6] = (uint8_t)((num>>48) & 0xFF); + data[7] = (uint8_t)((num>>56) & 0xFF); + return data; +} diff --git a/src/misc/serialize.h b/src/misc/serialize.h new file mode 100644 index 000000000..54747fe68 --- /dev/null +++ b/src/misc/serialize.h @@ -0,0 +1,45 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef SERIALIZE_H +#define SERIALIZE_H + +#include +#include +#include + +/// Most of those functions are unsafe unless otherwise specified +/// Do not use them on untrusted data (e.g. check a signature first) + +QByteArray doubleToData(double num); +QByteArray floatToData(float num); +float dataToFloat(QByteArray data); +QByteArray stringToData(QString str); +QString dataToString(QByteArray data); +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data); +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits); +uint8_t dataToUint8(QByteArray data); +uint16_t dataToUint16(QByteArray data); +uint32_t dataToUint32(QByteArray data); +uint64_t dataToUint64(QByteArray data); +unsigned getVUint32Size(QByteArray data); +QByteArray uint8ToData(uint8_t num); +QByteArray uint16ToData(uint16_t num); +QByteArray uint32ToData(uint32_t num); +QByteArray uint64ToData(uint64_t num); + +#endif // SERIALIZE_H diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 6494adad3..a3482ef14 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -31,6 +31,9 @@ #include "src/video/camera.h" #include "form/chatform.h" #include "maskablepixmapwidget.h" +#include "src/historykeeper.h" +#include "form/inputpassworddialog.h" +#include "src/autoupdate.h" #include #include #include @@ -45,9 +48,7 @@ #include #include #include -#include "src/historykeeper.h" #include -#include "form/inputpassworddialog.h" Widget *Widget::instance{nullptr}; @@ -195,6 +196,16 @@ void Widget::init() addFriendForm = new AddFriendForm; settingsWidget = new SettingsWidget(); + // Check for updates + { + QString newVersion = AutoUpdater::getUpdateVersion(); + if (!newVersion.isEmpty() && newVersion != GIT_VERSION) + { + qWarning() << "New update:"< Date: Sun, 9 Nov 2014 18:04:00 +0100 Subject: [PATCH 3/5] Add qtox update-server tools qtox-updater-sign takes data from stdin or as argument, signs it with the updater's private key, and output the signature + the data qtox-updater-genflist generates a flist, a serialized list of AutoUpdater::UpdateFileMeta for the qTox updater --- .../qtox-updater-genflist/main.cpp | 97 +++++++ .../qtox-updater-genflist.pro | 10 + .../qtox-updater-genflist/serialize.cpp | 254 ++++++++++++++++++ .../qtox-updater-genflist/serialize.h | 45 ++++ .../update-server/qtox-updater-sign/main.cpp | 43 +++ .../qtox-updater-sign/qtox-updater-sign.pro | 7 + 6 files changed, 456 insertions(+) create mode 100644 tools/update-server/qtox-updater-genflist/main.cpp create mode 100644 tools/update-server/qtox-updater-genflist/qtox-updater-genflist.pro create mode 100644 tools/update-server/qtox-updater-genflist/serialize.cpp create mode 100644 tools/update-server/qtox-updater-genflist/serialize.h create mode 100644 tools/update-server/qtox-updater-sign/main.cpp create mode 100644 tools/update-server/qtox-updater-sign/qtox-updater-sign.pro diff --git a/tools/update-server/qtox-updater-genflist/main.cpp b/tools/update-server/qtox-updater-genflist/main.cpp new file mode 100644 index 000000000..721751d90 --- /dev/null +++ b/tools/update-server/qtox-updater-genflist/main.cpp @@ -0,0 +1,97 @@ +#include +#include +#include +#include +#include +#include +#include "serialize.h" + +using namespace std; + +/// Pass the target folder as first argument, no spaces allowed. We'll call that dir $TARGET +/// Update the content of $TARGET/files/ before calling this tool +/// We'll generate $TARGET/flist and exit +/// We need qtox-updater-skey in our working directory to sign the flist +/// +/// The generated flist is very simple and just installs everything in the working directory ... + +int main(int argc, char* argv[]) +{ + cout << "qTox updater flist generator" << endl; + + /// First some basic error handling, prepare our handles, ... + if (argc != 2) + { + cout << "ERROR: qtox-updater-genflist takes the target path in argument" << endl; + return 1; + } + + QFile skeyFile("qtox-updater-skey"); + if (!skeyFile.open(QIODevice::ReadOnly)) + { + cout << "ERROR: qtox-updater-genflist can't open the secret (private) key file" << endl; + return 1; + } + QByteArray skeyData = skeyFile.readAll(); + skeyData = QByteArray::fromHex(skeyData); + skeyFile.close(); + + QString target(argv[1]); + + QFile flistFile(target+"/flist"); + if (!flistFile.open(QIODevice::Truncate | QIODevice::WriteOnly)) + { + cout << "ERROR: qtox-updater-genflist can't open the target flist" << endl; + return 1; + } + + QDir fdir(target+"/files/"); + if (!fdir.isReadable()) + { + cout << "ERROR: qtox-updater-genflist can't open the target files directory" << endl; + return 1; + } + + QStringList filesListStr = fdir.entryList(QDir::Files); + + /// Serialize the flist data + QByteArray flistData; + for (QString fileStr : filesListStr) + { + cout << "Adding "< + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "serialize.h" + +QByteArray doubleToData(double num) +{ + union + { + char tab[8]; + double n; + } castUnion; + //char n[8]; + //*((double*) n) = num; + + castUnion.n=num; + return QByteArray(castUnion.tab,8); +} + +QByteArray floatToData(float num) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.n=num; + return QByteArray(castUnion.tab,4); +} + +float dataToFloat(QByteArray data) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.tab[0]=data.data()[0]; + castUnion.tab[1]=data.data()[1]; + castUnion.tab[2]=data.data()[2]; + castUnion.tab[3]=data.data()[3]; + return castUnion.n; +} + +// Converts a string into PNet string data +QByteArray stringToData(QString str) +{ + QByteArray data(4,0); + // Write the size in a Uint of variable lenght (8-32 bits) + int i=0; + uint num1 = (uint)str.toUtf8().size(); + while (num1 >= 0x80) + { + data[i] = (unsigned char)(num1 | 0x80); i++; + num1 = num1 >> 7; + } + data[i]=num1; + data.resize(i+1); + data+=str.toUtf8(); + return data; +} + +QString dataToString(QByteArray data) +{ + // Variable UInt32 + unsigned char num3; + int num = 0; + int num2 = 0; + int i=0; + do + { + num3 = data[i]; i++; + num |= (num3 & 0x7f) << num2; + num2 += 7; + } while ((num3 & 0x80) != 0); + unsigned int strlen = (uint) num; + + if (!strlen) + return QString(); + + data = data.right(data.size()-i); // Remove the strlen + data.truncate(strlen); + return QString(data); +} + +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data) +{ + uint endvalue=0; + uint value=0; + if (numberOfBits <= 8) + { + endvalue = (uchar)data[0]; + goto done; + } + value = (uchar)data[0]; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + endvalue = (value | ((uint) ((uchar)data[1]) << 8)); + goto done; + } + value |= (uint) (((uchar)data[1]) << 8); + numberOfBits -= 8; + if (numberOfBits <= 8) + { + uint num2 = (uint) (((uchar)data[2]) << 0x10); + endvalue = (value | num2); + goto done; + } + value |= (uint) (((uchar)data[2]) << 0x10); + numberOfBits -= 8; + endvalue = (value | ((uint) (((uchar)data[3]) << 0x18))); + goto done; + + done: + + float num = max - min; + int num2 = (((int) 1) << numberOfBits) - 1; + float num3 = endvalue; + float num4 = num3 / ((float) num2); + return (min + (num4 * num)); +} + +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits) +{ + QByteArray data; + float num = max - min; + float num2 = (value - min) / num; + int num3 = (((int) 1) << numberOfBits) - 1; + uint source = num3 * num2; + + if (numberOfBits <= 8) + { + data += (unsigned char)source; + return data; + } + data += (unsigned char)source; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>8; + return data; + } + data += (unsigned char)source>>8; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>16; + return data; + } + data += (unsigned char)source>>16; + data += (unsigned char)source>>24; + + return data; +} + +uint8_t dataToUint8(QByteArray data) +{ + return (uint8_t)data[0]; +} + +uint16_t dataToUint16(QByteArray data) +{ + return ((uint16_t)(uint8_t)data[0]) + +(((uint16_t)(uint8_t)data[1])<<8); +} + +uint32_t dataToUint32(QByteArray data) +{ + return ((uint32_t)(uint8_t)data[0]) + +(((uint32_t)(uint8_t)data[1])<<8) + +(((uint32_t)(uint8_t)data[2])<<16) + +(((uint32_t)(uint8_t)data[3])<<24); +} + +uint64_t dataToUint64(QByteArray data) +{ + return ((uint64_t)(uint8_t)data[0]) + +(((uint64_t)(uint8_t)data[1])<<8) + +(((uint64_t)(uint8_t)data[2])<<16) + +(((uint64_t)(uint8_t)data[3])<<24) + +(((uint64_t)(uint8_t)data[4])<<32) + +(((uint64_t)(uint8_t)data[5])<<40) + +(((uint64_t)(uint8_t)data[6])<<48) + +(((uint64_t)(uint8_t)data[7])<<56); +} + +unsigned getVUint32Size(QByteArray data) +{ + unsigned lensize=0; + { + unsigned char num3; + do { + num3 = data[lensize]; + lensize++; + } while ((num3 & 0x80) != 0); + } + return lensize; +} + +QByteArray uint8ToData(uint8_t num) +{ + QByteArray data(1,0); + data[0] = (uint8_t)num; + return data; +} + +QByteArray uint16ToData(uint16_t num) +{ + QByteArray data(2,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + return data; +} + +QByteArray uint32ToData(uint32_t num) +{ + QByteArray data(4,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + return data; +} + +QByteArray uint64ToData(uint64_t num) +{ + QByteArray data(8,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + data[4] = (uint8_t)((num>>32) & 0xFF); + data[5] = (uint8_t)((num>>40) & 0xFF); + data[6] = (uint8_t)((num>>48) & 0xFF); + data[7] = (uint8_t)((num>>56) & 0xFF); + return data; +} diff --git a/tools/update-server/qtox-updater-genflist/serialize.h b/tools/update-server/qtox-updater-genflist/serialize.h new file mode 100644 index 000000000..54747fe68 --- /dev/null +++ b/tools/update-server/qtox-updater-genflist/serialize.h @@ -0,0 +1,45 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef SERIALIZE_H +#define SERIALIZE_H + +#include +#include +#include + +/// Most of those functions are unsafe unless otherwise specified +/// Do not use them on untrusted data (e.g. check a signature first) + +QByteArray doubleToData(double num); +QByteArray floatToData(float num); +float dataToFloat(QByteArray data); +QByteArray stringToData(QString str); +QString dataToString(QByteArray data); +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data); +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits); +uint8_t dataToUint8(QByteArray data); +uint16_t dataToUint16(QByteArray data); +uint32_t dataToUint32(QByteArray data); +uint64_t dataToUint64(QByteArray data); +unsigned getVUint32Size(QByteArray data); +QByteArray uint8ToData(uint8_t num); +QByteArray uint16ToData(uint16_t num); +QByteArray uint32ToData(uint32_t num); +QByteArray uint64ToData(uint64_t num); + +#endif // SERIALIZE_H diff --git a/tools/update-server/qtox-updater-sign/main.cpp b/tools/update-server/qtox-updater-sign/main.cpp new file mode 100644 index 000000000..21a0ef535 --- /dev/null +++ b/tools/update-server/qtox-updater-sign/main.cpp @@ -0,0 +1,43 @@ +#include +#include +#include + +using namespace std; + +int main(int argc, char* argv[]) +{ + QFile io; + QByteArray msg; + if (argc > 1) + { + msg = QByteArray(argv[1]); + } + else + { + io.open(stdin, QIODevice::ReadOnly); + msg = io.readAll(); + io.close(); + } + io.open(stdout, QIODevice::WriteOnly); + + QFile skeyFile("qtox-updater-skey"); + if (!skeyFile.open(QIODevice::ReadOnly)) + { + io.write("ERROR: qtox-updater-sign can't open the secret (private) key file\n"); + io.close(); + return 1; + } + QByteArray skeyData = skeyFile.readAll(); + skeyData = QByteArray::fromHex(skeyData); + skeyFile.close(); + + unsigned char sig[crypto_sign_BYTES]; + crypto_sign_detached(sig, nullptr, (unsigned char*)msg.data(), msg.size(), (unsigned char*)skeyData.data()); + + io.write((char*)sig, crypto_sign_BYTES); + io.write(msg); + io.close(); + + return 0; +} + diff --git a/tools/update-server/qtox-updater-sign/qtox-updater-sign.pro b/tools/update-server/qtox-updater-sign/qtox-updater-sign.pro new file mode 100644 index 000000000..5f9c65af3 --- /dev/null +++ b/tools/update-server/qtox-updater-sign/qtox-updater-sign.pro @@ -0,0 +1,7 @@ +TEMPLATE = app +CONFIG += console c++11 +QT += core + +SOURCES += main.cpp + +LIBS += -lsodium From 45e20187cc789b4416b15085bb74c6294a4527fa Mon Sep 17 00:00:00 2001 From: "Tux3 / Mlkj / !Lev.uXFMLA" Date: Sun, 9 Nov 2014 23:24:23 +0100 Subject: [PATCH 4/5] Download and applying updates, qtox-updater The auto-updater is essentially done, except for the GUI. For now this is only an API. The API works, but the tools to manage the update server would need some love --- .gitignore | 2 + qtox.pro | 2 +- src/autoupdate.cpp | 249 +++++++++++++++++++++++--- src/autoupdate.h | 44 ++++- src/main.cpp | 7 + src/widget/widget.cpp | 10 -- updater/main.cpp | 28 +++ updater/res.qrc | 5 + updater/res/qtox-256x256.png | Bin 0 -> 5630 bytes updater/serialize.cpp | 254 +++++++++++++++++++++++++++ updater/serialize.h | 45 +++++ updater/settingsDir.cpp | 52 ++++++ updater/settingsDir.h | 27 +++ updater/update.cpp | 115 ++++++++++++ updater/update.h | 55 ++++++ updater/updater.pro | 36 ++++ updater/widget.cpp | 156 ++++++++++++++++ updater/widget.h | 51 ++++++ updater/widget.ui | 139 +++++++++++++++ updater/windows/updater.exe.manifest | 12 ++ updater/windows/updater.ico | Bin 0 -> 16958 bytes updater/windows/updater.rc | 2 + 22 files changed, 1246 insertions(+), 45 deletions(-) create mode 100644 updater/main.cpp create mode 100644 updater/res.qrc create mode 100644 updater/res/qtox-256x256.png create mode 100644 updater/serialize.cpp create mode 100644 updater/serialize.h create mode 100644 updater/settingsDir.cpp create mode 100644 updater/settingsDir.h create mode 100644 updater/update.cpp create mode 100644 updater/update.h create mode 100644 updater/updater.pro create mode 100644 updater/widget.cpp create mode 100644 updater/widget.h create mode 100644 updater/widget.ui create mode 100644 updater/windows/updater.exe.manifest create mode 100644 updater/windows/updater.ico create mode 100644 updater/windows/updater.rc diff --git a/.gitignore b/.gitignore index 3727c045d..84ee6540d 100644 --- a/.gitignore +++ b/.gitignore @@ -7,3 +7,5 @@ qrc_* Makefile qtox *.qm +build-*-Release +build-*-Debug diff --git a/qtox.pro b/qtox.pro index 61393fae3..2b0f79d19 100644 --- a/qtox.pro +++ b/qtox.pro @@ -57,7 +57,7 @@ contains(JENKINS,YES) { # Rules for Windows, Mac OSX, and Linux win32 { RC_FILE = windows/qtox.rc - LIBS += -liphlpapi -L$$PWD/libs/lib -ltoxav -ltoxcore -ltoxencryptsave -ltoxdns -lvpx -lpthread -lsodium + LIBS += -liphlpapi -L$$PWD/libs/lib -lsodium -ltoxav -ltoxcore -ltoxencryptsave -ltoxdns -lvpx -lpthread LIBS += -L$$PWD/libs/lib -lopencv_core248 -lopencv_highgui248 -lopencv_imgproc248 -lOpenAL32 -lopus LIBS += -lz -lopengl32 -lole32 -loleaut32 -luuid -lvfw32 -ljpeg -ltiff -lpng -ljasper -lIlmImf -lHalf -lws2_32 } else { diff --git a/src/autoupdate.cpp b/src/autoupdate.cpp index 6e932f86d..f1ab56136 100644 --- a/src/autoupdate.cpp +++ b/src/autoupdate.cpp @@ -17,14 +17,25 @@ #include "src/autoupdate.h" #include "src/misc/serialize.h" +#include "src/misc/settings.h" #include #include #include +#include +#include +#include -#ifdef _WIN32 +#ifdef Q_OS_WIN +#include +#include +#endif + +#ifdef Q_OS_WIN const QString AutoUpdater::platform = "win32"; +const QString AutoUpdater::updaterBin = "qtox-updater.exe"; #else -const QString AutoUpdater::platform = "win32"; ///TODO: FIXME: undefine, we want an empty qstring +const QString AutoUpdater::platform; +const QString AutoUpdater::updaterBin; #endif const QString AutoUpdater::updateServer = "http://127.0.0.1"; const QString AutoUpdater::checkURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/version"; @@ -96,20 +107,7 @@ QString AutoUpdater::getUpdateVersion() return version; } -QList AutoUpdater::genUpdateDiff() -{ - QList diff; - - // Updates only for supported platforms - if (platform.isEmpty()) - return diff; - - QList newFlist = getUpdateFlist(); - - return diff; -} - -QList AutoUpdater::parseflist(QByteArray flistData) +QList AutoUpdater::parseFlist(QByteArray flistData) { QList flist; @@ -148,8 +146,6 @@ QList AutoUpdater::parseflist(QByteArray flistData) // Parse. We assume no errors handling needed since the signature is valid. while (!flistData.isEmpty()) { - qDebug() << "Got "< AutoUpdater::parseflist(QByteArray flistData) newFile.size = dataToUint64(flistData); flistData = flistData.mid(8); - qDebug() << "AutoUpdater::parseflist: New file:"; - qDebug() << "- Id: "< AutoUpdater::getUpdateFlist() +QByteArray AutoUpdater::getUpdateFlist() { - QList flist; + QByteArray flist; QNetworkAccessManager *manager = new QNetworkAccessManager; QNetworkReply* reply = manager->get(QNetworkRequest(QUrl(flistURI))); @@ -194,10 +183,212 @@ QList AutoUpdater::getUpdateFlist() return flist; } - QByteArray data = reply->readAll(); + flist = reply->readAll(); reply->deleteLater(); manager->deleteLater(); - flist = parseflist(data); return flist; } + +QByteArray AutoUpdater::getLocalFlist() +{ + QByteArray flist; + + QFile flistFile("flist"); + if (!flistFile.open(QIODevice::ReadOnly)) + { + qWarning() << "AutoUpdater::getLocalFlist: Can't open local flist"; + return flist; + } + + flist = flistFile.readAll(); + flistFile.close(); + + return flist; +} + +QList AutoUpdater::genUpdateDiff(QList updateFlist) +{ + QList diff; + QList localFlist = parseFlist(getLocalFlist()); + + for (UpdateFileMeta file : updateFlist) + if (!localFlist.contains(file)) + diff += file; + + return diff; +} + +AutoUpdater::UpdateFile AutoUpdater::getUpdateFile(UpdateFileMeta fileMeta) +{ + UpdateFile file; + file.metadata = fileMeta; + + QNetworkAccessManager *manager = new QNetworkAccessManager; + QNetworkReply* reply = manager->get(QNetworkRequest(QUrl(filesURI+fileMeta.id))); + while (!reply->isFinished()) + qApp->processEvents(); + + if (reply->error() != QNetworkReply::NoError) + { + qWarning() << "AutoUpdater: getUpdateFile: network error: "<errorString(); + reply->deleteLater(); + manager->deleteLater(); + return file; + } + + file.data = reply->readAll(); + reply->deleteLater(); + manager->deleteLater(); + + return file; +} + + +bool AutoUpdater::downloadUpdate() +{ + // Updates only for supported platforms + if (platform.isEmpty()) + return false; + + // Get a list of files to update + QByteArray newFlistData = getUpdateFlist(); + QList newFlist = parseFlist(newFlistData); + QList diff = genUpdateDiff(newFlist); + + qDebug() << "AutoUpdater: Need to update "< updateFlist = parseFlist(updateFlistData); + QList diff = genUpdateDiff(updateFlist); + + for (UpdateFileMeta fileMeta : diff) + if (!QFile::exists(updateDirStr+fileMeta.installpath)) + return false; + + return true; +} + +void AutoUpdater::installLocalUpdate() +{ + qDebug() << "AutoUpdater: About to start the qTox updater to install a local update"; + + // Delete the update if we fail so we don't fail again. + + // Updates only for supported platforms. + if (platform.isEmpty()) + { + qCritical() << "AutoUpdater: Failed to start the qTox updater, removing the update and exiting"; + QString updateDirStr = Settings::getInstance().getSettingsDirPath() + "/update/"; + QDir(updateDirStr).removeRecursively(); + exit(-1); + } + + // Workaround QTBUG-7645 + // QProcess fails silently when elevation is required instead of showing a UAC prompt on Win7/Vista +#ifdef Q_OS_WIN + int result = (int)::ShellExecuteA(0, "open", updaterBin.toUtf8().constData(), 0, 0, SW_SHOWNORMAL); + if (SE_ERR_ACCESSDENIED == result) + { + // Requesting elevation + result = (int)::ShellExecuteA(0, "runas", updaterBin.toUtf8().constData(), 0, 0, SW_SHOWNORMAL); + } + if (result <= 32) + { + goto fail; + } +#else + if (!QProcess::startDetached(updaterBin)) + goto fail; +#endif + + exit(0); + + // Centralized error handling +fail: + qCritical() << "AutoUpdater: Failed to start the qTox updater, removing the update and exiting"; + QString updateDirStr = Settings::getInstance().getSettingsDirPath() + "/update/"; + QDir(updateDirStr).removeRecursively(); + exit(-1); +} diff --git a/src/autoupdate.h b/src/autoupdate.h index 30502c9bc..07424dbc7 100644 --- a/src/autoupdate.h +++ b/src/autoupdate.h @@ -22,7 +22,16 @@ #include #include +/// For now we only support auto updates on Windows, although extending it is not a technical issue. +/// Linux and Mac users are expected to use their package managers or update manually through official channels. +#ifdef Q_OS_WIN +#define AUTOUPDATE_ENABLED 1 +#else +#define AUTOUPDATE_ENABLED 0 +#endif + /// Handles checking and applying updates for qTox +/// Do *NOT* use auto update unless AUTOUPDATE_ENABLED is defined to 1 class AutoUpdater { public: @@ -32,6 +41,13 @@ public: QString id; ///< Unique id of the file QString installpath; ///< Local path including the file name. May be relative to qtox-updater or absolute uint64_t size; ///< Size in bytes of the file + + bool operator==(const UpdateFileMeta& other) + { + return (size == other.size + && id == other.id && installpath == other.installpath + && memcmp(sig, other.sig, crypto_sign_BYTES) == 0); + } }; struct UpdateFile @@ -47,16 +63,33 @@ public: /// Fetch the version string of the last update available from the qTox update server /// Will try to follow qTox's proxy settings, may block and processEvents static QString getUpdateVersion(); - /// Generates a list of files we need to update + /// Will try to download an update, if successful returns true and qTox will apply it after a restart /// Will try to follow qTox's proxy settings, may block and processEvents - static QList genUpdateDiff(); + static bool downloadUpdate(); + /// Returns true if an update is downloaded and ready to be installed + /// If so, call installLocalUpdate. If not, call downloadUpdate. + /// This only checks that we downloaded an update and didn't stop in the middle, not that every file is still valid + static bool isLocalUpdateReady(); + /// Launches the qTox updater to try to install the local update and exits immediately + /// Will not check that the update actually exists, use isLocalUpdateReady first for that + /// The qTox updater will restart us after the update is done + /// Note: If we fail to start the qTox updater, we will delete the update and exit + [[ noreturn ]] static void installLocalUpdate(); protected: /// Parses and validates a flist file. Returns an empty list on error - static QList parseflist(QByteArray flistData); - /// Get the update server's flist and parse it. Returns an empty list on error + static QList parseFlist(QByteArray flistData); + /// Gets the update server's flist. Returns an empty array on error /// Will try to follow qTox's proxy settings, may block and processEvents - static QList getUpdateFlist(); + static QByteArray getUpdateFlist(); + /// Gets the local flist. Returns an empty array on error + static QByteArray getLocalFlist(); + /// Generates a list of files we need to update + static QList genUpdateDiff(QList updateFlist); + /// Tries to fetch the file from the update server. Returns a file with a null QByteArray on error. + /// Note that a file with an empty but non-null QByteArray is not an error, merely a file of size 0. + /// Will try to follow qTox's proxy settings, may block and processEvents + static UpdateFile getUpdateFile(UpdateFileMeta fileMeta); private: AutoUpdater() = delete; @@ -68,6 +101,7 @@ private: static const QString checkURI; ///< URI of the file containing the latest version string static const QString flistURI; ///< URI of the file containing info on each file (hash, signature, size, name, ..) static const QString filesURI; ///< URI of the actual files of the latest version + static const QString updaterBin; ///< Path to the qtox-updater binary static unsigned char key[]; }; diff --git a/src/main.cpp b/src/main.cpp index e014ea128..b737a699e 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -19,6 +19,7 @@ #include "src/ipc.h" #include "src/widget/toxuri.h" #include "src/widget/toxsave.h" +#include "src/autoupdate.h" #include #include #include @@ -87,6 +88,12 @@ int main(int argc, char *argv[]) // Install Unicode 6.1 supporting font QFontDatabase::addApplicationFont("://DejaVuSans.ttf"); + // Check whether we have an update waiting to be installed +#if AUTOUPDATE_ENABLED + if (AutoUpdater::isLocalUpdateReady()) + AutoUpdater::installLocalUpdate(); ///< NORETURN +#endif + // Inter-process communication IPC ipc; ipc.registerEventHandler(&toxURIEventHandler); diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index a3482ef14..28a30f448 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -196,16 +196,6 @@ void Widget::init() addFriendForm = new AddFriendForm; settingsWidget = new SettingsWidget(); - // Check for updates - { - QString newVersion = AutoUpdater::getUpdateVersion(); - if (!newVersion.isEmpty() && newVersion != GIT_VERSION) - { - qWarning() << "New update:"< + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "widget.h" +#include + +int main(int argc, char *argv[]) +{ + QApplication a(argc, argv); + Widget w; + w.show(); + + return a.exec(); +} diff --git a/updater/res.qrc b/updater/res.qrc new file mode 100644 index 000000000..bdcb8e3d3 --- /dev/null +++ b/updater/res.qrc @@ -0,0 +1,5 @@ + + + res/qtox-256x256.png + + diff --git a/updater/res/qtox-256x256.png b/updater/res/qtox-256x256.png new file mode 100644 index 0000000000000000000000000000000000000000..ea5cefa744a4ecdcb06e8e7f362ffe753ff274af GIT binary patch literal 5630 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 literal 0 HcmV?d00001 diff --git a/updater/serialize.cpp b/updater/serialize.cpp new file mode 100644 index 000000000..d16f68f60 --- /dev/null +++ b/updater/serialize.cpp @@ -0,0 +1,254 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "serialize.h" + +QByteArray doubleToData(double num) +{ + union + { + char tab[8]; + double n; + } castUnion; + //char n[8]; + //*((double*) n) = num; + + castUnion.n=num; + return QByteArray(castUnion.tab,8); +} + +QByteArray floatToData(float num) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.n=num; + return QByteArray(castUnion.tab,4); +} + +float dataToFloat(QByteArray data) +{ + union + { + char tab[4]; + float n; + } castUnion; + + castUnion.tab[0]=data.data()[0]; + castUnion.tab[1]=data.data()[1]; + castUnion.tab[2]=data.data()[2]; + castUnion.tab[3]=data.data()[3]; + return castUnion.n; +} + +// Converts a string into PNet string data +QByteArray stringToData(QString str) +{ + QByteArray data(4,0); + // Write the size in a Uint of variable lenght (8-32 bits) + int i=0; + uint num1 = (uint)str.toUtf8().size(); + while (num1 >= 0x80) + { + data[i] = (unsigned char)(num1 | 0x80); i++; + num1 = num1 >> 7; + } + data[i]=num1; + data.resize(i+1); + data+=str.toUtf8(); + return data; +} + +QString dataToString(QByteArray data) +{ + // Variable UInt32 + unsigned char num3; + int num = 0; + int num2 = 0; + int i=0; + do + { + num3 = data[i]; i++; + num |= (num3 & 0x7f) << num2; + num2 += 7; + } while ((num3 & 0x80) != 0); + unsigned int strlen = (uint) num; + + if (!strlen) + return QString(); + + data = data.right(data.size()-i); // Remove the strlen + data.truncate(strlen); + return QString(data); +} + +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data) +{ + uint endvalue=0; + uint value=0; + if (numberOfBits <= 8) + { + endvalue = (uchar)data[0]; + goto done; + } + value = (uchar)data[0]; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + endvalue = (value | ((uint) ((uchar)data[1]) << 8)); + goto done; + } + value |= (uint) (((uchar)data[1]) << 8); + numberOfBits -= 8; + if (numberOfBits <= 8) + { + uint num2 = (uint) (((uchar)data[2]) << 0x10); + endvalue = (value | num2); + goto done; + } + value |= (uint) (((uchar)data[2]) << 0x10); + numberOfBits -= 8; + endvalue = (value | ((uint) (((uchar)data[3]) << 0x18))); + goto done; + + done: + + float num = max - min; + int num2 = (((int) 1) << numberOfBits) - 1; + float num3 = endvalue; + float num4 = num3 / ((float) num2); + return (min + (num4 * num)); +} + +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits) +{ + QByteArray data; + float num = max - min; + float num2 = (value - min) / num; + int num3 = (((int) 1) << numberOfBits) - 1; + uint source = num3 * num2; + + if (numberOfBits <= 8) + { + data += (unsigned char)source; + return data; + } + data += (unsigned char)source; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>8; + return data; + } + data += (unsigned char)source>>8; + numberOfBits -= 8; + if (numberOfBits <= 8) + { + data += (unsigned char)source>>16; + return data; + } + data += (unsigned char)source>>16; + data += (unsigned char)source>>24; + + return data; +} + +uint8_t dataToUint8(QByteArray data) +{ + return (uint8_t)data[0]; +} + +uint16_t dataToUint16(QByteArray data) +{ + return ((uint16_t)(uint8_t)data[0]) + +(((uint16_t)(uint8_t)data[1])<<8); +} + +uint32_t dataToUint32(QByteArray data) +{ + return ((uint32_t)(uint8_t)data[0]) + +(((uint32_t)(uint8_t)data[1])<<8) + +(((uint32_t)(uint8_t)data[2])<<16) + +(((uint32_t)(uint8_t)data[3])<<24); +} + +uint64_t dataToUint64(QByteArray data) +{ + return ((uint64_t)(uint8_t)data[0]) + +(((uint64_t)(uint8_t)data[1])<<8) + +(((uint64_t)(uint8_t)data[2])<<16) + +(((uint64_t)(uint8_t)data[3])<<24) + +(((uint64_t)(uint8_t)data[4])<<32) + +(((uint64_t)(uint8_t)data[5])<<40) + +(((uint64_t)(uint8_t)data[6])<<48) + +(((uint64_t)(uint8_t)data[7])<<56); +} + +unsigned getVUint32Size(QByteArray data) +{ + unsigned lensize=0; + { + unsigned char num3; + do { + num3 = data[lensize]; + lensize++; + } while ((num3 & 0x80) != 0); + } + return lensize; +} + +QByteArray uint8ToData(uint8_t num) +{ + QByteArray data(1,0); + data[0] = (uint8_t)num; + return data; +} + +QByteArray uint16ToData(uint16_t num) +{ + QByteArray data(2,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + return data; +} + +QByteArray uint32ToData(uint32_t num) +{ + QByteArray data(4,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + return data; +} + +QByteArray uint64ToData(uint64_t num) +{ + QByteArray data(8,0); + data[0] = (uint8_t)(num & 0xFF); + data[1] = (uint8_t)((num>>8) & 0xFF); + data[2] = (uint8_t)((num>>16) & 0xFF); + data[3] = (uint8_t)((num>>24) & 0xFF); + data[4] = (uint8_t)((num>>32) & 0xFF); + data[5] = (uint8_t)((num>>40) & 0xFF); + data[6] = (uint8_t)((num>>48) & 0xFF); + data[7] = (uint8_t)((num>>56) & 0xFF); + return data; +} diff --git a/updater/serialize.h b/updater/serialize.h new file mode 100644 index 000000000..54747fe68 --- /dev/null +++ b/updater/serialize.h @@ -0,0 +1,45 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef SERIALIZE_H +#define SERIALIZE_H + +#include +#include +#include + +/// Most of those functions are unsafe unless otherwise specified +/// Do not use them on untrusted data (e.g. check a signature first) + +QByteArray doubleToData(double num); +QByteArray floatToData(float num); +float dataToFloat(QByteArray data); +QByteArray stringToData(QString str); +QString dataToString(QByteArray data); +float dataToRangedSingle(float min, float max, int numberOfBits, QByteArray data); +QByteArray rangedSingleToData(float value, float min, float max, int numberOfBits); +uint8_t dataToUint8(QByteArray data); +uint16_t dataToUint16(QByteArray data); +uint32_t dataToUint32(QByteArray data); +uint64_t dataToUint64(QByteArray data); +unsigned getVUint32Size(QByteArray data); +QByteArray uint8ToData(uint8_t num); +QByteArray uint16ToData(uint16_t num); +QByteArray uint32ToData(uint32_t num); +QByteArray uint64ToData(uint64_t num); + +#endif // SERIALIZE_H diff --git a/updater/settingsDir.cpp b/updater/settingsDir.cpp new file mode 100644 index 000000000..2a5581b44 --- /dev/null +++ b/updater/settingsDir.cpp @@ -0,0 +1,52 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "settingsDir.h" +#include +#include +#include +#include + +const QString FILENAME = "settings.ini"; + +QString getSettingsDirPath() +{ + if (isToxPortableEnabled()) + return "."; + +#ifdef Q_OS_WIN + return QDir::cleanPath(QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + + QDir::separator() + "AppData" + QDir::separator() + "Roaming" + QDir::separator() + "tox"); +#else + return QDir::cleanPath(QStandardPaths::writableLocation(QStandardPaths::ConfigLocation) + QDir::separator() + "tox"); +#endif +} + +bool isToxPortableEnabled() +{ + QFile portableSettings(FILENAME); + if (portableSettings.exists()) + { + QSettings ps(FILENAME, QSettings::IniFormat); + ps.beginGroup("General"); + return ps.value("makeToxPortable", false).toBool(); + } + else + { + return false; + } +} diff --git a/updater/settingsDir.h b/updater/settingsDir.h new file mode 100644 index 000000000..b695fe29f --- /dev/null +++ b/updater/settingsDir.h @@ -0,0 +1,27 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef SETTINGSDIR_H +#define SETTINGSDIR_H + +#include + +QString getSettingsDirPath(); + +bool isToxPortableEnabled(); + +#endif // SETTINGSDIR_H diff --git a/updater/update.cpp b/updater/update.cpp new file mode 100644 index 000000000..a2c8e7da4 --- /dev/null +++ b/updater/update.cpp @@ -0,0 +1,115 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "update.h" +#include "serialize.h" +#include +#include + +unsigned char key[crypto_sign_PUBLICKEYBYTES] = +{ + 0xa5, 0x80, 0xf3, 0xb7, 0xd0, 0x10, 0xc0, 0xf9, 0xd6, 0xcf, 0x48, 0x15, 0x99, 0x70, 0x92, 0x49, + 0xf6, 0xe8, 0xe5, 0xe2, 0x6c, 0x73, 0x8c, 0x48, 0x25, 0xed, 0x01, 0x72, 0xf7, 0x6c, 0x17, 0x28 +}; + +QByteArray getLocalFlist() +{ + QByteArray flist; + + QFile flistFile("flist"); + if (!flistFile.open(QIODevice::ReadOnly)) + { + qWarning() << "getLocalFlist: Can't open local flist"; + return flist; + } + + flist = flistFile.readAll(); + flistFile.close(); + + return flist; +} + +QList genUpdateDiff(QList updateFlist) +{ + QList diff; + QList localFlist = parseFlist(getLocalFlist()); + + for (UpdateFileMeta file : updateFlist) + if (!localFlist.contains(file)) + diff += file; + + return diff; +} + +QList parseFlist(QByteArray flistData) +{ + QList flist; + + if (flistData.isEmpty()) + { + qWarning() << "AutoUpdater::parseflist: Empty data"; + return flist; + } + + // Check version + if (flistData[0] != '1') + { + qWarning() << "AutoUpdater: parseflist: Bad version "<<(uint8_t)flistData[0]; + return flist; + } + flistData = flistData.mid(1); + + // Check signature + if (flistData.size() < (int)(crypto_sign_BYTES)) + { + qWarning() << "AutoUpdater::parseflist: Truncated data"; + return flist; + } + else + { + QByteArray msgData = flistData.mid(crypto_sign_BYTES); + unsigned char* msg = (unsigned char*)msgData.data(); + if (crypto_sign_verify_detached((unsigned char*)flistData.data(), msg, msgData.size(), key) != 0) + { + qCritical() << "AutoUpdater: parseflist: FORGED FLIST FILE"; + return flist; + } + flistData = flistData.mid(crypto_sign_BYTES); + } + + // Parse. We assume no errors handling needed since the signature is valid. + while (!flistData.isEmpty()) + { + UpdateFileMeta newFile; + + memcpy(newFile.sig, flistData.data(), crypto_sign_BYTES); + flistData = flistData.mid(crypto_sign_BYTES); + + newFile.id = dataToString(flistData); + flistData = flistData.mid(newFile.id.size() + getVUint32Size(flistData)); + + newFile.installpath = dataToString(flistData); + flistData = flistData.mid(newFile.installpath.size() + getVUint32Size(flistData)); + + newFile.size = dataToUint64(flistData); + flistData = flistData.mid(8); + + flist += newFile; + } + + return flist; +} diff --git a/updater/update.h b/updater/update.h new file mode 100644 index 000000000..aa2ce0c7e --- /dev/null +++ b/updater/update.h @@ -0,0 +1,55 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef UPDATE_H +#define UPDATE_H + +#include +#include +#include + +struct UpdateFileMeta +{ + unsigned char sig[crypto_sign_BYTES]; ///< Signature of the file (ed25519) + QString id; ///< Unique id of the file + QString installpath; ///< Local path including the file name. May be relative to qtox-updater or absolute + uint64_t size; ///< Size in bytes of the file + + bool operator==(const UpdateFileMeta& other) + { + return (size == other.size + && id == other.id && installpath == other.installpath + && memcmp(sig, other.sig, crypto_sign_BYTES) == 0); + } +}; + +struct UpdateFile +{ + UpdateFileMeta metadata; + QByteArray data; +}; + +/// Gets the local flist. Returns an empty array on error +QByteArray getLocalFlist(); +/// Parses and validates a flist file. Returns an empty list on error +QList parseFlist(QByteArray flistData); +/// Generates a list of files we need to update +QList genUpdateDiff(QList updateFlist); + +extern unsigned char key[crypto_sign_PUBLICKEYBYTES]; + +#endif // UPDATE_H diff --git a/updater/updater.pro b/updater/updater.pro new file mode 100644 index 000000000..e67e149c6 --- /dev/null +++ b/updater/updater.pro @@ -0,0 +1,36 @@ +#------------------------------------------------- +# +# Project created by QtCreator 2014-11-09T21:09:08 +# +#------------------------------------------------- + +QT += core gui + +greaterThan(QT_MAJOR_VERSION, 4): QT += widgets + +TARGET = qtox-updater +TEMPLATE = app + +CONFIG += c++11 + +SOURCES += main.cpp\ + widget.cpp \ + settingsDir.cpp \ + update.cpp \ + serialize.cpp + +HEADERS += widget.h \ + settingsDir.h \ + update.h \ + serialize.h + +FORMS += widget.ui + +RESOURCES += \ + res.qrc + +INCLUDEPATH += libs/include + +RC_FILE = windows/updater.rc + +LIBS += -L$$PWD/libs/lib/ -lsodium diff --git a/updater/widget.cpp b/updater/widget.cpp new file mode 100644 index 000000000..a87813204 --- /dev/null +++ b/updater/widget.cpp @@ -0,0 +1,156 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#include "widget.h" +#include "ui_widget.h" + +#include +#include +#include +#include +#include + +#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); + qApp->processEvents(); +} + +void Widget::fatalError(QString message) +{ + QMessageBox::critical(this,tr("Error"), message+'\n'+tr("qTox will restart now.")); + deleteUpdate(); + startQToxAndExit(); +} + +void Widget::deleteUpdate() +{ + QDir updateDir(getSettingsDirPath()+"/update/"); + updateDir.removeRecursively(); +} + +void Widget::startQToxAndExit() +{ + QProcess::startDetached(QTOX_PATH); + exit(0); +} + +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 updateFlist = parseFlist(updateFlistData); + setProgress(2); + QList diff = genUpdateDiff(updateFlist); + setProgress(4); + for (UpdateFileMeta fileMeta : diff) + if (!QFile::exists(updateDirStr+fileMeta.installpath)) + fatalError(tr("The update is incomplete.")); + setProgress(5); + + /// 2. Check the update (5-50%) + float checkProgressStep = 45/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/diff.size(); + float installProgress = 50; + for (UpdateFileMeta fileMeta : diff) + { + QFile fileFile(updateDirStr+fileMeta.installpath); + if (!fileFile.copy(fileMeta.installpath)) + fatalError(tr("Unable to copy the update's files.")); + + installProgress += installProgressStep; + setProgress(installProgress); + } + setProgress(95); + + /// 4. Delete the update (95-100%) + deleteUpdate(); + setProgress(100); + + /// 5. Start qTox and exit + startQToxAndExit(); +} diff --git a/updater/widget.h b/updater/widget.h new file mode 100644 index 000000000..c757fa7a0 --- /dev/null +++ b/updater/widget.h @@ -0,0 +1,51 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program 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. + This program 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 COPYING file for more details. +*/ + + +#ifndef WIDGET_H +#define WIDGET_H + +#include + +namespace Ui { +class Widget; +} + +class Widget : public QWidget +{ + Q_OBJECT + +public: + explicit Widget(QWidget *parent = 0); + ~Widget(); + + // Utilities + void setProgress(int value); + + // Noreturn + void fatalError(QString message); ///< Calls deleteUpdate and startQToxAndExit + void deleteUpdate(); + void startQToxAndExit(); + +public slots: + // Finds and applies the update + void update(); + +private: + Ui::Widget *ui; +}; + +#endif // WIDGET_H diff --git a/updater/widget.ui b/updater/widget.ui new file mode 100644 index 000000000..25eec0c22 --- /dev/null +++ b/updater/widget.ui @@ -0,0 +1,139 @@ + + + Widget + + + + 0 + 0 + 401 + 224 + + + + qTox Updater + + + + :/res/qtox-256x256.png:/res/qtox-256x256.png + + + + + 0 + 13 + 191 + 191 + + + + + + + :/res/qtox-256x256.png + + + true + + + + + + 206 + 95 + 171 + 20 + + + + 0 + + + 0 + + + Qt::AlignCenter + + + false + + + + + + 205 + 115 + 171 + 20 + + + + Updating qTox ... + + + Qt::AlignCenter + + + + + + 201 + 170 + 181 + 20 + + + + <a href="https://tox.im">https://tox.im</a> + + + Qt::AlignCenter + + + true + + + + + + 200 + 183 + 181 + 20 + + + + <a href="https://github.com/tux3/qtox">https://github.com/tux3/qtox</a> + + + Qt::AlignCenter + + + + + + 195 + 32 + 191 + 31 + + + + + 14 + + + + qTox Update + + + Qt::AlignCenter + + + + + + + + + diff --git a/updater/windows/updater.exe.manifest b/updater/windows/updater.exe.manifest new file mode 100644 index 000000000..626d284d9 --- /dev/null +++ b/updater/windows/updater.exe.manifest @@ -0,0 +1,12 @@ + + + + + + + + + + \ No newline at end of file diff --git a/updater/windows/updater.ico b/updater/windows/updater.ico new file mode 100644 index 0000000000000000000000000000000000000000..b322a59aa40aff41d33f4f855cf3608c3d3702e5 GIT binary patch literal 16958 zcmeI2Yiv|S6vuDz1BqYC7fcit6QwMDKl<xpeN%y`8(;-MDai z?#w-N=KRj=&fc9FNs7R~#fv5Uy-l**ElD>>l4OCPO0q+!ej=|~x6jIgybd`9`3)jZ zV)+iS^*|~hQzpoLGkkPICNr;AYy*gWwvim5-89JeA&*e$yb%WMN)Gnv z;Rkno7_lokNZ`!@*vrC@f29uotpfR7DM=~k7o;xuLXXU!H(&JJvrPwouL%ByX$wE= zeYba&EX&zOWY^AJ&&5W^{wg9C_~_Na?-0>vOLNOcBiz1ia*e*eA08ggpEG;TAEC-maW^+LH~KQ^xn7>X zyh9K>{RRm5{713V_!*f8yz`zb)cGm?xv_IE4GavF@{T>_%QD+CDv&*BN zxu&w+>2&6>@-Q|wmQ%8#WRE4va#d+3PzUAxuZ}-8CG}j_(XOY}IXKvHuqrt*`Mla* zU^(dg^jmI6`;K}wyHeiNw#hE=_nxxuq3EFU$Hm0`()WH}xx&iQC;Cs6&x@P)lfZGG z6#VPfu4`j4tLrzsu%RWK_^rv-uhoqH%4ycLF9qiyXz%~*%u6qRi{a;sG zw^_~Sue`Qy?WS-Ue)HR=E~rm4#?h!xJ^O#F!jNCj{%>-i_!Ht2z70@c-249$ zUWn;nAmYk+DC0QolhPi!iT&w1|Ka7Ifp{*P7?b;R9Cb(3-$xD~d4|@-*4M9pDSmGJ z4}$wNbpFSigDAxEk>Gf)nZI(fa{m2&c>2`o60Cm=3+nR&_%U?;i}>k<_ER7$AbTKZ zA!Cqh7$#Cj>we1f59ywsp4C_f8!_OMo*h1g9Do!R0J`$M(GKMcU){<)`cY`}WsxtOMK4BjfYT%TH&2T3cFmod?NyIODjV+=c{yaZ&MJ z_4x9XJ3BioL!E!gy-PmiVP)CS9;!wTCiVCy&QH8Zd$=q%p8AUNiq}GwKkt#eP98=r zo2JI5rcmXlJzOm|o_mK59jcn-{PnE(PM$nj!o#R#Gc+_*!19m!dcJQd^Q;g9^Q)%giI=^NiRsSn)M(Y;59T)Uu)TBP{>aH(!+ISs?~1E53rq3Ocmd zcif;5;`feN%2{#ZH_V^M-Z}QvczjxQbY95MTJReF2+g$w`*a9% z%^qtSKelggXP)~|=YhWSlM<7@VdWI{bSCg7bE))sM?7^J@yb}|ab+&c?ACf79vKzfzywbXb-fp3X;Z-j{0N9>268G$Dd(<5SE4Dq*)=R4NQH9b?^BPNRAdjZQW5c*uy zLiR#FfsA^EuHS^vcj$wVTLTrb?4X{mlVwkbvdg1wPLEDk>#Lsnnb6e6M@M+e&~~c# z(QZ5CzPg=cL!ab_`Xo(#l8zkl=&2XHWv`}tzM|OkYTveWt@xMlqOx17xs@{I=Ly70 z`AVf6I3(##${#B3LJ>OB<B-eGf5cd{eh k>DCt_M?4xNRFAAhlBO(?M&#+F5xI@DY literal 0 HcmV?d00001 diff --git a/updater/windows/updater.rc b/updater/windows/updater.rc new file mode 100644 index 000000000..e47abcc22 --- /dev/null +++ b/updater/windows/updater.rc @@ -0,0 +1,2 @@ +ID_ICON ICON DISCARDABLE "updater.ico" +1 24 "updater.exe.manifest" \ No newline at end of file From 0054787826c3471f9318cde71198f19585e8dae8 Mon Sep 17 00:00:00 2001 From: "Tux3 / Mlkj / !Lev.uXFMLA" Date: Mon, 10 Nov 2014 00:31:29 +0100 Subject: [PATCH 5/5] Add an option for auto-updates on Windows MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit With the update server hosted on my laptop... Fun fact: Updates are really easy to mess up right now. If I forget to update the update server, everyone gets downgraded. Upload a bad file and everyone will crash. Fun stuff. Upates Are Experimental™ and off by default. --- src/autoupdate.cpp | 22 ++++++++++++++++++++- src/autoupdate.h | 7 +++++++ src/misc/settings.cpp | 12 +++++++++++ src/misc/settings.h | 4 ++++ src/widget/form/settings/generalform.cpp | 11 +++++++++++ src/widget/form/settings/generalform.h | 1 + src/widget/form/settings/generalsettings.ui | 9 ++++++++- src/widget/widget.cpp | 22 +++++++++++++++++++++ src/widget/widget.h | 1 + 9 files changed, 87 insertions(+), 2 deletions(-) diff --git a/src/autoupdate.cpp b/src/autoupdate.cpp index f1ab56136..bb63eaa4e 100644 --- a/src/autoupdate.cpp +++ b/src/autoupdate.cpp @@ -18,12 +18,15 @@ #include "src/autoupdate.h" #include "src/misc/serialize.h" #include "src/misc/settings.h" +#include "src/widget/widget.h" #include #include #include #include #include #include +#include +#include #ifdef Q_OS_WIN #include @@ -37,7 +40,7 @@ const QString AutoUpdater::updaterBin = "qtox-updater.exe"; const QString AutoUpdater::platform; const QString AutoUpdater::updaterBin; #endif -const QString AutoUpdater::updateServer = "http://127.0.0.1"; +const QString AutoUpdater::updateServer = "http://mlkj.bounceme.net"; const QString AutoUpdater::checkURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/version"; const QString AutoUpdater::flistURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/flist"; const QString AutoUpdater::filesURI = AutoUpdater::updateServer+"/qtox/"+AutoUpdater::platform+"/files/"; @@ -392,3 +395,20 @@ fail: QDir(updateDirStr).removeRecursively(); exit(-1); } + +void AutoUpdater::checkUpdatesAsyncInteractive() +{ + QtConcurrent::run(&AutoUpdater::checkUpdatesAsyncInteractiveWorker); +} + +void AutoUpdater::checkUpdatesAsyncInteractiveWorker() +{ + if (!isUpdateAvailable()) + return; + + if (Widget::getInstance()->askMsgboxQuestion(QObject::tr("Update", "The title of a message box"), + QObject::tr("An update is available, do you want to download it now ?\nIt will be installed when qTox restarts."))) + { + downloadUpdate(); + } +} diff --git a/src/autoupdate.h b/src/autoupdate.h index 07424dbc7..a5df59b28 100644 --- a/src/autoupdate.h +++ b/src/autoupdate.h @@ -57,6 +57,10 @@ public: }; public: + /// Connects to the qTox update server, if an updat is found shows a dialog to the user asking to download it + /// Runs asynchronously in its own thread, and will return immediatly + /// Will call isUpdateAvailable, and as such may processEvents + static void checkUpdatesAsyncInteractive(); /// Connects to the qTox update server, returns true if an update is available for download /// Will call getUpdateVersion, and as such may block and processEvents static bool isUpdateAvailable(); @@ -90,6 +94,9 @@ protected: /// Note that a file with an empty but non-null QByteArray is not an error, merely a file of size 0. /// Will try to follow qTox's proxy settings, may block and processEvents static UpdateFile getUpdateFile(UpdateFileMeta fileMeta); + /// Does the actual work for checkUpdatesAsyncInteractive + /// Blocking, but otherwise has the same properties than checkUpdatesAsyncInteractive + static void checkUpdatesAsyncInteractiveWorker(); private: AutoUpdater() = delete; diff --git a/src/misc/settings.cpp b/src/misc/settings.cpp index 8e09051e1..43181ef5b 100644 --- a/src/misc/settings.cpp +++ b/src/misc/settings.cpp @@ -119,6 +119,7 @@ void Settings::load() proxyPort = s.value("proxyPort", 0).toInt(); currentProfile = s.value("currentProfile", "").toString(); autoAwayTime = s.value("autoAwayTime", 10).toInt(); + checkUpdates = s.value("checkUpdates", false).toBool(); s.endGroup(); s.beginGroup("Widgets"); @@ -255,6 +256,7 @@ void Settings::save(QString path) s.setValue("proxyPort", proxyPort); s.setValue("currentProfile", currentProfile); s.setValue("autoAwayTime", autoAwayTime); + s.setValue("checkUpdates", checkUpdates); s.endGroup(); s.beginGroup("Widgets"); @@ -727,6 +729,16 @@ void Settings::setWindowState(const QByteArray &value) windowState = value; } +bool Settings::getCheckUpdates() const +{ + return checkUpdates; +} + +void Settings::setCheckUpdates(bool newValue) +{ + checkUpdates = newValue; +} + QByteArray Settings::getSplitterState() const { return splitterState; diff --git a/src/misc/settings.h b/src/misc/settings.h index e278f307b..813ce7390 100644 --- a/src/misc/settings.h +++ b/src/misc/settings.h @@ -99,6 +99,9 @@ public: int getAutoAwayTime() const; void setAutoAwayTime(int newValue); + bool getCheckUpdates() const; + void setCheckUpdates(bool newValue); + QPixmap getSavedAvatar(const QString& ownerId); void saveAvatar(QPixmap& pic, const QString& ownerId); @@ -219,6 +222,7 @@ private: bool closeToTray; bool minimizeToTray; bool useEmoticons; + bool checkUpdates; bool forceTCP; diff --git a/src/widget/form/settings/generalform.cpp b/src/widget/form/settings/generalform.cpp index d300574c4..91af2ad1e 100644 --- a/src/widget/form/settings/generalform.cpp +++ b/src/widget/form/settings/generalform.cpp @@ -27,6 +27,8 @@ #include #include +#include "src/autoupdate.h" + static QStringList locales = {"bg", "de", "en", "fr", "it", "mannol", "pirate", "pl", "ru", "fi", "sv", "uk"}; static QStringList langs = {"Български", "Deustch", "English", "Français", "Italiano", "mannol", "Pirate", "Polski", "Русский", "Suomi", "Svenska", "Українська"}; @@ -39,6 +41,9 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : bodyUI = new Ui::GeneralSettings; bodyUI->setupUi(this); + + bodyUI->checkUpdates->setVisible(AUTOUPDATE_ENABLED); + bodyUI->checkUpdates->setChecked(Settings::getInstance().getCheckUpdates()); bodyUI->cbEnableIPv6->setChecked(Settings::getInstance().getEnableIPv6()); for (int i = 0; i < langs.size(); i++) @@ -93,6 +98,7 @@ GeneralForm::GeneralForm(SettingsWidget *myParent) : onUseProxyUpdated(); //general + connect(bodyUI->checkUpdates, &QCheckBox::stateChanged, this, &GeneralForm::onCheckUpdateChanged); connect(bodyUI->transComboBox, SIGNAL(currentIndexChanged(int)), this, SLOT(onTranslationUpdated())); connect(bodyUI->cbMakeToxPortable, &QCheckBox::stateChanged, this, &GeneralForm::onMakeToxPortableUpdated); connect(bodyUI->startInTray, &QCheckBox::stateChanged, this, &GeneralForm::onSetAutostartInTray); @@ -280,3 +286,8 @@ void GeneralForm::reloadSmiles() bodyUI->smile4->setToolTip(smiles[3]); bodyUI->smile5->setToolTip(smiles[4]); } + +void GeneralForm::onCheckUpdateChanged() +{ + Settings::getInstance().setCheckUpdates(bodyUI->checkUpdates->isChecked()); +} diff --git a/src/widget/form/settings/generalform.h b/src/widget/form/settings/generalform.h index 0701661f4..fcebef7fe 100644 --- a/src/widget/form/settings/generalform.h +++ b/src/widget/form/settings/generalform.h @@ -51,6 +51,7 @@ private slots: void onReconnectClicked(); void onAutoAcceptFileChange(); void onAutoSaveDirChange(); + void onCheckUpdateChanged(); private: Ui::GeneralSettings *bodyUI; diff --git a/src/widget/form/settings/generalsettings.ui b/src/widget/form/settings/generalsettings.ui index 7bdc0a97c..e7a86f1a9 100644 --- a/src/widget/form/settings/generalsettings.ui +++ b/src/widget/form/settings/generalsettings.ui @@ -40,7 +40,7 @@ 0 0 511 - 739 + 720 @@ -121,6 +121,13 @@ + + + + Check for updates on startup (unstable) + + + diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index 28a30f448..2a789649c 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -245,6 +245,11 @@ void Widget::init() coreThread->start(); addFriendForm->show(*ui); + +#if (AUTOUPDATE_ENABLED) + if (Settings::getInstance().getCheckUpdates()) + AutoUpdater::checkUpdatesAsyncInteractive(); +#endif } void Widget::setTranslation() @@ -1096,3 +1101,20 @@ void Widget::setEnabledThreadsafe(bool enabled) return setEnabled(enabled); } } + +bool Widget::askMsgboxQuestion(const QString& title, const QString& msg) +{ + // We can only display widgets from the GUI thread + if (QThread::currentThread() != qApp->thread()) + { + bool ret; + QMetaObject::invokeMethod(this, "askMsgboxQuestion", Qt::BlockingQueuedConnection, + Q_RETURN_ARG(bool, ret), + Q_ARG(const QString&, title), Q_ARG(const QString&, msg)); + return ret; + } + else + { + return QMessageBox::question(this, title, msg) == QMessageBox::StandardButton::Yes; + } +} diff --git a/src/widget/widget.h b/src/widget/widget.h index 2193fee55..969e8a3bb 100644 --- a/src/widget/widget.h +++ b/src/widget/widget.h @@ -67,6 +67,7 @@ public: Q_INVOKABLE QMessageBox::StandardButton showWarningMsgBox(const QString& title, const QString& msg, QMessageBox::StandardButtons buttonss = QMessageBox::Ok); Q_INVOKABLE void setEnabledThreadsafe(bool enabled); + Q_INVOKABLE bool askMsgboxQuestion(const QString& title, const QString& msg); ~Widget(); virtual void closeEvent(QCloseEvent *event);