2023-01-23 11:03:55 -03:00
|
|
|
// SPDX-FileCopyrightText: 2023 flowln <flowlnlnln@gmail.com>
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
|
2022-06-03 19:06:51 -03:00
|
|
|
#include "FlameAPI.h"
|
|
|
|
#include "FlameModIndex.h"
|
|
|
|
|
|
|
|
#include "Application.h"
|
|
|
|
#include "Json.h"
|
2023-08-14 18:16:53 +02:00
|
|
|
#include "net/ApiDownload.h"
|
2023-06-25 12:02:46 -07:00
|
|
|
#include "net/ApiUpload.h"
|
2023-01-24 14:37:40 +01:00
|
|
|
#include "net/NetJob.h"
|
2022-06-03 19:06:51 -03:00
|
|
|
#include "net/Upload.h"
|
|
|
|
|
2023-06-15 22:59:41 +03:00
|
|
|
Task::Ptr FlameAPI::matchFingerprints(const QList<uint>& fingerprints, std::shared_ptr<QByteArray> response)
|
2022-06-03 19:06:51 -03:00
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::MatchFingerprints"), APPLICATION->network());
|
2022-06-03 19:06:51 -03:00
|
|
|
|
|
|
|
QJsonObject body_obj;
|
|
|
|
QJsonArray fingerprints_arr;
|
|
|
|
for (auto& fp : fingerprints) {
|
|
|
|
fingerprints_arr.append(QString("%1").arg(fp));
|
|
|
|
}
|
|
|
|
|
|
|
|
body_obj["fingerprints"] = fingerprints_arr;
|
|
|
|
|
|
|
|
QJsonDocument body(body_obj);
|
|
|
|
auto body_raw = body.toJson();
|
|
|
|
|
2023-06-25 12:02:46 -07:00
|
|
|
netJob->addNetAction(Net::ApiUpload::makeByteArray(QString("https://api.curseforge.com/v1/fingerprints"), response, body_raw));
|
2022-06-03 19:06:51 -03:00
|
|
|
|
|
|
|
return netJob;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto FlameAPI::getModFileChangelog(int modId, int fileId) -> QString
|
|
|
|
{
|
|
|
|
QEventLoop lock;
|
|
|
|
QString changelog;
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::FileChangelog"), APPLICATION->network());
|
|
|
|
auto response = std::make_shared<QByteArray>();
|
2023-06-01 16:39:04 -07:00
|
|
|
netJob->addNetAction(Net::ApiDownload::makeByteArray(
|
2022-06-03 19:06:51 -03:00
|
|
|
QString("https://api.curseforge.com/v1/mods/%1/files/%2/changelog")
|
|
|
|
.arg(QString::fromStdString(std::to_string(modId)), QString::fromStdString(std::to_string(fileId))),
|
|
|
|
response));
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::succeeded, [&netJob, response, &changelog] {
|
2022-06-03 19:06:51 -03:00
|
|
|
QJsonParseError parse_error{};
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(*response, &parse_error);
|
|
|
|
if (parse_error.error != QJsonParseError::NoError) {
|
|
|
|
qWarning() << "Error while parsing JSON response from Flame::FileChangelog at " << parse_error.offset
|
|
|
|
<< " reason: " << parse_error.errorString();
|
|
|
|
qWarning() << *response;
|
|
|
|
|
|
|
|
netJob->failed(parse_error.errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
changelog = Json::ensureString(doc.object(), "data");
|
|
|
|
});
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::finished, [&lock] { lock.quit(); });
|
2022-06-03 19:06:51 -03:00
|
|
|
|
|
|
|
netJob->start();
|
|
|
|
lock.exec();
|
|
|
|
|
|
|
|
return changelog;
|
|
|
|
}
|
|
|
|
|
2022-07-19 11:50:38 -03:00
|
|
|
auto FlameAPI::getModDescription(int modId) -> QString
|
|
|
|
{
|
|
|
|
QEventLoop lock;
|
|
|
|
QString description;
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::ModDescription"), APPLICATION->network());
|
|
|
|
auto response = std::make_shared<QByteArray>();
|
2023-08-14 18:16:53 +02:00
|
|
|
netJob->addNetAction(Net::ApiDownload::makeByteArray(
|
|
|
|
QString("https://api.curseforge.com/v1/mods/%1/description").arg(QString::number(modId)), response));
|
2022-07-19 11:50:38 -03:00
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::succeeded, [&netJob, response, &description] {
|
2022-07-19 11:50:38 -03:00
|
|
|
QJsonParseError parse_error{};
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(*response, &parse_error);
|
|
|
|
if (parse_error.error != QJsonParseError::NoError) {
|
|
|
|
qWarning() << "Error while parsing JSON response from Flame::ModDescription at " << parse_error.offset
|
|
|
|
<< " reason: " << parse_error.errorString();
|
|
|
|
qWarning() << *response;
|
|
|
|
|
|
|
|
netJob->failed(parse_error.errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
description = Json::ensureString(doc.object(), "data");
|
|
|
|
});
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::finished, [&lock] { lock.quit(); });
|
2022-07-19 11:50:38 -03:00
|
|
|
|
|
|
|
netJob->start();
|
|
|
|
lock.exec();
|
|
|
|
|
|
|
|
return description;
|
|
|
|
}
|
|
|
|
|
2022-06-03 19:06:51 -03:00
|
|
|
auto FlameAPI::getLatestVersion(VersionSearchArgs&& args) -> ModPlatform::IndexedVersion
|
|
|
|
{
|
2022-11-25 09:23:46 -03:00
|
|
|
auto versions_url_optional = getVersionsURL(args);
|
|
|
|
if (!versions_url_optional.has_value())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto versions_url = versions_url_optional.value();
|
|
|
|
|
2022-06-03 19:06:51 -03:00
|
|
|
QEventLoop loop;
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::GetLatestVersion(%1)").arg(args.pack.name), APPLICATION->network());
|
|
|
|
auto response = std::make_shared<QByteArray>();
|
2022-06-03 19:06:51 -03:00
|
|
|
ModPlatform::IndexedVersion ver;
|
|
|
|
|
2023-06-25 12:36:27 -07:00
|
|
|
netJob->addNetAction(Net::ApiDownload::makeByteArray(versions_url, response));
|
2022-06-03 19:06:51 -03:00
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::succeeded, [response, args, &ver] {
|
2022-06-03 19:06:51 -03:00
|
|
|
QJsonParseError parse_error{};
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(*response, &parse_error);
|
|
|
|
if (parse_error.error != QJsonParseError::NoError) {
|
|
|
|
qWarning() << "Error while parsing JSON response from latest mod version at " << parse_error.offset
|
|
|
|
<< " reason: " << parse_error.errorString();
|
|
|
|
qWarning() << *response;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto obj = Json::requireObject(doc);
|
|
|
|
auto arr = Json::requireArray(obj, "data");
|
|
|
|
|
|
|
|
for (auto file : arr) {
|
|
|
|
auto file_obj = Json::requireObject(file);
|
|
|
|
auto file_tmp = FlameMod::loadIndexedPackVersion(file_obj);
|
2023-10-03 17:23:26 +03:00
|
|
|
if (file_tmp.date > ver.date && (!args.loaders.has_value() || !file_tmp.loaders || args.loaders.value() & file_tmp.loaders))
|
2023-08-23 12:52:51 +03:00
|
|
|
ver = file_tmp;
|
2022-06-03 19:06:51 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
} catch (Json::JsonException& e) {
|
|
|
|
qCritical() << "Failed to parse response from a version request.";
|
|
|
|
qCritical() << e.what();
|
|
|
|
qDebug() << doc;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-05-21 01:47:54 -07:00
|
|
|
QObject::connect(netJob.get(), &NetJob::finished, [&loop] { loop.quit(); });
|
2022-06-03 19:06:51 -03:00
|
|
|
|
|
|
|
netJob->start();
|
|
|
|
|
|
|
|
loop.exec();
|
|
|
|
|
|
|
|
return ver;
|
|
|
|
}
|
2022-06-19 14:29:21 -03:00
|
|
|
|
2023-06-15 22:59:41 +03:00
|
|
|
Task::Ptr FlameAPI::getProjects(QStringList addonIds, std::shared_ptr<QByteArray> response) const
|
2022-06-19 14:29:21 -03:00
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::GetProjects"), APPLICATION->network());
|
2022-06-19 14:29:21 -03:00
|
|
|
|
|
|
|
QJsonObject body_obj;
|
|
|
|
QJsonArray addons_arr;
|
|
|
|
for (auto& addonId : addonIds) {
|
|
|
|
addons_arr.append(addonId);
|
|
|
|
}
|
|
|
|
|
|
|
|
body_obj["modIds"] = addons_arr;
|
|
|
|
|
|
|
|
QJsonDocument body(body_obj);
|
|
|
|
auto body_raw = body.toJson();
|
|
|
|
|
2023-06-25 12:02:46 -07:00
|
|
|
netJob->addNetAction(Net::ApiUpload::makeByteArray(QString("https://api.curseforge.com/v1/mods"), response, body_raw));
|
2022-06-19 14:29:21 -03:00
|
|
|
|
2023-01-24 16:52:09 -03:00
|
|
|
QObject::connect(netJob.get(), &NetJob::failed, [body_raw] { qDebug() << body_raw; });
|
2022-06-19 14:29:21 -03:00
|
|
|
|
|
|
|
return netJob;
|
|
|
|
}
|
2022-07-21 16:40:28 -03:00
|
|
|
|
2023-06-15 22:59:41 +03:00
|
|
|
Task::Ptr FlameAPI::getFiles(const QStringList& fileIds, std::shared_ptr<QByteArray> response) const
|
2022-07-21 16:40:28 -03:00
|
|
|
{
|
2023-01-24 16:52:09 -03:00
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::GetFiles"), APPLICATION->network());
|
2022-07-21 16:40:28 -03:00
|
|
|
|
|
|
|
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();
|
|
|
|
|
2023-06-25 12:02:46 -07:00
|
|
|
netJob->addNetAction(Net::ApiUpload::makeByteArray(QString("https://api.curseforge.com/v1/mods/files"), response, body_raw));
|
2022-07-21 16:40:28 -03:00
|
|
|
|
2023-01-24 16:52:09 -03:00
|
|
|
QObject::connect(netJob.get(), &NetJob::failed, [body_raw] { qDebug() << body_raw; });
|
2022-07-21 16:40:28 -03:00
|
|
|
|
2023-05-15 16:34:33 -07:00
|
|
|
return netJob;
|
|
|
|
}
|
|
|
|
|
2023-08-17 00:23:53 +03:00
|
|
|
Task::Ptr FlameAPI::getFile(const QString& addonId, const QString& fileId, std::shared_ptr<QByteArray> response) const
|
2023-05-15 16:34:33 -07:00
|
|
|
{
|
|
|
|
auto netJob = makeShared<NetJob>(QString("Flame::GetFile"), APPLICATION->network());
|
|
|
|
netJob->addNetAction(
|
2023-08-12 11:11:58 +02:00
|
|
|
Net::ApiDownload::makeByteArray(QUrl(QString("https://api.curseforge.com/v1/mods/%1/files/%2").arg(addonId, fileId)), response));
|
2023-05-15 16:34:33 -07:00
|
|
|
|
|
|
|
QObject::connect(netJob.get(), &NetJob::failed, [addonId, fileId] { qDebug() << "Flame API file failure" << addonId << fileId; });
|
|
|
|
|
2022-07-21 16:40:28 -03:00
|
|
|
return netJob;
|
|
|
|
}
|
2022-12-20 12:15:17 -03:00
|
|
|
|
|
|
|
// https://docs.curseforge.com/?python#tocS_ModsSearchSortField
|
|
|
|
static QList<ResourceAPI::SortingMethod> s_sorts = { { 1, "Featured", QObject::tr("Sort by Featured") },
|
|
|
|
{ 2, "Popularity", QObject::tr("Sort by Popularity") },
|
|
|
|
{ 3, "LastUpdated", QObject::tr("Sort by Last Updated") },
|
|
|
|
{ 4, "Name", QObject::tr("Sort by Name") },
|
|
|
|
{ 5, "Author", QObject::tr("Sort by Author") },
|
|
|
|
{ 6, "TotalDownloads", QObject::tr("Sort by Downloads") },
|
|
|
|
{ 7, "Category", QObject::tr("Sort by Category") },
|
|
|
|
{ 8, "GameVersion", QObject::tr("Sort by Game Version") } };
|
|
|
|
|
|
|
|
QList<ResourceAPI::SortingMethod> FlameAPI::getSortingMethods() const
|
|
|
|
{
|
|
|
|
return s_sorts;
|
|
|
|
}
|