2015-04-19 18:17:47 +03:00
/*
* Bittorrent Client using Qt and libtorrent .
* Copyright ( C ) 2015 Vladimir Golovnev < glassez @ yandex . ru >
* Copyright ( C ) 2006 Christophe Dumez < chris @ qbittorrent . org >
*
* 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 .
*/
2016-04-19 09:54:48 +03:00
# include "session.h"
2018-02-02 10:12:49 +03:00
# include <algorithm>
2017-09-07 03:00:04 +03:00
# include <queue>
2017-10-08 09:59:52 +03:00
# include <string>
2019-10-02 07:52:51 +03:00
# include <utility>
2017-09-07 03:00:04 +03:00
2019-04-25 22:02:47 +03:00
# ifdef Q_OS_WIN
2019-09-28 09:21:53 +03:00
# include <Windows.h>
2019-04-25 22:02:47 +03:00
# include <wincrypt.h>
# include <iphlpapi.h>
# endif
2015-04-19 18:17:47 +03:00
# include <QDebug>
# include <QDir>
2019-10-01 10:33:11 +03:00
# include <QFile>
2015-04-19 18:17:47 +03:00
# include <QHostAddress>
# include <QNetworkAddressEntry>
2019-10-01 10:33:11 +03:00
# include <QNetworkConfigurationManager>
2016-04-19 09:54:48 +03:00
# include <QNetworkInterface>
2018-05-24 18:41:03 +03:00
# include <QRegularExpression>
2016-04-19 09:54:48 +03:00
# include <QString>
# include <QThread>
# include <QTimer>
2017-08-13 00:58:22 +03:00
# include <QUuid>
2015-04-19 18:17:47 +03:00
2016-04-19 09:54:48 +03:00
# include <libtorrent/alert_types.hpp>
2019-01-08 15:52:12 +03:00
# include <libtorrent/bdecode.hpp>
2015-04-19 18:17:47 +03:00
# include <libtorrent/bencode.hpp>
# include <libtorrent/error_code.hpp>
# include <libtorrent/extensions/ut_metadata.hpp>
# include <libtorrent/extensions/ut_pex.hpp>
# include <libtorrent/extensions/smart_ban.hpp>
2016-04-19 09:54:48 +03:00
# include <libtorrent/ip_filter.hpp>
# include <libtorrent/magnet_uri.hpp>
# include <libtorrent/session.hpp>
2019-01-08 15:52:12 +03:00
# include <libtorrent/session_stats.hpp>
2017-04-29 14:45:30 +03:00
# include <libtorrent/session_status.hpp>
2016-04-19 09:54:48 +03:00
# include <libtorrent/torrent_info.hpp>
2019-03-06 08:58:07 +03:00
# include <libtorrent/version.hpp>
2015-04-19 18:17:47 +03:00
2019-06-29 11:15:41 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200)
# include <libtorrent/read_resume_data.hpp>
# endif
2018-03-06 17:50:10 +03:00
# include "base/algorithm.h"
2018-11-29 17:25:38 +03:00
# include "base/exceptions.h"
2018-11-19 12:53:29 +03:00
# include "base/global.h"
2015-09-25 11:10:05 +03:00
# include "base/logger.h"
2016-04-19 09:54:48 +03:00
# include "base/net/downloadmanager.h"
2016-05-01 11:05:52 +03:00
# include "base/net/proxyconfigurationmanager.h"
2017-09-07 03:00:04 +03:00
# include "base/profile.h"
2016-04-07 17:58:30 +03:00
# include "base/torrentfileguard.h"
2016-04-19 09:54:48 +03:00
# include "base/torrentfilter.h"
# include "base/unicodestrings.h"
# include "base/utils/fs.h"
2017-09-07 03:00:04 +03:00
# include "base/utils/misc.h"
# include "base/utils/net.h"
2017-02-10 15:33:21 +03:00
# include "base/utils/random.h"
2016-04-19 09:54:48 +03:00
# include "magneturi.h"
2015-04-19 18:17:47 +03:00
# include "private/bandwidthscheduler.h"
2017-09-07 03:00:04 +03:00
# include "private/filterparserthread.h"
2019-07-18 19:53:04 +03:00
# include "private/ltunderlyingtype.h"
2019-03-06 08:58:07 +03:00
# include "private/portforwarderimpl.h"
2015-12-13 15:38:19 +03:00
# include "private/resumedatasavingmanager.h"
2017-09-07 03:00:04 +03:00
# include "private/statistics.h"
2015-04-19 18:17:47 +03:00
# include "torrenthandle.h"
2016-04-19 09:54:48 +03:00
# include "tracker.h"
# include "trackerentry.h"
2015-04-19 18:17:47 +03:00
2016-10-30 00:11:52 +03:00
static const char PEER_ID [ ] = " qB " ;
static const char RESUME_FOLDER [ ] = " BT_backup " ;
2017-03-08 00:41:29 +03:00
static const char USER_AGENT [ ] = " qBittorrent/ " QBT_VERSION_2 ;
2015-04-19 18:17:47 +03:00
using namespace BitTorrent ;
2016-02-09 11:56:48 +03:00
namespace
{
2019-03-06 08:58:07 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-07-18 19:53:04 +03:00
using LTAlertCategory = int ;
using LTPeerClass = int ;
using LTQueuePosition = int ;
2019-03-06 08:58:07 +03:00
using LTSessionFlags = int ;
using LTStatusFlags = int ;
using LTString = std : : string ;
# else
2019-07-18 19:53:04 +03:00
using LTAlertCategory = lt : : alert_category_t ;
using LTPeerClass = lt : : peer_class_t ;
using LTQueuePosition = lt : : queue_position_t ;
2019-03-06 08:58:07 +03:00
using LTSessionFlags = lt : : session_flags_t ;
using LTStatusFlags = lt : : status_flags_t ;
using LTString = lt : : string_view ;
# endif
2019-11-03 19:15:45 +03:00
template < typename LTStr >
QString fromLTString ( const LTStr & str )
{
return QString : : fromUtf8 ( str . data ( ) , static_cast < int > ( str . size ( ) ) ) ;
}
bool readFile ( const QString & path , QByteArray & buf )
{
QFile file ( path ) ;
if ( ! file . open ( QIODevice : : ReadOnly ) ) {
qDebug ( " Cannot read file %s: %s " , qUtf8Printable ( path ) , qUtf8Printable ( file . errorString ( ) ) ) ;
return false ;
}
2016-02-09 11:56:48 +03:00
2019-11-03 19:15:45 +03:00
buf = file . readAll ( ) ;
return true ;
}
2016-02-09 11:56:48 +03:00
2019-11-03 19:15:45 +03:00
bool loadTorrentResumeData ( const QByteArray & data , CreateTorrentParams & torrentParams , int & queuePos , MagnetUri & magnetUri )
{
lt : : error_code ec ;
lt : : bdecode_node root ;
lt : : bdecode ( data . constData ( ) , ( data . constData ( ) + data . size ( ) ) , root , ec ) ;
if ( ec | | ( root . type ( ) ! = lt : : bdecode_node : : dict_t ) ) return false ;
torrentParams = CreateTorrentParams ( ) ;
torrentParams . restored = true ;
torrentParams . skipChecking = false ;
torrentParams . name = fromLTString ( root . dict_find_string_value ( " qBt-name " ) ) ;
torrentParams . savePath = Profile : : instance ( ) . fromPortablePath (
Utils : : Fs : : toUniformPath ( fromLTString ( root . dict_find_string_value ( " qBt-savePath " ) ) ) ) ;
torrentParams . disableTempPath = root . dict_find_int_value ( " qBt-tempPathDisabled " ) ;
torrentParams . sequential = root . dict_find_int_value ( " qBt-sequential " ) ;
torrentParams . hasSeedStatus = root . dict_find_int_value ( " qBt-seedStatus " ) ;
torrentParams . firstLastPiecePriority = root . dict_find_int_value ( " qBt-firstLastPiecePriority " ) ;
torrentParams . hasRootFolder = root . dict_find_int_value ( " qBt-hasRootFolder " ) ;
torrentParams . seedingTimeLimit = root . dict_find_int_value ( " qBt-seedingTimeLimit " , TorrentHandle : : USE_GLOBAL_SEEDING_TIME ) ;
const bool isAutoManaged = root . dict_find_int_value ( " auto_managed " ) ;
const bool isPaused = root . dict_find_int_value ( " paused " ) ;
torrentParams . paused = root . dict_find_int_value ( " qBt-paused " , ( isPaused & & ! isAutoManaged ) ) ;
torrentParams . forced = root . dict_find_int_value ( " qBt-forced " , ( ! isPaused & & ! isAutoManaged ) ) ;
const LTString ratioLimitString = root . dict_find_string_value ( " qBt-ratioLimit " ) ;
if ( ratioLimitString . empty ( ) )
torrentParams . ratioLimit = root . dict_find_int_value ( " qBt-ratioLimit " , TorrentHandle : : USE_GLOBAL_RATIO * 1000 ) / 1000.0 ;
else
torrentParams . ratioLimit = fromLTString ( ratioLimitString ) . toDouble ( ) ;
// **************************************************************************************
// Workaround to convert legacy label to category
// TODO: Should be removed in future
torrentParams . category = fromLTString ( root . dict_find_string_value ( " qBt-label " ) ) ;
if ( torrentParams . category . isEmpty ( ) )
// **************************************************************************************
torrentParams . category = fromLTString ( root . dict_find_string_value ( " qBt-category " ) ) ;
const lt : : bdecode_node tagsNode = root . dict_find ( " qBt-tags " ) ;
if ( tagsNode . type ( ) = = lt : : bdecode_node : : list_t ) {
for ( int i = 0 ; i < tagsNode . list_size ( ) ; + + i ) {
const QString tag = fromLTString ( tagsNode . list_string_value_at ( i ) ) ;
if ( Session : : isValidTag ( tag ) )
torrentParams . tags < < tag ;
}
}
const lt : : bdecode_node addedTimeNode = root . dict_find ( " qBt-addedTime " ) ;
if ( addedTimeNode . type ( ) = = lt : : bdecode_node : : int_t )
torrentParams . addedTime = QDateTime : : fromSecsSinceEpoch ( addedTimeNode . int_value ( ) ) ;
queuePos = root . dict_find_int_value ( " qBt-queuePosition " ) ;
magnetUri = MagnetUri ( fromLTString ( root . dict_find_string_value ( " qBt-magnetUri " ) ) ) ;
return true ;
}
void torrentQueuePositionUp ( const lt : : torrent_handle & handle )
{
try {
handle . queue_position_up ( ) ;
}
catch ( const std : : exception & exc ) {
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
}
void torrentQueuePositionDown ( const lt : : torrent_handle & handle )
{
try {
handle . queue_position_down ( ) ;
}
catch ( const std : : exception & exc ) {
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
}
void torrentQueuePositionTop ( const lt : : torrent_handle & handle )
{
try {
handle . queue_position_top ( ) ;
}
catch ( const std : : exception & exc ) {
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
}
void torrentQueuePositionBottom ( const lt : : torrent_handle & handle )
{
try {
handle . queue_position_bottom ( ) ;
}
catch ( const std : : exception & exc ) {
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
}
2017-08-13 00:58:22 +03:00
2016-02-09 11:56:48 +03:00
QStringMap map_cast ( const QVariantMap & map )
{
QStringMap result ;
2018-03-06 17:50:10 +03:00
for ( auto i = map . cbegin ( ) ; i ! = map . cend ( ) ; + + i )
result [ i . key ( ) ] = i . value ( ) . toString ( ) ;
2016-02-09 11:56:48 +03:00
return result ;
}
QVariantMap map_cast ( const QStringMap & map )
{
QVariantMap result ;
2018-03-06 17:50:10 +03:00
for ( auto i = map . cbegin ( ) ; i ! = map . cend ( ) ; + + i )
result [ i . key ( ) ] = i . value ( ) ;
2016-02-09 11:56:48 +03:00
return result ;
}
2016-05-01 11:05:52 +03:00
QString normalizePath ( const QString & path )
{
2019-06-16 20:14:15 +03:00
QString tmp = Utils : : Fs : : toUniformPath ( path . trimmed ( ) ) ;
2016-05-01 11:05:52 +03:00
if ( ! tmp . isEmpty ( ) & & ! tmp . endsWith ( ' / ' ) )
return tmp + ' / ' ;
return tmp ;
}
2016-05-11 14:25:29 +03:00
QString normalizeSavePath ( QString path , const QString & defaultPath = specialFolderLocation ( SpecialFolder : : Downloads ) )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
path = path . trimmed ( ) ;
2016-03-06 09:25:55 +03:00
if ( path . isEmpty ( ) )
2019-06-16 20:14:15 +03:00
path = Utils : : Fs : : toUniformPath ( defaultPath . trimmed ( ) ) ;
2016-03-06 09:25:55 +03:00
2016-05-01 11:05:52 +03:00
return normalizePath ( path ) ;
2016-02-09 11:56:48 +03:00
}
QStringMap expandCategories ( const QStringMap & categories )
{
QStringMap expanded = categories ;
2015-04-19 18:17:47 +03:00
2018-03-06 17:50:10 +03:00
for ( auto i = categories . cbegin ( ) ; i ! = categories . cend ( ) ; + + i ) {
const QString & category = i . key ( ) ;
2018-11-27 23:15:04 +03:00
for ( const QString & subcat : asConst ( Session : : expandCategory ( category ) ) ) {
2016-02-09 11:56:48 +03:00
if ( ! expanded . contains ( subcat ) )
expanded [ subcat ] = " " ;
}
}
return expanded ;
}
2016-04-19 09:54:48 +03:00
2016-05-01 11:05:52 +03:00
template < typename T >
struct LowerLimited
{
LowerLimited ( T limit , T ret )
: m_limit ( limit )
, m_ret ( ret )
{
}
explicit LowerLimited ( T limit )
: LowerLimited ( limit , limit )
{
}
2017-05-09 09:21:55 +03:00
T operator ( ) ( T val ) const
2016-05-01 11:05:52 +03:00
{
return val < = m_limit ? m_ret : val ;
}
private :
const T m_limit ;
const T m_ret ;
} ;
template < typename T >
LowerLimited < T > lowerLimited ( T limit ) { return LowerLimited < T > ( limit ) ; }
template < typename T >
LowerLimited < T > lowerLimited ( T limit , T ret ) { return LowerLimited < T > ( limit , ret ) ; }
2017-08-15 15:26:28 +03:00
template < typename T >
std : : function < T ( const T & ) > clampValue ( const T lower , const T upper )
{
2019-02-04 08:09:21 +03:00
// TODO: change return type to `auto` when using C++17
2017-08-15 15:26:28 +03:00
return [ lower , upper ] ( const T value ) - > T
{
if ( value < lower )
return lower ;
if ( value > upper )
return upper ;
return value ;
} ;
}
2019-11-03 19:15:45 +03:00
# ifdef Q_OS_WIN
QString convertIfaceNameToGuid ( const QString & name )
{
// Under Windows XP or on Qt version <= 5.5 'name' will be a GUID already.
const QUuid uuid ( name ) ;
if ( ! uuid . isNull ( ) )
return uuid . toString ( ) . toUpper ( ) ; // Libtorrent expects the GUID in uppercase
NET_LUID luid { } ;
const LONG res = : : ConvertInterfaceNameToLuidW ( name . toStdWString ( ) . c_str ( ) , & luid ) ;
if ( res = = 0 ) {
GUID guid ;
if ( : : ConvertInterfaceLuidToGuid ( & luid , & guid ) = = 0 )
return QUuid ( guid ) . toString ( ) . toUpper ( ) ;
}
return { } ;
}
# endif
2016-02-09 11:56:48 +03:00
}
2015-04-19 18:17:47 +03:00
// Session
2016-02-09 11:56:48 +03:00
Session * Session : : m_instance = nullptr ;
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
# define BITTORRENT_KEY(name) "BitTorrent / " name
# define BITTORRENT_SESSION_KEY(name) BITTORRENT_KEY("Session / ") name
2015-04-19 18:17:47 +03:00
Session : : Session ( QObject * parent )
: QObject ( parent )
2016-05-01 11:05:52 +03:00
, m_isDHTEnabled ( BITTORRENT_SESSION_KEY ( " DHTEnabled " ) , true )
, m_isLSDEnabled ( BITTORRENT_SESSION_KEY ( " LSDEnabled " ) , true )
, m_isPeXEnabled ( BITTORRENT_SESSION_KEY ( " PeXEnabled " ) , true )
2016-10-29 19:14:27 +03:00
, m_isIPFilteringEnabled ( BITTORRENT_SESSION_KEY ( " IPFilteringEnabled " ) , false )
2016-05-01 11:05:52 +03:00
, m_isTrackerFilteringEnabled ( BITTORRENT_SESSION_KEY ( " TrackerFilteringEnabled " ) , false )
, m_IPFilterFile ( BITTORRENT_SESSION_KEY ( " IPFilter " ) )
2017-10-31 00:04:14 +03:00
, m_announceToAllTrackers ( BITTORRENT_SESSION_KEY ( " AnnounceToAllTrackers " ) , false )
2017-10-30 21:04:14 +03:00
, m_announceToAllTiers ( BITTORRENT_SESSION_KEY ( " AnnounceToAllTiers " ) , true )
2018-06-18 19:10:57 +03:00
, m_asyncIOThreads ( BITTORRENT_SESSION_KEY ( " AsyncIOThreadsCount " ) , 4 )
2019-07-03 12:44:16 +03:00
, m_filePoolSize ( BITTORRENT_SESSION_KEY ( " FilePoolSize " ) , 40 )
2018-08-28 13:18:07 +03:00
, m_checkingMemUsage ( BITTORRENT_SESSION_KEY ( " CheckingMemUsageSize " ) , 16 )
2017-09-10 16:04:40 +03:00
, m_diskCacheSize ( BITTORRENT_SESSION_KEY ( " DiskCacheSize " ) , 64 )
2016-05-01 11:05:52 +03:00
, m_diskCacheTTL ( BITTORRENT_SESSION_KEY ( " DiskCacheTTL " ) , 60 )
, m_useOSCache ( BITTORRENT_SESSION_KEY ( " UseOSCache " ) , true )
2018-01-29 12:03:14 +03:00
# ifdef Q_OS_WIN
, m_coalesceReadWriteEnabled ( BITTORRENT_SESSION_KEY ( " CoalesceReadWrite " ) , true )
# else
, m_coalesceReadWriteEnabled ( BITTORRENT_SESSION_KEY ( " CoalesceReadWrite " ) , false )
# endif
2017-08-11 14:15:18 +03:00
, m_isSuggestMode ( BITTORRENT_SESSION_KEY ( " SuggestMode " ) , false )
2017-08-11 14:48:58 +03:00
, m_sendBufferWatermark ( BITTORRENT_SESSION_KEY ( " SendBufferWatermark " ) , 500 )
, m_sendBufferLowWatermark ( BITTORRENT_SESSION_KEY ( " SendBufferLowWatermark " ) , 10 )
, m_sendBufferWatermarkFactor ( BITTORRENT_SESSION_KEY ( " SendBufferWatermarkFactor " ) , 50 )
2019-07-02 07:03:44 +03:00
, m_socketBacklogSize ( BITTORRENT_SESSION_KEY ( " SocketBacklogSize " ) , 30 )
2016-05-01 11:05:52 +03:00
, m_isAnonymousModeEnabled ( BITTORRENT_SESSION_KEY ( " AnonymousModeEnabled " ) , false )
, m_isQueueingEnabled ( BITTORRENT_SESSION_KEY ( " QueueingSystemEnabled " ) , true )
2016-10-30 00:10:42 +03:00
, m_maxActiveDownloads ( BITTORRENT_SESSION_KEY ( " MaxActiveDownloads " ) , 3 , lowerLimited ( - 1 ) )
, m_maxActiveUploads ( BITTORRENT_SESSION_KEY ( " MaxActiveUploads " ) , 3 , lowerLimited ( - 1 ) )
, m_maxActiveTorrents ( BITTORRENT_SESSION_KEY ( " MaxActiveTorrents " ) , 5 , lowerLimited ( - 1 ) )
2016-05-01 11:05:52 +03:00
, m_ignoreSlowTorrentsForQueueing ( BITTORRENT_SESSION_KEY ( " IgnoreSlowTorrentsForQueueing " ) , false )
2018-02-02 14:22:24 +03:00
, m_downloadRateForSlowTorrents ( BITTORRENT_SESSION_KEY ( " SlowTorrentsDownloadRate " ) , 2 )
, m_uploadRateForSlowTorrents ( BITTORRENT_SESSION_KEY ( " SlowTorrentsUploadRate " ) , 2 )
, m_slowTorrentsInactivityTimer ( BITTORRENT_SESSION_KEY ( " SlowTorrentsInactivityTimer " ) , 60 )
2016-05-01 11:05:52 +03:00
, m_outgoingPortsMin ( BITTORRENT_SESSION_KEY ( " OutgoingPortsMin " ) , 0 )
, m_outgoingPortsMax ( BITTORRENT_SESSION_KEY ( " OutgoingPortsMax " ) , 0 )
, m_ignoreLimitsOnLAN ( BITTORRENT_SESSION_KEY ( " IgnoreLimitsOnLAN " ) , true )
, m_includeOverheadInLimits ( BITTORRENT_SESSION_KEY ( " IncludeOverheadInLimits " ) , false )
2016-10-31 02:22:11 +03:00
, m_announceIP ( BITTORRENT_SESSION_KEY ( " AnnounceIP " ) )
2016-05-01 11:05:52 +03:00
, m_isSuperSeedingEnabled ( BITTORRENT_SESSION_KEY ( " SuperSeedingEnabled " ) , false )
, m_maxConnections ( BITTORRENT_SESSION_KEY ( " MaxConnections " ) , 500 , lowerLimited ( 0 , - 1 ) )
, m_maxUploads ( BITTORRENT_SESSION_KEY ( " MaxUploads " ) , - 1 , lowerLimited ( 0 , - 1 ) )
, m_maxConnectionsPerTorrent ( BITTORRENT_SESSION_KEY ( " MaxConnectionsPerTorrent " ) , 100 , lowerLimited ( 0 , - 1 ) )
, m_maxUploadsPerTorrent ( BITTORRENT_SESSION_KEY ( " MaxUploadsPerTorrent " ) , - 1 , lowerLimited ( 0 , - 1 ) )
2017-09-13 23:29:54 +03:00
, m_btProtocol ( BITTORRENT_SESSION_KEY ( " BTProtocol " ) , BTProtocol : : Both
, clampValue ( BTProtocol : : Both , BTProtocol : : UTP ) )
2016-05-01 11:05:52 +03:00
, m_isUTPRateLimited ( BITTORRENT_SESSION_KEY ( " uTPRateLimited " ) , true )
2017-11-22 17:37:08 +03:00
, m_utpMixedMode ( BITTORRENT_SESSION_KEY ( " uTPMixedMode " ) , MixedModeAlgorithm : : TCP
2017-08-15 21:23:07 +03:00
, clampValue ( MixedModeAlgorithm : : TCP , MixedModeAlgorithm : : Proportional ) )
2017-08-11 11:37:32 +03:00
, m_multiConnectionsPerIpEnabled ( BITTORRENT_SESSION_KEY ( " MultiConnectionsPerIp " ) , false )
2016-05-01 11:05:52 +03:00
, m_isAddTrackersEnabled ( BITTORRENT_SESSION_KEY ( " AddTrackersEnabled " ) , false )
, m_additionalTrackers ( BITTORRENT_SESSION_KEY ( " AdditionalTrackers " ) )
, m_globalMaxRatio ( BITTORRENT_SESSION_KEY ( " GlobalMaxRatio " ) , - 1 , [ ] ( qreal r ) { return r < 0 ? - 1. : r ; } )
2016-02-07 20:31:50 +03:00
, m_globalMaxSeedingMinutes ( BITTORRENT_SESSION_KEY ( " GlobalMaxSeedingMinutes " ) , - 1 , lowerLimited ( - 1 ) )
2016-05-01 11:05:52 +03:00
, m_isAddTorrentPaused ( BITTORRENT_SESSION_KEY ( " AddTorrentPaused " ) , false )
2017-04-15 17:21:24 +03:00
, m_isCreateTorrentSubfolder ( BITTORRENT_SESSION_KEY ( " CreateTorrentSubfolder " ) , true )
2016-05-01 11:05:52 +03:00
, m_isAppendExtensionEnabled ( BITTORRENT_SESSION_KEY ( " AddExtensionToIncompleteFiles " ) , false )
, m_refreshInterval ( BITTORRENT_SESSION_KEY ( " RefreshInterval " ) , 1500 )
, m_isPreallocationEnabled ( BITTORRENT_SESSION_KEY ( " Preallocation " ) , false )
, m_torrentExportDirectory ( BITTORRENT_SESSION_KEY ( " TorrentExportDirectory " ) )
, m_finishedTorrentExportDirectory ( BITTORRENT_SESSION_KEY ( " FinishedTorrentExportDirectory " ) )
, m_globalDownloadSpeedLimit ( BITTORRENT_SESSION_KEY ( " GlobalDLSpeedLimit " ) , 0 , lowerLimited ( 0 ) )
, m_globalUploadSpeedLimit ( BITTORRENT_SESSION_KEY ( " GlobalUPSpeedLimit " ) , 0 , lowerLimited ( 0 ) )
, m_altGlobalDownloadSpeedLimit ( BITTORRENT_SESSION_KEY ( " AlternativeGlobalDLSpeedLimit " ) , 10 , lowerLimited ( 0 ) )
, m_altGlobalUploadSpeedLimit ( BITTORRENT_SESSION_KEY ( " AlternativeGlobalUPSpeedLimit " ) , 10 , lowerLimited ( 0 ) )
, m_isAltGlobalSpeedLimitEnabled ( BITTORRENT_SESSION_KEY ( " UseAlternativeGlobalSpeedLimit " ) , false )
, m_isBandwidthSchedulerEnabled ( BITTORRENT_SESSION_KEY ( " BandwidthSchedulerEnabled " ) , false )
2018-07-04 07:58:09 +03:00
, m_saveResumeDataInterval ( BITTORRENT_SESSION_KEY ( " SaveResumeDataInterval " ) , 60 )
2019-04-22 06:02:31 +03:00
, m_port ( BITTORRENT_SESSION_KEY ( " Port " ) , - 1 )
2016-05-01 11:05:52 +03:00
, m_useRandomPort ( BITTORRENT_SESSION_KEY ( " UseRandomPort " ) , false )
, m_networkInterface ( BITTORRENT_SESSION_KEY ( " Interface " ) )
2016-10-31 02:40:26 +03:00
, m_networkInterfaceName ( BITTORRENT_SESSION_KEY ( " InterfaceName " ) )
2016-05-01 11:05:52 +03:00
, m_networkInterfaceAddress ( BITTORRENT_SESSION_KEY ( " InterfaceAddress " ) )
, m_isIPv6Enabled ( BITTORRENT_SESSION_KEY ( " IPv6Enabled " ) , false )
, m_encryption ( BITTORRENT_SESSION_KEY ( " Encryption " ) , 0 )
, m_isProxyPeerConnectionsEnabled ( BITTORRENT_SESSION_KEY ( " ProxyPeerConnections " ) , false )
2017-08-15 21:23:07 +03:00
, m_chokingAlgorithm ( BITTORRENT_SESSION_KEY ( " ChokingAlgorithm " ) , ChokingAlgorithm : : FixedSlots
, clampValue ( ChokingAlgorithm : : FixedSlots , ChokingAlgorithm : : RateBased ) )
, m_seedChokingAlgorithm ( BITTORRENT_SESSION_KEY ( " SeedChokingAlgorithm " ) , SeedChokingAlgorithm : : FastestUpload
, clampValue ( SeedChokingAlgorithm : : RoundRobin , SeedChokingAlgorithm : : AntiLeech ) )
2016-05-01 11:05:52 +03:00
, m_storedCategories ( BITTORRENT_SESSION_KEY ( " Categories " ) )
2017-06-05 03:22:17 +03:00
, m_storedTags ( BITTORRENT_SESSION_KEY ( " Tags " ) )
2016-05-01 11:05:52 +03:00
, m_maxRatioAction ( BITTORRENT_SESSION_KEY ( " MaxRatioAction " ) , Pause )
2016-05-11 14:25:29 +03:00
, m_defaultSavePath ( BITTORRENT_SESSION_KEY ( " DefaultSavePath " ) , specialFolderLocation ( SpecialFolder : : Downloads ) , normalizePath )
2016-05-01 11:05:52 +03:00
, m_tempPath ( BITTORRENT_SESSION_KEY ( " TempPath " ) , defaultSavePath ( ) + " temp/ " , normalizePath )
, m_isSubcategoriesEnabled ( BITTORRENT_SESSION_KEY ( " SubcategoriesEnabled " ) , false )
, m_isTempPathEnabled ( BITTORRENT_SESSION_KEY ( " TempPathEnabled " ) , false )
, m_isAutoTMMDisabledByDefault ( BITTORRENT_SESSION_KEY ( " DisableAutoTMMByDefault " ) , true )
, m_isDisableAutoTMMWhenCategoryChanged ( BITTORRENT_SESSION_KEY ( " DisableAutoTMMTriggers/CategoryChanged " ) , false )
, m_isDisableAutoTMMWhenDefaultSavePathChanged ( BITTORRENT_SESSION_KEY ( " DisableAutoTMMTriggers/DefaultSavePathChanged " ) , true )
, m_isDisableAutoTMMWhenCategorySavePathChanged ( BITTORRENT_SESSION_KEY ( " DisableAutoTMMTriggers/CategorySavePathChanged " ) , true )
, m_isTrackerEnabled ( BITTORRENT_KEY ( " TrackerEnabled " ) , false )
2017-03-06 15:40:34 +03:00
, m_bannedIPs ( " State/BannedIPs "
, QStringList ( )
, [ ] ( const QStringList & value )
{
QStringList tmp = value ;
tmp . sort ( ) ;
return tmp ;
}
)
2019-09-28 09:21:53 +03:00
# if defined(Q_OS_WIN)
, m_OSMemoryPriority ( BITTORRENT_KEY ( " OSMemoryPriority " ) , OSMemoryPriority : : BelowNormal )
# endif
2019-10-01 10:33:11 +03:00
, m_resumeFolderLock { new QFile { this } }
, m_refreshTimer { new QTimer { this } }
, m_seedingLimitTimer { new QTimer { this } }
, m_resumeDataTimer { new QTimer { this } }
, m_statistics { new Statistics { this } }
, m_ioThread { new QThread { this } }
, m_recentErroredTorrentsTimer { new QTimer { this } }
, m_networkManager { new QNetworkConfigurationManager { this } }
2015-04-19 18:17:47 +03:00
{
2019-04-23 06:33:33 +03:00
if ( port ( ) < 0 )
2019-04-22 06:02:31 +03:00
m_port = Utils : : Random : : rand ( 1024 , 65535 ) ;
2015-04-19 18:17:47 +03:00
initResumeFolder ( ) ;
2018-05-21 01:09:58 +03:00
m_recentErroredTorrentsTimer - > setSingleShot ( true ) ;
m_recentErroredTorrentsTimer - > setInterval ( 1000 ) ;
2019-10-01 10:33:11 +03:00
connect ( m_recentErroredTorrentsTimer , & QTimer : : timeout
, this , [ this ] ( ) { m_recentErroredTorrents . clear ( ) ; } ) ;
2018-05-21 01:09:58 +03:00
2016-02-07 20:31:50 +03:00
m_seedingLimitTimer - > setInterval ( 10000 ) ;
2017-10-30 18:59:13 +03:00
connect ( m_seedingLimitTimer , & QTimer : : timeout , this , & Session : : processShareLimits ) ;
2015-04-19 18:17:47 +03:00
2019-10-01 10:05:26 +03:00
initializeNativeSession ( ) ;
configureComponents ( ) ;
2016-06-03 17:03:17 +03:00
2017-08-20 18:00:23 +03:00
if ( isBandwidthSchedulerEnabled ( ) )
enableBandwidthScheduler ( ) ;
2016-05-01 11:05:52 +03:00
m_categories = map_cast ( m_storedCategories ) ;
2016-02-09 11:56:48 +03:00
if ( isSubcategoriesEnabled ( ) ) {
// if subcategories support changed manually
m_categories = expandCategories ( m_categories ) ;
2016-05-01 11:05:52 +03:00
m_storedCategories = map_cast ( m_categories ) ;
2016-02-09 11:56:48 +03:00
}
2017-06-05 03:22:17 +03:00
m_tags = QSet < QString > : : fromList ( m_storedTags . value ( ) ) ;
2016-05-01 11:05:52 +03:00
m_refreshTimer - > setInterval ( refreshInterval ( ) ) ;
2017-10-30 18:59:13 +03:00
connect ( m_refreshTimer , & QTimer : : timeout , this , & Session : : refresh ) ;
2015-04-19 18:17:47 +03:00
m_refreshTimer - > start ( ) ;
2016-02-07 20:31:50 +03:00
updateSeedingLimitTimer ( ) ;
2016-05-01 11:05:52 +03:00
populateAdditionalTrackers ( ) ;
enableTracker ( isTrackerEnabled ( ) ) ;
2019-10-01 10:33:11 +03:00
connect ( Net : : ProxyConfigurationManager : : instance ( )
, & Net : : ProxyConfigurationManager : : proxyConfigurationChanged
, this , & Session : : configureDeferred ) ;
2016-02-09 11:56:48 +03:00
2015-06-15 01:06:56 +03:00
// Network configuration monitor
2019-10-01 10:33:11 +03:00
connect ( m_networkManager , & QNetworkConfigurationManager : : onlineStateChanged , this , & Session : : networkOnlineStateChanged ) ;
connect ( m_networkManager , & QNetworkConfigurationManager : : configurationAdded , this , & Session : : networkConfigurationChange ) ;
connect ( m_networkManager , & QNetworkConfigurationManager : : configurationRemoved , this , & Session : : networkConfigurationChange ) ;
connect ( m_networkManager , & QNetworkConfigurationManager : : configurationChanged , this , & Session : : networkConfigurationChange ) ;
2015-06-15 01:06:56 +03:00
2018-11-19 12:53:29 +03:00
m_resumeDataSavingManager = new ResumeDataSavingManager { m_resumeFolderPath } ;
2015-12-13 15:38:19 +03:00
m_resumeDataSavingManager - > moveToThread ( m_ioThread ) ;
2017-10-30 18:59:13 +03:00
connect ( m_ioThread , & QThread : : finished , m_resumeDataSavingManager , & QObject : : deleteLater ) ;
2015-12-13 15:38:19 +03:00
m_ioThread - > start ( ) ;
2018-07-04 09:02:05 +03:00
// Regular saving of fastresume data
connect ( m_resumeDataTimer , & QTimer : : timeout , this , [ this ] ( ) { generateResumeData ( ) ; } ) ;
const uint saveInterval = saveResumeDataInterval ( ) ;
if ( saveInterval > 0 ) {
m_resumeDataTimer - > setInterval ( saveInterval * 60 * 1000 ) ;
m_resumeDataTimer - > start ( ) ;
}
2015-04-19 18:17:47 +03:00
// initialize PortForwarder instance
2019-03-06 08:58:07 +03:00
new PortForwarderImpl { m_nativeSession } ;
2015-04-19 18:17:47 +03:00
2017-04-29 14:52:28 +03:00
initMetrics ( ) ;
2015-04-19 18:17:47 +03:00
}
bool Session : : isDHTEnabled ( ) const
{
2016-05-01 11:05:52 +03:00
return m_isDHTEnabled ;
}
void Session : : setDHTEnabled ( bool enabled )
{
if ( enabled ! = m_isDHTEnabled ) {
m_isDHTEnabled = enabled ;
configureDeferred ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " DHT support [%1] " ) . arg ( enabled ? tr ( " ON " ) : tr ( " OFF " ) ) , Log : : INFO ) ;
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
}
bool Session : : isLSDEnabled ( ) const
{
2016-05-01 11:05:52 +03:00
return m_isLSDEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setLSDEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = m_isLSDEnabled ) {
m_isLSDEnabled = enabled ;
configureDeferred ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Local Peer Discovery support [%1] " ) . arg ( enabled ? tr ( " ON " ) : tr ( " OFF " ) )
, Log : : INFO ) ;
2016-05-01 11:05:52 +03:00
}
}
bool Session : : isPeXEnabled ( ) const
{
return m_isPeXEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setPeXEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
2016-10-31 03:06:29 +03:00
m_isPeXEnabled = enabled ;
if ( m_wasPexEnabled ! = enabled )
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Restart is required to toggle PeX support " ) , Log : : WARNING ) ;
2015-04-19 18:17:47 +03:00
}
bool Session : : isTempPathEnabled ( ) const
{
2016-05-01 11:05:52 +03:00
return m_isTempPathEnabled ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setTempPathEnabled ( const bool enabled )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( enabled ! = isTempPathEnabled ( ) ) {
m_isTempPathEnabled = enabled ;
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( m_torrents ) )
2016-05-01 11:05:52 +03:00
torrent - > handleTempPathChanged ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2016-02-09 11:56:48 +03:00
bool Session : : isAppendExtensionEnabled ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isAppendExtensionEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setAppendExtensionEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( isAppendExtensionEnabled ( ) ! = enabled ) {
// append or remove .!qB extension for incomplete files
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( m_torrents ) )
2016-05-01 11:05:52 +03:00
torrent - > handleAppendExtensionToggled ( ) ;
m_isAppendExtensionEnabled = enabled ;
}
}
uint Session : : refreshInterval ( ) const
{
return m_refreshInterval ;
}
2019-02-09 18:40:14 +03:00
void Session : : setRefreshInterval ( const uint value )
2016-05-01 11:05:52 +03:00
{
if ( value ! = refreshInterval ( ) ) {
m_refreshTimer - > setInterval ( value ) ;
m_refreshInterval = value ;
}
}
bool Session : : isPreallocationEnabled ( ) const
{
return m_isPreallocationEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setPreallocationEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
m_isPreallocationEnabled = enabled ;
}
QString Session : : torrentExportDirectory ( ) const
{
2019-06-16 20:14:15 +03:00
return Utils : : Fs : : toUniformPath ( m_torrentExportDirectory ) ;
2016-05-01 11:05:52 +03:00
}
2016-10-31 03:22:51 +03:00
void Session : : setTorrentExportDirectory ( QString path )
2016-05-01 11:05:52 +03:00
{
2019-06-16 20:14:15 +03:00
path = Utils : : Fs : : toUniformPath ( path ) ;
2016-10-31 03:22:51 +03:00
if ( path ! = torrentExportDirectory ( ) )
m_torrentExportDirectory = path ;
2016-05-01 11:05:52 +03:00
}
QString Session : : finishedTorrentExportDirectory ( ) const
{
2019-06-16 20:14:15 +03:00
return Utils : : Fs : : toUniformPath ( m_finishedTorrentExportDirectory ) ;
2016-05-01 11:05:52 +03:00
}
2016-10-31 03:22:51 +03:00
void Session : : setFinishedTorrentExportDirectory ( QString path )
2016-05-01 11:05:52 +03:00
{
2019-06-16 20:14:15 +03:00
path = Utils : : Fs : : toUniformPath ( path ) ;
2016-10-31 03:22:51 +03:00
if ( path ! = finishedTorrentExportDirectory ( ) )
m_finishedTorrentExportDirectory = path ;
2015-04-19 18:17:47 +03:00
}
QString Session : : defaultSavePath ( ) const
{
2019-06-16 20:14:15 +03:00
return Utils : : Fs : : toUniformPath ( m_defaultSavePath ) ;
2015-04-19 18:17:47 +03:00
}
QString Session : : tempPath ( ) const
{
2019-06-16 20:14:15 +03:00
return Utils : : Fs : : toUniformPath ( m_tempPath ) ;
2015-04-19 18:17:47 +03:00
}
2017-07-28 12:13:57 +03:00
QString Session : : torrentTempPath ( const TorrentInfo & torrentInfo ) const
2016-04-23 19:34:01 +03:00
{
2017-07-28 12:13:57 +03:00
if ( ( torrentInfo . filesCount ( ) > 1 ) & & ! torrentInfo . hasRootFolder ( ) )
return tempPath ( )
+ QString : : fromStdString ( torrentInfo . nativeInfo ( ) - > orig_files ( ) . name ( ) )
2018-07-21 08:28:13 +03:00
+ ' / ' ;
2017-07-28 12:13:57 +03:00
return tempPath ( ) ;
2016-04-23 19:34:01 +03:00
}
2016-02-09 11:56:48 +03:00
bool Session : : isValidCategoryName ( const QString & name )
{
2018-05-24 18:41:03 +03:00
static const QRegularExpression re ( R " (^([^ \\ \ /]|[^ \\ \ /]([^ \\ \ /]| \ /(?=[^ \ /]) ) * [ ^ \ \ \ / ] ) $ ) " ) ;
if ( ! name . isEmpty ( ) & & ( name . indexOf ( re ) ! = 0 ) ) {
2016-02-09 11:56:48 +03:00
qDebug ( ) < < " Incorrect category name: " < < name ;
return false ;
}
return true ;
}
QStringList Session : : expandCategory ( const QString & category )
{
QStringList result ;
if ( ! isValidCategoryName ( category ) )
return result ;
int index = 0 ;
while ( ( index = category . indexOf ( ' / ' , index ) ) > = 0 ) {
result < < category . left ( index ) ;
+ + index ;
}
result < < category ;
return result ;
}
2019-10-01 11:23:32 +03:00
QStringMap Session : : categories ( ) const
2016-02-09 11:56:48 +03:00
{
2017-09-24 14:54:42 +03:00
return m_categories ;
2016-02-09 11:56:48 +03:00
}
QString Session : : categorySavePath ( const QString & categoryName ) const
{
2019-02-09 18:40:14 +03:00
const QString basePath = m_defaultSavePath ;
2016-02-09 11:56:48 +03:00
if ( categoryName . isEmpty ( ) ) return basePath ;
2016-03-06 09:25:55 +03:00
QString path = m_categories . value ( categoryName ) ;
2016-02-09 11:56:48 +03:00
if ( path . isEmpty ( ) ) // use implicit save path
path = Utils : : Fs : : toValidFileSystemName ( categoryName , true ) ;
if ( ! QDir : : isAbsolutePath ( path ) )
2016-03-06 09:25:55 +03:00
path . prepend ( basePath ) ;
2016-02-09 11:56:48 +03:00
2016-03-06 09:25:55 +03:00
return normalizeSavePath ( path ) ;
2016-02-09 11:56:48 +03:00
}
bool Session : : addCategory ( const QString & name , const QString & savePath )
{
if ( name . isEmpty ( ) ) return false ;
if ( ! isValidCategoryName ( name ) | | m_categories . contains ( name ) )
return false ;
if ( isSubcategoriesEnabled ( ) ) {
2018-11-27 23:15:04 +03:00
for ( const QString & parent : asConst ( expandCategory ( name ) ) ) {
2016-02-09 11:56:48 +03:00
if ( ( parent ! = name ) & & ! m_categories . contains ( parent ) ) {
m_categories [ parent ] = " " ;
emit categoryAdded ( parent ) ;
}
}
}
m_categories [ name ] = savePath ;
2016-05-01 11:05:52 +03:00
m_storedCategories = map_cast ( m_categories ) ;
2016-02-09 11:56:48 +03:00
emit categoryAdded ( name ) ;
return true ;
}
bool Session : : editCategory ( const QString & name , const QString & savePath )
{
if ( ! m_categories . contains ( name ) ) return false ;
if ( categorySavePath ( name ) = = savePath ) return false ;
m_categories [ name ] = savePath ;
2017-09-24 14:54:42 +03:00
m_storedCategories = map_cast ( m_categories ) ;
2016-05-08 22:47:50 +03:00
if ( isDisableAutoTMMWhenCategorySavePathChanged ( ) ) {
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2016-02-09 11:56:48 +03:00
if ( torrent - > category ( ) = = name )
2016-05-08 22:47:50 +03:00
torrent - > setAutoTMMEnabled ( false ) ;
2016-02-09 11:56:48 +03:00
}
else {
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2016-02-09 11:56:48 +03:00
if ( torrent - > category ( ) = = name )
torrent - > handleCategorySavePathChanged ( ) ;
}
return true ;
}
bool Session : : removeCategory ( const QString & name )
{
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2016-02-09 11:56:48 +03:00
if ( torrent - > belongsToCategory ( name ) )
torrent - > setCategory ( " " ) ;
// remove stored category and its subcategories if exist
bool result = false ;
if ( isSubcategoriesEnabled ( ) ) {
// remove subcategories
2018-07-21 08:28:13 +03:00
const QString test = name + ' / ' ;
2019-05-12 08:26:06 +03:00
Algorithm : : removeIf ( m_categories , [ this , & test , & result ] ( const QString & category , const QString & )
2018-03-06 17:50:10 +03:00
{
2016-02-09 11:56:48 +03:00
if ( category . startsWith ( test ) ) {
result = true ;
emit categoryRemoved ( category ) ;
2018-03-06 17:50:10 +03:00
return true ;
2016-02-09 11:56:48 +03:00
}
2018-03-06 17:50:10 +03:00
return false ;
} ) ;
2016-02-09 11:56:48 +03:00
}
result = ( m_categories . remove ( name ) > 0 ) | | result ;
if ( result ) {
// update stored categories
2016-05-01 11:05:52 +03:00
m_storedCategories = map_cast ( m_categories ) ;
2016-02-09 11:56:48 +03:00
emit categoryRemoved ( name ) ;
}
return result ;
}
bool Session : : isSubcategoriesEnabled ( ) const
{
2016-05-01 11:05:52 +03:00
return m_isSubcategoriesEnabled ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setSubcategoriesEnabled ( const bool value )
2016-02-09 11:56:48 +03:00
{
if ( isSubcategoriesEnabled ( ) = = value ) return ;
if ( value ) {
// expand categories to include all parent categories
m_categories = expandCategories ( m_categories ) ;
// update stored categories
2016-05-01 11:05:52 +03:00
m_storedCategories = map_cast ( m_categories ) ;
2016-02-09 11:56:48 +03:00
}
else {
// reload categories
2016-05-01 11:05:52 +03:00
m_categories = map_cast ( m_storedCategories ) ;
2016-02-09 11:56:48 +03:00
}
2016-05-01 11:05:52 +03:00
m_isSubcategoriesEnabled = value ;
2016-02-09 11:56:48 +03:00
emit subcategoriesSupportChanged ( ) ;
}
2017-06-05 03:22:17 +03:00
QSet < QString > Session : : tags ( ) const
{
return m_tags ;
}
bool Session : : isValidTag ( const QString & tag )
{
return ( ! tag . trimmed ( ) . isEmpty ( ) & & ! tag . contains ( ' , ' ) ) ;
}
bool Session : : hasTag ( const QString & tag ) const
{
return m_tags . contains ( tag ) ;
}
bool Session : : addTag ( const QString & tag )
{
if ( ! isValidTag ( tag ) )
return false ;
if ( ! hasTag ( tag ) ) {
m_tags . insert ( tag ) ;
2019-10-31 07:38:11 +03:00
m_storedTags = m_tags . values ( ) ;
2017-06-05 03:22:17 +03:00
emit tagAdded ( tag ) ;
return true ;
}
return false ;
}
bool Session : : removeTag ( const QString & tag )
{
if ( m_tags . remove ( tag ) ) {
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2017-06-05 03:22:17 +03:00
torrent - > removeTag ( tag ) ;
2019-10-31 07:38:11 +03:00
m_storedTags = m_tags . values ( ) ;
2017-06-05 03:22:17 +03:00
emit tagRemoved ( tag ) ;
return true ;
}
return false ;
}
2016-05-08 22:47:50 +03:00
bool Session : : isAutoTMMDisabledByDefault ( ) const
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isAutoTMMDisabledByDefault ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setAutoTMMDisabledByDefault ( const bool value )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_isAutoTMMDisabledByDefault = value ;
2016-02-09 11:56:48 +03:00
}
2016-05-08 22:47:50 +03:00
bool Session : : isDisableAutoTMMWhenCategoryChanged ( ) const
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isDisableAutoTMMWhenCategoryChanged ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setDisableAutoTMMWhenCategoryChanged ( const bool value )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_isDisableAutoTMMWhenCategoryChanged = value ;
2016-02-09 11:56:48 +03:00
}
2016-05-08 22:47:50 +03:00
bool Session : : isDisableAutoTMMWhenDefaultSavePathChanged ( ) const
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isDisableAutoTMMWhenDefaultSavePathChanged ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setDisableAutoTMMWhenDefaultSavePathChanged ( const bool value )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_isDisableAutoTMMWhenDefaultSavePathChanged = value ;
2016-02-09 11:56:48 +03:00
}
2016-05-08 22:47:50 +03:00
bool Session : : isDisableAutoTMMWhenCategorySavePathChanged ( ) const
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isDisableAutoTMMWhenCategorySavePathChanged ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setDisableAutoTMMWhenCategorySavePathChanged ( const bool value )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_isDisableAutoTMMWhenCategorySavePathChanged = value ;
2016-02-09 11:56:48 +03:00
}
bool Session : : isAddTorrentPaused ( ) const
{
2016-05-01 11:05:52 +03:00
return m_isAddTorrentPaused ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setAddTorrentPaused ( const bool value )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_isAddTorrentPaused = value ;
}
bool Session : : isTrackerEnabled ( ) const
{
return m_isTrackerEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setTrackerEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
2019-08-08 18:21:07 +03:00
if ( m_isTrackerEnabled ! = enabled )
2016-05-01 11:05:52 +03:00
m_isTrackerEnabled = enabled ;
2019-08-08 18:21:07 +03:00
// call enableTracker() unconditionally, otherwise port change won't trigger
// tracker restart
enableTracker ( enabled ) ;
2016-02-09 11:56:48 +03:00
}
2015-04-19 18:17:47 +03:00
qreal Session : : globalMaxRatio ( ) const
{
return m_globalMaxRatio ;
}
2018-04-14 22:53:45 +03:00
// Torrents with a ratio superior to the given value will
2016-05-01 11:05:52 +03:00
// be automatically deleted
void Session : : setGlobalMaxRatio ( qreal ratio )
{
if ( ratio < 0 )
ratio = - 1. ;
if ( ratio ! = globalMaxRatio ( ) ) {
m_globalMaxRatio = ratio ;
2016-02-07 20:31:50 +03:00
updateSeedingLimitTimer ( ) ;
}
}
int Session : : globalMaxSeedingMinutes ( ) const
{
return m_globalMaxSeedingMinutes ;
}
void Session : : setGlobalMaxSeedingMinutes ( int minutes )
{
if ( minutes < 0 )
minutes = - 1 ;
if ( minutes ! = globalMaxSeedingMinutes ( ) ) {
m_globalMaxSeedingMinutes = minutes ;
updateSeedingLimitTimer ( ) ;
2016-05-01 11:05:52 +03:00
}
}
2015-04-19 18:17:47 +03:00
// Main destructor
Session : : ~ Session ( )
{
// Do some BT related saving
saveResumeData ( ) ;
// We must delete FilterParserThread
2019-05-09 07:45:52 +03:00
// before we delete lt::session
2015-04-19 18:17:47 +03:00
if ( m_filterParser )
delete m_filterParser ;
// We must delete PortForwarderImpl before
2019-05-09 07:45:52 +03:00
// we delete lt::session
2019-03-06 08:58:07 +03:00
delete Net : : PortForwarder : : instance ( ) ;
2015-04-19 18:17:47 +03:00
qDebug ( " Deleting the session " ) ;
delete m_nativeSession ;
2015-12-13 15:38:19 +03:00
m_ioThread - > quit ( ) ;
m_ioThread - > wait ( ) ;
2019-10-01 10:33:11 +03:00
m_resumeFolderLock - > close ( ) ;
m_resumeFolderLock - > remove ( ) ;
2015-04-19 18:17:47 +03:00
}
void Session : : initInstance ( )
{
2017-08-20 18:00:23 +03:00
if ( ! m_instance )
2015-04-19 18:17:47 +03:00
m_instance = new Session ;
}
void Session : : freeInstance ( )
{
if ( m_instance ) {
delete m_instance ;
2018-04-15 13:06:31 +03:00
m_instance = nullptr ;
2015-04-19 18:17:47 +03:00
}
}
Session * Session : : instance ( )
{
return m_instance ;
}
2016-05-01 11:05:52 +03:00
void Session : : adjustLimits ( )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( isQueueingSystemEnabled ( ) ) {
2019-05-07 06:22:39 +03:00
lt : : settings_pack settingsPack = m_nativeSession - > get_settings ( ) ;
2016-06-03 17:03:17 +03:00
adjustLimits ( settingsPack ) ;
m_nativeSession - > apply_settings ( settingsPack ) ;
2015-04-19 18:17:47 +03:00
}
2016-06-03 17:03:17 +03:00
}
2015-04-19 18:17:47 +03:00
2017-08-20 18:00:23 +03:00
void Session : : applyBandwidthLimits ( )
{
2019-05-07 06:22:39 +03:00
lt : : settings_pack settingsPack = m_nativeSession - > get_settings ( ) ;
2017-08-20 18:00:23 +03:00
applyBandwidthLimits ( settingsPack ) ;
m_nativeSession - > apply_settings ( settingsPack ) ;
}
2016-06-03 17:03:17 +03:00
void Session : : configure ( )
{
2019-05-07 06:22:39 +03:00
lt : : settings_pack settingsPack = m_nativeSession - > get_settings ( ) ;
2019-10-01 10:05:26 +03:00
loadLTSettings ( settingsPack ) ;
2016-06-03 17:03:17 +03:00
m_nativeSession - > apply_settings ( settingsPack ) ;
2019-10-01 10:05:26 +03:00
configureComponents ( ) ;
m_deferredConfigureScheduled = false ;
}
void Session : : configureComponents ( )
{
// This function contains components/actions that:
// 1. Need to be setup at start up
// 2. When deferred configure is called
2017-05-01 19:19:34 +03:00
configurePeerClasses ( ) ;
2016-05-08 15:45:53 +03:00
2019-10-29 18:16:22 +03:00
if ( ! m_IPFilteringConfigured ) {
2016-10-29 19:14:27 +03:00
if ( isIPFilteringEnabled ( ) )
2016-06-03 17:03:17 +03:00
enableIPFilter ( ) ;
else
disableIPFilter ( ) ;
2019-10-29 18:16:22 +03:00
m_IPFilteringConfigured = true ;
2016-06-03 17:03:17 +03:00
}
2019-09-28 09:21:53 +03:00
# if defined(Q_OS_WIN)
applyOSMemoryPriority ( ) ;
# endif
2019-10-01 10:05:26 +03:00
}
2016-06-03 17:03:17 +03:00
2019-10-01 10:05:26 +03:00
void Session : : initializeNativeSession ( )
{
const LTAlertCategory alertMask = lt : : alert : : error_notification
| lt : : alert : : file_progress_notification
| lt : : alert : : ip_block_notification
| lt : : alert : : peer_notification
| lt : : alert : : performance_warning
| lt : : alert : : port_mapping_notification
| lt : : alert : : status_notification
| lt : : alert : : storage_notification
| lt : : alert : : tracker_notification ;
const std : : string peerId = lt : : generate_fingerprint ( PEER_ID , QBT_VERSION_MAJOR , QBT_VERSION_MINOR , QBT_VERSION_BUGFIX , QBT_VERSION_BUILD ) ;
lt : : settings_pack pack ;
pack . set_int ( lt : : settings_pack : : alert_mask , alertMask ) ;
pack . set_str ( lt : : settings_pack : : peer_fingerprint , peerId ) ;
pack . set_bool ( lt : : settings_pack : : listen_system_port_fallback , false ) ;
pack . set_str ( lt : : settings_pack : : user_agent , USER_AGENT ) ;
pack . set_bool ( lt : : settings_pack : : use_dht_as_fallback , false ) ;
// Speed up exit
pack . set_int ( lt : : settings_pack : : stop_tracker_timeout , 1 ) ;
pack . set_int ( lt : : settings_pack : : auto_scrape_interval , 1200 ) ; // 20 minutes
pack . set_int ( lt : : settings_pack : : auto_scrape_min_interval , 900 ) ; // 15 minutes
pack . set_int ( lt : : settings_pack : : connection_speed , 20 ) ; // default is 10
pack . set_bool ( lt : : settings_pack : : no_connect_privileged_ports , false ) ;
// libtorrent 1.1 enables UPnP & NAT-PMP by default
// turn them off before `lt::session` ctor to avoid split second effects
pack . set_bool ( lt : : settings_pack : : enable_upnp , false ) ;
pack . set_bool ( lt : : settings_pack : : enable_natpmp , false ) ;
pack . set_bool ( lt : : settings_pack : : upnp_ignore_nonrouters , true ) ;
# if (LIBTORRENT_VERSION_NUM < 10200)
// Disable support for SSL torrents for now
pack . set_int ( lt : : settings_pack : : ssl_listen , 0 ) ;
// To prevent ISPs from blocking seeding
pack . set_bool ( lt : : settings_pack : : lazy_bitfields , true ) ;
// Disk cache pool is rarely tested in libtorrent and doesn't free buffers
// Soon to be deprecated there
// More info: https://github.com/arvidn/libtorrent/issues/2251
pack . set_bool ( lt : : settings_pack : : use_disk_cache_pool , false ) ;
# endif
loadLTSettings ( pack ) ;
m_nativeSession = new lt : : session { pack , LTSessionFlags { 0 } } ;
LogMsg ( tr ( " Peer ID: " ) + QString : : fromStdString ( peerId ) ) ;
LogMsg ( tr ( " HTTP User-Agent is '%1' " ) . arg ( USER_AGENT ) ) ;
LogMsg ( tr ( " DHT support [%1] " ) . arg ( isDHTEnabled ( ) ? tr ( " ON " ) : tr ( " OFF " ) ) , Log : : INFO ) ;
LogMsg ( tr ( " Local Peer Discovery support [%1] " ) . arg ( isLSDEnabled ( ) ? tr ( " ON " ) : tr ( " OFF " ) ) , Log : : INFO ) ;
LogMsg ( tr ( " PeX support [%1] " ) . arg ( isPeXEnabled ( ) ? tr ( " ON " ) : tr ( " OFF " ) ) , Log : : INFO ) ;
LogMsg ( tr ( " Anonymous mode [%1] " ) . arg ( isAnonymousModeEnabled ( ) ? tr ( " ON " ) : tr ( " OFF " ) ) , Log : : INFO ) ;
LogMsg ( tr ( " Encryption support [%1] " ) . arg ( ( encryption ( ) = = 0 ) ? tr ( " ON " ) :
( ( encryption ( ) = = 1 ) ? tr ( " FORCED " ) : tr ( " OFF " ) ) ) , Log : : INFO ) ;
m_nativeSession - > set_alert_notify ( [ this ] ( )
{
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( this , & Session : : readAlerts , Qt : : QueuedConnection ) ;
# else
QMetaObject : : invokeMethod ( this , " readAlerts " , Qt : : QueuedConnection ) ;
# endif
} ) ;
// Enabling plugins
m_nativeSession - > add_extension ( & lt : : create_smart_ban_plugin ) ;
m_nativeSession - > add_extension ( & lt : : create_ut_metadata_plugin ) ;
if ( isPeXEnabled ( ) )
m_nativeSession - > add_extension ( & lt : : create_ut_pex_plugin ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : processBannedIPs ( lt : : ip_filter & filter )
2015-04-19 18:17:47 +03:00
{
2016-06-03 17:03:17 +03:00
// First, import current filter
2018-11-27 23:15:04 +03:00
for ( const QString & ip : asConst ( m_bannedIPs . value ( ) ) ) {
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-05-07 06:22:39 +03:00
const lt : : address addr = lt : : address : : from_string ( ip . toLatin1 ( ) . constData ( ) , ec ) ;
2016-06-03 17:03:17 +03:00
Q_ASSERT ( ! ec ) ;
2016-10-30 00:11:52 +03:00
if ( ! ec )
2019-05-07 06:22:39 +03:00
filter . add_rule ( addr , addr , lt : : ip_filter : : blocked ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-05-07 06:22:39 +03:00
void Session : : adjustLimits ( lt : : settings_pack & settingsPack )
2015-04-19 18:17:47 +03:00
{
2018-04-14 22:53:45 +03:00
// Internally increase the queue limits to ensure that the magnet is started
2019-02-09 18:40:14 +03:00
const int maxDownloads = maxActiveDownloads ( ) ;
const int maxActive = maxActiveTorrents ( ) ;
2015-09-16 21:57:50 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : active_downloads
2016-06-03 17:03:17 +03:00
, maxDownloads > - 1 ? maxDownloads + m_extraLimit : maxDownloads ) ;
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : active_limit
2016-06-03 17:03:17 +03:00
, maxActive > - 1 ? maxActive + m_extraLimit : maxActive ) ;
2015-04-19 18:17:47 +03:00
}
2019-06-02 09:33:46 +03:00
void Session : : applyBandwidthLimits ( lt : : settings_pack & settingsPack ) const
2017-08-20 18:00:23 +03:00
{
const bool altSpeedLimitEnabled = isAltGlobalSpeedLimitEnabled ( ) ;
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : download_rate_limit , altSpeedLimitEnabled ? altGlobalDownloadSpeedLimit ( ) : globalDownloadSpeedLimit ( ) ) ;
settingsPack . set_int ( lt : : settings_pack : : upload_rate_limit , altSpeedLimitEnabled ? altGlobalUploadSpeedLimit ( ) : globalUploadSpeedLimit ( ) ) ;
2017-08-20 18:00:23 +03:00
}
2017-04-29 14:52:28 +03:00
void Session : : initMetrics ( )
{
2019-05-07 06:22:39 +03:00
m_metricIndices . net . hasIncomingConnections = lt : : find_metric_idx ( " net.has_incoming_connections " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . hasIncomingConnections > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . sentPayloadBytes = lt : : find_metric_idx ( " net.sent_payload_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . sentPayloadBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvPayloadBytes = lt : : find_metric_idx ( " net.recv_payload_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvPayloadBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . sentBytes = lt : : find_metric_idx ( " net.sent_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . sentBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvBytes = lt : : find_metric_idx ( " net.recv_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . sentIPOverheadBytes = lt : : find_metric_idx ( " net.sent_ip_overhead_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . sentIPOverheadBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvIPOverheadBytes = lt : : find_metric_idx ( " net.recv_ip_overhead_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvIPOverheadBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . sentTrackerBytes = lt : : find_metric_idx ( " net.sent_tracker_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . sentTrackerBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvTrackerBytes = lt : : find_metric_idx ( " net.recv_tracker_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvTrackerBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvRedundantBytes = lt : : find_metric_idx ( " net.recv_redundant_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvRedundantBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . net . recvFailedBytes = lt : : find_metric_idx ( " net.recv_failed_bytes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . net . recvFailedBytes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . peer . numPeersConnected = lt : : find_metric_idx ( " peer.num_peers_connected " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . peer . numPeersConnected > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . peer . numPeersDownDisk = lt : : find_metric_idx ( " peer.num_peers_down_disk " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . peer . numPeersDownDisk > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . peer . numPeersUpDisk = lt : : find_metric_idx ( " peer.num_peers_up_disk " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . peer . numPeersUpDisk > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . dht . dhtBytesIn = lt : : find_metric_idx ( " dht.dht_bytes_in " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . dht . dhtBytesIn > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . dht . dhtBytesOut = lt : : find_metric_idx ( " dht.dht_bytes_out " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . dht . dhtBytesOut > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . dht . dhtNodes = lt : : find_metric_idx ( " dht.dht_nodes " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . dht . dhtNodes > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . diskBlocksInUse = lt : : find_metric_idx ( " disk.disk_blocks_in_use " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . disk . diskBlocksInUse > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . numBlocksRead = lt : : find_metric_idx ( " disk.num_blocks_read " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . disk . numBlocksRead > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . numBlocksCacheHits = lt : : find_metric_idx ( " disk.num_blocks_cache_hits " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . disk . numBlocksCacheHits > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . writeJobs = lt : : find_metric_idx ( " disk.num_write_ops " ) ;
2017-08-14 17:27:31 +03:00
Q_ASSERT ( m_metricIndices . disk . writeJobs > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . readJobs = lt : : find_metric_idx ( " disk.num_read_ops " ) ;
2017-08-14 17:27:31 +03:00
Q_ASSERT ( m_metricIndices . disk . readJobs > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . hashJobs = lt : : find_metric_idx ( " disk.num_blocks_hashed " ) ;
2017-08-14 17:27:31 +03:00
Q_ASSERT ( m_metricIndices . disk . hashJobs > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . queuedDiskJobs = lt : : find_metric_idx ( " disk.queued_disk_jobs " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . disk . queuedDiskJobs > = 0 ) ;
2019-05-07 06:22:39 +03:00
m_metricIndices . disk . diskJobTime = lt : : find_metric_idx ( " disk.disk_job_time " ) ;
2017-04-29 14:52:28 +03:00
Q_ASSERT ( m_metricIndices . disk . diskJobTime > = 0 ) ;
}
2019-10-01 10:05:26 +03:00
void Session : : loadLTSettings ( lt : : settings_pack & settingsPack )
2015-04-19 18:17:47 +03:00
{
2019-07-02 07:03:44 +03:00
// from libtorrent doc:
// It will not take affect until the listen_interfaces settings is updated
settingsPack . set_int ( lt : : settings_pack : : listen_queue_size , socketBacklogSize ( ) ) ;
2017-08-13 00:58:22 +03:00
# ifdef Q_OS_WIN
QString chosenIP ;
# endif
2019-10-29 18:16:22 +03:00
if ( ! m_listenInterfaceConfigured ) {
2019-04-23 06:33:33 +03:00
const int port = useRandomPort ( ) ? 0 : this - > port ( ) ;
if ( port > 0 ) // user specified port
settingsPack . set_int ( lt : : settings_pack : : max_retry_port_bind , 0 ) ;
2019-10-16 18:38:51 +03:00
for ( const QString & ip : asConst ( getListeningIPs ( ) ) ) {
2016-06-03 17:03:17 +03:00
if ( ip . isEmpty ( ) ) {
2019-10-16 18:38:51 +03:00
const QString anyIP = QHostAddress ( QHostAddress : : AnyIPv4 ) . toString ( ) ;
const std : : string endpoint = anyIP . toStdString ( ) + ' : ' + std : : to_string ( port ) ;
settingsPack . set_str ( lt : : settings_pack : : listen_interfaces , endpoint ) ;
LogMsg ( tr ( " Trying to listen on IP: %1, port: %2 "
, " e.g: Trying to listen on IP: 192.168.0.1, port: 6881 " )
. arg ( anyIP , QString : : number ( port ) )
2019-07-02 06:49:35 +03:00
, Log : : INFO ) ;
2016-06-03 17:03:17 +03:00
break ;
}
2019-10-16 18:38:51 +03:00
lt : : error_code ec ;
const lt : : address addr = lt : : address : : from_string ( ip . toStdString ( ) , ec ) ;
2016-06-03 17:03:17 +03:00
if ( ! ec ) {
2019-10-16 18:38:51 +03:00
const std : : string endpoint = ( addr . is_v6 ( )
? ( ' [ ' + addr . to_string ( ) + ' ] ' )
: addr . to_string ( ) )
+ ' : ' + std : : to_string ( port ) ;
settingsPack . set_str ( lt : : settings_pack : : listen_interfaces , endpoint ) ;
LogMsg ( tr ( " Trying to listen on IP: %1, port: %2 "
, " e.g: Trying to listen on IP: 192.168.0.1, port: 6881 " )
. arg ( ip , QString : : number ( port ) )
2019-07-02 06:49:35 +03:00
, Log : : INFO ) ;
2017-08-13 00:58:22 +03:00
# ifdef Q_OS_WIN
chosenIP = ip ;
# endif
2016-06-03 17:03:17 +03:00
break ;
}
}
2015-04-19 18:17:47 +03:00
2017-08-13 00:58:22 +03:00
# ifdef Q_OS_WIN
// On Vista+ versions and after Qt 5.5 QNetworkInterface::name() returns
2019-10-16 18:38:51 +03:00
// the interface's LUID and not the GUID.
2017-08-13 00:58:22 +03:00
// Libtorrent expects GUIDs for the 'outgoing_interfaces' setting.
2019-10-16 18:38:51 +03:00
const QString netInterface = networkInterface ( ) ;
if ( ! netInterface . isEmpty ( ) ) {
const QString guid = convertIfaceNameToGuid ( netInterface ) ;
2017-08-13 00:58:22 +03:00
if ( ! guid . isEmpty ( ) ) {
2019-05-07 06:22:39 +03:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , guid . toStdString ( ) ) ;
2017-08-13 00:58:22 +03:00
}
else {
2019-05-07 06:22:39 +03:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , chosenIP . toStdString ( ) ) ;
2019-10-16 18:38:51 +03:00
LogMsg ( tr ( " Could not get GUID of configured network interface. Binding to IP: %1 " ) . arg ( chosenIP )
2017-08-13 00:58:22 +03:00
, Log : : WARNING ) ;
}
}
# else
2019-05-07 06:22:39 +03:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , networkInterface ( ) . toStdString ( ) ) ;
2018-11-06 18:49:17 +03:00
# endif // Q_OS_WIN
2019-10-16 18:38:51 +03:00
2019-10-29 18:16:22 +03:00
m_listenInterfaceConfigured = true ;
2015-04-19 18:17:47 +03:00
}
2017-08-20 18:00:23 +03:00
applyBandwidthLimits ( settingsPack ) ;
2015-04-19 18:17:47 +03:00
2016-06-03 17:03:17 +03:00
// The most secure, rc4 only so that all streams are encrypted
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : allowed_enc_level , lt : : settings_pack : : pe_rc4 ) ;
settingsPack . set_bool ( lt : : settings_pack : : prefer_rc4 , true ) ;
2016-06-03 17:03:17 +03:00
switch ( encryption ( ) ) {
2018-04-14 22:53:45 +03:00
case 0 : // Enabled
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : out_enc_policy , lt : : settings_pack : : pe_enabled ) ;
settingsPack . set_int ( lt : : settings_pack : : in_enc_policy , lt : : settings_pack : : pe_enabled ) ;
2016-06-03 17:03:17 +03:00
break ;
case 1 : // Forced
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : out_enc_policy , lt : : settings_pack : : pe_forced ) ;
settingsPack . set_int ( lt : : settings_pack : : in_enc_policy , lt : : settings_pack : : pe_forced ) ;
2016-06-03 17:03:17 +03:00
break ;
default : // Disabled
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : out_enc_policy , lt : : settings_pack : : pe_disabled ) ;
settingsPack . set_int ( lt : : settings_pack : : in_enc_policy , lt : : settings_pack : : pe_disabled ) ;
2016-06-03 17:03:17 +03:00
}
2019-06-16 12:50:55 +03:00
// proxy
2019-02-09 18:40:14 +03:00
const auto proxyManager = Net : : ProxyConfigurationManager : : instance ( ) ;
const Net : : ProxyConfiguration proxyConfig = proxyManager - > proxyConfiguration ( ) ;
2015-04-19 18:17:47 +03:00
2019-06-16 12:50:55 +03:00
switch ( proxyConfig . type ) {
case Net : : ProxyType : : HTTP :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : http ) ;
break ;
case Net : : ProxyType : : HTTP_PW :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : http_pw ) ;
break ;
case Net : : ProxyType : : SOCKS4 :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : socks4 ) ;
break ;
case Net : : ProxyType : : SOCKS5 :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : socks5 ) ;
break ;
case Net : : ProxyType : : SOCKS5_PW :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : socks5_pw ) ;
break ;
case Net : : ProxyType : : None :
default :
settingsPack . set_int ( lt : : settings_pack : : proxy_type , lt : : settings_pack : : none ) ;
}
if ( proxyConfig . type ! = Net : : ProxyType : : None ) {
settingsPack . set_str ( lt : : settings_pack : : proxy_hostname , proxyConfig . ip . toStdString ( ) ) ;
settingsPack . set_int ( lt : : settings_pack : : proxy_port , proxyConfig . port ) ;
if ( proxyManager - > isAuthenticationRequired ( ) ) {
settingsPack . set_str ( lt : : settings_pack : : proxy_username , proxyConfig . username . toStdString ( ) ) ;
settingsPack . set_str ( lt : : settings_pack : : proxy_password , proxyConfig . password . toStdString ( ) ) ;
2016-06-03 17:03:17 +03:00
}
2015-04-19 18:17:47 +03:00
2019-06-16 12:50:55 +03:00
settingsPack . set_bool ( lt : : settings_pack : : proxy_peer_connections , isProxyPeerConnectionsEnabled ( ) ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : announce_to_all_trackers , announceToAllTrackers ( ) ) ;
settingsPack . set_bool ( lt : : settings_pack : : announce_to_all_tiers , announceToAllTiers ( ) ) ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : aio_threads , asyncIOThreads ( ) ) ;
2018-06-18 19:10:57 +03:00
2019-07-03 12:44:16 +03:00
settingsPack . set_int ( lt : : settings_pack : : file_pool_size , filePoolSize ( ) ) ;
2018-08-28 13:18:07 +03:00
const int checkingMemUsageSize = checkingMemUsage ( ) * 64 ;
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : checking_mem_usage , checkingMemUsageSize ) ;
2018-08-28 13:18:07 +03:00
2018-04-14 22:53:45 +03:00
const int cacheSize = ( diskCacheSize ( ) > - 1 ) ? ( diskCacheSize ( ) * 64 ) : - 1 ;
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : cache_size , cacheSize ) ;
settingsPack . set_int ( lt : : settings_pack : : cache_expiry , diskCacheTTL ( ) ) ;
2016-06-03 17:03:17 +03:00
qDebug ( ) < < " Using a disk cache size of " < < cacheSize < < " MiB " ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
lt : : settings_pack : : io_buffer_mode_t mode = useOSCache ( ) ? lt : : settings_pack : : enable_os_cache
: lt : : settings_pack : : disable_os_cache ;
settingsPack . set_int ( lt : : settings_pack : : disk_io_read_mode , mode ) ;
settingsPack . set_int ( lt : : settings_pack : : disk_io_write_mode , mode ) ;
2018-01-29 12:03:14 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : coalesce_reads , isCoalesceReadWriteEnabled ( ) ) ;
settingsPack . set_bool ( lt : : settings_pack : : coalesce_writes , isCoalesceReadWriteEnabled ( ) ) ;
2018-01-29 12:03:14 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : suggest_mode , isSuggestModeEnabled ( )
? lt : : settings_pack : : suggest_read_cache : lt : : settings_pack : : no_piece_suggestions ) ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : send_buffer_watermark , sendBufferWatermark ( ) * 1024 ) ;
settingsPack . set_int ( lt : : settings_pack : : send_buffer_low_watermark , sendBufferLowWatermark ( ) * 1024 ) ;
settingsPack . set_int ( lt : : settings_pack : : send_buffer_watermark_factor , sendBufferWatermarkFactor ( ) ) ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : anonymous_mode , isAnonymousModeEnabled ( ) ) ;
2015-04-19 18:17:47 +03:00
2016-06-03 17:03:17 +03:00
// Queueing System
if ( isQueueingSystemEnabled ( ) ) {
adjustLimits ( settingsPack ) ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : active_seeds , maxActiveUploads ( ) ) ;
settingsPack . set_bool ( lt : : settings_pack : : dont_count_slow_torrents , ignoreSlowTorrentsForQueueing ( ) ) ;
settingsPack . set_int ( lt : : settings_pack : : inactive_down_rate , downloadRateForSlowTorrents ( ) * 1024 ) ; // KiB to Bytes
settingsPack . set_int ( lt : : settings_pack : : inactive_up_rate , uploadRateForSlowTorrents ( ) * 1024 ) ; // KiB to Bytes
settingsPack . set_int ( lt : : settings_pack : : auto_manage_startup , slowTorrentsInactivityTimer ( ) ) ;
2015-04-19 18:17:47 +03:00
}
else {
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : active_downloads , - 1 ) ;
settingsPack . set_int ( lt : : settings_pack : : active_seeds , - 1 ) ;
settingsPack . set_int ( lt : : settings_pack : : active_limit , - 1 ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : active_tracker_limit , - 1 ) ;
settingsPack . set_int ( lt : : settings_pack : : active_dht_limit , - 1 ) ;
settingsPack . set_int ( lt : : settings_pack : : active_lsd_limit , - 1 ) ;
settingsPack . set_int ( lt : : settings_pack : : alert_queue_size , std : : numeric_limits < int > : : max ( ) / 2 ) ;
2015-04-19 18:17:47 +03:00
2016-06-03 17:03:17 +03:00
// Outgoing ports
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : outgoing_port , outgoingPortsMin ( ) ) ;
settingsPack . set_int ( lt : : settings_pack : : num_outgoing_ports , outgoingPortsMax ( ) - outgoingPortsMin ( ) + 1 ) ;
2016-06-03 17:03:17 +03:00
// Include overhead in transfer limits
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : rate_limit_ip_overhead , includeOverheadInLimits ( ) ) ;
2016-06-03 17:03:17 +03:00
// IP address to announce to trackers
2019-05-07 06:22:39 +03:00
settingsPack . set_str ( lt : : settings_pack : : announce_ip , announceIP ( ) . toStdString ( ) ) ;
2016-06-03 17:03:17 +03:00
// Super seeding
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : strict_super_seeding , isSuperSeedingEnabled ( ) ) ;
2016-06-03 17:03:17 +03:00
// * Max connections limit
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : connections_limit , maxConnections ( ) ) ;
2016-06-03 17:03:17 +03:00
// * Global max upload slots
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : unchoke_slots_limit , maxUploads ( ) ) ;
2016-06-03 17:03:17 +03:00
// uTP
2017-09-13 23:29:54 +03:00
switch ( btProtocol ( ) ) {
case BTProtocol : : Both :
default :
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_tcp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_tcp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_utp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_utp , true ) ;
2017-09-13 23:29:54 +03:00
break ;
case BTProtocol : : TCP :
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_tcp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_tcp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_utp , false ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_utp , false ) ;
2017-09-13 23:29:54 +03:00
break ;
case BTProtocol : : UTP :
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_tcp , false ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_tcp , false ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_incoming_utp , true ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_outgoing_utp , true ) ;
2017-09-13 23:29:54 +03:00
break ;
}
2017-08-15 21:23:07 +03:00
switch ( utpMixedMode ( ) ) {
case MixedModeAlgorithm : : TCP :
default :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : mixed_mode_algorithm , lt : : settings_pack : : prefer_tcp ) ;
2017-08-15 21:23:07 +03:00
break ;
case MixedModeAlgorithm : : Proportional :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : mixed_mode_algorithm , lt : : settings_pack : : peer_proportional ) ;
2017-08-15 21:23:07 +03:00
break ;
}
2016-06-03 17:03:17 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : allow_multiple_connections_per_ip , multiConnectionsPerIpEnabled ( ) ) ;
2016-06-03 17:03:17 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : apply_ip_filter_to_trackers , isTrackerFilteringEnabled ( ) ) ;
2016-06-03 17:03:17 +03:00
2019-05-07 06:22:39 +03:00
settingsPack . set_bool ( lt : : settings_pack : : enable_dht , isDHTEnabled ( ) ) ;
2016-10-31 03:52:29 +03:00
if ( isDHTEnabled ( ) )
2019-05-07 06:22:39 +03:00
settingsPack . set_str ( lt : : settings_pack : : dht_bootstrap_nodes , " dht.libtorrent.org:25401,router.bittorrent.com:6881,router.utorrent.com:6881,dht.transmissionbt.com:6881,dht.aelitis.com:6881 " ) ;
settingsPack . set_bool ( lt : : settings_pack : : enable_lsd , isLSDEnabled ( ) ) ;
2017-08-11 12:06:31 +03:00
switch ( chokingAlgorithm ( ) ) {
2017-08-15 21:23:07 +03:00
case ChokingAlgorithm : : FixedSlots :
2017-08-11 12:06:31 +03:00
default :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : choking_algorithm , lt : : settings_pack : : fixed_slots_choker ) ;
2017-08-11 12:06:31 +03:00
break ;
2017-08-15 21:23:07 +03:00
case ChokingAlgorithm : : RateBased :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : choking_algorithm , lt : : settings_pack : : rate_based_choker ) ;
2017-08-11 12:06:31 +03:00
break ;
}
2017-08-15 21:23:07 +03:00
switch ( seedChokingAlgorithm ( ) ) {
case SeedChokingAlgorithm : : RoundRobin :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : round_robin ) ;
2017-08-15 21:23:07 +03:00
break ;
case SeedChokingAlgorithm : : FastestUpload :
default :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : fastest_upload ) ;
2017-08-15 21:23:07 +03:00
break ;
case SeedChokingAlgorithm : : AntiLeech :
2019-05-07 06:22:39 +03:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : anti_leech ) ;
2017-08-15 21:23:07 +03:00
break ;
}
2016-05-01 11:05:52 +03:00
}
2017-05-01 19:19:34 +03:00
void Session : : configurePeerClasses ( )
{
2019-05-07 06:22:39 +03:00
lt : : ip_filter f ;
2017-11-24 02:03:11 +03:00
// address_v4::from_string("255.255.255.255") crashes on some people's systems
// so instead we use address_v4::broadcast()
// Proactively do the same for 0.0.0.0 and address_v4::any()
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : any ( )
, lt : : address_v4 : : broadcast ( )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : global_peer_class_id ) ) ;
2019-06-02 09:33:46 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200) || TORRENT_USE_IPV6
2017-08-11 23:12:56 +03:00
// IPv6 may not be available on OS and the parsing
// would result in an exception -> abnormal program termination
// Affects Windows XP
try {
2019-06-02 09:33:46 +03:00
f . add_rule ( lt : : address_v6 : : any ( )
2019-05-07 06:22:39 +03:00
, lt : : address_v6 : : from_string ( " ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : global_peer_class_id ) ) ;
2017-08-11 23:12:56 +03:00
}
2019-05-16 10:02:17 +03:00
catch ( const std : : exception & ) { }
2019-06-02 09:33:46 +03:00
# endif
2017-05-01 19:19:34 +03:00
if ( ignoreLimitsOnLAN ( ) ) {
2017-05-02 18:11:23 +03:00
// local networks
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : from_string ( " 10.0.0.0 " )
, lt : : address_v4 : : from_string ( " 10.255.255.255 " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : from_string ( " 172.16.0.0 " )
, lt : : address_v4 : : from_string ( " 172.31.255.255 " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : from_string ( " 192.168.0.0 " )
, lt : : address_v4 : : from_string ( " 192.168.255.255 " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2017-05-02 18:11:23 +03:00
// link local
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : from_string ( " 169.254.0.0 " )
, lt : : address_v4 : : from_string ( " 169.254.255.255 " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2017-05-02 18:11:23 +03:00
// loopback
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v4 : : from_string ( " 127.0.0.0 " )
, lt : : address_v4 : : from_string ( " 127.255.255.255 " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2019-06-02 09:33:46 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200) || TORRENT_USE_IPV6
2017-08-11 23:12:56 +03:00
// IPv6 may not be available on OS and the parsing
// would result in an exception -> abnormal program termination
// Affects Windows XP
try {
// link local
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v6 : : from_string ( " fe80:: " )
, lt : : address_v6 : : from_string ( " febf:ffff:ffff:ffff:ffff:ffff:ffff:ffff " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2017-08-11 23:12:56 +03:00
// unique local addresses
2019-05-07 06:22:39 +03:00
f . add_rule ( lt : : address_v6 : : from_string ( " fc00:: " )
, lt : : address_v6 : : from_string ( " fdff:ffff:ffff:ffff:ffff:ffff:ffff:ffff " )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2017-08-11 23:12:56 +03:00
// loopback
2019-06-02 09:33:46 +03:00
f . add_rule ( lt : : address_v6 : : loopback ( )
, lt : : address_v6 : : loopback ( )
2019-07-18 19:53:04 +03:00
, 1 < < static_cast < LTUnderlyingType < LTPeerClass > > ( lt : : session : : local_peer_class_id ) ) ;
2017-08-11 23:12:56 +03:00
}
2019-05-16 10:02:17 +03:00
catch ( const std : : exception & ) { }
2019-06-02 09:33:46 +03:00
# endif
2017-05-01 19:19:34 +03:00
}
m_nativeSession - > set_peer_class_filter ( f ) ;
2019-05-07 06:22:39 +03:00
lt : : peer_class_type_filter peerClassTypeFilter ;
peerClassTypeFilter . add ( lt : : peer_class_type_filter : : tcp_socket , lt : : session : : tcp_peer_class_id ) ;
peerClassTypeFilter . add ( lt : : peer_class_type_filter : : ssl_tcp_socket , lt : : session : : tcp_peer_class_id ) ;
peerClassTypeFilter . add ( lt : : peer_class_type_filter : : i2p_socket , lt : : session : : tcp_peer_class_id ) ;
2017-12-04 02:56:00 +03:00
if ( ! isUTPRateLimited ( ) ) {
2019-05-07 06:22:39 +03:00
peerClassTypeFilter . disallow ( lt : : peer_class_type_filter : : utp_socket
, lt : : session : : global_peer_class_id ) ;
peerClassTypeFilter . disallow ( lt : : peer_class_type_filter : : ssl_utp_socket
, lt : : session : : global_peer_class_id ) ;
2017-05-01 19:19:34 +03:00
}
m_nativeSession - > set_peer_class_type_filter ( peerClassTypeFilter ) ;
}
2019-02-09 18:40:14 +03:00
void Session : : enableTracker ( const bool enable )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( enable ) {
2015-04-19 18:17:47 +03:00
if ( ! m_tracker )
m_tracker = new Tracker ( this ) ;
2019-08-08 18:21:07 +03:00
m_tracker - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
else {
if ( m_tracker )
delete m_tracker ;
}
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
void Session : : enableBandwidthScheduler ( )
{
if ( ! m_bwScheduler ) {
m_bwScheduler = new BandwidthScheduler ( this ) ;
2017-08-20 18:00:23 +03:00
connect ( m_bwScheduler . data ( ) , & BandwidthScheduler : : bandwidthLimitRequested
, this , & Session : : setAltGlobalSpeedLimitEnabled ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
m_bwScheduler - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
void Session : : populateAdditionalTrackers ( )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
m_additionalTrackerList . clear ( ) ;
2019-08-06 18:07:57 +03:00
const QString trackers = additionalTrackers ( ) ;
for ( QStringRef tracker : asConst ( trackers . splitRef ( ' \n ' ) ) ) {
2016-05-01 11:05:52 +03:00
tracker = tracker . trimmed ( ) ;
if ( ! tracker . isEmpty ( ) )
2019-08-06 18:07:57 +03:00
m_additionalTrackerList < < tracker . toString ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-02-07 20:31:50 +03:00
void Session : : processShareLimits ( )
2015-04-19 18:17:47 +03:00
{
2016-02-07 20:31:50 +03:00
qDebug ( " Processing share limits... " ) ;
2015-04-19 18:17:47 +03:00
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) ) {
2016-02-07 20:31:50 +03:00
if ( torrent - > isSeed ( ) & & ! torrent - > isForced ( ) ) {
if ( torrent - > ratioLimit ( ) ! = TorrentHandle : : NO_RATIO_LIMIT ) {
const qreal ratio = torrent - > realRatio ( ) ;
qreal ratioLimit = torrent - > ratioLimit ( ) ;
if ( ratioLimit = = TorrentHandle : : USE_GLOBAL_RATIO )
// If Global Max Ratio is really set...
ratioLimit = globalMaxRatio ( ) ;
if ( ratioLimit > = 0 ) {
qDebug ( " Ratio: %f (limit: %f) " , ratio , ratioLimit ) ;
if ( ( ratio < = TorrentHandle : : MAX_RATIO ) & & ( ratio > = ratioLimit ) ) {
if ( m_maxRatioAction = = Remove ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Removed. " ) . arg ( torrent - > name ( ) ) ) ;
2017-09-11 08:11:09 +03:00
deleteTorrent ( torrent - > hash ( ) ) ;
2016-02-07 20:31:50 +03:00
}
2019-09-18 19:26:01 +03:00
else if ( m_maxRatioAction = = DeleteFiles ) {
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Removed torrent and its files. " ) . arg ( torrent - > name ( ) ) ) ;
deleteTorrent ( torrent - > hash ( ) , TorrentAndFiles ) ;
}
2019-08-09 13:06:16 +03:00
else if ( ( m_maxRatioAction = = Pause ) & & ! torrent - > isPaused ( ) ) {
2016-02-07 20:31:50 +03:00
torrent - > pause ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Paused. " ) . arg ( torrent - > name ( ) ) ) ;
2016-02-07 20:31:50 +03:00
}
2019-08-09 13:06:16 +03:00
else if ( ( m_maxRatioAction = = EnableSuperSeeding ) & & ! torrent - > isPaused ( ) & & ! torrent - > superSeeding ( ) ) {
torrent - > setSuperSeeding ( true ) ;
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Enabled super seeding for it. " ) . arg ( torrent - > name ( ) ) ) ;
}
2018-02-23 02:55:28 +03:00
continue ;
2016-02-07 20:31:50 +03:00
}
2015-04-19 18:17:47 +03:00
}
2016-02-07 20:31:50 +03:00
}
if ( torrent - > seedingTimeLimit ( ) ! = TorrentHandle : : NO_SEEDING_TIME_LIMIT ) {
2019-06-18 18:36:45 +03:00
const qlonglong seedingTimeInMinutes = torrent - > seedingTime ( ) / 60 ;
2016-02-07 20:31:50 +03:00
int seedingTimeLimit = torrent - > seedingTimeLimit ( ) ;
2019-06-20 14:49:19 +03:00
if ( seedingTimeLimit = = TorrentHandle : : USE_GLOBAL_SEEDING_TIME ) {
2016-02-07 20:31:50 +03:00
// If Global Seeding Time Limit is really set...
seedingTimeLimit = globalMaxSeedingMinutes ( ) ;
2019-06-20 14:49:19 +03:00
}
2016-02-07 20:31:50 +03:00
if ( seedingTimeLimit > = 0 ) {
if ( ( seedingTimeInMinutes < = TorrentHandle : : MAX_SEEDING_TIME ) & & ( seedingTimeInMinutes > = seedingTimeLimit ) ) {
if ( m_maxRatioAction = = Remove ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Removed. " ) . arg ( torrent - > name ( ) ) ) ;
2017-09-11 08:11:09 +03:00
deleteTorrent ( torrent - > hash ( ) ) ;
2016-02-07 20:31:50 +03:00
}
2019-09-18 19:26:01 +03:00
else if ( m_maxRatioAction = = DeleteFiles ) {
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Removed torrent and its files. " ) . arg ( torrent - > name ( ) ) ) ;
deleteTorrent ( torrent - > hash ( ) , TorrentAndFiles ) ;
}
2019-08-09 13:06:16 +03:00
else if ( ( m_maxRatioAction = = Pause ) & & ! torrent - > isPaused ( ) ) {
2016-02-07 20:31:50 +03:00
torrent - > pause ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Paused. " ) . arg ( torrent - > name ( ) ) ) ;
2016-02-07 20:31:50 +03:00
}
2019-08-09 13:06:16 +03:00
else if ( ( m_maxRatioAction = = EnableSuperSeeding ) & & ! torrent - > isPaused ( ) & & ! torrent - > superSeeding ( ) ) {
torrent - > setSuperSeeding ( true ) ;
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Enabled super seeding for it. " ) . arg ( torrent - > name ( ) ) ) ;
}
2015-04-19 18:17:47 +03:00
}
}
}
}
}
}
// Add to BitTorrent session the downloaded torrent file
2019-03-01 10:38:16 +03:00
void Session : : handleDownloadFinished ( const Net : : DownloadResult & result )
2015-04-19 18:17:47 +03:00
{
2019-03-01 10:38:16 +03:00
switch ( result . status ) {
case Net : : DownloadStatus : : Success :
emit downloadFromUrlFinished ( result . url ) ;
addTorrent_impl ( CreateTorrentParams ( m_downloadedTorrents . take ( result . url ) )
, MagnetUri ( ) , TorrentInfo : : load ( result . data ) ) ;
break ;
case Net : : DownloadStatus : : RedirectedToMagnet :
addTorrent_impl ( CreateTorrentParams ( m_downloadedTorrents . take ( result . url ) ) , MagnetUri ( result . magnet ) ) ;
break ;
default :
emit downloadFromUrlFailed ( result . url , result . errorString ) ;
}
2015-04-19 18:17:47 +03:00
}
// Return the torrent handle, given its hash
TorrentHandle * Session : : findTorrent ( const InfoHash & hash ) const
{
return m_torrents . value ( hash ) ;
}
bool Session : : hasActiveTorrents ( ) const
{
2018-08-02 22:45:21 +03:00
return std : : any_of ( m_torrents . begin ( ) , m_torrents . end ( ) , [ ] ( TorrentHandle * torrent )
{
return TorrentFilter : : ActiveTorrent . match ( torrent ) ;
} ) ;
2015-04-19 18:17:47 +03:00
}
bool Session : : hasUnfinishedTorrents ( ) const
{
2018-08-02 22:45:21 +03:00
return std : : any_of ( m_torrents . begin ( ) , m_torrents . end ( ) , [ ] ( const TorrentHandle * torrent )
{
return ( ! torrent - > isSeed ( ) & & ! torrent - > isPaused ( ) ) ;
} ) ;
}
2015-04-19 18:17:47 +03:00
2018-08-02 22:45:21 +03:00
bool Session : : hasRunningSeed ( ) const
{
return std : : any_of ( m_torrents . begin ( ) , m_torrents . end ( ) , [ ] ( const TorrentHandle * torrent )
{
return ( torrent - > isSeed ( ) & & ! torrent - > isPaused ( ) ) ;
} ) ;
2015-04-19 18:17:47 +03:00
}
void Session : : banIP ( const QString & ip )
{
2016-05-01 11:05:52 +03:00
QStringList bannedIPs = m_bannedIPs ;
if ( ! bannedIPs . contains ( ip ) ) {
2019-05-07 06:22:39 +03:00
lt : : ip_filter filter = m_nativeSession - > get_ip_filter ( ) ;
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-05-07 06:22:39 +03:00
const lt : : address addr = lt : : address : : from_string ( ip . toLatin1 ( ) . constData ( ) , ec ) ;
2016-06-03 17:03:17 +03:00
Q_ASSERT ( ! ec ) ;
2016-10-30 00:11:52 +03:00
if ( ec ) return ;
2019-05-07 06:22:39 +03:00
filter . add_rule ( addr , addr , lt : : ip_filter : : blocked ) ;
2016-06-03 17:03:17 +03:00
m_nativeSession - > set_ip_filter ( filter ) ;
2016-05-01 11:05:52 +03:00
bannedIPs < < ip ;
2017-03-06 15:40:34 +03:00
bannedIPs . sort ( ) ;
2016-05-01 11:05:52 +03:00
m_bannedIPs = bannedIPs ;
}
2015-04-19 18:17:47 +03:00
}
// Delete a torrent from the session, given its hash
2019-09-18 19:26:01 +03:00
// and from the disk, if the corresponding deleteOption is chosen
2019-09-29 04:47:06 +03:00
bool Session : : deleteTorrent ( const InfoHash & hash , const DeleteOption deleteOption )
2015-04-19 18:17:47 +03:00
{
TorrentHandle * const torrent = m_torrents . take ( hash ) ;
if ( ! torrent ) return false ;
2017-08-13 13:56:03 +03:00
qDebug ( " Deleting torrent with hash: %s " , qUtf8Printable ( torrent - > hash ( ) ) ) ;
2015-04-19 18:17:47 +03:00
emit torrentAboutToBeRemoved ( torrent ) ;
// Remove it from session
2019-09-18 19:26:01 +03:00
if ( deleteOption = = Torrent ) {
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , " " , deleteOption } ;
2015-04-19 18:17:47 +03:00
QStringList unwantedFiles ;
if ( torrent - > hasMetadata ( ) )
unwantedFiles = torrent - > absoluteFilePathsUnwanted ( ) ;
2019-05-07 06:22:39 +03:00
m_nativeSession - > remove_torrent ( torrent - > nativeHandle ( ) , lt : : session : : delete_partfile ) ;
2015-04-19 18:17:47 +03:00
// Remove unwanted and incomplete files
2018-11-27 23:15:04 +03:00
for ( const QString & unwantedFile : asConst ( unwantedFiles ) ) {
2017-08-13 13:56:03 +03:00
qDebug ( " Removing unwanted file: %s " , qUtf8Printable ( unwantedFile ) ) ;
2015-05-06 14:53:27 +03:00
Utils : : Fs : : forceRemove ( unwantedFile ) ;
const QString parentFolder = Utils : : Fs : : branchPath ( unwantedFile ) ;
2017-08-13 13:56:03 +03:00
qDebug ( " Attempt to remove parent folder (if empty): %s " , qUtf8Printable ( parentFolder ) ) ;
2019-06-10 19:15:06 +03:00
QDir ( ) . rmdir ( parentFolder ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-09-18 19:26:01 +03:00
else {
const QString rootPath = torrent - > rootPath ( true ) ;
if ( ! rootPath . isEmpty ( ) ) {
// torrent with root folder
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , rootPath , deleteOption } ;
}
else if ( torrent - > useTempPath ( ) ) {
// torrent without root folder still has it in its temporary save path
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , torrent - > savePath ( true ) , deleteOption } ;
}
else {
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , " " , deleteOption } ;
}
m_nativeSession - > remove_torrent ( torrent - > nativeHandle ( ) , lt : : session : : delete_files ) ;
}
2015-04-19 18:17:47 +03:00
// Remove it from torrent resume directory
2019-02-09 18:40:14 +03:00
const QDir resumeDataDir ( m_resumeFolderPath ) ;
2015-04-19 18:17:47 +03:00
QStringList filters ;
filters < < QString ( " %1.* " ) . arg ( torrent - > hash ( ) ) ;
const QStringList files = resumeDataDir . entryList ( filters , QDir : : Files , QDir : : Unsorted ) ;
2018-11-18 21:40:37 +03:00
for ( const QString & file : files )
2015-05-06 14:53:27 +03:00
Utils : : Fs : : forceRemove ( resumeDataDir . absoluteFilePath ( file ) ) ;
2015-04-19 18:17:47 +03:00
delete torrent ;
qDebug ( " Torrent deleted. " ) ;
return true ;
}
bool Session : : cancelLoadMetadata ( const InfoHash & hash )
{
2019-09-29 05:12:56 +03:00
const auto loadedMetadataIter = m_loadedMetadata . find ( hash ) ;
if ( loadedMetadataIter = = m_loadedMetadata . end ( ) ) return false ;
2015-04-19 18:17:47 +03:00
2019-09-29 05:12:56 +03:00
m_loadedMetadata . erase ( loadedMetadataIter ) ;
2019-05-07 06:22:39 +03:00
const lt : : torrent_handle torrent = m_nativeSession - > find_torrent ( hash ) ;
2015-04-19 18:17:47 +03:00
if ( ! torrent . is_valid ( ) ) return false ;
2019-03-06 08:58:07 +03:00
if ( ! torrent . status ( LTStatusFlags { 0 } ) . has_metadata ) {
2015-04-19 18:17:47 +03:00
// if hidden torrent is still loading metadata...
- - m_extraLimit ;
adjustLimits ( ) ;
}
// Remove it from session
2019-05-07 06:22:39 +03:00
m_nativeSession - > remove_torrent ( torrent , lt : : session : : delete_files ) ;
2015-04-19 18:17:47 +03:00
qDebug ( " Preloaded torrent deleted. " ) ;
return true ;
}
2019-10-02 07:52:51 +03:00
void Session : : increaseTorrentsQueuePos ( const QVector < InfoHash > & hashes )
2015-04-19 18:17:47 +03:00
{
2019-10-02 07:52:51 +03:00
using ElementType = std : : pair < int , TorrentHandle * > ;
std : : priority_queue < ElementType
, std : : vector < ElementType >
, std : : greater < ElementType > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 02:01:09 +03:00
// Sort torrents by queue position
2019-10-02 07:52:51 +03:00
for ( const InfoHash & infoHash : hashes ) {
2018-11-06 18:49:17 +03:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
2019-10-02 07:52:51 +03:00
torrentQueue . emplace ( torrent - > queuePosition ( ) , torrent ) ;
2015-04-19 18:17:47 +03:00
}
2018-12-08 02:01:09 +03:00
// Increase torrents queue position (starting with the one in the highest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
2019-10-02 07:52:51 +03:00
const TorrentHandle * torrent = torrentQueue . top ( ) . second ;
2015-04-19 18:17:47 +03:00
torrentQueuePositionUp ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-08-23 07:38:08 +03:00
2018-11-19 12:53:29 +03:00
saveTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-02 07:52:51 +03:00
void Session : : decreaseTorrentsQueuePos ( const QVector < InfoHash > & hashes )
2015-04-19 18:17:47 +03:00
{
2019-10-02 07:52:51 +03:00
using ElementType = std : : pair < int , TorrentHandle * > ;
std : : priority_queue < ElementType > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 02:01:09 +03:00
// Sort torrents by queue position
2019-10-02 07:52:51 +03:00
for ( const InfoHash & infoHash : hashes ) {
2018-11-06 18:49:17 +03:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
2019-10-02 07:52:51 +03:00
torrentQueue . emplace ( torrent - > queuePosition ( ) , torrent ) ;
2015-04-19 18:17:47 +03:00
}
2018-12-08 02:01:09 +03:00
// Decrease torrents queue position (starting with the one in the lowest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
2019-10-02 07:52:51 +03:00
const TorrentHandle * torrent = torrentQueue . top ( ) . second ;
2015-04-19 18:17:47 +03:00
torrentQueuePositionDown ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-03-06 17:50:10 +03:00
for ( auto i = m_loadedMetadata . cbegin ( ) ; i ! = m_loadedMetadata . cend ( ) ; + + i )
torrentQueuePositionBottom ( m_nativeSession - > find_torrent ( i . key ( ) ) ) ;
2018-08-23 07:38:08 +03:00
2018-11-19 12:53:29 +03:00
saveTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-02 07:52:51 +03:00
void Session : : topTorrentsQueuePos ( const QVector < InfoHash > & hashes )
2015-04-19 18:17:47 +03:00
{
2019-10-02 07:52:51 +03:00
using ElementType = std : : pair < int , TorrentHandle * > ;
2019-10-02 08:58:12 +03:00
std : : priority_queue < ElementType > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 02:01:09 +03:00
// Sort torrents by queue position
2019-10-02 07:52:51 +03:00
for ( const InfoHash & infoHash : hashes ) {
2018-11-06 18:49:17 +03:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
2019-10-02 07:52:51 +03:00
torrentQueue . emplace ( torrent - > queuePosition ( ) , torrent ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-02 08:58:12 +03:00
// Top torrents queue position (starting with the one in the lowest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
2019-10-02 07:52:51 +03:00
const TorrentHandle * torrent = torrentQueue . top ( ) . second ;
2015-04-19 18:17:47 +03:00
torrentQueuePositionTop ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-08-23 07:38:08 +03:00
2018-11-19 12:53:29 +03:00
saveTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-02 07:52:51 +03:00
void Session : : bottomTorrentsQueuePos ( const QVector < InfoHash > & hashes )
2015-04-19 18:17:47 +03:00
{
2019-10-02 07:52:51 +03:00
using ElementType = std : : pair < int , TorrentHandle * > ;
std : : priority_queue < ElementType
, std : : vector < ElementType >
2019-10-02 08:58:12 +03:00
, std : : greater < ElementType > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 02:01:09 +03:00
// Sort torrents by queue position
2019-10-02 07:52:51 +03:00
for ( const InfoHash & infoHash : hashes ) {
2018-11-06 18:49:17 +03:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
2019-10-02 07:52:51 +03:00
torrentQueue . emplace ( torrent - > queuePosition ( ) , torrent ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-02 08:58:12 +03:00
// Bottom torrents queue position (starting with the one in the highest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
2019-10-02 07:52:51 +03:00
const TorrentHandle * torrent = torrentQueue . top ( ) . second ;
2015-04-19 18:17:47 +03:00
torrentQueuePositionBottom ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-03-06 17:50:10 +03:00
for ( auto i = m_loadedMetadata . cbegin ( ) ; i ! = m_loadedMetadata . cend ( ) ; + + i )
torrentQueuePositionBottom ( m_nativeSession - > find_torrent ( i . key ( ) ) ) ;
2018-08-23 07:38:08 +03:00
2018-11-19 12:53:29 +03:00
saveTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-07-22 14:29:14 +03:00
void Session : : handleTorrentSaveResumeDataRequested ( const TorrentHandle * torrent )
2019-07-22 14:22:26 +03:00
{
qDebug ( " Saving resume data is requested for torrent '%s'... " , qUtf8Printable ( torrent - > name ( ) ) ) ;
+ + m_numResumeData ;
}
2015-04-19 18:17:47 +03:00
QHash < InfoHash , TorrentHandle * > Session : : torrents ( ) const
{
return m_torrents ;
}
2018-12-28 08:38:08 +03:00
bool Session : : addTorrent ( const QString & source , const AddTorrentParams & params )
2015-04-19 18:17:47 +03:00
{
2018-12-28 08:38:08 +03:00
// `source`: .torrent file path/url or magnet uri
2018-11-04 22:21:31 +03:00
2018-12-29 15:38:51 +03:00
if ( Net : : DownloadManager : : hasSupportedScheme ( source ) ) {
2018-06-25 20:31:32 +03:00
LogMsg ( tr ( " Downloading '%1', please wait... " , " e.g: Downloading 'xxx.torrent', please wait... " ) . arg ( source ) ) ;
2015-04-19 18:17:47 +03:00
// Launch downloader
2019-04-08 20:09:16 +03:00
Net : : DownloadManager : : instance ( ) - > download ( Net : : DownloadRequest ( source ) . limit ( MAX_TORRENT_SIZE )
2019-03-03 12:43:42 +03:00
, this , & Session : : handleDownloadFinished ) ;
m_downloadedTorrents [ source ] = params ;
2018-11-04 22:21:31 +03:00
return true ;
2015-04-19 18:17:47 +03:00
}
2018-11-04 22:21:31 +03:00
2018-12-28 08:38:08 +03:00
const MagnetUri magnetUri { source } ;
if ( magnetUri . isValid ( ) )
return addTorrent_impl ( CreateTorrentParams ( params ) , magnetUri ) ;
2018-11-04 22:21:31 +03:00
TorrentFileGuard guard ( source ) ;
2018-12-16 10:51:36 +03:00
if ( addTorrent_impl ( CreateTorrentParams ( params )
, MagnetUri ( ) , TorrentInfo : : loadFromFile ( source ) ) ) {
2018-11-04 22:21:31 +03:00
guard . markAsAddedToSession ( ) ;
return true ;
2015-04-19 18:17:47 +03:00
}
return false ;
}
bool Session : : addTorrent ( const TorrentInfo & torrentInfo , const AddTorrentParams & params )
{
if ( ! torrentInfo . isValid ( ) ) return false ;
2018-12-16 10:51:36 +03:00
return addTorrent_impl ( CreateTorrentParams ( params ) , MagnetUri ( ) , torrentInfo ) ;
2015-04-19 18:17:47 +03:00
}
// Add a torrent to the BitTorrent session
2018-07-11 15:44:15 +03:00
bool Session : : addTorrent_impl ( CreateTorrentParams params , const MagnetUri & magnetUri ,
2016-04-19 09:54:48 +03:00
TorrentInfo torrentInfo , const QByteArray & fastresumeData )
2015-04-19 18:17:47 +03:00
{
2018-07-11 15:44:15 +03:00
params . savePath = normalizeSavePath ( params . savePath , " " ) ;
2016-02-09 11:56:48 +03:00
2018-07-11 15:44:15 +03:00
if ( ! params . category . isEmpty ( ) ) {
2019-06-29 11:15:41 +03:00
if ( ! m_categories . contains ( params . category ) & & ! addCategory ( params . category ) )
2018-07-11 15:44:15 +03:00
params . category = " " ;
2016-01-01 16:28:40 +03:00
}
2019-06-29 11:15:41 +03:00
const bool fromMagnetUri = magnetUri . isValid ( ) ;
2018-07-29 11:02:57 +03:00
// If empty then Automatic mode, otherwise Manual mode
QString savePath = params . savePath . isEmpty ( ) ? categorySavePath ( params . category ) : params . savePath ;
2019-05-07 06:22:39 +03:00
lt : : add_torrent_params p ;
2015-04-19 18:17:47 +03:00
InfoHash hash ;
if ( fromMagnetUri ) {
hash = magnetUri . hash ( ) ;
2016-01-01 16:28:40 +03:00
2019-07-10 19:54:10 +03:00
const auto it = m_loadedMetadata . constFind ( hash ) ;
if ( it ! = m_loadedMetadata . constEnd ( ) ) {
2019-06-28 21:24:39 +03:00
// Adding preloaded torrent...
2019-07-10 19:54:10 +03:00
const TorrentInfo metadata = it . value ( ) ;
if ( metadata . isValid ( ) ) {
// Metadata is received and torrent_handle is being deleted
// so we can't reuse it. Just add torrent using its metadata.
return addTorrent_impl ( params
, MagnetUri { } , metadata , fastresumeData ) ;
}
2016-01-01 16:28:40 +03:00
2019-07-10 19:54:10 +03:00
// Reuse existing torrent_handle
2019-06-28 21:24:39 +03:00
lt : : torrent_handle handle = m_nativeSession - > find_torrent ( hash ) ;
// We need to pause it first to create TorrentHandle within the same
// underlying state as in other cases.
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-07-10 19:54:10 +03:00
handle . auto_managed ( false ) ;
2019-04-03 09:23:35 +03:00
# else
2019-07-10 19:54:10 +03:00
handle . unset_flags ( lt : : torrent_flags : : auto_managed ) ;
2019-04-03 09:23:35 +03:00
# endif
2019-07-10 19:54:10 +03:00
handle . pause ( ) ;
2016-01-01 16:28:40 +03:00
2019-07-10 19:54:10 +03:00
m_loadedMetadata . remove ( hash ) ;
m_addingTorrents . insert ( hash , params ) ;
2016-01-01 16:28:40 +03:00
return true ;
}
p = magnetUri . addTorrentParams ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-06-29 11:15:41 +03:00
else {
if ( ! torrentInfo . isValid ( ) ) {
// We can have an invalid torrentInfo when there isn't a matching
// .torrent file to the .fastresume we loaded. Possibly from a
// failed upgrade.
return false ;
}
hash = torrentInfo . hash ( ) ;
}
// We should not add the torrent if it is already
// processed or is pending to add to session
if ( m_addingTorrents . contains ( hash ) | | m_loadedMetadata . contains ( hash ) )
return false ;
TorrentHandle * const torrent = m_torrents . value ( hash ) ;
if ( torrent ) { // a duplicate torrent is added
if ( torrent - > isPrivate ( ) | | ( ! fromMagnetUri & & torrentInfo . isPrivate ( ) ) )
return false ;
// merge trackers and web seeds
torrent - > addTrackers ( fromMagnetUri ? magnetUri . trackers ( ) : torrentInfo . trackers ( ) ) ;
torrent - > addUrlSeeds ( fromMagnetUri ? magnetUri . urlSeeds ( ) : torrentInfo . urlSeeds ( ) ) ;
return true ;
}
// Record if .fastresume is complete, that is whether it contains
// the required fields to resume a torrent.
bool hasCompleteFastresume = false ;
if ( ! fromMagnetUri ) {
if ( params . restored ) { // load from existing fastresume
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-06-28 21:24:39 +03:00
// Make sure the torrent will be initially checked and then paused
// to perform some service jobs on it. We will start it if needed.
// (Workaround to easily support libtorrent-1.1
QByteArray patchedFastresumeData = fastresumeData ;
patchedFastresumeData . replace ( " 6:pausedi0e " , " 6:pausedi1e " ) ;
patchedFastresumeData . replace ( " 12:auto_managedi0e " , " 12:auto_managedi1e " ) ;
p . resume_data = std : : vector < char > { patchedFastresumeData . constData ( )
, ( patchedFastresumeData . constData ( ) + patchedFastresumeData . size ( ) ) } ;
2019-06-29 11:15:41 +03:00
p . flags | = lt : : add_torrent_params : : flag_use_resume_save_path ;
2019-08-21 10:41:04 +03:00
// load from .torrent file when fastresume doesn't contain the required `info` dict
if ( ! patchedFastresumeData . contains ( " 4:infod " ) )
p . ti = torrentInfo . nativeInfo ( ) ;
2019-06-29 11:15:41 +03:00
// Still setup the default parameters and let libtorrent handle
// the parameter merging
hasCompleteFastresume = false ;
# else
lt : : error_code ec ;
p = lt : : read_resume_data ( fastresumeData , ec ) ;
2019-08-21 10:41:04 +03:00
// load from .torrent file when fastresume doesn't contain the required `info` dict
2019-09-03 07:35:15 +03:00
if ( ! p . ti | | ! p . ti - > is_valid ( ) )
2019-08-21 10:41:04 +03:00
p . ti = torrentInfo . nativeInfo ( ) ;
2019-06-29 11:15:41 +03:00
// libtorrent will always apply `file_priorities` to torrents,
// if the field is present then the fastresume is considered to
// be correctly generated and should be complete.
hasCompleteFastresume = ! p . file_priorities . empty ( ) ;
# endif
}
else { // new torrent
2018-07-11 15:44:15 +03:00
if ( ! params . hasRootFolder )
2018-05-19 09:41:33 +03:00
torrentInfo . stripRootFolder ( ) ;
// Metadata
2018-07-11 15:44:15 +03:00
if ( ! params . hasSeedStatus )
2018-05-19 09:41:33 +03:00
findIncompleteFiles ( torrentInfo , savePath ) ;
// if torrent name wasn't explicitly set we handle the case of
// initial renaming of torrent content and rename torrent accordingly
2018-07-11 15:44:15 +03:00
if ( params . name . isEmpty ( ) ) {
2018-05-19 09:41:33 +03:00
QString contentName = torrentInfo . rootFolder ( ) ;
if ( contentName . isEmpty ( ) & & ( torrentInfo . filesCount ( ) = = 1 ) )
contentName = torrentInfo . fileName ( 0 ) ;
if ( ! contentName . isEmpty ( ) & & ( contentName ! = torrentInfo . name ( ) ) )
2018-07-11 15:44:15 +03:00
params . name = contentName ;
2018-05-19 09:41:33 +03:00
}
2019-06-29 11:15:41 +03:00
Q_ASSERT ( p . file_priorities . empty ( ) ) ;
std : : transform ( params . filePriorities . cbegin ( ) , params . filePriorities . cend ( )
, std : : back_inserter ( p . file_priorities ) , [ ] ( const DownloadPriority priority )
{
# if (LIBTORRENT_VERSION_NUM < 10200)
return static_cast < boost : : uint8_t > ( priority ) ;
# else
return static_cast < lt : : download_priority_t > (
static_cast < lt : : download_priority_t : : underlying_type > ( priority ) ) ;
# endif
} ) ;
2016-01-04 16:00:50 +03:00
2019-08-21 10:41:04 +03:00
p . ti = torrentInfo . nativeInfo ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2019-06-28 21:24:39 +03:00
// Common
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-06-28 21:24:39 +03:00
p . flags & = ~ lt : : add_torrent_params : : flag_duplicate_is_error ; // Already checked
2019-04-03 09:23:35 +03:00
# else
2019-06-28 21:24:39 +03:00
p . flags & = ~ lt : : torrent_flags : : duplicate_is_error ; // Already checked
# endif
// Make sure the torrent will be initially checked and then paused
// to perform some service jobs on it. We will start it if needed.
# if (LIBTORRENT_VERSION_NUM < 10200)
p . flags | = lt : : add_torrent_params : : flag_paused ;
p . flags | = lt : : add_torrent_params : : flag_auto_managed ;
p . flags | = lt : : add_torrent_params : : flag_stop_when_ready ;
# else
p . flags | = lt : : torrent_flags : : paused ;
p . flags | = lt : : torrent_flags : : auto_managed ;
p . flags | = lt : : torrent_flags : : stop_when_ready ;
2019-04-03 09:23:35 +03:00
# endif
2015-04-19 18:17:47 +03:00
2019-06-28 21:24:39 +03:00
if ( ! hasCompleteFastresume ) {
2019-06-29 11:15:41 +03:00
// Limits
p . max_connections = maxConnectionsPerTorrent ( ) ;
p . max_uploads = maxUploadsPerTorrent ( ) ;
p . save_path = Utils : : Fs : : toNativePath ( savePath ) . toStdString ( ) ;
p . upload_limit = params . uploadLimit ;
p . download_limit = params . downloadLimit ;
2019-08-26 07:30:22 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200)
if ( params . addedTime . isValid ( ) )
p . added_time = params . addedTime . toSecsSinceEpoch ( ) ;
# endif
2019-06-29 11:15:41 +03:00
// Preallocation mode
p . storage_mode = isPreallocationEnabled ( )
? lt : : storage_mode_allocate : lt : : storage_mode_sparse ;
// Seeding mode
// Skip checking and directly start seeding
if ( params . skipChecking ) {
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-06-29 11:15:41 +03:00
p . flags | = lt : : add_torrent_params : : flag_seed_mode ;
2019-04-03 09:23:35 +03:00
# else
2019-06-29 11:15:41 +03:00
p . flags | = lt : : torrent_flags : : seed_mode ;
2019-04-03 09:23:35 +03:00
# endif
2018-09-05 15:59:22 +03:00
}
else {
2019-03-06 08:58:07 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-06-29 11:15:41 +03:00
p . flags & = ~ lt : : add_torrent_params : : flag_seed_mode ;
2019-03-06 08:58:07 +03:00
# else
2019-06-29 11:15:41 +03:00
p . flags & = ~ lt : : torrent_flags : : seed_mode ;
2019-04-03 09:23:35 +03:00
# endif
2019-06-29 11:15:41 +03:00
}
2018-09-05 15:59:22 +03:00
}
2018-07-11 15:44:15 +03:00
m_addingTorrents . insert ( hash , params ) ;
2015-04-19 18:17:47 +03:00
// Adding torrent to BitTorrent session
m_nativeSession - > async_add_torrent ( p ) ;
2019-06-29 11:15:41 +03:00
2015-04-19 18:17:47 +03:00
return true ;
}
2016-04-19 09:54:48 +03:00
bool Session : : findIncompleteFiles ( TorrentInfo & torrentInfo , QString & savePath ) const
{
auto findInDir = [ ] ( const QString & dirPath , TorrentInfo & torrentInfo ) - > bool
{
2017-04-27 22:31:30 +03:00
const QDir dir ( dirPath ) ;
2016-04-19 09:54:48 +03:00
bool found = false ;
2017-04-27 22:31:30 +03:00
for ( int i = 0 ; i < torrentInfo . filesCount ( ) ; + + i ) {
const QString filePath = torrentInfo . filePath ( i ) ;
if ( dir . exists ( filePath ) ) {
2016-04-19 09:54:48 +03:00
found = true ;
}
2017-04-27 22:31:30 +03:00
else if ( dir . exists ( filePath + QB_EXT ) ) {
2016-04-19 09:54:48 +03:00
found = true ;
2017-04-27 22:31:30 +03:00
torrentInfo . renameFile ( i , filePath + QB_EXT ) ;
2016-04-19 09:54:48 +03:00
}
}
return found ;
} ;
bool found = findInDir ( savePath , torrentInfo ) ;
if ( ! found & & isTempPathEnabled ( ) ) {
2017-07-28 12:13:57 +03:00
savePath = torrentTempPath ( torrentInfo ) ;
2016-04-19 09:54:48 +03:00
found = findInDir ( savePath , torrentInfo ) ;
}
return found ;
}
2015-04-19 18:17:47 +03:00
// Add a torrent to the BitTorrent session in hidden mode
// and force it to load its metadata
2016-01-07 14:22:35 +03:00
bool Session : : loadMetadata ( const MagnetUri & magnetUri )
2015-04-19 18:17:47 +03:00
{
2016-01-07 14:22:35 +03:00
if ( ! magnetUri . isValid ( ) ) return false ;
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
const InfoHash hash = magnetUri . hash ( ) ;
const QString name = magnetUri . name ( ) ;
2015-04-19 18:17:47 +03:00
2018-04-14 22:53:45 +03:00
// We should not add torrent if it's already
2015-04-19 18:17:47 +03:00
// processed or adding to session
if ( m_torrents . contains ( hash ) ) return false ;
if ( m_addingTorrents . contains ( hash ) ) return false ;
if ( m_loadedMetadata . contains ( hash ) ) return false ;
qDebug ( " Adding torrent to preload metadata... " ) ;
2017-08-13 13:56:03 +03:00
qDebug ( " -> Hash: %s " , qUtf8Printable ( hash ) ) ;
qDebug ( " -> Name: %s " , qUtf8Printable ( name ) ) ;
2015-04-19 18:17:47 +03:00
2019-05-07 06:22:39 +03:00
lt : : add_torrent_params p = magnetUri . addTorrentParams ( ) ;
2015-04-19 18:17:47 +03:00
// Flags
// Preallocation mode
2016-05-01 11:05:52 +03:00
if ( isPreallocationEnabled ( ) )
2019-05-07 06:22:39 +03:00
p . storage_mode = lt : : storage_mode_allocate ;
2015-04-19 18:17:47 +03:00
else
2019-05-07 06:22:39 +03:00
p . storage_mode = lt : : storage_mode_sparse ;
2015-04-19 18:17:47 +03:00
// Limits
2016-05-01 11:05:52 +03:00
p . max_connections = maxConnectionsPerTorrent ( ) ;
p . max_uploads = maxUploadsPerTorrent ( ) ;
2015-04-19 18:17:47 +03:00
2018-08-10 07:43:25 +03:00
const QString savePath = Utils : : Fs : : tempPath ( ) + static_cast < QString > ( hash ) ;
2017-03-06 18:51:35 +03:00
p . save_path = Utils : : Fs : : toNativePath ( savePath ) . toStdString ( ) ;
2015-04-19 18:17:47 +03:00
// Forced start
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-05-07 06:22:39 +03:00
p . flags & = ~ lt : : add_torrent_params : : flag_paused ;
p . flags & = ~ lt : : add_torrent_params : : flag_auto_managed ;
2019-04-03 09:23:35 +03:00
# else
p . flags & = ~ lt : : torrent_flags : : paused ;
p . flags & = ~ lt : : torrent_flags : : auto_managed ;
# endif
2019-06-28 21:24:39 +03:00
2015-04-19 18:17:47 +03:00
// Solution to avoid accidental file writes
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-05-07 06:22:39 +03:00
p . flags | = lt : : add_torrent_params : : flag_upload_mode ;
2019-04-03 09:23:35 +03:00
# else
p . flags | = lt : : torrent_flags : : upload_mode ;
# endif
2015-04-19 18:17:47 +03:00
// Adding torrent to BitTorrent session
2019-05-07 06:22:39 +03:00
lt : : error_code ec ;
lt : : torrent_handle h = m_nativeSession - > add_torrent ( p , ec ) ;
2015-04-19 18:17:47 +03:00
if ( ec ) return false ;
// waiting for metadata...
m_loadedMetadata . insert ( h . info_hash ( ) , TorrentInfo ( ) ) ;
+ + m_extraLimit ;
adjustLimits ( ) ;
return true ;
}
void Session : : exportTorrentFile ( TorrentHandle * const torrent , TorrentExportFolder folder )
{
2016-05-01 11:05:52 +03:00
Q_ASSERT ( ( ( folder = = TorrentExportFolder : : Regular ) & & ! torrentExportDirectory ( ) . isEmpty ( ) ) | |
( ( folder = = TorrentExportFolder : : Finished ) & & ! finishedTorrentExportDirectory ( ) . isEmpty ( ) ) ) ;
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
const QString validName = Utils : : Fs : : toValidFileSystemName ( torrent - > name ( ) ) ;
const QString torrentFilename = QString ( " %1.torrent " ) . arg ( torrent - > hash ( ) ) ;
2015-12-01 17:41:56 +03:00
QString torrentExportFilename = QString ( " %1.torrent " ) . arg ( validName ) ;
2019-02-09 18:40:14 +03:00
const QString torrentPath = QDir ( m_resumeFolderPath ) . absoluteFilePath ( torrentFilename ) ;
const QDir exportPath ( folder = = TorrentExportFolder : : Regular ? torrentExportDirectory ( ) : finishedTorrentExportDirectory ( ) ) ;
2015-04-19 18:17:47 +03:00
if ( exportPath . exists ( ) | | exportPath . mkpath ( exportPath . absolutePath ( ) ) ) {
2015-12-01 17:41:56 +03:00
QString newTorrentPath = exportPath . absoluteFilePath ( torrentExportFilename ) ;
int counter = 0 ;
while ( QFile : : exists ( newTorrentPath ) & & ! Utils : : Fs : : sameFiles ( torrentPath , newTorrentPath ) ) {
// Append number to torrent name to make it unique
torrentExportFilename = QString ( " %1 %2.torrent " ) . arg ( validName ) . arg ( + + counter ) ;
newTorrentPath = exportPath . absoluteFilePath ( torrentExportFilename ) ;
2015-04-19 18:17:47 +03:00
}
2015-12-01 17:41:56 +03:00
if ( ! QFile : : exists ( newTorrentPath ) )
QFile : : copy ( torrentPath , newTorrentPath ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-02-09 18:40:14 +03:00
void Session : : generateResumeData ( const bool final )
2015-04-19 18:17:47 +03:00
{
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( m_torrents ) ) {
2015-04-19 18:17:47 +03:00
if ( ! torrent - > isValid ( ) ) continue ;
2019-06-12 05:54:43 +03:00
2015-04-19 18:17:47 +03:00
if ( ! final & & ! torrent - > needSaveResumeData ( ) ) continue ;
2019-06-12 05:54:43 +03:00
if ( torrent - > isChecking ( )
| | torrent - > isPaused ( )
| | torrent - > hasError ( )
| | torrent - > hasMissingFiles ( ) )
continue ;
2015-04-19 18:17:47 +03:00
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
}
}
// Called on exit
void Session : : saveResumeData ( )
{
// Pause session
m_nativeSession - > pause ( ) ;
2018-11-19 12:53:29 +03:00
if ( isQueueingSystemEnabled ( ) )
saveTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
generateResumeData ( true ) ;
while ( m_numResumeData > 0 ) {
2019-10-27 08:46:52 +03:00
const std : : vector < lt : : alert * > alerts = getPendingAlerts ( lt : : seconds ( 30 ) ) ;
2015-04-19 18:17:47 +03:00
if ( alerts . empty ( ) ) {
2019-10-27 09:32:32 +03:00
LogMsg ( tr ( " Error: Aborted saving resume data for %1 outstanding torrents. " ) . arg ( QString : : number ( m_numResumeData ) )
, Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
break ;
}
2019-10-27 08:46:52 +03:00
for ( const lt : : alert * a : alerts ) {
2015-04-19 18:17:47 +03:00
switch ( a - > type ( ) ) {
2019-05-07 06:22:39 +03:00
case lt : : save_resume_data_failed_alert : : alert_type :
case lt : : save_resume_data_alert : : alert_type :
2016-11-18 19:04:04 +03:00
dispatchTorrentAlert ( a ) ;
2015-04-19 18:17:47 +03:00
break ;
}
}
}
}
2018-11-19 12:53:29 +03:00
void Session : : saveTorrentsQueue ( )
{
QMap < int , QString > queue ; // Use QMap since it should be ordered by key
2018-11-27 23:15:04 +03:00
for ( const TorrentHandle * torrent : asConst ( torrents ( ) ) ) {
2018-11-19 12:53:29 +03:00
// We require actual (non-cached) queue position here!
2019-07-18 19:53:04 +03:00
const int queuePos = LTUnderlyingType < LTQueuePosition > { torrent - > nativeHandle ( ) . queue_position ( ) } ;
2018-11-19 12:53:29 +03:00
if ( queuePos > = 0 )
queue [ queuePos ] = torrent - > hash ( ) ;
}
QByteArray data ;
2018-11-27 23:15:04 +03:00
for ( const QString & hash : asConst ( queue ) )
2018-11-19 12:53:29 +03:00
data + = ( hash . toLatin1 ( ) + ' \n ' ) ;
const QString filename = QLatin1String { " queue " } ;
2019-06-19 10:35:29 +03:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( m_resumeDataSavingManager
, [ this , data , filename ] ( ) { m_resumeDataSavingManager - > save ( filename , data ) ; } ) ;
# else
2018-11-19 12:53:29 +03:00
QMetaObject : : invokeMethod ( m_resumeDataSavingManager , " save "
, Q_ARG ( QString , filename ) , Q_ARG ( QByteArray , data ) ) ;
2019-06-19 10:35:29 +03:00
# endif
2018-11-19 12:53:29 +03:00
}
void Session : : removeTorrentsQueue ( )
{
const QString filename = QLatin1String { " queue " } ;
2019-06-19 10:35:29 +03:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( m_resumeDataSavingManager
, [ this , filename ] ( ) { m_resumeDataSavingManager - > remove ( filename ) ; } ) ;
# else
2018-11-19 12:53:29 +03:00
QMetaObject : : invokeMethod ( m_resumeDataSavingManager , " remove " , Q_ARG ( QString , filename ) ) ;
2019-06-19 10:35:29 +03:00
# endif
2018-11-19 12:53:29 +03:00
}
2016-02-09 11:56:48 +03:00
void Session : : setDefaultSavePath ( QString path )
2015-04-19 18:17:47 +03:00
{
2016-03-06 09:25:55 +03:00
path = normalizeSavePath ( path ) ;
2016-05-01 11:05:52 +03:00
if ( path = = m_defaultSavePath ) return ;
2015-04-19 18:17:47 +03:00
2016-02-09 11:56:48 +03:00
m_defaultSavePath = path ;
2015-04-19 18:17:47 +03:00
2016-05-08 22:47:50 +03:00
if ( isDisableAutoTMMWhenDefaultSavePathChanged ( ) )
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2016-05-08 22:47:50 +03:00
torrent - > setAutoTMMEnabled ( false ) ;
2016-02-09 11:56:48 +03:00
else
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2016-02-09 11:56:48 +03:00
torrent - > handleCategorySavePathChanged ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-02-09 11:56:48 +03:00
void Session : : setTempPath ( QString path )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
path = normalizeSavePath ( path , defaultSavePath ( ) + " temp/ " ) ;
if ( path = = m_tempPath ) return ;
2015-08-16 20:03:32 +03:00
2016-02-09 11:56:48 +03:00
m_tempPath = path ;
2015-08-16 20:03:32 +03:00
2018-11-27 23:15:04 +03:00
for ( TorrentHandle * const torrent : asConst ( m_torrents ) )
2016-02-09 11:56:48 +03:00
torrent - > handleTempPathChanged ( ) ;
2015-04-19 18:17:47 +03:00
}
2015-06-15 01:06:56 +03:00
void Session : : networkOnlineStateChanged ( const bool online )
{
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " System network status changed to %1 " , " e.g: System network status changed to ONLINE " ) . arg ( online ? tr ( " ONLINE " ) : tr ( " OFFLINE " ) ) , Log : : INFO ) ;
2015-06-15 01:06:56 +03:00
}
2018-04-14 22:53:45 +03:00
void Session : : networkConfigurationChange ( const QNetworkConfiguration & cfg )
2015-06-15 01:06:56 +03:00
{
2016-05-01 11:05:52 +03:00
const QString configuredInterfaceName = networkInterface ( ) ;
2015-12-05 22:22:01 +03:00
// Empty means "Any Interface". In this case libtorrent has binded to 0.0.0.0 so any change to any interface will
// be automatically picked up. Otherwise we would rebinding here to 0.0.0.0 again.
2016-05-01 11:05:52 +03:00
if ( configuredInterfaceName . isEmpty ( ) ) return ;
2015-06-15 01:06:56 +03:00
const QString changedInterface = cfg . name ( ) ;
2016-05-24 01:27:53 +03:00
2015-11-29 15:30:42 +03:00
if ( configuredInterfaceName = = changedInterface ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Network configuration of %1 has changed, refreshing session binding " , " e.g: Network configuration of tun0 has changed, refreshing session binding " ) . arg ( changedInterface ) , Log : : INFO ) ;
2016-05-01 11:05:52 +03:00
configureListeningInterface ( ) ;
2015-06-15 01:06:56 +03:00
}
}
2019-10-16 18:38:51 +03:00
QStringList Session : : getListeningIPs ( ) const
2015-04-19 18:17:47 +03:00
{
2015-11-07 03:06:07 +03:00
QStringList IPs ;
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
const QString ifaceName = networkInterface ( ) ;
const QString ifaceAddr = networkInterfaceAddress ( ) ;
const bool listenIPv6 = isIPv6Enabled ( ) ;
2015-04-19 18:17:47 +03:00
2016-05-27 01:35:58 +03:00
if ( ! ifaceAddr . isEmpty ( ) ) {
QHostAddress addr ( ifaceAddr ) ;
if ( addr . isNull ( ) ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Configured network interface address %1 isn't valid. " , " Configured network interface address 124.5.1568.1 isn't valid. " ) . arg ( ifaceAddr ) , Log : : CRITICAL ) ;
2016-05-27 01:35:58 +03:00
IPs . append ( " 127.0.0.1 " ) ; // Force listening to localhost and avoid accidental connection that will expose user data.
return IPs ;
}
}
if ( ifaceName . isEmpty ( ) ) {
if ( ! ifaceAddr . isEmpty ( ) )
IPs . append ( ifaceAddr ) ;
else
IPs . append ( QString ( ) ) ;
2015-11-07 03:06:07 +03:00
return IPs ;
2015-04-19 18:17:47 +03:00
}
// Attempt to listen on provided interface
2015-11-07 03:06:07 +03:00
const QNetworkInterface networkIFace = QNetworkInterface : : interfaceFromName ( ifaceName ) ;
if ( ! networkIFace . isValid ( ) ) {
2017-08-13 13:56:03 +03:00
qDebug ( " Invalid network interface: %s " , qUtf8Printable ( ifaceName ) ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " The network interface defined is invalid: %1 " ) . arg ( ifaceName ) , Log : : CRITICAL ) ;
2015-11-07 03:06:07 +03:00
IPs . append ( " 127.0.0.1 " ) ; // Force listening to localhost and avoid accidental connection that will expose user data.
return IPs ;
2015-04-19 18:17:47 +03:00
}
2015-11-07 03:06:07 +03:00
const QList < QNetworkAddressEntry > addresses = networkIFace . addressEntries ( ) ;
2015-06-15 01:33:48 +03:00
qDebug ( " This network interface has %d IP addresses " , addresses . size ( ) ) ;
QHostAddress ip ;
QString ipString ;
QAbstractSocket : : NetworkLayerProtocol protocol ;
2018-11-18 21:40:37 +03:00
for ( const QNetworkAddressEntry & entry : addresses ) {
2015-06-15 01:33:48 +03:00
ip = entry . ip ( ) ;
ipString = ip . toString ( ) ;
protocol = ip . protocol ( ) ;
2018-04-14 22:53:45 +03:00
Q_ASSERT ( ( protocol = = QAbstractSocket : : IPv4Protocol ) | | ( protocol = = QAbstractSocket : : IPv6Protocol ) ) ;
2015-11-07 03:06:07 +03:00
if ( ( ! listenIPv6 & & ( protocol = = QAbstractSocket : : IPv6Protocol ) )
| | ( listenIPv6 & & ( protocol = = QAbstractSocket : : IPv4Protocol ) ) )
2015-04-19 18:17:47 +03:00
continue ;
2016-04-13 11:51:29 +03:00
2018-04-14 22:53:45 +03:00
// If an iface address has been defined to only allow ip's that match it to go through
2016-04-13 11:51:29 +03:00
if ( ! ifaceAddr . isEmpty ( ) ) {
2016-05-27 01:35:58 +03:00
if ( ifaceAddr = = ipString ) {
IPs . append ( ipString ) ;
break ;
2016-04-13 11:51:29 +03:00
}
}
2016-05-27 01:35:58 +03:00
else {
IPs . append ( ipString ) ;
}
2015-11-07 03:06:07 +03:00
}
// Make sure there is at least one IP
// At this point there was a valid network interface, with no suitable IP.
if ( IPs . size ( ) = = 0 ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " qBittorrent didn't find an %1 local address to listen on "
, " qBittorrent didn't find an IPv4 local address to listen on " )
. arg ( listenIPv6 ? " IPv6 " : " IPv4 " ) , Log : : CRITICAL ) ;
2015-11-07 03:06:07 +03:00
IPs . append ( " 127.0.0.1 " ) ; // Force listening to localhost and avoid accidental connection that will expose user data.
return IPs ;
}
return IPs ;
}
// Set the ports range in which is chosen the port
// the BitTorrent session will listen to
2016-05-01 11:05:52 +03:00
void Session : : configureListeningInterface ( )
2015-11-07 03:06:07 +03:00
{
2019-10-29 18:16:22 +03:00
m_listenInterfaceConfigured = false ;
2016-06-03 17:03:17 +03:00
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : globalDownloadSpeedLimit ( ) const
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
return m_globalDownloadSpeedLimit * 1024 ;
2015-04-19 18:17:47 +03:00
}
2019-11-16 21:11:17 +03:00
void Session : : setGlobalDownloadSpeedLimit ( const int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
2019-11-20 07:20:17 +03:00
if ( limit = = globalDownloadSpeedLimit ( ) )
return ;
2019-11-16 21:11:17 +03:00
if ( limit < = 0 )
m_globalDownloadSpeedLimit = 0 ;
else if ( limit < = 1024 )
m_globalDownloadSpeedLimit = 1 ;
else
m_globalDownloadSpeedLimit = ( limit / 1024 ) ;
2016-05-01 11:05:52 +03:00
if ( ! isAltGlobalSpeedLimitEnabled ( ) )
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : globalUploadSpeedLimit ( ) const
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
return m_globalUploadSpeedLimit * 1024 ;
2015-04-19 18:17:47 +03:00
}
2019-11-16 21:11:17 +03:00
void Session : : setGlobalUploadSpeedLimit ( const int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
2019-11-20 07:20:17 +03:00
if ( limit = = globalUploadSpeedLimit ( ) )
return ;
2019-11-16 21:11:17 +03:00
if ( limit < = 0 )
m_globalUploadSpeedLimit = 0 ;
else if ( limit < = 1024 )
m_globalUploadSpeedLimit = 1 ;
else
m_globalUploadSpeedLimit = ( limit / 1024 ) ;
2016-05-01 11:05:52 +03:00
if ( ! isAltGlobalSpeedLimitEnabled ( ) )
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : altGlobalDownloadSpeedLimit ( ) const
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
return m_altGlobalDownloadSpeedLimit * 1024 ;
2015-04-19 18:17:47 +03:00
}
2019-11-16 21:11:17 +03:00
void Session : : setAltGlobalDownloadSpeedLimit ( const int limit )
2016-02-09 11:56:48 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
2019-11-20 07:20:17 +03:00
if ( limit = = altGlobalDownloadSpeedLimit ( ) )
return ;
2019-11-16 21:11:17 +03:00
if ( limit < = 0 )
m_altGlobalDownloadSpeedLimit = 0 ;
else if ( limit < = 1024 )
m_altGlobalDownloadSpeedLimit = 1 ;
else
m_altGlobalDownloadSpeedLimit = ( limit / 1024 ) ;
2016-05-01 11:05:52 +03:00
if ( isAltGlobalSpeedLimitEnabled ( ) )
configureDeferred ( ) ;
2016-02-09 11:56:48 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : altGlobalUploadSpeedLimit ( ) const
2016-02-09 11:56:48 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
return m_altGlobalUploadSpeedLimit * 1024 ;
2016-02-09 11:56:48 +03:00
}
2019-11-16 21:11:17 +03:00
void Session : : setAltGlobalUploadSpeedLimit ( const int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 17:52:32 +03:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
2019-11-20 07:20:17 +03:00
if ( limit = = altGlobalUploadSpeedLimit ( ) )
return ;
2019-11-16 21:11:17 +03:00
if ( limit < = 0 )
m_altGlobalUploadSpeedLimit = 0 ;
else if ( limit < = 1024 )
m_altGlobalUploadSpeedLimit = 1 ;
else
m_altGlobalUploadSpeedLimit = ( limit / 1024 ) ;
2016-05-01 11:05:52 +03:00
if ( isAltGlobalSpeedLimitEnabled ( ) )
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : downloadSpeedLimit ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return isAltGlobalSpeedLimitEnabled ( )
? altGlobalDownloadSpeedLimit ( )
: globalDownloadSpeedLimit ( ) ;
2015-04-19 18:17:47 +03:00
}
2018-10-21 08:46:48 +03:00
void Session : : setDownloadSpeedLimit ( const int limit )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( isAltGlobalSpeedLimitEnabled ( ) )
setAltGlobalDownloadSpeedLimit ( limit ) ;
else
setGlobalDownloadSpeedLimit ( limit ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : uploadSpeedLimit ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return isAltGlobalSpeedLimitEnabled ( )
? altGlobalUploadSpeedLimit ( )
: globalUploadSpeedLimit ( ) ;
2015-04-19 18:17:47 +03:00
}
2018-10-21 08:46:48 +03:00
void Session : : setUploadSpeedLimit ( const int limit )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( isAltGlobalSpeedLimitEnabled ( ) )
setAltGlobalUploadSpeedLimit ( limit ) ;
else
setGlobalUploadSpeedLimit ( limit ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
bool Session : : isAltGlobalSpeedLimitEnabled ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isAltGlobalSpeedLimitEnabled ;
2015-04-19 18:17:47 +03:00
}
2018-10-21 08:46:48 +03:00
void Session : : setAltGlobalSpeedLimitEnabled ( const bool enabled )
2015-06-07 15:03:30 +03:00
{
2017-08-20 18:00:23 +03:00
if ( enabled = = isAltGlobalSpeedLimitEnabled ( ) ) return ;
2016-05-01 11:05:52 +03:00
2017-08-20 18:00:23 +03:00
// Save new state to remember it on startup
m_isAltGlobalSpeedLimitEnabled = enabled ;
applyBandwidthLimits ( ) ;
// Notify
emit speedLimitModeChanged ( m_isAltGlobalSpeedLimitEnabled ) ;
2016-02-09 11:56:48 +03:00
}
2015-08-16 20:03:32 +03:00
2016-05-01 11:05:52 +03:00
bool Session : : isBandwidthSchedulerEnabled ( ) const
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isBandwidthSchedulerEnabled ;
2015-06-07 15:03:30 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setBandwidthSchedulerEnabled ( const bool enabled )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( enabled ! = isBandwidthSchedulerEnabled ( ) ) {
m_isBandwidthSchedulerEnabled = enabled ;
if ( enabled )
enableBandwidthScheduler ( ) ;
else
delete m_bwScheduler ;
}
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
uint Session : : saveResumeDataInterval ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_saveResumeDataInterval ;
2015-04-19 18:17:47 +03:00
}
2018-07-04 09:02:05 +03:00
void Session : : setSaveResumeDataInterval ( const uint value )
2015-04-19 18:17:47 +03:00
{
2018-07-04 09:02:05 +03:00
if ( value = = m_saveResumeDataInterval )
return ;
m_saveResumeDataInterval = value ;
if ( value > 0 ) {
2016-05-01 11:05:52 +03:00
m_resumeDataTimer - > setInterval ( value * 60 * 1000 ) ;
2018-07-04 09:02:05 +03:00
m_resumeDataTimer - > start ( ) ;
}
else {
m_resumeDataTimer - > stop ( ) ;
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : port ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_port ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setPort ( const int port )
2015-04-19 18:17:47 +03:00
{
2019-04-22 06:02:31 +03:00
if ( port ! = m_port ) {
2016-05-01 11:05:52 +03:00
m_port = port ;
configureListeningInterface ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
bool Session : : useRandomPort ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_useRandomPort ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setUseRandomPort ( const bool value )
2016-05-01 11:05:52 +03:00
{
m_useRandomPort = value ;
}
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
QString Session : : networkInterface ( ) const
{
return m_networkInterface ;
2015-04-19 18:17:47 +03:00
}
2017-08-13 00:58:22 +03:00
void Session : : setNetworkInterface ( const QString & iface )
2015-04-19 18:17:47 +03:00
{
2017-08-13 00:58:22 +03:00
if ( iface ! = networkInterface ( ) ) {
m_networkInterface = iface ;
2016-05-01 11:05:52 +03:00
configureListeningInterface ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2016-10-31 02:40:26 +03:00
QString Session : : networkInterfaceName ( ) const
{
return m_networkInterfaceName ;
}
void Session : : setNetworkInterfaceName ( const QString & name )
{
m_networkInterfaceName = name ;
}
2016-05-01 11:05:52 +03:00
QString Session : : networkInterfaceAddress ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_networkInterfaceAddress ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
void Session : : setNetworkInterfaceAddress ( const QString & address )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( address ! = networkInterfaceAddress ( ) ) {
m_networkInterfaceAddress = address ;
configureListeningInterface ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
bool Session : : isIPv6Enabled ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isIPv6Enabled ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setIPv6Enabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = isIPv6Enabled ( ) ) {
m_isIPv6Enabled = enabled ;
configureListeningInterface ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
int Session : : encryption ( ) const
{
return m_encryption ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setEncryption ( const int state )
2016-05-01 11:05:52 +03:00
{
if ( state ! = encryption ( ) ) {
m_encryption = state ;
2016-06-03 17:03:17 +03:00
configureDeferred ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Encryption support [%1] " ) . arg (
state = = 0 ? tr ( " ON " ) : ( ( state = = 1 ) ? tr ( " FORCED " ) : tr ( " OFF " ) ) )
, Log : : INFO ) ;
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
bool Session : : isProxyPeerConnectionsEnabled ( ) const
{
return m_isProxyPeerConnectionsEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setProxyPeerConnectionsEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = isProxyPeerConnectionsEnabled ( ) ) {
m_isProxyPeerConnectionsEnabled = enabled ;
configureDeferred ( ) ;
}
}
2017-08-15 21:23:07 +03:00
ChokingAlgorithm Session : : chokingAlgorithm ( ) const
2017-08-11 12:06:31 +03:00
{
return m_chokingAlgorithm ;
}
2019-02-09 18:40:14 +03:00
void Session : : setChokingAlgorithm ( const ChokingAlgorithm mode )
2017-08-11 12:06:31 +03:00
{
if ( mode = = m_chokingAlgorithm ) return ;
m_chokingAlgorithm = mode ;
configureDeferred ( ) ;
}
2017-08-15 21:23:07 +03:00
SeedChokingAlgorithm Session : : seedChokingAlgorithm ( ) const
2017-08-11 12:06:31 +03:00
{
return m_seedChokingAlgorithm ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSeedChokingAlgorithm ( const SeedChokingAlgorithm mode )
2017-08-11 12:06:31 +03:00
{
if ( mode = = m_seedChokingAlgorithm ) return ;
m_seedChokingAlgorithm = mode ;
configureDeferred ( ) ;
}
2016-05-01 11:05:52 +03:00
bool Session : : isAddTrackersEnabled ( ) const
{
return m_isAddTrackersEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setAddTrackersEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
m_isAddTrackersEnabled = enabled ;
}
QString Session : : additionalTrackers ( ) const
{
return m_additionalTrackers ;
}
void Session : : setAdditionalTrackers ( const QString & trackers )
{
if ( trackers ! = additionalTrackers ( ) ) {
m_additionalTrackers = trackers ;
populateAdditionalTrackers ( ) ;
}
}
2016-10-29 19:14:27 +03:00
bool Session : : isIPFilteringEnabled ( ) const
2016-05-01 11:05:52 +03:00
{
2016-10-29 19:14:27 +03:00
return m_isIPFilteringEnabled ;
2016-05-01 11:05:52 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setIPFilteringEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
2016-10-29 19:14:27 +03:00
if ( enabled ! = m_isIPFilteringEnabled ) {
m_isIPFilteringEnabled = enabled ;
2019-10-29 18:16:22 +03:00
m_IPFilteringConfigured = false ;
2016-05-01 11:05:52 +03:00
configureDeferred ( ) ;
}
}
QString Session : : IPFilterFile ( ) const
{
2019-06-16 20:14:15 +03:00
return Utils : : Fs : : toUniformPath ( m_IPFilterFile ) ;
2016-05-01 11:05:52 +03:00
}
void Session : : setIPFilterFile ( QString path )
{
2019-06-16 20:14:15 +03:00
path = Utils : : Fs : : toUniformPath ( path ) ;
2016-05-01 11:05:52 +03:00
if ( path ! = IPFilterFile ( ) ) {
m_IPFilterFile = path ;
2019-10-29 18:16:22 +03:00
m_IPFilteringConfigured = false ;
2016-05-01 11:05:52 +03:00
configureDeferred ( ) ;
}
}
2017-03-06 15:40:34 +03:00
void Session : : setBannedIPs ( const QStringList & newList )
{
if ( newList = = m_bannedIPs )
return ; // do nothing
// here filter out incorrect IP
QStringList filteredList ;
for ( const QString & ip : newList ) {
if ( Utils : : Net : : isValidIP ( ip ) ) {
// the same IPv6 addresses could be written in different forms;
// QHostAddress::toString() result format follows RFC5952;
// thus we avoid duplicate entries pointing to the same address
filteredList < < QHostAddress ( ip ) . toString ( ) ;
}
else {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " %1 is not a valid IP address and was rejected while applying the list of banned addresses. " )
. arg ( ip )
, Log : : WARNING ) ;
2017-03-06 15:40:34 +03:00
}
}
// now we have to sort IPs and make them unique
filteredList . sort ( ) ;
filteredList . removeDuplicates ( ) ;
// Again ensure that the new list is different from the stored one.
if ( filteredList = = m_bannedIPs )
return ; // do nothing
// store to session settings
// also here we have to recreate filter list including 3rd party ban file
// and install it again into m_session
m_bannedIPs = filteredList ;
2019-10-29 18:16:22 +03:00
m_IPFilteringConfigured = false ;
2017-03-06 15:40:34 +03:00
configureDeferred ( ) ;
}
QStringList Session : : bannedIPs ( ) const
{
return m_bannedIPs ;
}
2019-09-28 09:21:53 +03:00
# if defined(Q_OS_WIN)
OSMemoryPriority Session : : getOSMemoryPriority ( ) const
{
return m_OSMemoryPriority ;
}
void Session : : setOSMemoryPriority ( const OSMemoryPriority priority )
{
if ( m_OSMemoryPriority = = priority )
return ;
m_OSMemoryPriority = priority ;
configureDeferred ( ) ;
}
void Session : : applyOSMemoryPriority ( ) const
{
using SETPROCESSINFORMATION = BOOL ( WINAPI * ) ( HANDLE , PROCESS_INFORMATION_CLASS , LPVOID , DWORD ) ;
const auto setProcessInformation = Utils : : Misc : : loadWinAPI < SETPROCESSINFORMATION > ( " Kernel32.dll " , " SetProcessInformation " ) ;
if ( ! setProcessInformation ) // only available on Windows >= 8
return ;
# if (_WIN32_WINNT < _WIN32_WINNT_WIN8)
// this dummy struct is required to compile successfully when targeting older Windows version
struct MEMORY_PRIORITY_INFORMATION
{
ULONG MemoryPriority ;
} ;
2019-11-21 19:31:04 +03:00
# define MEMORY_PRIORITY_LOWEST 0
# define MEMORY_PRIORITY_VERY_LOW 1
# define MEMORY_PRIORITY_LOW 2
# define MEMORY_PRIORITY_MEDIUM 3
# define MEMORY_PRIORITY_BELOW_NORMAL 4
# define MEMORY_PRIORITY_NORMAL 5
2019-09-28 09:21:53 +03:00
# endif
MEMORY_PRIORITY_INFORMATION prioInfo { } ;
switch ( getOSMemoryPriority ( ) ) {
case OSMemoryPriority : : Normal :
default :
prioInfo . MemoryPriority = MEMORY_PRIORITY_NORMAL ;
break ;
case OSMemoryPriority : : BelowNormal :
prioInfo . MemoryPriority = MEMORY_PRIORITY_BELOW_NORMAL ;
break ;
case OSMemoryPriority : : Medium :
prioInfo . MemoryPriority = MEMORY_PRIORITY_MEDIUM ;
break ;
case OSMemoryPriority : : Low :
prioInfo . MemoryPriority = MEMORY_PRIORITY_LOW ;
break ;
case OSMemoryPriority : : VeryLow :
prioInfo . MemoryPriority = MEMORY_PRIORITY_VERY_LOW ;
break ;
}
setProcessInformation ( : : GetCurrentProcess ( ) , ProcessMemoryPriority , & prioInfo , sizeof ( prioInfo ) ) ;
}
# endif
2016-05-01 11:05:52 +03:00
int Session : : maxConnectionsPerTorrent ( ) const
{
return m_maxConnectionsPerTorrent ;
}
void Session : : setMaxConnectionsPerTorrent ( int max )
{
max = ( max > 0 ) ? max : - 1 ;
if ( max ! = maxConnectionsPerTorrent ( ) ) {
m_maxConnectionsPerTorrent = max ;
// Apply this to all session torrents
2019-04-03 09:23:35 +03:00
for ( const lt : : torrent_handle & handle : m_nativeSession - > get_torrents ( ) ) {
2016-05-01 11:05:52 +03:00
if ( ! handle . is_valid ( ) ) continue ;
try {
handle . set_max_connections ( max ) ;
}
2018-05-07 14:44:11 +03:00
catch ( const std : : exception & ) { }
2016-05-01 11:05:52 +03:00
}
}
}
int Session : : maxUploadsPerTorrent ( ) const
{
return m_maxUploadsPerTorrent ;
}
void Session : : setMaxUploadsPerTorrent ( int max )
{
max = ( max > 0 ) ? max : - 1 ;
if ( max ! = maxUploadsPerTorrent ( ) ) {
m_maxUploadsPerTorrent = max ;
// Apply this to all session torrents
2019-04-03 09:23:35 +03:00
for ( const lt : : torrent_handle & handle : m_nativeSession - > get_torrents ( ) ) {
2016-05-01 11:05:52 +03:00
if ( ! handle . is_valid ( ) ) continue ;
try {
handle . set_max_uploads ( max ) ;
}
2018-05-07 14:44:11 +03:00
catch ( const std : : exception & ) { }
2016-05-01 11:05:52 +03:00
}
}
}
bool Session : : announceToAllTrackers ( ) const
{
return m_announceToAllTrackers ;
}
2019-02-09 18:40:14 +03:00
void Session : : setAnnounceToAllTrackers ( const bool val )
2016-05-01 11:05:52 +03:00
{
if ( val ! = m_announceToAllTrackers ) {
m_announceToAllTrackers = val ;
configureDeferred ( ) ;
}
}
2017-10-30 21:04:14 +03:00
bool Session : : announceToAllTiers ( ) const
{
return m_announceToAllTiers ;
}
2019-02-09 18:40:14 +03:00
void Session : : setAnnounceToAllTiers ( const bool val )
2017-10-30 21:04:14 +03:00
{
if ( val ! = m_announceToAllTiers ) {
m_announceToAllTiers = val ;
configureDeferred ( ) ;
}
}
2018-06-18 19:10:57 +03:00
int Session : : asyncIOThreads ( ) const
{
return qBound ( 1 , m_asyncIOThreads . value ( ) , 1024 ) ;
}
void Session : : setAsyncIOThreads ( const int num )
{
if ( num = = m_asyncIOThreads )
return ;
m_asyncIOThreads = num ;
configureDeferred ( ) ;
}
2019-07-03 12:44:16 +03:00
int Session : : filePoolSize ( ) const
{
return m_filePoolSize ;
}
void Session : : setFilePoolSize ( const int size )
{
if ( size = = m_filePoolSize )
return ;
m_filePoolSize = size ;
configureDeferred ( ) ;
}
2018-08-28 13:18:07 +03:00
int Session : : checkingMemUsage ( ) const
{
return qMax ( 1 , m_checkingMemUsage . value ( ) ) ;
}
void Session : : setCheckingMemUsage ( int size )
{
size = qMax ( size , 1 ) ;
if ( size = = m_checkingMemUsage )
return ;
m_checkingMemUsage = size ;
configureDeferred ( ) ;
}
2017-08-18 03:05:36 +03:00
int Session : : diskCacheSize ( ) const
2016-05-01 11:05:52 +03:00
{
2019-01-16 07:52:41 +03:00
# ifdef QBT_APP_64BIT
2019-06-19 17:01:25 +03:00
return qMin ( m_diskCacheSize . value ( ) , 33554431 ) ; // 32768GiB
2016-05-01 11:05:52 +03:00
# else
// When build as 32bit binary, set the maximum at less than 2GB to prevent crashes
// allocate 1536MiB and leave 512MiB to the rest of program data in RAM
2019-06-19 17:01:25 +03:00
return qMin ( m_diskCacheSize . value ( ) , 1536 ) ;
2016-05-01 11:05:52 +03:00
# endif
}
2017-08-18 03:05:36 +03:00
void Session : : setDiskCacheSize ( int size )
2016-05-01 11:05:52 +03:00
{
2019-01-16 07:52:41 +03:00
# ifdef QBT_APP_64BIT
2019-06-19 17:01:25 +03:00
size = qMin ( size , 33554431 ) ; // 32768GiB
2016-05-01 11:05:52 +03:00
# else
// allocate 1536MiB and leave 512MiB to the rest of program data in RAM
2017-08-18 03:05:36 +03:00
size = qMin ( size , 1536 ) ;
2016-05-01 11:05:52 +03:00
# endif
if ( size ! = m_diskCacheSize ) {
m_diskCacheSize = size ;
configureDeferred ( ) ;
}
}
2017-08-18 20:20:53 +03:00
int Session : : diskCacheTTL ( ) const
2016-05-01 11:05:52 +03:00
{
return m_diskCacheTTL ;
}
2019-02-09 18:40:14 +03:00
void Session : : setDiskCacheTTL ( const int ttl )
2016-05-01 11:05:52 +03:00
{
if ( ttl ! = m_diskCacheTTL ) {
m_diskCacheTTL = ttl ;
configureDeferred ( ) ;
}
}
bool Session : : useOSCache ( ) const
{
return m_useOSCache ;
}
2019-02-09 18:40:14 +03:00
void Session : : setUseOSCache ( const bool use )
2016-05-01 11:05:52 +03:00
{
if ( use ! = m_useOSCache ) {
m_useOSCache = use ;
configureDeferred ( ) ;
}
}
2018-01-29 12:03:14 +03:00
bool Session : : isCoalesceReadWriteEnabled ( ) const
{
return m_coalesceReadWriteEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setCoalesceReadWriteEnabled ( const bool enabled )
2018-01-29 12:03:14 +03:00
{
if ( enabled = = m_coalesceReadWriteEnabled ) return ;
m_coalesceReadWriteEnabled = enabled ;
configureDeferred ( ) ;
}
2017-08-11 14:15:18 +03:00
bool Session : : isSuggestModeEnabled ( ) const
{
return m_isSuggestMode ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSuggestMode ( const bool mode )
2017-08-11 14:15:18 +03:00
{
if ( mode = = m_isSuggestMode ) return ;
m_isSuggestMode = mode ;
configureDeferred ( ) ;
}
2017-08-11 14:48:58 +03:00
int Session : : sendBufferWatermark ( ) const
{
return m_sendBufferWatermark ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSendBufferWatermark ( const int value )
2017-08-11 14:48:58 +03:00
{
if ( value = = m_sendBufferWatermark ) return ;
m_sendBufferWatermark = value ;
configureDeferred ( ) ;
}
int Session : : sendBufferLowWatermark ( ) const
{
return m_sendBufferLowWatermark ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSendBufferLowWatermark ( const int value )
2017-08-11 14:48:58 +03:00
{
if ( value = = m_sendBufferLowWatermark ) return ;
m_sendBufferLowWatermark = value ;
configureDeferred ( ) ;
}
int Session : : sendBufferWatermarkFactor ( ) const
{
return m_sendBufferWatermarkFactor ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSendBufferWatermarkFactor ( const int value )
2017-08-11 14:48:58 +03:00
{
if ( value = = m_sendBufferWatermarkFactor ) return ;
m_sendBufferWatermarkFactor = value ;
configureDeferred ( ) ;
}
2019-07-02 07:03:44 +03:00
int Session : : socketBacklogSize ( ) const
{
return m_socketBacklogSize ;
}
void Session : : setSocketBacklogSize ( const int value )
{
if ( value = = m_socketBacklogSize ) return ;
m_socketBacklogSize = value ;
configureDeferred ( ) ;
}
2016-05-01 11:05:52 +03:00
bool Session : : isAnonymousModeEnabled ( ) const
{
return m_isAnonymousModeEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setAnonymousModeEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = m_isAnonymousModeEnabled ) {
m_isAnonymousModeEnabled = enabled ;
configureDeferred ( ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Anonymous mode [%1] " ) . arg ( isAnonymousModeEnabled ( ) ? tr ( " ON " ) : tr ( " OFF " ) )
, Log : : INFO ) ;
2016-05-01 11:05:52 +03:00
}
}
bool Session : : isQueueingSystemEnabled ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isQueueingEnabled ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setQueueingSystemEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = m_isQueueingEnabled ) {
m_isQueueingEnabled = enabled ;
configureDeferred ( ) ;
2018-11-19 12:53:29 +03:00
if ( enabled )
saveTorrentsQueue ( ) ;
else
removeTorrentsQueue ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
int Session : : maxActiveDownloads ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_maxActiveDownloads ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
void Session : : setMaxActiveDownloads ( int max )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
max = std : : max ( max , - 1 ) ;
if ( max ! = m_maxActiveDownloads ) {
m_maxActiveDownloads = max ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
int Session : : maxActiveUploads ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_maxActiveUploads ;
}
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
void Session : : setMaxActiveUploads ( int max )
{
max = std : : max ( max , - 1 ) ;
if ( max ! = m_maxActiveUploads ) {
m_maxActiveUploads = max ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
int Session : : maxActiveTorrents ( ) const
2015-07-22 08:51:23 +03:00
{
2016-05-01 11:05:52 +03:00
return m_maxActiveTorrents ;
2015-07-22 08:51:23 +03:00
}
2016-05-01 11:05:52 +03:00
void Session : : setMaxActiveTorrents ( int max )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
max = std : : max ( max , - 1 ) ;
if ( max ! = m_maxActiveTorrents ) {
m_maxActiveTorrents = max ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
bool Session : : ignoreSlowTorrentsForQueueing ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_ignoreSlowTorrentsForQueueing ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setIgnoreSlowTorrentsForQueueing ( const bool ignore )
2016-05-01 11:05:52 +03:00
{
if ( ignore ! = m_ignoreSlowTorrentsForQueueing ) {
m_ignoreSlowTorrentsForQueueing = ignore ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2018-02-02 14:22:24 +03:00
int Session : : downloadRateForSlowTorrents ( ) const
{
return m_downloadRateForSlowTorrents ;
}
2019-02-09 18:40:14 +03:00
void Session : : setDownloadRateForSlowTorrents ( const int rateInKibiBytes )
2018-02-02 14:22:24 +03:00
{
if ( rateInKibiBytes = = m_downloadRateForSlowTorrents )
return ;
m_downloadRateForSlowTorrents = rateInKibiBytes ;
configureDeferred ( ) ;
}
int Session : : uploadRateForSlowTorrents ( ) const
{
return m_uploadRateForSlowTorrents ;
}
2019-02-09 18:40:14 +03:00
void Session : : setUploadRateForSlowTorrents ( const int rateInKibiBytes )
2018-02-02 14:22:24 +03:00
{
if ( rateInKibiBytes = = m_uploadRateForSlowTorrents )
return ;
m_uploadRateForSlowTorrents = rateInKibiBytes ;
configureDeferred ( ) ;
}
int Session : : slowTorrentsInactivityTimer ( ) const
{
return m_slowTorrentsInactivityTimer ;
}
2019-02-09 18:40:14 +03:00
void Session : : setSlowTorrentsInactivityTimer ( const int timeInSeconds )
2018-02-02 14:22:24 +03:00
{
if ( timeInSeconds = = m_slowTorrentsInactivityTimer )
return ;
m_slowTorrentsInactivityTimer = timeInSeconds ;
configureDeferred ( ) ;
}
2017-08-18 20:20:53 +03:00
int Session : : outgoingPortsMin ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_outgoingPortsMin ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setOutgoingPortsMin ( const int min )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( min ! = m_outgoingPortsMin ) {
m_outgoingPortsMin = min ;
configureDeferred ( ) ;
}
}
2015-08-16 20:03:32 +03:00
2017-08-18 20:20:53 +03:00
int Session : : outgoingPortsMax ( ) const
2016-05-01 11:05:52 +03:00
{
return m_outgoingPortsMax ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setOutgoingPortsMax ( const int max )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( max ! = m_outgoingPortsMax ) {
m_outgoingPortsMax = max ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
bool Session : : ignoreLimitsOnLAN ( ) const
2015-06-15 01:06:56 +03:00
{
2016-05-01 11:05:52 +03:00
return m_ignoreLimitsOnLAN ;
2015-06-15 01:06:56 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setIgnoreLimitsOnLAN ( const bool ignore )
2015-06-15 01:06:56 +03:00
{
2016-05-01 11:05:52 +03:00
if ( ignore ! = m_ignoreLimitsOnLAN ) {
m_ignoreLimitsOnLAN = ignore ;
configureDeferred ( ) ;
2015-06-15 01:06:56 +03:00
}
}
2016-05-01 11:05:52 +03:00
bool Session : : includeOverheadInLimits ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_includeOverheadInLimits ;
}
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setIncludeOverheadInLimits ( const bool include )
2016-05-01 11:05:52 +03:00
{
if ( include ! = m_includeOverheadInLimits ) {
m_includeOverheadInLimits = include ;
configureDeferred ( ) ;
2016-05-27 01:35:58 +03:00
}
2016-05-01 11:05:52 +03:00
}
2016-05-27 01:35:58 +03:00
2016-10-31 02:22:11 +03:00
QString Session : : announceIP ( ) const
2016-05-01 11:05:52 +03:00
{
2016-10-31 02:22:11 +03:00
return m_announceIP ;
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
2016-10-31 02:22:11 +03:00
void Session : : setAnnounceIP ( const QString & ip )
2016-05-01 11:05:52 +03:00
{
2016-10-31 02:22:11 +03:00
if ( ip ! = m_announceIP ) {
m_announceIP = ip ;
2016-05-01 11:05:52 +03:00
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
2016-05-01 11:05:52 +03:00
bool Session : : isSuperSeedingEnabled ( ) const
{
return m_isSuperSeedingEnabled ;
}
2016-04-13 11:51:29 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setSuperSeedingEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = m_isSuperSeedingEnabled ) {
m_isSuperSeedingEnabled = enabled ;
configureDeferred ( ) ;
2015-11-07 03:06:07 +03:00
}
2016-05-01 11:05:52 +03:00
}
2015-11-07 03:06:07 +03:00
2016-05-01 11:05:52 +03:00
int Session : : maxConnections ( ) const
{
return m_maxConnections ;
}
2015-11-07 03:06:07 +03:00
2016-05-01 11:05:52 +03:00
void Session : : setMaxConnections ( int max )
{
max = ( max > 0 ) ? max : - 1 ;
if ( max ! = m_maxConnections ) {
m_maxConnections = max ;
configureDeferred ( ) ;
}
2015-11-07 03:06:07 +03:00
}
2016-05-01 11:05:52 +03:00
int Session : : maxUploads ( ) const
{
return m_maxUploads ;
}
2015-11-07 03:06:07 +03:00
2016-05-01 11:05:52 +03:00
void Session : : setMaxUploads ( int max )
{
max = ( max > 0 ) ? max : - 1 ;
if ( max ! = m_maxUploads ) {
m_maxUploads = max ;
configureDeferred ( ) ;
}
}
2015-11-07 03:06:07 +03:00
2017-09-13 23:29:54 +03:00
BTProtocol Session : : btProtocol ( ) const
2016-05-01 11:05:52 +03:00
{
2017-09-13 23:29:54 +03:00
return m_btProtocol ;
2016-05-01 11:05:52 +03:00
}
2015-11-07 03:06:07 +03:00
2019-02-09 18:40:14 +03:00
void Session : : setBTProtocol ( const BTProtocol protocol )
2016-05-01 11:05:52 +03:00
{
2017-09-13 23:29:54 +03:00
if ( ( protocol < BTProtocol : : Both ) | | ( BTProtocol : : UTP < protocol ) )
return ;
if ( protocol = = m_btProtocol ) return ;
m_btProtocol = protocol ;
configureDeferred ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-05-01 11:05:52 +03:00
bool Session : : isUTPRateLimited ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isUTPRateLimited ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setUTPRateLimited ( const bool limited )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( limited ! = m_isUTPRateLimited ) {
m_isUTPRateLimited = limited ;
configureDeferred ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2017-08-15 21:23:07 +03:00
MixedModeAlgorithm Session : : utpMixedMode ( ) const
2017-08-11 11:28:52 +03:00
{
return m_utpMixedMode ;
}
2019-02-09 18:40:14 +03:00
void Session : : setUtpMixedMode ( const MixedModeAlgorithm mode )
2017-08-11 11:28:52 +03:00
{
if ( mode = = m_utpMixedMode ) return ;
m_utpMixedMode = mode ;
configureDeferred ( ) ;
}
2017-08-11 11:37:32 +03:00
bool Session : : multiConnectionsPerIpEnabled ( ) const
{
return m_multiConnectionsPerIpEnabled ;
}
2019-02-09 18:40:14 +03:00
void Session : : setMultiConnectionsPerIpEnabled ( const bool enabled )
2017-08-11 11:37:32 +03:00
{
if ( enabled = = m_multiConnectionsPerIpEnabled ) return ;
m_multiConnectionsPerIpEnabled = enabled ;
configureDeferred ( ) ;
}
2016-05-01 11:05:52 +03:00
bool Session : : isTrackerFilteringEnabled ( ) const
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
return m_isTrackerFilteringEnabled ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setTrackerFilteringEnabled ( const bool enabled )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( enabled ! = m_isTrackerFilteringEnabled ) {
m_isTrackerFilteringEnabled = enabled ;
configureDeferred ( ) ;
}
2015-04-19 18:17:47 +03:00
}
bool Session : : isListening ( ) const
{
return m_nativeSession - > is_listening ( ) ;
}
2016-02-09 11:56:48 +03:00
MaxRatioAction Session : : maxRatioAction ( ) const
{
2016-05-01 11:05:52 +03:00
return static_cast < MaxRatioAction > ( m_maxRatioAction . value ( ) ) ;
2016-02-09 11:56:48 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : setMaxRatioAction ( const MaxRatioAction act )
2016-02-09 11:56:48 +03:00
{
2016-05-01 11:05:52 +03:00
m_maxRatioAction = static_cast < int > ( act ) ;
2015-04-19 18:17:47 +03:00
}
// If this functions returns true, we cannot add torrent to session,
2018-07-15 05:19:24 +03:00
// but it is still possible to merge trackers in some cases
2015-04-19 18:17:47 +03:00
bool Session : : isKnownTorrent ( const InfoHash & hash ) const
{
return ( m_torrents . contains ( hash )
| | m_addingTorrents . contains ( hash )
| | m_loadedMetadata . contains ( hash ) ) ;
}
2016-02-07 20:31:50 +03:00
void Session : : updateSeedingLimitTimer ( )
2015-04-19 18:17:47 +03:00
{
2017-09-06 22:47:05 +03:00
if ( ( globalMaxRatio ( ) = = TorrentHandle : : NO_RATIO_LIMIT ) & & ! hasPerTorrentRatioLimit ( )
2016-02-07 20:31:50 +03:00
& & ( globalMaxSeedingMinutes ( ) = = TorrentHandle : : NO_SEEDING_TIME_LIMIT ) & & ! hasPerTorrentSeedingTimeLimit ( ) ) {
if ( m_seedingLimitTimer - > isActive ( ) )
m_seedingLimitTimer - > stop ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-02-07 20:31:50 +03:00
else if ( ! m_seedingLimitTimer - > isActive ( ) ) {
m_seedingLimitTimer - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-02-07 20:31:50 +03:00
void Session : : handleTorrentShareLimitChanged ( TorrentHandle * const torrent )
2015-04-19 18:17:47 +03:00
{
2018-08-23 14:55:27 +03:00
torrent - > saveResumeData ( ) ;
2019-07-22 14:22:26 +03:00
updateSeedingLimitTimer ( ) ;
2015-04-19 18:17:47 +03:00
}
2018-07-10 14:37:45 +03:00
void Session : : handleTorrentNameChanged ( TorrentHandle * const torrent )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2018-07-10 14:37:45 +03:00
}
2015-04-19 18:17:47 +03:00
void Session : : handleTorrentSavePathChanged ( TorrentHandle * const torrent )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
emit torrentSavePathChanged ( torrent ) ;
}
2016-02-09 11:56:48 +03:00
void Session : : handleTorrentCategoryChanged ( TorrentHandle * const torrent , const QString & oldCategory )
2015-06-07 15:03:30 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2016-02-09 11:56:48 +03:00
emit torrentCategoryChanged ( torrent , oldCategory ) ;
}
2015-08-16 20:03:32 +03:00
2017-06-05 03:22:17 +03:00
void Session : : handleTorrentTagAdded ( TorrentHandle * const torrent , const QString & tag )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2017-06-05 03:22:17 +03:00
emit torrentTagAdded ( torrent , tag ) ;
}
void Session : : handleTorrentTagRemoved ( TorrentHandle * const torrent , const QString & tag )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2017-06-05 03:22:17 +03:00
emit torrentTagRemoved ( torrent , tag ) ;
}
2018-04-14 22:53:45 +03:00
void Session : : handleTorrentSavingModeChanged ( TorrentHandle * const torrent )
2016-02-09 11:56:48 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2016-02-09 11:56:48 +03:00
emit torrentSavingModeChanged ( torrent ) ;
2015-06-07 15:03:30 +03:00
}
2019-05-21 07:21:17 +03:00
void Session : : handleTorrentTrackersAdded ( TorrentHandle * const torrent , const QVector < TrackerEntry > & newTrackers )
2015-04-19 18:17:47 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2018-07-10 14:37:45 +03:00
for ( const TrackerEntry & newTracker : newTrackers )
LogMsg ( tr ( " Tracker '%1' was added to torrent '%2' " ) . arg ( newTracker . url ( ) , torrent - > name ( ) ) ) ;
2015-04-19 18:17:47 +03:00
emit trackersAdded ( torrent , newTrackers ) ;
if ( torrent - > trackers ( ) . size ( ) = = newTrackers . size ( ) )
emit trackerlessStateChanged ( torrent , false ) ;
emit trackersChanged ( torrent ) ;
}
2019-05-21 07:21:17 +03:00
void Session : : handleTorrentTrackersRemoved ( TorrentHandle * const torrent , const QVector < TrackerEntry > & deletedTrackers )
2015-04-19 18:17:47 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2018-07-10 14:37:45 +03:00
for ( const TrackerEntry & deletedTracker : deletedTrackers )
LogMsg ( tr ( " Tracker '%1' was deleted from torrent '%2' " ) . arg ( deletedTracker . url ( ) , torrent - > name ( ) ) ) ;
2015-04-19 18:17:47 +03:00
emit trackersRemoved ( torrent , deletedTrackers ) ;
if ( torrent - > trackers ( ) . size ( ) = = 0 )
emit trackerlessStateChanged ( torrent , true ) ;
emit trackersChanged ( torrent ) ;
}
void Session : : handleTorrentTrackersChanged ( TorrentHandle * const torrent )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
emit trackersChanged ( torrent ) ;
}
2019-08-02 07:55:06 +03:00
void Session : : handleTorrentUrlSeedsAdded ( TorrentHandle * const torrent , const QVector < QUrl > & newUrlSeeds )
2015-04-19 18:17:47 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2018-07-10 14:37:45 +03:00
for ( const QUrl & newUrlSeed : newUrlSeeds )
LogMsg ( tr ( " URL seed '%1' was added to torrent '%2' " ) . arg ( newUrlSeed . toString ( ) , torrent - > name ( ) ) ) ;
2015-04-19 18:17:47 +03:00
}
2019-08-02 07:55:06 +03:00
void Session : : handleTorrentUrlSeedsRemoved ( TorrentHandle * const torrent , const QVector < QUrl > & urlSeeds )
2015-04-19 18:17:47 +03:00
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2018-07-10 14:37:45 +03:00
for ( const QUrl & urlSeed : urlSeeds )
LogMsg ( tr ( " URL seed '%1' was removed from torrent '%2' " ) . arg ( urlSeed . toString ( ) , torrent - > name ( ) ) ) ;
2015-04-19 18:17:47 +03:00
}
void Session : : handleTorrentMetadataReceived ( TorrentHandle * const torrent )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
// Save metadata
const QDir resumeDataDir ( m_resumeFolderPath ) ;
QString torrentFile = resumeDataDir . absoluteFilePath ( QString ( " %1.torrent " ) . arg ( torrent - > hash ( ) ) ) ;
if ( torrent - > saveTorrentFile ( torrentFile ) ) {
// Copy the torrent file to the export folder
2016-05-01 11:05:52 +03:00
if ( ! torrentExportDirectory ( ) . isEmpty ( ) )
2015-04-19 18:17:47 +03:00
exportTorrentFile ( torrent ) ;
}
emit torrentMetadataLoaded ( torrent ) ;
}
void Session : : handleTorrentPaused ( TorrentHandle * const torrent )
{
if ( ! torrent - > hasError ( ) & & ! torrent - > hasMissingFiles ( ) )
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
emit torrentPaused ( torrent ) ;
}
void Session : : handleTorrentResumed ( TorrentHandle * const torrent )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
emit torrentResumed ( torrent ) ;
}
void Session : : handleTorrentChecked ( TorrentHandle * const torrent )
{
emit torrentFinishedChecking ( torrent ) ;
}
void Session : : handleTorrentFinished ( TorrentHandle * const torrent )
{
2018-11-19 12:53:29 +03:00
if ( ! torrent - > hasError ( ) & & ! torrent - > hasMissingFiles ( ) )
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
emit torrentFinished ( torrent ) ;
qDebug ( " Checking if the torrent contains torrent files to download " ) ;
// Check if there are torrent files inside
for ( int i = 0 ; i < torrent - > filesCount ( ) ; + + i ) {
const QString torrentRelpath = torrent - > filePath ( i ) ;
if ( torrentRelpath . endsWith ( " .torrent " , Qt : : CaseInsensitive ) ) {
qDebug ( " Found possible recursive torrent download. " ) ;
2018-07-21 08:28:13 +03:00
const QString torrentFullpath = torrent - > savePath ( true ) + ' / ' + torrentRelpath ;
2017-08-13 13:56:03 +03:00
qDebug ( " Full subtorrent path is %s " , qUtf8Printable ( torrentFullpath ) ) ;
2015-04-19 18:17:47 +03:00
TorrentInfo torrentInfo = TorrentInfo : : loadFromFile ( torrentFullpath ) ;
if ( torrentInfo . isValid ( ) ) {
qDebug ( " emitting recursiveTorrentDownloadPossible() " ) ;
emit recursiveTorrentDownloadPossible ( torrent ) ;
break ;
}
else {
qDebug ( " Caught error loading torrent " ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Unable to decode '%1' torrent file. " ) . arg ( Utils : : Fs : : toNativePath ( torrentFullpath ) ) , Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
}
}
}
// Move .torrent file to another folder
2016-05-01 11:05:52 +03:00
if ( ! finishedTorrentExportDirectory ( ) . isEmpty ( ) )
2015-04-19 18:17:47 +03:00
exportTorrentFile ( torrent , TorrentExportFolder : : Finished ) ;
if ( ! hasUnfinishedTorrents ( ) )
emit allTorrentsFinished ( ) ;
}
2019-05-09 07:45:52 +03:00
void Session : : handleTorrentResumeDataReady ( TorrentHandle * const torrent , const lt : : entry & data )
2015-04-19 18:17:47 +03:00
{
- - m_numResumeData ;
2015-12-13 15:38:19 +03:00
// Separated thread is used for the blocking IO which results in slow processing of many torrents.
2019-05-09 07:45:52 +03:00
// Encoding data in parallel while doing IO saves time. Copying lt::entry objects around
2015-12-13 15:38:19 +03:00
// isn't cheap too.
QByteArray out ;
2019-10-27 16:37:23 +03:00
out . reserve ( 1024 * 1024 ) ; // most fastresume file sizes are under 1 MB
2019-05-07 06:22:39 +03:00
lt : : bencode ( std : : back_inserter ( out ) , data ) ;
2015-12-13 15:38:19 +03:00
2018-11-19 12:53:29 +03:00
const QString filename = QString ( " %1.fastresume " ) . arg ( torrent - > hash ( ) ) ;
2019-06-19 10:35:29 +03:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( m_resumeDataSavingManager
, [ this , filename , out ] ( ) { m_resumeDataSavingManager - > save ( filename , out ) ; } ) ;
# else
2018-11-19 12:53:29 +03:00
QMetaObject : : invokeMethod ( m_resumeDataSavingManager , " save " ,
Q_ARG ( QString , filename ) , Q_ARG ( QByteArray , out ) ) ;
2019-06-19 10:35:29 +03:00
# endif
2015-04-19 18:17:47 +03:00
}
void Session : : handleTorrentResumeDataFailed ( TorrentHandle * const torrent )
{
Q_UNUSED ( torrent )
- - m_numResumeData ;
}
void Session : : handleTorrentTrackerReply ( TorrentHandle * const torrent , const QString & trackerUrl )
{
emit trackerSuccess ( torrent , trackerUrl ) ;
}
void Session : : handleTorrentTrackerError ( TorrentHandle * const torrent , const QString & trackerUrl )
{
emit trackerError ( torrent , trackerUrl ) ;
}
void Session : : handleTorrentTrackerWarning ( TorrentHandle * const torrent , const QString & trackerUrl )
{
emit trackerWarning ( torrent , trackerUrl ) ;
}
bool Session : : hasPerTorrentRatioLimit ( ) const
{
2019-01-11 11:05:57 +03:00
return std : : any_of ( m_torrents . cbegin ( ) , m_torrents . cend ( ) , [ ] ( const TorrentHandle * torrent )
{
return ( torrent - > ratioLimit ( ) > = 0 ) ;
} ) ;
2015-04-19 18:17:47 +03:00
}
2016-02-07 20:31:50 +03:00
bool Session : : hasPerTorrentSeedingTimeLimit ( ) const
{
2019-01-11 11:05:57 +03:00
return std : : any_of ( m_torrents . cbegin ( ) , m_torrents . cend ( ) , [ ] ( const TorrentHandle * torrent )
{
return ( torrent - > seedingTimeLimit ( ) > = 0 ) ;
} ) ;
2016-02-07 20:31:50 +03:00
}
2015-04-19 18:17:47 +03:00
void Session : : initResumeFolder ( )
{
2016-05-11 14:25:29 +03:00
m_resumeFolderPath = Utils : : Fs : : expandPathAbs ( specialFolderLocation ( SpecialFolder : : Data ) + RESUME_FOLDER ) ;
2019-02-09 18:40:14 +03:00
const QDir resumeFolderDir ( m_resumeFolderPath ) ;
2015-04-19 18:17:47 +03:00
if ( resumeFolderDir . exists ( ) | | resumeFolderDir . mkpath ( resumeFolderDir . absolutePath ( ) ) ) {
2019-10-01 10:33:11 +03:00
m_resumeFolderLock - > setFileName ( resumeFolderDir . absoluteFilePath ( " session.lock " ) ) ;
if ( ! m_resumeFolderLock - > open ( QFile : : WriteOnly ) ) {
2018-11-29 17:25:38 +03:00
throw RuntimeError { tr ( " Cannot write to torrent resume folder. " ) } ;
2015-04-19 18:17:47 +03:00
}
}
else {
2018-11-29 17:25:38 +03:00
throw RuntimeError { tr ( " Cannot create torrent resume folder. " ) } ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
void Session : : configureDeferred ( )
{
2019-10-01 10:05:26 +03:00
if ( m_deferredConfigureScheduled )
return ;
m_deferredConfigureScheduled = true ;
2019-06-19 10:35:29 +03:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
2019-10-01 10:05:26 +03:00
QMetaObject : : invokeMethod ( this
, qOverload < > ( & Session : : configure )
, Qt : : QueuedConnection ) ;
2019-06-19 10:35:29 +03:00
# else
2019-10-01 10:05:26 +03:00
QMetaObject : : invokeMethod ( this , " configure " , Qt : : QueuedConnection ) ;
2019-06-19 10:35:29 +03:00
# endif
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
// Enable IP Filtering
2017-03-06 15:40:34 +03:00
// this method creates ban list from scratch combining user ban list and 3rd party ban list file
2016-06-03 17:03:17 +03:00
void Session : : enableIPFilter ( )
2015-04-19 18:17:47 +03:00
{
2015-09-04 22:56:08 +03:00
qDebug ( " Enabling IPFilter " ) ;
2017-03-07 02:34:55 +03:00
// 1. Parse the IP filter
2019-05-09 07:45:52 +03:00
// 2. In the slot add the manually banned IPs to the provided lt::ip_filter
2017-03-07 02:34:55 +03:00
// 3. Set the ip_filter in one go so there isn't a time window where there isn't an ip_filter
// set between clearing the old one and setting the new one.
2015-04-19 18:17:47 +03:00
if ( ! m_filterParser ) {
2017-03-07 02:34:55 +03:00
m_filterParser = new FilterParserThread ( this ) ;
2018-04-18 16:59:41 +03:00
connect ( m_filterParser . data ( ) , & FilterParserThread : : IPFilterParsed , this , & Session : : handleIPFilterParsed ) ;
connect ( m_filterParser . data ( ) , & FilterParserThread : : IPFilterError , this , & Session : : handleIPFilterError ) ;
2015-04-19 18:17:47 +03:00
}
2016-06-03 17:03:17 +03:00
m_filterParser - > processFilterFile ( IPFilterFile ( ) ) ;
2015-04-19 18:17:47 +03:00
}
// Disable IP Filtering
void Session : : disableIPFilter ( )
{
qDebug ( " Disabling IPFilter " ) ;
if ( m_filterParser ) {
2019-02-09 18:40:14 +03:00
disconnect ( m_filterParser . data ( ) , nullptr , this , nullptr ) ;
2015-04-19 18:17:47 +03:00
delete m_filterParser ;
}
2016-06-03 17:03:17 +03:00
// Add the banned IPs after the IPFilter disabling
// which creates an empty filter and overrides all previously
// applied bans.
2019-05-07 06:22:39 +03:00
lt : : ip_filter filter ;
2017-03-07 02:34:55 +03:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
2015-04-19 18:17:47 +03:00
}
void Session : : recursiveTorrentDownload ( const InfoHash & hash )
{
TorrentHandle * const torrent = m_torrents . value ( hash ) ;
if ( ! torrent ) return ;
for ( int i = 0 ; i < torrent - > filesCount ( ) ; + + i ) {
const QString torrentRelpath = torrent - > filePath ( i ) ;
if ( torrentRelpath . endsWith ( " .torrent " ) ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Recursive download of file '%1' embedded in torrent '%2' "
, " Recursive download of 'test.torrent' embedded in torrent 'test2' " )
. arg ( Utils : : Fs : : toNativePath ( torrentRelpath ) , torrent - > name ( ) ) ) ;
2018-07-21 08:28:13 +03:00
const QString torrentFullpath = torrent - > savePath ( ) + ' / ' + torrentRelpath ;
2015-04-19 18:17:47 +03:00
AddTorrentParams params ;
// Passing the save path along to the sub torrent file
params . savePath = torrent - > savePath ( ) ;
addTorrent ( TorrentInfo : : loadFromFile ( torrentFullpath ) , params ) ;
}
}
}
2017-04-29 14:45:30 +03:00
const SessionStatus & Session : : status ( ) const
2015-04-19 18:17:47 +03:00
{
2017-04-29 14:45:30 +03:00
return m_status ;
2015-04-19 18:17:47 +03:00
}
2017-04-29 14:45:30 +03:00
const CacheStatus & Session : : cacheStatus ( ) const
2015-04-19 18:17:47 +03:00
{
2017-04-29 14:45:30 +03:00
return m_cacheStatus ;
2015-04-19 18:17:47 +03:00
}
// Will resume torrents in backup directory
void Session : : startUpTorrents ( )
{
qDebug ( " Resuming torrents... " ) ;
const QDir resumeDataDir ( m_resumeFolderPath ) ;
QStringList fastresumes = resumeDataDir . entryList (
2015-12-16 17:08:27 +03:00
QStringList ( QLatin1String ( " *.fastresume " ) ) , QDir : : Files , QDir : : Unsorted ) ;
2015-04-19 18:17:47 +03:00
2019-08-26 09:25:16 +03:00
struct TorrentResumeData
2015-12-16 17:08:27 +03:00
{
QString hash ;
MagnetUri magnetUri ;
2018-07-11 15:44:15 +03:00
CreateTorrentParams addTorrentData ;
2015-12-16 17:08:27 +03:00
QByteArray data ;
2019-08-26 09:25:16 +03:00
} ;
2015-12-16 17:08:27 +03:00
2017-01-23 18:45:37 +03:00
int resumedTorrentsCount = 0 ;
2019-07-02 06:49:35 +03:00
const auto startupTorrent = [ this , & resumeDataDir , & resumedTorrentsCount ] ( const TorrentResumeData & params )
2015-12-17 16:47:34 +03:00
{
2019-08-21 10:37:39 +03:00
// TODO: Remove loading of .torrent files when starting up existing torrents
// Starting from v4.2.0, the required `info` dict will be stored in fastresume too
// (besides .torrent file), that means we can remove loading of .torrent files in
// a later release, such as v4.3.0.
2019-02-09 18:40:14 +03:00
const QString filePath = resumeDataDir . filePath ( QString ( " %1.torrent " ) . arg ( params . hash ) ) ;
2015-12-17 16:47:34 +03:00
qDebug ( ) < < " Starting up torrent " < < params . hash < < " ... " ;
if ( ! addTorrent_impl ( params . addTorrentData , params . magnetUri , TorrentInfo : : loadFromFile ( filePath ) , params . data ) )
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Unable to resume torrent '%1'. " , " e.g: Unable to resume torrent 'hash'. " )
. arg ( params . hash ) , Log : : CRITICAL ) ;
2017-01-23 18:45:37 +03:00
// process add torrent messages before message queue overflow
2018-04-14 22:53:45 +03:00
if ( ( resumedTorrentsCount % 100 ) = = 0 ) readAlerts ( ) ;
2017-01-23 18:45:37 +03:00
+ + resumedTorrentsCount ;
2015-12-17 16:47:34 +03:00
} ;
2018-11-19 12:53:29 +03:00
qDebug ( " Starting up torrents... " ) ;
2015-12-16 17:08:27 +03:00
qDebug ( " Queue size: %d " , fastresumes . size ( ) ) ;
2018-11-19 12:53:29 +03:00
2018-05-24 18:41:03 +03:00
const QRegularExpression rx ( QLatin1String ( " ^([A-Fa-f0-9]{40}) \ \ . fastresume $ " )) ;
2015-04-19 18:17:47 +03:00
2018-11-19 12:53:29 +03:00
if ( isQueueingSystemEnabled ( ) ) {
QFile queueFile { resumeDataDir . absoluteFilePath ( QLatin1String { " queue " } ) } ;
// TODO: The following code is deprecated in 4.1.5. Remove after several releases in 4.2.x.
// === BEGIN DEPRECATED CODE === //
if ( ! queueFile . exists ( ) ) {
// Resume downloads in a legacy manner
QMap < int , TorrentResumeData > queuedResumeData ;
int nextQueuePosition = 1 ;
int numOfRemappedFiles = 0 ;
2018-11-27 23:15:04 +03:00
for ( const QString & fastresumeName : asConst ( fastresumes ) ) {
2018-11-19 12:53:29 +03:00
const QRegularExpressionMatch rxMatch = rx . match ( fastresumeName ) ;
if ( ! rxMatch . hasMatch ( ) ) continue ;
QString hash = rxMatch . captured ( 1 ) ;
QString fastresumePath = resumeDataDir . absoluteFilePath ( fastresumeName ) ;
QByteArray data ;
CreateTorrentParams torrentParams ;
MagnetUri magnetUri ;
int queuePosition ;
if ( readFile ( fastresumePath , data ) & & loadTorrentResumeData ( data , torrentParams , queuePosition , magnetUri ) ) {
if ( queuePosition < = nextQueuePosition ) {
startupTorrent ( { hash , magnetUri , torrentParams , data } ) ;
if ( queuePosition = = nextQueuePosition ) {
+ + nextQueuePosition ;
while ( queuedResumeData . contains ( nextQueuePosition ) ) {
startupTorrent ( queuedResumeData . take ( nextQueuePosition ) ) ;
+ + nextQueuePosition ;
}
}
}
else {
int q = queuePosition ;
for ( ; queuedResumeData . contains ( q ) ; + + q ) { }
if ( q ! = queuePosition )
+ + numOfRemappedFiles ;
queuedResumeData [ q ] = { hash , magnetUri , torrentParams , data } ;
2015-12-17 16:47:34 +03:00
}
}
2015-12-16 17:08:27 +03:00
}
2018-11-19 12:53:29 +03:00
if ( numOfRemappedFiles > 0 ) {
2019-07-02 06:49:35 +03:00
LogMsg ( QString ( tr ( " Queue positions were corrected in %1 resume files " ) )
. arg ( numOfRemappedFiles ) , Log : : CRITICAL ) ;
2015-12-16 17:08:27 +03:00
}
2018-11-19 12:53:29 +03:00
// starting up downloading torrents (queue position > 0)
2018-11-27 23:15:04 +03:00
for ( const TorrentResumeData & torrentResumeData : asConst ( queuedResumeData ) )
2018-11-19 12:53:29 +03:00
startupTorrent ( torrentResumeData ) ;
return ;
2015-04-19 18:17:47 +03:00
}
2018-11-19 12:53:29 +03:00
// === END DEPRECATED CODE === //
2015-12-16 17:08:27 +03:00
2018-11-19 12:53:29 +03:00
QStringList queue ;
if ( queueFile . open ( QFile : : ReadOnly ) ) {
QByteArray line ;
while ( ! ( line = queueFile . readLine ( ) ) . isEmpty ( ) )
queue . append ( QString : : fromLatin1 ( line . trimmed ( ) ) + QLatin1String { " .fastresume " } ) ;
}
else {
LogMsg ( tr ( " Couldn't load torrents queue from '%1'. Error: %2 " )
2019-07-02 06:49:35 +03:00
. arg ( queueFile . fileName ( ) , queueFile . errorString ( ) ) , Log : : WARNING ) ;
2018-11-19 12:53:29 +03:00
}
if ( ! queue . empty ( ) )
2019-10-31 07:38:11 +03:00
fastresumes = queue + fastresumes . toSet ( ) . subtract ( queue . toSet ( ) ) . values ( ) ;
2017-01-18 19:17:51 +03:00
}
2018-11-27 23:15:04 +03:00
for ( const QString & fastresumeName : asConst ( fastresumes ) ) {
2018-11-19 12:53:29 +03:00
const QRegularExpressionMatch rxMatch = rx . match ( fastresumeName ) ;
if ( ! rxMatch . hasMatch ( ) ) continue ;
const QString hash = rxMatch . captured ( 1 ) ;
const QString fastresumePath = resumeDataDir . absoluteFilePath ( fastresumeName ) ;
QByteArray data ;
CreateTorrentParams torrentParams ;
MagnetUri magnetUri ;
int queuePosition ;
if ( readFile ( fastresumePath , data )
& & loadTorrentResumeData ( data , torrentParams , queuePosition , magnetUri ) ) {
startupTorrent ( { hash , magnetUri , torrentParams , data } ) ;
}
}
2015-04-19 18:17:47 +03:00
}
quint64 Session : : getAlltimeDL ( ) const
{
return m_statistics - > getAlltimeDL ( ) ;
}
quint64 Session : : getAlltimeUL ( ) const
{
return m_statistics - > getAlltimeUL ( ) ;
}
void Session : : refresh ( )
{
m_nativeSession - > post_torrent_updates ( ) ;
2017-04-29 14:52:28 +03:00
m_nativeSession - > post_session_stats ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 18:40:14 +03:00
void Session : : handleIPFilterParsed ( const int ruleCount )
2015-04-19 18:17:47 +03:00
{
2017-04-17 17:07:12 +03:00
if ( m_filterParser ) {
2019-05-07 06:22:39 +03:00
lt : : ip_filter filter = m_filterParser - > IPfilter ( ) ;
2017-03-07 02:34:55 +03:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
}
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Successfully parsed the provided IP filter: %1 rules were applied. " , " %1 is a number " ) . arg ( ruleCount ) ) ;
2016-05-01 11:05:52 +03:00
emit IPFilterParsed ( false , ruleCount ) ;
2015-04-19 18:17:47 +03:00
}
void Session : : handleIPFilterError ( )
{
2019-05-07 06:22:39 +03:00
lt : : ip_filter filter ;
2017-03-07 02:34:55 +03:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Error: Failed to parse the provided IP filter. " ) , Log : : CRITICAL ) ;
2016-05-01 11:05:52 +03:00
emit IPFilterParsed ( true , 0 ) ;
2015-04-19 18:17:47 +03:00
}
2019-10-27 08:46:52 +03:00
std : : vector < lt : : alert * > Session : : getPendingAlerts ( const lt : : time_duration time ) const
2015-04-19 18:17:47 +03:00
{
2019-10-27 08:46:52 +03:00
if ( time > lt : : time_duration : : zero ( ) )
m_nativeSession - > wait_for_alert ( time ) ;
2015-04-19 18:17:47 +03:00
2019-10-27 08:46:52 +03:00
std : : vector < lt : : alert * > alerts ;
m_nativeSession - > pop_alerts ( & alerts ) ;
return alerts ;
2015-04-19 18:17:47 +03:00
}
2017-04-15 17:21:24 +03:00
bool Session : : isCreateTorrentSubfolder ( ) const
{
return m_isCreateTorrentSubfolder ;
}
2019-02-09 18:40:14 +03:00
void Session : : setCreateTorrentSubfolder ( const bool value )
2017-04-15 17:21:24 +03:00
{
m_isCreateTorrentSubfolder = value ;
}
2015-04-19 18:17:47 +03:00
// Read alerts sent by the BitTorrent session
void Session : : readAlerts ( )
{
2019-10-27 08:46:52 +03:00
const std : : vector < lt : : alert * > alerts = getPendingAlerts ( ) ;
for ( const lt : : alert * a : alerts )
2015-04-19 18:17:47 +03:00
handleAlert ( a ) ;
}
2019-05-07 06:22:39 +03:00
void Session : : handleAlert ( const lt : : alert * a )
2015-04-19 18:17:47 +03:00
{
try {
switch ( a - > type ( ) ) {
2019-05-07 06:22:39 +03:00
case lt : : file_renamed_alert : : alert_type :
case lt : : file_completed_alert : : alert_type :
case lt : : torrent_finished_alert : : alert_type :
case lt : : save_resume_data_alert : : alert_type :
case lt : : save_resume_data_failed_alert : : alert_type :
case lt : : storage_moved_alert : : alert_type :
case lt : : storage_moved_failed_alert : : alert_type :
case lt : : torrent_paused_alert : : alert_type :
case lt : : torrent_resumed_alert : : alert_type :
case lt : : tracker_error_alert : : alert_type :
case lt : : tracker_reply_alert : : alert_type :
case lt : : tracker_warning_alert : : alert_type :
case lt : : fastresume_rejected_alert : : alert_type :
case lt : : torrent_checked_alert : : alert_type :
case lt : : metadata_received_alert : : alert_type :
2015-04-19 18:17:47 +03:00
dispatchTorrentAlert ( a ) ;
break ;
2019-05-07 06:22:39 +03:00
case lt : : state_update_alert : : alert_type :
handleStateUpdateAlert ( static_cast < const lt : : state_update_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : session_stats_alert : : alert_type :
handleSessionStatsAlert ( static_cast < const lt : : session_stats_alert * > ( a ) ) ;
2017-04-29 14:52:28 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : file_error_alert : : alert_type :
handleFileErrorAlert ( static_cast < const lt : : file_error_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : add_torrent_alert : : alert_type :
handleAddTorrentAlert ( static_cast < const lt : : add_torrent_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : torrent_removed_alert : : alert_type :
handleTorrentRemovedAlert ( static_cast < const lt : : torrent_removed_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : torrent_deleted_alert : : alert_type :
handleTorrentDeletedAlert ( static_cast < const lt : : torrent_deleted_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : torrent_delete_failed_alert : : alert_type :
handleTorrentDeleteFailedAlert ( static_cast < const lt : : torrent_delete_failed_alert * > ( a ) ) ;
2015-08-09 10:27:56 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : portmap_error_alert : : alert_type :
handlePortmapWarningAlert ( static_cast < const lt : : portmap_error_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : portmap_alert : : alert_type :
handlePortmapAlert ( static_cast < const lt : : portmap_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : alert_type :
handlePeerBlockedAlert ( static_cast < const lt : : peer_blocked_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_ban_alert : : alert_type :
handlePeerBanAlert ( static_cast < const lt : : peer_ban_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : url_seed_alert : : alert_type :
handleUrlSeedAlert ( static_cast < const lt : : url_seed_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : listen_succeeded_alert : : alert_type :
handleListenSucceededAlert ( static_cast < const lt : : listen_succeeded_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : listen_failed_alert : : alert_type :
handleListenFailedAlert ( static_cast < const lt : : listen_failed_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : external_ip_alert : : alert_type :
handleExternalIPAlert ( static_cast < const lt : : external_ip_alert * > ( a ) ) ;
2015-04-19 18:17:47 +03:00
break ;
2019-09-13 07:31:26 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200)
case lt : : alerts_dropped_alert : : alert_type :
handleAlertsDroppedAlert ( static_cast < const lt : : alerts_dropped_alert * > ( a ) ) ;
break ;
# endif
2015-04-19 18:17:47 +03:00
}
}
2019-05-16 10:02:17 +03:00
catch ( const std : : exception & exc ) {
2017-03-07 14:41:38 +03:00
qWarning ( ) < < " Caught exception in " < < Q_FUNC_INFO < < " : " < < QString : : fromStdString ( exc . what ( ) ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-05-07 06:22:39 +03:00
void Session : : dispatchTorrentAlert ( const lt : : alert * a )
2015-04-19 18:17:47 +03:00
{
2019-05-07 06:22:39 +03:00
TorrentHandle * const torrent = m_torrents . value ( static_cast < const lt : : torrent_alert * > ( a ) - > handle . info_hash ( ) ) ;
2019-06-28 21:24:39 +03:00
if ( torrent ) {
2015-04-19 18:17:47 +03:00
torrent - > handleAlert ( a ) ;
2019-06-28 21:24:39 +03:00
return ;
}
switch ( a - > type ( ) ) {
case lt : : torrent_paused_alert : : alert_type :
handleTorrentPausedAlert ( static_cast < const lt : : torrent_paused_alert * > ( a ) ) ;
break ;
case lt : : metadata_received_alert : : alert_type :
handleMetadataReceivedAlert ( static_cast < const lt : : metadata_received_alert * > ( a ) ) ;
break ;
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : createTorrentHandle ( const lt : : torrent_handle & nativeHandle )
2015-04-19 18:17:47 +03:00
{
// Magnet added for preload its metadata
2016-01-20 09:57:02 +03:00
if ( ! m_addingTorrents . contains ( nativeHandle . info_hash ( ) ) ) return ;
2015-04-19 18:17:47 +03:00
2019-02-09 18:40:14 +03:00
const CreateTorrentParams params = m_addingTorrents . take ( nativeHandle . info_hash ( ) ) ;
2015-04-19 18:17:47 +03:00
2018-07-11 15:44:15 +03:00
TorrentHandle * const torrent = new TorrentHandle ( this , nativeHandle , params ) ;
2015-04-19 18:17:47 +03:00
m_torrents . insert ( torrent - > hash ( ) , torrent ) ;
2019-02-09 18:40:14 +03:00
const bool fromMagnetUri = ! torrent - > hasMetadata ( ) ;
2015-04-19 18:17:47 +03:00
2018-07-11 15:44:15 +03:00
if ( params . restored ) {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' restored. " , " 'torrent name' restored. " ) . arg ( torrent - > name ( ) ) ) ;
2015-04-19 18:17:47 +03:00
}
else {
// The following is useless for newly added magnet
if ( ! fromMagnetUri ) {
// Backup torrent file
const QDir resumeDataDir ( m_resumeFolderPath ) ;
const QString newFile = resumeDataDir . absoluteFilePath ( QString ( " %1.torrent " ) . arg ( torrent - > hash ( ) ) ) ;
if ( torrent - > saveTorrentFile ( newFile ) ) {
// Copy the torrent file to the export folder
2016-05-01 11:05:52 +03:00
if ( ! torrentExportDirectory ( ) . isEmpty ( ) )
2015-04-19 18:17:47 +03:00
exportTorrentFile ( torrent ) ;
}
else {
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Couldn't save '%1.torrent' " ) . arg ( torrent - > hash ( ) ) , Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-05-01 11:05:52 +03:00
if ( isAddTrackersEnabled ( ) & & ! torrent - > isPrivate ( ) )
torrent - > addTrackers ( m_additionalTrackerList ) ;
2015-06-30 21:03:17 +03:00
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " '%1' added to download list. " , " 'torrent name' was added to download list. " )
. arg ( torrent - > name ( ) ) ) ;
2015-11-29 19:40:24 +03:00
// In case of crash before the scheduled generation
// of the fastresumes.
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-02-07 20:31:50 +03:00
if ( ( ( torrent - > ratioLimit ( ) > = 0 ) | | ( torrent - > seedingTimeLimit ( ) > = 0 ) )
& & ! m_seedingLimitTimer - > isActive ( ) )
m_seedingLimitTimer - > start ( ) ;
2015-04-19 18:17:47 +03:00
// Send torrent addition signal
emit torrentAdded ( torrent ) ;
2016-04-30 01:38:24 +03:00
// Send new torrent signal
2018-07-11 15:44:15 +03:00
if ( ! params . restored )
2016-04-30 01:38:24 +03:00
emit torrentNew ( torrent ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleAddTorrentAlert ( const lt : : add_torrent_alert * p )
2016-01-20 09:57:02 +03:00
{
if ( p - > error ) {
qDebug ( " /! \\ Error: Failed to add torrent! " ) ;
2017-03-07 14:41:38 +03:00
QString msg = QString : : fromStdString ( p - > message ( ) ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " Couldn't add torrent. Reason: %1 " ) . arg ( msg ) , Log : : WARNING ) ;
2016-01-20 09:57:02 +03:00
emit addTorrentFailed ( msg ) ;
}
else {
createTorrentHandle ( p - > handle ) ;
}
}
2019-05-07 06:22:39 +03:00
void Session : : handleTorrentRemovedAlert ( const lt : : torrent_removed_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 10:51:36 +03:00
const InfoHash infoHash { p - > info_hash } ;
2017-09-14 19:55:14 +03:00
2019-09-29 05:12:56 +03:00
const auto loadedMetadataIter = m_loadedMetadata . find ( infoHash ) ;
if ( loadedMetadataIter ! = m_loadedMetadata . end ( ) ) {
emit metadataLoaded ( * loadedMetadataIter ) ;
m_loadedMetadata . erase ( loadedMetadataIter ) ;
}
2018-12-16 10:51:36 +03:00
2019-09-29 05:12:56 +03:00
const auto removingTorrentDataIter = m_removingTorrents . find ( infoHash ) ;
if ( removingTorrentDataIter ! = m_removingTorrents . end ( ) ) {
if ( removingTorrentDataIter - > deleteOption = = Torrent ) {
LogMsg ( tr ( " '%1' was removed from the transfer list. " , " 'xxx.avi' was removed... " ) . arg ( removingTorrentDataIter - > name ) ) ;
m_removingTorrents . erase ( removingTorrentDataIter ) ;
2017-09-14 19:55:14 +03:00
}
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleTorrentDeletedAlert ( const lt : : torrent_deleted_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 10:51:36 +03:00
const InfoHash infoHash { p - > info_hash } ;
2019-09-29 05:12:56 +03:00
const auto removingTorrentDataIter = m_removingTorrents . find ( infoHash ) ;
2018-12-16 10:51:36 +03:00
2019-09-29 05:12:56 +03:00
if ( removingTorrentDataIter = = m_removingTorrents . end ( ) )
2017-09-14 19:55:14 +03:00
return ;
2019-09-29 05:12:56 +03:00
Utils : : Fs : : smartRemoveEmptyFolderTree ( removingTorrentDataIter - > savePathToRemove ) ;
LogMsg ( tr ( " '%1' was removed from the transfer list and hard disk. " , " 'xxx.avi' was removed... " ) . arg ( removingTorrentDataIter - > name ) ) ;
m_removingTorrents . erase ( removingTorrentDataIter ) ;
2015-08-09 10:27:56 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleTorrentDeleteFailedAlert ( const lt : : torrent_delete_failed_alert * p )
2015-08-09 10:27:56 +03:00
{
2018-12-16 10:51:36 +03:00
const InfoHash infoHash { p - > info_hash } ;
2019-09-29 05:12:56 +03:00
const auto removingTorrentDataIter = m_removingTorrents . find ( infoHash ) ;
2018-12-16 10:51:36 +03:00
2019-09-29 05:12:56 +03:00
if ( removingTorrentDataIter = = m_removingTorrents . end ( ) )
2017-09-14 19:55:14 +03:00
return ;
2019-09-29 05:12:56 +03:00
2015-08-09 10:27:56 +03:00
// libtorrent won't delete the directory if it contains files not listed in the torrent,
// so we remove the directory ourselves
2019-09-29 05:12:56 +03:00
Utils : : Fs : : smartRemoveEmptyFolderTree ( removingTorrentDataIter - > savePathToRemove ) ;
2017-09-14 19:55:14 +03:00
2019-06-30 15:39:49 +03:00
if ( p - > error ) {
LogMsg ( tr ( " '%1' was removed from the transfer list but the files couldn't be deleted. Error: %2 " , " 'xxx.avi' was removed... " )
2019-09-29 05:12:56 +03:00
. arg ( removingTorrentDataIter - > name , QString : : fromLocal8Bit ( p - > error . message ( ) . c_str ( ) ) )
2019-06-30 15:39:49 +03:00
, Log : : WARNING ) ;
}
else {
2019-09-29 05:12:56 +03:00
LogMsg ( tr ( " '%1' was removed from the transfer list. " , " 'xxx.avi' was removed... " ) . arg ( removingTorrentDataIter - > name ) ) ;
2019-06-30 15:39:49 +03:00
}
2019-09-29 05:12:56 +03:00
m_removingTorrents . erase ( removingTorrentDataIter ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleMetadataReceivedAlert ( const lt : : metadata_received_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 10:51:36 +03:00
const InfoHash hash { p - > handle . info_hash ( ) } ;
2019-09-29 05:12:56 +03:00
const auto loadedMetadataIter = m_loadedMetadata . find ( hash ) ;
2015-04-19 18:17:47 +03:00
2019-09-29 05:12:56 +03:00
if ( loadedMetadataIter ! = m_loadedMetadata . end ( ) ) {
2015-04-19 18:17:47 +03:00
- - m_extraLimit ;
adjustLimits ( ) ;
2019-09-29 05:12:56 +03:00
* loadedMetadataIter = TorrentInfo ( p - > handle . torrent_file ( ) ) ;
2019-05-07 06:22:39 +03:00
m_nativeSession - > remove_torrent ( p - > handle , lt : : session : : delete_files ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-06-28 21:24:39 +03:00
void Session : : handleTorrentPausedAlert ( const libtorrent : : torrent_paused_alert * p )
{
const InfoHash hash { p - > handle . info_hash ( ) } ;
2019-07-10 19:54:10 +03:00
if ( m_addingTorrents . contains ( hash ) ) {
2019-06-28 21:24:39 +03:00
// Adding preloaded torrent
lt : : torrent_handle handle = p - > handle ;
- - m_extraLimit ;
// Preloaded torrent is in "Upload mode" so we need to disable it
// otherwise the torrent never be downloaded (until application restart)
# if (LIBTORRENT_VERSION_NUM < 10200)
handle . set_upload_mode ( false ) ;
# else
handle . unset_flags ( lt : : torrent_flags : : upload_mode ) ;
# endif
adjustLimits ( ) ;
// use common 2nd step of torrent addition
createTorrentHandle ( handle ) ;
}
}
2019-05-07 06:22:39 +03:00
void Session : : handleFileErrorAlert ( const lt : : file_error_alert * p )
2015-04-19 18:17:47 +03:00
{
TorrentHandle * const torrent = m_torrents . value ( p - > handle . info_hash ( ) ) ;
2019-06-15 17:21:26 +03:00
if ( ! torrent )
return ;
2018-05-21 01:09:58 +03:00
2019-06-15 17:21:26 +03:00
const InfoHash hash = torrent - > hash ( ) ;
if ( ! m_recentErroredTorrents . contains ( hash ) ) {
m_recentErroredTorrents . insert ( hash ) ;
const QString msg = QString : : fromStdString ( p - > message ( ) ) ;
LogMsg ( tr ( " File error alert. Torrent: \" %1 \" . File: \" %2 \" . Reason: %3 " )
. arg ( torrent - > name ( ) , p - > filename ( ) , msg )
, Log : : WARNING ) ;
emit fullDiskError ( torrent , msg ) ;
2015-04-19 18:17:47 +03:00
}
2019-06-15 17:21:26 +03:00
m_recentErroredTorrentsTimer - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handlePortmapWarningAlert ( const lt : : portmap_error_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " UPnP/NAT-PMP: Port mapping failure, message: %1 " ) . arg ( QString : : fromStdString ( p - > message ( ) ) ) , Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handlePortmapAlert ( const lt : : portmap_alert * p )
2015-04-19 18:17:47 +03:00
{
qDebug ( " UPnP Success, msg: %s " , p - > message ( ) . c_str ( ) ) ;
2019-07-02 06:49:35 +03:00
LogMsg ( tr ( " UPnP/NAT-PMP: Port mapping successful, message: %1 " ) . arg ( QString : : fromStdString ( p - > message ( ) ) ) , Log : : INFO ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handlePeerBlockedAlert ( const lt : : peer_blocked_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-03-06 08:58:07 +03:00
# if LIBTORRENT_VERSION_NUM < 10200
2019-02-09 18:40:14 +03:00
const std : : string ip = p - > ip . to_string ( ec ) ;
2019-03-06 08:58:07 +03:00
# else
const std : : string ip = p - > endpoint . address ( ) . to_string ( ec ) ;
# endif
2015-04-19 18:17:47 +03:00
QString reason ;
switch ( p - > reason ) {
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : ip_filter :
2015-04-19 18:17:47 +03:00
reason = tr ( " due to IP filter. " , " this peer was blocked due to ip filter. " ) ;
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : port_filter :
2015-04-19 18:17:47 +03:00
reason = tr ( " due to port filter. " , " this peer was blocked due to port filter. " ) ;
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : i2p_mixed :
2015-04-19 18:17:47 +03:00
reason = tr ( " due to i2p mixed mode restrictions. " , " this peer was blocked due to i2p mixed mode restrictions. " ) ;
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : privileged_ports :
2015-04-19 18:17:47 +03:00
reason = tr ( " because it has a low port. " , " this peer was blocked because it has a low port. " ) ;
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : utp_disabled :
2019-02-27 10:22:12 +03:00
reason = tr ( " because %1 is disabled. " , " this peer was blocked because uTP is disabled. " ) . arg ( QString : : fromUtf8 ( C_UTP ) ) ; // don't translate μTP
2015-04-19 18:17:47 +03:00
break ;
2019-05-07 06:22:39 +03:00
case lt : : peer_blocked_alert : : tcp_disabled :
2015-09-04 22:56:08 +03:00
reason = tr ( " because %1 is disabled. " , " this peer was blocked because TCP is disabled. " ) . arg ( " TCP " ) ; // don't translate TCP
2015-04-19 18:17:47 +03:00
break ;
}
if ( ! ec )
Logger : : instance ( ) - > addPeer ( QString : : fromLatin1 ( ip . c_str ( ) ) , true , reason ) ;
}
2019-05-07 06:22:39 +03:00
void Session : : handlePeerBanAlert ( const lt : : peer_ban_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-02-09 18:40:14 +03:00
const std : : string ip = p - > ip . address ( ) . to_string ( ec ) ;
2019-04-03 09:23:35 +03:00
# else
const std : : string ip = p - > endpoint . address ( ) . to_string ( ec ) ;
# endif
2015-04-19 18:17:47 +03:00
if ( ! ec )
Logger : : instance ( ) - > addPeer ( QString : : fromLatin1 ( ip . c_str ( ) ) , false ) ;
}
2019-05-07 06:22:39 +03:00
void Session : : handleUrlSeedAlert ( const lt : : url_seed_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-09-18 09:56:56 +03:00
const TorrentHandle * torrent = m_torrents . value ( p - > handle . info_hash ( ) ) ;
if ( ! torrent )
return ;
if ( p - > error ) {
LogMsg ( tr ( " URL seed name lookup failed. Torrent: \" %1 \" . URL: \" %2 \" . Error: \" %3 \" " )
. arg ( torrent - > name ( ) , p - > server_url ( ) , QString : : fromStdString ( p - > message ( ) ) )
, Log : : WARNING ) ;
}
else {
LogMsg ( tr ( " Received error message from a URL seed. Torrent: \" %1 \" . URL: \" %2 \" . Message: \" %3 \" " )
. arg ( torrent - > name ( ) , p - > server_url ( ) , p - > error_message ( ) )
, Log : : WARNING ) ;
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleListenSucceededAlert ( const lt : : listen_succeeded_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-04-03 09:23:35 +03:00
QString proto = " INVALID " ;
# if (LIBTORRENT_VERSION_NUM < 10200)
switch ( p - > sock_type )
{
case lt : : listen_succeeded_alert : : udp :
2015-04-19 18:17:47 +03:00
proto = " UDP " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_succeeded_alert : : tcp :
2015-04-19 18:17:47 +03:00
proto = " TCP " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_succeeded_alert : : tcp_ssl :
2015-04-19 18:17:47 +03:00
proto = " TCP_SSL " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_succeeded_alert : : i2p :
proto = " I2P " ;
break ;
case lt : : listen_succeeded_alert : : socks5 :
proto = " SOCKS5 " ;
break ;
case lt : : listen_succeeded_alert : : utp_ssl :
proto = " UTP_SSL " ;
break ;
}
# else
switch ( p - > socket_type )
{
case lt : : socket_type_t : : udp :
proto = " UDP " ;
break ;
case lt : : socket_type_t : : tcp :
proto = " TCP " ;
break ;
case lt : : socket_type_t : : tcp_ssl :
proto = " TCP_SSL " ;
break ;
case lt : : socket_type_t : : i2p :
proto = " I2P " ;
break ;
case lt : : socket_type_t : : socks5 :
proto = " SOCKS5 " ;
break ;
case lt : : socket_type_t : : utp_ssl :
proto = " UTP_SSL " ;
break ;
}
# endif
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-10-16 18:38:51 +03:00
LogMsg ( tr ( " Successfully listening on IP: %1, port: %2/%3 "
, " e.g: Successfully listening on IP: 192.168.0.1, port: TCP/6881 " )
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2018-03-06 18:49:12 +03:00
. arg ( p - > endpoint . address ( ) . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > endpoint . port ( ) ) ) , Log : : INFO ) ;
2019-04-03 09:23:35 +03:00
# else
. arg ( p - > address . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > port ) ) , Log : : INFO ) ;
# endif
2015-04-19 18:17:47 +03:00
// Force reannounce on all torrents because some trackers blacklist some ports
2019-04-03 09:23:35 +03:00
for ( const lt : : torrent_handle & torrent : m_nativeSession - > get_torrents ( ) )
torrent . force_reannounce ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleListenFailedAlert ( const lt : : listen_failed_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-04-03 09:23:35 +03:00
QString proto = " INVALID " ;
# if (LIBTORRENT_VERSION_NUM < 10200)
switch ( p - > sock_type )
{
case lt : : listen_failed_alert : : udp :
2015-04-19 18:17:47 +03:00
proto = " UDP " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_failed_alert : : tcp :
2015-04-19 18:17:47 +03:00
proto = " TCP " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_failed_alert : : tcp_ssl :
2015-04-19 18:17:47 +03:00
proto = " TCP_SSL " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : listen_failed_alert : : i2p :
proto = " I2P " ;
break ;
case lt : : listen_failed_alert : : socks5 :
proto = " SOCKS5 " ;
break ;
case lt : : listen_failed_alert : : utp_ssl :
proto = " UTP_SSL " ;
break ;
}
# else
switch ( p - > socket_type )
{
case lt : : socket_type_t : : udp :
proto = " UDP " ;
break ;
case lt : : socket_type_t : : tcp :
proto = " TCP " ;
break ;
case lt : : socket_type_t : : tcp_ssl :
proto = " TCP_SSL " ;
break ;
case lt : : socket_type_t : : i2p :
2015-04-19 18:17:47 +03:00
proto = " I2P " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : socket_type_t : : socks5 :
2015-04-19 18:17:47 +03:00
proto = " SOCKS5 " ;
2019-04-03 09:23:35 +03:00
break ;
case lt : : socket_type_t : : utp_ssl :
proto = " UTP_SSL " ;
break ;
}
# endif
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-10-16 18:38:51 +03:00
LogMsg ( tr ( " Failed to listen on IP: %1, port: %2/%3. Reason: %4 "
, " e.g: Failed to listen on IP: 192.168.0.1, port: TCP/6881. Reason: already in use " )
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2018-03-06 18:49:12 +03:00
. arg ( p - > endpoint . address ( ) . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > endpoint . port ( ) )
2019-04-03 09:23:35 +03:00
, QString : : fromLocal8Bit ( p - > error . message ( ) . c_str ( ) ) ) , Log : : CRITICAL ) ;
# else
. arg ( p - > address . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > port )
, QString : : fromLocal8Bit ( p - > error . message ( ) . c_str ( ) ) ) , Log : : CRITICAL ) ;
# endif
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleExternalIPAlert ( const lt : : external_ip_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-10-16 19:29:26 +03:00
lt : : error_code ec ;
2019-10-16 18:38:51 +03:00
LogMsg ( tr ( " Detected external IP: %1 " , " e.g. Detected external IP: 1.1.1.1 " )
. arg ( p - > external_address . to_string ( ec ) . c_str ( ) ) , Log : : INFO ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 06:22:39 +03:00
void Session : : handleSessionStatsAlert ( const lt : : session_stats_alert * p )
2017-04-29 14:52:28 +03:00
{
2019-06-01 06:28:37 +03:00
const qreal interval = lt : : total_milliseconds ( p - > timestamp ( ) - m_statsLastTimestamp ) / 1000. ;
m_statsLastTimestamp = p - > timestamp ( ) ;
2019-04-03 09:23:35 +03:00
# if (LIBTORRENT_VERSION_NUM < 10200)
const auto & stats = p - > values ;
# else
const auto stats = p - > counters ( ) ;
# endif
2017-04-29 14:52:28 +03:00
2019-04-03 09:23:35 +03:00
m_status . hasIncomingConnections = static_cast < bool > ( stats [ m_metricIndices . net . hasIncomingConnections ] ) ;
2017-04-29 14:52:28 +03:00
2019-04-03 09:23:35 +03:00
const auto ipOverheadDownload = stats [ m_metricIndices . net . recvIPOverheadBytes ] ;
const auto ipOverheadUpload = stats [ m_metricIndices . net . sentIPOverheadBytes ] ;
const auto totalDownload = stats [ m_metricIndices . net . recvBytes ] + ipOverheadDownload ;
const auto totalUpload = stats [ m_metricIndices . net . sentBytes ] + ipOverheadUpload ;
const auto totalPayloadDownload = stats [ m_metricIndices . net . recvPayloadBytes ] ;
const auto totalPayloadUpload = stats [ m_metricIndices . net . sentPayloadBytes ] ;
const auto trackerDownload = stats [ m_metricIndices . net . recvTrackerBytes ] ;
const auto trackerUpload = stats [ m_metricIndices . net . sentTrackerBytes ] ;
const auto dhtDownload = stats [ m_metricIndices . dht . dhtBytesIn ] ;
const auto dhtUpload = stats [ m_metricIndices . dht . dhtBytesOut ] ;
2017-04-29 14:52:28 +03:00
2019-02-09 18:40:14 +03:00
auto calcRate = [ interval ] ( const quint64 previous , const quint64 current )
2017-04-29 14:52:28 +03:00
{
Q_ASSERT ( current > = previous ) ;
return static_cast < quint64 > ( ( current - previous ) / interval ) ;
} ;
m_status . payloadDownloadRate = calcRate ( m_status . totalPayloadDownload , totalPayloadDownload ) ;
m_status . payloadUploadRate = calcRate ( m_status . totalPayloadUpload , totalPayloadUpload ) ;
m_status . downloadRate = calcRate ( m_status . totalDownload , totalDownload ) ;
m_status . uploadRate = calcRate ( m_status . totalUpload , totalUpload ) ;
m_status . ipOverheadDownloadRate = calcRate ( m_status . ipOverheadDownload , ipOverheadDownload ) ;
m_status . ipOverheadUploadRate = calcRate ( m_status . ipOverheadUpload , ipOverheadUpload ) ;
m_status . dhtDownloadRate = calcRate ( m_status . dhtDownload , dhtDownload ) ;
m_status . dhtUploadRate = calcRate ( m_status . dhtUpload , dhtUpload ) ;
m_status . trackerDownloadRate = calcRate ( m_status . trackerDownload , trackerDownload ) ;
m_status . trackerUploadRate = calcRate ( m_status . trackerUpload , trackerUpload ) ;
m_status . totalDownload = totalDownload ;
m_status . totalUpload = totalUpload ;
m_status . totalPayloadDownload = totalPayloadDownload ;
m_status . totalPayloadUpload = totalPayloadUpload ;
m_status . ipOverheadDownload = ipOverheadDownload ;
m_status . ipOverheadUpload = ipOverheadUpload ;
m_status . trackerDownload = trackerDownload ;
m_status . trackerUpload = trackerUpload ;
m_status . dhtDownload = dhtDownload ;
m_status . dhtUpload = dhtUpload ;
2019-04-03 09:23:35 +03:00
m_status . totalWasted = stats [ m_metricIndices . net . recvRedundantBytes ]
+ stats [ m_metricIndices . net . recvFailedBytes ] ;
m_status . dhtNodes = stats [ m_metricIndices . dht . dhtNodes ] ;
m_status . diskReadQueue = stats [ m_metricIndices . peer . numPeersUpDisk ] ;
m_status . diskWriteQueue = stats [ m_metricIndices . peer . numPeersDownDisk ] ;
m_status . peersCount = stats [ m_metricIndices . peer . numPeersConnected ] ;
const int numBlocksRead = stats [ m_metricIndices . disk . numBlocksRead ] ;
const int numBlocksCacheHits = stats [ m_metricIndices . disk . numBlocksCacheHits ] ;
m_cacheStatus . totalUsedBuffers = stats [ m_metricIndices . disk . diskBlocksInUse ] ;
2018-02-02 10:12:49 +03:00
m_cacheStatus . readRatio = static_cast < qreal > ( numBlocksCacheHits ) / std : : max ( numBlocksCacheHits + numBlocksRead , 1 ) ;
2019-04-03 09:23:35 +03:00
m_cacheStatus . jobQueueLength = stats [ m_metricIndices . disk . queuedDiskJobs ] ;
2017-08-14 17:27:31 +03:00
2019-04-03 09:23:35 +03:00
const quint64 totalJobs = stats [ m_metricIndices . disk . writeJobs ] + stats [ m_metricIndices . disk . readJobs ]
+ stats [ m_metricIndices . disk . hashJobs ] ;
m_cacheStatus . averageJobTime = ( totalJobs > 0 )
? ( stats [ m_metricIndices . disk . diskJobTime ] / totalJobs ) : 0 ;
2017-04-29 14:52:28 +03:00
emit statsUpdated ( ) ;
}
2017-04-29 14:45:30 +03:00
2019-09-13 07:31:26 +03:00
# if (LIBTORRENT_VERSION_NUM >= 10200)
void Session : : handleAlertsDroppedAlert ( const lt : : alerts_dropped_alert * p ) const
{
LogMsg ( tr ( " Error: Internal alert queue full and alerts were dropped, you might see degraded performance. Dropped alert types: %1. Message: %2 " )
. arg ( QString : : fromStdString ( p - > dropped_alerts . to_string ( ) ) , QString : : fromStdString ( p - > message ( ) ) ) , Log : : CRITICAL ) ;
}
# endif
2019-05-07 06:22:39 +03:00
void Session : : handleStateUpdateAlert ( const lt : : state_update_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-08-17 09:58:10 +03:00
QVector < BitTorrent : : TorrentHandle * > updatedTorrents ;
updatedTorrents . reserve ( p - > status . size ( ) ) ;
2019-05-07 06:22:39 +03:00
for ( const lt : : torrent_status & status : p - > status ) {
2015-04-19 18:17:47 +03:00
TorrentHandle * const torrent = m_torrents . value ( status . info_hash ) ;
2019-05-27 10:15:59 +03:00
if ( ! torrent )
continue ;
torrent - > handleStateUpdate ( status ) ;
2019-08-17 09:58:10 +03:00
updatedTorrents . push_back ( torrent ) ;
2015-04-19 18:17:47 +03:00
}
2019-09-24 06:29:07 +03:00
if ( ! updatedTorrents . isEmpty ( ) )
emit torrentsUpdated ( updatedTorrents ) ;
2015-04-19 18:17:47 +03:00
}