Merge branch 'feature_updater' of github.com:MultiMC/MultiMC5 into feature_updater
This commit is contained in:
commit
220e07aef4
69
MultiMC.cpp
69
MultiMC.cpp
@ -2,10 +2,12 @@
|
|||||||
#include "MultiMC.h"
|
#include "MultiMC.h"
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QFileInfo>
|
||||||
#include <QNetworkAccessManager>
|
#include <QNetworkAccessManager>
|
||||||
#include <QTranslator>
|
#include <QTranslator>
|
||||||
#include <QLibraryInfo>
|
#include <QLibraryInfo>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
|
#include <QStringList>
|
||||||
|
|
||||||
#include "gui/MainWindow.h"
|
#include "gui/MainWindow.h"
|
||||||
#include "gui/dialogs/VersionSelectDialog.h"
|
#include "gui/dialogs/VersionSelectDialog.h"
|
||||||
@ -409,6 +411,65 @@ std::shared_ptr<JavaVersionList> MultiMC::javalist()
|
|||||||
return m_javalist;
|
return m_javalist;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef WINDOWS
|
||||||
|
#define UPDATER_BIN "updater.exe"
|
||||||
|
#elif LINUX
|
||||||
|
#define UPDATER_BIN "updater"
|
||||||
|
#elif OSX
|
||||||
|
#define UPDATER_BIN "updater"
|
||||||
|
#else
|
||||||
|
#error Unsupported operating system.
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void MultiMC::installUpdates(const QString& updateFilesDir, bool restartOnFinish)
|
||||||
|
{
|
||||||
|
QLOG_INFO() << "Installing updates.";
|
||||||
|
#if LINUX
|
||||||
|
// On Linux, the MultiMC executable file is actually in the bin folder inside the installation directory.
|
||||||
|
// This means that MultiMC's *actual* install path is the parent folder.
|
||||||
|
// We need to tell the updater to run with this directory as the install path, rather than the bin folder where the executable is.
|
||||||
|
// On other operating systems, we'll just use the path to the executable.
|
||||||
|
QString appDir = QFileInfo(MMC->applicationDirPath()).dir().path();
|
||||||
|
|
||||||
|
// On Linux, we also need to set the finish command to the launch script, rather than the binary.
|
||||||
|
QString finishCmd = PathCombine(appDir, "MultiMC");
|
||||||
|
#else
|
||||||
|
QString appDir = MMC->applicationDirPath();
|
||||||
|
QString finishCmd = MMC->applicationFilePath();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Build the command we'll use to run the updater.
|
||||||
|
// Note, the above comment about the app dir path on Linux is irrelevant here because the updater binary is always in the
|
||||||
|
// same folder as the main binary.
|
||||||
|
QString updaterBinary = PathCombine(MMC->applicationDirPath(), UPDATER_BIN);
|
||||||
|
QStringList args;
|
||||||
|
// ./updater --install-dir $INSTALL_DIR --package-dir $UPDATEFILES_DIR --script $UPDATEFILES_DIR/file_list.xml --wait $PID --mode main
|
||||||
|
args << "--install-dir" << appDir;
|
||||||
|
args << "--package-dir" << updateFilesDir;
|
||||||
|
args << "--script" << PathCombine(updateFilesDir, "file_list.xml");
|
||||||
|
args << "--wait" << QString::number(MMC->applicationPid());
|
||||||
|
|
||||||
|
if (restartOnFinish)
|
||||||
|
args << "--finish-cmd" << finishCmd;
|
||||||
|
|
||||||
|
QLOG_INFO() << "Running updater with command" << updaterBinary << args.join(" ");
|
||||||
|
|
||||||
|
QProcess::startDetached(updaterBinary, args);
|
||||||
|
|
||||||
|
// Now that we've started the updater, quit MultiMC.
|
||||||
|
MMC->quit();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MultiMC::setUpdateOnExit(const QString& updateFilesDir)
|
||||||
|
{
|
||||||
|
m_updateOnExitPath = updateFilesDir;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString MultiMC::getExitUpdatePath() const
|
||||||
|
{
|
||||||
|
return m_updateOnExitPath;
|
||||||
|
}
|
||||||
|
|
||||||
int main_gui(MultiMC &app)
|
int main_gui(MultiMC &app)
|
||||||
{
|
{
|
||||||
// show main window
|
// show main window
|
||||||
@ -417,7 +478,13 @@ int main_gui(MultiMC &app)
|
|||||||
mainWin.restoreGeometry(QByteArray::fromBase64(MMC->settings()->get("MainWindowGeometry").toByteArray()));
|
mainWin.restoreGeometry(QByteArray::fromBase64(MMC->settings()->get("MainWindowGeometry").toByteArray()));
|
||||||
mainWin.show();
|
mainWin.show();
|
||||||
mainWin.checkSetDefaultJava();
|
mainWin.checkSetDefaultJava();
|
||||||
return app.exec();
|
auto exitCode = app.exec();
|
||||||
|
|
||||||
|
// Update if necessary.
|
||||||
|
if (!app.getExitUpdatePath().isEmpty())
|
||||||
|
app.installUpdates(app.getExitUpdatePath(), false);
|
||||||
|
|
||||||
|
return exitCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
|
18
MultiMC.h
18
MultiMC.h
@ -98,6 +98,22 @@ public:
|
|||||||
|
|
||||||
std::shared_ptr<JavaVersionList> javalist();
|
std::shared_ptr<JavaVersionList> javalist();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Installs update from the given update files directory.
|
||||||
|
*/
|
||||||
|
void installUpdates(const QString& updateFilesDir, bool restartOnFinish=false);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Sets MultiMC to install updates from the given directory when it exits.
|
||||||
|
*/
|
||||||
|
void setUpdateOnExit(const QString& updateFilesDir);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Gets the path to install updates from on exit.
|
||||||
|
* If this is an empty string, no updates should be installed on exit.
|
||||||
|
*/
|
||||||
|
QString getExitUpdatePath() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void initLogger();
|
void initLogger();
|
||||||
|
|
||||||
@ -124,6 +140,8 @@ private:
|
|||||||
QsLogging::DestinationPtr m_fileDestination;
|
QsLogging::DestinationPtr m_fileDestination;
|
||||||
QsLogging::DestinationPtr m_debugDestination;
|
QsLogging::DestinationPtr m_debugDestination;
|
||||||
|
|
||||||
|
QString m_updateOnExitPath;
|
||||||
|
|
||||||
Status m_status = MultiMC::Failed;
|
Status m_status = MultiMC::Failed;
|
||||||
MultiMCVersion m_version;
|
MultiMCVersion m_version;
|
||||||
};
|
};
|
||||||
|
@ -439,20 +439,31 @@ void MainWindow::updateAvailable(QString repo, QString versionName, int versionI
|
|||||||
QLOG_INFO() << "Update will be installed later.";
|
QLOG_INFO() << "Update will be installed later.";
|
||||||
break;
|
break;
|
||||||
case UPDATE_NOW:
|
case UPDATE_NOW:
|
||||||
{
|
downloadUpdates(repo, versionId);
|
||||||
QLOG_INFO() << "Installing update.";
|
|
||||||
ProgressDialog updateDlg(this);
|
|
||||||
DownloadUpdateTask updateTask(repo, versionId, &updateDlg);
|
|
||||||
updateDlg.exec(&updateTask);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case UPDATE_ONEXIT:
|
case UPDATE_ONEXIT:
|
||||||
// TODO: Implement installing updates on exit.
|
downloadUpdates(repo, versionId, true);
|
||||||
QLOG_INFO() << "Installing on exit is not implemented yet.";
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::downloadUpdates(QString repo, int versionId, bool installOnExit)
|
||||||
|
{
|
||||||
|
QLOG_INFO() << "Downloading updates.";
|
||||||
|
// TODO: If the user chooses to update on exit, we should download updates in the background.
|
||||||
|
// Doing so is a bit complicated, because we'd have to make sure it finished downloading before actually exiting MultiMC.
|
||||||
|
ProgressDialog updateDlg(this);
|
||||||
|
DownloadUpdateTask updateTask(repo, versionId, &updateDlg);
|
||||||
|
// If the task succeeds, install the updates.
|
||||||
|
if (updateDlg.exec(&updateTask))
|
||||||
|
{
|
||||||
|
if (installOnExit)
|
||||||
|
MMC->setUpdateOnExit(updateTask.updateFilesDir());
|
||||||
|
else
|
||||||
|
MMC->installUpdates(updateTask.updateFilesDir());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::onCatToggled(bool state)
|
void MainWindow::onCatToggled(bool state)
|
||||||
{
|
{
|
||||||
setCatBackground(state);
|
setCatBackground(state);
|
||||||
|
@ -169,6 +169,11 @@ slots:
|
|||||||
|
|
||||||
void repopulateAccountsMenu();
|
void repopulateAccountsMenu();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Runs the DownloadUpdateTask and installs updates.
|
||||||
|
*/
|
||||||
|
void downloadUpdates(QString repo, int versionId, bool installOnExit=false);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool eventFilter(QObject *obj, QEvent *ev);
|
bool eventFilter(QObject *obj, QEvent *ev);
|
||||||
void setCatBackground(bool enabled);
|
void setCatBackground(bool enabled);
|
||||||
|
@ -41,6 +41,13 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="btnUpdateOnExit">
|
||||||
|
<property name="text">
|
||||||
|
<string>Update after MultiMC closes</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="btnUpdateLater">
|
<widget class="QPushButton" name="btnUpdateLater">
|
||||||
<property name="sizePolicy">
|
<property name="sizePolicy">
|
||||||
|
@ -63,6 +63,15 @@ void FileDownload::start()
|
|||||||
request.setRawHeader(QString("If-None-Match").toLatin1(), m_expected_md5.toLatin1());
|
request.setRawHeader(QString("If-None-Match").toLatin1(), m_expected_md5.toLatin1());
|
||||||
request.setHeader(QNetworkRequest::UserAgentHeader, "MultiMC/5.0 (Uncached)");
|
request.setHeader(QNetworkRequest::UserAgentHeader, "MultiMC/5.0 (Uncached)");
|
||||||
|
|
||||||
|
// Go ahead and try to open the file.
|
||||||
|
// If we don't do this, empty files won't be created, which breaks the updater.
|
||||||
|
// Plus, this way, we don't end up starting a download for a file we can't open.
|
||||||
|
if (!m_output_file.open(QIODevice::WriteOnly))
|
||||||
|
{
|
||||||
|
emit failed(index_within_job);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
auto worker = MMC->qnam();
|
auto worker = MMC->qnam();
|
||||||
QNetworkReply *rep = worker->get(request);
|
QNetworkReply *rep = worker->get(request);
|
||||||
|
|
||||||
|
@ -24,6 +24,8 @@
|
|||||||
#include <QTemporaryDir>
|
#include <QTemporaryDir>
|
||||||
#include <QCryptographicHash>
|
#include <QCryptographicHash>
|
||||||
|
|
||||||
|
#include <QDomDocument>
|
||||||
|
|
||||||
|
|
||||||
DownloadUpdateTask::DownloadUpdateTask(QString repoUrl, int versionId, QObject* parent) :
|
DownloadUpdateTask::DownloadUpdateTask(QString repoUrl, int versionId, QObject* parent) :
|
||||||
Task(parent)
|
Task(parent)
|
||||||
@ -197,10 +199,11 @@ void DownloadUpdateTask::parseVersionInfo(VersionInfoFileEnum vfile, VersionFile
|
|||||||
|
|
||||||
VersionFileEntry file{
|
VersionFileEntry file{
|
||||||
fileObj.value("Path").toString(),
|
fileObj.value("Path").toString(),
|
||||||
fileObj.value("Executable").toBool(false),
|
fileObj.value("Perms").toVariant().toInt(),
|
||||||
FileSourceList(),
|
FileSourceList(),
|
||||||
fileObj.value("MD5").toString(),
|
fileObj.value("MD5").toString(),
|
||||||
};
|
};
|
||||||
|
QLOG_DEBUG() << "File" << file.path << "with perms" << file.mode;
|
||||||
|
|
||||||
QJsonArray sourceArray = fileObj.value("Sources").toArray();
|
QJsonArray sourceArray = fileObj.value("Sources").toArray();
|
||||||
for (QJsonValue val : sourceArray)
|
for (QJsonValue val : sourceArray)
|
||||||
@ -274,7 +277,7 @@ void DownloadUpdateTask::processFileLists()
|
|||||||
QLOG_DEBUG() << "Will download" << entry.path << "from" << source.url;
|
QLOG_DEBUG() << "Will download" << entry.path << "from" << source.url;
|
||||||
|
|
||||||
// Download it to updatedir/<filepath>-<md5> where filepath is the file's path with slashes replaced by underscores.
|
// Download it to updatedir/<filepath>-<md5> where filepath is the file's path with slashes replaced by underscores.
|
||||||
QString dlPath = PathCombine(m_updateFilesDir.path(), entry.path.replace("/", "_"));
|
QString dlPath = PathCombine(m_updateFilesDir.path(), QString(entry.path).replace("/", "_"));
|
||||||
|
|
||||||
// We need to download the file to the updatefiles folder and add a task to copy it to its install path.
|
// We need to download the file to the updatefiles folder and add a task to copy it to its install path.
|
||||||
FileDownloadPtr download = FileDownload::make(source.url, dlPath);
|
FileDownloadPtr download = FileDownload::make(source.url, dlPath);
|
||||||
@ -283,7 +286,7 @@ void DownloadUpdateTask::processFileLists()
|
|||||||
netJob->addNetAction(download);
|
netJob->addNetAction(download);
|
||||||
|
|
||||||
// Now add a copy operation to our operations list to install the file.
|
// Now add a copy operation to our operations list to install the file.
|
||||||
m_operationList.append(UpdateOperation::CopyOp(dlPath, entry.path));
|
m_operationList.append(UpdateOperation::CopyOp(dlPath, entry.path, entry.mode));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -300,7 +303,75 @@ void DownloadUpdateTask::processFileLists()
|
|||||||
m_filesNetJob.reset(netJob);
|
m_filesNetJob.reset(netJob);
|
||||||
netJob->start();
|
netJob->start();
|
||||||
|
|
||||||
// TODO: Write update operations to a file for the update installer.
|
writeInstallScript(m_operationList, PathCombine(m_updateFilesDir.path(), "file_list.xml"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void DownloadUpdateTask::writeInstallScript(UpdateOperationList& opsList, QString scriptFile)
|
||||||
|
{
|
||||||
|
// Build the base structure of the XML document.
|
||||||
|
QDomDocument doc;
|
||||||
|
|
||||||
|
QDomElement root = doc.createElement("update");
|
||||||
|
root.setAttribute("version", "3");
|
||||||
|
doc.appendChild(root);
|
||||||
|
|
||||||
|
QDomElement installFiles = doc.createElement("install");
|
||||||
|
root.appendChild(installFiles);
|
||||||
|
|
||||||
|
QDomElement removeFiles = doc.createElement("uninstall");
|
||||||
|
root.appendChild(removeFiles);
|
||||||
|
|
||||||
|
// Write the operation list to the XML document.
|
||||||
|
for (UpdateOperation op : opsList)
|
||||||
|
{
|
||||||
|
QDomElement file = doc.createElement("file");
|
||||||
|
|
||||||
|
switch (op.type)
|
||||||
|
{
|
||||||
|
case UpdateOperation::OP_COPY:
|
||||||
|
{
|
||||||
|
// Install the file.
|
||||||
|
QDomElement name = doc.createElement("source");
|
||||||
|
QDomElement path = doc.createElement("dest");
|
||||||
|
QDomElement mode = doc.createElement("mode");
|
||||||
|
name.appendChild(doc.createTextNode(op.file));
|
||||||
|
path.appendChild(doc.createTextNode(op.dest));
|
||||||
|
// We need to add a 0 at the beginning here, because Qt doesn't convert to octal correctly.
|
||||||
|
mode.appendChild(doc.createTextNode("0" + QString::number(op.mode, 8)));
|
||||||
|
file.appendChild(name);
|
||||||
|
file.appendChild(path);
|
||||||
|
file.appendChild(mode);
|
||||||
|
installFiles.appendChild(file);
|
||||||
|
QLOG_DEBUG() << "Will install file" << op.file;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case UpdateOperation::OP_DELETE:
|
||||||
|
{
|
||||||
|
// Delete the file.
|
||||||
|
file.appendChild(doc.createTextNode(op.file));
|
||||||
|
removeFiles.appendChild(file);
|
||||||
|
QLOG_DEBUG() << "Will remove file" << op.file;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
QLOG_WARN() << "Can't write update operation of type" << op.type << "to file. Not implemented.";
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write the XML document to the file.
|
||||||
|
QFile outFile(scriptFile);
|
||||||
|
|
||||||
|
if (outFile.open(QIODevice::WriteOnly))
|
||||||
|
{
|
||||||
|
outFile.write(doc.toByteArray());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
emitFailed(tr("Failed to write update script file."));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DownloadUpdateTask::fileDownloadFinished()
|
void DownloadUpdateTask::fileDownloadFinished()
|
||||||
@ -320,3 +391,8 @@ void DownloadUpdateTask::fileDownloadProgressChanged(qint64 current, qint64 tota
|
|||||||
setProgress((int)(((float)current / (float)total)*100));
|
setProgress((int)(((float)current / (float)total)*100));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString DownloadUpdateTask::updateFilesDir()
|
||||||
|
{
|
||||||
|
return m_updateFilesDir.path();
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -29,6 +29,11 @@ class DownloadUpdateTask : public Task
|
|||||||
public:
|
public:
|
||||||
explicit DownloadUpdateTask(QString repoUrl, int versionId, QObject* parent=0);
|
explicit DownloadUpdateTask(QString repoUrl, int versionId, QObject* parent=0);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Gets the directory that contains the update files.
|
||||||
|
*/
|
||||||
|
QString updateFilesDir();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// TODO: We should probably put these data structures into a separate header...
|
// TODO: We should probably put these data structures into a separate header...
|
||||||
|
|
||||||
@ -57,7 +62,7 @@ protected:
|
|||||||
struct VersionFileEntry
|
struct VersionFileEntry
|
||||||
{
|
{
|
||||||
QString path;
|
QString path;
|
||||||
bool isExecutable;
|
int mode;
|
||||||
FileSourceList sources;
|
FileSourceList sources;
|
||||||
QString md5;
|
QString md5;
|
||||||
};
|
};
|
||||||
@ -70,9 +75,9 @@ protected:
|
|||||||
*/
|
*/
|
||||||
struct UpdateOperation
|
struct UpdateOperation
|
||||||
{
|
{
|
||||||
static UpdateOperation CopyOp(QString fsource, QString fdest) { return UpdateOperation{OP_COPY, fsource, fdest, 644}; }
|
static UpdateOperation CopyOp(QString fsource, QString fdest, int fmode=0644) { return UpdateOperation{OP_COPY, fsource, fdest, fmode}; }
|
||||||
static UpdateOperation MoveOp(QString fsource, QString fdest) { return UpdateOperation{OP_MOVE, fsource, fdest, 644}; }
|
static UpdateOperation MoveOp(QString fsource, QString fdest, int fmode=0644) { return UpdateOperation{OP_MOVE, fsource, fdest, fmode}; }
|
||||||
static UpdateOperation DeleteOp(QString file) { return UpdateOperation{OP_DELETE, file, "", 644}; }
|
static UpdateOperation DeleteOp(QString file) { return UpdateOperation{OP_DELETE, file, "", 0644}; }
|
||||||
static UpdateOperation ChmodOp(QString file, int fmode) { return UpdateOperation{OP_CHMOD, file, "", fmode}; }
|
static UpdateOperation ChmodOp(QString file, int fmode) { return UpdateOperation{OP_CHMOD, file, "", fmode}; }
|
||||||
|
|
||||||
//! Specifies the type of operation that this is.
|
//! Specifies the type of operation that this is.
|
||||||
@ -84,8 +89,8 @@ protected:
|
|||||||
OP_CHMOD,
|
OP_CHMOD,
|
||||||
} type;
|
} type;
|
||||||
|
|
||||||
//! The source file. If this is a DELETE or CHMOD operation, this is the file that will be modified.
|
//! The file to operate on. If this is a DELETE or CHMOD operation, this is the file that will be modified.
|
||||||
QString source;
|
QString file;
|
||||||
|
|
||||||
//! The destination file. If this is a DELETE or CHMOD operation, this field will be ignored.
|
//! The destination file. If this is a DELETE or CHMOD operation, this field will be ignored.
|
||||||
QString dest;
|
QString dest;
|
||||||
@ -145,6 +150,11 @@ protected:
|
|||||||
*/
|
*/
|
||||||
virtual void processFileLists();
|
virtual void processFileLists();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Takes the operations list and writes an install script for the updater to the update files directory.
|
||||||
|
*/
|
||||||
|
virtual void writeInstallScript(UpdateOperationList& opsList, QString scriptFile);
|
||||||
|
|
||||||
VersionFileList m_downloadList;
|
VersionFileList m_downloadList;
|
||||||
UpdateOperationList m_operationList;
|
UpdateOperationList m_operationList;
|
||||||
|
|
||||||
|
@ -7,7 +7,9 @@
|
|||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
#define PLATFORM_WINDOWS
|
#define PLATFORM_WINDOWS
|
||||||
|
#define WIN32_LEAN_AND_MEAN
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
#include <shellapi.h>
|
||||||
|
|
||||||
// disable warnings about exception specifications,
|
// disable warnings about exception specifications,
|
||||||
// which are not implemented in Visual C++
|
// which are not implemented in Visual C++
|
||||||
|
@ -51,6 +51,11 @@ void UpdateInstaller::setForceElevated(bool elevated)
|
|||||||
m_forceElevated = elevated;
|
m_forceElevated = elevated;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void UpdateInstaller::setFinishCmd(const std::string& cmd)
|
||||||
|
{
|
||||||
|
m_finishCmd = cmd;
|
||||||
|
}
|
||||||
|
|
||||||
std::list<std::string> UpdateInstaller::updaterArgs() const
|
std::list<std::string> UpdateInstaller::updaterArgs() const
|
||||||
{
|
{
|
||||||
std::list<std::string> args;
|
std::list<std::string> args;
|
||||||
@ -266,7 +271,7 @@ void UpdateInstaller::revert()
|
|||||||
|
|
||||||
void UpdateInstaller::installFile(const UpdateScriptFile& file)
|
void UpdateInstaller::installFile(const UpdateScriptFile& file)
|
||||||
{
|
{
|
||||||
std::string destPath = m_installDir + '/' + file.path;
|
std::string destPath = file.dest;
|
||||||
std::string target = file.linkTarget;
|
std::string target = file.linkTarget;
|
||||||
|
|
||||||
// backup the existing file if any
|
// backup the existing file if any
|
||||||
@ -279,9 +284,7 @@ void UpdateInstaller::installFile(const UpdateScriptFile& file)
|
|||||||
FileUtils::mkpath(destDir.c_str());
|
FileUtils::mkpath(destDir.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (target.empty())
|
std::string sourceFile = file.source;
|
||||||
{
|
|
||||||
std::string sourceFile = m_packageDir + '/' + FileUtils::fileName(file.path.c_str());
|
|
||||||
if (!FileUtils::fileExists(sourceFile.c_str()))
|
if (!FileUtils::fileExists(sourceFile.c_str()))
|
||||||
{
|
{
|
||||||
throw "Source file does not exist: " + sourceFile;
|
throw "Source file does not exist: " + sourceFile;
|
||||||
@ -290,12 +293,6 @@ void UpdateInstaller::installFile(const UpdateScriptFile& file)
|
|||||||
|
|
||||||
// set the permissions on the newly extracted file
|
// set the permissions on the newly extracted file
|
||||||
FileUtils::chmod(destPath.c_str(),file.permissions);
|
FileUtils::chmod(destPath.c_str(),file.permissions);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// create the symlink
|
|
||||||
FileUtils::createSymLink(destPath.c_str(),target.c_str());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void UpdateInstaller::installFiles()
|
void UpdateInstaller::installFiles()
|
||||||
@ -391,19 +388,9 @@ void UpdateInstaller::restartMainApp()
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
std::string command;
|
std::string command = m_finishCmd;
|
||||||
std::list<std::string> args;
|
std::list<std::string> args;
|
||||||
|
|
||||||
for (std::vector<UpdateScriptFile>::const_iterator iter = m_script->filesToInstall().begin();
|
|
||||||
iter != m_script->filesToInstall().end();
|
|
||||||
iter++)
|
|
||||||
{
|
|
||||||
if (iter->isMainBinary)
|
|
||||||
{
|
|
||||||
command = m_installDir + '/' + iter->path;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!command.empty())
|
if (!command.empty())
|
||||||
{
|
{
|
||||||
LOG(Info,"Starting main application " + command);
|
LOG(Info,"Starting main application " + command);
|
||||||
@ -411,7 +398,7 @@ void UpdateInstaller::restartMainApp()
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
LOG(Error,"No main binary specified in update script");
|
LOG(Error,"No main binary specified");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (const std::exception& ex)
|
catch (const std::exception& ex)
|
||||||
|
@ -33,6 +33,7 @@ class UpdateInstaller
|
|||||||
void setWaitPid(PLATFORM_PID pid);
|
void setWaitPid(PLATFORM_PID pid);
|
||||||
void setForceElevated(bool elevated);
|
void setForceElevated(bool elevated);
|
||||||
void setAutoClose(bool autoClose);
|
void setAutoClose(bool autoClose);
|
||||||
|
void setFinishCmd(const std::string& cmd);
|
||||||
|
|
||||||
void setObserver(UpdateObserver* observer);
|
void setObserver(UpdateObserver* observer);
|
||||||
|
|
||||||
@ -60,6 +61,7 @@ class UpdateInstaller
|
|||||||
std::string m_installDir;
|
std::string m_installDir;
|
||||||
std::string m_packageDir;
|
std::string m_packageDir;
|
||||||
std::string m_backupDir;
|
std::string m_backupDir;
|
||||||
|
std::string m_finishCmd;
|
||||||
PLATFORM_PID m_waitPid;
|
PLATFORM_PID m_waitPid;
|
||||||
UpdateScript* m_script;
|
UpdateScript* m_script;
|
||||||
UpdateObserver* m_observer;
|
UpdateObserver* m_observer;
|
||||||
|
@ -71,13 +71,14 @@ void UpdateScript::parseUpdate(const TiXmlElement* updateNode)
|
|||||||
UpdateScriptFile UpdateScript::parseFile(const TiXmlElement* element)
|
UpdateScriptFile UpdateScript::parseFile(const TiXmlElement* element)
|
||||||
{
|
{
|
||||||
UpdateScriptFile file;
|
UpdateScriptFile file;
|
||||||
file.path = elementText(element->FirstChildElement("name"));
|
// The name within the update files folder.
|
||||||
|
file.source = elementText(element->FirstChildElement("source"));
|
||||||
|
// The path to install to.
|
||||||
|
file.dest = elementText(element->FirstChildElement("dest"));
|
||||||
|
|
||||||
std::string modeString = elementText(element->FirstChildElement("permissions"));
|
std::string modeString = elementText(element->FirstChildElement("mode"));
|
||||||
sscanf(modeString.c_str(),"%i",&file.permissions);
|
sscanf(modeString.c_str(),"%i",&file.permissions);
|
||||||
|
|
||||||
file.linkTarget = elementText(element->FirstChildElement("target"));
|
|
||||||
file.isMainBinary = strToBool(elementText(element->FirstChildElement("is-main-binary")));
|
|
||||||
return file;
|
return file;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,10 +35,12 @@ class UpdateScriptFile
|
|||||||
public:
|
public:
|
||||||
UpdateScriptFile()
|
UpdateScriptFile()
|
||||||
: permissions(0)
|
: permissions(0)
|
||||||
, isMainBinary(false)
|
|
||||||
{}
|
{}
|
||||||
|
|
||||||
std::string path;
|
/// Path to copy from.
|
||||||
|
std::string source;
|
||||||
|
/// The path to copy to.
|
||||||
|
std::string dest;
|
||||||
std::string linkTarget;
|
std::string linkTarget;
|
||||||
|
|
||||||
/** The permissions for this file, specified
|
/** The permissions for this file, specified
|
||||||
@ -46,14 +48,11 @@ class UpdateScriptFile
|
|||||||
*/
|
*/
|
||||||
int permissions;
|
int permissions;
|
||||||
|
|
||||||
bool isMainBinary;
|
|
||||||
|
|
||||||
bool operator==(const UpdateScriptFile& other) const
|
bool operator==(const UpdateScriptFile& other) const
|
||||||
{
|
{
|
||||||
return path == other.path &&
|
return source == other.source &&
|
||||||
permissions == other.permissions &&
|
dest == other.dest &&
|
||||||
linkTarget == other.linkTarget &&
|
permissions == other.permissions;
|
||||||
isMainBinary == other.isMainBinary;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -104,6 +104,7 @@ void UpdaterOptions::parse(int argc, char** argv)
|
|||||||
AnyOption parser;
|
AnyOption parser;
|
||||||
parser.setOption("install-dir");
|
parser.setOption("install-dir");
|
||||||
parser.setOption("package-dir");
|
parser.setOption("package-dir");
|
||||||
|
parser.setOption("finish-cmd");
|
||||||
parser.setOption("script");
|
parser.setOption("script");
|
||||||
parser.setOption("wait");
|
parser.setOption("wait");
|
||||||
parser.setOption("mode");
|
parser.setOption("mode");
|
||||||
@ -133,6 +134,10 @@ void UpdaterOptions::parse(int argc, char** argv)
|
|||||||
{
|
{
|
||||||
waitPid = static_cast<PLATFORM_PID>(atoll(parser.getValue("wait")));
|
waitPid = static_cast<PLATFORM_PID>(atoll(parser.getValue("wait")));
|
||||||
}
|
}
|
||||||
|
if (parser.getValue("finish-cmd"))
|
||||||
|
{
|
||||||
|
finishCmd = parser.getValue("finish-cmd");
|
||||||
|
}
|
||||||
|
|
||||||
showVersion = parser.getFlag("version");
|
showVersion = parser.getFlag("version");
|
||||||
forceElevated = parser.getFlag("force-elevated");
|
forceElevated = parser.getFlag("force-elevated");
|
||||||
|
@ -14,6 +14,7 @@ class UpdaterOptions
|
|||||||
std::string installDir;
|
std::string installDir;
|
||||||
std::string packageDir;
|
std::string packageDir;
|
||||||
std::string scriptPath;
|
std::string scriptPath;
|
||||||
|
std::string finishCmd;
|
||||||
PLATFORM_PID waitPid;
|
PLATFORM_PID waitPid;
|
||||||
std::string logFile;
|
std::string logFile;
|
||||||
bool showVersion;
|
bool showVersion;
|
||||||
|
@ -137,7 +137,8 @@ int main(int argc, char** argv)
|
|||||||
+ ", package-dir: " + options.packageDir
|
+ ", package-dir: " + options.packageDir
|
||||||
+ ", wait-pid: " + intToStr(options.waitPid)
|
+ ", wait-pid: " + intToStr(options.waitPid)
|
||||||
+ ", script-path: " + options.scriptPath
|
+ ", script-path: " + options.scriptPath
|
||||||
+ ", mode: " + intToStr(options.mode));
|
+ ", mode: " + intToStr(options.mode)
|
||||||
|
+ ", finish-cmd: " + options.finishCmd);
|
||||||
|
|
||||||
installer.setMode(options.mode);
|
installer.setMode(options.mode);
|
||||||
installer.setInstallDir(options.installDir);
|
installer.setInstallDir(options.installDir);
|
||||||
@ -146,6 +147,7 @@ int main(int argc, char** argv)
|
|||||||
installer.setWaitPid(options.waitPid);
|
installer.setWaitPid(options.waitPid);
|
||||||
installer.setForceElevated(options.forceElevated);
|
installer.setForceElevated(options.forceElevated);
|
||||||
installer.setAutoClose(options.autoClose);
|
installer.setAutoClose(options.autoClose);
|
||||||
|
installer.setFinishCmd(options.finishCmd);
|
||||||
|
|
||||||
if (options.mode == UpdateInstaller::Main)
|
if (options.mode == UpdateInstaller::Main)
|
||||||
{
|
{
|
||||||
|
@ -18,31 +18,10 @@ void TestUpdateScript::testV2Script()
|
|||||||
TEST_COMPARE(newFormat.filesToUninstall(),oldFormat.filesToUninstall());
|
TEST_COMPARE(newFormat.filesToUninstall(),oldFormat.filesToUninstall());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestUpdateScript::testPermissions()
|
|
||||||
{
|
|
||||||
UpdateScript script;
|
|
||||||
script.parse("file_list.xml");
|
|
||||||
|
|
||||||
for (std::vector<UpdateScriptFile>::const_iterator iter = script.filesToInstall().begin();
|
|
||||||
iter != script.filesToInstall().end();
|
|
||||||
iter++)
|
|
||||||
{
|
|
||||||
if (iter->isMainBinary)
|
|
||||||
{
|
|
||||||
TEST_COMPARE(iter->permissions,0755);
|
|
||||||
}
|
|
||||||
if (!iter->linkTarget.empty())
|
|
||||||
{
|
|
||||||
TEST_COMPARE(iter->permissions,0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int,char**)
|
int main(int,char**)
|
||||||
{
|
{
|
||||||
TestList<TestUpdateScript> tests;
|
TestList<TestUpdateScript> tests;
|
||||||
tests.addTest(&TestUpdateScript::testV2Script);
|
tests.addTest(&TestUpdateScript::testV2Script);
|
||||||
tests.addTest(&TestUpdateScript::testPermissions);
|
|
||||||
return TestUtils::runTest(tests);
|
return TestUtils::runTest(tests);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4,6 +4,5 @@ class TestUpdateScript
|
|||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
void testV2Script();
|
void testV2Script();
|
||||||
void testPermissions();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user