qBittorrent/src/base/bittorrent/peerinfo.cpp

347 lines
9.5 KiB
C++
Raw Normal View History

2015-04-19 18:17:47 +03:00
/*
* Bittorrent Client using Qt and libtorrent.
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru>
*
* 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.
*/
#include "peerinfo.h"
2019-06-02 12:13:34 +03:00
#include <QBitArray>
#include "base/bittorrent/torrent.h"
2015-09-25 11:10:05 +03:00
#include "base/net/geoipmanager.h"
#include "base/unicodestrings.h"
#include "peeraddress.h"
2015-04-19 18:17:47 +03:00
using namespace BitTorrent;
PeerInfo::PeerInfo(const Torrent *torrent, const lt::peer_info &nativeInfo)
2015-04-19 18:17:47 +03:00
: m_nativeInfo(nativeInfo)
{
2015-11-12 22:19:44 +03:00
calcRelevance(torrent);
determineFlags();
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::fromDHT() const
{
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::dht);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::fromPeX() const
{
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::pex);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::fromLSD() const
{
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::lsd);
2015-04-19 18:17:47 +03:00
}
QString PeerInfo::country() const
{
if (m_country.isEmpty())
m_country = Net::GeoIPManager::instance()->lookup(address().ip);
return m_country;
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isInteresting() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::interesting);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isChocked() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::choked);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isRemoteInterested() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::remote_interested);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isRemoteChocked() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::remote_choked);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isSupportsExtensions() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::supports_extensions);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isLocalConnection() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::local_connection);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isHandshake() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::handshake);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isConnecting() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::connecting);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isOnParole() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::on_parole);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isSeed() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::seed);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::optimisticUnchoke() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::optimistic_unchoke);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isSnubbed() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::snubbed);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isUploadOnly() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::upload_only);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isEndgameMode() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::endgame_mode);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isHolepunched() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::holepunched);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::useI2PSocket() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::i2p_socket);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::useUTPSocket() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::utp_socket);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::useSSLSocket() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::ssl_socket);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isRC4Encrypted() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::rc4_encrypted);
2015-04-19 18:17:47 +03:00
}
bool PeerInfo::isPlaintextEncrypted() const
{
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::plaintext_encrypted);
2015-04-19 18:17:47 +03:00
}
PeerAddress PeerInfo::address() const
{
// fast path for platforms which boost.asio internal struct maps to `sockaddr`
return {QHostAddress(m_nativeInfo.ip.data()), m_nativeInfo.ip.port()};
// slow path for the others
//return {QHostAddress(QString::fromStdString(m_nativeInfo.ip.address().to_string()))
// , m_nativeInfo.ip.port()};
2015-04-19 18:17:47 +03:00
}
QString PeerInfo::client() const
{
return QString::fromStdString(m_nativeInfo.client);
2015-04-19 18:17:47 +03:00
}
qreal PeerInfo::progress() const
{
return m_nativeInfo.progress;
}
int PeerInfo::payloadUpSpeed() const
{
return m_nativeInfo.payload_up_speed;
}
int PeerInfo::payloadDownSpeed() const
{
return m_nativeInfo.payload_down_speed;
}
qlonglong PeerInfo::totalUpload() const
{
return m_nativeInfo.total_upload;
}
qlonglong PeerInfo::totalDownload() const
{
return m_nativeInfo.total_download;
}
QBitArray PeerInfo::pieces() const
{
QBitArray result(m_nativeInfo.pieces.size());
2020-11-16 10:02:11 +03:00
for (int i = 0; i < result.size(); ++i)
{
if (m_nativeInfo.pieces[lt::piece_index_t {i}])
result.setBit(i, true);
}
2015-04-19 18:17:47 +03:00
return result;
}
QString PeerInfo::connectionType() const
{
if (m_nativeInfo.flags & lt::peer_info::utp_socket)
return QString::fromUtf8(C_UTP);
2015-04-19 18:17:47 +03:00
return (m_nativeInfo.connection_type == lt::peer_info::standard_bittorrent)
? QLatin1String {"BT"}
: QLatin1String {"Web"};
2015-04-19 18:17:47 +03:00
}
2015-11-12 22:19:44 +03:00
void PeerInfo::calcRelevance(const Torrent *torrent)
2015-11-12 22:19:44 +03:00
{
const QBitArray allPieces = torrent->pieces();
const int localMissing = allPieces.count(false);
if (localMissing <= 0)
2020-11-16 10:02:11 +03:00
{
m_relevance = 0;
return;
2015-11-12 22:19:44 +03:00
}
const QBitArray peerPieces = pieces();
const int remoteHaves = (peerPieces & (~allPieces)).count(true);
m_relevance = static_cast<qreal>(remoteHaves) / localMissing;
2015-11-12 22:19:44 +03:00
}
qreal PeerInfo::relevance() const
{
return m_relevance;
}
void PeerInfo::determineFlags()
{
const auto updateFlags = [this](const QChar specifier, const QString &explanation)
{
m_flags += (specifier + QLatin1Char(' '));
m_flagsDescription += QString::fromLatin1("%1 = %2\n").arg(specifier, explanation);
};
2020-11-16 10:02:11 +03:00
if (isInteresting())
{
if (isRemoteChocked())
{
// d = Your client wants to download, but peer doesn't want to send (interested and choked)
updateFlags(QLatin1Char('d'), tr("Interested (local) and choked (peer)"));
2015-11-12 22:19:44 +03:00
}
2020-11-16 10:02:11 +03:00
else
{
// D = Currently downloading (interested and not choked)
updateFlags(QLatin1Char('D'), tr("Interested (local) and unchoked (peer)"));
2015-11-12 22:19:44 +03:00
}
}
2020-11-16 10:02:11 +03:00
if (isRemoteInterested())
{
if (isChocked())
{
// u = Peer wants your client to upload, but your client doesn't want to (interested and choked)
updateFlags(QLatin1Char('u'), tr("Interested (peer) and choked (local)"));
2015-11-12 22:19:44 +03:00
}
2020-11-16 10:02:11 +03:00
else
{
// U = Currently uploading (interested and not choked)
updateFlags(QLatin1Char('U'), tr("Interested (peer) and unchoked (local)"));
2015-11-12 22:19:44 +03:00
}
}
// K = Peer is unchoking your client, but your client is not interested
if (!isRemoteChocked() && !isInteresting())
updateFlags(QLatin1Char('K'), tr("Not interested (local) and unchoked (peer)"));
// ? = Your client unchoked the peer but the peer is not interested
if (!isChocked() && !isRemoteInterested())
updateFlags(QLatin1Char('?'), tr("Not interested (peer) and unchoked (local)"));
// O = Optimistic unchoke
2020-11-16 10:02:11 +03:00
if (optimisticUnchoke())
updateFlags(QLatin1Char('O'), tr("Optimistic unchoke"));
2015-11-12 22:19:44 +03:00
// S = Peer is snubbed
2020-11-16 10:02:11 +03:00
if (isSnubbed())
updateFlags(QLatin1Char('S'), tr("Peer snubbed"));
2015-11-12 22:19:44 +03:00
// I = Peer is an incoming connection
2020-11-16 10:02:11 +03:00
if (!isLocalConnection())
updateFlags(QLatin1Char('I'), tr("Incoming connection"));
2015-11-12 22:19:44 +03:00
// H = Peer was obtained through DHT
if (fromDHT())
updateFlags(QLatin1Char('H'), tr("Peer from DHT"));
2015-11-12 22:19:44 +03:00
// X = Peer was included in peerlists obtained through Peer Exchange (PEX)
2020-11-16 10:02:11 +03:00
if (fromPeX())
updateFlags(QLatin1Char('X'), tr("Peer from PEX"));
2015-11-12 22:19:44 +03:00
// L = Peer is local
if (fromLSD())
updateFlags(QLatin1Char('L'), tr("Peer from LSD"));
2015-11-12 22:19:44 +03:00
// E = Peer is using Protocol Encryption (all traffic)
2020-11-16 10:02:11 +03:00
if (isRC4Encrypted())
updateFlags(QLatin1Char('E'), tr("Encrypted traffic"));
2015-11-12 22:19:44 +03:00
// e = Peer is using Protocol Encryption (handshake)
2020-11-16 10:02:11 +03:00
if (isPlaintextEncrypted())
updateFlags(QLatin1Char('e'), tr("Encrypted handshake"));
2015-11-12 22:19:44 +03:00
// P = Peer is using uTorrent uTP
2020-11-16 10:02:11 +03:00
if (useUTPSocket())
updateFlags(QLatin1Char('P'), QString::fromUtf8(C_UTP));
2015-11-12 22:19:44 +03:00
m_flags.chop(1);
m_flagsDescription.chop(1);
2015-11-12 22:19:44 +03:00
}
QString PeerInfo::flags() const
{
return m_flags;
}
QString PeerInfo::flagsDescription() const
{
return m_flagsDescription;
}
int PeerInfo::downloadingPieceIndex() const
{
return static_cast<int>(m_nativeInfo.downloading_piece_index);
}