2019-06-27 03:20:11 +02:00
|
|
|
#include "UrlResolvingTask.h"
|
|
|
|
#include <QtXml>
|
2019-06-30 11:03:59 +02:00
|
|
|
#include <Json.h>
|
|
|
|
|
2019-06-27 03:20:11 +02:00
|
|
|
|
|
|
|
namespace {
|
2019-06-30 11:03:59 +02:00
|
|
|
const char * metabase = "https://cursemeta.dries007.net";
|
2019-06-27 03:20:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Flame::UrlResolvingTask::UrlResolvingTask(const QString& toProcess)
|
|
|
|
: m_url(toProcess)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Flame::UrlResolvingTask::executeTask()
|
2019-06-30 11:03:59 +02:00
|
|
|
{
|
|
|
|
resolveUrl();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Flame::UrlResolvingTask::resolveUrl()
|
2019-06-27 03:20:11 +02:00
|
|
|
{
|
|
|
|
setStatus(tr("Resolving URL..."));
|
|
|
|
setProgress(0, 1);
|
2019-07-09 22:04:52 +02:00
|
|
|
QUrl actualUrl(m_url);
|
|
|
|
if(actualUrl.host() != "www.curseforge.com") {
|
|
|
|
emitFailed(tr("Not a Twitch URL."));
|
|
|
|
return;
|
|
|
|
}
|
2019-06-27 03:20:11 +02:00
|
|
|
m_dljob.reset(new NetJob("URL resolver"));
|
|
|
|
|
2019-06-30 11:03:59 +02:00
|
|
|
bool weAreDigging = false;
|
2019-06-27 03:20:11 +02:00
|
|
|
needle = QString();
|
|
|
|
|
|
|
|
if(m_url.startsWith("https://")) {
|
|
|
|
if(m_url.endsWith("?client=y")) {
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download?client=y
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download/2697088?client=y
|
|
|
|
m_url.chop(9);
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download/2697088
|
|
|
|
}
|
|
|
|
if(m_url.endsWith("/download")) {
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download -> need to dig inside html...
|
|
|
|
weAreDigging = true;
|
|
|
|
needle = m_url;
|
|
|
|
needle.replace("https://", "twitch://");
|
|
|
|
needle.replace("/download", "/download-client/");
|
|
|
|
m_url.append("?client=y");
|
|
|
|
} else if (m_url.contains("/download/")) {
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download/2697088
|
|
|
|
m_url.replace("/download/", "/download-client/");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(m_url.startsWith("twitch://")) {
|
|
|
|
// twitch://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download-client/2697088
|
|
|
|
m_url.replace(0, 9, "https://");
|
|
|
|
// https://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download-client/2697088
|
|
|
|
}
|
|
|
|
auto dl = Net::Download::makeByteArray(QUrl(m_url), &results);
|
|
|
|
m_dljob->addNetAction(dl);
|
|
|
|
if(weAreDigging) {
|
2019-06-30 11:03:59 +02:00
|
|
|
connect(m_dljob.get(), &NetJob::finished, this, &Flame::UrlResolvingTask::processHTML);
|
2019-06-27 03:20:11 +02:00
|
|
|
} else {
|
2019-06-30 11:03:59 +02:00
|
|
|
connect(m_dljob.get(), &NetJob::finished, this, &Flame::UrlResolvingTask::processCCIP);
|
2019-06-27 03:20:11 +02:00
|
|
|
}
|
2019-06-30 11:03:59 +02:00
|
|
|
m_dljob->start();
|
2019-06-27 03:20:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Flame::UrlResolvingTask::processHTML()
|
|
|
|
{
|
|
|
|
QString htmlDoc = QString::fromUtf8(results);
|
|
|
|
auto index = htmlDoc.indexOf(needle);
|
|
|
|
if(index < 0) {
|
|
|
|
emitFailed(tr("Couldn't find the needle in the haystack..."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto indexStart = index;
|
|
|
|
int indexEnd = -1;
|
|
|
|
while((index + 1) < htmlDoc.size() && htmlDoc[index] != '"') {
|
|
|
|
index ++;
|
|
|
|
if(htmlDoc[index] == '"') {
|
|
|
|
indexEnd = index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(indexEnd > 0) {
|
|
|
|
QString found = htmlDoc.mid(indexStart, indexEnd - indexStart);
|
|
|
|
qDebug() << "Found needle: " << found;
|
|
|
|
// twitch://www.curseforge.com/minecraft/modpacks/ftb-sky-odyssey/download-client/2697088
|
|
|
|
m_url = found;
|
2019-06-30 11:03:59 +02:00
|
|
|
resolveUrl();
|
2019-06-27 03:20:11 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
emitFailed(tr("Couldn't find the end of the needle in the haystack..."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Flame::UrlResolvingTask::processCCIP()
|
|
|
|
{
|
|
|
|
QDomDocument doc;
|
|
|
|
if (!doc.setContent(results)) {
|
|
|
|
qDebug() << results;
|
|
|
|
emitFailed(tr("Resolving failed."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto packageNode = doc.namedItem("package");
|
|
|
|
if(!packageNode.isElement()) {
|
|
|
|
emitFailed(tr("Resolving failed: missing package root element."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto projectNode = packageNode.namedItem("project");
|
|
|
|
if(!projectNode.isElement()) {
|
|
|
|
emitFailed(tr("Resolving failed: missing project element."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto attribs = projectNode.attributes();
|
|
|
|
|
|
|
|
auto projectIdNode = attribs.namedItem("id");
|
|
|
|
if(!projectIdNode.isAttr()) {
|
|
|
|
emitFailed(tr("Resolving failed: missing id attribute."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto fileIdNode = attribs.namedItem("file");
|
|
|
|
if(!fileIdNode.isAttr()) {
|
|
|
|
emitFailed(tr("Resolving failed: missing file attribute."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto projectId = projectIdNode.nodeValue();
|
|
|
|
auto fileId = fileIdNode.nodeValue();
|
|
|
|
bool success = true;
|
|
|
|
m_result.projectId = projectId.toInt(&success);
|
|
|
|
if(!success) {
|
2019-09-25 22:31:09 +02:00
|
|
|
emitFailed(tr("Failed to resolve projectId as a number."));
|
2019-06-27 03:20:11 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_result.fileId = fileId.toInt(&success);
|
|
|
|
if(!success) {
|
2019-09-25 22:31:09 +02:00
|
|
|
emitFailed(tr("Failed to resolve fileId as a number."));
|
2019-06-27 03:20:11 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
qDebug() << "Resolved" << m_url << "as" << m_result.projectId << "/" << m_result.fileId;
|
2019-06-30 11:03:59 +02:00
|
|
|
resolveIDs();
|
2019-06-27 03:20:11 +02:00
|
|
|
}
|
|
|
|
|
2019-06-30 11:03:59 +02:00
|
|
|
void Flame::UrlResolvingTask::resolveIDs()
|
|
|
|
{
|
|
|
|
setStatus(tr("Resolving mod IDs..."));
|
|
|
|
m_dljob.reset(new NetJob("Mod id resolver"));
|
|
|
|
auto projectIdStr = QString::number(m_result.projectId);
|
|
|
|
auto fileIdStr = QString::number(m_result.fileId);
|
|
|
|
QString metaurl = QString("%1/%2/%3.json").arg(metabase, projectIdStr, fileIdStr);
|
|
|
|
auto dl = Net::Download::makeByteArray(QUrl(metaurl), &results);
|
|
|
|
m_dljob->addNetAction(dl);
|
|
|
|
connect(m_dljob.get(), &NetJob::finished, this, &Flame::UrlResolvingTask::processCursemeta);
|
|
|
|
m_dljob->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Flame::UrlResolvingTask::processCursemeta()
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
if(m_result.parseFromBytes(results)) {
|
|
|
|
emitSucceeded();
|
|
|
|
qDebug() << results;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} catch (const JSONValidationError &e) {
|
|
|
|
|
|
|
|
qCritical() << "Resolving of" << m_result.projectId << m_result.fileId << "failed because of a parsing error:";
|
|
|
|
qCritical() << e.cause();
|
|
|
|
qCritical() << "JSON:";
|
|
|
|
qCritical() << results;
|
|
|
|
}
|
|
|
|
emitFailed(tr("Failed to resolve the modpack file."));
|
|
|
|
}
|