2013-11-04 02:53:05 +01:00
|
|
|
/* 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
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2013-09-22 04:21:36 +02:00
|
|
|
#include "ForgeInstaller.h"
|
2014-01-24 18:12:02 +01:00
|
|
|
#include "OneSixVersion.h"
|
|
|
|
#include "OneSixLibrary.h"
|
2013-09-22 14:00:37 +02:00
|
|
|
#include "net/HttpMetaCache.h"
|
2013-09-22 04:21:36 +02:00
|
|
|
#include <quazip.h>
|
|
|
|
#include <quazipfile.h>
|
|
|
|
#include <pathutils.h>
|
|
|
|
#include <QStringList>
|
2014-01-27 19:20:07 +01:00
|
|
|
#include <QRegularExpression>
|
|
|
|
#include <QRegularExpressionMatch>
|
2013-09-22 14:00:37 +02:00
|
|
|
#include "MultiMC.h"
|
2014-01-24 18:12:02 +01:00
|
|
|
#include "OneSixInstance.h"
|
2013-09-22 04:21:36 +02:00
|
|
|
|
2014-01-22 07:33:32 +01:00
|
|
|
#include <QJsonDocument>
|
2014-01-23 21:31:41 +01:00
|
|
|
#include <QJsonArray>
|
2014-01-22 07:33:32 +01:00
|
|
|
#include <QSaveFile>
|
|
|
|
#include <QCryptographicHash>
|
|
|
|
|
2013-09-22 04:21:36 +02:00
|
|
|
ForgeInstaller::ForgeInstaller(QString filename, QString universal_url)
|
|
|
|
{
|
2014-01-24 18:12:02 +01:00
|
|
|
std::shared_ptr<OneSixVersion> newVersion;
|
2013-09-22 04:21:36 +02:00
|
|
|
m_universal_url = universal_url;
|
|
|
|
|
|
|
|
QuaZip zip(filename);
|
|
|
|
if (!zip.open(QuaZip::mdUnzip))
|
|
|
|
return;
|
|
|
|
|
|
|
|
QuaZipFile file(&zip);
|
|
|
|
|
|
|
|
// read the install profile
|
|
|
|
if (!zip.setCurrentFile("install_profile.json"))
|
|
|
|
return;
|
|
|
|
|
|
|
|
QJsonParseError jsonError;
|
|
|
|
if (!file.open(QIODevice::ReadOnly))
|
|
|
|
return;
|
|
|
|
QJsonDocument jsonDoc = QJsonDocument::fromJson(file.readAll(), &jsonError);
|
|
|
|
file.close();
|
|
|
|
if (jsonError.error != QJsonParseError::NoError)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!jsonDoc.isObject())
|
|
|
|
return;
|
|
|
|
|
|
|
|
QJsonObject root = jsonDoc.object();
|
|
|
|
|
|
|
|
auto installVal = root.value("install");
|
|
|
|
auto versionInfoVal = root.value("versionInfo");
|
|
|
|
if (!installVal.isObject() || !versionInfoVal.isObject())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// read the forge version info
|
|
|
|
{
|
2014-01-24 18:12:02 +01:00
|
|
|
newVersion = OneSixVersion::fromJson(versionInfoVal.toObject());
|
2013-09-22 04:21:36 +02:00
|
|
|
if (!newVersion)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject installObj = installVal.toObject();
|
|
|
|
QString libraryName = installObj.value("path").toString();
|
|
|
|
internalPath = installObj.value("filePath").toString();
|
2014-02-01 14:52:21 +01:00
|
|
|
m_forgeVersionString = installObj.value("version").toString().remove("Forge").trimmed();
|
2013-09-22 04:21:36 +02:00
|
|
|
|
|
|
|
// where do we put the library? decode the mojang path
|
2014-01-24 18:12:02 +01:00
|
|
|
OneSixLibrary lib(libraryName);
|
2013-09-22 04:21:36 +02:00
|
|
|
lib.finalize();
|
2013-09-22 14:00:37 +02:00
|
|
|
|
|
|
|
auto cacheentry = MMC->metacache()->resolveEntry("libraries", lib.storagePath());
|
2013-09-22 04:21:36 +02:00
|
|
|
finalPath = "libraries/" + lib.storagePath();
|
|
|
|
if (!ensureFilePathExists(finalPath))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!zip.setCurrentFile(internalPath))
|
|
|
|
return;
|
|
|
|
if (!file.open(QIODevice::ReadOnly))
|
|
|
|
return;
|
|
|
|
{
|
|
|
|
QByteArray data = file.readAll();
|
|
|
|
// extract file
|
|
|
|
QSaveFile extraction(finalPath);
|
|
|
|
if (!extraction.open(QIODevice::WriteOnly))
|
|
|
|
return;
|
|
|
|
if (extraction.write(data) != data.size())
|
|
|
|
return;
|
|
|
|
if (!extraction.commit())
|
|
|
|
return;
|
2013-09-22 14:00:37 +02:00
|
|
|
QCryptographicHash md5sum(QCryptographicHash::Md5);
|
|
|
|
md5sum.addData(data);
|
|
|
|
|
|
|
|
cacheentry->stale = false;
|
|
|
|
cacheentry->md5sum = md5sum.result().toHex().constData();
|
|
|
|
MMC->metacache()->updateEntry(cacheentry);
|
2013-09-22 04:21:36 +02:00
|
|
|
}
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
m_forge_version = newVersion;
|
|
|
|
realVersionId = m_forge_version->id = installObj.value("minecraft").toString();
|
|
|
|
}
|
|
|
|
|
2014-01-24 18:12:02 +01:00
|
|
|
bool ForgeInstaller::add(OneSixInstance *to)
|
2013-09-22 04:21:36 +02:00
|
|
|
{
|
2014-01-23 21:31:41 +01:00
|
|
|
if (!BaseInstaller::add(to))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject obj;
|
|
|
|
obj.insert("order", 5);
|
|
|
|
|
2013-09-22 04:21:36 +02:00
|
|
|
if (!m_forge_version)
|
|
|
|
return false;
|
|
|
|
int sliding_insert_window = 0;
|
|
|
|
{
|
2014-01-23 21:31:41 +01:00
|
|
|
QJsonArray librariesPlus;
|
|
|
|
|
2013-09-22 04:21:36 +02:00
|
|
|
// for each library in the version we are adding (except for the blacklisted)
|
|
|
|
QSet<QString> blacklist{"lwjgl", "lwjgl_util", "lwjgl-platform"};
|
|
|
|
for (auto lib : m_forge_version->libraries)
|
|
|
|
{
|
|
|
|
QString libName = lib->name();
|
2013-09-30 02:34:46 +02:00
|
|
|
// WARNING: This could actually break.
|
2013-09-22 04:21:36 +02:00
|
|
|
// if this is the actual forge lib, set an absolute url for the download
|
2013-09-22 14:00:37 +02:00
|
|
|
if (libName.contains("minecraftforge"))
|
2013-09-22 04:21:36 +02:00
|
|
|
{
|
|
|
|
lib->setAbsoluteUrl(m_universal_url);
|
|
|
|
}
|
2013-09-30 02:34:46 +02:00
|
|
|
else if (libName.contains("scala"))
|
|
|
|
{
|
|
|
|
lib->setHint("forge-pack-xz");
|
|
|
|
}
|
2013-09-22 04:21:36 +02:00
|
|
|
if (blacklist.contains(libName))
|
|
|
|
continue;
|
|
|
|
|
2014-01-23 21:31:41 +01:00
|
|
|
QJsonObject libObj = lib->toJson();
|
|
|
|
|
2013-09-22 04:21:36 +02:00
|
|
|
bool found = false;
|
2014-01-27 19:20:07 +01:00
|
|
|
bool equals = false;
|
2014-01-23 21:31:41 +01:00
|
|
|
// find an entry that matches this one
|
2014-02-02 14:17:44 +01:00
|
|
|
for (auto tolib : to->getVanillaVersion()->libraries)
|
2013-09-22 04:21:36 +02:00
|
|
|
{
|
|
|
|
if (tolib->name() != libName)
|
|
|
|
continue;
|
|
|
|
found = true;
|
2014-01-27 19:20:07 +01:00
|
|
|
if (tolib->toJson() == libObj)
|
|
|
|
{
|
|
|
|
equals = true;
|
|
|
|
}
|
2013-09-22 04:21:36 +02:00
|
|
|
// replace lib
|
2014-01-27 22:23:07 +01:00
|
|
|
libObj.insert("insert", QString("replace"));
|
2013-09-22 04:21:36 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-01-27 19:20:07 +01:00
|
|
|
if (equals)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2013-09-22 04:21:36 +02:00
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
// add lib
|
2014-02-02 14:05:07 +01:00
|
|
|
libObj.insert("insert", QString("prepend"));
|
|
|
|
if (lib->name() == "minecraftforge")
|
|
|
|
{
|
|
|
|
libObj.insert("MMC-depend", QString("hard"));
|
|
|
|
}
|
2013-09-22 04:21:36 +02:00
|
|
|
sliding_insert_window++;
|
|
|
|
}
|
2014-01-27 19:20:07 +01:00
|
|
|
librariesPlus.prepend(libObj);
|
2013-09-22 04:21:36 +02:00
|
|
|
}
|
2014-01-23 21:31:41 +01:00
|
|
|
obj.insert("+libraries", librariesPlus);
|
|
|
|
obj.insert("mainClass", m_forge_version->mainClass);
|
2014-01-27 19:20:07 +01:00
|
|
|
QString args = m_forge_version->minecraftArguments;
|
|
|
|
QStringList tweakers;
|
|
|
|
{
|
|
|
|
QRegularExpression expression("--tweakClass ([a-zA-Z0-9\\.]*)");
|
|
|
|
QRegularExpressionMatch match = expression.match(args);
|
|
|
|
while (match.hasMatch())
|
|
|
|
{
|
|
|
|
tweakers.append(match.captured(1));
|
|
|
|
args.remove(match.capturedStart(), match.capturedLength());
|
|
|
|
match = expression.match(args);
|
|
|
|
}
|
|
|
|
}
|
2014-02-02 14:17:44 +01:00
|
|
|
if (!args.isEmpty() && args != to->getVanillaVersion()->minecraftArguments)
|
2014-01-27 19:20:07 +01:00
|
|
|
{
|
|
|
|
obj.insert("minecraftArguments", args);
|
|
|
|
}
|
|
|
|
if (!tweakers.isEmpty())
|
|
|
|
{
|
|
|
|
obj.insert("+tweakers", QJsonArray::fromStringList(tweakers));
|
|
|
|
}
|
|
|
|
if (!m_forge_version->processArguments.isEmpty() &&
|
2014-02-02 14:17:44 +01:00
|
|
|
m_forge_version->processArguments != to->getVanillaVersion()->processArguments)
|
2014-01-27 19:20:07 +01:00
|
|
|
{
|
|
|
|
obj.insert("processArguments", m_forge_version->processArguments);
|
|
|
|
}
|
2014-01-23 21:31:41 +01:00
|
|
|
}
|
|
|
|
|
2014-02-01 14:52:21 +01:00
|
|
|
obj.insert("name", QString("Forge"));
|
2014-02-01 16:26:38 +01:00
|
|
|
obj.insert("fileId", id());
|
2014-02-01 14:52:21 +01:00
|
|
|
obj.insert("version", m_forgeVersionString);
|
|
|
|
obj.insert("mcVersion", to->intendedVersionId());
|
|
|
|
|
2014-01-23 21:31:41 +01:00
|
|
|
QFile file(filename(to->instanceRoot()));
|
|
|
|
if (!file.open(QFile::WriteOnly))
|
|
|
|
{
|
2014-01-27 19:20:07 +01:00
|
|
|
QLOG_ERROR() << "Error opening" << file.fileName()
|
|
|
|
<< "for reading:" << file.errorString();
|
2014-01-23 21:31:41 +01:00
|
|
|
return false;
|
2013-09-22 04:21:36 +02:00
|
|
|
}
|
2014-01-23 21:31:41 +01:00
|
|
|
file.write(QJsonDocument(obj).toJson());
|
|
|
|
file.close();
|
|
|
|
|
2014-01-22 07:33:32 +01:00
|
|
|
return true;
|
2013-09-22 04:21:36 +02:00
|
|
|
}
|