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.
|
|
|
|
*/
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
#include "peerinfo.h"
|
|
|
|
|
2019-06-02 12:13:34 +03:00
|
|
|
#include <QBitArray>
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
#include "base/bittorrent/torrenthandle.h"
|
2015-09-25 11:10:05 +03:00
|
|
|
#include "base/net/geoipmanager.h"
|
|
|
|
#include "base/unicodestrings.h"
|
2017-09-07 03:00:04 +03:00
|
|
|
#include "base/utils/string.h"
|
2015-04-19 18:17:47 +03:00
|
|
|
|
|
|
|
using namespace BitTorrent;
|
|
|
|
|
|
|
|
// PeerAddress
|
|
|
|
|
|
|
|
PeerAddress::PeerAddress()
|
|
|
|
: port(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-02-12 02:45:30 +02:00
|
|
|
PeerAddress::PeerAddress(const QHostAddress &ip, ushort port)
|
2015-04-19 18:17:47 +03:00
|
|
|
: ip(ip)
|
|
|
|
, port(port)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// PeerInfo
|
|
|
|
|
2019-05-07 11:22:39 +08:00
|
|
|
PeerInfo::PeerInfo(const TorrentHandle *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
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::dht);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::fromPeX() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::pex);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::fromLSD() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.source & lt::peer_info::lsd);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
2015-05-18 17:02:48 +03:00
|
|
|
#ifndef DISABLE_COUNTRIES_RESOLUTION
|
2015-04-19 18:17:47 +03:00
|
|
|
QString PeerInfo::country() const
|
|
|
|
{
|
2015-05-18 17:02:48 +03:00
|
|
|
return Net::GeoIPManager::instance()->lookup(address().ip);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
2015-05-18 17:02:48 +03:00
|
|
|
#endif
|
2015-04-19 18:17:47 +03:00
|
|
|
|
|
|
|
bool PeerInfo::isInteresting() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::interesting);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isChocked() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::choked);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isRemoteInterested() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::remote_interested);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isRemoteChocked() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::remote_choked);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isSupportsExtensions() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::supports_extensions);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isLocalConnection() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::local_connection);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isHandshake() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::handshake);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isConnecting() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::connecting);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isOnParole() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::on_parole);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isSeed() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::seed);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::optimisticUnchoke() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::optimistic_unchoke);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isSnubbed() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::snubbed);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isUploadOnly() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::upload_only);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isEndgameMode() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::endgame_mode);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isHolepunched() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::holepunched);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::useI2PSocket() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::i2p_socket);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::useUTPSocket() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::utp_socket);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::useSSLSocket() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::ssl_socket);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isRC4Encrypted() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::rc4_encrypted);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PeerInfo::isPlaintextEncrypted() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
return static_cast<bool>(m_nativeInfo.flags & lt::peer_info::plaintext_encrypted);
|
2015-04-19 18:17:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
PeerAddress PeerInfo::address() const
|
|
|
|
{
|
|
|
|
return PeerAddress(QHostAddress(QString::fromStdString(m_nativeInfo.ip.address().to_string())),
|
|
|
|
m_nativeInfo.ip.port());
|
|
|
|
}
|
|
|
|
|
|
|
|
QString PeerInfo::client() const
|
|
|
|
{
|
2017-03-07 19:41:38 +08:00
|
|
|
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());
|
|
|
|
|
2019-03-06 08:58:07 +03:00
|
|
|
int i = 0;
|
|
|
|
for (const bool bit : m_nativeInfo.pieces)
|
|
|
|
result.setBit(i++, bit);
|
2015-04-19 18:17:47 +03:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString PeerInfo::connectionType() const
|
|
|
|
{
|
2019-05-07 11:22:39 +08:00
|
|
|
if (m_nativeInfo.flags & lt::peer_info::utp_socket)
|
2015-09-04 22:56:08 +03:00
|
|
|
return QString::fromUtf8(C_UTP);
|
2015-04-19 18:17:47 +03:00
|
|
|
|
|
|
|
QString connection;
|
2017-09-07 03:00:04 +03:00
|
|
|
switch (m_nativeInfo.connection_type) {
|
2019-05-07 11:22:39 +08:00
|
|
|
case lt::peer_info::http_seed:
|
|
|
|
case lt::peer_info::web_seed:
|
2015-04-19 18:17:47 +03:00
|
|
|
connection = "Web";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
connection = "BT";
|
|
|
|
}
|
|
|
|
|
|
|
|
return connection;
|
|
|
|
}
|
2015-11-12 22:19:44 +03:00
|
|
|
|
|
|
|
void PeerInfo::calcRelevance(const TorrentHandle *torrent)
|
|
|
|
{
|
2018-07-21 15:07:11 +08:00
|
|
|
const QBitArray allPieces = torrent->pieces();
|
|
|
|
const QBitArray peerPieces = pieces();
|
2015-11-12 22:19:44 +03:00
|
|
|
|
|
|
|
int localMissing = 0;
|
|
|
|
int remoteHaves = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < allPieces.size(); ++i) {
|
|
|
|
if (!allPieces[i]) {
|
|
|
|
++localMissing;
|
|
|
|
if (peerPieces[i])
|
|
|
|
++remoteHaves;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (localMissing == 0)
|
|
|
|
m_relevance = 0.0;
|
|
|
|
else
|
|
|
|
m_relevance = static_cast<qreal>(remoteHaves) / localMissing;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal PeerInfo::relevance() const
|
|
|
|
{
|
|
|
|
return m_relevance;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PeerInfo::determineFlags()
|
|
|
|
{
|
2017-12-18 12:52:03 +02:00
|
|
|
QStringList flagsDescriptionList;
|
|
|
|
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isInteresting()) {
|
2017-09-07 03:00:04 +03:00
|
|
|
// d = Your client wants to download, but peer doesn't want to send (interested and choked)
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isRemoteChocked()) {
|
|
|
|
m_flags += "d ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "d = "
|
|
|
|
+ tr("Interested(local) and Choked(peer)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
else {
|
2017-09-07 03:00:04 +03:00
|
|
|
// D = Currently downloading (interested and not choked)
|
2015-11-12 22:19:44 +03:00
|
|
|
m_flags += "D ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "D = "
|
|
|
|
+ tr("interested(local) and unchoked(peer)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isRemoteInterested()) {
|
2017-09-07 03:00:04 +03:00
|
|
|
// u = Peer wants your client to upload, but your client doesn't want to (interested and choked)
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isChocked()) {
|
|
|
|
m_flags += "u ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "u = "
|
|
|
|
+ tr("interested(peer) and choked(local)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
else {
|
2017-09-07 03:00:04 +03:00
|
|
|
// U = Currently uploading (interested and not choked)
|
2015-11-12 22:19:44 +03:00
|
|
|
m_flags += "U ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "U = "
|
|
|
|
+ tr("interested(peer) and unchoked(local)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// O = Optimistic unchoke
|
2015-11-12 22:19:44 +03:00
|
|
|
if (optimisticUnchoke()) {
|
|
|
|
m_flags += "O ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "O = "
|
|
|
|
+ tr("optimistic unchoke");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// S = Peer is snubbed
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isSnubbed()) {
|
|
|
|
m_flags += "S ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "S = "
|
|
|
|
+ tr("peer snubbed");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// I = Peer is an incoming connection
|
2015-11-12 22:19:44 +03:00
|
|
|
if (!isLocalConnection()) {
|
|
|
|
m_flags += "I ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "I = "
|
|
|
|
+ tr("incoming connection");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// K = Peer is unchoking your client, but your client is not interested
|
2015-11-12 22:19:44 +03:00
|
|
|
if (!isRemoteChocked() && !isInteresting()) {
|
|
|
|
m_flags += "K ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "K = "
|
|
|
|
+ tr("not interested(local) and unchoked(peer)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// ? = Your client unchoked the peer but the peer is not interested
|
2015-11-12 22:19:44 +03:00
|
|
|
if (!isChocked() && !isRemoteInterested()) {
|
|
|
|
m_flags += "? ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "? = "
|
|
|
|
+ tr("not interested(peer) and unchoked(local)");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// X = Peer was included in peerlists obtained through Peer Exchange (PEX)
|
2015-11-12 22:19:44 +03:00
|
|
|
if (fromPeX()) {
|
|
|
|
m_flags += "X ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "X = "
|
|
|
|
+ tr("peer from PEX");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// H = Peer was obtained through DHT
|
2015-11-12 22:19:44 +03:00
|
|
|
if (fromDHT()) {
|
|
|
|
m_flags += "H ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "H = "
|
|
|
|
+ tr("peer from DHT");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// E = Peer is using Protocol Encryption (all traffic)
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isRC4Encrypted()) {
|
|
|
|
m_flags += "E ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "E = "
|
|
|
|
+ tr("encrypted traffic");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// e = Peer is using Protocol Encryption (handshake)
|
2015-11-12 22:19:44 +03:00
|
|
|
if (isPlaintextEncrypted()) {
|
|
|
|
m_flags += "e ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "e = "
|
|
|
|
+ tr("encrypted handshake");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// P = Peer is using uTorrent uTP
|
2015-11-12 22:19:44 +03:00
|
|
|
if (useUTPSocket()) {
|
|
|
|
m_flags += "P ";
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "P = "
|
|
|
|
+ QString::fromUtf8(C_UTP);
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
2017-09-07 03:00:04 +03:00
|
|
|
// L = Peer is local
|
2015-11-12 22:19:44 +03:00
|
|
|
if (fromLSD()) {
|
2018-07-21 13:28:13 +08:00
|
|
|
m_flags += 'L';
|
2017-12-18 12:52:03 +02:00
|
|
|
flagsDescriptionList += "L = "
|
|
|
|
+ tr("peer from LSD");
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
m_flags = m_flags.trimmed();
|
2017-12-18 12:52:03 +02:00
|
|
|
m_flagsDescription = flagsDescriptionList.join('\n');
|
2015-11-12 22:19:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString PeerInfo::flags() const
|
|
|
|
{
|
|
|
|
return m_flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString PeerInfo::flagsDescription() const
|
|
|
|
{
|
|
|
|
return m_flagsDescription;
|
|
|
|
}
|
2016-02-27 20:51:39 +01:00
|
|
|
|
|
|
|
int PeerInfo::downloadingPieceIndex() const
|
|
|
|
{
|
2019-03-06 08:58:07 +03:00
|
|
|
return static_cast<int>(m_nativeInfo.downloading_piece_index);
|
2016-02-27 20:51:39 +01:00
|
|
|
}
|