aboutsummaryrefslogtreecommitdiff
path: root/launcher/modplatform
diff options
context:
space:
mode:
Diffstat (limited to 'launcher/modplatform')
-rw-r--r--launcher/modplatform/atlauncher/ATLPackInstallTask.cpp20
-rw-r--r--launcher/modplatform/atlauncher/ATLPackInstallTask.h1
-rw-r--r--launcher/modplatform/flame/FlameAPI.cpp23
-rw-r--r--launcher/modplatform/flame/FlameAPI.h1
-rw-r--r--launcher/modplatform/flame/FlameInstanceCreationTask.cpp457
-rw-r--r--launcher/modplatform/flame/FlameInstanceCreationTask.h44
-rw-r--r--launcher/modplatform/flame/PackManifest.cpp28
-rw-r--r--launcher/modplatform/flame/PackManifest.h10
-rw-r--r--launcher/modplatform/helpers/OverrideUtils.cpp59
-rw-r--r--launcher/modplatform/helpers/OverrideUtils.h20
-rw-r--r--launcher/modplatform/legacy_ftb/PackFetchTask.cpp14
-rw-r--r--launcher/modplatform/legacy_ftb/PackFetchTask.h2
-rw-r--r--launcher/modplatform/legacy_ftb/PackInstallTask.cpp8
-rw-r--r--launcher/modplatform/legacy_ftb/PackInstallTask.h1
-rw-r--r--launcher/modplatform/modpacksch/FTBPackInstallTask.cpp5
-rw-r--r--launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp407
-rw-r--r--launcher/modplatform/modrinth/ModrinthInstanceCreationTask.h44
-rw-r--r--launcher/modplatform/packwiz/Packwiz.cpp156
-rw-r--r--launcher/modplatform/technic/SingleZipPackInstallTask.cpp2
-rw-r--r--launcher/modplatform/technic/SolderPackInstallTask.cpp10
-rw-r--r--launcher/modplatform/technic/SolderPackInstallTask.h1
21 files changed, 1209 insertions, 104 deletions
diff --git a/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp b/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
index 70a35395..a553eafd 100644
--- a/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
+++ b/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
@@ -90,6 +90,7 @@ void PackInstallTask::executeTask()
QObject::connect(netJob, &NetJob::succeeded, this, &PackInstallTask::onDownloadSucceeded);
QObject::connect(netJob, &NetJob::failed, this, &PackInstallTask::onDownloadFailed);
+ QObject::connect(netJob, &NetJob::aborted, this, &PackInstallTask::onDownloadAborted);
}
void PackInstallTask::onDownloadSucceeded()
@@ -169,6 +170,12 @@ void PackInstallTask::onDownloadFailed(QString reason)
emitFailed(reason);
}
+void PackInstallTask::onDownloadAborted()
+{
+ jobPtr.reset();
+ emitAborted();
+}
+
void PackInstallTask::deleteExistingFiles()
{
setStatus(tr("Deleting existing files..."));
@@ -675,6 +682,11 @@ void PackInstallTask::installConfigs()
abortable = true;
setProgress(current, total);
});
+ connect(jobPtr.get(), &NetJob::aborted, [&]{
+ abortable = false;
+ jobPtr.reset();
+ emitAborted();
+ });
jobPtr->start();
}
@@ -831,6 +843,12 @@ void PackInstallTask::downloadMods()
abortable = true;
setProgress(current, total);
});
+ connect(jobPtr.get(), &NetJob::aborted, [&]
+ {
+ abortable = false;
+ jobPtr.reset();
+ emitAborted();
+ });
jobPtr->start();
}
@@ -1005,7 +1023,7 @@ void PackInstallTask::install()
components->saveNow();
- instance.setName(m_instName);
+ instance.setName(name());
instance.setIconKey(m_instIcon);
instance.setManagedPack("atlauncher", m_pack_safe_name, m_pack_name, m_version_name, m_version_name);
instanceSettings->resumeSave();
diff --git a/launcher/modplatform/atlauncher/ATLPackInstallTask.h b/launcher/modplatform/atlauncher/ATLPackInstallTask.h
index a7124d59..ed4436f0 100644
--- a/launcher/modplatform/atlauncher/ATLPackInstallTask.h
+++ b/launcher/modplatform/atlauncher/ATLPackInstallTask.h
@@ -93,6 +93,7 @@ protected:
private slots:
void onDownloadSucceeded();
void onDownloadFailed(QString reason);
+ void onDownloadAborted();
void onModsDownloaded();
void onModsExtracted();
diff --git a/launcher/modplatform/flame/FlameAPI.cpp b/launcher/modplatform/flame/FlameAPI.cpp
index 9c74918b..4d71da21 100644
--- a/launcher/modplatform/flame/FlameAPI.cpp
+++ b/launcher/modplatform/flame/FlameAPI.cpp
@@ -183,3 +183,26 @@ auto FlameAPI::getProjects(QStringList addonIds, QByteArray* response) const ->
return netJob;
}
+
+auto FlameAPI::getFiles(const QStringList& fileIds, QByteArray* response) const -> NetJob*
+{
+ auto* netJob = new NetJob(QString("Flame::GetFiles"), APPLICATION->network());
+
+ QJsonObject body_obj;
+ QJsonArray files_arr;
+ for (auto& fileId : fileIds) {
+ files_arr.append(fileId);
+ }
+
+ body_obj["fileIds"] = files_arr;
+
+ QJsonDocument body(body_obj);
+ auto body_raw = body.toJson();
+
+ netJob->addNetAction(Net::Upload::makeByteArray(QString("https://api.curseforge.com/v1/mods/files"), response, body_raw));
+
+ QObject::connect(netJob, &NetJob::finished, [response, netJob] { delete response; netJob->deleteLater(); });
+ QObject::connect(netJob, &NetJob::failed, [body_raw] { qDebug() << body_raw; });
+
+ return netJob;
+}
diff --git a/launcher/modplatform/flame/FlameAPI.h b/launcher/modplatform/flame/FlameAPI.h
index 4eac0664..4c6ca64c 100644
--- a/launcher/modplatform/flame/FlameAPI.h
+++ b/launcher/modplatform/flame/FlameAPI.h
@@ -12,6 +12,7 @@ class FlameAPI : public NetworkModAPI {
auto getLatestVersion(VersionSearchArgs&& args) -> ModPlatform::IndexedVersion;
auto getProjects(QStringList addonIds, QByteArray* response) const -> NetJob* override;
+ auto getFiles(const QStringList& fileIds, QByteArray* response) const -> NetJob*;
private:
inline auto getSortFieldInt(QString sortString) const -> int
diff --git a/launcher/modplatform/flame/FlameInstanceCreationTask.cpp b/launcher/modplatform/flame/FlameInstanceCreationTask.cpp
new file mode 100644
index 00000000..48ac02e0
--- /dev/null
+++ b/launcher/modplatform/flame/FlameInstanceCreationTask.cpp
@@ -0,0 +1,457 @@
+#include "FlameInstanceCreationTask.h"
+
+#include "modplatform/flame/FlameAPI.h"
+#include "modplatform/flame/PackManifest.h"
+
+#include "Application.h"
+#include "FileSystem.h"
+#include "InstanceList.h"
+#include "Json.h"
+
+#include "minecraft/MinecraftInstance.h"
+#include "minecraft/PackProfile.h"
+
+#include "modplatform/helpers/OverrideUtils.h"
+
+#include "settings/INISettingsObject.h"
+
+#include "ui/dialogs/BlockedModsDialog.h"
+#include "ui/dialogs/CustomMessageBox.h"
+
+const static QMap<QString, QString> forgemap = { { "1.2.5", "3.4.9.171" },
+ { "1.4.2", "6.0.1.355" },
+ { "1.4.7", "6.6.2.534" },
+ { "1.5.2", "7.8.1.737" } };
+
+static const FlameAPI api;
+
+bool FlameCreationTask::abort()
+{
+ if (!canAbort())
+ return false;
+
+ m_abort = true;
+ if (m_process_update_file_info_job)
+ m_process_update_file_info_job->abort();
+ if (m_files_job)
+ m_files_job->abort();
+ if (m_mod_id_resolver)
+ m_mod_id_resolver->abort();
+
+ return Task::abort();
+}
+
+bool FlameCreationTask::updateInstance()
+{
+ auto instance_list = APPLICATION->instances();
+
+ // FIXME: How to handle situations when there's more than one install already for a given modpack?
+ auto inst = instance_list->getInstanceByManagedName(originalName());
+
+ if (!inst) {
+ inst = instance_list->getInstanceById(originalName());
+
+ if (!inst)
+ return false;
+ }
+
+ QString index_path(FS::PathCombine(m_stagingPath, "manifest.json"));
+
+ try {
+ Flame::loadManifest(m_pack, index_path);
+ } catch (const JSONValidationError& e) {
+ setError(tr("Could not understand pack manifest:\n") + e.cause());
+ return false;
+ }
+
+ auto version_id = inst->getManagedPackVersionName();
+ auto version_str = !version_id.isEmpty() ? tr(" (version %1)").arg(version_id) : "";
+
+ auto info = CustomMessageBox::selectable(
+ m_parent, tr("Similar modpack was found!"),
+ tr("One or more of your instances are from this same modpack%1. Do you want to create a "
+ "separate instance, or update the existing one?\n\nNOTE: Make sure you made a backup of your important instance data before "
+ "updating, as worlds can be corrupted and some configuration may be lost (due to pack overrides).")
+ .arg(version_str), QMessageBox::Information, QMessageBox::Ok | QMessageBox::Reset | QMessageBox::Abort);
+ info->setButtonText(QMessageBox::Ok, tr("Update existing instance"));
+ info->setButtonText(QMessageBox::Abort, tr("Create new instance"));
+ info->setButtonText(QMessageBox::Reset, tr("Cancel"));
+
+ info->exec();
+
+ if (info->clickedButton() == info->button(QMessageBox::Abort))
+ return false;
+
+ if (info->clickedButton() == info->button(QMessageBox::Reset)) {
+ m_abort = true;
+ return false;
+ }
+
+ QDir old_inst_dir(inst->instanceRoot());
+
+ QString old_index_folder(FS::PathCombine(old_inst_dir.absolutePath(), "flame"));
+ QString old_index_path(FS::PathCombine(old_index_folder, "manifest.json"));
+
+ QFileInfo old_index_file(old_index_path);
+ if (old_index_file.exists()) {
+ Flame::Manifest old_pack;
+ Flame::loadManifest(old_pack, old_index_path);
+
+ auto& old_files = old_pack.files;
+
+ auto& files = m_pack.files;
+
+ // Remove repeated files, we don't need to download them!
+ auto files_iterator = files.begin();
+ while (files_iterator != files.end()) {
+ auto const& file = files_iterator;
+
+ auto old_file = old_files.find(file.key());
+ if (old_file != old_files.end()) {
+ // We found a match, but is it a different version?
+ if (old_file->fileId == file->fileId) {
+ qDebug() << "Removed file at" << file->targetFolder << "with id" << file->fileId << "from list of downloads";
+
+ old_files.remove(file.key());
+ files_iterator = files.erase(files_iterator);
+ }
+ }
+
+ files_iterator++;
+ }
+
+ QDir old_minecraft_dir(inst->gameRoot());
+
+ // We will remove all the previous overrides, to prevent duplicate files!
+ // TODO: Currently 'overrides' will always override the stuff on update. How do we preserve unchanged overrides?
+ // FIXME: We may want to do something about disabled mods.
+ auto old_overrides = Override::readOverrides("overrides", old_index_folder);
+ for (const auto& entry : old_overrides) {
+ if (entry.isEmpty())
+ continue;
+ qDebug() << "Scheduling" << entry << "for removal";
+ m_files_to_remove.append(old_minecraft_dir.absoluteFilePath(entry));
+ }
+
+ // Remove remaining old files (we need to do an API request to know which ids are which files...)
+ QStringList fileIds;
+
+ for (auto& file : old_files) {
+ fileIds.append(QString::number(file.fileId));
+ }
+
+ auto* raw_response = new QByteArray;
+ auto job = api.getFiles(fileIds, raw_response);
+
+ QEventLoop loop;
+
+ connect(job, &NetJob::succeeded, this, [this, raw_response, fileIds, old_inst_dir, &old_files, old_minecraft_dir] {
+ // Parse the API response
+ QJsonParseError parse_error{};
+ auto doc = QJsonDocument::fromJson(*raw_response, &parse_error);
+ if (parse_error.error != QJsonParseError::NoError) {
+ qWarning() << "Error while parsing JSON response from Flame files task at " << parse_error.offset
+ << " reason: " << parse_error.errorString();
+ qWarning() << *raw_response;
+ return;
+ }
+
+ try {
+ QJsonArray entries;
+ if (fileIds.size() == 1)
+ entries = { Json::requireObject(Json::requireObject(doc), "data") };
+ else
+ entries = Json::requireArray(Json::requireObject(doc), "data");
+
+ for (auto entry : entries) {
+ auto entry_obj = Json::requireObject(entry);
+
+ Flame::File file;
+ // We don't care about blocked mods, we just need local data to delete the file
+ file.parseFromObject(entry_obj, false);
+
+ auto id = Json::requireInteger(entry_obj, "id");
+ old_files.insert(id, file);
+ }
+ } catch (Json::JsonException& e) {
+ qCritical() << e.cause() << e.what();
+ }
+
+ // Delete the files
+ for (auto& file : old_files) {
+ if (file.fileName.isEmpty() || file.targetFolder.isEmpty())
+ continue;
+
+ QString relative_path(FS::PathCombine(file.targetFolder, file.fileName));
+ qDebug() << "Scheduling" << relative_path << "for removal";
+ m_files_to_remove.append(old_minecraft_dir.absoluteFilePath(relative_path));
+ }
+ });
+ connect(job, &NetJob::finished, &loop, &QEventLoop::quit);
+
+ m_process_update_file_info_job = job;
+ job->start();
+
+ loop.exec();
+
+ m_process_update_file_info_job = nullptr;
+ } else {
+ // We don't have an old index file, so we may duplicate stuff!
+ auto dialog = CustomMessageBox::selectable(m_parent,
+ tr("No index file."),
+ tr("We couldn't find a suitable index file for the older version. This may cause some of the files to be duplicated. Do you want to continue?"),
+ QMessageBox::Warning, QMessageBox::Ok | QMessageBox::Cancel);
+
+ if (dialog->exec() == QDialog::DialogCode::Rejected) {
+ m_abort = true;
+ return false;
+ }
+ }
+
+ setOverride(true);
+ qDebug() << "Will override instance!";
+
+ m_instance = inst;
+
+ // We let it go through the createInstance() stage, just with a couple modifications for updating
+ return false;
+}
+
+bool FlameCreationTask::createInstance()
+{
+ QEventLoop loop;
+
+ QString parent_folder(FS::PathCombine(m_stagingPath, "flame"));
+
+ try {
+ QString index_path(FS::PathCombine(m_stagingPath, "manifest.json"));
+ if (!m_pack.is_loaded)
+ Flame::loadManifest(m_pack, index_path);
+
+ // Keep index file in case we need it some other time (like when changing versions)
+ QString new_index_place(FS::PathCombine(parent_folder, "manifest.json"));
+ FS::ensureFilePathExists(new_index_place);
+ QFile::rename(index_path, new_index_place);
+
+ } catch (const JSONValidationError& e) {
+ setError(tr("Could not understand pack manifest:\n") + e.cause());
+ return false;
+ }
+
+ if (!m_pack.overrides.isEmpty()) {
+ QString overridePath = FS::PathCombine(m_stagingPath, m_pack.overrides);
+ if (QFile::exists(overridePath)) {
+ // Create a list of overrides in "overrides.txt" inside flame/
+ Override::createOverrides("overrides", parent_folder, overridePath);
+
+ QString mcPath = FS::PathCombine(m_stagingPath, "minecraft");
+ if (!QFile::rename(overridePath, mcPath)) {
+ setError(tr("Could not rename the overrides folder:\n") + m_pack.overrides);
+ return false;
+ }
+ } else {
+ logWarning(
+ tr("The specified overrides folder (%1) is missing. Maybe the modpack was already used before?").arg(m_pack.overrides));
+ }
+ }
+
+ QString forgeVersion;
+ QString fabricVersion;
+ // TODO: is Quilt relevant here?
+ for (auto& loader : m_pack.minecraft.modLoaders) {
+ auto id = loader.id;
+ if (id.startsWith("forge-")) {
+ id.remove("forge-");
+ forgeVersion = id;
+ continue;
+ }
+ if (id.startsWith("fabric-")) {
+ id.remove("fabric-");
+ fabricVersion = id;
+ continue;
+ }
+ logWarning(tr("Unknown mod loader in manifest: %1").arg(id));
+ }
+
+ QString configPath = FS::PathCombine(m_stagingPath, "instance.cfg");
+ auto instanceSettings = std::make_shared<INISettingsObject>(configPath);
+ MinecraftInstance instance(m_globalSettings, instanceSettings, m_stagingPath);
+ auto mcVersion = m_pack.minecraft.version;
+
+ // Hack to correct some 'special sauce'...
+ if (mcVersion.endsWith('.')) {
+ mcVersion.remove(QRegularExpression("[.]+$"));
+ logWarning(tr("Mysterious trailing dots removed from Minecraft version while importing pack."));
+ }
+
+ auto components = instance.getPackProfile();
+ components->buildingFromScratch();
+ components->setComponentVersion("net.minecraft", mcVersion, true);
+ if (!forgeVersion.isEmpty()) {
+ // FIXME: dirty, nasty, hack. Proper solution requires dependency resolution and knowledge of the metadata.
+ if (forgeVersion == "recommended") {
+ if (forgemap.contains(mcVersion)) {
+ forgeVersion = forgemap[mcVersion];
+ } else {
+ logWarning(tr("Could not map recommended Forge version for Minecraft %1").arg(mcVersion));
+ }
+ }
+ components->setComponentVersion("net.minecraftforge", forgeVersion);
+ }
+ if (!fabricVersion.isEmpty())
+ components->setComponentVersion("net.fabricmc.fabric-loader", fabricVersion);
+
+ if (m_instIcon != "default") {
+ instance.setIconKey(m_instIcon);
+ } else {
+ if (m_pack.name.contains("Direwolf20")) {
+ instance.setIconKey("steve");
+ } else if (m_pack.name.contains("FTB") || m_pack.name.contains("Feed The Beast")) {
+ instance.setIconKey("ftb_logo");
+ } else {
+ instance.setIconKey("flame");
+ }
+ }
+
+ QString jarmodsPath = FS::PathCombine(m_stagingPath, "minecraft", "jarmods");
+ QFileInfo jarmodsInfo(jarmodsPath);
+ if (jarmodsInfo.isDir()) {
+ // install all the jar mods
+ qDebug() << "Found jarmods:";
+ QDir jarmodsDir(jarmodsPath);
+ QStringList jarMods;
+ for (const auto& info : jarmodsDir.entryInfoList(QDir::NoDotAndDotDot | QDir::Files)) {
+ qDebug() << info.fileName();
+ jarMods.push_back(info.absoluteFilePath());
+ }
+ auto profile = instance.getPackProfile();
+ profile->installJarMods(jarMods);
+ // nuke the original files
+ FS::deletePath(jarmodsPath);
+ }
+
+ instance.setManagedPack("flame", {}, m_pack.name, {}, m_pack.version);
+ instance.setName(name());
+
+ m_mod_id_resolver = new Flame::FileResolvingTask(APPLICATION->network(), m_pack);
+ connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::succeeded, this, [this, &loop] { idResolverSucceeded(loop); });
+ connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::failed, [&](QString reason) {
+ m_mod_id_resolver.reset();
+ setError(tr("Unable to resolve mod IDs:\n") + reason);
+ });
+ connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::progress, this, &FlameCreationTask::setProgress);
+ connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::status, this, &FlameCreationTask::setStatus);
+
+ m_mod_id_resolver->start();
+
+ loop.exec();
+
+ bool did_succeed = getError().isEmpty();
+
+ // Update information of the already installed instance, if any.
+ if (m_instance && did_succeed) {
+ setAbortable(false);
+ auto inst = m_instance.value();
+
+ // Only change the name if it didn't use a custom name, so that the previous custom name
+ // is preserved, but if we're using the original one, we update the version string.
+ // NOTE: This needs to come before the copyManagedPack call!
+ if (inst->name().contains(inst->getManagedPackVersionName())) {
+ if (askForChangingInstanceName(m_parent, inst->name(), instance.name()) == InstanceNameChange::ShouldChange)
+ inst->setName(instance.name());
+ }
+
+ inst->copyManagedPack(instance);
+ }
+
+ return did_succeed;
+}
+
+void FlameCreationTask::idResolverSucceeded(QEventLoop& loop)
+{
+ auto results = m_mod_id_resolver->getResults();
+
+ // first check for blocked mods
+ QString text;
+ QList<QUrl> urls;
+ auto anyBlocked = false;
+ for (const auto& result : results.files.values()) {
+ if (!result.resolved || result.url.isEmpty()) {
+ text += QString("%1: <a href='%2'>%2</a><br/>").arg(result.fileName, result.websiteUrl);
+ urls.append(QUrl(result.websiteUrl));
+ anyBlocked = true;
+ }
+ }
+ if (anyBlocked) {
+ qWarning() << "Blocked mods found, displaying mod list";
+
+ auto message_dialog = new BlockedModsDialog(m_parent, tr("Blocked mods found"),
+ tr("The following mods were blocked on third party launchers.<br/>"
+ "You will need to manually download them and add them to the modpack"),
+ text,
+ urls);
+ message_dialog->setModal(true);
+
+ if (message_dialog->exec()) {
+ setupDownloadJob(loop);
+ } else {
+ m_mod_id_resolver.reset();
+ setError("Canceled");
+ loop.quit();
+ }
+ } else {
+ setupDownloadJob(loop);
+ }
+}
+
+void FlameCreationTask::setupDownloadJob(QEventLoop& loop)
+{
+ m_files_job = new NetJob(tr("Mod download"), APPLICATION->network());
+ for (const auto& result : m_mod_id_resolver->getResults().files) {
+ QString filename = result.fileName;
+ if (!result.required) {
+ filename += ".disabled";
+ }
+
+ auto relpath = FS::PathCombine("minecraft", result.targetFolder, filename);
+ auto path = FS::PathCombine(m_stagingPath, relpath);
+
+ switch (result.type) {
+ case Flame::File::Type::Folder: {
+ logWarning(tr("This 'Folder' may need extracting: %1").arg(relpath));
+ // fall-through intentional, we treat these as plain old mods and dump them wherever.
+ }
+ case Flame::File::Type::SingleFile:
+ case Flame::File::Type::Mod: {
+ if (!result.url.isEmpty()) {
+ qDebug() << "Will download" << result.url << "to" << path;
+ auto dl = Net::Download::makeFile(result.url, path);
+ m_files_job->addNetAction(dl);
+ }
+ break;
+ }
+ case Flame::File::Type::Modpack:
+ logWarning(tr("Nesting modpacks in modpacks is not implemented, nothing was downloaded: %1").arg(relpath));
+ break;
+ case Flame::File::Type::Cmod2:
+ case Flame::File::Type::Ctoc:
+ case Flame::File::Type::Unknown:
+ logWarning(tr("Unrecognized/unhandled PackageType for: %1").arg(relpath));
+ break;
+ }
+ }
+
+ m_mod_id_resolver.reset();
+ connect(m_files_job.get(), &NetJob::succeeded, this, [&]() {
+ m_files_job.reset();
+ });
+ connect(m_files_job.get(), &NetJob::failed, [&](QString reason) {
+ m_files_job.reset();
+ setError(reason);
+ });
+ connect(m_files_job.get(), &NetJob::progress, [&](qint64 current, qint64 total) { setProgress(current, total); });
+ connect(m_files_job.get(), &NetJob::finished, &loop, &QEventLoop::quit);
+
+ setStatus(tr("Downloading mods..."));
+ m_files_job->start();
+}
diff --git a/launcher/modplatform/flame/FlameInstanceCreationTask.h b/launcher/modplatform/flame/FlameInstanceCreationTask.h
new file mode 100644
index 00000000..ded0e2ce
--- /dev/null
+++ b/launcher/modplatform/flame/FlameInstanceCreationTask.h
@@ -0,0 +1,44 @@
+#pragma once
+
+#include "InstanceCreationTask.h"
+
+#include <optional>
+
+#include "minecraft/MinecraftInstance.h"
+
+#include "modplatform/flame/FileResolvingTask.h"
+
+#include "net/NetJob.h"
+
+class FlameCreationTask final : public InstanceCreationTask {
+ Q_OBJECT
+
+ public:
+ FlameCreationTask(const QString& staging_path, SettingsObjectPtr global_settings, QWidget* parent)
+ : InstanceCreationTask(), m_parent(parent)
+ {
+ setStagingPath(staging_path);
+ setParentSettings(global_settings);
+ }
+
+ bool abort() override;
+
+ bool updateInstance() override;
+ bool createInstance() override;
+
+ private slots:
+ void idResolverSucceeded(QEventLoop&);
+ void setupDownloadJob(QEventLoop&);
+
+ private:
+ QWidget* m_parent = nullptr;
+
+ shared_qobject_ptr<Flame::FileResolvingTask> m_mod_id_resolver;
+ Flame::Manifest m_pack;
+
+ // Handle to allow aborting
+ NetJob* m_process_update_file_info_job = nullptr;
+ NetJob::Ptr m_files_job = nullptr;
+
+ std::optional<InstancePtr> m_instance;
+};
diff --git a/launcher/modplatform/flame/PackManifest.cpp b/launcher/modplatform/flame/PackManifest.cpp
index 12a4b990..22008297 100644
--- a/launcher/modplatform/flame/PackManifest.cpp
+++ b/launcher/modplatform/flame/PackManifest.cpp
@@ -29,21 +29,29 @@ static void loadMinecraftV1(Flame::Minecraft& m, QJsonObject& minecraft)
}
}
-static void loadManifestV1(Flame::Manifest& m, QJsonObject& manifest)
+static void loadManifestV1(Flame::Manifest& pack, QJsonObject& manifest)
{
auto mc = Json::requireObject(manifest, "minecraft");
- loadMinecraftV1(m.minecraft, mc);
- m.name = Json::ensureString(manifest, QString("name"), "Unnamed");
- m.version = Json::ensureString(manifest, QString("version"), QString());
- m.author = Json::ensureString(manifest, QString("author"), "Anonymous");
+
+ loadMinecraftV1(pack.minecraft, mc);
+
+ pack.name = Json::ensureString(manifest, QString("name"), "Unnamed");
+ pack.version = Json::ensureString(manifest, QString("version"), QString());
+ pack.author = Json::ensureString(manifest, QString("author"), "Anonymous");
+
auto arr = Json::ensureArray(manifest, "files", QJsonArray());
- for (QJsonValueRef item : arr) {
+ for (auto item : arr) {
auto obj = Json::requireObject(item);
+
Flame::File file;
loadFileV1(file, obj);
- m.files.insert(file.fileId,file);
+
+ pack.files.insert(file.fileId,file);
}
- m.overrides = Json::ensureString(manifest, "overrides", "overrides");
+
+ pack.overrides = Json::ensureString(manifest, "overrides", "overrides");
+
+ pack.is_loaded = true;
}
void Flame::loadManifest(Flame::Manifest& m, const QString& filepath)
@@ -61,7 +69,7 @@ void Flame::loadManifest(Flame::Manifest& m, const QString& filepath)
loadManifestV1(m, obj);
}
-bool Flame::File::parseFromObject(const QJsonObject& obj)
+bool Flame::File::parseFromObject(const QJsonObject& obj, bool throw_on_blocked)
{
fileName = Json::requireString(obj, "fileName");
// This is a piece of a Flame project JSON pulled out into the file metadata (here) for convenience
@@ -91,7 +99,7 @@ bool Flame::File::parseFromObject(const QJsonObject& obj)
// may throw, if the project is blocked
QString rawUrl = Json::ensureString(obj, "downloadUrl");
url = QUrl(rawUrl, QUrl::TolerantMode);
- if (!url.isValid()) {
+ if (!url.isValid() && throw_on_blocked) {
throw JSONValidationError(QString("Invalid URL: %1").arg(rawUrl));
}
diff --git a/launcher/modplatform/flame/PackManifest.h b/launcher/modplatform/flame/PackManifest.h
index 677db1c3..0b7461d8 100644
--- a/launcher/modplatform/flame/PackManifest.h
+++ b/launcher/modplatform/flame/PackManifest.h
@@ -35,18 +35,18 @@
#pragma once
-#include <QString>
-#include <QVector>
+#include <QJsonObject>
#include <QMap>
+#include <QString>
#include <QUrl>
-#include <QJsonObject>
+#include <QVector>
namespace Flame
{
struct File
{
// NOTE: throws JSONValidationError
- bool parseFromObject(const QJsonObject& object);
+ bool parseFromObject(const QJsonObject& object, bool throw_on_blocked = true);
int projectId = 0;
int fileId = 0;
@@ -97,6 +97,8 @@ struct Manifest
//File id -> File
QMap<int,Flame::File> files;
QString overrides;
+
+ bool is_loaded = false;
};
void loadManifest(Flame::Manifest & m, const QString &filepath);
diff --git a/launcher/modplatform/helpers/OverrideUtils.cpp b/launcher/modplatform/helpers/OverrideUtils.cpp
new file mode 100644
index 00000000..65b5f760
--- /dev/null
+++ b/launcher/modplatform/helpers/OverrideUtils.cpp
@@ -0,0 +1,59 @@
+#include "OverrideUtils.h"
+
+#include <QDirIterator>
+
+#include "FileSystem.h"
+
+namespace Override {
+
+void createOverrides(const QString& name, const QString& parent_folder, const QString& override_path)
+{
+ QString file_path(FS::PathCombine(parent_folder, name + ".txt"));
+ if (QFile::exists(file_path))
+ QFile::remove(file_path);
+
+ FS::ensureFilePathExists(file_path);
+
+ QFile file(file_path);
+ file.open(QFile::WriteOnly);
+
+ QDirIterator override_iterator(override_path, QDirIterator::Subdirectories);
+ while (override_iterator.hasNext()) {
+ auto override_file_path = override_iterator.next();
+ QFileInfo info(override_file_path);
+ if (info.isFile()) {
+ // Absolute path with temp directory -> relative path
+ override_file_path = override_file_path.split(name).last().remove(0, 1);
+
+ file.write(override_file_path.toUtf8());
+ file.write("\n");
+ }
+ }
+
+ file.close();
+}
+
+QStringList readOverrides(const QString& name, const QString& parent_folder)
+{
+ QString file_path(FS::PathCombine(parent_folder, name + ".txt"));
+
+ QFile file(file_path);
+ if (!file.exists())
+ return {};
+
+ QStringList previous_overrides;
+
+ file.open(QFile::ReadOnly);
+
+ QString entry;
+ do {
+ entry = file.readLine();
+ previous_overrides.append(entry.trimmed());
+ } while (!entry.isEmpty());
+
+ file.close();
+
+ return previous_overrides;
+}
+
+} // namespace Override
diff --git a/launcher/modplatform/helpers/OverrideUtils.h b/launcher/modplatform/helpers/OverrideUtils.h
new file mode 100644
index 00000000..536261a2
--- /dev/null
+++ b/launcher/modplatform/helpers/OverrideUtils.h
@@ -0,0 +1,20 @@
+#pragma once
+
+#include <QString>
+
+namespace Override {
+
+/** This creates a file in `parent_folder` that holds information about which
+ * overrides are in `override_path`.
+ *
+ * If there's already an existing such file, it will be ovewritten.
+ */
+void createOverrides(const QString& name, const QString& parent_folder, const QString& override_path);
+
+/** This reads an existing overrides archive, returning a list of overrides.
+ *
+ * If there's no such file in `parent_folder`, it will return an empty list.
+ */
+QStringList readOverrides(const QString& name, const QString& parent_folder);
+
+} // namespace Override
diff --git a/launcher/modplatform/legacy_ftb/PackFetchTask.cpp b/launcher/modplatform/legacy_ftb/PackFetchTask.cpp
index 4da6a866..36aa60c7 100644
--- a/launcher/modplatform/legacy_ftb/PackFetchTask.cpp
+++ b/launcher/modplatform/legacy_ftb/PackFetchTask.cpp
@@ -59,6 +59,7 @@ void PackFetchTask::fetch()
QObject::connect(jobPtr.get(), &NetJob::succeeded, this, &PackFetchTask::fileDownloadFinished);
QObject::connect(jobPtr.get(), &NetJob::failed, this, &PackFetchTask::fileDownloadFailed);
+ QObject::connect(jobPtr.get(), &NetJob::aborted, this, &PackFetchTask::fileDownloadAborted);
jobPtr->start();
}
@@ -98,6 +99,14 @@ void PackFetchTask::fetchPrivate(const QStringList & toFetch)
delete data;
});
+ QObject::connect(job, &NetJob::aborted, this, [this, job, data]{
+ emit aborted();
+ job->deleteLater();
+
+ data->clear();
+ delete data;
+ });
+
job->start();
}
}
@@ -204,4 +213,9 @@ void PackFetchTask::fileDownloadFailed(QString reason)
emit failed(reason);
}
+void PackFetchTask::fileDownloadAborted()
+{
+ emit aborted();
+}
+
}
diff --git a/launcher/modplatform/legacy_ftb/PackFetchTask.h b/launcher/modplatform/legacy_ftb/PackFetchTask.h
index f1667e90..8f3c4f3b 100644
--- a/launcher/modplatform/legacy_ftb/PackFetchTask.h
+++ b/launcher/modplatform/legacy_ftb/PackFetchTask.h
@@ -33,10 +33,12 @@ private:
protected slots:
void fileDownloadFinished();
void fileDownloadFailed(QString reason);
+ void fileDownloadAborted();
signals:
void finished(ModpackList publicPacks, ModpackList thirdPartyPacks);
void failed(QString reason);
+ void aborted();
void privateFileDownloadFinished(Modpack modpack);
void privateFileDownloadFailed(QString reason, QString packCode);
diff --git a/launcher/modplatform/legacy_ftb/PackInstallTask.cpp b/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
index 83e14969..209ad884 100644
--- a/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
+++ b/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
@@ -86,6 +86,7 @@ void PackInstallTask::downloadPack()
connect(netJobContainer.get(), &NetJob::succeeded, this, &PackInstallTask::onDownloadSucceeded);
connect(netJobContainer.get(), &NetJob::failed, this, &PackInstallTask::onDownloadFailed);
connect(netJobContainer.get(), &NetJob::progress, this, &PackInstallTask::onDownloadProgress);
+ connect(netJobContainer.get(), &NetJob::aborted, this, &PackInstallTask::onDownloadAborted);
netJobContainer->start();
progress(1, 4);
@@ -110,6 +111,11 @@ void PackInstallTask::onDownloadProgress(qint64 current, qint64 total)
setStatus(tr("Downloading zip for %1 (%2%)").arg(m_pack.name).arg(current / 10));
}
+void PackInstallTask::onDownloadAborted()
+{
+ emitAborted();
+}
+
void PackInstallTask::unzip()
{
progress(2, 4);
@@ -228,7 +234,7 @@ void PackInstallTask::install()
progress(4, 4);
- instance.setName(m_instName);
+ instance.setName(name());
if(m_instIcon == "default")
{
m_instIcon = "ftb_logo";
diff --git a/launcher/modplatform/legacy_ftb/PackInstallTask.h b/launcher/modplatform/legacy_ftb/PackInstallTask.h
index da4c0da5..da791e06 100644
--- a/launcher/modplatform/legacy_ftb/PackInstallTask.h
+++ b/launcher/modplatform/legacy_ftb/PackInstallTask.h
@@ -38,6 +38,7 @@ private slots:
void onDownloadSucceeded();
void onDownloadFailed(QString reason);
void onDownloadProgress(qint64 current, qint64 total);
+ void onDownloadAborted();
void onUnzipFinished();
void onUnzipCanceled();
diff --git a/launcher/modplatform/modpacksch/FTBPackInstallTask.cpp b/launcher/modplatform/modpacksch/FTBPackInstallTask.cpp
index 3c15667c..97ce1dc6 100644
--- a/launcher/modplatform/modpacksch/FTBPackInstallTask.cpp
+++ b/launcher/modplatform/modpacksch/FTBPackInstallTask.cpp
@@ -65,9 +65,8 @@ bool PackInstallTask::abort()
if (m_mod_id_resolver_task)
aborted &= m_mod_id_resolver_task->abort();
- // FIXME: This should be 'emitAborted()', but InstanceStaging doesn't connect to the abort signal yet...
if (aborted)
- emitFailed(tr("Aborted"));
+ emitAborted();
return aborted;
}
@@ -335,7 +334,7 @@ void PackInstallTask::install()
components->saveNow();
- instance.setName(m_instName);
+ instance.setName(name());
instance.setIconKey(m_instIcon);
instance.setManagedPack("modpacksch", QString::number(m_pack.id), m_pack.name, QString::number(m_version.id), m_version.name);
instanceSettings->resumeSave();
diff --git a/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp
new file mode 100644
index 00000000..ddeea224
--- /dev/null
+++ b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp
@@ -0,0 +1,407 @@
+#include "ModrinthInstanceCreationTask.h"
+
+#include "Application.h"
+#include "FileSystem.h"
+#include "InstanceList.h"
+#include "Json.h"
+
+#include "minecraft/PackProfile.h"
+
+#include "modplatform/helpers/OverrideUtils.h"
+
+#include "net/ChecksumValidator.h"
+
+#include "settings/INISettingsObject.h"
+
+#include "ui/dialogs/CustomMessageBox.h"
+
+#include <QAbstractButton>
+
+bool ModrinthCreationTask::abort()
+{
+ if (!canAbort())
+ return false;
+
+ m_abort = true;
+ if (m_files_job)
+ m_files_job->abort();
+ return Task::abort();
+}
+
+bool ModrinthCreationTask::updateInstance()
+{
+ auto instance_list = APPLICATION->instances();
+
+ // FIXME: How to handle situations when there's more than one install already for a given modpack?
+ auto inst = instance_list->getInstanceByManagedName(originalName());
+
+ if (!inst) {
+ inst = instance_list->getInstanceById(originalName());
+
+ if (!inst)
+ return false;
+ }
+
+ QString index_path = FS::PathCombine(m_stagingPath, "modrinth.index.json");
+ if (!parseManifest(index_path, m_files, true, false))
+ return false;
+
+ auto version_name = inst->getManagedPackVersionName();
+ auto version_str = !version_name.isEmpty() ? tr(" (version %1)").arg(version_name) : "";
+
+ auto info = CustomMessageBox::selectable(
+ m_parent, tr("Similar modpack was found!"),
+ tr("One or more of your instances are from this same modpack%1. Do you want to create a "
+ "separate instance, or update the existing one?\n\nNOTE: Make sure you made a backup of your important instance data before "
+ "updating, as worlds can be corrupted and some configuration may be lost (due to pack overrides).")
+ .arg(version_str),
+ QMessageBox::Information, QMessageBox::Ok | QMessageBox::Reset | QMessageBox::Abort);
+ info->setButtonText(QMessageBox::Ok, tr("Create new instance"));
+ info->setButtonText(QMessageBox::Abort, tr("Update existing instance"));
+ info->setButtonText(QMessageBox::Reset, tr("Cancel"));
+
+ info->exec();
+
+ if (info->clickedButton() == info->button(QMessageBox::Ok))
+ return false;
+
+ if (info->clickedButton() == info->button(QMessageBox::Reset)) {
+ m_abort = true;
+ return false;
+ }
+
+ // Remove repeated files, we don't need to download them!
+ QDir old_inst_dir(inst->instanceRoot());
+
+ QString old_index_folder(FS::PathCombine(old_inst_dir.absolutePath(), "mrpack"));
+
+ QString old_index_path(FS::PathCombine(old_index_folder, "modrinth.index.json"));
+ QFileInfo old_index_file(old_index_path);
+ if (old_index_file.exists()) {
+ std::vector<Modrinth::File> old_files;
+ parseManifest(old_index_path, old_files, false, false);
+
+ // Let's remove all duplicated, identical resources!
+ auto files_iterator = m_files.begin();
+ begin:
+ while (files_iterator != m_files.end()) {
+ auto const& file = *files_iterator;
+
+ auto old_files_iterator = old_files.begin();
+ while (old_files_iterator != old_files.end()) {
+ auto const& old_file = *old_files_iterator;
+
+ if (old_file.hash == file.hash) {
+ qDebug() << "Removed file at" << file.path << "from list of downloads";
+ files_iterator = m_files.erase(files_iterator);
+ old_files_iterator = old_files.erase(old_files_iterator);
+ goto begin; // Sorry :c
+ }
+
+ old_files_iterator++;
+ }
+
+ files_iterator++;
+ }
+
+ QDir old_minecraft_dir(inst->gameRoot());
+
+ // Some files were removed from the old version, and some will be downloaded in an updated version,
+ // so we're fine removing them!
+ if (!old_files.empty()) {
+ for (auto const& file : old_files) {
+ if (file.path.isEmpty())
+ continue;
+ qDebug() << "Scheduling" << file.path << "for removal";
+ m_files_to_remove.append(old_minecraft_dir.absoluteFilePath(file.path));
+ }
+ }
+
+ // We will remove all the previous overrides, to prevent duplicate files!
+ // TODO: Currently 'overrides' will always override the stuff on update. How do we preserve unchanged overrides?
+ // FIXME: We may want to do something about disabled mods.
+ auto old_overrides = Override::readOverrides("overrides", old_index_folder);
+ for (const auto& entry : old_overrides) {
+ if (entry.isEmpty())
+ continue;
+ qDebug() << "Scheduling" << entry << "for removal";
+ m_files_to_remove.append(old_minecraft_dir.absoluteFilePath(entry));
+ }
+
+ auto old_client_overrides = Override::readOverrides("client-overrides", old_index_folder);
+ for (const auto& entry : old_overrides) {
+ if (entry.isEmpty())
+ continue;
+ qDebug() << "Scheduling" << entry << "for removal";
+ m_files_to_remove.append(old_minecraft_dir.absoluteFilePath(entry));
+ }
+ } else {
+ // We don't have an old index file, so we may duplicate stuff!
+ auto dialog = CustomMessageBox::selectable(m_parent,
+ tr("No index file."),
+ tr("We couldn't find a suitable index file for the older version. This may cause some of the files to be duplicated. Do you want to continue?"),
+ QMessageBox::Warning, QMessageBox::Ok | QMessageBox::Cancel);
+
+ if (dialog->exec() == QDialog::DialogCode::Rejected) {
+ m_abort = true;
+ return false;
+ }
+ }
+
+
+ setOverride(true);
+ qDebug() << "Will override instance!";
+
+ m_instance = inst;
+
+ // We let it go through the createInstance() stage, just with a couple modifications for updating
+ return false;
+}
+
+// https://docs.modrinth.com/docs/modpacks/format_definition/
+bool ModrinthCreationTask::createInstance()
+{
+ QEventLoop loop;
+
+ QString parent_folder(FS::PathCombine(m_stagingPath, "mrpack"));
+
+ QString index_path = FS::PathCombine(m_stagingPath, "modrinth.index.json");
+ if (m_files.empty() && !parseManifest(index_path, m_files, true, true))
+ return false;
+
+ // Keep index file in case we need it some other time (like when changing versions)
+ QString new_index_place(FS::PathCombine(parent_folder, "modrinth.index.json"));
+ FS::ensureFilePathExists(new_index_place);
+ QFile::rename(index_path, new_index_place);
+
+ auto mcPath = FS::PathCombine(m_stagingPath, ".minecraft");
+
+ auto override_path = FS::PathCombine(m_stagingPath, "overrides");
+ if (QFile::exists(override_path)) {
+ // Create a list of overrides in "overrides.txt" inside mrpack/
+ Override::createOverrides("overrides", parent_folder, override_path);
+
+ // Apply the overrides
+ if (!QFile::rename(override_path, mcPath)) {
+ setError(tr("Could not rename the overrides folder:\n") + "overrides");
+ return false;
+ }
+ }
+
+ // Do client overrides
+ auto client_override_path = FS::PathCombine(m_stagingPath, "client-overrides");
+ if (QFile::exists(client_override_path)) {
+ // Create a list of overrides in "client-overrides.txt" inside mrpack/
+ Override::createOverrides("client-overrides", parent_folder, client_override_path);
+
+ // Apply the overrides
+ if (!FS::overrideFolder(mcPath, client_override_path)) {
+ setError(tr("Could not rename the client overrides folder:\n") + "client overrides");
+ return false;
+ }
+ }
+
+ QString configPath = FS::PathCombine(m_stagingPath, "instance.cfg");
+ auto instanceSettings = std::make_shared<INISettingsObject>(configPath);
+ MinecraftInstance instance(m_globalSettings, instanceSettings, m_stagingPath);
+
+ auto components = instance.getPackProfile();
+ components->buildingFromScratch();
+ components->setComponentVersion("net.minecraft", minecraftVersion, true);
+
+ if (!fabricVersion.isEmpty())
+ components->setComponentVersion("net.fabricmc.fabric-loader", fabricVersion);
+ if (!quiltVersion.isEmpty())
+ components->setComponentVersion("org.quiltmc.quilt-loader", quiltVersion);
+ if (!forgeVersion.isEmpty())
+ components->setComponentVersion("net.minecraftforge", forgeVersion);
+
+ if (m_instIcon != "default") {
+ instance.setIconKey(m_instIcon);
+ } else {
+ instance.setIconKey("modrinth");
+ }
+
+ instance.setManagedPack("modrinth", getManagedPackID(), m_managed_name, m_managed_version_id, version());
+ instance.setName(name());
+ instance.saveNow();
+
+ m_files_job = new NetJob(tr("Mod download"), APPLICATION->network());
+
+ for (auto file : m_files) {
+ auto path = FS::PathCombine(m_stagingPath, ".minecraft", file.path);
+ qDebug() << "Will try to download" << file.downloads.front() << "to" << path;
+ auto dl = Net::Download::makeFile(file.downloads.dequeue(), path);
+ dl->addValidator(new Net::ChecksumValidator(file.hashAlgorithm, file.hash));
+ m_files_job->addNetAction(dl);
+
+ if (!file.downloads.empty()) {
+ // FIXME: This really needs to be put into a ConcurrentTask of
+ // MultipleOptionsTask's , once those exist :)
+ auto param = dl.toWeakRef();
+ connect(dl.get(), &NetAction::failed, [this, &file, path, param] {
+ auto ndl = Net::Download::makeFile(file.downloads.dequeue(), path);
+ ndl->addValidator(new Net::ChecksumValidator(file.hashAlgorithm, file.hash));
+ m_files_job->addNetAction(ndl);
+ if (auto shared = param.lock()) shared->succeeded();
+ });
+ }
+ }
+
+ bool ended_well = false;
+
+ connect(m_files_job.get(), &NetJob::succeeded, this, [&]() { ended_well = true; });
+ connect(m_files_job.get(), &NetJob::failed, [&](const QString& reason) {
+ ended_well = false;
+ setError(reason);
+ });
+ connect(m_files_job.get(), &NetJob::finished, &loop, &QEventLoop::quit);
+ connect(m_files_job.get(), &NetJob::progress, [&](qint64 current, qint64 total) { setProgress(current, total); });
+
+ setStatus(tr("Downloading mods..."));
+ m_files_job->start();
+
+ loop.exec();
+
+ // Update information of the already installed instance, if any.
+ if (m_instance && ended_well) {
+ setAbortable(false);
+ auto inst = m_instance.value();
+
+ // Only change the name if it didn't use a custom name, so that the previous custom name
+ // is preserved, but if we're using the original one, we update the version string.
+ // NOTE: This needs to come before the copyManagedPack call!
+ if (inst->name().contains(inst->getManagedPackVersionName())) {
+ if (askForChangingInstanceName(m_parent, inst->name(), instance.name()) == InstanceNameChange::ShouldChange)
+ inst->setName(instance.name());
+ }
+
+ inst->copyManagedPack(instance);
+ }
+
+ return ended_well;
+}
+
+bool ModrinthCreationTask::parseManifest(const QString& index_path, std::vector<Modrinth::File>& files, bool set_managed_info, bool show_optional_dialog)
+{
+ try {
+ auto doc = Json::requireDocument(index_path);
+ auto obj = Json::requireObject(doc, "modrinth.index.json");
+ int formatVersion = Json::requireInteger(obj, "formatVersion", "modrinth.index.json");
+ if (formatVersion == 1) {
+ auto game = Json::requireString(obj, "game", "modrinth.index.json");
+ if (game != "minecraft") {
+ throw JSONValidationError("Unknown game: " + game);
+ }
+
+ if (set_managed_info) {
+ m_managed_version_id = Json::ensureString(obj, "versionId", {}, "Managed ID");
+ m_managed_name = Json::ensureString(obj, "name", {}, "Managed Name");
+ }
+
+ auto jsonFiles = Json::requireIsArrayOf<QJsonObject>(obj, "files", "modrinth.index.json");
+ bool had_optional = false;
+ for (const auto& modInfo : jsonFiles) {
+ Modrinth::File file;
+ file.path = Json::requireString(modInfo, "path");
+
+ auto env = Json::ensureObject(modInfo, "env");
+ // 'env' field is optional
+ if (!env.isEmpty()) {
+ QString support = Json::ensureString(env, "client", "unsupported");
+ if (support == "unsupported") {
+ continue;
+ } else if (support == "optional") {
+ // TODO: Make a review dialog for choosing which ones the user wants!
+ if (!had_optional && show_optional_dialog) {
+ had_optional = true;
+ auto info = CustomMessageBox::selectable(
+ m_parent, tr("Optional mod detected!"),
+ tr("One or more mods from this modpack are optional. They will be downloaded, but disabled by default!"),
+ QMessageBox::Information);
+ info->exec();
+ }
+
+ if (file.path.endsWith(".jar"))
+ file.path += ".disabled";
+ }
+ }
+
+ QJsonObject hashes = Json::requireObject(modInfo, "hashes");
+ QString hash;
+ QCryptographicHash::Algorithm hashAlgorithm;
+ hash = Json::ensureString(hashes, "sha1");
+ hashAlgorithm = QCryptographicHash::Sha1;
+ if (hash.isEmpty()) {
+ hash = Json::ensureString(hashes, "sha512");
+ hashAlgorithm = QCryptographicHash::Sha512;
+ if (hash.isEmpty()) {
+ hash = Json::ensureString(hashes, "sha256");
+ hashAlgorithm = QCryptographicHash::Sha256;
+ if (hash.isEmpty()) {
+ throw JSONValidationError("No hash found for: " + file.path);
+ }
+ }
+ }
+ file.hash = QByteArray::fromHex(hash.toLatin1());
+ file.hashAlgorithm = hashAlgorithm;
+
+ // Do not use requireUrl, which uses StrictMode, instead use QUrl's default TolerantMode
+ // (as Modrinth seems to incorrectly handle spaces)
+
+ auto download_arr = Json::ensureArray(modInfo, "downloads");
+ for (auto download : download_arr) {
+ qWarning() << download.toString();
+ bool is_last = download.toString() == download_arr.last().toString();
+
+ auto download_url = QUrl(download.toString());
+
+ if (!download_url.isValid()) {
+ qDebug()
+ << QString("Download URL (%1) for %2 is not a correctly formatted URL").arg(download_url.toString(), file.path);
+ if (is_last && file.downloads.isEmpty())
+ throw JSONValidationError(tr("Download URL for %1 is not a correctly formatted URL").arg(file.path));
+ } else {
+ file.downloads.push_back(download_url);
+ }
+ }
+
+ files.push_back(file);
+ }
+
+ auto dependencies = Json::requireObject(obj, "dependencies", "modrinth.index.json");
+ for (auto it = dependencies.begin(), end = dependencies.end(); it != end; ++it) {
+ QString name = it.key();
+ if (name == "minecraft") {
+ minecraftVersion = Json::requireString(*it, "Minecraft version");
+ } else if (name == "fabric-loader") {
+ fabricVersion = Json::requireString(*it, "Fabric Loader version");
+ } else if (name == "quilt-loader") {
+ quiltVersion = Json::requireString(*it, "Quilt Loader version");
+ } else if (name == "forge") {
+ forgeVersion = Json::requireString(*it, "Forge version");
+ } else {
+ throw JSONValidationError("Unknown dependency type: " + name);
+ }
+ }
+ } else {
+ throw JSONValidationError(QStringLiteral("Unknown format version: %s").arg(formatVersion));
+ }
+
+ } catch (const JSONValidationError& e) {
+ setError(tr("Could not understand pack index:\n") + e.cause());
+ return false;
+ }
+
+ return true;
+}
+
+QString ModrinthCreationTask::getManagedPackID() const
+{
+ if (!m_source_url.isEmpty()) {
+ QRegularExpression regex(R"(data\/(.*)\/versions)");
+ return regex.match(m_source_url).captured(1);
+ }
+
+ return {};
+}
diff --git a/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.h b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.h
new file mode 100644
index 00000000..e459aadf
--- /dev/null
+++ b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.h
@@ -0,0 +1,44 @@
+#pragma once
+
+#include "InstanceCreationTask.h"
+
+#include <optional>
+
+#include "minecraft/MinecraftInstance.h"
+
+#include "modplatform/modrinth/ModrinthPackManifest.h"
+
+#include "net/NetJob.h"
+
+class ModrinthCreationTask final : public InstanceCreationTask {
+ Q_OBJECT
+
+ public:
+ ModrinthCreationTask(QString staging_path, SettingsObjectPtr global_settings, QWidget* parent, QString source_url = {})
+ : InstanceCreationTask(), m_parent(parent), m_source_url(std::move(source_url))
+ {
+ setStagingPath(staging_path);
+ setParentSettings(global_settings);
+ }
+
+ bool abort() override;
+
+ bool updateInstance() override;
+ bool createInstance() override;
+
+ private:
+ bool parseManifest(const QString&, std::vector<Modrinth::File>&, bool set_managed_info = true, bool show_optional_dialog = true);
+ QString getManagedPackID() const;
+
+ private:
+ QWidget* m_parent = nullptr;
+
+ QString minecraftVersion, fabricVersion, quiltVersion, forgeVersion;
+ QString m_managed_id, m_managed_version_id, m_managed_name;
+ QString m_source_url;
+
+ std::vector<Modrinth::File> m_files;
+ NetJob::Ptr m_files_job;
+
+ std::optional<InstancePtr> m_instance;
+};
diff --git a/launcher/modplatform/packwiz/Packwiz.cpp b/launcher/modplatform/packwiz/Packwiz.cpp
index c3561093..b1fe963e 100644
--- a/launcher/modplatform/packwiz/Packwiz.cpp
+++ b/launcher/modplatform/packwiz/Packwiz.cpp
@@ -1,20 +1,20 @@
// SPDX-License-Identifier: GPL-3.0-only
/*
-* PolyMC - Minecraft Launcher
-* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
-*
-* This program is free software: you can redistribute it and/or modify
-* it under the terms of the GNU General Public License as published by
-* the Free Software Foundation, version 3.
-*
-* This program is distributed in the hope that it will be useful,
-* but WITHOUT ANY WARRANTY; without even the implied warranty of
-* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-* GNU General Public License for more details.
-*
-* You should have received a copy of the GNU General Public License
-* along with this program. If not, see <https://www.gnu.org/licenses/>.
-*/
+ * PolyMC - Minecraft Launcher
+ * Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
#include "Packwiz.h"
@@ -22,9 +22,7 @@
#include <QDir>
#include <QObject>
-#include "toml.h"
-#include "FileSystem.h"
-
+#include <toml++/toml.h>
#include "minecraft/mod/Mod.h"
#include "modplatform/ModIndex.h"
@@ -44,7 +42,7 @@ auto getRealIndexName(QDir& index_dir, QString normalized_fname, bool should_fin
}
}
- if(should_find_match && !QString::compare(normalized_fname, real_fname, Qt::CaseSensitive)){
+ if (should_find_match && !QString::compare(normalized_fname, real_fname, Qt::CaseSensitive)) {
qCritical() << "Could not find a match for a valid metadata file!";
qCritical() << "File: " << normalized_fname;
return {};
@@ -57,7 +55,7 @@ auto getRealIndexName(QDir& index_dir, QString normalized_fname, bool should_fin
// Helpers
static inline auto indexFileName(QString const& mod_slug) -> QString
{
- if(mod_slug.endsWith(".pw.toml"))
+ if (mod_slug.endsWith(".pw.toml"))
return mod_slug;
return QString("%1.pw.toml").arg(mod_slug);
}
@@ -65,32 +63,28 @@ static inline auto indexFileName(QString const& mod_slug) -> QString
static ModPlatform::ProviderCapabilities ProviderCaps;
// Helper functions for extracting data from the TOML file
-auto stringEntry(toml_table_t* parent, const char* entry_name) -> QString
+auto stringEntry(toml::table table, const std::string entry_name) -> QString
{
- toml_datum_t var = toml_string_in(parent, entry_name);
- if (!var.ok) {
- qCritical() << QString("Failed to read str property '%1' in mod metadata.").arg(entry_name);
+ auto node = table[entry_name];
+ if (!node) {
+ qCritical() << QString::fromStdString("Failed to read str property '" + entry_name + "' in mod metadata.");
return {};
}
- QString tmp = var.u.s;
- free(var.u.s);
-
- return tmp;
+ return QString::fromStdString(node.value_or(""));
}
-auto intEntry(toml_table_t* parent, const char* entry_name) -> int
+auto intEntry(toml::table table, const std::string entry_name) -> int
{
- toml_datum_t var = toml_int_in(parent, entry_name);
- if (!var.ok) {
- qCritical() << QString("Failed to read int property '%1' in mod metadata.").arg(entry_name);
+ auto node = table[entry_name];
+ if (!node) {
+ qCritical() << QString::fromStdString("Failed to read int property '" + entry_name + "' in mod metadata.");
return {};
}
- return var.u.i;
+ return node.value_or(0);
}
-
auto V1::createModFormat(QDir& index_dir, ModPlatform::IndexedPack& mod_pack, ModPlatform::IndexedVersion& mod_version) -> Mod
{
Mod mod;
@@ -99,10 +93,9 @@ auto V1::createModFormat(QDir& index_dir, ModPlatform::IndexedPack& mod_pack, Mo
mod.name = mod_pack.name;
mod.filename = mod_version.fileName;
- if(mod_pack.provider == ModPlatform::Provider::FLAME){
+ if (mod_pack.provider == ModPlatform::Provider::FLAME) {
mod.mode = "metadata:curseforge";
- }
- else {
+ } else {
mod.mode = "url";
mod.url = mod_version.downloadUrl;
}
@@ -120,8 +113,8 @@ auto V1::createModFormat(QDir& index_dir, ModPlatform::IndexedPack& mod_pack, Mo
auto V1::createModFormat(QDir& index_dir, ::Mod& internal_mod, QString slug) -> Mod
{
// Try getting metadata if it exists
- Mod mod { getIndexForMod(index_dir, slug) };
- if(mod.isValid())
+ Mod mod{ getIndexForMod(index_dir, slug) };
+ if (mod.isValid())
return mod;
qWarning() << QString("Tried to create mod metadata with a Mod without metadata!");
@@ -131,7 +124,7 @@ auto V1::createModFormat(QDir& index_dir, ::Mod& internal_mod, QString slug) ->
void V1::updateModIndex(QDir& index_dir, Mod& mod)
{
- if(!mod.isValid()){
+ if (!mod.isValid()) {
qCritical() << QString("Tried to update metadata of an invalid mod!");
return;
}
@@ -150,7 +143,9 @@ void V1::updateModIndex(QDir& index_dir, Mod& mod)
// TODO: We should do more stuff here, as the user is likely trying to
// override a file. In this case, check versions and ask the user what
// they want to do!
- if (index_file.exists()) { index_file.remove(); }
+ if (index_file.exists()) {
+ index_file.remove();
+ }
if (!index_file.open(QIODevice::ReadWrite)) {
qCritical() << QString("Could not open file %1!").arg(indexFileName(mod.name));
@@ -174,15 +169,15 @@ void V1::updateModIndex(QDir& index_dir, Mod& mod)
in_stream << QString("\n[update]\n");
in_stream << QString("[update.%1]\n").arg(ProviderCaps.name(mod.provider));
- switch(mod.provider){
- case(ModPlatform::Provider::FLAME):
- in_stream << QString("file-id = %1\n").arg(mod.file_id.toString());
- in_stream << QString("project-id = %1\n").arg(mod.project_id.toString());
- break;
- case(ModPlatform::Provider::MODRINTH):
- addToStream("mod-id", mod.mod_id().toString());
- addToStream("version", mod.version().toString());
- break;
+ switch (mod.provider) {
+ case (ModPlatform::Provider::FLAME):
+ in_stream << QString("file-id = %1\n").arg(mod.file_id.toString());
+ in_stream << QString("project-id = %1\n").arg(mod.project_id.toString());
+ break;
+ case (ModPlatform::Provider::MODRINTH):
+ addToStream("mod-id", mod.mod_id().toString());
+ addToStream("version", mod.version().toString());
+ break;
}
}
@@ -230,27 +225,25 @@ auto V1::getIndexForMod(QDir& index_dir, QString slug) -> Mod
if (real_fname.isEmpty())
return {};
- QFile index_file(index_dir.absoluteFilePath(real_fname));
-
- if (!index_file.open(QIODevice::ReadOnly)) {
- qWarning() << QString("Failed to open mod metadata for %1").arg(slug);
+ toml::table table;
+#if TOML_EXCEPTIONS
+ try {
+ table = toml::parse_file(index_dir.absoluteFilePath(real_fname).toStdString());
+ } catch (const toml::parse_error& err) {
+ qWarning() << QString("Could not open file %1!").arg(normalized_fname);
+ qWarning() << "Reason: " << QString(err.what());
return {};
}
-
- toml_table_t* table = nullptr;
-
- // NOLINTNEXTLINE(modernize-avoid-c-arrays)
- char errbuf[200];
- auto file_bytearray = index_file.readAll();
- table = toml_parse(file_bytearray.data(), errbuf, sizeof(errbuf));
-
- index_file.close();
-
+#else
+ table = toml::parse_file(index_dir.absoluteFilePath(real_fname).toStdString());
if (!table) {
qWarning() << QString("Could not open file %1!").arg(normalized_fname);
- qWarning() << "Reason: " << QString(errbuf);
+ qWarning() << "Reason: " << QString(table.error().what());
return {};
}
+#endif
+
+ // index_file.close();
mod.slug = slug;
@@ -261,45 +254,42 @@ auto V1::getIndexForMod(QDir& index_dir, QString slug) -> Mod
}
{ // [download] info
- toml_table_t* download_table = toml_table_in(table, "download");
+ auto download_table = table["download"].as_table();
if (!download_table) {
qCritical() << QString("No [download] section found on mod metadata!");
return {};
}
- mod.mode = stringEntry(download_table, "mode");
- mod.url = stringEntry(download_table, "url");
- mod.hash_format = stringEntry(download_table, "hash-format");
- mod.hash = stringEntry(download_table, "hash");
+ mod.mode = stringEntry(*download_table, "mode");
+ mod.url = stringEntry(*download_table, "url");
+ mod.hash_format = stringEntry(*download_table, "hash-format");
+ mod.hash = stringEntry(*download_table, "hash");
}
- { // [update] info
+ { // [update] info
using Provider = ModPlatform::Provider;
- toml_table_t* update_table = toml_table_in(table, "update");
- if (!update_table) {
+ auto update_table = table["update"];
+ if (!update_table || !update_table.is_table()) {
qCritical() << QString("No [update] section found on mod metadata!");
return {};
}
- toml_table_t* mod_provider_table = nullptr;
- if ((mod_provider_table = toml_table_in(update_table, ProviderCaps.name(Provider::FLAME)))) {
+ toml::table* mod_provider_table = nullptr;
+ if ((mod_provider_table = update_table[ProviderCaps.name(Provider::FLAME)].as_table())) {
mod.provider = Provider::FLAME;
- mod.file_id = intEntry(mod_provider_table, "file-id");
- mod.project_id = intEntry(mod_provider_table, "project-id");
- } else if ((mod_provider_table = toml_table_in(update_table, ProviderCaps.name(Provider::MODRINTH)))) {
+ mod.file_id = intEntry(*mod_provider_table, "file-id");
+ mod.project_id = intEntry(*mod_provider_table, "project-id");
+ } else if ((mod_provider_table = update_table[ProviderCaps.name(Provider::MODRINTH)].as_table())) {
mod.provider = Provider::MODRINTH;
- mod.mod_id() = stringEntry(mod_provider_table, "mod-id");
- mod.version() = stringEntry(mod_provider_table, "version");
+ mod.mod_id() = stringEntry(*mod_provider_table, "mod-id");
+ mod.version() = stringEntry(*mod_provider_table, "version");
} else {
qCritical() << QString("No mod provider on mod metadata!");
return {};
}
-
}
- toml_free(table);
-
return mod;
}
diff --git a/launcher/modplatform/technic/SingleZipPackInstallTask.cpp b/launcher/modplatform/technic/SingleZipPackInstallTask.cpp
index 9093b245..6438d9ef 100644
--- a/launcher/modplatform/technic/SingleZipPackInstallTask.cpp
+++ b/launcher/modplatform/technic/SingleZipPackInstallTask.cpp
@@ -133,7 +133,7 @@ void Technic::SingleZipPackInstallTask::extractFinished()
shared_qobject_ptr<Technic::TechnicPackProcessor> packProcessor = new Technic::TechnicPackProcessor();
connect(packProcessor.get(), &Technic::TechnicPackProcessor::succeeded, this, &Technic::SingleZipPackInstallTask::emitSucceeded);
connect(packProcessor.get(), &Technic::TechnicPackProcessor::failed, this, &Technic::SingleZipPackInstallTask::emitFailed);
- packProcessor->run(m_globalSettings, m_instName, m_instIcon, m_stagingPath, m_minecraftVersion);
+ packProcessor->run(m_globalSettings, name(), m_instIcon, m_stagingPath, m_minecraftVersion);
}
void Technic::SingleZipPackInstallTask::extractAborted()
diff --git a/launcher/modplatform/technic/SolderPackInstallTask.cpp b/launcher/modplatform/technic/SolderPackInstallTask.cpp
index 89dbf4ca..19731b38 100644
--- a/launcher/modplatform/technic/SolderPackInstallTask.cpp
+++ b/launcher/modplatform/technic/SolderPackInstallTask.cpp
@@ -77,6 +77,7 @@ void Technic::SolderPackInstallTask::executeTask()
auto job = m_filesNetJob.get();
connect(job, &NetJob::succeeded, this, &Technic::SolderPackInstallTask::fileListSucceeded);
connect(job, &NetJob::failed, this, &Technic::SolderPackInstallTask::downloadFailed);
+ connect(job, &NetJob::aborted, this, &Technic::SolderPackInstallTask::downloadAborted);
m_filesNetJob->start();
}
@@ -127,6 +128,7 @@ void Technic::SolderPackInstallTask::fileListSucceeded()
connect(m_filesNetJob.get(), &NetJob::succeeded, this, &Technic::SolderPackInstallTask::downloadSucceeded);
connect(m_filesNetJob.get(), &NetJob::progress, this, &Technic::SolderPackInstallTask::downloadProgressChanged);
connect(m_filesNetJob.get(), &NetJob::failed, this, &Technic::SolderPackInstallTask::downloadFailed);
+ connect(m_filesNetJob.get(), &NetJob::aborted, this, &Technic::SolderPackInstallTask::downloadAborted);
m_filesNetJob->start();
}
@@ -171,6 +173,12 @@ void Technic::SolderPackInstallTask::downloadProgressChanged(qint64 current, qin
setProgress(current / 2, total);
}
+void Technic::SolderPackInstallTask::downloadAborted()
+{
+ emitAborted();
+ m_filesNetJob.reset();
+}
+
void Technic::SolderPackInstallTask::extractFinished()
{
if (!m_extractFuture.result())
@@ -214,7 +222,7 @@ void Technic::SolderPackInstallTask::extractFinished()
shared_qobject_ptr<Technic::TechnicPackProcessor> packProcessor = new Technic::TechnicPackProcessor();
connect(packProcessor.get(), &Technic::TechnicPackProcessor::succeeded, this, &Technic::SolderPackInstallTask::emitSucceeded);
connect(packProcessor.get(), &Technic::TechnicPackProcessor::failed, this, &Technic::SolderPackInstallTask::emitFailed);
- packProcessor->run(m_globalSettings, m_instName, m_instIcon, m_stagingPath, m_minecraftVersion, true);
+ packProcessor->run(m_globalSettings, name(), m_instIcon, m_stagingPath, m_minecraftVersion, true);
}
void Technic::SolderPackInstallTask::extractAborted()
diff --git a/launcher/modplatform/technic/SolderPackInstallTask.h b/launcher/modplatform/technic/SolderPackInstallTask.h
index 117a7bd6..aa14ce88 100644
--- a/launcher/modplatform/technic/SolderPackInstallTask.h
+++ b/launcher/modplatform/technic/SolderPackInstallTask.h
@@ -61,6 +61,7 @@ namespace Technic
void downloadSucceeded();
void downloadFailed(QString reason);
void downloadProgressChanged(qint64 current, qint64 total);
+ void downloadAborted();
void extractFinished();
void extractAborted();