mirror of
https://github.com/PrismLauncher/PrismLauncher.git
synced 2025-04-29 22:24:26 +02:00
197 lines
7.6 KiB
C++
197 lines
7.6 KiB
C++
#include "ModrinthCheckUpdate.h"
|
|
#include "Application.h"
|
|
#include "ModrinthAPI.h"
|
|
#include "ModrinthPackIndex.h"
|
|
|
|
#include "Json.h"
|
|
|
|
#include "QObjectPtr.h"
|
|
#include "ResourceDownloadTask.h"
|
|
|
|
#include "modplatform/helpers/HashUtils.h"
|
|
|
|
#include "tasks/ConcurrentTask.h"
|
|
|
|
static ModrinthAPI api;
|
|
|
|
bool ModrinthCheckUpdate::abort()
|
|
{
|
|
if (m_job)
|
|
return m_job->abort();
|
|
return true;
|
|
}
|
|
|
|
/* Check for update:
|
|
* - Get latest version available
|
|
* - Compare hash of the latest version with the current hash
|
|
* - If equal, no updates, else, there's updates, so add to the list
|
|
* */
|
|
void ModrinthCheckUpdate::executeTask()
|
|
{
|
|
setStatus(tr("Preparing resources for Modrinth..."));
|
|
setProgress(0, (m_loaders_list.isEmpty() ? 1 : m_loaders_list.length()) * 2 + 1);
|
|
|
|
auto hashing_task =
|
|
makeShared<ConcurrentTask>("MakeModrinthHashesTask", APPLICATION->settings()->get("NumberOfConcurrentTasks").toInt());
|
|
for (auto* resource : m_resources) {
|
|
auto hash = resource->metadata()->hash;
|
|
|
|
// Sadly the API can only handle one hash type per call, se we
|
|
// need to generate a new hash if the current one is innadequate
|
|
// (though it will rarely happen, if at all)
|
|
if (resource->metadata()->hash_format != m_hash_type) {
|
|
auto hash_task = Hashing::createHasher(resource->fileinfo().absoluteFilePath(), ModPlatform::ResourceProvider::MODRINTH);
|
|
connect(hash_task.get(), &Hashing::Hasher::resultsReady, [this, resource](QString hash) { m_mappings.insert(hash, resource); });
|
|
connect(hash_task.get(), &Task::failed, [this] { failed("Failed to generate hash"); });
|
|
hashing_task->addTask(hash_task);
|
|
} else {
|
|
m_mappings.insert(hash, resource);
|
|
}
|
|
}
|
|
|
|
connect(hashing_task.get(), &Task::finished, this, &ModrinthCheckUpdate::checkNextLoader);
|
|
m_job = hashing_task;
|
|
hashing_task->start();
|
|
}
|
|
|
|
void ModrinthCheckUpdate::getUpdateModsForLoader(std::optional<ModPlatform::ModLoaderTypes> loader)
|
|
{
|
|
setStatus(tr("Waiting for the API response from Modrinth..."));
|
|
setProgress(m_progress + 1, m_progressTotal);
|
|
|
|
auto response = std::make_shared<QByteArray>();
|
|
QStringList hashes = m_mappings.keys();
|
|
auto job = api.latestVersions(hashes, m_hash_type, m_game_versions, loader, response);
|
|
|
|
connect(job.get(), &Task::succeeded, this, [this, response, loader] { checkVersionsResponse(response, loader); });
|
|
|
|
connect(job.get(), &Task::failed, this, &ModrinthCheckUpdate::checkNextLoader);
|
|
|
|
m_job = job;
|
|
job->start();
|
|
}
|
|
|
|
void ModrinthCheckUpdate::checkVersionsResponse(std::shared_ptr<QByteArray> response, std::optional<ModPlatform::ModLoaderTypes> loader)
|
|
{
|
|
setStatus(tr("Parsing the API response from Modrinth..."));
|
|
setProgress(m_progress + 1, m_progressTotal);
|
|
|
|
QJsonParseError parse_error{};
|
|
QJsonDocument doc = QJsonDocument::fromJson(*response, &parse_error);
|
|
if (parse_error.error != QJsonParseError::NoError) {
|
|
qWarning() << "Error while parsing JSON response from ModrinthCheckUpdate at " << parse_error.offset
|
|
<< " reason: " << parse_error.errorString();
|
|
qWarning() << *response;
|
|
|
|
emitFailed(parse_error.errorString());
|
|
return;
|
|
}
|
|
|
|
try {
|
|
auto iter = m_mappings.begin();
|
|
|
|
while (iter != m_mappings.end()) {
|
|
const QString hash = iter.key();
|
|
Resource* resource = iter.value();
|
|
|
|
auto project_obj = doc[hash].toObject();
|
|
|
|
// If the returned project is empty, but we have Modrinth metadata,
|
|
// it means this specific version is not available
|
|
if (project_obj.isEmpty()) {
|
|
qDebug() << "Mod " << m_mappings.find(hash).value()->name() << " got an empty response." << "Hash: " << hash;
|
|
++iter;
|
|
continue;
|
|
}
|
|
|
|
// Sometimes a version may have multiple files, one with "forge" and one with "fabric",
|
|
// so we may want to filter it
|
|
QString loader_filter;
|
|
static auto flags = { ModPlatform::ModLoaderType::NeoForge, ModPlatform::ModLoaderType::Forge,
|
|
ModPlatform::ModLoaderType::Quilt, ModPlatform::ModLoaderType::Fabric };
|
|
for (auto flag : flags) {
|
|
if (loader.has_value() && loader->testFlag(flag)) {
|
|
loader_filter = ModPlatform::getModLoaderAsString(flag);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Currently, we rely on a couple heuristics to determine whether an update is actually available or not:
|
|
// - The file needs to be preferred: It is either the primary file, or the one found via (explicit) usage of the
|
|
// loader_filter
|
|
// - The version reported by the JAR is different from the version reported by the indexed version (it's usually the case)
|
|
// Such is the pain of having arbitrary files for a given version .-.
|
|
|
|
auto project_ver = Modrinth::loadIndexedPackVersion(project_obj, m_hash_type, loader_filter);
|
|
if (project_ver.downloadUrl.isEmpty()) {
|
|
qCritical() << "Modrinth mod without download url!" << project_ver.fileName;
|
|
++iter;
|
|
continue;
|
|
}
|
|
|
|
// Fake pack with the necessary info to pass to the download task :)
|
|
auto pack = std::make_shared<ModPlatform::IndexedPack>();
|
|
pack->name = resource->name();
|
|
pack->slug = resource->metadata()->slug;
|
|
pack->addonId = resource->metadata()->project_id;
|
|
pack->provider = ModPlatform::ResourceProvider::MODRINTH;
|
|
if ((project_ver.hash != hash && project_ver.is_preferred) || (resource->status() == ResourceStatus::NOT_INSTALLED)) {
|
|
auto download_task = makeShared<ResourceDownloadTask>(pack, project_ver, m_resource_model);
|
|
|
|
QString old_version = resource->metadata()->version_number;
|
|
if (old_version.isEmpty()) {
|
|
if (resource->status() == ResourceStatus::NOT_INSTALLED)
|
|
old_version = tr("Not installed");
|
|
else
|
|
old_version = tr("Unknown");
|
|
}
|
|
|
|
m_updates.emplace_back(pack->name, hash, old_version, project_ver.version_number, project_ver.version_type,
|
|
project_ver.changelog, ModPlatform::ResourceProvider::MODRINTH, download_task, resource->enabled());
|
|
}
|
|
m_deps.append(std::make_shared<GetModDependenciesTask::PackDependency>(pack, project_ver));
|
|
|
|
iter = m_mappings.erase(iter);
|
|
}
|
|
} catch (Json::JsonException& e) {
|
|
emitFailed(e.cause() + ": " + e.what());
|
|
return;
|
|
}
|
|
checkNextLoader();
|
|
}
|
|
|
|
void ModrinthCheckUpdate::checkNextLoader()
|
|
{
|
|
if (m_mappings.isEmpty()) {
|
|
emitSucceeded();
|
|
return;
|
|
}
|
|
|
|
if (m_loaders_list.isEmpty() && m_loader_idx == 0) {
|
|
getUpdateModsForLoader({});
|
|
m_loader_idx++;
|
|
return;
|
|
}
|
|
|
|
if (m_loader_idx < m_loaders_list.size()) {
|
|
getUpdateModsForLoader(m_loaders_list.at(m_loader_idx));
|
|
m_loader_idx++;
|
|
return;
|
|
}
|
|
|
|
for (auto resource : m_mappings) {
|
|
QString reason;
|
|
|
|
if (dynamic_cast<Mod*>(resource) != nullptr)
|
|
reason =
|
|
tr("No valid version found for this resource. It's probably unavailable for the current game "
|
|
"version / mod loader.");
|
|
else
|
|
reason = tr("No valid version found for this resource. It's probably unavailable for the current game version.");
|
|
|
|
emit checkFailed(resource, reason);
|
|
}
|
|
|
|
emitSucceeded();
|
|
}
|