2015-02-08 17:56:14 +01:00
|
|
|
#include "GoUpdate.h"
|
|
|
|
#include <pathutils.h>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QDomDocument>
|
|
|
|
#include <QFile>
|
2015-02-09 01:51:14 +01:00
|
|
|
#include <Env.h>
|
2015-02-08 17:56:14 +01:00
|
|
|
|
|
|
|
namespace GoUpdate
|
|
|
|
{
|
|
|
|
|
|
|
|
bool parseVersionInfo(const QByteArray &data, VersionFileList &list, QString &error)
|
|
|
|
{
|
|
|
|
QJsonParseError jsonError;
|
|
|
|
QJsonDocument jsonDoc = QJsonDocument::fromJson(data, &jsonError);
|
|
|
|
if (jsonError.error != QJsonParseError::NoError)
|
|
|
|
{
|
|
|
|
error = QString("Failed to parse version info JSON: %1 at %2")
|
|
|
|
.arg(jsonError.errorString())
|
|
|
|
.arg(jsonError.offset);
|
|
|
|
qCritical() << error;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject json = jsonDoc.object();
|
|
|
|
|
|
|
|
qDebug() << data;
|
|
|
|
qDebug() << "Loading version info from JSON.";
|
|
|
|
QJsonArray filesArray = json.value("Files").toArray();
|
|
|
|
for (QJsonValue fileValue : filesArray)
|
|
|
|
{
|
|
|
|
QJsonObject fileObj = fileValue.toObject();
|
|
|
|
|
|
|
|
QString file_path = fileObj.value("Path").toString();
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
// On OSX, the paths for the updater need to be fixed.
|
|
|
|
// basically, anything that isn't in the .app folder is ignored.
|
|
|
|
// everything else is changed so the code that processes the files actually finds
|
|
|
|
// them and puts the replacements in the right spots.
|
2015-03-21 18:35:42 +01:00
|
|
|
fixPathForOSX(file_path);
|
2015-02-08 17:56:14 +01:00
|
|
|
#endif
|
|
|
|
VersionFileEntry file{file_path, fileObj.value("Perms").toVariant().toInt(),
|
|
|
|
FileSourceList(), fileObj.value("MD5").toString(), };
|
|
|
|
qDebug() << "File" << file.path << "with perms" << file.mode;
|
|
|
|
|
|
|
|
QJsonArray sourceArray = fileObj.value("Sources").toArray();
|
|
|
|
for (QJsonValue val : sourceArray)
|
|
|
|
{
|
|
|
|
QJsonObject sourceObj = val.toObject();
|
|
|
|
|
|
|
|
QString type = sourceObj.value("SourceType").toString();
|
|
|
|
if (type == "http")
|
|
|
|
{
|
|
|
|
file.sources.append(FileSource("http", sourceObj.value("Url").toString()));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qWarning() << "Unknown source type" << type << "ignored.";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qDebug() << "Loaded info for" << file.path;
|
|
|
|
|
|
|
|
list.append(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool processFileLists
|
|
|
|
(
|
|
|
|
const VersionFileList ¤tVersion,
|
|
|
|
const VersionFileList &newVersion,
|
|
|
|
const QString &rootPath,
|
|
|
|
const QString &tempPath,
|
|
|
|
NetJobPtr job,
|
2015-06-07 23:42:22 +02:00
|
|
|
OperationList &ops
|
2015-02-08 17:56:14 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// First, if we've loaded the current version's file list, we need to iterate through it and
|
|
|
|
// delete anything in the current one version's list that isn't in the new version's list.
|
|
|
|
for (VersionFileEntry entry : currentVersion)
|
|
|
|
{
|
|
|
|
QFileInfo toDelete(PathCombine(rootPath, entry.path));
|
|
|
|
if (!toDelete.exists())
|
|
|
|
{
|
|
|
|
qCritical() << "Expected file " << toDelete.absoluteFilePath()
|
|
|
|
<< " doesn't exist!";
|
|
|
|
}
|
|
|
|
bool keep = false;
|
|
|
|
|
|
|
|
//
|
|
|
|
for (VersionFileEntry newEntry : newVersion)
|
|
|
|
{
|
|
|
|
if (newEntry.path == entry.path)
|
|
|
|
{
|
|
|
|
qDebug() << "Not deleting" << entry.path
|
|
|
|
<< "because it is still present in the new version.";
|
|
|
|
keep = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the loop reaches the end and we didn't find a match, delete the file.
|
|
|
|
if (!keep)
|
|
|
|
{
|
|
|
|
if (toDelete.exists())
|
|
|
|
ops.append(Operation::DeleteOp(entry.path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, check each file in MultiMC's folder and see if we need to update them.
|
|
|
|
for (VersionFileEntry entry : newVersion)
|
|
|
|
{
|
|
|
|
// TODO: Let's not MD5sum a ton of files on the GUI thread. We should probably find a
|
|
|
|
// way to do this in the background.
|
|
|
|
QString fileMD5;
|
|
|
|
QString realEntryPath = PathCombine(rootPath, entry.path);
|
|
|
|
QFile entryFile(realEntryPath);
|
|
|
|
QFileInfo entryInfo(realEntryPath);
|
|
|
|
|
|
|
|
bool needs_upgrade = false;
|
|
|
|
if (!entryFile.exists())
|
|
|
|
{
|
|
|
|
needs_upgrade = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bool pass = true;
|
|
|
|
if (!entryInfo.isReadable())
|
|
|
|
{
|
|
|
|
qCritical() << "File " << realEntryPath << " is not readable.";
|
|
|
|
pass = false;
|
|
|
|
}
|
|
|
|
if (!entryInfo.isWritable())
|
|
|
|
{
|
|
|
|
qCritical() << "File " << realEntryPath << " is not writable.";
|
|
|
|
pass = false;
|
|
|
|
}
|
|
|
|
if (!entryFile.open(QFile::ReadOnly))
|
|
|
|
{
|
|
|
|
qCritical() << "File " << realEntryPath << " cannot be opened for reading.";
|
|
|
|
pass = false;
|
|
|
|
}
|
|
|
|
if (!pass)
|
|
|
|
{
|
|
|
|
ops.clear();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!needs_upgrade)
|
|
|
|
{
|
|
|
|
QCryptographicHash hash(QCryptographicHash::Md5);
|
|
|
|
auto foo = entryFile.readAll();
|
|
|
|
|
|
|
|
hash.addData(foo);
|
|
|
|
fileMD5 = hash.result().toHex();
|
|
|
|
if ((fileMD5 != entry.md5))
|
|
|
|
{
|
|
|
|
qDebug() << "MD5Sum does not match!";
|
|
|
|
qDebug() << "Expected:'" << entry.md5 << "'";
|
|
|
|
qDebug() << "Got: '" << fileMD5 << "'";
|
|
|
|
needs_upgrade = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip file. it doesn't need an upgrade.
|
|
|
|
if (!needs_upgrade)
|
|
|
|
{
|
|
|
|
qDebug() << "File" << realEntryPath << " does not need updating.";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// yep. this file actually needs an upgrade. PROCEED.
|
|
|
|
qDebug() << "Found file" << realEntryPath << " that needs updating.";
|
|
|
|
|
|
|
|
// Go through the sources list and find one to use.
|
|
|
|
// TODO: Make a NetAction that takes a source list and tries each of them until one
|
|
|
|
// works. For now, we'll just use the first http one.
|
|
|
|
for (FileSource source : entry.sources)
|
|
|
|
{
|
|
|
|
if (source.type != "http")
|
|
|
|
continue;
|
|
|
|
|
|
|
|
qDebug() << "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.
|
|
|
|
QString dlPath = PathCombine(tempPath, QString(entry.path).replace("/", "_"));
|
|
|
|
|
2015-06-07 23:42:22 +02:00
|
|
|
// We need to download the file to the updatefiles folder and add a task
|
|
|
|
// to copy it to its install path.
|
|
|
|
auto download = MD5EtagDownload::make(source.url, dlPath);
|
|
|
|
download->m_expected_md5 = entry.md5;
|
|
|
|
job->addNetAction(download);
|
|
|
|
ops.append(Operation::CopyOp(dlPath, entry.path, entry.mode));
|
2015-02-08 17:56:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fixPathForOSX(QString &path)
|
|
|
|
{
|
|
|
|
if (path.startsWith("MultiMC.app/"))
|
|
|
|
{
|
|
|
|
// remove the prefix and add a new, more appropriate one.
|
|
|
|
path.remove(0, 12);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qCritical() << "Update path not within .app: " << path;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|