NOISSUE fix bad redirect URLs provided by the curse CDN

MultiMC now parses the HTTP Location header in a (more) tolerant mode.
This commit is contained in:
Petr Mrázek 2018-01-21 03:49:54 +01:00
parent 0942867ecc
commit c33b4e252f
4 changed files with 50 additions and 4 deletions

View File

@ -327,6 +327,7 @@ void InstanceImportTask::processFlame()
case Flame::File::Type::SingleFile: case Flame::File::Type::SingleFile:
case Flame::File::Type::Mod: case Flame::File::Type::Mod:
{ {
qDebug() << "Will download" << result.url << "to" << path;
auto dl = Net::Download::makeFile(result.url, path); auto dl = Net::Download::makeFile(result.url, path);
m_filesNetJob->addNetAction(dl); m_filesNetJob->addNetAction(dl);
break; break;

View File

@ -48,7 +48,13 @@ void Flame::FileResolvingTask::netJobFinished()
continue; continue;
} }
out.fileName = Json::requireString(obj, "FileNameOnDisk"); out.fileName = Json::requireString(obj, "FileNameOnDisk");
out.url = Json::requireString(obj, "DownloadURL"); auto urlString = Json::requireString(obj, "DownloadURL");
urlString.replace(' ', "%20");
out.url = QUrl(urlString, QUrl::StrictMode);
if(!out.url.isValid())
{
throw "Perkele!";
}
// This is a piece of a Flame project JSON pulled out into the file metadata (here) for convenience // This is a piece of a Flame project JSON pulled out into the file metadata (here) for convenience
// It is also optional // It is also optional
QJsonObject projObj = Json::ensureObject(obj, "_Project", {}); QJsonObject projObj = Json::ensureObject(obj, "_Project", {});

View File

@ -2,6 +2,7 @@
#include <QString> #include <QString>
#include <QVector> #include <QVector>
#include <QUrl>
namespace Flame namespace Flame
{ {
@ -15,7 +16,7 @@ struct File
// our // our
bool resolved = false; bool resolved = false;
QString fileName; QString fileName;
QString url; QUrl url;
QString targetFolder = QLatin1Literal("mods"); QString targetFolder = QLatin1Literal("mods");
enum class Type enum class Type
{ {

View File

@ -150,13 +150,50 @@ void Download::sslErrors(const QList<QSslError> & errors)
bool Download::handleRedirect() bool Download::handleRedirect()
{ {
QVariant redirect = m_reply->header(QNetworkRequest::LocationHeader); QUrl redirect = m_reply->header(QNetworkRequest::LocationHeader).toUrl();
if(!redirect.isValid())
{
if(!m_reply->hasRawHeader("Location"))
{
// no redirect -> it's fine to continue
return false;
}
// there is a Location header, but it's not correct. we need to apply some workarounds...
QByteArray redirectBA = m_reply->rawHeader("Location");
if(redirectBA.size() == 0)
{
// empty, yet present redirect header? WTF?
return false;
}
QString redirectStr = QString::fromUtf8(redirectBA);
/*
* IF the URL begins with //, we need to insert the URL scheme.
* See: https://bugreports.qt-project.org/browse/QTBUG-41061
*/
if(redirectStr.startsWith("//"))
{
redirectStr = m_reply->url().scheme() + ":" + redirectStr;
}
/*
* Next, make sure the URL is parsed in tolerant mode. Qt doesn't parse the location header in tolerant mode, which causes issues.
* FIXME: report Qt bug for this
*/
redirect = QUrl(redirectStr, QUrl::TolerantMode);
qDebug() << "Fixed location header:" << redirect;
}
else
{
qDebug() << "Location header:" << redirect;
}
QString redirectURL; QString redirectURL;
if(redirect.isValid()) if(redirect.isValid())
{ {
redirectURL = redirect.toString(); redirectURL = redirect.toString();
} }
// FIXME: This is a hack for https://bugreports.qt-project.org/browse/QTBUG-41061 // FIXME: This is a hack for
else if(m_reply->hasRawHeader("Location")) else if(m_reply->hasRawHeader("Location"))
{ {
auto data = m_reply->rawHeader("Location"); auto data = m_reply->rawHeader("Location");
@ -165,6 +202,7 @@ bool Download::handleRedirect()
redirectURL = m_reply->url().scheme() + ":" + data; redirectURL = m_reply->url().scheme() + ":" + data;
} }
} }
if (!redirectURL.isEmpty()) if (!redirectURL.isEmpty())
{ {
m_url = QUrl(redirect.toString()); m_url = QUrl(redirect.toString());