From 193748f8367a416409562e6bc2d620901165b1f5 Mon Sep 17 00:00:00 2001 From: Martin Rotter Date: Sun, 24 Jan 2021 20:07:09 +0100 Subject: [PATCH] Proper HTTP redirection even for some totally exotic redirect URLs. --- resources/scripts/7za | 2 +- .../network-web/basenetworkaccessmanager.cpp | 13 +++- src/librssguard/network-web/downloader.cpp | 73 ++++--------------- 3 files changed, 27 insertions(+), 61 deletions(-) diff --git a/resources/scripts/7za b/resources/scripts/7za index 47f412575..9c10723bf 160000 --- a/resources/scripts/7za +++ b/resources/scripts/7za @@ -1 +1 @@ -Subproject commit 47f4125753452eff8800dbd6600c5a05540b15d9 +Subproject commit 9c10723bfbaf6cb85107d6ee16e0324e9e487749 diff --git a/src/librssguard/network-web/basenetworkaccessmanager.cpp b/src/librssguard/network-web/basenetworkaccessmanager.cpp index 5954d7c67..0ed338fb0 100644 --- a/src/librssguard/network-web/basenetworkaccessmanager.cpp +++ b/src/librssguard/network-web/basenetworkaccessmanager.cpp @@ -49,11 +49,18 @@ QNetworkReply* BaseNetworkAccessManager::createRequest(QNetworkAccessManager::Op QIODevice* outgoingData) { QNetworkRequest new_request = request; - // This rapidly speeds up loading of web sites. - // NOTE: https://en.wikipedia.org/wiki/HTTP_pipelining new_request.setAttribute(QNetworkRequest::Attribute::HttpPipeliningAllowedAttribute, true); +#if QT_VERSION >= 0x050900 + new_request.setAttribute(QNetworkRequest::Attribute::RedirectPolicyAttribute, + QNetworkRequest::RedirectPolicy::NoLessSafeRedirectPolicy); +#elif QT_VERSION >= 0x050600 + new_request.setAttribute(QNetworkRequest::Attribute::FollowRedirectsAttribute, true); +#endif + // Setup custom user-agent. new_request.setRawHeader(HTTP_HEADERS_USER_AGENT, QString(APP_USERAGENT).toLocal8Bit()); - return QNetworkAccessManager::createRequest(op, new_request, outgoingData); + + auto reply = QNetworkAccessManager::createRequest(op, new_request, outgoingData); + return reply; } diff --git a/src/librssguard/network-web/downloader.cpp b/src/librssguard/network-web/downloader.cpp index efcfb06b2..6befbf1a4 100644 --- a/src/librssguard/network-web/downloader.cpp +++ b/src/librssguard/network-web/downloader.cpp @@ -104,71 +104,30 @@ void Downloader::manipulateData(const QString& url, void Downloader::finished() { auto* reply = qobject_cast(sender()); - QNetworkAccessManager::Operation reply_operation = reply->operation(); - m_timer->stop(); // In this phase, some part of downloading process is completed. const QUrl redirection_url = reply->attribute(QNetworkRequest::RedirectionTargetAttribute).toUrl(); - if (redirection_url.isValid()) { - // Communication indicates that HTTP redirection is needed. - // Setup redirection URL and download again. - QNetworkRequest request = reply->request(); - - if (redirection_url.host().isEmpty()) { - request.setUrl(QUrl(reply->request().url().scheme() + QSL("://") + reply->request().url().host() + redirection_url.toString())); - } - else { - request.setUrl(redirection_url); - } - - qWarningNN << LOGSEC_NETWORK - << "We are redirecting URL request to:" - << QUOTE_W_SPACE_DOT(request.url()); - - m_activeReply->deleteLater(); - m_activeReply = nullptr; - - if (reply_operation == QNetworkAccessManager::GetOperation) { - runGetRequest(request); - } - else if (reply_operation == QNetworkAccessManager::Operation::PostOperation) { - if (m_inputMultipartData == nullptr) { - runPostRequest(request, m_inputData); - } - else { - runPostRequest(request, m_inputMultipartData); - } - } - else if (reply_operation == QNetworkAccessManager::PutOperation) { - runPutRequest(request, m_inputData); - } - else if (reply_operation == QNetworkAccessManager::DeleteOperation) { - runDeleteRequest(request); - } + // No redirection is indicated. Final file is obtained in our "reply" object. + // Read the data into output buffer. + if (m_inputMultipartData == nullptr) { + m_lastOutputData = reply->readAll(); } else { - // No redirection is indicated. Final file is obtained in our "reply" object. - // Read the data into output buffer. - if (m_inputMultipartData == nullptr) { - m_lastOutputData = reply->readAll(); - } - else { - m_lastOutputMultipartData = decodeMultipartAnswer(reply); - } - - m_lastContentType = reply->header(QNetworkRequest::ContentTypeHeader); - m_lastOutputError = reply->error(); - m_activeReply->deleteLater(); - m_activeReply = nullptr; - - if (m_inputMultipartData != nullptr) { - m_inputMultipartData->deleteLater(); - } - - emit completed(m_lastOutputError, m_lastOutputData); + m_lastOutputMultipartData = decodeMultipartAnswer(reply); } + + m_lastContentType = reply->header(QNetworkRequest::ContentTypeHeader); + m_lastOutputError = reply->error(); + m_activeReply->deleteLater(); + m_activeReply = nullptr; + + if (m_inputMultipartData != nullptr) { + m_inputMultipartData->deleteLater(); + } + + emit completed(m_lastOutputError, m_lastOutputData); } void Downloader::progressInternal(qint64 bytes_received, qint64 bytes_total) {