/* * Bittorrent Client using Qt4 and libtorrent. * Copyright (C) 2011 Christophe Dumez * * 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; either version 2 * of the License, or (at your option) any later version. * * 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, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. * * Contact : chris@qbittorrent.org */ #include #include #include #include "qbtsession.h" #include "misc.h" #include "torrentspeedmonitor.h" #include "qinisettings.h" using namespace libtorrent; template struct Sample { Sample(const T down = 0, const T up = 0) : download(down), upload(up) {} T download; T upload; }; class SpeedSample { public: SpeedSample() {} void addSample(int speedDL, int speedUL); Sample average() const; void clear(); private: static const int max_samples = 30; private: QList > m_speedSamples; }; TorrentSpeedMonitor::TorrentSpeedMonitor(QBtSession* session) : QThread(session), m_abort(false), m_session(session) { connect(m_session, SIGNAL(deletedTorrent(QString)), SLOT(removeSamples(QString))); connect(m_session, SIGNAL(pausedTorrent(QTorrentHandle)), SLOT(removeSamples(QTorrentHandle))); loadStats(); } TorrentSpeedMonitor::~TorrentSpeedMonitor() { m_abort = true; m_abortCond.wakeOne(); wait(); saveStats(); } void TorrentSpeedMonitor::run() { do { m_mutex.lock(); getSamples(); saveStats(); m_abortCond.wait(&m_mutex, 1000); m_mutex.unlock(); } while(!m_abort); } void SpeedSample::addSample(int speedDL, int speedUL) { m_speedSamples << Sample(speedDL, speedUL); if (m_speedSamples.size() > max_samples) m_speedSamples.removeFirst(); } Sample SpeedSample::average() const { if (m_speedSamples.empty()) return Sample(); qlonglong sumDL = 0; qlonglong sumUL = 0; foreach (const Sample& s, m_speedSamples) { sumDL += s.download; sumUL += s.upload; } const qreal numSamples = m_speedSamples.size(); return Sample(sumDL/numSamples, sumUL/numSamples); } void SpeedSample::clear() { m_speedSamples.clear(); } void TorrentSpeedMonitor::removeSamples(const QString &hash) { m_samples.remove(hash); } void TorrentSpeedMonitor::removeSamples(const QTorrentHandle& h) { try { m_samples.remove(h.hash()); } catch(invalid_handle&) {} } qlonglong TorrentSpeedMonitor::getETA(const QString &hash) const { QMutexLocker locker(&m_mutex); QTorrentHandle h = m_session->getTorrentHandle(hash); if (h.is_paused() || !m_samples.contains(hash)) return MAX_ETA; const Sample speed_average = m_samples[hash].average(); if (h.is_seed()) { if (!speed_average.upload) return MAX_ETA; bool _unused; qreal max_ratio = m_session->getMaxRatioPerTorrent(hash, &_unused); if (max_ratio < 0) return MAX_ETA; libtorrent::size_type realDL = h.all_time_download(); if (realDL <= 0) realDL = h.total_wanted(); return (realDL * max_ratio - h.all_time_upload()) / speed_average.upload; } if (!speed_average.download) return MAX_ETA; return (h.total_wanted() - h.total_wanted_done()) / speed_average.download; } quint64 TorrentSpeedMonitor::getAlltimeDL() const { QMutexLocker l(&m_mutex); return alltimeDL + sessionDL; } quint64 TorrentSpeedMonitor::getAlltimeUL() const { QMutexLocker l(&m_mutex); return alltimeUL + sessionUL; } void TorrentSpeedMonitor::getSamples() { const std::vector torrents = m_session->getSession()->get_torrents(); std::vector::const_iterator it = torrents.begin(); std::vector::const_iterator itend = torrents.end(); for ( ; it != itend; ++it) { try { torrent_status st = it->status(0x0); if (!st.paused) { int up = st.upload_payload_rate; int down = st.download_payload_rate; m_samples[misc::toQString(it->info_hash())].addSample(down, up); } } catch(invalid_handle&) {} } libtorrent::session_status ss = m_session->getSessionStatus(); sessionDL = ss.total_download; sessionUL = ss.total_upload; } void TorrentSpeedMonitor::saveStats() const { QIniSettings s; QVariantHash v; v.insert("AlltimeDL", alltimeDL + sessionDL); v.insert("AlltimeUL", alltimeUL + sessionUL); s.setValue("Stats/AllStats", v); } void TorrentSpeedMonitor::loadStats() { QIniSettings s; QVariantHash v(s.value("Stats/AllStats", QVariantHash()).toHash()); alltimeDL = v["AlltimeDL"].toULongLong(); alltimeUL = v["AlltimeUL"].toULongLong(); sessionDL = sessionUL = 0; }