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 15:12:49 +08:00
# include <algorithm>
2017-09-07 03:00:04 +03:00
# include <cstdlib>
# include <queue>
2017-10-08 09:59:52 +03:00
# include <string>
2017-09-07 03:00:04 +03:00
2019-04-26 03:02:47 +08:00
# ifdef Q_OS_WIN
# include <wincrypt.h>
# include <iphlpapi.h>
# endif
2015-04-19 18:17:47 +03:00
# include <QDebug>
# include <QDir>
# include <QHostAddress>
# include <QNetworkAddressEntry>
2016-04-19 09:54:48 +03:00
# include <QNetworkInterface>
2018-05-24 23:41:03 +08: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 16:15:41 +08:00
# if (LIBTORRENT_VERSION_NUM >= 10200)
# include <libtorrent/read_resume_data.hpp>
# endif
2018-03-06 14:50:10 +00: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 16:58:30 +02: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 20:33:21 +08: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
2017-08-15 21:57:34 +03:00
# if defined(Q_OS_WIN) && (_WIN32_WINNT < 0x0600)
2017-08-13 00:58:22 +03:00
using NETIO_STATUS = LONG ;
# endif
2016-10-30 00:11:52 +03:00
static const char PEER_ID [ ] = " qB " ;
static const char RESUME_FOLDER [ ] = " BT_backup " ;
2017-03-07 23:41:29 +02: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
2016-02-09 11:56:48 +03:00
bool readFile ( const QString & path , QByteArray & buf ) ;
2018-12-08 01:01:09 +02:00
bool loadTorrentResumeData ( const QByteArray & data , CreateTorrentParams & torrentParams , int & queuePos , MagnetUri & magnetUri ) ;
2016-02-09 11:56:48 +03:00
2019-05-07 11:22:39 +08:00
void torrentQueuePositionUp ( const lt : : torrent_handle & handle ) ;
void torrentQueuePositionDown ( const lt : : torrent_handle & handle ) ;
void torrentQueuePositionTop ( const lt : : torrent_handle & handle ) ;
void torrentQueuePositionBottom ( const lt : : torrent_handle & handle ) ;
2016-02-09 11:56:48 +03:00
2017-08-13 00:58:22 +03:00
# ifdef Q_OS_WIN
QString convertIfaceNameToGuid ( const QString & name ) ;
# endif
2016-02-09 11:56:48 +03:00
QStringMap map_cast ( const QVariantMap & map )
{
QStringMap result ;
2018-03-06 14:50:10 +00: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 14:50:10 +00: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 ;
}
2019-03-06 08:58:07 +03:00
template < typename LTStr >
QString fromLTString ( const LTStr & str )
{
return QString : : fromUtf8 ( str . data ( ) , static_cast < int > ( str . size ( ) ) ) ;
}
2017-06-04 19:22:17 -05:00
template < typename Entry >
QSet < QString > entryListToSetImpl ( const Entry & entry )
{
Q_ASSERT ( entry . type ( ) = = Entry : : list_t ) ;
QSet < QString > output ;
for ( int i = 0 ; i < entry . list_size ( ) ; + + i ) {
2019-03-06 08:58:07 +03:00
const QString tag = fromLTString ( entry . list_string_value_at ( i ) ) ;
2017-06-04 19:22:17 -05:00
if ( Session : : isValidTag ( tag ) )
output . insert ( tag ) ;
else
qWarning ( ) < < QString ( " Dropping invalid stored tag: %1 " ) . arg ( tag ) ;
}
return output ;
}
2019-05-07 11:22:39 +08:00
bool isList ( const lt : : bdecode_node & entry )
2017-06-04 19:22:17 -05:00
{
2019-05-07 11:22:39 +08:00
return entry . type ( ) = = lt : : bdecode_node : : list_t ;
2017-06-04 19:22:17 -05:00
}
2019-05-07 11:22:39 +08:00
QSet < QString > entryListToSet ( const lt : : bdecode_node & entry )
2017-06-04 19:22:17 -05:00
{
return entryListToSetImpl ( entry ) ;
}
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 13:25:29 +02: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 14:50:10 +00:00
for ( auto i = categories . cbegin ( ) ; i ! = categories . cend ( ) ; + + i ) {
const QString & category = i . key ( ) ;
2018-11-27 22:15:04 +02: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 14:21:55 +08: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 20:26:28 +08:00
template < typename T >
std : : function < T ( const T & ) > clampValue ( const T lower , const T upper )
{
2019-02-04 13:09:21 +08:00
// TODO: change return type to `auto` when using C++17
2017-08-15 20:26:28 +08:00
return [ lower , upper ] ( const T value ) - > T
{
if ( value < lower )
return lower ;
if ( value > upper )
return upper ;
return value ;
} ;
}
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_deferredConfigureScheduled ( false )
2016-06-03 17:03:17 +03:00
, m_IPFilteringChanged ( false )
, m_listenInterfaceChanged ( true )
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-30 23:04:14 +02:00
, m_announceToAllTrackers ( BITTORRENT_SESSION_KEY ( " AnnounceToAllTrackers " ) , false )
2017-10-30 20:04:14 +02:00
, m_announceToAllTiers ( BITTORRENT_SESSION_KEY ( " AnnounceToAllTiers " ) , true )
2018-06-19 00:10:57 +08:00
, m_asyncIOThreads ( BITTORRENT_SESSION_KEY ( " AsyncIOThreadsCount " ) , 4 )
2019-07-03 17:44:16 +08:00
, m_filePoolSize ( BITTORRENT_SESSION_KEY ( " FilePoolSize " ) , 40 )
2018-08-28 11:18:07 +01:00
, m_checkingMemUsage ( BITTORRENT_SESSION_KEY ( " CheckingMemUsageSize " ) , 16 )
2017-09-10 21:04:40 +08: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 17:03:14 +08:00
# ifdef Q_OS_WIN
, m_coalesceReadWriteEnabled ( BITTORRENT_SESSION_KEY ( " CoalesceReadWrite " ) , true )
# else
, m_coalesceReadWriteEnabled ( BITTORRENT_SESSION_KEY ( " CoalesceReadWrite " ) , false )
# endif
2017-08-11 19:15:18 +08:00
, m_isSuggestMode ( BITTORRENT_SESSION_KEY ( " SuggestMode " ) , false )
2017-08-11 19:48:58 +08: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 12:03:44 +08: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 13:22:24 +02: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 01:22:11 +02: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-14 04:29:54 +08: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 22:37:08 +08:00
, m_utpMixedMode ( BITTORRENT_SESSION_KEY ( " uTPMixedMode " ) , MixedModeAlgorithm : : TCP
2017-08-16 02:23:07 +08:00
, clampValue ( MixedModeAlgorithm : : TCP , MixedModeAlgorithm : : Proportional ) )
2017-08-11 16:37:32 +08: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 13:01:50 -04:30
, 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 12:58:09 +08:00
, m_saveResumeDataInterval ( BITTORRENT_SESSION_KEY ( " SaveResumeDataInterval " ) , 60 )
2016-05-01 11:05:52 +03:00
, m_port ( BITTORRENT_SESSION_KEY ( " Port " ) , 8999 )
, m_useRandomPort ( BITTORRENT_SESSION_KEY ( " UseRandomPort " ) , false )
, m_networkInterface ( BITTORRENT_SESSION_KEY ( " Interface " ) )
2016-10-31 01:40:26 +02: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-16 02:23:07 +08: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-04 19:22:17 -05: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 13:25:29 +02: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 19:40:34 +07:00
, m_bannedIPs ( " State/BannedIPs "
, QStringList ( )
, [ ] ( const QStringList & value )
{
QStringList tmp = value ;
tmp . sort ( ) ;
return tmp ;
}
)
2016-10-31 02:06:29 +02:00
, m_wasPexEnabled ( m_isPeXEnabled )
2015-04-19 18:17:47 +03:00
, m_numResumeData ( 0 )
, m_extraLimit ( 0 )
2018-05-21 01:09:58 +03:00
, m_recentErroredTorrentsTimer ( new QTimer ( this ) )
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 ) ;
connect ( m_recentErroredTorrentsTimer , & QTimer : : timeout , this , [ this ] ( ) { m_recentErroredTorrents . clear ( ) ; } ) ;
2016-02-07 13:01:50 -04:30
m_seedingLimitTimer = new QTimer ( this ) ;
m_seedingLimitTimer - > setInterval ( 10000 ) ;
2017-10-30 23:59:13 +08:00
connect ( m_seedingLimitTimer , & QTimer : : timeout , this , & Session : : processShareLimits ) ;
2015-04-19 18:17:47 +03:00
2015-11-10 00:22:18 +02:00
// Set severity level of libtorrent session
2019-07-18 19:53:04 +03:00
const LTAlertCategory alertMask = lt : : alert : : error_notification
2019-05-07 11:22:39 +08:00
| lt : : alert : : peer_notification
| lt : : alert : : port_mapping_notification
| lt : : alert : : storage_notification
| lt : : alert : : tracker_notification
| lt : : alert : : status_notification
| lt : : alert : : ip_block_notification
2019-05-27 09:30:49 +08:00
| lt : : alert : : performance_warning
| lt : : alert : : file_progress_notification ;
2019-05-07 11:22:39 +08:00
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 ) ;
2016-06-03 17:03:17 +03:00
// Speed up exit
2019-05-07 11:22:39 +08:00
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 ) ;
2017-10-30 20:38:41 +08:00
// libtorrent 1.1 enables UPnP & NAT-PMP by default
2019-05-07 11:22:39 +08:00
// 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 ) ;
2019-04-03 14:23:35 +08:00
# 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
2016-06-03 17:03:17 +03:00
configure ( pack ) ;
2019-03-06 08:58:07 +03:00
m_nativeSession = new lt : : session { pack , LTSessionFlags { 0 } } ;
2016-04-28 10:56:58 +03:00
m_nativeSession - > set_alert_notify ( [ this ] ( )
{
2019-06-19 15:35:29 +08:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( this , & Session : : readAlerts , Qt : : QueuedConnection ) ;
# else
2016-04-28 10:56:58 +03:00
QMetaObject : : invokeMethod ( this , " readAlerts " , Qt : : QueuedConnection ) ;
2019-06-19 15:35:29 +08:00
# endif
2016-04-28 10:56:58 +03:00
} ) ;
2017-05-01 19:19:34 +03:00
configurePeerClasses ( ) ;
2015-04-19 18:17:47 +03:00
// Enabling plugins
2019-05-07 11:22:39 +08:00
//m_nativeSession->add_extension(<::create_metadata_plugin);
m_nativeSession - > add_extension ( & lt : : create_ut_metadata_plugin ) ;
2016-06-03 17:03:17 +03:00
if ( isPeXEnabled ( ) )
2019-05-07 11:22:39 +08:00
m_nativeSession - > add_extension ( & lt : : create_ut_pex_plugin ) ;
m_nativeSession - > add_extension ( & lt : : create_smart_ban_plugin ) ;
2015-04-19 18:17:47 +03:00
2019-07-02 11:49:35 +08:00
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 ) ;
2016-06-03 17:03:17 +03:00
2017-08-20 18:00:23 +03:00
if ( isBandwidthSchedulerEnabled ( ) )
enableBandwidthScheduler ( ) ;
2017-03-07 01:34:55 +02:00
if ( isIPFilteringEnabled ( ) ) {
// Manually banned IPs are handled in that function too(in the slots)
2016-06-03 17:03:17 +03:00
enableIPFilter ( ) ;
2017-03-07 01:34:55 +02:00
}
else {
// Add the banned IPs
2019-05-07 11:22:39 +08:00
lt : : ip_filter filter ;
2017-03-07 01:34:55 +02:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
}
2016-06-03 17:03:17 +03:00
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-04 19:22:17 -05:00
m_tags = QSet < QString > : : fromList ( m_storedTags . value ( ) ) ;
2015-04-19 18:17:47 +03:00
m_refreshTimer = new QTimer ( this ) ;
2016-05-01 11:05:52 +03:00
m_refreshTimer - > setInterval ( refreshInterval ( ) ) ;
2017-10-30 23:59:13 +08:00
connect ( m_refreshTimer , & QTimer : : timeout , this , & Session : : refresh ) ;
2015-04-19 18:17:47 +03:00
m_refreshTimer - > start ( ) ;
m_statistics = new Statistics ( this ) ;
2016-02-07 13:01:50 -04:30
updateSeedingLimitTimer ( ) ;
2016-05-01 11:05:52 +03:00
populateAdditionalTrackers ( ) ;
enableTracker ( isTrackerEnabled ( ) ) ;
2018-04-18 16:59:41 +03:00
connect ( Net : : ProxyConfigurationManager : : instance ( ) , & Net : : ProxyConfigurationManager : : proxyConfigurationChanged
, this , & Session : : configureDeferred ) ;
2016-02-09 11:56:48 +03:00
2015-06-15 00:06:56 +02:00
// Network configuration monitor
2018-04-18 16:59:41 +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 00:06:56 +02:00
2015-12-13 15:38:19 +03:00
m_ioThread = new QThread ( this ) ;
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 23:59:13 +08:00
connect ( m_ioThread , & QThread : : finished , m_resumeDataSavingManager , & QObject : : deleteLater ) ;
2015-12-13 15:38:19 +03:00
m_ioThread - > start ( ) ;
2018-07-04 14:02:05 +08:00
// Regular saving of fastresume data
m_resumeDataTimer = new QTimer ( this ) ;
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
qDebug ( " * BitTorrent Session constructed " ) ;
}
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 11:49:35 +08: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 17:40:14 +02: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 11:49:35 +08: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 17:40:14 +02:00
void Session : : setPeXEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
2016-10-31 02:06:29 +02:00
m_isPeXEnabled = enabled ;
if ( m_wasPexEnabled ! = enabled )
2019-07-02 11:49:35 +08: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 17:40:14 +02: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 22:15:04 +02: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 17:40:14 +02: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 22:15:04 +02: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 17:40:14 +02: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 17:40:14 +02: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 02:22:51 +02: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 02:22:51 +02: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 02:22:51 +02: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 02:22:51 +02: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 13:28:13 +08: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 23:41:03 +08: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 ;
}
2017-09-24 14:54:42 +03:00
const 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 17:40:14 +02: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 22:15:04 +02: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 22:15:04 +02: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 22:15:04 +02: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 22:15:04 +02: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 13:28:13 +08:00
const QString test = name + ' / ' ;
2019-05-12 13:26:06 +08:00
Algorithm : : removeIf ( m_categories , [ this , & test , & result ] ( const QString & category , const QString & )
2018-03-06 14:50:10 +00:00
{
2016-02-09 11:56:48 +03:00
if ( category . startsWith ( test ) ) {
result = true ;
emit categoryRemoved ( category ) ;
2018-03-06 14:50:10 +00:00
return true ;
2016-02-09 11:56:48 +03:00
}
2018-03-06 14:50:10 +00: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 17:40:14 +02: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-04 19:22:17 -05: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 ) ;
m_storedTags = m_tags . toList ( ) ;
emit tagAdded ( tag ) ;
return true ;
}
return false ;
}
bool Session : : removeTag ( const QString & tag )
{
if ( m_tags . remove ( tag ) ) {
2018-11-27 22:15:04 +02:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) )
2017-06-04 19:22:17 -05:00
torrent - > removeTag ( tag ) ;
m_storedTags = m_tags . toList ( ) ;
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 17:40:14 +02: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 17:40:14 +02: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 17:40:14 +02: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 17:40:14 +02: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 17:40:14 +02: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 17:40:14 +02:00
void Session : : setTrackerEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( isTrackerEnabled ( ) ! = enabled ) {
enableTracker ( enabled ) ;
m_isTrackerEnabled = 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 13:01:50 -04:30
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 12:45:52 +08: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 12:45:52 +08: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 ( ) ;
2015-04-19 18:17:47 +03:00
m_resumeFolderLock . close ( ) ;
m_resumeFolderLock . remove ( ) ;
}
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 11:22:39 +08: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 11:22:39 +08: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
// Set BitTorrent session configuration
void Session : : configure ( )
{
qDebug ( " Configuring session " ) ;
2019-05-07 11:22:39 +08:00
lt : : settings_pack settingsPack = m_nativeSession - > get_settings ( ) ;
2016-06-03 17:03:17 +03:00
configure ( settingsPack ) ;
m_nativeSession - > apply_settings ( settingsPack ) ;
2017-05-01 19:19:34 +03:00
configurePeerClasses ( ) ;
2016-05-08 20:45:53 +08:00
2016-06-03 17:03:17 +03:00
if ( m_IPFilteringChanged ) {
2016-10-29 19:14:27 +03:00
if ( isIPFilteringEnabled ( ) )
2016-06-03 17:03:17 +03:00
enableIPFilter ( ) ;
else
disableIPFilter ( ) ;
m_IPFilteringChanged = false ;
}
m_deferredConfigureScheduled = false ;
qDebug ( " Session configured " ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08: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 22:15:04 +02:00
for ( const QString & ip : asConst ( m_bannedIPs . value ( ) ) ) {
2016-06-03 17:03:17 +03:00
boost : : system : : error_code ec ;
2019-05-07 11:22:39 +08: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 11:22:39 +08:00
filter . add_rule ( addr , addr , lt : : ip_filter : : blocked ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-05-07 11:22:39 +08: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 17:40:14 +02:00
const int maxDownloads = maxActiveDownloads ( ) ;
const int maxActive = maxActiveTorrents ( ) ;
2015-09-16 21:57:50 +03:00
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 14:33:46 +08:00
void Session : : applyBandwidthLimits ( lt : : settings_pack & settingsPack ) const
2017-08-20 18:00:23 +03:00
{
const bool altSpeedLimitEnabled = isAltGlobalSpeedLimitEnabled ( ) ;
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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-05-09 12:45:52 +08:00
void Session : : configure ( lt : : settings_pack & settingsPack )
2015-04-19 18:17:47 +03:00
{
2019-07-02 12:03:44 +08: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
2016-06-03 17:03:17 +03:00
if ( m_listenInterfaceChanged ) {
const ushort port = this - > port ( ) ;
2019-02-09 17:40:14 +02:00
const std : : pair < int , int > ports ( port , port ) ;
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : max_retry_port_bind , ports . second - ports . first ) ;
2018-11-18 20:40:37 +02:00
for ( QString ip : getListeningIPs ( ) ) {
2019-05-07 11:22:39 +08:00
lt : : error_code ec ;
2016-06-03 17:03:17 +03:00
std : : string interfacesStr ;
if ( ip . isEmpty ( ) ) {
ip = QLatin1String ( " 0.0.0.0 " ) ;
interfacesStr = std : : string ( ( QString ( " %1:%2 " ) . arg ( ip ) . arg ( port ) ) . toLatin1 ( ) . constData ( ) ) ;
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " qBittorrent is trying to listen on any interface port: %1 "
, " e.g: qBittorrent is trying to listen on any interface port: TCP/6881 " )
. arg ( QString : : number ( port ) )
, Log : : INFO ) ;
2016-06-03 17:03:17 +03:00
2019-05-07 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : listen_interfaces , interfacesStr ) ;
2016-06-03 17:03:17 +03:00
break ;
}
2019-05-07 11:22:39 +08:00
const lt : : address addr = lt : : address : : from_string ( ip . toLatin1 ( ) . constData ( ) , ec ) ;
2016-06-03 17:03:17 +03:00
if ( ! ec ) {
interfacesStr = std : : string ( ( addr . is_v6 ( ) ? QString ( " [%1]:%2 " ) : QString ( " %1:%2 " ) )
. arg ( ip ) . arg ( port ) . toLatin1 ( ) . constData ( ) ) ;
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " qBittorrent is trying to listen on interface %1 port: %2 "
, " e.g: qBittorrent is trying to listen on interface 192.168.0.1 port: TCP/6881 " )
. arg ( ip ) . arg ( port )
, Log : : INFO ) ;
2019-05-07 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : listen_interfaces , interfacesStr ) ;
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
// the interface's Luid and not the GUID.
// Libtorrent expects GUIDs for the 'outgoing_interfaces' setting.
if ( ! networkInterface ( ) . isEmpty ( ) ) {
2019-02-09 17:40:14 +02:00
const QString guid = convertIfaceNameToGuid ( networkInterface ( ) ) ;
2017-08-13 00:58:22 +03:00
if ( ! guid . isEmpty ( ) ) {
2019-05-07 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , guid . toStdString ( ) ) ;
2017-08-13 00:58:22 +03:00
}
else {
2019-05-07 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , chosenIP . toStdString ( ) ) ;
2017-08-13 00:58:22 +03:00
LogMsg ( tr ( " Could not get GUID of configured network interface. Binding to IP %1 " ) . arg ( chosenIP )
, Log : : WARNING ) ;
}
}
# else
2019-05-07 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : outgoing_interfaces , networkInterface ( ) . toStdString ( ) ) ;
2018-11-06 17:49:17 +02:00
# endif // Q_OS_WIN
2016-06-03 17:03:17 +03:00
m_listenInterfaceChanged = false ;
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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 17:50:55 +08:00
// proxy
2019-02-09 17:40:14 +02:00
const auto proxyManager = Net : : ProxyConfigurationManager : : instance ( ) ;
const Net : : ProxyConfiguration proxyConfig = proxyManager - > proxyConfiguration ( ) ;
2015-04-19 18:17:47 +03:00
2019-06-16 17:50:55 +08: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 17:50:55 +08:00
settingsPack . set_bool ( lt : : settings_pack : : proxy_peer_connections , isProxyPeerConnectionsEnabled ( ) ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08: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 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : aio_threads , asyncIOThreads ( ) ) ;
2018-06-19 00:10:57 +08:00
2019-07-03 17:44:16 +08:00
settingsPack . set_int ( lt : : settings_pack : : file_pool_size , filePoolSize ( ) ) ;
2018-08-28 11:18:07 +01:00
const int checkingMemUsageSize = checkingMemUsage ( ) * 64 ;
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : checking_mem_usage , checkingMemUsageSize ) ;
2018-08-28 11:18:07 +01:00
2018-04-14 22:53:45 +03:00
const int cacheSize = ( diskCacheSize ( ) > - 1 ) ? ( diskCacheSize ( ) * 64 ) : - 1 ;
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 17:03:14 +08:00
2019-05-07 11:22:39 +08:00
settingsPack . set_bool ( lt : : settings_pack : : coalesce_reads , isCoalesceReadWriteEnabled ( ) ) ;
settingsPack . set_bool ( lt : : settings_pack : : coalesce_writes , isCoalesceReadWriteEnabled ( ) ) ;
2018-01-29 17:03:14 +08:00
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08:00
settingsPack . set_str ( lt : : settings_pack : : announce_ip , announceIP ( ) . toStdString ( ) ) ;
2016-06-03 17:03:17 +03:00
// Super seeding
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : unchoke_slots_limit , maxUploads ( ) ) ;
2016-06-03 17:03:17 +03:00
// uTP
2017-09-14 04:29:54 +08:00
switch ( btProtocol ( ) ) {
case BTProtocol : : Both :
default :
2019-05-07 11:22:39 +08: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-14 04:29:54 +08:00
break ;
case BTProtocol : : TCP :
2019-05-07 11:22:39 +08: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-14 04:29:54 +08:00
break ;
case BTProtocol : : UTP :
2019-05-07 11:22:39 +08: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-14 04:29:54 +08:00
break ;
}
2017-08-16 02:23:07 +08:00
switch ( utpMixedMode ( ) ) {
case MixedModeAlgorithm : : TCP :
default :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : mixed_mode_algorithm , lt : : settings_pack : : prefer_tcp ) ;
2017-08-16 02:23:07 +08:00
break ;
case MixedModeAlgorithm : : Proportional :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : mixed_mode_algorithm , lt : : settings_pack : : peer_proportional ) ;
2017-08-16 02:23:07 +08:00
break ;
}
2016-06-03 17:03:17 +03:00
2019-05-07 11:22:39 +08:00
settingsPack . set_bool ( lt : : settings_pack : : allow_multiple_connections_per_ip , multiConnectionsPerIpEnabled ( ) ) ;
2016-06-03 17:03:17 +03:00
2019-05-07 11:22:39 +08:00
settingsPack . set_bool ( lt : : settings_pack : : apply_ip_filter_to_trackers , isTrackerFilteringEnabled ( ) ) ;
2016-06-03 17:03:17 +03:00
2019-05-07 11:22:39 +08:00
settingsPack . set_bool ( lt : : settings_pack : : enable_dht , isDHTEnabled ( ) ) ;
2016-10-31 02:52:29 +02:00
if ( isDHTEnabled ( ) )
2019-05-07 11:22:39 +08: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 17:06:31 +08:00
switch ( chokingAlgorithm ( ) ) {
2017-08-16 02:23:07 +08:00
case ChokingAlgorithm : : FixedSlots :
2017-08-11 17:06:31 +08:00
default :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : choking_algorithm , lt : : settings_pack : : fixed_slots_choker ) ;
2017-08-11 17:06:31 +08:00
break ;
2017-08-16 02:23:07 +08:00
case ChokingAlgorithm : : RateBased :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : choking_algorithm , lt : : settings_pack : : rate_based_choker ) ;
2017-08-11 17:06:31 +08:00
break ;
}
2017-08-16 02:23:07 +08:00
switch ( seedChokingAlgorithm ( ) ) {
case SeedChokingAlgorithm : : RoundRobin :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : round_robin ) ;
2017-08-16 02:23:07 +08:00
break ;
case SeedChokingAlgorithm : : FastestUpload :
default :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : fastest_upload ) ;
2017-08-16 02:23:07 +08:00
break ;
case SeedChokingAlgorithm : : AntiLeech :
2019-05-07 11:22:39 +08:00
settingsPack . set_int ( lt : : settings_pack : : seed_choking_algorithm , lt : : settings_pack : : anti_leech ) ;
2017-08-16 02:23:07 +08:00
break ;
}
2016-05-01 11:05:52 +03:00
}
2017-05-01 19:19:34 +03:00
void Session : : configurePeerClasses ( )
{
2019-05-07 11:22:39 +08:00
lt : : ip_filter f ;
2017-11-24 01:03:11 +02: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 11:22:39 +08: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 14:33:46 +08: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 14:33:46 +08:00
f . add_rule ( lt : : address_v6 : : any ( )
2019-05-07 11:22:39 +08: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 15:02:17 +08:00
catch ( const std : : exception & ) { }
2019-06-02 14:33:46 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 14:33:46 +08: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 11:22:39 +08: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 11:22:39 +08: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 14:33:46 +08: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 15:02:17 +08:00
catch ( const std : : exception & ) { }
2019-06-02 14:33:46 +08:00
# endif
2017-05-01 19:19:34 +03:00
}
m_nativeSession - > set_peer_class_filter ( f ) ;
2019-05-07 11:22:39 +08: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 01:56:00 +02:00
if ( ! isUTPRateLimited ( ) ) {
2019-05-07 11:22:39 +08: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 17:40:14 +02: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 ) ;
if ( m_tracker - > start ( ) )
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " Embedded Tracker [ON] " ) , Log : : INFO ) ;
2015-04-19 18:17:47 +03:00
else
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " Failed to start the embedded tracker! " ) , Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
}
else {
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " Embedded Tracker [OFF] " ) , Log : : INFO ) ;
2015-04-19 18:17:47 +03:00
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 ( ) ;
2018-11-27 22:15:04 +02:00
for ( QString tracker : asConst ( additionalTrackers ( ) . split ( ' \n ' ) ) ) {
2016-05-01 11:05:52 +03:00
tracker = tracker . trimmed ( ) ;
if ( ! tracker . isEmpty ( ) )
m_additionalTrackerList < < tracker ;
2015-04-19 18:17:47 +03:00
}
}
2016-02-07 13:01:50 -04:30
void Session : : processShareLimits ( )
2015-04-19 18:17:47 +03:00
{
2016-02-07 13:01:50 -04:30
qDebug ( " Processing share limits... " ) ;
2015-04-19 18:17:47 +03:00
2018-11-27 22:15:04 +02:00
for ( TorrentHandle * const torrent : asConst ( torrents ( ) ) ) {
2016-02-07 13:01:50 -04:30
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 11:49:35 +08:00
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Removed. " ) . arg ( torrent - > name ( ) ) ) ;
2017-09-11 13:11:09 +08:00
deleteTorrent ( torrent - > hash ( ) ) ;
2016-02-07 13:01:50 -04:30
}
else if ( ! torrent - > isPaused ( ) ) {
torrent - > pause ( ) ;
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " '%1' reached the maximum ratio you set. Paused. " ) . arg ( torrent - > name ( ) ) ) ;
2016-02-07 13:01:50 -04:30
}
2018-02-23 01:55:28 +02:00
continue ;
2016-02-07 13:01:50 -04:30
}
2015-04-19 18:17:47 +03:00
}
2016-02-07 13:01:50 -04:30
}
if ( torrent - > seedingTimeLimit ( ) ! = TorrentHandle : : NO_SEEDING_TIME_LIMIT ) {
2019-06-18 23:36:45 +08:00
const qlonglong seedingTimeInMinutes = torrent - > seedingTime ( ) / 60 ;
2016-02-07 13:01:50 -04:30
int seedingTimeLimit = torrent - > seedingTimeLimit ( ) ;
2019-06-20 19:49:19 +08:00
if ( seedingTimeLimit = = TorrentHandle : : USE_GLOBAL_SEEDING_TIME ) {
2016-02-07 13:01:50 -04:30
// If Global Seeding Time Limit is really set...
seedingTimeLimit = globalMaxSeedingMinutes ( ) ;
2019-06-20 19:49:19 +08:00
}
2016-02-07 13:01:50 -04:30
if ( seedingTimeLimit > = 0 ) {
if ( ( seedingTimeInMinutes < = TorrentHandle : : MAX_SEEDING_TIME ) & & ( seedingTimeInMinutes > = seedingTimeLimit ) ) {
if ( m_maxRatioAction = = Remove ) {
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Removed. " ) . arg ( torrent - > name ( ) ) ) ;
2017-09-11 13:11:09 +08:00
deleteTorrent ( torrent - > hash ( ) ) ;
2016-02-07 13:01:50 -04:30
}
else if ( ! torrent - > isPaused ( ) ) {
torrent - > pause ( ) ;
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " '%1' reached the maximum seeding time you set. Paused. " ) . arg ( torrent - > name ( ) ) ) ;
2016-02-07 13:01:50 -04:30
}
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 15:45:21 -04: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 15:45:21 -04: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 15:45:21 -04: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 11:22:39 +08:00
lt : : ip_filter filter = m_nativeSession - > get_ip_filter ( ) ;
2016-06-03 17:03:17 +03:00
boost : : system : : error_code ec ;
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 19:40:34 +07: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
// deleteLocalFiles = true means that the torrent will be removed from the hard-drive too
2019-02-09 17:40:14 +02:00
bool Session : : deleteTorrent ( const QString & hash , const bool deleteLocalFiles )
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
if ( deleteLocalFiles ) {
2019-02-09 17:40:14 +02:00
const QString rootPath = torrent - > rootPath ( true ) ;
2017-08-04 22:29:17 +03:00
if ( ! rootPath . isEmpty ( ) )
// torrent with root folder
2017-09-14 19:55:14 +03:00
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , rootPath , deleteLocalFiles } ;
2017-08-04 22:29:17 +03:00
else if ( torrent - > useTempPath ( ) )
// torrent without root folder still has it in its temporary save path
2017-09-14 19:55:14 +03:00
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , torrent - > savePath ( true ) , deleteLocalFiles } ;
else
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , " " , deleteLocalFiles } ;
2019-05-07 11:22:39 +08:00
m_nativeSession - > remove_torrent ( torrent - > nativeHandle ( ) , lt : : session : : delete_files ) ;
2015-04-19 18:17:47 +03:00
}
else {
2017-09-14 19:55:14 +03:00
m_removingTorrents [ torrent - > hash ( ) ] = { torrent - > name ( ) , " " , deleteLocalFiles } ;
2015-04-19 18:17:47 +03:00
QStringList unwantedFiles ;
if ( torrent - > hasMetadata ( ) )
unwantedFiles = torrent - > absoluteFilePathsUnwanted ( ) ;
2019-05-07 11:22:39 +08: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 22:15:04 +02: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-11 00:15:06 +08:00
QDir ( ) . rmdir ( parentFolder ) ;
2015-04-19 18:17:47 +03:00
}
}
// Remove it from torrent resume directory
2019-02-09 17:40:14 +02: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 20:40:37 +02: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 )
{
if ( ! m_loadedMetadata . contains ( hash ) ) return false ;
m_loadedMetadata . remove ( hash ) ;
2019-05-07 11:22:39 +08: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 11:22:39 +08:00
m_nativeSession - > remove_torrent ( torrent , lt : : session : : delete_files ) ;
2015-04-19 18:17:47 +03:00
qDebug ( " Preloaded torrent deleted. " ) ;
return true ;
}
2018-12-08 01:01:09 +02:00
void Session : : increaseTorrentsQueuePos ( const QStringList & hashes )
2015-04-19 18:17:47 +03:00
{
std : : priority_queue < QPair < int , TorrentHandle * > ,
2018-11-19 12:53:29 +03:00
std : : vector < QPair < int , TorrentHandle * > > ,
std : : greater < QPair < int , TorrentHandle * > > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 01:01:09 +02:00
// Sort torrents by queue position
2018-11-18 20:40:37 +02:00
for ( const InfoHash infoHash : hashes ) {
2018-11-06 17:49:17 +02:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
torrentQueue . push ( qMakePair ( torrent - > queuePosition ( ) , torrent ) ) ;
}
2018-12-08 01:01:09 +02: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 ( ) ) {
TorrentHandle * const torrent = torrentQueue . top ( ) . second ;
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
}
2018-12-08 01:01:09 +02:00
void Session : : decreaseTorrentsQueuePos ( const QStringList & hashes )
2015-04-19 18:17:47 +03:00
{
std : : priority_queue < QPair < int , TorrentHandle * > ,
2018-11-19 12:53:29 +03:00
std : : vector < QPair < int , TorrentHandle * > > ,
std : : less < QPair < int , TorrentHandle * > > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 01:01:09 +02:00
// Sort torrents by queue position
2018-11-18 20:40:37 +02:00
for ( const InfoHash infoHash : hashes ) {
2018-11-06 17:49:17 +02:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
torrentQueue . push ( qMakePair ( torrent - > queuePosition ( ) , torrent ) ) ;
}
2018-12-08 01:01:09 +02: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 ( ) ) {
TorrentHandle * const torrent = torrentQueue . top ( ) . second ;
torrentQueuePositionDown ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-03-06 14:50:10 +00: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
}
2018-12-08 01:01:09 +02:00
void Session : : topTorrentsQueuePos ( const QStringList & hashes )
2015-04-19 18:17:47 +03:00
{
std : : priority_queue < QPair < int , TorrentHandle * > ,
2018-11-19 12:53:29 +03:00
std : : vector < QPair < int , TorrentHandle * > > ,
std : : greater < QPair < int , TorrentHandle * > > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 01:01:09 +02:00
// Sort torrents by queue position
2018-11-18 20:40:37 +02:00
for ( const InfoHash infoHash : hashes ) {
2018-11-06 17:49:17 +02:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
torrentQueue . push ( qMakePair ( torrent - > queuePosition ( ) , torrent ) ) ;
}
2018-12-08 01:01:09 +02:00
// Top torrents queue position (starting with the one in the highest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
TorrentHandle * const torrent = torrentQueue . top ( ) . second ;
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
}
2018-12-08 01:01:09 +02:00
void Session : : bottomTorrentsQueuePos ( const QStringList & hashes )
2015-04-19 18:17:47 +03:00
{
std : : priority_queue < QPair < int , TorrentHandle * > ,
2018-11-19 12:53:29 +03:00
std : : vector < QPair < int , TorrentHandle * > > ,
std : : less < QPair < int , TorrentHandle * > > > torrentQueue ;
2015-04-19 18:17:47 +03:00
2018-12-08 01:01:09 +02:00
// Sort torrents by queue position
2018-11-18 20:40:37 +02:00
for ( const InfoHash infoHash : hashes ) {
2018-11-06 17:49:17 +02:00
TorrentHandle * const torrent = m_torrents . value ( infoHash ) ;
2015-04-19 18:17:47 +03:00
if ( torrent & & ! torrent - > isSeed ( ) )
torrentQueue . push ( qMakePair ( torrent - > queuePosition ( ) , torrent ) ) ;
}
2018-12-08 01:01:09 +02:00
// Bottom torrents queue position (starting with the one in the lowest queue position)
2015-04-19 18:17:47 +03:00
while ( ! torrentQueue . empty ( ) ) {
TorrentHandle * const torrent = torrentQueue . top ( ) . second ;
torrentQueuePositionBottom ( torrent - > nativeHandle ( ) ) ;
torrentQueue . pop ( ) ;
}
2018-03-06 14:50:10 +00: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 ;
}
2015-11-05 19:17:10 +03:00
TorrentStatusReport Session : : torrentStatusReport ( ) const
{
return m_torrentStatusReport ;
}
2018-12-28 13:38:08 +08:00
bool Session : : addTorrent ( const QString & source , const AddTorrentParams & params )
2015-04-19 18:17:47 +03:00
{
2018-12-28 13:38:08 +08:00
// `source`: .torrent file path/url or magnet uri
2018-11-04 14:21:31 -05:00
2018-12-29 20:38:51 +08: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 14:21:31 -05:00
return true ;
2015-04-19 18:17:47 +03:00
}
2018-11-04 14:21:31 -05:00
2018-12-28 13:38:08 +08:00
const MagnetUri magnetUri { source } ;
if ( magnetUri . isValid ( ) )
return addTorrent_impl ( CreateTorrentParams ( params ) , magnetUri ) ;
2018-11-04 14:21:31 -05:00
TorrentFileGuard guard ( source ) ;
2018-12-16 15:51:36 +08:00
if ( addTorrent_impl ( CreateTorrentParams ( params )
, MagnetUri ( ) , TorrentInfo : : loadFromFile ( source ) ) ) {
2018-11-04 14:21:31 -05: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 15:51:36 +08: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 16:15:41 +08: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 16:15:41 +08:00
const bool fromMagnetUri = magnetUri . isValid ( ) ;
2018-07-29 16:02:57 +08:00
// If empty then Automatic mode, otherwise Manual mode
QString savePath = params . savePath . isEmpty ( ) ? categorySavePath ( params . category ) : params . savePath ;
2019-05-07 11:22:39 +08: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 14:23:35 +08:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-07-10 19:54:10 +03:00
handle . auto_managed ( false ) ;
2019-04-03 14:23:35 +08:00
# else
2019-07-10 19:54:10 +03:00
handle . unset_flags ( lt : : torrent_flags : : auto_managed ) ;
2019-04-03 14:23:35 +08: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 16:15:41 +08: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 16:15:41 +08:00
p . flags | = lt : : add_torrent_params : : flag_use_resume_save_path ;
// 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 ) ;
// 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 16:15:41 +08: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
} ) ;
2018-05-19 09:41:33 +03:00
}
2016-01-04 16:00:50 +03:00
2015-04-19 18:17:47 +03:00
p . ti = torrentInfo . nativeInfo ( ) ;
}
2019-06-28 21:24:39 +03:00
// Common
2019-04-03 14:23:35 +08: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 14:23:35 +08: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 14:23:35 +08:00
# endif
2015-04-19 18:17:47 +03:00
2019-06-28 21:24:39 +03:00
if ( ! hasCompleteFastresume ) {
2019-06-29 16:15:41 +08: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 ;
// 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 14:23:35 +08:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-06-29 16:15:41 +08:00
p . flags | = lt : : add_torrent_params : : flag_seed_mode ;
2019-04-03 14:23:35 +08:00
# else
2019-06-29 16:15:41 +08:00
p . flags | = lt : : torrent_flags : : seed_mode ;
2019-04-03 14:23:35 +08: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 16:15:41 +08:00
p . flags & = ~ lt : : add_torrent_params : : flag_seed_mode ;
2019-03-06 08:58:07 +03:00
# else
2019-06-29 16:15:41 +08:00
p . flags & = ~ lt : : torrent_flags : : seed_mode ;
2019-04-03 14:23:35 +08:00
# endif
2019-06-29 16:15:41 +08: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 16:15:41 +08: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 15:31:30 -04:00
const QDir dir ( dirPath ) ;
2016-04-19 09:54:48 +03:00
bool found = false ;
2017-04-27 15:31:30 -04: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 15:31:30 -04:00
else if ( dir . exists ( filePath + QB_EXT ) ) {
2016-04-19 09:54:48 +03:00
found = true ;
2017-04-27 15:31:30 -04: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 17:40:14 +02: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 11:22:39 +08: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 11:22:39 +08:00
p . storage_mode = lt : : storage_mode_allocate ;
2015-04-19 18:17:47 +03:00
else
2019-05-07 11:22:39 +08: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 12:43:25 +08:00
const QString savePath = Utils : : Fs : : tempPath ( ) + static_cast < QString > ( hash ) ;
2017-03-06 23:51:35 +08:00
p . save_path = Utils : : Fs : : toNativePath ( savePath ) . toStdString ( ) ;
2015-04-19 18:17:47 +03:00
// Forced start
2019-04-03 14:23:35 +08:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-05-07 11:22:39 +08:00
p . flags & = ~ lt : : add_torrent_params : : flag_paused ;
p . flags & = ~ lt : : add_torrent_params : : flag_auto_managed ;
2019-04-03 14:23:35 +08: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 14:23:35 +08:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-05-07 11:22:39 +08:00
p . flags | = lt : : add_torrent_params : : flag_upload_mode ;
2019-04-03 14:23:35 +08: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 11:22:39 +08: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 17:40:14 +02: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 17:40:14 +02: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 17:40:14 +02:00
void Session : : generateResumeData ( const bool final )
2015-04-19 18:17:47 +03:00
{
2018-11-27 22:15:04 +02:00
for ( TorrentHandle * const torrent : asConst ( m_torrents ) ) {
2015-04-19 18:17:47 +03:00
if ( ! torrent - > isValid ( ) ) continue ;
2019-06-12 10:54:43 +08:00
2015-04-19 18:17:47 +03:00
if ( ! final & & ! torrent - > needSaveResumeData ( ) ) continue ;
2019-06-12 10:54:43 +08: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 ( )
{
2018-11-19 12:53:29 +03:00
qDebug ( " Saving resume data... " ) ;
2015-04-19 18:17:47 +03:00
// 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-05-07 11:22:39 +08:00
std : : vector < lt : : alert * > alerts ;
2015-04-19 18:17:47 +03:00
getPendingAlerts ( alerts , 30 * 1000 ) ;
if ( alerts . empty ( ) ) {
2018-03-15 00:13:47 +08:00
fprintf ( stderr , " aborting with %d outstanding torrents to save resume data for \n " , m_numResumeData ) ;
2015-04-19 18:17:47 +03:00
break ;
}
2018-11-06 17:49:17 +02:00
for ( const auto a : alerts ) {
2015-04-19 18:17:47 +03:00
switch ( a - > type ( ) ) {
2019-05-07 11:22:39 +08:00
case lt : : save_resume_data_failed_alert : : alert_type :
case lt : : save_resume_data_alert : : alert_type :
2016-11-18 23:04:04 +07: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 22:15:04 +02: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 22:15:04 +02: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 15:35:29 +08: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 15:35:29 +08:00
# endif
2018-11-19 12:53:29 +03:00
}
void Session : : removeTorrentsQueue ( )
{
const QString filename = QLatin1String { " queue " } ;
2019-06-19 15:35:29 +08: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 15:35:29 +08: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 22:15:04 +02: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 22:15:04 +02: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 22:15:04 +02: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 00:06:56 +02:00
void Session : : networkOnlineStateChanged ( const bool online )
{
2019-07-02 11:49:35 +08: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 00:06:56 +02:00
}
2018-04-14 22:53:45 +03:00
void Session : : networkConfigurationChange ( const QNetworkConfiguration & cfg )
2015-06-15 00:06:56 +02:00
{
2016-05-01 11:05:52 +03:00
const QString configuredInterfaceName = networkInterface ( ) ;
2015-12-05 21:22:01 +02: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 00:06:56 +02:00
const QString changedInterface = cfg . name ( ) ;
2016-05-24 00:27:53 +02:00
2015-11-29 14:30:42 +02:00
if ( configuredInterfaceName = = changedInterface ) {
2019-07-02 11:49:35 +08: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 00:06:56 +02:00
}
}
2015-11-07 02:06:07 +02:00
const QStringList Session : : getListeningIPs ( )
2015-04-19 18:17:47 +03:00
{
2015-11-07 02:06:07 +02: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 11:49:35 +08: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 02:06:07 +02:00
return IPs ;
2015-04-19 18:17:47 +03:00
}
// Attempt to listen on provided interface
2015-11-07 02:06:07 +02: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 11:49:35 +08:00
LogMsg ( tr ( " The network interface defined is invalid: %1 " ) . arg ( ifaceName ) , Log : : CRITICAL ) ;
2015-11-07 02:06:07 +02: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 02:06:07 +02:00
const QList < QNetworkAddressEntry > addresses = networkIFace . addressEntries ( ) ;
2015-06-15 00:33:48 +02:00
qDebug ( " This network interface has %d IP addresses " , addresses . size ( ) ) ;
QHostAddress ip ;
QString ipString ;
QAbstractSocket : : NetworkLayerProtocol protocol ;
2018-11-18 20:40:37 +02:00
for ( const QNetworkAddressEntry & entry : addresses ) {
2015-06-15 00:33:48 +02: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 02:06:07 +02:00
if ( ( ! listenIPv6 & & ( protocol = = QAbstractSocket : : IPv6Protocol ) )
| | ( listenIPv6 & & ( protocol = = QAbstractSocket : : IPv4Protocol ) ) )
2015-04-19 18:17:47 +03:00
continue ;
2016-04-13 10:51:29 +02: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 10:51:29 +02:00
if ( ! ifaceAddr . isEmpty ( ) ) {
2016-05-27 01:35:58 +03:00
if ( ifaceAddr = = ipString ) {
IPs . append ( ipString ) ;
break ;
2016-04-13 10:51:29 +02:00
}
}
2016-05-27 01:35:58 +03:00
else {
IPs . append ( ipString ) ;
}
2015-11-07 02:06:07 +02: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 11:49:35 +08: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 02:06:07 +02: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 02:06:07 +02:00
{
2016-06-03 17:03:17 +03:00
m_listenInterfaceChanged = true ;
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 16:52:32 +02: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
}
2016-05-01 11:05:52 +03:00
void Session : : setGlobalDownloadSpeedLimit ( int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 16:52:32 +02:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
limit / = 1024 ;
2016-05-01 11:05:52 +03:00
if ( limit < 0 ) limit = 0 ;
if ( limit = = globalDownloadSpeedLimit ( ) ) return ;
m_globalDownloadSpeedLimit = limit ;
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 16:52:32 +02: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
}
2016-05-01 11:05:52 +03:00
void Session : : setGlobalUploadSpeedLimit ( int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 16:52:32 +02:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
limit / = 1024 ;
2016-05-01 11:05:52 +03:00
if ( limit < 0 ) limit = 0 ;
if ( limit = = globalUploadSpeedLimit ( ) ) return ;
m_globalUploadSpeedLimit = limit ;
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 16:52:32 +02: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
}
2016-05-01 11:05:52 +03:00
void Session : : setAltGlobalDownloadSpeedLimit ( int limit )
2016-02-09 11:56:48 +03:00
{
2016-11-01 16:52:32 +02:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
limit / = 1024 ;
2016-05-01 11:05:52 +03:00
if ( limit < 0 ) limit = 0 ;
if ( limit = = altGlobalDownloadSpeedLimit ( ) ) return ;
m_altGlobalDownloadSpeedLimit = limit ;
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 16:52:32 +02: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
}
2016-05-01 11:05:52 +03:00
void Session : : setAltGlobalUploadSpeedLimit ( int limit )
2015-04-19 18:17:47 +03:00
{
2016-11-01 16:52:32 +02:00
// Unfortunately the value was saved as KiB instead of B.
// But it is better to pass it around internally(+ webui) as Bytes.
limit / = 1024 ;
2016-05-01 11:05:52 +03:00
if ( limit < 0 ) limit = 0 ;
if ( limit = = altGlobalUploadSpeedLimit ( ) ) return ;
m_altGlobalUploadSpeedLimit = limit ;
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 13:46:48 +08: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 13:46:48 +08: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 13:46:48 +08: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 17:40:14 +02: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 14:02:05 +08:00
void Session : : setSaveResumeDataInterval ( const uint value )
2015-04-19 18:17:47 +03:00
{
2018-07-04 14:02:05 +08: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 14:02:05 +08: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
{
2017-02-10 20:33:21 +08:00
static int randomPort = Utils : : Random : : rand ( 1024 , 65535 ) ;
2016-05-01 11:05:52 +03:00
if ( useRandomPort ( ) )
return randomPort ;
return m_port ;
2015-04-19 18:17:47 +03:00
}
2019-02-09 17:40:14 +02:00
void Session : : setPort ( const int port )
2015-04-19 18:17:47 +03:00
{
2016-05-01 11:05:52 +03:00
if ( port ! = this - > port ( ) ) {
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 17:40:14 +02: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 01:40:26 +02: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 17:40:14 +02: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 17:40:14 +02: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 11:49:35 +08: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 17:40:14 +02:00
void Session : : setProxyPeerConnectionsEnabled ( const bool enabled )
2016-05-01 11:05:52 +03:00
{
if ( enabled ! = isProxyPeerConnectionsEnabled ( ) ) {
m_isProxyPeerConnectionsEnabled = enabled ;
configureDeferred ( ) ;
}
}
2017-08-16 02:23:07 +08:00
ChokingAlgorithm Session : : chokingAlgorithm ( ) const
2017-08-11 17:06:31 +08:00
{
return m_chokingAlgorithm ;
}
2019-02-09 17:40:14 +02:00
void Session : : setChokingAlgorithm ( const ChokingAlgorithm mode )
2017-08-11 17:06:31 +08:00
{
if ( mode = = m_chokingAlgorithm ) return ;
m_chokingAlgorithm = mode ;
configureDeferred ( ) ;
}
2017-08-16 02:23:07 +08:00
SeedChokingAlgorithm Session : : seedChokingAlgorithm ( ) const
2017-08-11 17:06:31 +08:00
{
return m_seedChokingAlgorithm ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSeedChokingAlgorithm ( const SeedChokingAlgorithm mode )
2017-08-11 17:06:31 +08: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 17:40:14 +02: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 17:40:14 +02: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 ;
2016-06-03 17:03:17 +03:00
m_IPFilteringChanged = true ;
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 ;
2016-06-03 17:03:17 +03:00
m_IPFilteringChanged = true ;
2016-05-01 11:05:52 +03:00
configureDeferred ( ) ;
}
}
2017-03-06 19:40:34 +07: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 11:49:35 +08: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 19:40:34 +07: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 ;
m_IPFilteringChanged = true ;
configureDeferred ( ) ;
}
QStringList Session : : bannedIPs ( ) const
{
return m_bannedIPs ;
}
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 14:23:35 +08: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 19:44:11 +08: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 14:23:35 +08: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 19:44:11 +08:00
catch ( const std : : exception & ) { }
2016-05-01 11:05:52 +03:00
}
}
}
bool Session : : announceToAllTrackers ( ) const
{
return m_announceToAllTrackers ;
}
2019-02-09 17:40:14 +02: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 20:04:14 +02:00
bool Session : : announceToAllTiers ( ) const
{
return m_announceToAllTiers ;
}
2019-02-09 17:40:14 +02:00
void Session : : setAnnounceToAllTiers ( const bool val )
2017-10-30 20:04:14 +02:00
{
if ( val ! = m_announceToAllTiers ) {
m_announceToAllTiers = val ;
configureDeferred ( ) ;
}
}
2018-06-19 00:10:57 +08: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 17:44:16 +08: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 11:18:07 +01: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 12:52:41 +08:00
# ifdef QBT_APP_64BIT
2019-06-19 22:01:25 +08: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 22:01:25 +08: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 12:52:41 +08:00
# ifdef QBT_APP_64BIT
2019-06-19 22:01:25 +08: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 17:40:14 +02: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 17:40:14 +02: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 17:03:14 +08:00
bool Session : : isCoalesceReadWriteEnabled ( ) const
{
return m_coalesceReadWriteEnabled ;
}
2019-02-09 17:40:14 +02:00
void Session : : setCoalesceReadWriteEnabled ( const bool enabled )
2018-01-29 17:03:14 +08:00
{
if ( enabled = = m_coalesceReadWriteEnabled ) return ;
m_coalesceReadWriteEnabled = enabled ;
configureDeferred ( ) ;
}
2017-08-11 19:15:18 +08:00
bool Session : : isSuggestModeEnabled ( ) const
{
return m_isSuggestMode ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSuggestMode ( const bool mode )
2017-08-11 19:15:18 +08:00
{
if ( mode = = m_isSuggestMode ) return ;
m_isSuggestMode = mode ;
configureDeferred ( ) ;
}
2017-08-11 19:48:58 +08:00
int Session : : sendBufferWatermark ( ) const
{
return m_sendBufferWatermark ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSendBufferWatermark ( const int value )
2017-08-11 19:48:58 +08:00
{
if ( value = = m_sendBufferWatermark ) return ;
m_sendBufferWatermark = value ;
configureDeferred ( ) ;
}
int Session : : sendBufferLowWatermark ( ) const
{
return m_sendBufferLowWatermark ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSendBufferLowWatermark ( const int value )
2017-08-11 19:48:58 +08:00
{
if ( value = = m_sendBufferLowWatermark ) return ;
m_sendBufferLowWatermark = value ;
configureDeferred ( ) ;
}
int Session : : sendBufferWatermarkFactor ( ) const
{
return m_sendBufferWatermarkFactor ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSendBufferWatermarkFactor ( const int value )
2017-08-11 19:48:58 +08:00
{
if ( value = = m_sendBufferWatermarkFactor ) return ;
m_sendBufferWatermarkFactor = value ;
configureDeferred ( ) ;
}
2019-07-02 12:03:44 +08: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 17:40:14 +02: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 11:49:35 +08: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 17:40:14 +02: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 17:40:14 +02: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 13:22:24 +02:00
int Session : : downloadRateForSlowTorrents ( ) const
{
return m_downloadRateForSlowTorrents ;
}
2019-02-09 17:40:14 +02:00
void Session : : setDownloadRateForSlowTorrents ( const int rateInKibiBytes )
2018-02-02 13:22:24 +02:00
{
if ( rateInKibiBytes = = m_downloadRateForSlowTorrents )
return ;
m_downloadRateForSlowTorrents = rateInKibiBytes ;
configureDeferred ( ) ;
}
int Session : : uploadRateForSlowTorrents ( ) const
{
return m_uploadRateForSlowTorrents ;
}
2019-02-09 17:40:14 +02:00
void Session : : setUploadRateForSlowTorrents ( const int rateInKibiBytes )
2018-02-02 13:22:24 +02:00
{
if ( rateInKibiBytes = = m_uploadRateForSlowTorrents )
return ;
m_uploadRateForSlowTorrents = rateInKibiBytes ;
configureDeferred ( ) ;
}
int Session : : slowTorrentsInactivityTimer ( ) const
{
return m_slowTorrentsInactivityTimer ;
}
2019-02-09 17:40:14 +02:00
void Session : : setSlowTorrentsInactivityTimer ( const int timeInSeconds )
2018-02-02 13:22:24 +02: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 17:40:14 +02: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 17:40:14 +02: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 00:06:56 +02:00
{
2016-05-01 11:05:52 +03:00
return m_ignoreLimitsOnLAN ;
2015-06-15 00:06:56 +02:00
}
2019-02-09 17:40:14 +02:00
void Session : : setIgnoreLimitsOnLAN ( const bool ignore )
2015-06-15 00:06:56 +02:00
{
2016-05-01 11:05:52 +03:00
if ( ignore ! = m_ignoreLimitsOnLAN ) {
m_ignoreLimitsOnLAN = ignore ;
configureDeferred ( ) ;
2015-06-15 00:06:56 +02: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 17:40:14 +02: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 01:22:11 +02:00
QString Session : : announceIP ( ) const
2016-05-01 11:05:52 +03:00
{
2016-10-31 01:22:11 +02:00
return m_announceIP ;
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
2016-10-31 01:22:11 +02:00
void Session : : setAnnounceIP ( const QString & ip )
2016-05-01 11:05:52 +03:00
{
2016-10-31 01:22:11 +02: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 10:51:29 +02:00
2019-02-09 17:40:14 +02: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 02:06:07 +02:00
}
2016-05-01 11:05:52 +03:00
}
2015-11-07 02:06:07 +02:00
2016-05-01 11:05:52 +03:00
int Session : : maxConnections ( ) const
{
return m_maxConnections ;
}
2015-11-07 02:06:07 +02: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 02:06:07 +02:00
}
2016-05-01 11:05:52 +03:00
int Session : : maxUploads ( ) const
{
return m_maxUploads ;
}
2015-11-07 02:06:07 +02: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 02:06:07 +02:00
2017-09-14 04:29:54 +08:00
BTProtocol Session : : btProtocol ( ) const
2016-05-01 11:05:52 +03:00
{
2017-09-14 04:29:54 +08:00
return m_btProtocol ;
2016-05-01 11:05:52 +03:00
}
2015-11-07 02:06:07 +02:00
2019-02-09 17:40:14 +02:00
void Session : : setBTProtocol ( const BTProtocol protocol )
2016-05-01 11:05:52 +03:00
{
2017-09-14 04:29:54 +08: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 17:40:14 +02: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-16 02:23:07 +08:00
MixedModeAlgorithm Session : : utpMixedMode ( ) const
2017-08-11 16:28:52 +08:00
{
return m_utpMixedMode ;
}
2019-02-09 17:40:14 +02:00
void Session : : setUtpMixedMode ( const MixedModeAlgorithm mode )
2017-08-11 16:28:52 +08:00
{
if ( mode = = m_utpMixedMode ) return ;
m_utpMixedMode = mode ;
configureDeferred ( ) ;
}
2017-08-11 16:37:32 +08:00
bool Session : : multiConnectionsPerIpEnabled ( ) const
{
return m_multiConnectionsPerIpEnabled ;
}
2019-02-09 17:40:14 +02:00
void Session : : setMultiConnectionsPerIpEnabled ( const bool enabled )
2017-08-11 16:37:32 +08: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 17:40:14 +02: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 17:40:14 +02: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 10:19:24 +08: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 13:01:50 -04:30
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 13:01:50 -04:30
& & ( 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 13:01:50 -04:30
else if ( ! m_seedingLimitTimer - > isActive ( ) ) {
m_seedingLimitTimer - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
}
2016-02-07 13:01:50 -04:30
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-04 19:22:17 -05:00
void Session : : handleTorrentTagAdded ( TorrentHandle * const torrent , const QString & tag )
{
2019-07-22 14:22:26 +03:00
torrent - > saveResumeData ( ) ;
2017-06-04 19:22:17 -05: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-04 19:22:17 -05: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 12:21:17 +08: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 12:21:17 +08: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 12:55:06 +08: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 12:55:06 +08: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 13:28:13 +08: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 11:49:35 +08: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 12:45:52 +08: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 12:45:52 +08: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-05-07 11:22:39 +08: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 15:35:29 +08: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 15:35:29 +08: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 16:05:57 +08: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 13:01:50 -04:30
bool Session : : hasPerTorrentSeedingTimeLimit ( ) const
{
2019-01-11 16:05:57 +08:00
return std : : any_of ( m_torrents . cbegin ( ) , m_torrents . cend ( ) , [ ] ( const TorrentHandle * torrent )
{
return ( torrent - > seedingTimeLimit ( ) > = 0 ) ;
} ) ;
2016-02-07 13:01:50 -04:30
}
2015-04-19 18:17:47 +03:00
void Session : : initResumeFolder ( )
{
2016-05-11 13:25:29 +02:00
m_resumeFolderPath = Utils : : Fs : : expandPathAbs ( specialFolderLocation ( SpecialFolder : : Data ) + RESUME_FOLDER ) ;
2019-02-09 17:40:14 +02:00
const QDir resumeFolderDir ( m_resumeFolderPath ) ;
2015-04-19 18:17:47 +03:00
if ( resumeFolderDir . exists ( ) | | resumeFolderDir . mkpath ( resumeFolderDir . absolutePath ( ) ) ) {
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 ( )
{
2017-05-02 14:39:47 +03:00
if ( ! m_deferredConfigureScheduled ) {
2019-06-19 15:35:29 +08:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
QMetaObject : : invokeMethod ( this
2019-07-25 01:41:09 +08:00
, qOverload < > ( & Session : : configure )
2019-06-19 15:35:29 +08:00
, Qt : : QueuedConnection ) ;
# else
2017-05-02 14:39:47 +03:00
QMetaObject : : invokeMethod ( this , " configure " , Qt : : QueuedConnection ) ;
2019-06-19 15:35:29 +08:00
# endif
2017-05-02 14:39:47 +03:00
m_deferredConfigureScheduled = true ;
}
2016-05-01 11:05:52 +03:00
}
2015-04-19 18:17:47 +03:00
// Enable IP Filtering
2017-03-06 19:40:34 +07: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 01:34:55 +02:00
// 1. Parse the IP filter
2019-05-09 12:45:52 +08:00
// 2. In the slot add the manually banned IPs to the provided lt::ip_filter
2017-03-07 01:34:55 +02: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 01:34:55 +02: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 17:40:14 +02: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 11:22:39 +08:00
lt : : ip_filter filter ;
2017-03-07 01:34:55 +02: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 11:49:35 +08: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 13:28:13 +08: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
2015-12-16 17:08:27 +03:00
typedef struct
{
QString hash ;
MagnetUri magnetUri ;
2018-07-11 15:44:15 +03:00
CreateTorrentParams addTorrentData ;
2015-12-16 17:08:27 +03:00
QByteArray data ;
} TorrentResumeData ;
2017-01-23 15:45:37 +00:00
int resumedTorrentsCount = 0 ;
2019-07-02 11:49:35 +08:00
const auto startupTorrent = [ this , & resumeDataDir , & resumedTorrentsCount ] ( const TorrentResumeData & params )
2015-12-17 16:47:34 +03:00
{
2019-02-09 17:40:14 +02: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 11:49:35 +08:00
LogMsg ( tr ( " Unable to resume torrent '%1'. " , " e.g: Unable to resume torrent 'hash'. " )
. arg ( params . hash ) , Log : : CRITICAL ) ;
2017-01-23 15:45:37 +00: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 15:45:37 +00: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 23:41:03 +08: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 22:15:04 +02: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 11:49:35 +08: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 22:15:04 +02: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 11:49:35 +08:00
. arg ( queueFile . fileName ( ) , queueFile . errorString ( ) ) , Log : : WARNING ) ;
2018-11-19 12:53:29 +03:00
}
if ( ! queue . empty ( ) )
fastresumes = queue + fastresumes . toSet ( ) . subtract ( queue . toSet ( ) ) . toList ( ) ;
2017-01-18 17:17:51 +01:00
}
2018-11-27 22:15:04 +02: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 17:40:14 +02: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 11:22:39 +08:00
lt : : ip_filter filter = m_filterParser - > IPfilter ( ) ;
2017-03-07 01:34:55 +02:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
}
2019-07-02 11:49:35 +08: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 11:22:39 +08:00
lt : : ip_filter filter ;
2017-03-07 01:34:55 +02:00
processBannedIPs ( filter ) ;
m_nativeSession - > set_ip_filter ( filter ) ;
2019-07-02 11:49:35 +08: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-05-07 11:22:39 +08:00
void Session : : getPendingAlerts ( std : : vector < lt : : alert * > & out , const ulong time )
2015-04-19 18:17:47 +03:00
{
Q_ASSERT ( out . empty ( ) ) ;
2016-04-28 10:56:58 +03:00
if ( time > 0 )
2019-05-07 11:22:39 +08:00
m_nativeSession - > wait_for_alert ( lt : : milliseconds ( time ) ) ;
2016-04-28 10:56:58 +03:00
m_nativeSession - > pop_alerts ( & out ) ;
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 17:40:14 +02: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-05-07 11:22:39 +08:00
std : : vector < lt : : alert * > alerts ;
2015-04-19 18:17:47 +03:00
getPendingAlerts ( alerts ) ;
2019-01-08 15:52:12 +03:00
for ( const auto a : alerts )
2015-04-19 18:17:47 +03:00
handleAlert ( a ) ;
}
2019-05-07 11:22:39 +08:00
void Session : : handleAlert ( const lt : : alert * a )
2015-04-19 18:17:47 +03:00
{
try {
switch ( a - > type ( ) ) {
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08:00
case lt : : torrent_delete_failed_alert : : alert_type :
handleTorrentDeleteFailedAlert ( static_cast < const lt : : torrent_delete_failed_alert * > ( a ) ) ;
2015-08-09 15:27:56 +08:00
break ;
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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-05-16 15:02:17 +08:00
catch ( const std : : exception & exc ) {
2017-03-07 19:41:38 +08:00
qWarning ( ) < < " Caught exception in " < < Q_FUNC_INFO < < " : " < < QString : : fromStdString ( exc . what ( ) ) ;
2015-04-19 18:17:47 +03:00
}
}
2019-05-07 11:22:39 +08:00
void Session : : dispatchTorrentAlert ( const lt : : alert * a )
2015-04-19 18:17:47 +03:00
{
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 17:40:14 +02: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 17:40:14 +02: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 11:49:35 +08: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 11:49:35 +08: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 20:03:17 +02:00
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " '%1' added to download list. " , " 'torrent name' was added to download list. " )
. arg ( torrent - > name ( ) ) ) ;
2015-11-29 18:40:24 +02: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 13:01:50 -04:30
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 11:22:39 +08: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 19:41:38 +08:00
QString msg = QString : : fromStdString ( p - > message ( ) ) ;
2019-07-02 11:49:35 +08: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 11:22:39 +08:00
void Session : : handleTorrentRemovedAlert ( const lt : : torrent_removed_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 15:51:36 +08:00
const InfoHash infoHash { p - > info_hash } ;
2017-09-14 19:55:14 +03:00
2018-12-16 15:51:36 +08:00
if ( m_loadedMetadata . contains ( infoHash ) )
emit metadataLoaded ( m_loadedMetadata . take ( infoHash ) ) ;
if ( m_removingTorrents . contains ( infoHash ) ) {
const RemovingTorrentData tmpRemovingTorrentData = m_removingTorrents [ infoHash ] ;
2017-09-14 19:55:14 +03:00
if ( ! tmpRemovingTorrentData . requestedFileDeletion ) {
LogMsg ( tr ( " '%1' was removed from the transfer list. " , " 'xxx.avi' was removed... " ) . arg ( tmpRemovingTorrentData . name ) ) ;
2018-12-16 15:51:36 +08:00
m_removingTorrents . remove ( infoHash ) ;
2017-09-14 19:55:14 +03:00
}
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void Session : : handleTorrentDeletedAlert ( const lt : : torrent_deleted_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 15:51:36 +08:00
const InfoHash infoHash { p - > info_hash } ;
if ( ! m_removingTorrents . contains ( infoHash ) )
2017-09-14 19:55:14 +03:00
return ;
2018-12-16 15:51:36 +08:00
const RemovingTorrentData tmpRemovingTorrentData = m_removingTorrents . take ( infoHash ) ;
2017-09-14 19:55:14 +03:00
Utils : : Fs : : smartRemoveEmptyFolderTree ( tmpRemovingTorrentData . savePathToRemove ) ;
LogMsg ( tr ( " '%1' was removed from the transfer list and hard disk. " , " 'xxx.avi' was removed... " ) . arg ( tmpRemovingTorrentData . name ) ) ;
2015-08-09 15:27:56 +08:00
}
2019-05-07 11:22:39 +08:00
void Session : : handleTorrentDeleteFailedAlert ( const lt : : torrent_delete_failed_alert * p )
2015-08-09 15:27:56 +08:00
{
2018-12-16 15:51:36 +08:00
const InfoHash infoHash { p - > info_hash } ;
if ( ! m_removingTorrents . contains ( infoHash ) )
2017-09-14 19:55:14 +03:00
return ;
2018-12-16 15:51:36 +08:00
const RemovingTorrentData tmpRemovingTorrentData = m_removingTorrents . take ( infoHash ) ;
2015-08-09 15:27:56 +08:00
// libtorrent won't delete the directory if it contains files not listed in the torrent,
// so we remove the directory ourselves
2017-09-14 19:55:14 +03:00
Utils : : Fs : : smartRemoveEmptyFolderTree ( tmpRemovingTorrentData . savePathToRemove ) ;
2019-06-30 20:39:49 +08: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-07-27 00:24:36 +08:00
. arg ( tmpRemovingTorrentData . name , QString : : fromLocal8Bit ( p - > error . message ( ) . c_str ( ) ) )
2019-06-30 20:39:49 +08:00
, Log : : WARNING ) ;
}
else {
LogMsg ( tr ( " '%1' was removed from the transfer list. " , " 'xxx.avi' was removed... " ) . arg ( tmpRemovingTorrentData . name ) ) ;
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void Session : : handleMetadataReceivedAlert ( const lt : : metadata_received_alert * p )
2015-04-19 18:17:47 +03:00
{
2018-12-16 15:51:36 +08:00
const InfoHash hash { p - > handle . info_hash ( ) } ;
2015-04-19 18:17:47 +03:00
if ( m_loadedMetadata . contains ( hash ) ) {
- - m_extraLimit ;
adjustLimits ( ) ;
m_loadedMetadata [ hash ] = TorrentInfo ( p - > handle . torrent_file ( ) ) ;
2019-05-07 11:22:39 +08: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 11:22:39 +08: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 22:21:26 +08:00
if ( ! torrent )
return ;
2018-05-21 01:09:58 +03:00
2019-06-15 22:21:26 +08: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 22:21:26 +08:00
m_recentErroredTorrentsTimer - > start ( ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void Session : : handlePortmapWarningAlert ( const lt : : portmap_error_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-07-02 11:49:35 +08: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 11:22:39 +08: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 11:49:35 +08: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 11:22:39 +08:00
void Session : : handlePeerBlockedAlert ( const lt : : peer_blocked_alert * p )
2015-04-19 18:17:47 +03:00
{
boost : : system : : error_code ec ;
2019-03-06 08:58:07 +03:00
# if LIBTORRENT_VERSION_NUM < 10200
2019-02-09 17:40:14 +02: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08: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 11:22:39 +08:00
case lt : : peer_blocked_alert : : utp_disabled :
2019-02-27 15:22:12 +08: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 11:22:39 +08: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 11:22:39 +08:00
void Session : : handlePeerBanAlert ( const lt : : peer_ban_alert * p )
2015-04-19 18:17:47 +03:00
{
boost : : system : : error_code ec ;
2019-04-03 14:23:35 +08:00
# if (LIBTORRENT_VERSION_NUM < 10200)
2019-02-09 17:40:14 +02:00
const std : : string ip = p - > ip . address ( ) . to_string ( ec ) ;
2019-04-03 14:23:35 +08: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 11:22:39 +08:00
void Session : : handleUrlSeedAlert ( const lt : : url_seed_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " URL seed lookup failed for URL: '%1', message: %2 " )
. arg ( QString : : fromStdString ( p - > server_url ( ) ) , QString : : fromStdString ( p - > message ( ) ) )
, Log : : CRITICAL ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void Session : : handleListenSucceededAlert ( const lt : : listen_succeeded_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-04-03 14:23:35 +08: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 14:23:35 +08:00
break ;
case lt : : listen_succeeded_alert : : tcp :
2015-04-19 18:17:47 +03:00
proto = " TCP " ;
2019-04-03 14:23:35 +08:00
break ;
case lt : : listen_succeeded_alert : : tcp_ssl :
2015-04-19 18:17:47 +03:00
proto = " TCP_SSL " ;
2019-04-03 14:23:35 +08: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
boost : : system : : error_code ec ;
LogMsg ( tr ( " qBittorrent is successfully listening on interface %1 port: %2/%3 "
, " e.g: qBittorrent is successfully listening on interface 192.168.0.1 port: TCP/6881 " )
# if (LIBTORRENT_VERSION_NUM < 10200)
2018-03-06 23:49:12 +08:00
. arg ( p - > endpoint . address ( ) . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > endpoint . port ( ) ) ) , Log : : INFO ) ;
2019-04-03 14:23:35 +08: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 14:23:35 +08: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 11:22:39 +08:00
void Session : : handleListenFailedAlert ( const lt : : listen_failed_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-04-03 14:23:35 +08: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 14:23:35 +08:00
break ;
case lt : : listen_failed_alert : : tcp :
2015-04-19 18:17:47 +03:00
proto = " TCP " ;
2019-04-03 14:23:35 +08:00
break ;
case lt : : listen_failed_alert : : tcp_ssl :
2015-04-19 18:17:47 +03:00
proto = " TCP_SSL " ;
2019-04-03 14:23:35 +08: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 14:23:35 +08:00
break ;
case lt : : socket_type_t : : socks5 :
2015-04-19 18:17:47 +03:00
proto = " SOCKS5 " ;
2019-04-03 14:23:35 +08:00
break ;
case lt : : socket_type_t : : utp_ssl :
proto = " UTP_SSL " ;
break ;
}
# endif
boost : : system : : error_code ec ;
LogMsg ( tr ( " qBittorrent failed listening on interface %1 port: %2/%3. Reason: %4. "
, " e.g: qBittorrent failed listening on interface 192.168.0.1 port: TCP/6881. Reason: already in use. " )
# if (LIBTORRENT_VERSION_NUM < 10200)
2018-03-06 23:49:12 +08:00
. arg ( p - > endpoint . address ( ) . to_string ( ec ) . c_str ( ) , proto , QString : : number ( p - > endpoint . port ( ) )
2019-04-03 14:23:35 +08: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 11:22:39 +08:00
void Session : : handleExternalIPAlert ( const lt : : external_ip_alert * p )
2015-04-19 18:17:47 +03:00
{
boost : : system : : error_code ec ;
2019-07-02 11:49:35 +08:00
LogMsg ( tr ( " External IP: %1 " , " e.g. External IP: 192.168.0.1 " ) . arg ( p - > external_address . to_string ( ec ) . c_str ( ) ) , Log : : INFO ) ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void Session : : handleSessionStatsAlert ( const lt : : session_stats_alert * p )
2017-04-29 14:52:28 +03:00
{
2019-06-01 11:28:37 +08:00
const qreal interval = lt : : total_milliseconds ( p - > timestamp ( ) - m_statsLastTimestamp ) / 1000. ;
m_statsLastTimestamp = p - > timestamp ( ) ;
2019-04-03 14:23:35 +08: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 14:23:35 +08:00
m_status . hasIncomingConnections = static_cast < bool > ( stats [ m_metricIndices . net . hasIncomingConnections ] ) ;
2017-04-29 14:52:28 +03:00
2019-04-03 14:23:35 +08: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 17:40:14 +02: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 14:23:35 +08: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 15:12:49 +08:00
m_cacheStatus . readRatio = static_cast < qreal > ( numBlocksCacheHits ) / std : : max ( numBlocksCacheHits + numBlocksRead , 1 ) ;
2019-04-03 14:23:35 +08:00
m_cacheStatus . jobQueueLength = stats [ m_metricIndices . disk . queuedDiskJobs ] ;
2017-08-14 17:27:31 +03:00
2019-04-03 14:23:35 +08: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-05-07 11:22:39 +08:00
void Session : : handleStateUpdateAlert ( const lt : : state_update_alert * p )
2015-04-19 18:17:47 +03:00
{
2019-05-07 11:22:39 +08: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 15:15:59 +08:00
if ( ! torrent )
continue ;
torrent - > handleStateUpdate ( status ) ;
2015-04-19 18:17:47 +03:00
}
2015-11-05 19:17:10 +03:00
m_torrentStatusReport = TorrentStatusReport ( ) ;
2019-05-27 15:15:59 +08:00
for ( const TorrentHandle * torrent : asConst ( m_torrents ) ) {
2015-04-19 18:17:47 +03:00
if ( torrent - > isDownloading ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbDownloading ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isUploading ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbSeeding ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isCompleted ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbCompleted ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isPaused ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbPaused ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isResumed ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbResumed ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isActive ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbActive ;
2015-04-19 18:17:47 +03:00
if ( torrent - > isInactive ( ) )
2015-11-05 19:17:10 +03:00
+ + m_torrentStatusReport . nbInactive ;
2015-11-11 08:51:22 +02:00
if ( torrent - > isErrored ( ) )
+ + m_torrentStatusReport . nbErrored ;
2015-04-19 18:17:47 +03:00
}
2015-11-05 19:17:10 +03:00
emit torrentsUpdated ( ) ;
2015-04-19 18:17:47 +03:00
}
2016-02-09 11:56:48 +03:00
namespace
2015-04-19 18:17:47 +03:00
{
2016-02-09 11:56:48 +03:00
bool readFile ( const QString & path , QByteArray & buf )
{
QFile file ( path ) ;
if ( ! file . open ( QIODevice : : ReadOnly ) ) {
2017-08-13 13:56:03 +03:00
qDebug ( " Cannot read file %s: %s " , qUtf8Printable ( path ) , qUtf8Printable ( file . errorString ( ) ) ) ;
2016-02-09 11:56:48 +03:00
return false ;
}
2015-04-19 18:17:47 +03:00
2016-02-09 11:56:48 +03:00
buf = file . readAll ( ) ;
return true ;
2016-01-01 16:28:40 +03:00
}
2018-12-08 01:01:09 +02:00
bool loadTorrentResumeData ( const QByteArray & data , CreateTorrentParams & torrentParams , int & queuePos , MagnetUri & magnetUri )
2016-02-09 11:56:48 +03:00
{
2018-07-11 15:44:15 +03:00
torrentParams = CreateTorrentParams ( ) ;
torrentParams . restored = true ;
torrentParams . skipChecking = false ;
2016-02-09 11:56:48 +03:00
2019-05-07 11:22:39 +08:00
lt : : error_code ec ;
lt : : bdecode_node fast ;
lt : : bdecode ( data . constData ( ) , data . constData ( ) + data . size ( ) , fast , ec ) ;
if ( ec | | ( fast . type ( ) ! = lt : : bdecode_node : : dict_t ) ) return false ;
2016-02-09 11:56:48 +03:00
2018-07-11 15:44:15 +03:00
torrentParams . savePath = Profile : : instance ( ) . fromPortablePath (
2019-06-16 20:14:15 +03:00
Utils : : Fs : : toUniformPath ( fromLTString ( fast . dict_find_string_value ( " qBt-savePath " ) ) ) ) ;
2017-10-08 09:59:52 +03:00
2019-03-06 08:58:07 +03:00
LTString ratioLimitString = fast . dict_find_string_value ( " qBt-ratioLimit " ) ;
2017-10-08 09:59:52 +03:00
if ( ratioLimitString . empty ( ) )
2018-07-11 15:44:15 +03:00
torrentParams . ratioLimit = fast . dict_find_int_value ( " qBt-ratioLimit " , TorrentHandle : : USE_GLOBAL_RATIO * 1000 ) / 1000.0 ;
2017-10-08 09:59:52 +03:00
else
2019-03-06 08:58:07 +03:00
torrentParams . ratioLimit = fromLTString ( ratioLimitString ) . toDouble ( ) ;
2018-07-11 15:44:15 +03:00
torrentParams . seedingTimeLimit = fast . dict_find_int_value ( " qBt-seedingTimeLimit " , TorrentHandle : : USE_GLOBAL_SEEDING_TIME ) ;
2016-02-09 11:56:48 +03:00
// **************************************************************************************
// Workaround to convert legacy label to category
// TODO: Should be removed in future
2019-03-06 08:58:07 +03:00
torrentParams . category = fromLTString ( fast . dict_find_string_value ( " qBt-label " ) ) ;
2018-07-11 15:44:15 +03:00
if ( torrentParams . category . isEmpty ( ) )
2016-02-09 11:56:48 +03:00
// **************************************************************************************
2019-03-06 08:58:07 +03:00
torrentParams . category = fromLTString ( fast . dict_find_string_value ( " qBt-category " ) ) ;
2017-06-04 19:22:17 -05:00
// auto because the return type depends on the #if above.
const auto tagsEntry = fast . dict_find_list ( " qBt-tags " ) ;
if ( isList ( tagsEntry ) )
2018-07-11 15:44:15 +03:00
torrentParams . tags = entryListToSet ( tagsEntry ) ;
2019-03-06 08:58:07 +03:00
torrentParams . name = fromLTString ( fast . dict_find_string_value ( " qBt-name " ) ) ;
2018-07-11 15:44:15 +03:00
torrentParams . hasSeedStatus = fast . dict_find_int_value ( " qBt-seedStatus " ) ;
torrentParams . disableTempPath = fast . dict_find_int_value ( " qBt-tempPathDisabled " ) ;
torrentParams . hasRootFolder = fast . dict_find_int_value ( " qBt-hasRootFolder " ) ;
2016-02-09 11:56:48 +03:00
2019-03-06 08:58:07 +03:00
magnetUri = MagnetUri ( fromLTString ( fast . dict_find_string_value ( " qBt-magnetUri " ) ) ) ;
2018-09-05 15:59:22 +03:00
const bool isAutoManaged = fast . dict_find_int_value ( " auto_managed " ) ;
const bool isPaused = fast . dict_find_int_value ( " paused " ) ;
torrentParams . paused = fast . dict_find_int_value ( " qBt-paused " , ( isPaused & & ! isAutoManaged ) ) ;
torrentParams . forced = fast . dict_find_int_value ( " qBt-forced " , ( ! isPaused & & ! isAutoManaged ) ) ;
2018-07-11 15:44:15 +03:00
torrentParams . firstLastPiecePriority = fast . dict_find_int_value ( " qBt-firstLastPiecePriority " ) ;
torrentParams . sequential = fast . dict_find_int_value ( " qBt-sequential " ) ;
2016-02-09 11:56:48 +03:00
2018-12-08 01:01:09 +02:00
queuePos = fast . dict_find_int_value ( " qBt-queuePosition " ) ;
2015-04-19 18:17:47 +03:00
2016-02-09 11:56:48 +03:00
return true ;
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void torrentQueuePositionUp ( const lt : : torrent_handle & handle )
2016-02-09 11:56:48 +03:00
{
try {
handle . queue_position_up ( ) ;
}
2019-05-16 15:02:17 +08:00
catch ( const std : : exception & exc ) {
2016-02-09 11:56:48 +03:00
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void torrentQueuePositionDown ( const lt : : torrent_handle & handle )
2016-02-09 11:56:48 +03:00
{
try {
handle . queue_position_down ( ) ;
}
2019-05-16 15:02:17 +08:00
catch ( const std : : exception & exc ) {
2016-02-09 11:56:48 +03:00
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2019-05-07 11:22:39 +08:00
void torrentQueuePositionTop ( const lt : : torrent_handle & handle )
2016-02-09 11:56:48 +03:00
{
try {
handle . queue_position_top ( ) ;
}
2019-05-16 15:02:17 +08:00
catch ( const std : : exception & exc ) {
2016-02-09 11:56:48 +03:00
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2016-02-09 11:56:48 +03:00
2019-05-07 11:22:39 +08:00
void torrentQueuePositionBottom ( const lt : : torrent_handle & handle )
2016-02-09 11:56:48 +03:00
{
try {
handle . queue_position_bottom ( ) ;
}
2019-05-16 15:02:17 +08:00
catch ( const std : : exception & exc ) {
2016-02-09 11:56:48 +03:00
qDebug ( ) < < Q_FUNC_INFO < < " fails: " < < exc . what ( ) ;
}
2015-04-19 18:17:47 +03:00
}
2017-08-13 00:58:22 +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.
QUuid uuid ( name ) ;
if ( ! uuid . isNull ( ) )
return uuid . toString ( ) . toUpper ( ) ; // Libtorrent expects the GUID in uppercase
using PCONVERTIFACENAMETOLUID = NETIO_STATUS ( WINAPI * ) ( const WCHAR * , PNET_LUID ) ;
2018-04-07 15:35:35 +08:00
const auto ConvertIfaceNameToLuid = Utils : : Misc : : loadWinAPI < PCONVERTIFACENAMETOLUID > ( " Iphlpapi.dll " , " ConvertInterfaceNameToLuidW " ) ;
2019-02-14 19:16:42 +02:00
if ( ! ConvertIfaceNameToLuid ) return { } ;
2017-08-13 00:58:22 +03:00
using PCONVERTIFACELUIDTOGUID = NETIO_STATUS ( WINAPI * ) ( const NET_LUID * , GUID * ) ;
2018-04-07 15:35:35 +08:00
const auto ConvertIfaceLuidToGuid = Utils : : Misc : : loadWinAPI < PCONVERTIFACELUIDTOGUID > ( " Iphlpapi.dll " , " ConvertInterfaceLuidToGuid " ) ;
2019-02-14 19:16:42 +02:00
if ( ! ConvertIfaceLuidToGuid ) return { } ;
2017-08-13 00:58:22 +03:00
NET_LUID luid ;
2019-02-09 17:40:14 +02:00
const LONG res = ConvertIfaceNameToLuid ( name . toStdWString ( ) . c_str ( ) , & luid ) ;
2017-08-13 00:58:22 +03:00
if ( res = = 0 ) {
GUID guid ;
if ( ConvertIfaceLuidToGuid ( & luid , & guid ) = = 0 )
return QUuid ( guid ) . toString ( ) . toUpper ( ) ;
}
2019-02-14 19:16:42 +02:00
return { } ;
2017-08-13 00:58:22 +03:00
}
# endif
2015-04-19 18:17:47 +03:00
}