2023-02-04 21:41:24 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2023 arthomnix
|
|
|
|
*
|
|
|
|
* This source is subject to the Microsoft Public License (MS-PL).
|
|
|
|
* Please see the COPYING.md file for more information.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <QDir>
|
|
|
|
#include <QDirIterator>
|
|
|
|
#include <QCryptographicHash>
|
|
|
|
#include <QMap>
|
|
|
|
#include "Json.h"
|
|
|
|
#include "ModrinthInstanceExportTask.h"
|
|
|
|
#include "net/NetJob.h"
|
|
|
|
#include "Application.h"
|
|
|
|
#include "ui/dialogs/ModrinthExportDialog.h"
|
|
|
|
#include "JlCompress.h"
|
|
|
|
#include "FileSystem.h"
|
2023-02-05 21:09:16 +00:00
|
|
|
#include "ModrinthHashLookupRequest.h"
|
2023-02-04 21:41:24 +00:00
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
namespace Modrinth
|
|
|
|
{
|
|
|
|
|
|
|
|
InstanceExportTask::InstanceExportTask(InstancePtr instance, ExportSettings settings) : m_instance(instance), m_settings(settings) {}
|
2023-02-04 21:41:24 +00:00
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
void InstanceExportTask::executeTask()
|
2023-02-04 21:41:24 +00:00
|
|
|
{
|
2023-02-05 12:33:56 +00:00
|
|
|
setStatus(tr("Finding files to look up on Modrinth..."));
|
|
|
|
|
2023-02-04 21:41:24 +00:00
|
|
|
QDir modsDir(m_instance->gameRoot() + "/mods");
|
|
|
|
modsDir.setFilter(QDir::Files);
|
2023-05-23 19:51:48 +00:00
|
|
|
QStringList modsFilter = { "*.jar" };
|
|
|
|
if (m_settings.treatDisabledAsOptional) {
|
|
|
|
modsFilter << "*.jar.disabled";
|
|
|
|
}
|
|
|
|
modsDir.setNameFilters(modsFilter);
|
|
|
|
|
|
|
|
QStringList zipFilter = { "*.zip" };
|
|
|
|
if (m_settings.treatDisabledAsOptional) {
|
|
|
|
zipFilter << "*.zip.disabled";
|
|
|
|
}
|
2023-02-04 21:41:24 +00:00
|
|
|
|
|
|
|
QDir resourcePacksDir(m_instance->gameRoot() + "/resourcepacks");
|
|
|
|
resourcePacksDir.setFilter(QDir::Files);
|
2023-05-23 19:51:48 +00:00
|
|
|
resourcePacksDir.setNameFilters(zipFilter);
|
2023-02-04 21:41:24 +00:00
|
|
|
|
|
|
|
QDir shaderPacksDir(m_instance->gameRoot() + "/shaderpacks");
|
|
|
|
shaderPacksDir.setFilter(QDir::Files);
|
2023-05-23 19:51:48 +00:00
|
|
|
shaderPacksDir.setNameFilters(zipFilter);
|
2023-02-04 21:41:24 +00:00
|
|
|
|
|
|
|
QStringList filesToResolve;
|
|
|
|
|
|
|
|
if (modsDir.exists()) {
|
|
|
|
QDirIterator modsIterator(modsDir);
|
|
|
|
while (modsIterator.hasNext()) {
|
|
|
|
filesToResolve << modsIterator.next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_settings.includeResourcePacks && resourcePacksDir.exists()) {
|
|
|
|
QDirIterator resourcePacksIterator(resourcePacksDir);
|
|
|
|
while (resourcePacksIterator.hasNext()) {
|
|
|
|
filesToResolve << resourcePacksIterator.next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_settings.includeShaderPacks && shaderPacksDir.exists()) {
|
|
|
|
QDirIterator shaderPacksIterator(shaderPacksDir);
|
|
|
|
while (shaderPacksIterator.hasNext()) {
|
|
|
|
filesToResolve << shaderPacksIterator.next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-05 12:03:00 +00:00
|
|
|
if (!m_settings.datapacksPath.isEmpty()) {
|
|
|
|
QDir datapacksDir(m_instance->gameRoot() + "/" + m_settings.datapacksPath);
|
|
|
|
datapacksDir.setFilter(QDir::Files);
|
|
|
|
datapacksDir.setNameFilters(QStringList() << "*.zip");
|
|
|
|
|
|
|
|
if (datapacksDir.exists()) {
|
|
|
|
QDirIterator datapacksIterator(datapacksDir);
|
|
|
|
while (datapacksIterator.hasNext()) {
|
|
|
|
filesToResolve << datapacksIterator.next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-04 21:41:24 +00:00
|
|
|
m_netJob = new NetJob(tr("Modrinth pack export"), APPLICATION->network());
|
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
QList<HashLookupData> hashes;
|
|
|
|
|
|
|
|
qint64 progress = 0;
|
|
|
|
setProgress(progress, filesToResolve.length());
|
2023-02-05 10:13:13 +00:00
|
|
|
for (const QString &filePath: filesToResolve) {
|
|
|
|
qDebug() << "Attempting to resolve file hash from Modrinth API: " << filePath;
|
2023-02-04 21:41:24 +00:00
|
|
|
QFile file(filePath);
|
|
|
|
|
|
|
|
if (file.open(QFile::ReadOnly)) {
|
|
|
|
QByteArray contents = file.readAll();
|
|
|
|
QCryptographicHash hasher(QCryptographicHash::Sha512);
|
|
|
|
hasher.addData(contents);
|
|
|
|
QString hash = hasher.result().toHex();
|
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
hashes.append(HashLookupData {
|
|
|
|
QFileInfo(file),
|
|
|
|
hash
|
2023-02-04 21:41:24 +00:00
|
|
|
});
|
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
progress++;
|
|
|
|
setProgress(progress, filesToResolve.length());
|
2023-02-04 21:41:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
m_response.reset(new QList<HashLookupResponseData>);
|
|
|
|
|
|
|
|
m_netJob->addNetAction(HashLookupRequest::make(hashes, m_response.get()));
|
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
connect(m_netJob.get(), &NetJob::succeeded, this, &InstanceExportTask::lookupSucceeded);
|
|
|
|
connect(m_netJob.get(), &NetJob::failed, this, &InstanceExportTask::lookupFailed);
|
|
|
|
connect(m_netJob.get(), &NetJob::progress, this, &InstanceExportTask::lookupProgress);
|
2023-02-04 21:41:24 +00:00
|
|
|
|
|
|
|
m_netJob->start();
|
2023-02-05 12:33:56 +00:00
|
|
|
setStatus(tr("Looking up files on Modrinth..."));
|
2023-02-04 21:41:24 +00:00
|
|
|
}
|
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
void InstanceExportTask::lookupSucceeded()
|
2023-02-04 21:41:24 +00:00
|
|
|
{
|
2023-02-05 12:33:56 +00:00
|
|
|
setStatus(tr("Creating modpack metadata..."));
|
2023-02-05 14:28:18 +00:00
|
|
|
QList<ExportFile> resolvedFiles;
|
2023-02-04 21:41:24 +00:00
|
|
|
QFileInfoList failedFiles;
|
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
for (const auto &file : *m_response) {
|
|
|
|
if (file.found) {
|
|
|
|
try {
|
|
|
|
auto url = Json::requireString(file.fileJson, "url");
|
|
|
|
auto hashes = Json::requireObject(file.fileJson, "hashes");
|
2023-02-05 17:36:44 +00:00
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
QString sha512Hash = Json::requireString(hashes, "sha512");
|
|
|
|
QString sha1Hash = Json::requireString(hashes, "sha1");
|
2023-02-05 17:36:44 +00:00
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
ExportFile fileData;
|
2023-02-05 17:36:44 +00:00
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
QDir gameDir(m_instance->gameRoot());
|
2023-02-04 21:41:24 +00:00
|
|
|
|
2023-05-23 19:51:48 +00:00
|
|
|
QString path = file.fileInfo.absoluteFilePath();
|
|
|
|
if (path.endsWith(".disabled")) {
|
|
|
|
fileData.optional = true;
|
|
|
|
path = path.left(path.length() - QString(".disabled").length());
|
|
|
|
}
|
|
|
|
|
|
|
|
fileData.path = gameDir.relativeFilePath(path);
|
2023-02-05 21:09:16 +00:00
|
|
|
fileData.download = url;
|
|
|
|
fileData.sha512 = sha512Hash;
|
|
|
|
fileData.sha1 = sha1Hash;
|
|
|
|
fileData.fileSize = file.fileInfo.size();
|
2023-02-04 21:41:24 +00:00
|
|
|
|
2023-02-05 21:09:16 +00:00
|
|
|
resolvedFiles << fileData;
|
|
|
|
} catch (const Json::JsonException &e) {
|
|
|
|
qDebug() << "File " << file.fileInfo.absoluteFilePath() << " failed to process for reason " << e.cause() << ", adding to overrides";
|
|
|
|
failedFiles << file.fileInfo;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
failedFiles << file.fileInfo;
|
2023-02-04 21:41:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject indexJson;
|
|
|
|
indexJson.insert("formatVersion", QJsonValue(1));
|
|
|
|
indexJson.insert("game", QJsonValue("minecraft"));
|
|
|
|
indexJson.insert("versionId", QJsonValue(m_settings.version));
|
|
|
|
indexJson.insert("name", QJsonValue(m_settings.name));
|
|
|
|
|
|
|
|
if (!m_settings.description.isEmpty()) {
|
|
|
|
indexJson.insert("summary", QJsonValue(m_settings.description));
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonArray files;
|
|
|
|
|
|
|
|
for (const auto &file : resolvedFiles) {
|
|
|
|
QJsonObject fileObj;
|
|
|
|
fileObj.insert("path", file.path);
|
|
|
|
|
|
|
|
QJsonObject hashes;
|
|
|
|
hashes.insert("sha512", file.sha512);
|
|
|
|
hashes.insert("sha1", file.sha1);
|
|
|
|
fileObj.insert("hashes", hashes);
|
|
|
|
|
|
|
|
QJsonArray downloads;
|
|
|
|
downloads.append(file.download);
|
|
|
|
fileObj.insert("downloads", downloads);
|
|
|
|
|
|
|
|
fileObj.insert("fileSize", QJsonValue(file.fileSize));
|
|
|
|
|
2023-05-23 19:51:48 +00:00
|
|
|
if (file.optional) {
|
|
|
|
QJsonObject env;
|
|
|
|
env.insert("client", "optional");
|
|
|
|
env.insert("server", "optional");
|
|
|
|
fileObj.insert("env", env);
|
|
|
|
}
|
|
|
|
|
2023-02-04 21:41:24 +00:00
|
|
|
files.append(fileObj);
|
|
|
|
}
|
|
|
|
|
|
|
|
indexJson.insert("files", files);
|
|
|
|
|
|
|
|
QJsonObject dependencies;
|
|
|
|
dependencies.insert("minecraft", m_settings.gameVersion);
|
|
|
|
if (!m_settings.forgeVersion.isEmpty()) {
|
|
|
|
dependencies.insert("forge", m_settings.forgeVersion);
|
|
|
|
}
|
|
|
|
if (!m_settings.fabricVersion.isEmpty()) {
|
|
|
|
dependencies.insert("fabric-loader", m_settings.fabricVersion);
|
|
|
|
}
|
|
|
|
if (!m_settings.quiltVersion.isEmpty()) {
|
|
|
|
dependencies.insert("quilt-loader", m_settings.quiltVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
indexJson.insert("dependencies", dependencies);
|
|
|
|
|
2023-02-05 12:33:56 +00:00
|
|
|
setStatus(tr("Copying files to modpack..."));
|
|
|
|
|
2023-02-04 21:41:24 +00:00
|
|
|
QTemporaryDir tmp;
|
|
|
|
if (tmp.isValid()) {
|
2023-02-05 16:33:20 +00:00
|
|
|
Json::write(indexJson, tmp.path() + "/modrinth.index.json");
|
2023-02-04 21:41:24 +00:00
|
|
|
|
|
|
|
if (!failedFiles.isEmpty()) {
|
|
|
|
QDir tmpDir(tmp.path());
|
|
|
|
QDir gameDir(m_instance->gameRoot());
|
|
|
|
for (const auto &file : failedFiles) {
|
|
|
|
QString src = file.absoluteFilePath();
|
|
|
|
tmpDir.mkpath("overrides/" + gameDir.relativeFilePath(file.absolutePath()));
|
|
|
|
QString dest = tmpDir.path() + "/overrides/" + gameDir.relativeFilePath(src);
|
|
|
|
if (!QFile::copy(file.absoluteFilePath(), dest)) {
|
|
|
|
emitFailed(tr("Failed to copy file %1 to overrides").arg(src));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_settings.includeGameConfig) {
|
|
|
|
tmpDir.mkdir("overrides");
|
|
|
|
QFile::copy(gameDir.absoluteFilePath("options.txt"), tmpDir.absoluteFilePath("overrides/options.txt"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_settings.includeModConfigs) {
|
|
|
|
tmpDir.mkdir("overrides");
|
|
|
|
FS::copy copy(m_instance->gameRoot() + "/config", tmpDir.absoluteFilePath("overrides/config"));
|
|
|
|
copy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-05 12:33:56 +00:00
|
|
|
setStatus(tr("Zipping modpack..."));
|
2023-02-04 21:41:24 +00:00
|
|
|
if (!JlCompress::compressDir(m_settings.exportPath, tmp.path())) {
|
|
|
|
emitFailed(tr("Failed to create zip file"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
emitFailed(tr("Failed to create temporary directory"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-05 10:13:13 +00:00
|
|
|
qDebug() << "Successfully exported Modrinth pack to " << m_settings.exportPath;
|
2023-02-04 21:41:24 +00:00
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
void InstanceExportTask::lookupFailed(const QString &reason)
|
2023-02-04 21:41:24 +00:00
|
|
|
{
|
2023-02-05 10:13:13 +00:00
|
|
|
emitFailed(reason);
|
2023-02-04 21:41:24 +00:00
|
|
|
}
|
|
|
|
|
2023-02-05 14:28:18 +00:00
|
|
|
void InstanceExportTask::lookupProgress(qint64 current, qint64 total)
|
2023-02-04 21:41:24 +00:00
|
|
|
{
|
|
|
|
setProgress(current, total);
|
2023-02-05 14:28:18 +00:00
|
|
|
}
|
|
|
|
|
2023-02-04 21:41:24 +00:00
|
|
|
}
|