- There's now a notice reminding people to change the base URL if they had a custom base URL and change the paste type (that was something I personally had problems with when I was testing, so a reminder was helpful for me). - Broke down some of the long lines on APIPage.cpp to be more readable. - Added copyright headers where they were missing. - Changed the paste service display names to the names they are more commonly known by. - Changed the default hastebin base URL to https://hst.sh due to the acquisition of https://hastebin.com by Toptal.
		
			
				
	
	
		
			247 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
		
			9.1 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>
 | |
|  *
 | |
|  *  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 <QJsonArray>
 | |
| #include <QJsonDocument>
 | |
| #include <QFile>
 | |
| 
 | |
| 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, BuildConfig.USER_AGENT_UNCACHED);
 | |
| 
 | |
|     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, BuildConfig.USER_AGENT_UNCACHED);
 | |
|         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);
 | |
|     // This function call would be a lot shorter if we were using the latest Qt
 | |
|     connect(rep,
 | |
|             static_cast<void (QNetworkReply::*)(QNetworkReply::NetworkError)>(&QNetworkReply::error),
 | |
|             this, &PasteUpload::downloadError);
 | |
| 
 | |
|     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.
 | |
|     qCritical() << "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));
 | |
|         qCritical() << 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));
 | |
|             qCritical() << 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));
 | |
|                 qCritical() << m_uploadUrl << " returned error: " << error;
 | |
|                 qCritical() << "Response body: " << data;
 | |
|                 return;
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             emitFailed(tr("Error: %1 returned a malformed response body").arg(m_uploadUrl));
 | |
|             qCritical() << 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));
 | |
|                 qCritical() << m_uploadUrl << " returned error: " << error;
 | |
|                 qCritical() << "Error message: " << message;
 | |
|                 qCritical() << "Response body: " << data;
 | |
|                 return;
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             emitFailed(tr("Error: %1 returned a malformed response body").arg(m_uploadUrl));
 | |
|             qCritical() << m_uploadUrl << " returned malformed response body: " << data;
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     emitSucceeded();
 | |
| }
 |