pollymc/logic/OneSixUpdate.cpp

200 lines
5.7 KiB
C++
Raw Normal View History

2013-02-22 05:40:17 +05:30
/* Copyright 2013 MultiMC Contributors
*
* 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-22 05:40:17 +05:30
* 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.
*/
#include "MultiMC.h"
2013-08-04 18:16:33 +05:30
#include "OneSixUpdate.h"
2013-02-22 05:40:17 +05:30
#include <QtNetwork>
#include <QFile>
#include <QFileInfo>
#include <QTextStream>
#include <QDataStream>
2013-08-04 03:28:39 +05:30
#include "BaseInstance.h"
#include "lists/MinecraftVersionList.h"
#include "OneSixVersion.h"
2013-09-12 03:13:17 +05:30
#include "OneSixLibrary.h"
2013-08-05 06:59:50 +05:30
#include "OneSixInstance.h"
#include "pathutils.h"
OneSixUpdate::OneSixUpdate(BaseInstance *inst, QObject *parent) : BaseUpdate(inst, parent)
{
}
2013-08-04 18:16:33 +05:30
void OneSixUpdate::executeTask()
{
2013-08-05 06:59:50 +05:30
QString intendedVersion = m_inst->intendedVersionId();
// Make directories
QDir mcDir(m_inst->minecraftRoot());
if (!mcDir.exists() && !mcDir.mkpath("."))
{
emitFailed("Failed to create bin folder.");
return;
}
// Get a pointer to the version object that corresponds to the instance's version.
2013-10-06 04:43:40 +05:30
targetVersion = std::dynamic_pointer_cast<MinecraftVersion>(
MMC->minecraftlist()->findVersion(intendedVersion));
if (targetVersion == nullptr)
{
2013-08-05 06:59:50 +05:30
// don't do anything if it was invalid
emitSucceeded();
return;
}
if (m_inst->shouldUpdate())
2013-08-05 06:59:50 +05:30
{
versionFileStart();
}
else
{
jarlibStart();
}
}
void OneSixUpdate::versionFileStart()
{
2013-08-04 03:28:39 +05:30
setStatus("Getting the version files from Mojang.");
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
urlstr += targetVersion->descriptor() + "/" + targetVersion->descriptor() + ".json";
auto job = new NetJob("Version index");
job->addNetAction(ByteArrayDownload::make(QUrl(urlstr)));
specificVersionDownloadJob.reset(job);
2013-10-06 04:43:40 +05:30
connect(specificVersionDownloadJob.get(), SIGNAL(succeeded()), SLOT(versionFileFinished()));
connect(specificVersionDownloadJob.get(), SIGNAL(failed()), SLOT(versionFileFailed()));
connect(specificVersionDownloadJob.get(), SIGNAL(progress(qint64, qint64)),
SIGNAL(progress(qint64, qint64)));
2013-09-02 03:55:40 +05:30
specificVersionDownloadJob->start();
}
2013-08-04 18:16:33 +05:30
void OneSixUpdate::versionFileFinished()
{
NetActionPtr DlJob = specificVersionDownloadJob->first();
OneSixInstance *inst = (OneSixInstance *)m_inst;
QString version_id = targetVersion->descriptor();
QString inst_dir = m_inst->instanceRoot();
2013-08-05 06:59:50 +05:30
// save the version file in $instanceId/version.json
{
QString version1 = PathCombine(inst_dir, "/version.json");
ensureFilePathExists(version1);
2013-08-14 11:43:41 +05:30
// FIXME: detect errors here, download to a temp file, swap
QSaveFile vfile1(version1);
if (!vfile1.open(QIODevice::Truncate | QIODevice::WriteOnly))
{
emitFailed("Can't open " + version1 + " for writing.");
return;
}
2013-10-06 04:43:40 +05:30
auto data = std::dynamic_pointer_cast<ByteArrayDownload>(DlJob)->m_data;
qint64 actual = 0;
if ((actual = vfile1.write(data)) != data.size())
{
emitFailed("Failed to write into " + version1 + ". Written " + actual + " out of " +
data.size() + '.');
return;
}
if (!vfile1.commit())
{
emitFailed("Can't commit changes to " + version1);
return;
}
}
2013-08-14 11:43:41 +05:30
// the version is downloaded safely. update is 'done' at this point
m_inst->setShouldUpdate(false);
// delete any custom version inside the instance (it's no longer relevant, we did an update)
QString custom = PathCombine(inst_dir, "/custom.json");
QFile finfo(custom);
if (finfo.exists())
{
finfo.remove();
}
inst->reloadFullVersion();
2013-08-05 06:59:50 +05:30
jarlibStart();
}
void OneSixUpdate::versionFileFailed()
{
emitFailed("Failed to download the version description. Try again.");
2013-08-05 06:59:50 +05:30
}
void OneSixUpdate::jarlibStart()
{
OneSixInstance *inst = (OneSixInstance *)m_inst;
2013-08-05 06:59:50 +05:30
bool successful = inst->reloadFullVersion();
if (!successful)
2013-08-05 06:59:50 +05:30
{
emitFailed("Failed to load the version description file (version.json). It might be "
"corrupted, missing or simply too new.");
2013-08-05 06:59:50 +05:30
return;
}
2013-10-06 04:43:40 +05:30
std::shared_ptr<OneSixVersion> version = inst->getFullVersion();
// download the right jar, save it in versions/$version/$version.jar
QString urlstr("http://s3.amazonaws.com/Minecraft.Download/versions/");
2013-08-05 06:59:50 +05:30
urlstr += version->id + "/" + version->id + ".jar";
QString targetstr("versions/");
2013-08-05 06:59:50 +05:30
targetstr += version->id + "/" + version->id + ".jar";
auto job = new NetJob("Libraries for instance " + inst->name());
job->addNetAction(FileDownload::make(QUrl(urlstr), targetstr));
jarlibDownloadJob.reset(job);
2013-08-05 06:59:50 +05:30
auto libs = version->getActiveNativeLibs();
libs.append(version->getActiveNormalLibs());
auto metacache = MMC->metacache();
for (auto lib : libs)
2013-08-05 06:59:50 +05:30
{
if (lib->hint() == "local")
continue;
QString download_path = lib->downloadUrl();
auto entry = metacache->resolveEntry("libraries", lib->storagePath());
if (entry->stale)
{
2013-10-06 04:43:40 +05:30
if (lib->hint() == "forge-pack-xz")
jarlibDownloadJob->addNetAction(ForgeXzDownload::make(download_path, entry));
else
jarlibDownloadJob->addNetAction(CacheDownload::make(download_path, entry));
}
2013-08-05 06:59:50 +05:30
}
2013-10-06 04:43:40 +05:30
connect(jarlibDownloadJob.get(), SIGNAL(succeeded()), SLOT(jarlibFinished()));
connect(jarlibDownloadJob.get(), SIGNAL(failed()), SLOT(jarlibFailed()));
connect(jarlibDownloadJob.get(), SIGNAL(progress(qint64, qint64)),
SIGNAL(progress(qint64, qint64)));
2013-08-05 06:59:50 +05:30
2013-09-02 03:55:40 +05:30
jarlibDownloadJob->start();
}
2013-08-04 18:16:33 +05:30
void OneSixUpdate::jarlibFinished()
{
emitSucceeded();
}
2013-08-04 18:16:33 +05:30
void OneSixUpdate::jarlibFailed()
{
QStringList failed = jarlibDownloadJob->getFailedFiles();
QString failed_all = failed.join("\n");
emitFailed("Failed to download the following files:\n" + failed_all +
"\n\nPlease try again.");
}