b6452215c1
feat: add details to mod pack downloading feat: add logging rule sloading form `ligging.ini at data path root feat: add `launcher.task` `launcher.task.net` and `launcher.task.net.[down|up]load` logging categories fix: add new subtask progress to the end of the lay out not the beginning (cuts down on flickering) Signed-off-by: Rachel Powers <508861+Ryex@users.noreply.github.com>
255 lines
9.4 KiB
C++
255 lines
9.4 KiB
C++
// SPDX-License-Identifier: GPL-3.0-only
|
|
/*
|
|
* PolyMC - Minecraft Launcher
|
|
* Copyright (C) 2022 Lenny McLennington <lenny@sneed.church>
|
|
* Copyright (C) 2022 Swirl <swurl@swurl.xyz>
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
|
*
|
|
* 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/>.
|
|
*
|
|
* This file incorporates work covered by the following copyright and
|
|
* permission notice:
|
|
*
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "PasteUpload.h"
|
|
#include "BuildConfig.h"
|
|
#include "Application.h"
|
|
|
|
#include <QDebug>
|
|
#include <QJsonObject>
|
|
#include <QHttpPart>
|
|
#include <QJsonArray>
|
|
#include <QJsonDocument>
|
|
#include <QFile>
|
|
#include <QUrlQuery>
|
|
|
|
#include "logging.h"
|
|
|
|
std::array<PasteUpload::PasteTypeInfo, 4> PasteUpload::PasteTypes = {
|
|
{{"0x0.st", "https://0x0.st", ""},
|
|
{"hastebin", "https://hst.sh", "/documents"},
|
|
{"paste.gg", "https://paste.gg", "/api/v1/pastes"},
|
|
{"mclo.gs", "https://api.mclo.gs", "/1/log"}}};
|
|
|
|
PasteUpload::PasteUpload(QWidget *window, QString text, QString baseUrl, PasteType pasteType) : m_window(window), m_baseUrl(baseUrl), m_pasteType(pasteType), m_text(text.toUtf8())
|
|
{
|
|
if (m_baseUrl == "")
|
|
m_baseUrl = PasteTypes.at(pasteType).defaultBase;
|
|
|
|
// HACK: Paste's docs say the standard API path is at /api/<version> but the official instance paste.gg doesn't follow that??
|
|
if (pasteType == PasteGG && m_baseUrl == PasteTypes.at(pasteType).defaultBase)
|
|
m_uploadUrl = "https://api.paste.gg/v1/pastes";
|
|
else
|
|
m_uploadUrl = m_baseUrl + PasteTypes.at(pasteType).endpointPath;
|
|
}
|
|
|
|
PasteUpload::~PasteUpload()
|
|
{
|
|
}
|
|
|
|
void PasteUpload::executeTask()
|
|
{
|
|
QNetworkRequest request{QUrl(m_uploadUrl)};
|
|
QNetworkReply *rep{};
|
|
|
|
request.setHeader(QNetworkRequest::UserAgentHeader, APPLICATION->getUserAgentUncached().toUtf8());
|
|
|
|
switch (m_pasteType) {
|
|
case NullPointer: {
|
|
QHttpMultiPart *multiPart =
|
|
new QHttpMultiPart{QHttpMultiPart::FormDataType};
|
|
|
|
QHttpPart filePart;
|
|
filePart.setBody(m_text);
|
|
filePart.setHeader(QNetworkRequest::ContentTypeHeader, "text/plain");
|
|
filePart.setHeader(QNetworkRequest::ContentDispositionHeader,
|
|
"form-data; name=\"file\"; filename=\"log.txt\"");
|
|
multiPart->append(filePart);
|
|
|
|
rep = APPLICATION->network()->post(request, multiPart);
|
|
multiPart->setParent(rep);
|
|
|
|
break;
|
|
}
|
|
case Hastebin: {
|
|
request.setHeader(QNetworkRequest::UserAgentHeader, APPLICATION->getUserAgentUncached().toUtf8());
|
|
rep = APPLICATION->network()->post(request, m_text);
|
|
break;
|
|
}
|
|
case Mclogs: {
|
|
QUrlQuery postData;
|
|
postData.addQueryItem("content", m_text);
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
|
rep = APPLICATION->network()->post(request, postData.toString().toUtf8());
|
|
break;
|
|
}
|
|
case PasteGG: {
|
|
QJsonObject obj;
|
|
QJsonDocument doc;
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
|
|
|
obj.insert("expires", QDateTime::currentDateTimeUtc().addDays(100).toString(Qt::DateFormat::ISODate));
|
|
|
|
QJsonArray files;
|
|
QJsonObject logFileInfo;
|
|
QJsonObject logFileContentInfo;
|
|
logFileContentInfo.insert("format", "text");
|
|
logFileContentInfo.insert("value", QString::fromUtf8(m_text));
|
|
logFileInfo.insert("name", "log.txt");
|
|
logFileInfo.insert("content", logFileContentInfo);
|
|
files.append(logFileInfo);
|
|
|
|
obj.insert("files", files);
|
|
|
|
doc.setObject(obj);
|
|
rep = APPLICATION->network()->post(request, doc.toJson());
|
|
break;
|
|
}
|
|
}
|
|
|
|
connect(rep, &QNetworkReply::uploadProgress, this, &Task::setProgress);
|
|
connect(rep, &QNetworkReply::finished, this, &PasteUpload::downloadFinished);
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
|
|
connect(rep, &QNetworkReply::errorOccurred, this, &PasteUpload::downloadError);
|
|
#else
|
|
connect(rep, QOverload<QNetworkReply::NetworkError>::of(&QNetworkReply::error), this, &PasteUpload::downloadError);
|
|
#endif
|
|
|
|
|
|
m_reply = std::shared_ptr<QNetworkReply>(rep);
|
|
|
|
setStatus(tr("Uploading to %1").arg(m_uploadUrl));
|
|
}
|
|
|
|
void PasteUpload::downloadError(QNetworkReply::NetworkError error)
|
|
{
|
|
// error happened during download.
|
|
qCCritical(taskUploadLogC) << "Network error: " << error;
|
|
emitFailed(m_reply->errorString());
|
|
}
|
|
|
|
void PasteUpload::downloadFinished()
|
|
{
|
|
QByteArray data = m_reply->readAll();
|
|
int statusCode = m_reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
if (m_reply->error() != QNetworkReply::NetworkError::NoError)
|
|
{
|
|
emitFailed(tr("Network error: %1").arg(m_reply->errorString()));
|
|
m_reply.reset();
|
|
return;
|
|
}
|
|
else if (statusCode != 200 && statusCode != 201)
|
|
{
|
|
QString reasonPhrase = m_reply->attribute(QNetworkRequest::HttpReasonPhraseAttribute).toString();
|
|
emitFailed(tr("Error: %1 returned unexpected status code %2 %3").arg(m_uploadUrl).arg(statusCode).arg(reasonPhrase));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned unexpected status code " << statusCode << " with body: " << data;
|
|
m_reply.reset();
|
|
return;
|
|
}
|
|
|
|
switch (m_pasteType)
|
|
{
|
|
case NullPointer:
|
|
m_pasteLink = QString::fromUtf8(data).trimmed();
|
|
break;
|
|
case Hastebin: {
|
|
QJsonDocument jsonDoc{QJsonDocument::fromJson(data)};
|
|
QJsonObject jsonObj{jsonDoc.object()};
|
|
if (jsonObj.contains("key") && jsonObj["key"].isString())
|
|
{
|
|
QString key = jsonDoc.object()["key"].toString();
|
|
m_pasteLink = m_baseUrl + "/" + key;
|
|
}
|
|
else
|
|
{
|
|
emitFailed(tr("Error: %1 returned a malformed response body").arg(m_uploadUrl));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned malformed response body: " << data;
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
case Mclogs: {
|
|
QJsonDocument jsonDoc{QJsonDocument::fromJson(data)};
|
|
QJsonObject jsonObj{jsonDoc.object()};
|
|
if (jsonObj.contains("success") && jsonObj["success"].isBool())
|
|
{
|
|
bool success = jsonObj["success"].toBool();
|
|
if (success)
|
|
{
|
|
m_pasteLink = jsonObj["url"].toString();
|
|
}
|
|
else
|
|
{
|
|
QString error = jsonObj["error"].toString();
|
|
emitFailed(tr("Error: %1 returned an error: %2").arg(m_uploadUrl, error));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned error: " << error;
|
|
qCCritical(taskUploadLogC) << "Response body: " << data;
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
emitFailed(tr("Error: %1 returned a malformed response body").arg(m_uploadUrl));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned malformed response body: " << data;
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
case PasteGG:
|
|
QJsonDocument jsonDoc{QJsonDocument::fromJson(data)};
|
|
QJsonObject jsonObj{jsonDoc.object()};
|
|
if (jsonObj.contains("status") && jsonObj["status"].isString())
|
|
{
|
|
QString status = jsonObj["status"].toString();
|
|
if (status == "success")
|
|
{
|
|
m_pasteLink = m_baseUrl + "/p/anonymous/" + jsonObj["result"].toObject()["id"].toString();
|
|
}
|
|
else
|
|
{
|
|
QString error = jsonObj["error"].toString();
|
|
QString message = (jsonObj.contains("message") && jsonObj["message"].isString()) ? jsonObj["message"].toString() : "none";
|
|
emitFailed(tr("Error: %1 returned an error code: %2\nError message: %3").arg(m_uploadUrl, error, message));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned error: " << error;
|
|
qCCritical(taskUploadLogC) << "Error message: " << message;
|
|
qCCritical(taskUploadLogC) << "Response body: " << data;
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
emitFailed(tr("Error: %1 returned a malformed response body").arg(m_uploadUrl));
|
|
qCCritical(taskUploadLogC) << m_uploadUrl << " returned malformed response body: " << data;
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
emitSucceeded();
|
|
}
|