pollymc/api/logic/minecraft/onesix/OneSixUpdate.cpp

350 lines
11 KiB
C++
Raw Normal View History

2015-02-02 14:25:30 -08:00
/* Copyright 2013-2015 MultiMC Contributors
2013-02-21 18:10:17 -06:00
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
2013-02-21 18:10:17 -06:00
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2015-02-09 01:51:14 +01:00
#include "Env.h"
#include <minecraft/forge/ForgeXzDownload.h>
2013-08-04 14:46:33 +02:00
#include "OneSixUpdate.h"
#include "OneSixInstance.h"
2013-02-21 18:10:17 -06:00
#include <QtNetwork>
#include <QFile>
#include <QFileInfo>
#include <QTextStream>
#include <QDataStream>
2014-05-08 21:20:10 +02:00
#include <JlCompress.h>
2015-02-09 01:51:14 +01:00
#include "BaseInstance.h"
#include "minecraft/MinecraftVersionList.h"
#include "minecraft/MinecraftProfile.h"
#include "minecraft/Library.h"
2015-02-09 01:51:14 +01:00
#include "net/URLConstants.h"
#include "net/ChecksumValidator.h"
2015-02-09 01:51:14 +01:00
#include "minecraft/AssetsUtils.h"
#include "Exception.h"
#include "MMCZip.h"
2015-10-05 01:47:27 +02:00
#include <FileSystem.h>
OneSixUpdate::OneSixUpdate(OneSixInstance *inst, QObject *parent) : Task(parent), m_inst(inst)
{
}
2013-08-04 14:46:33 +02:00
void OneSixUpdate::executeTask()
{
// Make directories
QDir mcDir(m_inst->minecraftRoot());
if (!mcDir.exists() && !mcDir.mkpath("."))
{
emitFailed(tr("Failed to create folder for minecraft binaries."));
return;
}
// Get a pointer to the version object that corresponds to the instance's version.
targetVersion = std::dynamic_pointer_cast<MinecraftVersion>(ENV.getVersion("net.minecraft", m_inst->intendedVersionId()));
if (targetVersion == nullptr)
2013-08-05 03:29:50 +02:00
{
// don't do anything if it was invalid
emitFailed(tr("The specified Minecraft version is invalid. Choose a different one."));
return;
2013-08-05 03:29:50 +02:00
}
if (m_inst->providesVersionFile() || !targetVersion->needsUpdate())
2014-02-21 19:15:59 +01:00
{
qDebug() << "Instance either provides a version file or doesn't need an update.";
2014-02-21 19:15:59 +01:00
jarlibStart();
return;
}
versionUpdateTask = std::dynamic_pointer_cast<MinecraftVersionList>(ENV.getVersionList("net.minecraft"))->createUpdateTask(m_inst->intendedVersionId());
if (!versionUpdateTask)
{
qDebug() << "Didn't spawn an update task.";
jarlibStart();
return;
}
connect(versionUpdateTask.get(), SIGNAL(succeeded()), SLOT(jarlibStart()));
2015-04-26 23:04:50 +02:00
connect(versionUpdateTask.get(), &NetJob::failed, this, &OneSixUpdate::versionUpdateFailed);
connect(versionUpdateTask.get(), SIGNAL(progress(qint64, qint64)), SIGNAL(progress(qint64, qint64)));
setStatus(tr("Getting the version files from Mojang..."));
versionUpdateTask->start();
2013-08-05 03:29:50 +02:00
}
void OneSixUpdate::versionUpdateFailed(QString reason)
2013-08-05 03:29:50 +02:00
{
emitFailed(reason);
2013-08-05 03:29:50 +02:00
}
2013-12-10 07:12:52 +01:00
void OneSixUpdate::assetIndexStart()
{
2013-12-23 15:46:01 +00:00
setStatus(tr("Updating assets index..."));
2013-12-10 07:12:52 +01:00
OneSixInstance *inst = (OneSixInstance *)m_inst;
auto profile = inst->getMinecraftProfile();
auto assets = profile->getMinecraftAssets();
QUrl indexUrl = assets->url;
QString localPath = assets->id + ".json";
auto job = new NetJob(tr("Asset index for %1").arg(inst->name()));
2013-12-10 07:12:52 +01:00
auto metacache = ENV.metacache();
2013-12-10 07:12:52 +01:00
auto entry = metacache->resolveEntry("asset_indexes", localPath);
entry->setStale(true);
auto hexSha1 = assets->sha1.toLatin1();
qDebug() << "Asset index SHA1:" << hexSha1;
auto dl = Net::Download::makeCached(indexUrl, entry);
auto rawSha1 = QByteArray::fromHex(assets->sha1.toLatin1());
dl->addValidator(new Net::ChecksumValidator(QCryptographicHash::Sha1, rawSha1));
job->addNetAction(dl);
2013-12-10 07:12:52 +01:00
jarlibDownloadJob.reset(job);
connect(jarlibDownloadJob.get(), SIGNAL(succeeded()), SLOT(assetIndexFinished()));
2015-04-26 23:04:50 +02:00
connect(jarlibDownloadJob.get(), &NetJob::failed, this, &OneSixUpdate::assetIndexFailed);
connect(jarlibDownloadJob.get(), SIGNAL(progress(qint64, qint64)), SIGNAL(progress(qint64, qint64)));
2013-12-10 07:12:52 +01:00
2015-05-06 22:16:52 +02:00
qDebug() << m_inst->name() << ": Starting asset index download";
2013-12-10 07:12:52 +01:00
jarlibDownloadJob->start();
}
void OneSixUpdate::assetIndexFinished()
{
AssetsIndex index;
2015-05-06 22:16:52 +02:00
qDebug() << m_inst->name() << ": Finished asset index download";
2013-12-10 07:12:52 +01:00
OneSixInstance *inst = (OneSixInstance *)m_inst;
auto profile = inst->getMinecraftProfile();
auto assets = profile->getMinecraftAssets();
2013-12-10 07:12:52 +01:00
QString asset_fname = "assets/indexes/" + assets->id + ".json";
// FIXME: this looks like a job for a generic validator based on json schema?
if (!AssetsUtils::loadAssetsIndexJson(assets->id, asset_fname, &index))
2013-12-10 07:12:52 +01:00
{
auto metacache = ENV.metacache();
auto entry = metacache->resolveEntry("asset_indexes", assets->id + ".json");
metacache->evictEntry(entry);
emitFailed(tr("Failed to read the assets index!"));
2013-12-10 07:12:52 +01:00
}
auto job = index.getDownloadJob();
if(job)
2013-12-10 07:12:52 +01:00
{
2013-12-23 15:46:01 +00:00
setStatus(tr("Getting the assets files from Mojang..."));
jarlibDownloadJob = job;
2013-12-10 07:12:52 +01:00
connect(jarlibDownloadJob.get(), SIGNAL(succeeded()), SLOT(assetsFinished()));
2015-04-26 23:04:50 +02:00
connect(jarlibDownloadJob.get(), &NetJob::failed, this, &OneSixUpdate::assetsFailed);
connect(jarlibDownloadJob.get(), SIGNAL(progress(qint64, qint64)), SIGNAL(progress(qint64, qint64)));
2013-12-10 07:12:52 +01:00
jarlibDownloadJob->start();
return;
}
assetsFinished();
}
2015-04-26 23:04:50 +02:00
void OneSixUpdate::assetIndexFailed(QString reason)
2013-12-10 07:12:52 +01:00
{
2015-05-06 22:16:52 +02:00
qDebug() << m_inst->name() << ": Failed asset index download";
2015-04-26 23:04:50 +02:00
emitFailed(tr("Failed to download the assets index:\n%1").arg(reason));
2013-12-10 07:12:52 +01:00
}
void OneSixUpdate::assetsFinished()
{
emitSucceeded();
2013-12-10 07:12:52 +01:00
}
2015-04-26 23:04:50 +02:00
void OneSixUpdate::assetsFailed(QString reason)
2013-12-10 07:12:52 +01:00
{
2015-04-26 23:04:50 +02:00
emitFailed(tr("Failed to download assets:\n%1").arg(reason));
2013-12-10 07:12:52 +01:00
}
2013-08-05 03:29:50 +02:00
void OneSixUpdate::jarlibStart()
{
2013-12-23 15:46:01 +00:00
setStatus(tr("Getting the library files from Mojang..."));
qDebug() << m_inst->name() << ": downloading libraries";
OneSixInstance *inst = (OneSixInstance *)m_inst;
inst->reloadProfile();
if(inst->flags() & BaseInstance::VersionBrokenFlag)
{
emitFailed(tr("Failed to load the version description files - check the instance for errors."));
return;
2013-08-05 03:29:50 +02:00
}
// Build a list of URLs that will need to be downloaded.
std::shared_ptr<MinecraftProfile> profile = inst->getMinecraftProfile();
// minecraft.jar for this version
{
QString version_id = profile->getMinecraftVersion();
QString localPath = version_id + "/" + version_id + ".jar";
QString urlstr = profile->getMainJarUrl();
auto job = new NetJob(tr("Libraries for instance %1").arg(inst->name()));
auto metacache = ENV.metacache();
auto entry = metacache->resolveEntry("versions", localPath);
job->addNetAction(Net::Download::makeCached(QUrl(urlstr), entry));
jarlibDownloadJob.reset(job);
}
auto libs = profile->getLibraries();
auto metacache = ENV.metacache();
QList<LibraryPtr> brokenLocalLibs;
QStringList failedFiles;
for (auto lib : libs)
2013-08-05 03:29:50 +02:00
{
auto dls = lib->getDownloads(currentSystem, metacache.get(), failedFiles);
for(auto dl : dls)
{
jarlibDownloadJob->addNetAction(dl);
}
2013-08-05 03:29:50 +02:00
}
if (!brokenLocalLibs.empty())
{
jarlibDownloadJob.reset();
QString failed_all = failedFiles.join("\n");
emitFailed(tr("Some libraries marked as 'local' are missing their jar "
"files:\n%1\n\nYou'll have to correct this problem manually. If this is "
"an externally tracked instance, make sure to run it at least once "
"outside of MultiMC.").arg(failed_all));
return;
}
2013-10-06 01:13:40 +02:00
connect(jarlibDownloadJob.get(), SIGNAL(succeeded()), SLOT(jarlibFinished()));
2015-04-26 23:04:50 +02:00
connect(jarlibDownloadJob.get(), &NetJob::failed, this, &OneSixUpdate::jarlibFailed);
2013-10-06 01:13:40 +02:00
connect(jarlibDownloadJob.get(), SIGNAL(progress(qint64, qint64)),
SIGNAL(progress(qint64, qint64)));
2013-08-05 03:29:50 +02:00
2013-09-02 00:25:40 +02:00
jarlibDownloadJob->start();
}
2013-08-04 14:46:33 +02:00
void OneSixUpdate::jarlibFinished()
{
2014-05-05 00:10:59 +02:00
OneSixInstance *inst = (OneSixInstance *)m_inst;
std::shared_ptr<MinecraftProfile> profile = inst->getMinecraftProfile();
if (profile->hasTrait("legacyFML"))
2014-05-05 00:10:59 +02:00
{
fmllibsStart();
}
else
{
assetIndexStart();
}
}
2015-04-26 23:04:50 +02:00
void OneSixUpdate::jarlibFailed(QString reason)
{
QStringList failed = jarlibDownloadJob->getFailedFiles();
QString failed_all = failed.join("\n");
emitFailed(
2015-04-26 23:04:50 +02:00
tr("Failed to download the following files:\n%1\n\nReason:%2\nPlease try again.").arg(failed_all, reason));
}
2014-05-05 00:10:59 +02:00
void OneSixUpdate::fmllibsStart()
{
// Get the mod list
OneSixInstance *inst = (OneSixInstance *)m_inst;
std::shared_ptr<MinecraftProfile> profile = inst->getMinecraftProfile();
2014-05-05 00:10:59 +02:00
bool forge_present = false;
QString version = inst->intendedVersionId();
auto &fmlLibsMapping = g_VersionFilterData.fmlLibsMapping;
2014-05-05 00:10:59 +02:00
if (!fmlLibsMapping.contains(version))
{
assetIndexStart();
return;
}
auto &libList = fmlLibsMapping[version];
// determine if we need some libs for FML or forge
setStatus(tr("Checking for FML libraries..."));
forge_present = (profile->versionPatch("net.minecraftforge") != nullptr);
2014-05-05 00:10:59 +02:00
// we don't...
if (!forge_present)
{
assetIndexStart();
return;
}
// now check the lib folder inside the instance for files.
for (auto &lib : libList)
{
2015-10-05 01:47:27 +02:00
QFileInfo libInfo(FS::PathCombine(inst->libDir(), lib.filename));
2014-05-05 00:10:59 +02:00
if (libInfo.exists())
continue;
fmlLibsToProcess.append(lib);
}
// if everything is in place, there's nothing to do here...
if (fmlLibsToProcess.isEmpty())
{
assetIndexStart();
return;
}
// download missing libs to our place
setStatus(tr("Dowloading FML libraries..."));
auto dljob = new NetJob("FML libraries");
auto metacache = ENV.metacache();
2014-05-05 00:10:59 +02:00
for (auto &lib : fmlLibsToProcess)
{
auto entry = metacache->resolveEntry("fmllibs", lib.filename);
QString urlString = lib.ours ? URLConstants::FMLLIBS_OUR_BASE_URL + lib.filename
: URLConstants::FMLLIBS_FORGE_BASE_URL + lib.filename;
dljob->addNetAction(Net::Download::makeCached(QUrl(urlString), entry));
2014-05-05 00:10:59 +02:00
}
connect(dljob, SIGNAL(succeeded()), SLOT(fmllibsFinished()));
2015-04-26 23:04:50 +02:00
connect(dljob, &NetJob::failed, this, &OneSixUpdate::fmllibsFailed);
2014-05-05 00:10:59 +02:00
connect(dljob, SIGNAL(progress(qint64, qint64)), SIGNAL(progress(qint64, qint64)));
legacyDownloadJob.reset(dljob);
legacyDownloadJob->start();
}
void OneSixUpdate::fmllibsFinished()
{
legacyDownloadJob.reset();
if (!fmlLibsToProcess.isEmpty())
2014-05-05 00:10:59 +02:00
{
setStatus(tr("Copying FML libraries into the instance..."));
OneSixInstance *inst = (OneSixInstance *)m_inst;
auto metacache = ENV.metacache();
2014-05-05 00:10:59 +02:00
int index = 0;
for (auto &lib : fmlLibsToProcess)
{
progress(index, fmlLibsToProcess.size());
auto entry = metacache->resolveEntry("fmllibs", lib.filename);
2015-10-05 01:47:27 +02:00
auto path = FS::PathCombine(inst->libDir(), lib.filename);
if (!FS::ensureFilePathExists(path))
2014-05-05 00:10:59 +02:00
{
emitFailed(tr("Failed creating FML library folder inside the instance."));
return;
}
2015-10-05 01:47:27 +02:00
if (!QFile::copy(entry->getFullPath(), FS::PathCombine(inst->libDir(), lib.filename)))
2014-05-05 00:10:59 +02:00
{
emitFailed(tr("Failed copying Forge/FML library: %1.").arg(lib.filename));
return;
}
index++;
}
progress(index, fmlLibsToProcess.size());
}
assetIndexStart();
}
2015-04-26 23:04:50 +02:00
void OneSixUpdate::fmllibsFailed(QString reason)
2014-05-05 00:10:59 +02:00
{
2015-04-26 23:04:50 +02:00
emitFailed(tr("Game update failed: it was impossible to fetch the required FML libraries.\nReason:\n%1").arg(reason));
2014-05-05 00:10:59 +02:00
return;
}