/* This file is part of Clementine. Copyright 2010-2013, David Sansome Copyright 2010-2012, 2014, John Maguire Copyright 2014, Krzysztof Sobiecki Clementine 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, either version 3 of the License, or (at your option) any later version. Clementine 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 Clementine. If not, see . */ #include "network.h" #include #include #include #include #include #include "core/closure.h" #include "core/logging.h" #include "core/utilities.h" QMutex ThreadSafeNetworkDiskCache::sMutex; QNetworkDiskCache* ThreadSafeNetworkDiskCache::sCache = nullptr; QMutex NetworkAccessManager::sMutex; QMap* NetworkAccessManager::sTrafficAnalysis = nullptr; ThreadSafeNetworkDiskCache::ThreadSafeNetworkDiskCache(QObject* parent) { QMutexLocker l(&sMutex); if (!sCache) { sCache = new QNetworkDiskCache; sCache->setCacheDirectory( Utilities::GetConfigPath(Utilities::Path_NetworkCache)); } } qint64 ThreadSafeNetworkDiskCache::cacheSize() const { QMutexLocker l(&sMutex); return sCache->cacheSize(); } QIODevice* ThreadSafeNetworkDiskCache::data(const QUrl& url) { QMutexLocker l(&sMutex); return sCache->data(url); } void ThreadSafeNetworkDiskCache::insert(QIODevice* device) { QMutexLocker l(&sMutex); sCache->insert(device); } QNetworkCacheMetaData ThreadSafeNetworkDiskCache::metaData(const QUrl& url) { QMutexLocker l(&sMutex); return sCache->metaData(url); } QIODevice* ThreadSafeNetworkDiskCache::prepare( const QNetworkCacheMetaData& metaData) { QMutexLocker l(&sMutex); return sCache->prepare(metaData); } bool ThreadSafeNetworkDiskCache::remove(const QUrl& url) { QMutexLocker l(&sMutex); return sCache->remove(url); } void ThreadSafeNetworkDiskCache::updateMetaData( const QNetworkCacheMetaData& metaData) { QMutexLocker l(&sMutex); sCache->updateMetaData(metaData); } void ThreadSafeNetworkDiskCache::clear() { QMutexLocker l(&sMutex); sCache->clear(); } NetworkAccessManager::NetworkAccessManager(QObject* parent) : QNetworkAccessManager(parent) { StaticInit(); setCache(new ThreadSafeNetworkDiskCache(this)); } void NetworkAccessManager::StaticInit() { QMutexLocker l(&sMutex); if (sTrafficAnalysis == nullptr) { sTrafficAnalysis = new QMap; } } void NetworkAccessManager::PrintNetworkStatistics() { for (const auto& it : sTrafficAnalysis->toStdMap()) { qLog(Debug) << it.first << it.second; } } QNetworkReply* NetworkAccessManager::createRequest( Operation op, const QNetworkRequest& request, QIODevice* outgoingData) { QByteArray user_agent = QString("%1 %2") .arg(QCoreApplication::applicationName(), QCoreApplication::applicationVersion()) .toUtf8(); if (request.hasRawHeader("User-Agent")) { // Append the existing user-agent set by a client library (such as // libmygpo-qt). user_agent += " " + request.rawHeader("User-Agent"); } QNetworkRequest new_request(request); new_request.setRawHeader("User-Agent", user_agent); if (op == QNetworkAccessManager::PostOperation && !new_request.header(QNetworkRequest::ContentTypeHeader).isValid()) { new_request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded"); } // Prefer the cache unless the caller has changed the setting already if (request.attribute(QNetworkRequest::CacheLoadControlAttribute).toInt() == QNetworkRequest::PreferNetwork) { new_request.setAttribute(QNetworkRequest::CacheLoadControlAttribute, QNetworkRequest::PreferCache); } RecordRequest(new_request); return QNetworkAccessManager::createRequest(op, new_request, outgoingData); } void NetworkAccessManager::RecordRequest(const QNetworkRequest& request) { QMutexLocker l(&sMutex); if (sTrafficAnalysis == nullptr) return; const QUrl& url = request.url(); const QString recorded = QString("%1://%2:%3%4") .arg(url.scheme()) .arg(url.host()) .arg(url.port()) .arg(url.path()); if (sTrafficAnalysis->contains(recorded)) { ++(*sTrafficAnalysis)[recorded]; } else { (*sTrafficAnalysis)[recorded] = 1; } } NetworkTimeouts::NetworkTimeouts(int timeout_msec, QObject* parent) : timeout_msec_(timeout_msec) {} void NetworkTimeouts::AddReply(QNetworkReply* reply) { if (timers_.contains(reply)) return; connect(reply, SIGNAL(destroyed()), SLOT(ReplyFinished())); connect(reply, SIGNAL(finished()), SLOT(ReplyFinished())); timers_[reply] = startTimer(timeout_msec_); } void NetworkTimeouts::AddReply(RedirectFollower* reply) { if (redirect_timers_.contains(reply)) { return; } NewClosure(reply, SIGNAL(destroyed()), this, SLOT(RedirectFinished(RedirectFollower*)), reply); NewClosure(reply, SIGNAL(finished()), this, SLOT(RedirectFinished(RedirectFollower*)), reply); redirect_timers_[reply] = startTimer(timeout_msec_); } void NetworkTimeouts::ReplyFinished() { QNetworkReply* reply = reinterpret_cast(sender()); if (timers_.contains(reply)) { killTimer(timers_.take(reply)); } } void NetworkTimeouts::RedirectFinished(RedirectFollower* reply) { if (redirect_timers_.contains(reply)) { killTimer(redirect_timers_.take(reply)); } } void NetworkTimeouts::timerEvent(QTimerEvent* e) { QNetworkReply* reply = timers_.key(e->timerId()); if (reply) { reply->abort(); } RedirectFollower* redirect = redirect_timers_.key(e->timerId()); if (redirect) { redirect->abort(); } } RedirectFollower::RedirectFollower(QNetworkReply* first_reply, int max_redirects) : QObject(nullptr), current_reply_(first_reply), redirects_remaining_(max_redirects) { ConnectReply(first_reply); } void RedirectFollower::ConnectReply(QNetworkReply* reply) { connect(reply, SIGNAL(readyRead()), SLOT(ReadyRead())); connect(reply, SIGNAL(error(QNetworkReply::NetworkError)), SIGNAL(error(QNetworkReply::NetworkError))); connect(reply, SIGNAL(downloadProgress(qint64, qint64)), SIGNAL(downloadProgress(qint64, qint64))); connect(reply, SIGNAL(uploadProgress(qint64, qint64)), SIGNAL(uploadProgress(qint64, qint64))); connect(reply, SIGNAL(finished()), SLOT(ReplyFinished())); } void RedirectFollower::ReadyRead() { // Don't re-emit this signal for redirect replies. if (current_reply_->attribute(QNetworkRequest::RedirectionTargetAttribute) .isValid()) { return; } emit readyRead(); } void RedirectFollower::ReplyFinished() { current_reply_->deleteLater(); if (current_reply_->attribute(QNetworkRequest::RedirectionTargetAttribute) .isValid()) { if (redirects_remaining_-- == 0) { emit finished(); return; } const QUrl next_url = current_reply_->url().resolved( current_reply_->attribute(QNetworkRequest::RedirectionTargetAttribute) .toUrl()); QNetworkRequest req(current_reply_->request()); req.setUrl(next_url); current_reply_ = current_reply_->manager()->get(req); ConnectReply(current_reply_); return; } emit finished(); }