mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-21 22:24:07 +01:00
bf5a5b839c
https://github.com/Neargye/semver * Use semver library for version downgrade checking * Add test validating our current version is valid semver
372 lines
11 KiB
C++
372 lines
11 KiB
C++
#include "Updates.hpp"
|
|
|
|
#include "common/Modes.hpp"
|
|
#include "common/NetworkRequest.hpp"
|
|
#include "common/NetworkResult.hpp"
|
|
#include "common/Outcome.hpp"
|
|
#include "common/QLogging.hpp"
|
|
#include "common/Version.hpp"
|
|
#include "Settings.hpp"
|
|
#include "singletons/Paths.hpp"
|
|
#include "util/CombinePath.hpp"
|
|
#include "util/PostToThread.hpp"
|
|
|
|
#include <QDesktopServices>
|
|
#include <QMessageBox>
|
|
#include <QProcess>
|
|
#include <QRegularExpression>
|
|
#include <semver/semver.hpp>
|
|
|
|
namespace chatterino {
|
|
namespace {
|
|
QString currentBranch()
|
|
{
|
|
return getSettings()->betaUpdates ? "beta" : "stable";
|
|
}
|
|
|
|
} // namespace
|
|
|
|
Updates::Updates()
|
|
: currentVersion_(CHATTERINO_VERSION)
|
|
, updateGuideLink_("https://chatterino.com")
|
|
{
|
|
qCDebug(chatterinoUpdate) << "init UpdateManager";
|
|
}
|
|
|
|
Updates &Updates::instance()
|
|
{
|
|
// fourtf: don't add this class to the application class
|
|
static Updates instance;
|
|
|
|
return instance;
|
|
}
|
|
|
|
/// Checks if the online version is newer or older than the current version.
|
|
bool Updates::isDowngradeOf(const QString &online, const QString ¤t)
|
|
{
|
|
semver::version onlineVersion;
|
|
if (!onlineVersion.from_string_noexcept(online.toStdString()))
|
|
{
|
|
qCWarning(chatterinoUpdate) << "Unable to parse online version"
|
|
<< online << "into a proper semver string";
|
|
return false;
|
|
}
|
|
|
|
semver::version currentVersion;
|
|
if (!currentVersion.from_string_noexcept(current.toStdString()))
|
|
{
|
|
qCWarning(chatterinoUpdate) << "Unable to parse current version"
|
|
<< current << "into a proper semver string";
|
|
return false;
|
|
}
|
|
|
|
return onlineVersion < currentVersion;
|
|
}
|
|
|
|
const QString &Updates::getCurrentVersion() const
|
|
{
|
|
return currentVersion_;
|
|
}
|
|
|
|
const QString &Updates::getOnlineVersion() const
|
|
{
|
|
return onlineVersion_;
|
|
}
|
|
|
|
void Updates::installUpdates()
|
|
{
|
|
if (this->status_ != UpdateAvailable)
|
|
{
|
|
assert(false);
|
|
return;
|
|
}
|
|
|
|
#ifdef Q_OS_MACOS
|
|
QMessageBox *box = new QMessageBox(
|
|
QMessageBox::Information, "Chatterino Update",
|
|
"A link will open in your browser. Download and install to update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->exec();
|
|
QDesktopServices::openUrl(this->updateExe_);
|
|
#elif defined Q_OS_LINUX
|
|
QMessageBox *box =
|
|
new QMessageBox(QMessageBox::Information, "Chatterino Update",
|
|
"Automatic updates are currently not available on "
|
|
"linux. Please redownload the app to update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->exec();
|
|
QDesktopServices::openUrl(this->updateGuideLink_);
|
|
#elif defined Q_OS_WIN
|
|
if (getPaths()->isPortable())
|
|
{
|
|
QMessageBox *box =
|
|
new QMessageBox(QMessageBox::Information, "Chatterino Update",
|
|
"Chatterino is downloading the update "
|
|
"in the background and will run the "
|
|
"updater once it is finished.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->show();
|
|
|
|
NetworkRequest(this->updatePortable_)
|
|
.timeout(600000)
|
|
.onError([this](NetworkResult) {
|
|
this->setStatus_(DownloadFailed);
|
|
|
|
postToThread([] {
|
|
QMessageBox *box = new QMessageBox(
|
|
QMessageBox::Information, "Chatterino Update",
|
|
"Failed while trying to download the update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->show();
|
|
box->raise();
|
|
});
|
|
})
|
|
.onSuccess([this](auto result) -> Outcome {
|
|
QByteArray object = result.getData();
|
|
auto filename =
|
|
combinePath(getPaths()->miscDirectory, "update.zip");
|
|
|
|
QFile file(filename);
|
|
file.open(QIODevice::Truncate | QIODevice::WriteOnly);
|
|
|
|
if (file.write(object) == -1)
|
|
{
|
|
this->setStatus_(WriteFileFailed);
|
|
return Failure;
|
|
}
|
|
file.flush();
|
|
file.close();
|
|
|
|
QProcess::startDetached(
|
|
combinePath(QCoreApplication::applicationDirPath(),
|
|
"updater.1/ChatterinoUpdater.exe"),
|
|
{filename, "restart"});
|
|
|
|
QApplication::exit(0);
|
|
return Success;
|
|
})
|
|
.execute();
|
|
this->setStatus_(Downloading);
|
|
}
|
|
else
|
|
{
|
|
QMessageBox *box =
|
|
new QMessageBox(QMessageBox::Information, "Chatterino Update",
|
|
"Chatterino is downloading the update "
|
|
"in the background and will run the "
|
|
"updater once it is finished.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->show();
|
|
|
|
NetworkRequest(this->updateExe_)
|
|
.timeout(600000)
|
|
.onError([this](NetworkResult) {
|
|
this->setStatus_(DownloadFailed);
|
|
|
|
QMessageBox *box = new QMessageBox(
|
|
QMessageBox::Information, "Chatterino Update",
|
|
"Failed to download the update. \n\nTry manually "
|
|
"downloading the update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->exec();
|
|
})
|
|
.onSuccess([this](auto result) -> Outcome {
|
|
QByteArray object = result.getData();
|
|
auto filePath =
|
|
combinePath(getPaths()->miscDirectory, "Update.exe");
|
|
|
|
QFile file(filePath);
|
|
file.open(QIODevice::Truncate | QIODevice::WriteOnly);
|
|
|
|
if (file.write(object) == -1)
|
|
{
|
|
this->setStatus_(WriteFileFailed);
|
|
QMessageBox *box = new QMessageBox(
|
|
QMessageBox::Information, "Chatterino Update",
|
|
"Failed to save the update file. This could be due to "
|
|
"window settings or antivirus software.\n\nTry "
|
|
"manually "
|
|
"downloading the update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->exec();
|
|
|
|
QDesktopServices::openUrl(this->updateExe_);
|
|
return Failure;
|
|
}
|
|
file.flush();
|
|
file.close();
|
|
|
|
if (QProcess::startDetached(filePath, {}))
|
|
{
|
|
QApplication::exit(0);
|
|
}
|
|
else
|
|
{
|
|
QMessageBox *box = new QMessageBox(
|
|
QMessageBox::Information, "Chatterino Update",
|
|
"Failed to execute update binary. This could be due to "
|
|
"window "
|
|
"settings or antivirus software.\n\nTry manually "
|
|
"downloading "
|
|
"the update.");
|
|
box->setAttribute(Qt::WA_DeleteOnClose);
|
|
box->exec();
|
|
|
|
QDesktopServices::openUrl(this->updateExe_);
|
|
}
|
|
|
|
return Success;
|
|
})
|
|
.execute();
|
|
this->setStatus_(Downloading);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void Updates::checkForUpdates()
|
|
{
|
|
auto version = Version::instance();
|
|
|
|
if (!version.isSupportedOS())
|
|
{
|
|
qCDebug(chatterinoUpdate)
|
|
<< "Update checking disabled because OS doesn't appear to be one "
|
|
"of Windows, GNU/Linux or macOS.";
|
|
return;
|
|
}
|
|
|
|
// Disable updates on Flatpak
|
|
if (version.isFlatpak())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Disable updates if on nightly
|
|
if (Modes::instance().isNightly)
|
|
{
|
|
return;
|
|
}
|
|
|
|
QString url =
|
|
"https://notitia.chatterino.com/version/chatterino/" CHATTERINO_OS "/" +
|
|
currentBranch();
|
|
|
|
NetworkRequest(url)
|
|
.timeout(60000)
|
|
.onSuccess([this](auto result) -> Outcome {
|
|
auto object = result.parseJson();
|
|
/// Version available on every platform
|
|
QJsonValue version_val = object.value("version");
|
|
|
|
if (!version_val.isString())
|
|
{
|
|
this->setStatus_(SearchFailed);
|
|
qCDebug(chatterinoUpdate) << "error updating";
|
|
return Failure;
|
|
}
|
|
|
|
#if defined Q_OS_WIN || defined Q_OS_MACOS
|
|
/// Downloads an installer for the new version
|
|
QJsonValue updateExe_val = object.value("updateexe");
|
|
if (!updateExe_val.isString())
|
|
{
|
|
this->setStatus_(SearchFailed);
|
|
qCDebug(chatterinoUpdate) << "error updating";
|
|
return Failure;
|
|
}
|
|
this->updateExe_ = updateExe_val.toString();
|
|
|
|
# ifdef Q_OS_WIN
|
|
/// Windows portable
|
|
QJsonValue portable_val = object.value("portable_download");
|
|
if (!portable_val.isString())
|
|
{
|
|
this->setStatus_(SearchFailed);
|
|
qCDebug(chatterinoUpdate) << "error updating";
|
|
return Failure;
|
|
}
|
|
this->updatePortable_ = portable_val.toString();
|
|
# endif
|
|
|
|
#elif defined Q_OS_LINUX
|
|
QJsonValue updateGuide_val = object.value("updateguide");
|
|
if (updateGuide_val.isString())
|
|
{
|
|
this->updateGuideLink_ = updateGuide_val.toString();
|
|
}
|
|
#else
|
|
return Failure;
|
|
#endif
|
|
|
|
/// Current version
|
|
this->onlineVersion_ = version_val.toString();
|
|
|
|
/// Update available :)
|
|
if (this->currentVersion_ != this->onlineVersion_)
|
|
{
|
|
this->setStatus_(UpdateAvailable);
|
|
this->isDowngrade_ = Updates::isDowngradeOf(
|
|
this->onlineVersion_, this->currentVersion_);
|
|
}
|
|
else
|
|
{
|
|
this->setStatus_(NoUpdateAvailable);
|
|
}
|
|
return Failure;
|
|
})
|
|
.execute();
|
|
this->setStatus_(Searching);
|
|
}
|
|
|
|
Updates::Status Updates::getStatus() const
|
|
{
|
|
return this->status_;
|
|
}
|
|
|
|
bool Updates::shouldShowUpdateButton() const
|
|
{
|
|
switch (this->getStatus())
|
|
{
|
|
case UpdateAvailable:
|
|
case SearchFailed:
|
|
case Downloading:
|
|
case DownloadFailed:
|
|
case WriteFileFailed:
|
|
return true;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool Updates::isError() const
|
|
{
|
|
switch (this->getStatus())
|
|
{
|
|
case SearchFailed:
|
|
case DownloadFailed:
|
|
case WriteFileFailed:
|
|
return true;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool Updates::isDowngrade() const
|
|
{
|
|
return this->isDowngrade_;
|
|
}
|
|
|
|
void Updates::setStatus_(Status status)
|
|
{
|
|
if (this->status_ != status)
|
|
{
|
|
this->status_ = status;
|
|
postToThread([this, status] {
|
|
this->statusUpdated.invoke(status);
|
|
});
|
|
}
|
|
}
|
|
|
|
} // namespace chatterino
|