2007-03-05 13:55:23 +00:00
/*
* Bittorrent Client using Qt4 and libtorrent .
2007-07-14 14:31:59 +00:00
* Copyright ( C ) 2006 Christophe Dumez
2007-03-05 13:55:23 +00:00
*
2007-07-14 14:31:59 +00:00
* 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 .
2007-03-05 13:55:23 +00:00
*
* 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
2007-07-14 14:31:59 +00:00
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*
2009-04-05 17:00:55 +00:00
* 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 .
*
2007-07-14 14:31:59 +00:00
* Contact : chris @ qbittorrent . org
2007-03-05 13:55:23 +00:00
*/
2007-03-27 18:49:29 +00:00
# include <QDir>
2009-07-12 09:08:38 +00:00
# include <QDateTime>
2007-03-27 18:49:29 +00:00
# include <QString>
2008-05-18 09:50:35 +00:00
# include <QSettings>
2010-05-11 15:53:14 +00:00
# include <QNetworkInterface>
# include <QHostAddress>
# include <QNetworkAddressEntry>
2009-11-28 18:23:17 +00:00
# include <stdlib.h>
2007-03-27 18:49:29 +00:00
2009-11-12 18:24:51 +00:00
# include "filesystemwatcher.h"
2007-09-29 08:00:14 +00:00
# include "bittorrent.h"
# include "misc.h"
2009-11-20 07:48:44 +00:00
# include "downloadthread.h"
# include "filterparserthread.h"
2009-11-14 19:08:28 +00:00
# include "preferences.h"
2010-02-28 15:15:00 +00:00
# include "scannedfoldersmodel.h"
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2010-01-02 23:03:46 +00:00
# include "geoip.h"
2010-01-02 22:20:37 +00:00
# endif
2009-11-20 07:48:44 +00:00
# include "torrentpersistentdata.h"
2009-11-18 17:46:59 +00:00
# include "httpserver.h"
2010-01-24 11:57:15 +00:00
# include "bandwidthscheduler.h"
2008-11-02 19:22:10 +00:00
# include <libtorrent/extensions/ut_metadata.hpp>
2010-06-07 08:32:41 +00:00
# include <libtorrent/version.hpp>
# if LIBTORRENT_VERSION_MINOR > 14
2009-11-18 17:46:59 +00:00
# include <libtorrent/extensions/lt_trackers.hpp>
2009-11-18 10:29:20 +00:00
# endif
2007-07-23 12:46:36 +00:00
# include <libtorrent/extensions/ut_pex.hpp>
2008-11-02 19:22:10 +00:00
# include <libtorrent/extensions/smart_ban.hpp>
2009-10-24 14:36:17 +00:00
//#include <libtorrent/extensions/metadata_transfer.hpp>
2007-07-23 12:46:36 +00:00
# include <libtorrent/entry.hpp>
# include <libtorrent/bencode.hpp>
# include <libtorrent/identify_client.hpp>
# include <libtorrent/alert_types.hpp>
2007-08-26 16:25:22 +00:00
# include <libtorrent/torrent_info.hpp>
2007-08-01 20:01:06 +00:00
# include <boost/filesystem/exception.hpp>
2010-03-04 19:29:48 +00:00
# include <queue>
2007-07-23 12:46:36 +00:00
2010-03-03 17:27:25 +00:00
const int MAX_TRACKER_ERRORS = 2 ;
const float MAX_RATIO = 100. ;
2009-12-30 15:07:51 +00:00
enum ProxyType { HTTP = 1 , SOCKS5 = 2 , HTTP_PW = 3 , SOCKS5_PW = 4 , SOCKS4 = 5 } ;
2010-01-01 20:29:03 +00:00
enum VersionType { NORMAL , ALPHA , BETA , RELEASE_CANDIDATE , DEVEL } ;
2007-03-05 13:55:23 +00:00
// Main constructor
2010-02-28 15:15:00 +00:00
Bittorrent : : Bittorrent ( )
2010-03-20 11:30:11 +00:00
: m_scanFolders ( ScanFoldersModel : : instance ( this ) ) ,
preAllocateAll ( false ) , addInPause ( false ) , ratio_limit ( - 1 ) ,
UPnPEnabled ( false ) , NATPMPEnabled ( false ) , LSDEnabled ( false ) ,
DHTEnabled ( false ) , current_dht_port ( 0 ) , queueingEnabled ( false ) ,
torrentExport ( false ) , exiting ( false )
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2010-03-20 11:30:11 +00:00
, geoipDBLoaded ( false ) , resolve_countries ( false )
2010-01-02 22:20:37 +00:00
# endif
2010-03-03 17:27:25 +00:00
{
2007-04-07 07:43:57 +00:00
// To avoid some exceptions
fs : : path : : default_name_check ( fs : : no_check ) ;
2010-01-02 11:22:44 +00:00
// For backward compatibility
// Move .qBittorrent content to XDG folder
// TODO: Remove after some releases (introduced in v2.1.0)
misc : : moveToXDGFolders ( ) ;
2009-11-20 08:01:59 +00:00
// Creating Bittorrent session
2009-08-27 03:13:39 +00:00
// Check if we should spoof utorrent
2009-12-30 13:10:06 +00:00
QList < int > version ;
version < < VERSION_MAJOR ;
version < < VERSION_MINOR ;
version < < VERSION_BUGFIX ;
2010-01-01 20:29:03 +00:00
version < < VERSION_TYPE ;
2009-12-30 13:10:06 +00:00
QString peer_id = Preferences : : getPeerID ( ) ;
if ( peer_id . size ( ) ! = 2 ) peer_id = " qB " ;
if ( peer_id ! = " qB " ) {
QStringList peer_ver = Preferences : : getClientVersion ( ) . split ( ' . ' ) ;
2010-01-01 20:29:03 +00:00
while ( peer_ver . size ( ) < 4 ) {
2009-12-30 13:10:06 +00:00
peer_ver < < " 0 " ;
}
for ( int i = 0 ; i < peer_ver . size ( ) ; + + i ) {
QString ver = peer_ver . at ( i ) ;
2010-01-01 20:29:03 +00:00
if ( ver . size ( ) ! = 1 ) {
ver . truncate ( 1 ) ;
}
2009-12-30 13:10:06 +00:00
version . replace ( i , ver . toInt ( ) ) ;
}
2008-05-18 09:50:35 +00:00
}
2009-12-30 13:10:06 +00:00
// Construct session
2010-03-04 20:19:25 +00:00
s = new session ( fingerprint ( peer_id . toLocal8Bit ( ) . constData ( ) , version . at ( 0 ) , version . at ( 1 ) , version . at ( 2 ) , version . at ( 3 ) ) , 0 ) ;
std : : cout < < " Peer ID: " < < fingerprint ( peer_id . toLocal8Bit ( ) . constData ( ) , version . at ( 0 ) , version . at ( 1 ) , version . at ( 2 ) , version . at ( 3 ) ) . to_string ( ) < < std : : endl ;
addConsoleMessage ( " Peer ID: " + misc : : toQString ( fingerprint ( peer_id . toLocal8Bit ( ) . constData ( ) , version . at ( 0 ) , version . at ( 1 ) , version . at ( 2 ) , version . at ( 3 ) ) . to_string ( ) ) ) ;
2009-09-03 11:29:15 +00:00
2007-03-05 13:55:23 +00:00
// Set severity level of libtorrent session
2008-11-01 23:58:53 +00:00
s - > set_alert_mask ( alert : : error_notification | alert : : peer_notification | alert : : port_mapping_notification | alert : : storage_notification | alert : : tracker_notification | alert : : status_notification | alert : : ip_block_notification ) ;
2008-11-02 13:19:27 +00:00
// Load previous state
loadSessionState ( ) ;
2008-11-02 19:22:10 +00:00
// Enabling plugins
2009-10-24 14:28:00 +00:00
//s->add_extension(&create_metadata_plugin);
2008-11-02 19:22:10 +00:00
s - > add_extension ( & create_ut_metadata_plugin ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2009-10-24 14:28:00 +00:00
s - > add_extension ( create_lt_trackers_plugin ) ;
2009-11-18 10:29:20 +00:00
# endif
2009-12-17 18:00:21 +00:00
if ( Preferences : : isPeXEnabled ( ) ) {
PeXEnabled = true ;
s - > add_extension ( & create_ut_pex_plugin ) ;
} else {
PeXEnabled = false ;
}
2008-11-02 19:22:10 +00:00
s - > add_extension ( & create_smart_ban_plugin ) ;
2007-07-30 13:56:31 +00:00
timerAlerts = new QTimer ( ) ;
connect ( timerAlerts , SIGNAL ( timeout ( ) ) , this , SLOT ( readAlerts ( ) ) ) ;
timerAlerts - > start ( 3000 ) ;
2010-03-20 21:21:42 +00:00
connect ( & resumeDataTimer , SIGNAL ( timeout ( ) ) , this , SLOT ( saveTempFastResumeData ( ) ) ) ;
resumeDataTimer . start ( 180000 ) ; // 3min
2007-03-07 22:36:01 +00:00
// To download from urls
downloader = new downloadThread ( this ) ;
2007-07-22 09:47:27 +00:00
connect ( downloader , SIGNAL ( downloadFinished ( QString , QString ) ) , this , SLOT ( processDownloadedFile ( QString , QString ) ) ) ;
2007-08-26 16:25:22 +00:00
connect ( downloader , SIGNAL ( downloadFailure ( QString , QString ) ) , this , SLOT ( handleDownloadFailure ( QString , QString ) ) ) ;
2009-12-17 20:28:30 +00:00
appendLabelToSavePath = Preferences : : appendTorrentLabel ( ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2009-12-18 13:36:47 +00:00
appendqBExtension = Preferences : : useIncompleteFilesExtension ( ) ;
# endif
2010-03-08 19:19:17 +00:00
connect ( m_scanFolders , SIGNAL ( torrentsAdded ( QStringList & ) ) , this , SLOT ( addTorrentsFromScanFolder ( QStringList & ) ) ) ;
2009-11-14 19:08:28 +00:00
// Apply user settings to Bittorrent session
configureSession ( ) ;
2007-08-26 16:25:22 +00:00
qDebug ( " * BTSession constructed " ) ;
2007-03-07 22:36:01 +00:00
}
2009-11-30 21:30:14 +00:00
session_proxy Bittorrent : : asyncDeletion ( ) {
qDebug ( " Bittorrent session async deletion IN " ) ;
exiting = true ;
2008-11-02 13:19:27 +00:00
// Do some BT related saving
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2009-11-18 12:35:55 +00:00
saveDHTEntry ( ) ;
2010-04-06 16:52:17 +00:00
# endif
2008-11-02 13:19:27 +00:00
saveSessionState ( ) ;
2009-11-18 12:35:55 +00:00
saveFastResumeData ( ) ;
2009-11-28 15:03:27 +00:00
// Delete session
session_proxy sp = s - > abort ( ) ;
delete s ;
2009-11-30 21:30:14 +00:00
qDebug ( " Bittorrent session async deletion OUT " ) ;
return sp ;
}
// Main destructor
Bittorrent : : ~ Bittorrent ( ) {
qDebug ( " BTSession destructor IN " ) ;
if ( ! exiting ) {
// Do some BT related saving
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2009-11-30 21:30:14 +00:00
saveDHTEntry ( ) ;
2010-04-06 16:52:17 +00:00
# endif
2009-11-30 21:30:14 +00:00
saveSessionState ( ) ;
saveFastResumeData ( ) ;
// Delete session
session_proxy sp = s - > abort ( ) ;
delete s ;
}
2007-07-29 14:54:39 +00:00
// Delete our objects
2007-07-30 13:56:31 +00:00
delete timerAlerts ;
2008-09-13 07:33:41 +00:00
if ( BigRatioTimer )
2007-11-24 21:55:19 +00:00
delete BigRatioTimer ;
2008-09-13 07:33:41 +00:00
if ( filterParser )
2008-05-17 20:32:03 +00:00
delete filterParser ;
2007-03-07 22:36:01 +00:00
delete downloader ;
2010-01-24 11:57:15 +00:00
if ( bd_scheduler )
delete bd_scheduler ;
2009-11-18 17:46:59 +00:00
// HTTP Server
if ( httpServer )
delete httpServer ;
2009-11-16 20:28:58 +00:00
if ( timerETA )
delete timerETA ;
2009-11-30 21:30:14 +00:00
qDebug ( " BTSession destructor OUT " ) ;
2007-03-05 13:55:23 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : preAllocateAllFiles ( bool b ) {
2010-03-03 17:27:25 +00:00
const bool change = ( preAllocateAll ! = b ) ;
2007-11-02 22:56:07 +00:00
if ( change ) {
qDebug ( " PreAllocateAll changed, reloading all torrents! " ) ;
preAllocateAll = b ;
2007-09-08 17:07:29 +00:00
}
}
2010-03-18 19:25:16 +00:00
ScanFoldersModel * Bittorrent : : getScanFoldersModel ( ) const {
return m_scanFolders ;
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : deleteBigRatios ( ) {
2009-07-12 07:23:11 +00:00
if ( ratio_limit = = - 1 ) return ;
2009-08-16 03:09:20 +00:00
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( * torrentIT ) ;
2009-08-16 03:09:20 +00:00
if ( ! h . is_valid ( ) ) continue ;
if ( h . is_seed ( ) ) {
2010-03-03 17:27:25 +00:00
const QString hash = h . hash ( ) ;
const float ratio = getRealRatio ( hash ) ;
2009-08-16 03:09:20 +00:00
if ( ratio < = MAX_RATIO & & ratio > ratio_limit ) {
2010-03-03 17:27:25 +00:00
addConsoleMessage ( tr ( " %1 reached the maximum ratio you set. " ) . arg ( h . name ( ) ) ) ;
2009-08-16 03:09:20 +00:00
deleteTorrent ( hash ) ;
//emit torrent_ratio_deleted(fileName);
}
2007-09-09 09:09:24 +00:00
}
2009-08-16 03:09:20 +00:00
}
2007-09-09 09:09:24 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : setDownloadLimit ( QString hash , long val ) {
2007-08-20 06:29:18 +00:00
QTorrentHandle h = getTorrentHandle ( hash ) ;
2009-08-27 09:07:33 +00:00
if ( h . is_valid ( ) ) {
2007-07-30 16:57:28 +00:00
h . set_download_limit ( val ) ;
2009-08-27 09:07:33 +00:00
}
2007-07-14 13:38:29 +00:00
}
2009-11-20 08:01:59 +00:00
bool Bittorrent : : isQueueingEnabled ( ) const {
2008-07-14 22:01:05 +00:00
return queueingEnabled ;
2008-07-14 19:20:18 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : setUploadLimit ( QString hash , long val ) {
2007-08-04 06:23:44 +00:00
qDebug ( " Set upload limit rate to %ld " , val ) ;
2007-08-20 06:29:18 +00:00
QTorrentHandle h = getTorrentHandle ( hash ) ;
2009-08-27 09:07:33 +00:00
if ( h . is_valid ( ) ) {
2007-07-30 16:57:28 +00:00
h . set_upload_limit ( val ) ;
2009-08-27 09:07:33 +00:00
}
2007-07-14 13:38:29 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : handleDownloadFailure ( QString url , QString reason ) {
2007-07-20 20:18:18 +00:00
emit downloadFromUrlFailure ( url , reason ) ;
2009-08-21 07:40:57 +00:00
// Clean up
2010-03-03 17:27:25 +00:00
const QUrl & qurl = QUrl : : fromEncoded ( url . toLocal8Bit ( ) ) ;
const int index = url_skippingDlg . indexOf ( qurl ) ;
2009-08-21 07:40:57 +00:00
if ( index > = 0 )
url_skippingDlg . removeAt ( index ) ;
2009-12-06 14:41:05 +00:00
if ( savepath_fromurl . contains ( qurl ) )
savepath_fromurl . remove ( qurl ) ;
2007-07-20 20:18:18 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : startTorrentsInPause ( bool b ) {
2007-09-08 17:07:29 +00:00
addInPause = b ;
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : setQueueingEnabled ( bool enable ) {
2008-07-14 22:01:05 +00:00
if ( queueingEnabled ! = enable ) {
2008-08-19 00:28:44 +00:00
qDebug ( " Queueing system is changing state... " ) ;
2008-07-14 22:01:05 +00:00
queueingEnabled = enable ;
2008-07-14 19:20:18 +00:00
}
}
2009-11-14 19:08:28 +00:00
// Set BT session configuration
2009-11-20 08:01:59 +00:00
void Bittorrent : : configureSession ( ) {
2009-11-14 19:08:28 +00:00
qDebug ( " Configuring session " ) ;
// Downloads
// * Save path
2009-11-19 08:14:04 +00:00
defaultSavePath = Preferences : : getSavePath ( ) ;
2009-11-14 19:08:28 +00:00
if ( Preferences : : isTempPathEnabled ( ) ) {
setDefaultTempPath ( Preferences : : getTempPath ( ) ) ;
} else {
setDefaultTempPath ( QString : : null ) ;
}
2009-12-17 20:28:30 +00:00
setAppendLabelToSavePath ( Preferences : : appendTorrentLabel ( ) ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2009-12-18 13:36:47 +00:00
setAppendqBExtension ( Preferences : : useIncompleteFilesExtension ( ) ) ;
# endif
2009-11-14 19:08:28 +00:00
preAllocateAllFiles ( Preferences : : preAllocateAllFiles ( ) ) ;
startTorrentsInPause ( Preferences : : addTorrentsInPause ( ) ) ;
2010-02-28 15:15:00 +00:00
// * Scan dirs
const QStringList & scan_dirs = Preferences : : getScanDirs ( ) ;
2010-06-05 18:59:05 +00:00
QList < bool > downloadInDirList = Preferences : : getDownloadInScanDirs ( ) ;
2010-03-22 18:50:04 +00:00
while ( scan_dirs . size ( ) > downloadInDirList . size ( ) ) {
2010-06-05 18:59:05 +00:00
downloadInDirList < < false ;
2010-02-28 15:15:00 +00:00
}
2010-03-22 18:50:04 +00:00
int i = 0 ;
foreach ( const QString & dir , scan_dirs ) {
2010-06-05 18:59:05 +00:00
m_scanFolders - > addPath ( dir , downloadInDirList . at ( i ) ) ;
2010-03-22 18:50:04 +00:00
+ + i ;
2009-11-14 19:08:28 +00:00
}
2010-01-31 13:09:46 +00:00
// * Export Dir
2010-03-03 17:27:25 +00:00
const bool newTorrentExport = Preferences : : isTorrentExportEnabled ( ) ;
2010-01-31 13:09:46 +00:00
if ( torrentExport ! = newTorrentExport ) {
torrentExport = newTorrentExport ;
if ( torrentExport ) {
qDebug ( " Torrent export is enabled, exporting the current torrents " ) ;
exportTorrentFiles ( Preferences : : getExportDir ( ) ) ;
}
}
2009-11-14 19:08:28 +00:00
// Connection
// * Ports binding
2010-03-03 17:27:25 +00:00
const unsigned short old_listenPort = getListenPort ( ) ;
const unsigned short new_listenPort = Preferences : : getSessionPort ( ) ;
2009-12-29 09:59:32 +00:00
if ( old_listenPort ! = new_listenPort ) {
setListeningPort ( new_listenPort ) ;
2010-03-03 17:27:25 +00:00
addConsoleMessage ( tr ( " qBittorrent is bound to port: TCP/%1 " , " e.g: qBittorrent is bound to port: 6881 " ) . arg ( QString : : number ( new_listenPort ) ) ) ;
2009-11-14 19:08:28 +00:00
}
// * Global download limit
2010-03-03 17:27:25 +00:00
const bool alternative_speeds = Preferences : : isAltBandwidthEnabled ( ) ;
2010-01-23 20:45:14 +00:00
int down_limit ;
if ( alternative_speeds )
down_limit = Preferences : : getAltGlobalDownloadLimit ( ) ;
else
down_limit = Preferences : : getGlobalDownloadLimit ( ) ;
2009-11-14 19:08:28 +00:00
if ( down_limit < = 0 ) {
// Download limit disabled
setDownloadRateLimit ( - 1 ) ;
} else {
// Enabled
setDownloadRateLimit ( down_limit * 1024 ) ;
}
2010-01-23 20:45:14 +00:00
int up_limit ;
if ( alternative_speeds )
up_limit = Preferences : : getAltGlobalUploadLimit ( ) ;
else
up_limit = Preferences : : getGlobalUploadLimit ( ) ;
2009-11-14 19:08:28 +00:00
// * Global Upload limit
if ( up_limit < = 0 ) {
// Upload limit disabled
setUploadRateLimit ( - 1 ) ;
} else {
// Enabled
setUploadRateLimit ( up_limit * 1024 ) ;
}
2010-01-24 11:57:15 +00:00
if ( Preferences : : isSchedulerEnabled ( ) ) {
if ( ! bd_scheduler ) {
bd_scheduler = new BandwidthScheduler ( this ) ;
connect ( bd_scheduler , SIGNAL ( switchToAlternativeMode ( bool ) ) , this , SLOT ( useAlternativeSpeedsLimit ( bool ) ) ) ;
}
bd_scheduler - > start ( ) ;
} else {
if ( bd_scheduler ) delete bd_scheduler ;
}
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2009-11-15 10:59:11 +00:00
// Resolve countries
2009-11-15 20:03:23 +00:00
qDebug ( " Loading country resolution settings " ) ;
2010-03-03 17:27:25 +00:00
const bool new_resolv_countries = Preferences : : resolvePeerCountries ( ) ;
2009-11-15 10:59:11 +00:00
if ( resolve_countries ! = new_resolv_countries ) {
2009-11-15 20:03:23 +00:00
qDebug ( " in country reoslution settings " ) ;
2009-11-15 10:59:11 +00:00
resolve_countries = new_resolv_countries ;
2009-11-15 20:03:23 +00:00
if ( resolve_countries & & ! geoipDBLoaded ) {
qDebug ( " Loading geoip database " ) ;
GeoIP : : loadDatabase ( s ) ;
geoipDBLoaded = true ;
}
2009-11-15 10:59:11 +00:00
// Update torrent handles
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( h . is_valid ( ) )
h . resolve_countries ( resolve_countries ) ;
}
}
2010-01-02 22:20:37 +00:00
# endif
2009-11-14 19:08:28 +00:00
// * UPnP
if ( Preferences : : isUPnPEnabled ( ) ) {
enableUPnP ( true ) ;
addConsoleMessage ( tr ( " UPnP support [ON] " ) , QString : : fromUtf8 ( " blue " ) ) ;
} else {
enableUPnP ( false ) ;
addConsoleMessage ( tr ( " UPnP support [OFF] " ) , QString : : fromUtf8 ( " blue " ) ) ;
}
// * NAT-PMP
if ( Preferences : : isNATPMPEnabled ( ) ) {
enableNATPMP ( true ) ;
addConsoleMessage ( tr ( " NAT-PMP support [ON] " ) , QString : : fromUtf8 ( " blue " ) ) ;
} else {
enableNATPMP ( false ) ;
addConsoleMessage ( tr ( " NAT-PMP support [OFF] " ) , QString : : fromUtf8 ( " blue " ) ) ;
}
// * Session settings
session_settings sessionSettings ;
2009-12-30 13:10:06 +00:00
QString peer_id = Preferences : : getPeerID ( ) ;
if ( peer_id . size ( ) ! = 2 ) peer_id = " qB " ;
if ( peer_id = = " UT " ) {
QString version = Preferences : : getClientVersion ( ) . replace ( " . " , " " ) ;
while ( version . size ( ) < 4 )
version . append ( " 0 " ) ;
2010-03-03 17:27:25 +00:00
const QString build = Preferences : : getClientBuild ( ) ;
2009-12-30 13:10:06 +00:00
sessionSettings . user_agent = QString ( " uTorrent/ " + version + " ( " + build + " ) " ) . toStdString ( ) ;
2009-11-14 19:08:28 +00:00
} else {
2009-12-30 13:10:06 +00:00
if ( peer_id = = " AZ " ) {
QStringList version = Preferences : : getClientVersion ( ) . split ( " . " ) ;
while ( version . size ( ) < 4 )
version < < " 0 " ;
sessionSettings . user_agent = QString ( " Azureus " + version . join ( " . " ) ) . toStdString ( ) ;
} else {
2010-01-01 20:29:03 +00:00
if ( peer_id = = " KT " ) {
sessionSettings . user_agent = QString ( " KTorrent/ " + Preferences : : getClientVersion ( ) ) . toStdString ( ) ;
} else {
sessionSettings . user_agent = " qBittorrent " VERSION ;
}
2009-12-30 13:10:06 +00:00
}
2009-11-14 19:08:28 +00:00
}
2009-12-30 13:10:06 +00:00
std : : cout < < " HTTP user agent is " < < sessionSettings . user_agent < < std : : endl ;
addConsoleMessage ( tr ( " HTTP user agent is %1 " ) . arg ( misc : : toQString ( sessionSettings . user_agent ) ) ) ;
2009-11-14 19:08:28 +00:00
sessionSettings . upnp_ignore_nonrouters = true ;
sessionSettings . use_dht_as_fallback = false ;
2009-11-28 10:36:34 +00:00
// To prevent ISPs from blocking seeding
sessionSettings . lazy_bitfields = true ;
2009-11-28 15:03:27 +00:00
// Speed up exit
sessionSettings . stop_tracker_timeout = 1 ;
2009-11-19 15:50:57 +00:00
//sessionSettings.announce_to_all_trackers = true;
2009-12-02 18:19:16 +00:00
sessionSettings . auto_scrape_interval = 1200 ; // 20 minutes
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-04-09 18:24:57 +00:00
sessionSettings . announce_to_all_trackers = true ;
2009-11-19 08:43:18 +00:00
sessionSettings . announce_to_all_tiers = true ; //uTorrent behavior
2009-12-02 18:19:16 +00:00
sessionSettings . auto_scrape_min_interval = 900 ; // 15 minutes
2009-11-22 10:35:16 +00:00
# endif
2009-11-20 08:01:59 +00:00
// To keep same behavior as in qBittorrent v1.2.0
2009-11-14 19:08:28 +00:00
sessionSettings . rate_limit_ip_overhead = false ;
2009-12-17 17:37:54 +00:00
sessionSettings . cache_size = Preferences : : diskCacheSize ( ) * 64 ;
addConsoleMessage ( tr ( " Using a disk cache size of %1 MiB " ) . arg ( Preferences : : diskCacheSize ( ) ) ) ;
2009-11-14 19:08:28 +00:00
// Queueing System
if ( Preferences : : isQueueingSystemEnabled ( ) ) {
sessionSettings . active_downloads = Preferences : : getMaxActiveDownloads ( ) ;
sessionSettings . active_seeds = Preferences : : getMaxActiveUploads ( ) ;
sessionSettings . active_limit = Preferences : : getMaxActiveTorrents ( ) ;
sessionSettings . dont_count_slow_torrents = false ;
setQueueingEnabled ( true ) ;
} else {
sessionSettings . active_downloads = - 1 ;
sessionSettings . active_seeds = - 1 ;
sessionSettings . active_limit = - 1 ;
setQueueingEnabled ( false ) ;
}
2010-01-31 15:27:54 +00:00
// Outgoing ports
sessionSettings . outgoing_ports = std : : make_pair ( Preferences : : outgoingPortsMin ( ) , Preferences : : outgoingPortsMax ( ) ) ;
2009-11-14 19:08:28 +00:00
setSessionSettings ( sessionSettings ) ;
2010-01-31 15:42:24 +00:00
// Ignore limits on LAN
sessionSettings . ignore_limits_on_local_network = Preferences : : ignoreLimitsOnLAN ( ) ;
2010-01-31 15:57:07 +00:00
// Include overhead in transfer limits
sessionSettings . rate_limit_ip_overhead = Preferences : : includeOverheadInLimits ( ) ;
2009-11-14 19:08:28 +00:00
// Bittorrent
2010-03-25 19:16:43 +00:00
// * Max Half-open connections
s - > set_max_half_open_connections ( Preferences : : getMaxHalfOpenConnections ( ) ) ;
2009-11-14 19:08:28 +00:00
// * Max connections limit
setMaxConnections ( Preferences : : getMaxConnecs ( ) ) ;
// * Max connections per torrent limit
setMaxConnectionsPerTorrent ( Preferences : : getMaxConnecsPerTorrent ( ) ) ;
// * Max uploads per torrent limit
setMaxUploadsPerTorrent ( Preferences : : getMaxUploadsPerTorrent ( ) ) ;
// * DHT
if ( Preferences : : isDHTEnabled ( ) ) {
// Set DHT Port
if ( enableDHT ( true ) ) {
2009-12-29 09:59:32 +00:00
int dht_port ;
if ( Preferences : : isDHTPortSameAsBT ( ) )
2009-12-29 10:04:41 +00:00
dht_port = 0 ;
2009-12-29 09:59:32 +00:00
else
2009-11-14 19:08:28 +00:00
dht_port = Preferences : : getDHTPort ( ) ;
setDHTPort ( dht_port ) ;
2009-12-30 16:47:03 +00:00
if ( dht_port = = 0 ) dht_port = new_listenPort ;
2009-11-14 19:08:28 +00:00
addConsoleMessage ( tr ( " DHT support [ON], port: UDP/%1 " ) . arg ( dht_port ) , QString : : fromUtf8 ( " blue " ) ) ;
} else {
addConsoleMessage ( tr ( " DHT support [OFF] " ) , QString : : fromUtf8 ( " red " ) ) ;
}
} else {
enableDHT ( false ) ;
addConsoleMessage ( tr ( " DHT support [OFF] " ) , QString : : fromUtf8 ( " blue " ) ) ;
}
// * PeX
2009-12-17 18:00:21 +00:00
if ( PeXEnabled ) {
addConsoleMessage ( tr ( " PeX support [ON] " ) , QString : : fromUtf8 ( " blue " ) ) ;
} else {
addConsoleMessage ( tr ( " PeX support [OFF] " ) , QString : : fromUtf8 ( " red " ) ) ;
}
if ( PeXEnabled ! = Preferences : : isPeXEnabled ( ) ) {
addConsoleMessage ( tr ( " Restart is required to toggle PeX support " ) , QString : : fromUtf8 ( " red " ) ) ;
}
2009-11-14 19:08:28 +00:00
// * LSD
if ( Preferences : : isLSDEnabled ( ) ) {
enableLSD ( true ) ;
addConsoleMessage ( tr ( " Local Peer Discovery [ON] " ) , QString : : fromUtf8 ( " blue " ) ) ;
} else {
enableLSD ( false ) ;
addConsoleMessage ( tr ( " Local Peer Discovery support [OFF] " ) , QString : : fromUtf8 ( " blue " ) ) ;
}
// * Encryption
2010-03-03 17:27:25 +00:00
const int encryptionState = Preferences : : getEncryptionSetting ( ) ;
2009-11-14 19:08:28 +00:00
// The most secure, rc4 only so that all streams and encrypted
pe_settings encryptionSettings ;
encryptionSettings . allowed_enc_level = pe_settings : : rc4 ;
encryptionSettings . prefer_rc4 = true ;
switch ( encryptionState ) {
case 0 : //Enabled
encryptionSettings . out_enc_policy = pe_settings : : enabled ;
encryptionSettings . in_enc_policy = pe_settings : : enabled ;
addConsoleMessage ( tr ( " Encryption support [ON] " ) , QString : : fromUtf8 ( " blue " ) ) ;
break ;
case 1 : // Forced
encryptionSettings . out_enc_policy = pe_settings : : forced ;
encryptionSettings . in_enc_policy = pe_settings : : forced ;
addConsoleMessage ( tr ( " Encryption support [FORCED] " ) , QString : : fromUtf8 ( " blue " ) ) ;
break ;
default : // Disabled
encryptionSettings . out_enc_policy = pe_settings : : disabled ;
encryptionSettings . in_enc_policy = pe_settings : : disabled ;
addConsoleMessage ( tr ( " Encryption support [OFF] " ) , QString : : fromUtf8 ( " blue " ) ) ;
}
applyEncryptionSettings ( encryptionSettings ) ;
// * Desired ratio
setGlobalRatio ( Preferences : : getDesiredRatio ( ) ) ;
// * Maximum ratio
setDeleteRatio ( Preferences : : getDeleteRatio ( ) ) ;
// Ip Filter
2009-11-17 16:02:35 +00:00
FilterParserThread : : processFilterList ( s , Preferences : : bannedIPs ( ) ) ;
2009-11-14 19:08:28 +00:00
if ( Preferences : : isFilteringEnabled ( ) ) {
enableIPFilter ( Preferences : : getFilter ( ) ) ;
} else {
disableIPFilter ( ) ;
}
2009-11-18 17:46:59 +00:00
// Update Web UI
if ( Preferences : : isWebUiEnabled ( ) ) {
2010-03-03 17:27:25 +00:00
const quint16 port = Preferences : : getWebUiPort ( ) ;
const QString username = Preferences : : getWebUiUsername ( ) ;
const QString password = Preferences : : getWebUiPassword ( ) ;
2009-11-18 17:46:59 +00:00
initWebUi ( username , password , port ) ;
} else if ( httpServer ) {
delete httpServer ;
}
2009-11-14 19:08:28 +00:00
// * Proxy settings
proxy_settings proxySettings ;
2009-12-30 10:53:58 +00:00
if ( Preferences : : isPeerProxyEnabled ( ) ) {
2009-11-14 19:08:28 +00:00
qDebug ( " Enabling P2P proxy " ) ;
2009-12-30 10:53:58 +00:00
proxySettings . hostname = Preferences : : getPeerProxyIp ( ) . toStdString ( ) ;
2009-11-14 19:08:28 +00:00
qDebug ( " hostname is %s " , proxySettings . hostname . c_str ( ) ) ;
2009-12-30 10:53:58 +00:00
proxySettings . port = Preferences : : getPeerProxyPort ( ) ;
2009-11-14 19:08:28 +00:00
qDebug ( " port is %d " , proxySettings . port ) ;
2009-12-30 10:53:58 +00:00
if ( Preferences : : isPeerProxyAuthEnabled ( ) ) {
proxySettings . username = Preferences : : getPeerProxyUsername ( ) . toStdString ( ) ;
proxySettings . password = Preferences : : getPeerProxyPassword ( ) . toStdString ( ) ;
2009-11-14 19:08:28 +00:00
qDebug ( " username is %s " , proxySettings . username . c_str ( ) ) ;
qDebug ( " password is %s " , proxySettings . password . c_str ( ) ) ;
}
}
2009-12-30 10:53:58 +00:00
switch ( Preferences : : getPeerProxyType ( ) ) {
case HTTP :
qDebug ( " type: http " ) ;
proxySettings . type = proxy_settings : : http ;
break ;
case HTTP_PW :
qDebug ( " type: http_pw " ) ;
proxySettings . type = proxy_settings : : http_pw ;
break ;
case SOCKS4 :
proxySettings . type = proxy_settings : : socks4 ;
case SOCKS5 :
qDebug ( " type: socks5 " ) ;
proxySettings . type = proxy_settings : : socks5 ;
break ;
case SOCKS5_PW :
qDebug ( " type: socks5_pw " ) ;
proxySettings . type = proxy_settings : : socks5_pw ;
break ;
default :
proxySettings . type = proxy_settings : : none ;
}
setPeerProxySettings ( proxySettings ) ;
// HTTP Proxy
proxy_settings http_proxySettings ;
2010-01-08 20:15:08 +00:00
qDebug ( " HTTP Communications proxy type: %d " , Preferences : : getHTTPProxyType ( ) ) ;
2009-12-30 10:53:58 +00:00
switch ( Preferences : : getHTTPProxyType ( ) ) {
case HTTP_PW :
http_proxySettings . type = proxy_settings : : http_pw ;
http_proxySettings . username = Preferences : : getHTTPProxyUsername ( ) . toStdString ( ) ;
http_proxySettings . password = Preferences : : getHTTPProxyPassword ( ) . toStdString ( ) ;
http_proxySettings . hostname = Preferences : : getHTTPProxyIp ( ) . toStdString ( ) ;
http_proxySettings . port = Preferences : : getHTTPProxyPort ( ) ;
break ;
case HTTP :
http_proxySettings . type = proxy_settings : : http ;
http_proxySettings . hostname = Preferences : : getHTTPProxyIp ( ) . toStdString ( ) ;
http_proxySettings . port = Preferences : : getHTTPProxyPort ( ) ;
break ;
2010-01-07 20:35:12 +00:00
case SOCKS5 :
http_proxySettings . type = proxy_settings : : socks5 ;
http_proxySettings . hostname = Preferences : : getHTTPProxyIp ( ) . toStdString ( ) ;
http_proxySettings . port = Preferences : : getHTTPProxyPort ( ) ;
break ;
case SOCKS5_PW :
http_proxySettings . type = proxy_settings : : socks5_pw ;
http_proxySettings . username = Preferences : : getHTTPProxyUsername ( ) . toStdString ( ) ;
http_proxySettings . password = Preferences : : getHTTPProxyPassword ( ) . toStdString ( ) ;
http_proxySettings . hostname = Preferences : : getHTTPProxyIp ( ) . toStdString ( ) ;
http_proxySettings . port = Preferences : : getHTTPProxyPort ( ) ;
break ;
2009-12-30 10:53:58 +00:00
default :
http_proxySettings . type = proxy_settings : : none ;
}
setHTTPProxySettings ( http_proxySettings ) ;
2009-11-14 19:08:28 +00:00
qDebug ( " Session configured " ) ;
}
2009-11-20 08:01:59 +00:00
bool Bittorrent : : initWebUi ( QString username , QString password , int port ) {
2009-12-30 09:46:06 +00:00
if ( httpServer ) {
if ( httpServer - > serverPort ( ) ! = port ) {
httpServer - > close ( ) ;
}
} else {
2009-11-18 17:46:59 +00:00
httpServer = new HttpServer ( this , 3000 , this ) ;
2009-12-30 09:46:06 +00:00
}
2009-11-18 17:46:59 +00:00
httpServer - > setAuthorization ( username , password ) ;
2009-12-30 09:46:06 +00:00
bool success = true ;
if ( ! httpServer - > isListening ( ) ) {
success = httpServer - > listen ( QHostAddress : : Any , port ) ;
if ( success )
2009-12-30 09:49:56 +00:00
addConsoleMessage ( tr ( " The Web UI is listening on port %1 " ) . arg ( port ) ) ;
2009-12-30 09:46:06 +00:00
else
2010-01-02 22:20:37 +00:00
addConsoleMessage ( tr ( " Web User Interface Error - Unable to bind Web UI to port %1 " ) . arg ( port ) , " red " ) ;
2009-12-30 09:46:06 +00:00
}
return success ;
2009-11-18 17:46:59 +00:00
}
2010-01-24 11:57:15 +00:00
void Bittorrent : : useAlternativeSpeedsLimit ( bool alternative ) {
// Save new state to remember it on startup
Preferences : : setAltBandwidthEnabled ( alternative ) ;
// Apply settings to the bittorrent session
if ( alternative ) {
s - > set_download_rate_limit ( Preferences : : getAltGlobalDownloadLimit ( ) * 1024 ) ;
s - > set_upload_rate_limit ( Preferences : : getAltGlobalUploadLimit ( ) * 1024 ) ;
} else {
s - > set_download_rate_limit ( Preferences : : getGlobalDownloadLimit ( ) * 1024 ) ;
s - > set_upload_rate_limit ( Preferences : : getGlobalUploadLimit ( ) * 1024 ) ;
}
emit alternativeSpeedsModeChanged ( alternative ) ;
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : takeETASamples ( ) {
2009-11-16 20:28:58 +00:00
bool change = false ; ;
foreach ( const QString & hash , ETA_samples . keys ( ) ) {
2010-03-03 17:27:25 +00:00
const QTorrentHandle & h = getTorrentHandle ( hash ) ;
2009-11-16 20:28:58 +00:00
if ( h . is_valid ( ) & & ! h . is_paused ( ) & & ! h . is_seed ( ) ) {
QList < int > samples = ETA_samples . value ( h . hash ( ) , QList < int > ( ) ) ;
if ( samples . size ( ) > = MAX_SAMPLES )
samples . removeFirst ( ) ;
samples . append ( h . download_payload_rate ( ) ) ;
ETA_samples [ h . hash ( ) ] = samples ;
change = true ;
} else {
ETA_samples . remove ( hash ) ;
2007-06-29 15:23:15 +00:00
}
2009-11-16 20:28:58 +00:00
}
if ( ! change & & timerETA ) {
delete timerETA ;
}
}
// This algorithm was inspired from KTorrent - http://www.ktorrent.org
// Calculate the ETA using a combination of several algorithms:
// GASA: Global Average Speed Algorithm
// CSA: Current Speed Algorithm
// WINX: Window of X Algorithm
2009-11-20 08:01:59 +00:00
qlonglong Bittorrent : : getETA ( QString hash ) {
2010-03-03 17:27:25 +00:00
const QTorrentHandle & h = getTorrentHandle ( hash ) ;
2009-11-16 20:28:58 +00:00
if ( ! h . is_valid ( ) | | h . state ( ) ! = torrent_status : : downloading | | ! h . active_time ( ) )
2009-08-16 03:09:20 +00:00
return - 1 ;
2009-11-16 20:28:58 +00:00
// See if the torrent is going to be completed soon
2010-03-03 17:27:25 +00:00
const qulonglong bytes_left = h . actual_size ( ) - h . total_wanted_done ( ) ;
2009-11-16 20:28:58 +00:00
if ( h . actual_size ( ) > 10485760L ) { // Size > 10MiB
if ( h . progress ( ) > = ( float ) 0.99 & & bytes_left < 10485760L ) { // Progress>99% but less than 10MB left.
// Compute by taking samples
if ( ! ETA_samples . contains ( h . hash ( ) ) ) {
ETA_samples [ h . hash ( ) ] = QList < int > ( ) ;
}
if ( ! timerETA ) {
timerETA = new QTimer ( this ) ;
connect ( timerETA , SIGNAL ( timeout ( ) ) , this , SLOT ( takeETASamples ( ) ) ) ;
timerETA - > start ( ) ;
} else {
2010-03-03 17:27:25 +00:00
const QList < int > & samples = ETA_samples . value ( h . hash ( ) , QList < int > ( ) ) ;
const int nb_samples = samples . size ( ) ;
2009-11-16 20:28:58 +00:00
if ( nb_samples > 3 ) {
long sum_samples = 0 ;
2010-03-03 17:27:25 +00:00
foreach ( const int val , samples ) {
2009-11-16 20:28:58 +00:00
sum_samples + = val ;
}
// Use WINX
return ( qlonglong ) ( ( ( double ) bytes_left ) / ( ( ( double ) sum_samples ) / ( ( double ) nb_samples ) ) ) ;
}
}
}
2009-08-16 03:09:20 +00:00
}
2009-11-16 20:28:58 +00:00
// Normal case: Use GASA
double avg_speed = ( double ) h . all_time_download ( ) / h . active_time ( ) ;
return ( qlonglong ) floor ( ( double ) ( bytes_left ) / avg_speed ) ;
2007-06-29 15:23:15 +00:00
}
2009-11-20 08:01:59 +00:00
std : : vector < torrent_handle > Bittorrent : : getTorrents ( ) const {
2009-08-16 03:09:20 +00:00
return s - > get_torrents ( ) ;
2008-12-26 21:18:33 +00:00
}
2007-03-05 13:55:23 +00:00
// Return the torrent handle, given its hash
2009-11-20 08:01:59 +00:00
QTorrentHandle Bittorrent : : getTorrentHandle ( QString hash ) const {
2010-03-03 17:27:25 +00:00
return QTorrentHandle ( s - > find_torrent ( misc : : QStringToSha1 ( hash ) ) ) ;
2007-03-05 13:55:23 +00:00
}
2009-11-20 08:01:59 +00:00
bool Bittorrent : : hasActiveTorrents ( ) const {
2008-12-26 21:18:33 +00:00
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
2009-08-16 03:09:20 +00:00
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
2010-04-05 19:03:00 +00:00
const QTorrentHandle h ( * torrentIT ) ;
2009-11-07 19:55:33 +00:00
if ( h . is_valid ( ) & & ! h . is_paused ( ) & & ! h . is_queued ( ) )
return true ;
2009-08-16 03:09:20 +00:00
}
2009-11-07 19:55:33 +00:00
return false ;
2008-05-18 09:26:02 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : banIP ( QString ip ) {
2009-11-17 16:02:35 +00:00
FilterParserThread : : processFilterList ( s , QStringList ( ip ) ) ;
Preferences : : banIP ( ip ) ;
}
2007-03-05 13:55:23 +00:00
// Delete a torrent from the session, given its hash
// permanent = true means that the torrent will be removed from the hard-drive too
2009-11-20 08:01:59 +00:00
void Bittorrent : : deleteTorrent ( QString hash , bool delete_local_files ) {
2010-03-04 20:19:25 +00:00
qDebug ( " Deleting torrent with hash: %s " , qPrintable ( hash ) ) ;
2010-03-03 17:27:25 +00:00
const QTorrentHandle & h = getTorrentHandle ( hash ) ;
2007-08-20 06:29:18 +00:00
if ( ! h . is_valid ( ) ) {
2007-03-11 18:01:34 +00:00
qDebug ( " /! \\ Error: Invalid handle " ) ;
2007-03-05 17:35:38 +00:00
return ;
}
2010-03-03 17:27:25 +00:00
const QString & fileName = h . name ( ) ;
2007-10-19 16:00:42 +00:00
// Remove it from session
2009-11-18 12:11:15 +00:00
if ( delete_local_files )
2009-08-16 03:09:20 +00:00
s - > remove_torrent ( h . get_torrent_handle ( ) , session : : delete_files ) ;
2008-11-01 21:42:56 +00:00
else
s - > remove_torrent ( h . get_torrent_handle ( ) ) ;
2007-03-05 13:55:23 +00:00
// Remove it from torrent backup directory
2010-01-02 11:22:44 +00:00
QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2007-07-31 08:13:38 +00:00
QStringList filters ;
filters < < hash + " .* " ;
2010-03-03 17:27:25 +00:00
const QStringList & files = torrentBackup . entryList ( filters , QDir : : Files , QDir : : Unsorted ) ;
2009-01-24 19:20:09 +00:00
foreach ( const QString & file , files ) {
2007-07-31 08:13:38 +00:00
torrentBackup . remove ( file ) ;
}
2009-08-16 03:09:20 +00:00
TorrentPersistentData : : deletePersistentData ( hash ) ;
2007-07-27 13:58:12 +00:00
// Remove tracker errors
2009-11-19 15:04:43 +00:00
trackersInfos . remove ( hash ) ;
2009-11-18 12:11:15 +00:00
if ( delete_local_files )
addConsoleMessage ( tr ( " '%1' was removed from transfer list and hard disk. " , " 'xxx.avi' was removed... " ) . arg ( fileName ) ) ;
2008-09-07 11:31:29 +00:00
else
2009-11-18 12:11:15 +00:00
addConsoleMessage ( tr ( " '%1' was removed from transfer list. " , " 'xxx.avi' was removed... " ) . arg ( fileName ) ) ;
2008-05-16 07:10:50 +00:00
emit deletedTorrent ( hash ) ;
2007-03-05 13:55:23 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : pauseAllTorrents ( ) {
2009-08-16 03:09:20 +00:00
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( ! h . is_valid ( ) ) continue ;
if ( ! h . is_paused ( ) ) {
h . pause ( ) ;
emit pausedTorrent ( h ) ;
2008-12-26 21:18:33 +00:00
}
2009-08-16 03:09:20 +00:00
}
2008-05-16 07:10:50 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : resumeAllTorrents ( ) {
2009-08-16 03:09:20 +00:00
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( ! h . is_valid ( ) ) continue ;
if ( h . is_paused ( ) ) {
h . resume ( ) ;
emit resumedTorrent ( h ) ;
2008-12-26 21:18:33 +00:00
}
2009-08-16 03:09:20 +00:00
}
2008-05-16 07:10:50 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : pauseTorrent ( QString hash ) {
2009-08-16 03:09:20 +00:00
QTorrentHandle h = getTorrentHandle ( hash ) ;
if ( ! h . is_paused ( ) ) {
h . pause ( ) ;
emit pausedTorrent ( h ) ;
}
2008-12-27 10:14:16 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : resumeTorrent ( QString hash ) {
2009-08-16 03:09:20 +00:00
QTorrentHandle h = getTorrentHandle ( hash ) ;
if ( h . is_paused ( ) ) {
h . resume ( ) ;
emit resumedTorrent ( h ) ;
}
2008-12-27 10:14:16 +00:00
}
2009-11-20 08:01:59 +00:00
QTorrentHandle Bittorrent : : addMagnetUri ( QString magnet_uri , bool resumed ) {
2009-08-17 05:14:03 +00:00
QTorrentHandle h ;
2010-03-03 17:27:25 +00:00
const QString & hash = misc : : magnetUriToHash ( magnet_uri ) ;
2009-08-17 05:14:03 +00:00
if ( hash . isEmpty ( ) ) {
addConsoleMessage ( tr ( " '%1' is not a valid magnet URI. " ) . arg ( magnet_uri ) ) ;
return h ;
}
2010-04-05 19:03:00 +00:00
const QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2009-08-17 05:14:03 +00:00
if ( resumed ) {
2010-03-04 20:19:25 +00:00
qDebug ( " Resuming magnet URI: %s " , qPrintable ( hash ) ) ;
2010-01-19 21:47:53 +00:00
// Load metadata
if ( QFile : : exists ( torrentBackup . path ( ) + QDir : : separator ( ) + hash + QString ( " .torrent " ) ) )
return addTorrent ( torrentBackup . path ( ) + QDir : : separator ( ) + hash + QString ( " .torrent " ) , false , false , true ) ;
2009-08-17 05:14:03 +00:00
} else {
qDebug ( " Adding new magnet URI " ) ;
}
bool fastResume = false ;
Q_ASSERT ( magnet_uri . startsWith ( " magnet: " ) ) ;
// Check if torrent is already in download list
2010-03-04 20:19:25 +00:00
if ( s - > find_torrent ( sha1_hash ( hash . toLocal8Bit ( ) . constData ( ) ) ) . is_valid ( ) ) {
2009-08-17 05:14:03 +00:00
qDebug ( " /! \\ Torrent is already in download list " ) ;
// Update info Bar
addConsoleMessage ( tr ( " '%1' is already in download list. " , " e.g: 'xxx.avi' is already in download list. " ) . arg ( magnet_uri ) ) ;
return h ;
}
add_torrent_params p ;
//Getting fast resume data if existing
std : : vector < char > buf ;
if ( resumed ) {
2010-04-05 19:03:00 +00:00
const QString fastresume_path = torrentBackup . path ( ) + QDir : : separator ( ) + hash + QString ( " .fastresume " ) ;
2010-03-06 21:11:47 +00:00
qDebug ( " Trying to load fastresume data: %s " , qPrintable ( fastresume_path ) ) ;
if ( load_file ( fastresume_path . toLocal8Bit ( ) . constData ( ) , buf ) = = 0 ) {
2009-08-17 05:14:03 +00:00
fastResume = true ;
p . resume_data = & buf ;
qDebug ( " Successfuly loaded " ) ;
}
}
2010-04-06 09:17:33 +00:00
QString torrent_name = misc : : magnetUriToName ( magnet_uri ) ;
const QString & savePath = getSavePath ( hash , false , QString : : null , torrent_name ) ;
2010-01-04 21:16:33 +00:00
if ( ! defaultTempPath . isEmpty ( ) & & resumed & & ! TorrentPersistentData : : isSeed ( hash ) ) {
qDebug ( " addMagnetURI: Temp folder is enabled. " ) ;
2010-03-03 17:27:25 +00:00
p . save_path = defaultTempPath . toLocal8Bit ( ) . constData ( ) ;
qDebug ( " addMagnetURI: using save_path: %s " , qPrintable ( defaultTempPath ) ) ;
2010-01-04 21:16:33 +00:00
} else {
2010-03-03 17:27:25 +00:00
p . save_path = savePath . toLocal8Bit ( ) . constData ( ) ;
qDebug ( " addMagnetURI: using save_path: %s " , qPrintable ( savePath ) ) ;
2009-08-17 05:14:03 +00:00
}
2010-01-04 21:16:33 +00:00
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-01-03 14:20:31 +00:00
// Skip checking and directly start seeding (new in libtorrent v0.15)
if ( TorrentTempData : : isSeedingMode ( hash ) ) {
p . seed_mode = true ;
} else {
p . seed_mode = false ;
}
# endif
2009-08-17 05:14:03 +00:00
// Preallocate all?
if ( preAllocateAll )
p . storage_mode = storage_mode_allocate ;
else
p . storage_mode = storage_mode_sparse ;
// Start in pause
2010-01-20 08:35:36 +00:00
p . paused = true ;
2009-08-17 05:14:03 +00:00
p . duplicate_is_error = false ; // Already checked
p . auto_managed = false ; // Because it is added in paused state
2009-11-20 08:01:59 +00:00
// Adding torrent to Bittorrent session
2009-08-17 05:14:03 +00:00
try {
h = QTorrentHandle ( add_magnet_uri ( * s , magnet_uri . toStdString ( ) , p ) ) ;
} catch ( std : : exception e ) {
qDebug ( " Error: %s " , e . what ( ) ) ;
}
// Check if it worked
if ( ! h . is_valid ( ) ) {
// No need to keep on, it failed.
qDebug ( " /! \\ Error: Invalid handle " ) ;
return h ;
}
Q_ASSERT ( h . hash ( ) = = hash ) ;
2010-01-04 21:16:33 +00:00
// If temp path is enabled, move torrent
if ( ! defaultTempPath . isEmpty ( ) & & ! resumed ) {
qDebug ( " Temp folder is enabled, moving new torrent to temp folder " ) ;
h . move_storage ( defaultTempPath ) ;
}
2009-08-17 05:14:03 +00:00
// Connections limit per torrent
2009-11-17 13:11:32 +00:00
h . set_max_connections ( Preferences : : getMaxConnecsPerTorrent ( ) ) ;
2009-08-17 05:14:03 +00:00
// Uploads limit per torrent
2009-11-17 13:11:32 +00:00
h . set_max_uploads ( Preferences : : getMaxUploadsPerTorrent ( ) ) ;
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2009-11-15 10:59:11 +00:00
// Resolve countries
h . resolve_countries ( resolve_countries ) ;
2010-01-02 22:20:37 +00:00
# endif
2009-08-17 05:14:03 +00:00
// Load filtered files
2009-11-20 18:29:13 +00:00
if ( ! resumed ) {
2009-08-17 05:14:03 +00:00
// Sequential download
if ( TorrentTempData : : hasTempData ( hash ) ) {
2010-01-03 14:20:31 +00:00
qDebug ( " addMagnetURI Setting download as sequential (from tmp data) " ) ;
2009-08-17 05:14:03 +00:00
h . set_sequential_download ( TorrentTempData : : isSequential ( hash ) ) ;
}
2010-03-03 17:27:25 +00:00
const QString & label = TorrentTempData : : getLabel ( hash ) ;
2009-08-17 05:14:03 +00:00
// Save persistent data for new torrent
2010-01-19 23:43:29 +00:00
TorrentPersistentData : : saveTorrentPersistentData ( h , true ) ;
2010-01-03 14:20:31 +00:00
// Save Label
if ( ! label . isEmpty ( ) ) {
TorrentPersistentData : : saveLabel ( hash , label ) ;
}
2009-08-17 05:14:03 +00:00
// Save save_path
if ( ! defaultTempPath . isEmpty ( ) ) {
2010-03-03 17:27:25 +00:00
qDebug ( " addTorrent: Saving save_path in persistent data: %s " , qPrintable ( savePath ) ) ;
2009-08-17 05:14:03 +00:00
TorrentPersistentData : : saveSavePath ( hash , savePath ) ;
}
}
2010-01-20 18:53:11 +00:00
if ( ! fastResume & & ( ! addInPause | | ( Preferences : : useAdditionDialog ( ) ) ) ) {
2009-08-17 05:14:03 +00:00
// Start torrent because it was added in paused state
h . resume ( ) ;
}
// Send torrent addition signal
if ( fastResume )
addConsoleMessage ( tr ( " '%1' resumed. (fast resume) " , " '/home/y/xxx.torrent' was resumed. (fast resume) " ) . arg ( magnet_uri ) ) ;
else
addConsoleMessage ( tr ( " '%1' added to download list. " , " '/home/y/xxx.torrent' was added to download list. " ) . arg ( magnet_uri ) ) ;
emit addedTorrent ( h ) ;
return h ;
}
2009-11-20 08:01:59 +00:00
// Add a torrent to the Bittorrent session
QTorrentHandle Bittorrent : : addTorrent ( QString path , bool fromScanDir , QString from_url , bool resumed ) {
2007-08-20 06:29:18 +00:00
QTorrentHandle h ;
2007-03-05 13:55:23 +00:00
bool fastResume = false ;
2010-04-05 19:03:00 +00:00
const QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2010-03-03 17:27:25 +00:00
QString hash ;
2009-08-16 03:09:20 +00:00
boost : : intrusive_ptr < torrent_info > t ;
2007-03-05 13:55:23 +00:00
2010-05-30 17:51:40 +00:00
# ifdef Q_WS_WIN
2010-06-03 20:08:19 +00:00
// Windows hack
if ( ! path . endsWith ( " .torrent " ) ) {
if ( QFile : : rename ( path , path + " .torrent " ) )
path + = " .torrent " ;
}
qDebug ( " Downloading torrent at path: %s " , qPrintable ( path ) ) ;
2010-05-30 17:51:40 +00:00
# endif
2007-03-05 13:55:23 +00:00
// Checking if BT_backup Dir exists
// create it if it is not
2007-08-20 06:29:18 +00:00
if ( ! torrentBackup . exists ( ) ) {
if ( ! torrentBackup . mkpath ( torrentBackup . path ( ) ) ) {
2010-03-04 20:19:25 +00:00
std : : cerr < < " Couldn't create the directory: ' " < < qPrintable ( torrentBackup . path ( ) ) < < " ' \n " ;
2007-03-05 17:35:38 +00:00
exit ( 1 ) ;
2007-03-05 13:55:23 +00:00
}
}
// Processing torrents
2010-04-05 19:03:00 +00:00
const QString file = path . trimmed ( ) . replace ( " file:// " , " " , Qt : : CaseInsensitive ) ;
2007-08-20 06:29:18 +00:00
if ( file . isEmpty ( ) ) {
2008-12-30 00:34:41 +00:00
return h ;
2007-03-05 13:55:23 +00:00
}
2007-09-09 07:44:22 +00:00
Q_ASSERT ( ! file . startsWith ( " http:// " , Qt : : CaseInsensitive ) & & ! file . startsWith ( " https:// " , Qt : : CaseInsensitive ) & & ! file . startsWith ( " ftp:// " , Qt : : CaseInsensitive ) ) ;
2009-08-16 03:09:20 +00:00
2010-03-03 17:27:25 +00:00
qDebug ( " Adding %s to download list " , qPrintable ( file ) ) ;
2008-11-01 21:42:56 +00:00
try {
2009-08-16 03:09:20 +00:00
// Getting torrent file informations
2010-06-06 13:58:43 +00:00
t = new torrent_info ( file . toUtf8 ( ) . constData ( ) ) ;
2010-01-03 13:00:16 +00:00
if ( ! t - > is_valid ( ) )
throw std : : exception ( ) ;
2008-11-01 21:42:56 +00:00
} catch ( std : : exception & ) {
2009-08-16 03:09:20 +00:00
if ( ! from_url . isNull ( ) ) {
addConsoleMessage ( tr ( " Unable to decode torrent file: '%1' " , " e.g: Unable to decode torrent file: '/home/y/xxx.torrent' " ) . arg ( from_url ) , QString : : fromUtf8 ( " red " ) ) ;
//emit invalidTorrent(from_url);
QFile : : remove ( file ) ;
} else {
addConsoleMessage ( tr ( " Unable to decode torrent file: '%1' " , " e.g: Unable to decode torrent file: '/home/y/xxx.torrent' " ) . arg ( file ) , QString : : fromUtf8 ( " red " ) ) ;
//emit invalidTorrent(file);
}
addConsoleMessage ( tr ( " This file is either corrupted or this isn't a torrent. " ) , QString : : fromUtf8 ( " red " ) ) ;
if ( fromScanDir ) {
// Remove file
QFile : : remove ( file ) ;
}
return h ;
2008-11-01 21:42:56 +00:00
}
2009-08-16 03:09:20 +00:00
hash = misc : : toQString ( t - > info_hash ( ) ) ;
2009-08-17 05:14:03 +00:00
2010-03-03 17:27:25 +00:00
qDebug ( " -> Hash: %s " , qPrintable ( hash ) ) ;
qDebug ( " -> Name: %s " , t - > name ( ) . c_str ( ) ) ;
2008-11-01 21:42:56 +00:00
// Check if torrent is already in download list
if ( s - > find_torrent ( t - > info_hash ( ) ) . is_valid ( ) ) {
2009-08-16 03:09:20 +00:00
qDebug ( " /! \\ Torrent is already in download list " ) ;
// Update info Bar
2010-03-11 16:37:49 +00:00
if ( ! from_url . isNull ( ) ) {
// If download from url, remove temp file
QFile : : remove ( file ) ;
addConsoleMessage ( tr ( " '%1' is already in download list. " , " e.g: 'xxx.avi' is already in download list. " ) . arg ( from_url ) ) ;
//emit duplicateTorrent(from_url);
} else {
addConsoleMessage ( tr ( " '%1' is already in download list. " , " e.g: 'xxx.avi' is already in download list. " ) . arg ( file ) ) ;
//emit duplicateTorrent(file);
}
// Check if the torrent contains trackers or url seeds we don't know about
// and add them
QTorrentHandle h_ex = getTorrentHandle ( hash ) ;
if ( h_ex . is_valid ( ) ) {
std : : vector < announce_entry > old_trackers = h_ex . trackers ( ) ;
std : : vector < announce_entry > new_trackers = t - > trackers ( ) ;
bool trackers_added = false ;
for ( std : : vector < announce_entry > : : iterator it = new_trackers . begin ( ) ; it ! = new_trackers . end ( ) ; it + + ) {
std : : string tracker_url = it - > url ;
bool found = false ;
for ( std : : vector < announce_entry > : : iterator itold = old_trackers . begin ( ) ; itold ! = old_trackers . end ( ) ; itold + + ) {
if ( tracker_url = = itold - > url ) {
found = true ;
break ;
2010-02-07 14:31:45 +00:00
}
}
2010-03-11 16:37:49 +00:00
if ( found ) {
trackers_added = true ;
announce_entry entry ( tracker_url ) ;
h_ex . add_tracker ( entry ) ;
2010-02-15 18:45:39 +00:00
}
2010-03-11 16:37:49 +00:00
}
if ( trackers_added ) {
addConsoleMessage ( tr ( " Note: new trackers were added to the existing torrent. " ) ) ;
}
bool urlseeds_added = false ;
const QStringList & old_urlseeds = h_ex . url_seeds ( ) ;
std : : vector < std : : string > new_urlseeds = t - > url_seeds ( ) ;
for ( std : : vector < std : : string > : : iterator it = new_urlseeds . begin ( ) ; it ! = new_urlseeds . end ( ) ; it + + ) {
const QString & new_url = misc : : toQString ( it - > c_str ( ) ) ;
if ( ! old_urlseeds . contains ( new_url ) ) {
urlseeds_added = true ;
h_ex . add_url_seed ( new_url ) ;
2010-02-07 14:31:45 +00:00
}
}
2010-03-11 16:37:49 +00:00
if ( urlseeds_added ) {
addConsoleMessage ( tr ( " Note: new URL seeds were added to the existing torrent. " ) ) ;
}
}
if ( fromScanDir ) {
2009-08-16 03:09:20 +00:00
// Delete torrent from scan dir
QFile : : remove ( file ) ;
}
return h ;
2008-11-01 21:42:56 +00:00
}
2010-04-06 09:17:33 +00:00
QString root_folder = misc : : truncateRootFolder ( t ) ;
2008-11-01 21:42:56 +00:00
add_torrent_params p ;
//Getting fast resume data if existing
std : : vector < char > buf ;
2009-08-16 03:09:20 +00:00
if ( resumed ) {
2010-04-05 19:03:00 +00:00
const QString fastresume_path = torrentBackup . path ( ) + QDir : : separator ( ) + hash + QString ( " .fastresume " ) ;
2010-03-06 21:11:47 +00:00
qDebug ( " Trying to load fastresume data: %s " , qPrintable ( fastresume_path ) ) ;
if ( load_file ( fastresume_path . toLocal8Bit ( ) . constData ( ) , buf ) = = 0 ) {
2008-11-01 21:42:56 +00:00
fastResume = true ;
p . resume_data = & buf ;
2008-11-01 23:58:53 +00:00
qDebug ( " Successfuly loaded " ) ;
2009-08-16 03:09:20 +00:00
}
2008-11-01 21:42:56 +00:00
}
2009-08-21 07:40:57 +00:00
QString savePath ;
2009-12-06 14:41:05 +00:00
if ( ! from_url . isEmpty ( ) & & savepath_fromurl . contains ( QUrl : : fromEncoded ( from_url . toLocal8Bit ( ) ) ) ) {
2009-08-21 07:40:57 +00:00
// Enforcing the save path defined before URL download (from RSS for example)
2009-12-06 14:41:05 +00:00
savePath = savepath_fromurl . take ( QUrl : : fromEncoded ( from_url . toLocal8Bit ( ) ) ) ;
2009-08-21 07:40:57 +00:00
} else {
2010-04-06 09:17:33 +00:00
savePath = getSavePath ( hash , fromScanDir , path , root_folder ) ;
2009-08-21 07:40:57 +00:00
}
2010-01-04 21:16:33 +00:00
if ( ! defaultTempPath . isEmpty ( ) & & resumed & & ! TorrentPersistentData : : isSeed ( hash ) ) {
qDebug ( " addTorrent::Temp folder is enabled. " ) ;
2010-03-03 17:27:25 +00:00
p . save_path = defaultTempPath . toLocal8Bit ( ) . constData ( ) ;
qDebug ( " addTorrent: using save_path: %s " , qPrintable ( defaultTempPath ) ) ;
2010-01-04 21:16:33 +00:00
} else {
2010-03-03 17:27:25 +00:00
p . save_path = savePath . toLocal8Bit ( ) . constData ( ) ;
qDebug ( " addTorrent: using save_path: %s " , qPrintable ( savePath ) ) ;
2009-03-08 16:26:02 +00:00
}
2009-11-06 14:30:14 +00:00
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2009-11-06 14:30:14 +00:00
// Skip checking and directly start seeding (new in libtorrent v0.15)
if ( TorrentTempData : : isSeedingMode ( hash ) ) {
p . seed_mode = true ;
} else {
p . seed_mode = false ;
}
2009-11-18 10:29:20 +00:00
# endif
2008-11-01 21:42:56 +00:00
p . ti = t ;
// Preallocate all?
if ( preAllocateAll )
p . storage_mode = storage_mode_allocate ;
else
p . storage_mode = storage_mode_sparse ;
// Start in pause
p . paused = true ;
p . duplicate_is_error = false ; // Already checked
2008-12-27 13:04:06 +00:00
p . auto_managed = false ; // Because it is added in paused state
2009-11-20 08:01:59 +00:00
// Adding torrent to Bittorrent session
2008-12-27 21:25:58 +00:00
try {
2009-08-16 03:09:20 +00:00
h = QTorrentHandle ( s - > add_torrent ( p ) ) ;
2008-12-27 21:25:58 +00:00
} catch ( std : : exception e ) {
qDebug ( " Error: %s " , e . what ( ) ) ;
}
2008-11-01 21:42:56 +00:00
// Check if it worked
if ( ! h . is_valid ( ) ) {
2009-08-16 03:09:20 +00:00
// No need to keep on, it failed.
qDebug ( " /! \\ Error: Invalid handle " ) ;
// If download from url, remove temp file
if ( ! from_url . isNull ( ) ) QFile : : remove ( file ) ;
return h ;
2008-11-01 21:42:56 +00:00
}
2009-12-08 17:20:28 +00:00
2010-01-04 21:16:33 +00:00
// If temp path is enabled, move torrent
if ( ! defaultTempPath . isEmpty ( ) & & ! resumed ) {
qDebug ( " Temp folder is enabled, moving new torrent to temp folder " ) ;
h . move_storage ( defaultTempPath ) ;
}
2008-11-01 21:42:56 +00:00
// Connections limit per torrent
2009-11-17 13:11:32 +00:00
h . set_max_connections ( Preferences : : getMaxConnecsPerTorrent ( ) ) ;
2008-11-01 21:42:56 +00:00
// Uploads limit per torrent
2009-11-17 13:11:32 +00:00
h . set_max_uploads ( Preferences : : getMaxUploadsPerTorrent ( ) ) ;
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2009-11-15 10:59:11 +00:00
// Resolve countries
2009-11-15 12:57:25 +00:00
qDebug ( " AddTorrent: Resolve_countries: %d " , ( int ) resolve_countries ) ;
2009-11-15 10:59:11 +00:00
h . resolve_countries ( resolve_countries ) ;
2010-01-02 22:20:37 +00:00
# endif
2009-11-20 18:29:13 +00:00
if ( ! resumed ) {
2009-08-16 03:09:20 +00:00
// Sequential download
if ( TorrentTempData : : hasTempData ( hash ) ) {
qDebug ( " addTorrent: Setting download as sequential (from tmp data) " ) ;
2009-12-08 17:20:28 +00:00
h . prioritize_files ( TorrentTempData : : getFilesPriority ( hash ) ) ;
2009-08-16 03:09:20 +00:00
h . set_sequential_download ( TorrentTempData : : isSequential ( hash ) ) ;
2010-01-01 10:55:13 +00:00
// Import Files names from torrent addition dialog
2010-03-03 17:27:25 +00:00
const QStringList & files_path = TorrentTempData : : getFilesPath ( hash ) ;
2010-01-21 21:14:54 +00:00
bool force_recheck = false ;
2010-01-01 10:55:13 +00:00
if ( files_path . size ( ) = = h . num_files ( ) ) {
for ( int i = 0 ; i < h . num_files ( ) ; + + i ) {
2010-06-06 13:58:43 +00:00
QString old_path = h . files_path ( ) . at ( i ) ;
old_path = old_path . replace ( " \\ " , " / " ) ;
if ( ! QFile : : exists ( old_path ) ) {
// Remove old parent folder manually since we will
// not get a file_renamed alert
QStringList parts = old_path . split ( " / " , QString : : SkipEmptyParts ) ;
parts . removeLast ( ) ;
if ( ! parts . empty ( ) )
QDir ( ) . rmpath ( parts . join ( " / " ) ) ;
}
2010-03-03 17:27:25 +00:00
const QString & path = files_path . at ( i ) ;
2010-06-06 13:58:43 +00:00
if ( ! force_recheck & & QDir ( h . save_path ( ) ) . exists ( path ) )
2010-01-21 21:14:54 +00:00
force_recheck = true ;
2010-06-06 13:58:43 +00:00
qDebug ( " Renaming file to %s " , qPrintable ( path ) ) ;
2010-01-01 10:55:13 +00:00
h . rename_file ( i , path ) ;
}
2010-01-21 20:27:59 +00:00
// Force recheck
2010-01-21 21:14:54 +00:00
if ( force_recheck ) h . force_recheck ( ) ;
2010-01-01 10:55:13 +00:00
}
2009-08-16 03:09:20 +00:00
}
2010-03-03 17:27:25 +00:00
const QString & label = TorrentTempData : : getLabel ( hash ) ;
2009-08-16 03:09:20 +00:00
// Save persistent data for new torrent
TorrentPersistentData : : saveTorrentPersistentData ( h ) ;
2009-12-15 19:52:43 +00:00
// Save Label
2009-12-18 16:36:48 +00:00
if ( ! label . isEmpty ( ) ) {
TorrentPersistentData : : saveLabel ( hash , label ) ;
2009-12-15 19:52:43 +00:00
}
2009-08-16 03:09:20 +00:00
// Save save_path
if ( ! defaultTempPath . isEmpty ( ) ) {
2010-03-04 20:19:25 +00:00
qDebug ( " addTorrent: Saving save_path in persistent data: %s " , qPrintable ( savePath ) ) ;
2009-08-16 03:09:20 +00:00
TorrentPersistentData : : saveSavePath ( hash , savePath ) ;
}
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2009-12-18 13:36:47 +00:00
// Append .!qB to incomplete files
if ( appendqBExtension )
appendqBextensionToTorrent ( h , true ) ;
# endif
2010-01-31 13:09:46 +00:00
// Backup torrent file
2010-03-03 17:27:25 +00:00
const QString & newFile = torrentBackup . absoluteFilePath ( hash + " .torrent " ) ;
2010-01-31 13:09:46 +00:00
if ( file ! = newFile ) {
// Delete file from torrentBackup directory in case it exists because
// QFile::copy() do not overwrite
QFile : : remove ( newFile ) ;
// Copy it to torrentBackup directory
QFile : : copy ( file , newFile ) ;
}
// Copy the torrent file to the export folder
if ( torrentExport ) {
QDir exportPath ( Preferences : : getExportDir ( ) ) ;
if ( exportPath . exists ( ) | | exportPath . mkpath ( exportPath . absolutePath ( ) ) ) {
QString torrent_path = exportPath . absoluteFilePath ( h . name ( ) + " .torrent " ) ;
if ( QFile : : exists ( torrent_path ) & & misc : : sameFiles ( file , torrent_path ) ) {
// Append hash to torrent name to make it unique
torrent_path = exportPath . absoluteFilePath ( h . name ( ) + " - " + h . hash ( ) + " .torrent " ) ;
}
QFile : : copy ( file , torrent_path ) ;
//h.save_torrent_file(torrent_path);
}
}
2008-11-01 21:42:56 +00:00
}
2010-01-20 18:53:11 +00:00
if ( ! fastResume & & ( ! addInPause | | ( Preferences : : useAdditionDialog ( ) & & ! fromScanDir ) ) ) {
2009-08-16 03:09:20 +00:00
// Start torrent because it was added in paused state
h . resume ( ) ;
2008-11-01 21:42:56 +00:00
}
// If download from url, remove temp file
if ( ! from_url . isNull ( ) ) QFile : : remove ( file ) ;
// Delete from scan dir to avoid trying to download it again
if ( fromScanDir ) {
2009-08-16 03:09:20 +00:00
QFile : : remove ( file ) ;
2008-11-01 21:42:56 +00:00
}
// Send torrent addition signal
if ( ! from_url . isNull ( ) ) {
2009-08-16 03:09:20 +00:00
if ( fastResume )
addConsoleMessage ( tr ( " '%1' resumed. (fast resume) " , " '/home/y/xxx.torrent' was resumed. (fast resume) " ) . arg ( from_url ) ) ;
else
addConsoleMessage ( tr ( " '%1' added to download list. " , " '/home/y/xxx.torrent' was added to download list. " ) . arg ( from_url ) ) ;
2008-11-01 21:42:56 +00:00
} else {
2009-08-16 03:09:20 +00:00
if ( fastResume )
addConsoleMessage ( tr ( " '%1' resumed. (fast resume) " , " '/home/y/xxx.torrent' was resumed. (fast resume) " ) . arg ( file ) ) ;
else
addConsoleMessage ( tr ( " '%1' added to download list. " , " '/home/y/xxx.torrent' was added to download list. " ) . arg ( file ) ) ;
2008-01-04 20:43:28 +00:00
}
2008-12-29 18:36:54 +00:00
emit addedTorrent ( h ) ;
2008-12-30 00:34:41 +00:00
return h ;
2007-03-05 13:55:23 +00:00
}
2010-01-31 13:09:46 +00:00
void Bittorrent : : exportTorrentFiles ( QString path ) {
Q_ASSERT ( torrentExport ) ;
QDir exportDir ( path ) ;
if ( ! exportDir . exists ( ) ) {
if ( ! exportDir . mkpath ( exportDir . absolutePath ( ) ) ) {
2010-03-03 17:27:25 +00:00
std : : cerr < < " Error: Could not create torrent export directory: " < < qPrintable ( exportDir . absolutePath ( ) ) < < std : : endl ;
2010-01-31 13:09:46 +00:00
return ;
}
}
QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
std : : vector < torrent_handle > handles = s - > get_torrents ( ) ;
std : : vector < torrent_handle > : : iterator itr ;
for ( itr = handles . begin ( ) ; itr ! = handles . end ( ) ; itr + + ) {
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( * itr ) ;
2010-01-31 13:09:46 +00:00
if ( ! h . is_valid ( ) ) {
std : : cerr < < " Torrent Export: torrent is invalid, skipping... " < < std : : endl ;
continue ;
}
2010-03-03 17:27:25 +00:00
const QString & src_path = torrentBackup . absoluteFilePath ( h . hash ( ) + " .torrent " ) ;
2010-01-31 13:09:46 +00:00
if ( QFile : : exists ( src_path ) ) {
QString dst_path = exportDir . absoluteFilePath ( h . name ( ) + " .torrent " ) ;
if ( QFile : : exists ( dst_path ) ) {
if ( ! misc : : sameFiles ( src_path , dst_path ) ) {
dst_path = exportDir . absoluteFilePath ( h . name ( ) + " - " + h . hash ( ) + " .torrent " ) ;
} else {
qDebug ( " Torrent Export: Destination file exists, skipping... " ) ;
continue ;
}
}
2010-03-03 17:27:25 +00:00
qDebug ( " Export Torrent: %s -> %s " , qPrintable ( src_path ) , qPrintable ( dst_path ) ) ;
2010-01-31 13:09:46 +00:00
QFile : : copy ( src_path , dst_path ) ;
} else {
2010-03-03 17:27:25 +00:00
std : : cerr < < " Error: could not export torrent " < < qPrintable ( h . hash ( ) ) < < " , maybe it has not metadata yet. " < < std : : endl ;
2010-01-31 13:09:46 +00:00
}
}
}
2007-08-20 06:29:18 +00:00
// Set the maximum number of opened connections
2009-11-20 08:01:59 +00:00
void Bittorrent : : setMaxConnections ( int maxConnec ) {
2007-08-20 06:29:18 +00:00
s - > set_max_connections ( maxConnec ) ;
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : setMaxConnectionsPerTorrent ( int max ) {
2007-09-09 07:44:22 +00:00
// Apply this to all session torrents
std : : vector < torrent_handle > handles = s - > get_torrents ( ) ;
unsigned int nbHandles = handles . size ( ) ;
for ( unsigned int i = 0 ; i < nbHandles ; + + i ) {
2010-03-11 21:43:09 +00:00
QTorrentHandle h ( handles [ i ] ) ;
2007-09-09 07:44:22 +00:00
if ( ! h . is_valid ( ) ) {
qDebug ( " /! \\ Error: Invalid handle " ) ;
continue ;
}
h . set_max_connections ( max ) ;
}
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : setMaxUploadsPerTorrent ( int max ) {
2007-09-09 07:44:22 +00:00
// Apply this to all session torrents
std : : vector < torrent_handle > handles = s - > get_torrents ( ) ;
unsigned int nbHandles = handles . size ( ) ;
for ( unsigned int i = 0 ; i < nbHandles ; + + i ) {
2010-03-11 21:43:09 +00:00
QTorrentHandle h ( handles [ i ] ) ;
2007-09-09 07:44:22 +00:00
if ( ! h . is_valid ( ) ) {
qDebug ( " /! \\ Error: Invalid handle " ) ;
continue ;
}
h . set_max_uploads ( max ) ;
}
}
2007-03-05 13:55:23 +00:00
// Return DHT state
2009-11-20 08:01:59 +00:00
bool Bittorrent : : isDHTEnabled ( ) const {
2007-03-05 13:55:23 +00:00
return DHTEnabled ;
}
2009-11-21 15:18:09 +00:00
bool Bittorrent : : isLSDEnabled ( ) const {
return LSDEnabled ;
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : enableUPnP ( bool b ) {
2007-09-08 17:07:29 +00:00
if ( b ) {
2007-11-23 15:20:26 +00:00
if ( ! UPnPEnabled ) {
qDebug ( " Enabling UPnP " ) ;
s - > start_upnp ( ) ;
UPnPEnabled = true ;
}
2007-09-08 17:07:29 +00:00
} else {
2007-11-23 15:20:26 +00:00
if ( UPnPEnabled ) {
qDebug ( " Disabling UPnP " ) ;
s - > stop_upnp ( ) ;
UPnPEnabled = false ;
}
2007-03-05 13:55:23 +00:00
}
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : enableNATPMP ( bool b ) {
2007-09-08 17:07:29 +00:00
if ( b ) {
2007-11-23 15:20:26 +00:00
if ( ! NATPMPEnabled ) {
qDebug ( " Enabling NAT-PMP " ) ;
s - > start_natpmp ( ) ;
NATPMPEnabled = true ;
}
2007-09-08 17:07:29 +00:00
} else {
2007-11-23 15:20:26 +00:00
if ( NATPMPEnabled ) {
qDebug ( " Disabling NAT-PMP " ) ;
s - > stop_natpmp ( ) ;
NATPMPEnabled = false ;
}
2007-09-08 17:07:29 +00:00
}
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : enableLSD ( bool b ) {
2007-09-08 17:07:29 +00:00
if ( b ) {
2007-11-23 15:20:26 +00:00
if ( ! LSDEnabled ) {
qDebug ( " Enabling LSD " ) ;
s - > start_lsd ( ) ;
LSDEnabled = true ;
}
2007-09-08 17:07:29 +00:00
} else {
2007-11-23 15:20:26 +00:00
if ( LSDEnabled ) {
qDebug ( " Disabling LSD " ) ;
s - > stop_lsd ( ) ;
LSDEnabled = false ;
}
2007-09-08 17:07:29 +00:00
}
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : loadSessionState ( ) {
2010-04-05 19:32:44 +00:00
const QString state_path = misc : : cacheLocation ( ) + QDir : : separator ( ) + QString : : fromUtf8 ( " ses_state " ) ;
2010-06-05 13:43:36 +00:00
if ( ! QFile : : exists ( state_path ) ) return ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-04-06 16:52:17 +00:00
std : : vector < char > in ;
if ( load_file ( state_path . toLocal8Bit ( ) . constData ( ) , in ) = = 0 )
{
lazy_entry e ;
if ( lazy_bdecode ( & in [ 0 ] , & in [ 0 ] + in . size ( ) , e ) = = 0 )
s - > load_state ( e ) ;
}
# else
2010-03-06 21:11:47 +00:00
boost : : filesystem : : ifstream ses_state_file ( state_path . toLocal8Bit ( ) . constData ( )
2009-08-16 03:09:20 +00:00
, std : : ios_base : : binary ) ;
ses_state_file . unsetf ( std : : ios_base : : skipws ) ;
s - > load_state ( bdecode (
std : : istream_iterator < char > ( ses_state_file )
, std : : istream_iterator < char > ( ) ) ) ;
2010-04-06 16:52:17 +00:00
# endif
2008-11-02 13:19:27 +00:00
}
2009-11-20 08:01:59 +00:00
void Bittorrent : : saveSessionState ( ) {
2009-08-16 03:09:20 +00:00
qDebug ( " Saving session state to disk... " ) ;
2010-04-05 19:32:44 +00:00
const QString state_path = misc : : cacheLocation ( ) + QDir : : separator ( ) + QString : : fromUtf8 ( " ses_state " ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-04-06 16:52:17 +00:00
entry session_state ;
s - > save_state ( session_state ) ;
std : : vector < char > out ;
bencode ( std : : back_inserter ( out ) , session_state ) ;
file f ;
error_code ec ;
if ( f . open ( state_path . toLocal8Bit ( ) . data ( ) , file : : write_only , ec ) ) {
if ( ec ) {
file : : iovec_t b = { & out [ 0 ] , out . size ( ) } ;
f . writev ( 0 , & b , 1 , ec ) ;
}
}
# else
entry session_state = s - > state ( ) ;
2010-03-06 21:11:47 +00:00
boost : : filesystem : : ofstream out ( state_path . toLocal8Bit ( ) . constData ( )
2009-08-16 03:09:20 +00:00
, std : : ios_base : : binary ) ;
out . unsetf ( std : : ios_base : : skipws ) ;
bencode ( std : : ostream_iterator < char > ( out ) , session_state ) ;
2010-04-06 16:52:17 +00:00
# endif
2008-11-02 13:19:27 +00:00
}
2007-09-08 17:07:29 +00:00
// Enable DHT
2009-11-20 08:01:59 +00:00
bool Bittorrent : : enableDHT ( bool b ) {
2007-09-08 17:07:29 +00:00
if ( b ) {
if ( ! DHTEnabled ) {
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2007-09-08 17:07:29 +00:00
entry dht_state ;
2010-04-05 19:03:00 +00:00
const QString dht_state_path = misc : : cacheLocation ( ) + QDir : : separator ( ) + QString : : fromUtf8 ( " dht_state " ) ;
2008-12-15 17:28:05 +00:00
if ( QFile : : exists ( dht_state_path ) ) {
2010-03-04 20:19:25 +00:00
boost : : filesystem : : ifstream dht_state_file ( dht_state_path . toLocal8Bit ( ) . constData ( ) , std : : ios_base : : binary ) ;
2008-12-15 17:28:05 +00:00
dht_state_file . unsetf ( std : : ios_base : : skipws ) ;
try {
dht_state = bdecode ( std : : istream_iterator < char > ( dht_state_file ) , std : : istream_iterator < char > ( ) ) ;
} catch ( std : : exception & ) { }
2009-08-16 03:09:20 +00:00
}
2010-04-06 16:52:17 +00:00
# endif
2007-10-27 20:53:09 +00:00
try {
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-04-06 16:52:17 +00:00
s - > start_dht ( ) ;
# else
2009-08-16 03:09:20 +00:00
s - > start_dht ( dht_state ) ;
2010-04-06 16:52:17 +00:00
# endif
2009-08-16 03:09:20 +00:00
s - > add_dht_router ( std : : make_pair ( std : : string ( " router.bittorrent.com " ) , 6881 ) ) ;
s - > add_dht_router ( std : : make_pair ( std : : string ( " router.utorrent.com " ) , 6881 ) ) ;
s - > add_dht_router ( std : : make_pair ( std : : string ( " router.bitcomet.com " ) , 6881 ) ) ;
DHTEnabled = true ;
qDebug ( " DHT enabled " ) ;
} catch ( std : : exception e ) {
qDebug ( " Could not enable DHT, reason: %s " , e . what ( ) ) ;
return false ;
}
2007-09-08 17:07:29 +00:00
}
} else {
if ( DHTEnabled ) {
DHTEnabled = false ;
s - > stop_dht ( ) ;
qDebug ( " DHT disabled " ) ;
}
2007-03-05 13:55:23 +00:00
}
2007-10-27 20:53:09 +00:00
return true ;
2007-03-05 13:55:23 +00:00
}
2009-11-20 08:01:59 +00:00
float Bittorrent : : getRealRatio ( QString hash ) const {
2007-08-20 06:29:18 +00:00
QTorrentHandle h = getTorrentHandle ( hash ) ;
2010-03-03 17:27:25 +00:00
if ( ! h . is_valid ( ) ) {
return 0. ;
}
2010-02-17 08:10:57 +00:00
Q_ASSERT ( h . total_done ( ) > = 0 ) ;
2008-11-01 22:07:14 +00:00
Q_ASSERT ( h . all_time_upload ( ) > = 0 ) ;
2010-02-17 08:10:57 +00:00
if ( h . total_done ( ) = = 0 ) {
2009-07-12 08:21:06 +00:00
if ( h . all_time_upload ( ) = = 0 )
return 0 ;
return 101 ;
2008-11-01 22:07:14 +00:00
}
2010-02-17 08:10:57 +00:00
float ratio = ( float ) h . all_time_upload ( ) / ( float ) h . total_done ( ) ;
2007-08-20 06:29:18 +00:00
Q_ASSERT ( ratio > = 0. ) ;
2009-01-23 19:26:22 +00:00
if ( ratio > 100. )
ratio = 100. ;
2007-08-20 06:29:18 +00:00
return ratio ;
}
2010-03-20 21:21:42 +00:00
void Bittorrent : : saveTempFastResumeData ( ) {
std : : vector < torrent_handle > torrents = s - > get_torrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( ! h . is_valid ( ) | | ! h . has_metadata ( ) | | h . is_seed ( ) | | h . is_paused ( ) ) continue ;
if ( h . state ( ) = = torrent_status : : checking_files | | h . state ( ) = = torrent_status : : queued_for_checking ) continue ;
qDebug ( " Saving fastresume data for %s " , qPrintable ( h . name ( ) ) ) ;
h . save_resume_data ( ) ;
}
}
2007-11-22 21:50:22 +00:00
// Only save fast resume data for unfinished and unpaused torrents (Optimization)
// Called periodically and on exit
2009-11-20 08:01:59 +00:00
void Bittorrent : : saveFastResumeData ( ) {
2008-11-01 23:58:53 +00:00
// Stop listening for alerts
2010-03-20 21:21:42 +00:00
resumeDataTimer . stop ( ) ;
2008-11-01 23:58:53 +00:00
timerAlerts - > stop ( ) ;
int num_resume_data = 0 ;
2008-12-29 21:10:31 +00:00
// Pause session
2008-11-01 23:58:53 +00:00
s - > pause ( ) ;
std : : vector < torrent_handle > torrents = s - > get_torrents ( ) ;
2008-12-29 21:10:31 +00:00
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
2009-01-05 21:19:55 +00:00
if ( ! h . is_valid ( ) | | ! h . has_metadata ( ) ) continue ;
if ( isQueueingEnabled ( ) )
2009-08-16 03:09:20 +00:00
TorrentPersistentData : : savePriority ( h ) ;
2009-11-20 18:36:34 +00:00
// Actually with should save fast resume data for paused files too
//if(h.is_paused()) continue;
2008-11-02 00:04:38 +00:00
if ( h . state ( ) = = torrent_status : : checking_files | | h . state ( ) = = torrent_status : : queued_for_checking ) continue ;
2008-11-01 23:58:53 +00:00
h . save_resume_data ( ) ;
+ + num_resume_data ;
}
2008-12-29 21:10:31 +00:00
while ( num_resume_data > 0 ) {
2009-08-16 03:09:20 +00:00
alert const * a = s - > wait_for_alert ( seconds ( 30 ) ) ;
if ( a = = 0 ) {
std : : cerr < < " aborting with " < < num_resume_data < < " outstanding "
" torrents to save resume data for " < < std : : endl ;
break ;
}
// Saving fastresume data can fail
save_resume_data_failed_alert const * rda = dynamic_cast < save_resume_data_failed_alert const * > ( a ) ;
if ( rda ) {
2008-11-01 23:58:53 +00:00
- - num_resume_data ;
2009-08-16 03:09:20 +00:00
s - > pop_alert ( ) ;
2009-11-28 13:50:16 +00:00
try {
// Remove torrent from session
s - > remove_torrent ( rda - > handle ) ;
} catch ( libtorrent : : libtorrent_exception ) { }
2009-08-16 03:09:20 +00:00
continue ;
}
save_resume_data_alert const * rd = dynamic_cast < save_resume_data_alert const * > ( a ) ;
if ( ! rd ) {
2008-11-01 23:58:53 +00:00
s - > pop_alert ( ) ;
2009-08-16 03:09:20 +00:00
continue ;
}
// Saving fast resume data was successful
- - num_resume_data ;
if ( ! rd - > resume_data ) continue ;
2010-01-02 11:22:44 +00:00
QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( rd - > handle ) ;
2009-11-26 11:59:40 +00:00
if ( ! h . is_valid ( ) ) continue ;
2009-08-16 03:09:20 +00:00
// Remove old fastresume file if it exists
QFile : : remove ( torrentBackup . path ( ) + QDir : : separator ( ) + h . hash ( ) + " .fastresume " ) ;
2010-04-05 19:03:00 +00:00
const QString file = h . hash ( ) + " .fastresume " ;
2010-03-03 17:27:25 +00:00
boost : : filesystem : : ofstream out ( fs : : path ( torrentBackup . path ( ) . toLocal8Bit ( ) . constData ( ) ) / file . toLocal8Bit ( ) . constData ( ) , std : : ios_base : : binary ) ;
2009-08-16 03:09:20 +00:00
out . unsetf ( std : : ios_base : : skipws ) ;
bencode ( std : : ostream_iterator < char > ( out ) , * rd - > resume_data ) ;
// Remove torrent from session
s - > remove_torrent ( rd - > handle ) ;
s - > pop_alert ( ) ;
2008-08-26 06:39:57 +00:00
}
2007-03-05 13:55:23 +00:00
}
2009-11-20 08:01:59 +00:00
QStringList Bittorrent : : getConsoleMessages ( ) const {
2008-09-07 11:31:29 +00:00
return consoleMessages ;
}
2009-11-20 08:01:59 +00:00
QStringList Bittorrent : : getPeerBanMessages ( ) const {
2008-09-07 11:31:29 +00:00
return peerBanMessages ;
}
2010-01-02 22:20:37 +00:00
# ifdef DISABLE_GUI
void Bittorrent : : addConsoleMessage ( QString msg , QString ) {
# else
2010-01-02 23:03:46 +00:00
void Bittorrent : : addConsoleMessage ( QString msg , QColor color ) {
if ( consoleMessages . size ( ) > 100 ) {
consoleMessages . removeFirst ( ) ;
}
2010-06-03 21:08:28 +00:00
# ifdef Q_WS_WIN
msg = msg . replace ( " / " , " \\ " ) ;
# endif
2010-01-02 23:03:46 +00:00
consoleMessages . append ( QString : : fromUtf8 ( " <font color='grey'> " ) + QDateTime : : currentDateTime ( ) . toString ( QString : : fromUtf8 ( " dd/MM/yyyy hh:mm:ss " ) ) + QString : : fromUtf8 ( " </font> - <font color=' " ) + color . name ( ) + QString : : fromUtf8 ( " '><i> " ) + msg + QString : : fromUtf8 ( " </i></font> " ) ) ;
2010-01-02 22:20:37 +00:00
# endif
2010-01-02 23:03:46 +00:00
emit newConsoleMessage ( QDateTime : : currentDateTime ( ) . toString ( " dd/MM/yyyy hh:mm:ss " ) + " - " + msg ) ;
}
2008-09-07 11:31:29 +00:00
2010-01-02 23:03:46 +00:00
void Bittorrent : : addPeerBanMessage ( QString ip , bool from_ipfilter ) {
if ( peerBanMessages . size ( ) > 100 ) {
peerBanMessages . removeFirst ( ) ;
}
if ( from_ipfilter )
peerBanMessages . append ( QString : : fromUtf8 ( " <font color='grey'> " ) + QDateTime : : currentDateTime ( ) . toString ( QString : : fromUtf8 ( " dd/MM/yyyy hh:mm:ss " ) ) + QString : : fromUtf8 ( " </font> - " ) + tr ( " <font color='red'>%1</font> <i>was blocked due to your IP filter</i> " , " x.y.z.w was blocked " ) . arg ( ip ) ) ;
else
peerBanMessages . append ( QString : : fromUtf8 ( " <font color='grey'> " ) + QDateTime : : currentDateTime ( ) . toString ( QString : : fromUtf8 ( " dd/MM/yyyy hh:mm:ss " ) ) + QString : : fromUtf8 ( " </font> - " ) + tr ( " <font color='red'>%1</font> <i>was banned due to corrupt pieces</i> " , " x.y.z.w was banned " ) . arg ( ip ) ) ;
2008-09-07 11:31:29 +00:00
}
2010-01-02 23:03:46 +00:00
bool Bittorrent : : isFilePreviewPossible ( QString hash ) const {
// See if there are supported files in the torrent
2010-03-03 17:27:25 +00:00
const QTorrentHandle & h = getTorrentHandle ( hash ) ;
2010-01-02 23:03:46 +00:00
if ( ! h . is_valid ( ) ) {
qDebug ( " /! \\ Error: Invalid handle " ) ;
return false ;
}
2010-03-03 17:27:25 +00:00
const unsigned int nbFiles = h . num_files ( ) ;
2010-01-02 23:03:46 +00:00
for ( unsigned int i = 0 ; i < nbFiles ; + + i ) {
2010-03-04 21:10:46 +00:00
QString extension = h . file_at ( i ) . split ( ' . ' ) . last ( ) ;
2010-01-02 23:03:46 +00:00
if ( misc : : isPreviewable ( extension ) )
return true ;
}
2007-03-05 17:35:38 +00:00
return false ;
2007-03-05 13:55:23 +00:00
}
2010-01-02 23:03:46 +00:00
void Bittorrent : : addTorrentsFromScanFolder ( QStringList & pathList ) {
foreach ( const QString & file , pathList ) {
2010-02-28 15:15:00 +00:00
qDebug ( " File %s added " , qPrintable ( file ) ) ;
2010-01-02 23:03:46 +00:00
try {
2010-06-06 13:58:43 +00:00
torrent_info t ( file . toUtf8 ( ) . constData ( ) ) ;
2010-03-03 17:27:25 +00:00
if ( t . is_valid ( ) )
addTorrent ( file , true ) ;
2010-01-02 23:03:46 +00:00
} catch ( std : : exception & ) {
2010-03-03 17:27:25 +00:00
qDebug ( " Ignoring incomplete torrent file: %s " , qPrintable ( file ) ) ;
2010-01-02 23:03:46 +00:00
}
2009-08-16 03:09:20 +00:00
}
2007-03-05 13:55:23 +00:00
}
2010-01-02 23:03:46 +00:00
QString Bittorrent : : getDefaultSavePath ( ) const {
return defaultSavePath ;
}
2009-08-21 07:40:57 +00:00
2010-01-02 23:03:46 +00:00
bool Bittorrent : : useTemporaryFolder ( ) const {
return ! defaultTempPath . isEmpty ( ) ;
}
2009-08-16 03:09:20 +00:00
2010-01-02 23:03:46 +00:00
void Bittorrent : : setDefaultTempPath ( QString temppath ) {
if ( defaultTempPath = = temppath )
return ;
if ( temppath . isEmpty ( ) ) {
// Disabling temp dir
// Moving all torrents to their destination folder
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( ! h . is_valid ( ) ) continue ;
h . move_storage ( getSavePath ( h . hash ( ) ) ) ;
}
} else {
// Moving all downloading torrents to temporary save path
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
if ( ! h . is_valid ( ) ) continue ;
if ( ! h . is_seed ( ) )
h . move_storage ( temppath ) ;
}
2009-03-08 16:26:02 +00:00
}
2010-01-02 23:03:46 +00:00
defaultTempPath = temppath ;
2009-03-08 16:26:02 +00:00
}
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-01-02 23:03:46 +00:00
void Bittorrent : : appendqBextensionToTorrent ( QTorrentHandle h , bool append ) {
if ( ! h . is_valid ( ) | | ! h . has_metadata ( ) ) return ;
std : : vector < size_type > fp ;
h . file_progress ( fp ) ;
for ( int i = 0 ; i < h . num_files ( ) ; + + i ) {
if ( append ) {
2010-03-03 17:27:25 +00:00
const qulonglong file_size = h . filesize_at ( i ) ;
2010-01-02 23:03:46 +00:00
if ( file_size > 0 & & ( fp [ i ] / ( double ) file_size ) < 1. ) {
2010-06-03 20:08:19 +00:00
const QString & name = misc : : toQStringU ( h . get_torrent_info ( ) . file_at ( i ) . path . string ( ) ) ;
2010-01-02 23:03:46 +00:00
if ( ! name . endsWith ( " .!qB " ) ) {
2010-04-05 19:32:44 +00:00
const QString new_name = name + " .!qB " ;
2010-03-13 13:40:26 +00:00
qDebug ( " Renaming %s to %s " , qPrintable ( name ) , qPrintable ( new_name ) ) ;
h . rename_file ( i , new_name ) ;
2010-01-02 23:03:46 +00:00
}
}
} else {
2010-06-03 20:08:19 +00:00
QString name = misc : : toQStringU ( h . get_torrent_info ( ) . file_at ( i ) . path . string ( ) ) ;
2010-01-02 23:03:46 +00:00
if ( name . endsWith ( " .!qB " ) ) {
2010-03-13 13:40:26 +00:00
const QString old_name = name ;
2010-01-02 23:03:46 +00:00
name . chop ( 4 ) ;
2010-03-13 13:40:26 +00:00
qDebug ( " Renaming %s to %s " , qPrintable ( old_name ) , qPrintable ( name ) ) ;
2010-01-02 23:03:46 +00:00
h . rename_file ( i , name ) ;
2009-12-18 13:36:47 +00:00
}
}
}
}
# endif
2010-01-02 23:03:46 +00:00
void Bittorrent : : changeLabelInTorrentSavePath ( QTorrentHandle h , QString old_label , QString new_label ) {
if ( ! h . is_valid ( ) ) return ;
if ( ! appendLabelToSavePath ) return ;
2010-04-10 13:58:36 +00:00
QString old_save_path = TorrentPersistentData : : getSavePath ( h . hash ( ) ) ;
if ( ! old_save_path . startsWith ( defaultSavePath ) ) return ;
2010-04-10 14:17:26 +00:00
QString new_save_path = misc : : updateLabelInSavePath ( defaultSavePath , old_save_path , old_label , new_label ) ;
if ( new_save_path ! = old_save_path ) {
// Move storage
qDebug ( " Moving storage to %s " , qPrintable ( new_save_path ) ) ;
QDir ( ) . mkpath ( new_save_path ) ;
h . move_storage ( new_save_path ) ;
2010-01-05 18:11:52 +00:00
}
2009-12-17 20:28:30 +00:00
}
2010-01-02 23:03:46 +00:00
void Bittorrent : : appendLabelToTorrentSavePath ( QTorrentHandle h ) {
if ( ! h . is_valid ( ) ) return ;
2010-03-03 17:27:25 +00:00
const QString & label = TorrentPersistentData : : getLabel ( h . hash ( ) ) ;
2010-01-02 23:03:46 +00:00
if ( label . isEmpty ( ) ) return ;
// Current save path
2010-04-10 14:17:26 +00:00
QString old_save_path = TorrentPersistentData : : getSavePath ( h . hash ( ) ) ;
QString new_save_path = misc : : updateLabelInSavePath ( defaultSavePath , old_save_path , " " , label ) ;
if ( old_save_path ! = new_save_path ) {
// Move storage
2010-04-10 14:50:11 +00:00
QDir ( ) . mkpath ( new_save_path ) ;
2010-04-10 14:17:26 +00:00
h . move_storage ( new_save_path ) ;
2010-01-02 23:03:46 +00:00
}
}
void Bittorrent : : setAppendLabelToSavePath ( bool append ) {
if ( appendLabelToSavePath ! = append ) {
appendLabelToSavePath = ! appendLabelToSavePath ;
if ( appendLabelToSavePath ) {
// Move torrents storage to sub folder with label name
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
appendLabelToTorrentSavePath ( h ) ;
}
2009-12-17 20:28:30 +00:00
}
}
}
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-01-02 23:03:46 +00:00
void Bittorrent : : setAppendqBExtension ( bool append ) {
if ( appendqBExtension ! = append ) {
appendqBExtension = ! appendqBExtension ;
// append or remove .!qB extension for incomplete files
std : : vector < torrent_handle > torrents = getTorrents ( ) ;
std : : vector < torrent_handle > : : iterator torrentIT ;
for ( torrentIT = torrents . begin ( ) ; torrentIT ! = torrents . end ( ) ; torrentIT + + ) {
QTorrentHandle h = QTorrentHandle ( * torrentIT ) ;
appendqBextensionToTorrent ( h , appendqBExtension ) ;
}
2009-12-18 13:36:47 +00:00
}
}
# endif
2010-01-02 23:03:46 +00:00
// Set the ports range in which is chosen the port the Bittorrent
// session will listen to
void Bittorrent : : setListeningPort ( int port ) {
std : : pair < int , int > ports ( port , port ) ;
2010-05-11 15:53:14 +00:00
const QString & iface_name = Preferences : : getNetworkInterface ( ) ;
if ( iface_name . isEmpty ( ) ) {
s - > listen_on ( ports ) ;
return ;
}
QNetworkInterface network_iface = QNetworkInterface : : interfaceFromName ( iface_name ) ;
if ( ! network_iface . isValid ( ) ) {
s - > listen_on ( ports ) ;
return ;
}
QString ip = " 127.0.0.1 " ;
if ( ! network_iface . addressEntries ( ) . isEmpty ( ) ) {
ip = network_iface . addressEntries ( ) . first ( ) . ip ( ) . toString ( ) ;
}
qDebug ( " Listening on interface %s with ip %s " , qPrintable ( iface_name ) , qPrintable ( ip ) ) ;
s - > listen_on ( ports , ip . toLocal8Bit ( ) . constData ( ) ) ;
2010-01-02 23:03:46 +00:00
}
2007-03-05 13:55:23 +00:00
2010-01-02 23:03:46 +00:00
// Set download rate limit
// -1 to disable
void Bittorrent : : setDownloadRateLimit ( long rate ) {
qDebug ( " Setting a global download rate limit at %ld " , rate ) ;
s - > set_download_rate_limit ( rate ) ;
}
2007-07-14 10:50:38 +00:00
2010-01-02 23:03:46 +00:00
session * Bittorrent : : getSession ( ) const {
return s ;
}
2007-03-05 13:55:23 +00:00
2010-01-02 23:03:46 +00:00
// Set upload rate limit
// -1 to disable
void Bittorrent : : setUploadRateLimit ( long rate ) {
qDebug ( " set upload_limit to %fkb/s " , rate / 1024. ) ;
s - > set_upload_rate_limit ( rate ) ;
2007-09-16 14:16:42 +00:00
}
2010-01-02 23:03:46 +00:00
// libtorrent allow to adjust ratio for each torrent
// This function will apply to same ratio to all torrents
void Bittorrent : : setGlobalRatio ( float ratio ) {
if ( ratio ! = - 1 & & ratio < 1. ) ratio = 1. ;
if ( ratio = = - 1 ) {
// 0 means unlimited for libtorrent
ratio = 0 ;
}
std : : vector < torrent_handle > handles = s - > get_torrents ( ) ;
2010-03-03 17:27:25 +00:00
std : : vector < torrent_handle > : : iterator it ;
for ( it = handles . begin ( ) ; it ! = handles . end ( ) ; it + + ) {
QTorrentHandle h ( * it ) ;
2010-01-02 23:03:46 +00:00
if ( ! h . is_valid ( ) ) {
qDebug ( " /! \\ Error: Invalid handle " ) ;
continue ;
}
h . set_ratio ( ratio ) ;
2007-03-11 18:01:34 +00:00
}
2007-03-05 13:55:23 +00:00
}
2007-03-05 17:35:38 +00:00
2010-01-02 23:03:46 +00:00
// Torrents will a ratio superior to the given value will
// be automatically deleted
void Bittorrent : : setDeleteRatio ( float ratio ) {
if ( ratio ! = - 1 & & ratio < 1. ) ratio = 1. ;
if ( ratio_limit = = - 1 & & ratio ! = - 1 ) {
Q_ASSERT ( ! BigRatioTimer ) ;
BigRatioTimer = new QTimer ( this ) ;
connect ( BigRatioTimer , SIGNAL ( timeout ( ) ) , this , SLOT ( deleteBigRatios ( ) ) ) ;
BigRatioTimer - > start ( 5000 ) ;
} else {
if ( ratio_limit ! = - 1 & & ratio = = - 1 ) {
delete BigRatioTimer ;
}
}
if ( ratio_limit ! = ratio ) {
ratio_limit = ratio ;
qDebug ( " * Set deleteRatio to %.1f " , ratio_limit ) ;
deleteBigRatios ( ) ;
2007-11-24 21:55:19 +00:00
}
}
2007-09-09 09:09:24 +00:00
2010-01-20 09:54:51 +00:00
// Set DHT port (>= 1 or 0 if same as BT)
2010-01-02 23:03:46 +00:00
void Bittorrent : : setDHTPort ( int dht_port ) {
2010-01-20 09:54:51 +00:00
if ( dht_port > = 0 ) {
2010-01-02 23:03:46 +00:00
if ( dht_port = = current_dht_port ) return ;
struct dht_settings DHTSettings ;
DHTSettings . service_port = dht_port ;
s - > set_dht_settings ( DHTSettings ) ;
current_dht_port = dht_port ;
qDebug ( " Set DHT Port to %d " , dht_port ) ;
}
2007-03-05 17:35:38 +00:00
}
2010-01-02 23:03:46 +00:00
// Enable IP Filtering
void Bittorrent : : enableIPFilter ( QString filter ) {
qDebug ( " Enabling IPFiler " ) ;
if ( ! filterParser ) {
filterParser = new FilterParserThread ( this , s ) ;
}
if ( filterPath . isEmpty ( ) | | filterPath ! = filter ) {
filterPath = filter ;
filterParser - > processFilterFile ( filter ) ;
}
2008-05-17 20:32:03 +00:00
}
2007-03-05 17:35:38 +00:00
2010-01-02 23:03:46 +00:00
// Disable IP Filtering
void Bittorrent : : disableIPFilter ( ) {
qDebug ( " Disabling IPFilter " ) ;
s - > set_ip_filter ( ip_filter ( ) ) ;
if ( filterParser ) {
delete filterParser ;
}
filterPath = " " ;
2008-05-17 20:32:03 +00:00
}
2007-03-05 17:35:38 +00:00
2010-01-02 23:03:46 +00:00
// Set BT session settings (user_agent)
2010-03-03 17:27:25 +00:00
void Bittorrent : : setSessionSettings ( const session_settings & sessionSettings ) {
2010-01-02 23:03:46 +00:00
qDebug ( " Set session settings " ) ;
s - > set_settings ( sessionSettings ) ;
}
2007-03-05 17:35:38 +00:00
2010-01-02 23:03:46 +00:00
// Set Proxy
2010-03-03 17:27:25 +00:00
void Bittorrent : : setPeerProxySettings ( const proxy_settings & proxySettings ) {
2010-01-02 23:03:46 +00:00
qDebug ( " Set Peer Proxy settings " ) ;
s - > set_peer_proxy ( proxySettings ) ;
s - > set_dht_proxy ( proxySettings ) ;
}
2009-12-30 10:53:58 +00:00
2010-03-03 17:27:25 +00:00
void Bittorrent : : setHTTPProxySettings ( const proxy_settings & proxySettings ) {
2010-01-02 23:03:46 +00:00
s - > set_tracker_proxy ( proxySettings ) ;
s - > set_web_seed_proxy ( proxySettings ) ;
QString proxy_str ;
switch ( proxySettings . type ) {
case proxy_settings : : http_pw :
2010-01-08 20:15:08 +00:00
proxy_str = " http:// " + misc : : toQString ( proxySettings . username ) + " : " + misc : : toQString ( proxySettings . password ) + " @ " + misc : : toQString ( proxySettings . hostname ) + " : " + QString : : number ( proxySettings . port ) ;
break ;
case proxy_settings : : http :
2010-01-02 23:03:46 +00:00
proxy_str = " http:// " + misc : : toQString ( proxySettings . hostname ) + " : " + QString : : number ( proxySettings . port ) ;
2010-01-08 20:15:08 +00:00
break ;
case proxy_settings : : socks5 :
proxy_str = misc : : toQString ( proxySettings . hostname ) + " : " + QString : : number ( proxySettings . port ) ;
break ;
case proxy_settings : : socks5_pw :
proxy_str = misc : : toQString ( proxySettings . username ) + " : " + misc : : toQString ( proxySettings . password ) + " @ " + misc : : toQString ( proxySettings . hostname ) + " : " + QString : : number ( proxySettings . port ) ;
break ;
2010-01-02 23:03:46 +00:00
default :
2010-01-08 20:15:08 +00:00
qDebug ( " Disabling HTTP communications proxy " ) ;
2009-12-30 10:53:58 +00:00
# ifdef Q_WS_WIN
2010-06-06 10:10:45 +00:00
# ifdef MINGW
2010-01-02 23:03:46 +00:00
putenv ( " http_proxy= " ) ;
2010-01-19 21:47:53 +00:00
putenv ( " sock_proxy= " ) ;
2010-06-06 10:10:45 +00:00
# else
SetEnvironmentVariableA ( " http_proxy " , " " ) ;
SetEnvironmentVariableA ( " sock_proxy " , " " ) ;
# endif
2009-12-30 10:53:58 +00:00
# else
2010-01-02 23:03:46 +00:00
unsetenv ( " http_proxy " ) ;
2010-01-08 20:15:08 +00:00
unsetenv ( " sock_proxy " ) ;
2009-12-30 10:53:58 +00:00
# endif
2010-01-02 23:03:46 +00:00
return ;
}
// We need this for urllib in search engine plugins
2009-12-30 10:53:58 +00:00
# ifdef Q_WS_WIN
2010-06-06 10:10:45 +00:00
QString type_str ;
2010-01-08 20:15:08 +00:00
if ( proxySettings . type = = proxy_settings : : socks5 | | proxySettings . type = = proxy_settings : : socks5_pw )
2010-06-06 10:10:45 +00:00
type_str = " sock_proxy " ;
2010-01-08 20:15:08 +00:00
else
2010-06-06 10:10:45 +00:00
type_str = " http_proxy " ;
# ifdef MINGW
QString tmp = type_str + " = " + proxy_str ;
putenv ( tmp . toLocal8Bit ( ) . constData ( ) ) ;
# else
SetEnvironmentVariableA ( type_str . toLocal8Bit ( ) . constData ( ) , proxy_str . toLocal8Bit ( ) . constData ( ) ) ;
# endif
2009-12-30 10:53:58 +00:00
# else
2010-03-03 17:27:25 +00:00
qDebug ( " HTTP communications proxy string: %s " , qPrintable ( proxy_str ) ) ;
2010-01-08 20:15:08 +00:00
if ( proxySettings . type = = proxy_settings : : socks5 | | proxySettings . type = = proxy_settings : : socks5_pw )
2010-03-03 17:27:25 +00:00
setenv ( " sock_proxy " , proxy_str . toLocal8Bit ( ) . constData ( ) , 1 ) ;
2010-01-08 20:15:08 +00:00
else
2010-03-03 17:27:25 +00:00
setenv ( " http_proxy " , proxy_str . toLocal8Bit ( ) . constData ( ) , 1 ) ;
2009-12-30 10:53:58 +00:00
# endif
2010-01-02 23:03:46 +00:00
}
2007-05-02 13:52:29 +00:00
2010-03-20 11:30:11 +00:00
void Bittorrent : : recursiveTorrentDownload ( const QTorrentHandle & h ) {
for ( int i = 0 ; i < h . get_torrent_info ( ) . num_files ( ) ; + + i ) {
2010-06-06 13:58:43 +00:00
const QString & torrent_relpath = misc : : toQStringU ( h . get_torrent_info ( ) . file_at ( i ) . path . string ( ) ) ;
2010-03-20 11:30:11 +00:00
if ( torrent_relpath . endsWith ( " .torrent " ) ) {
addConsoleMessage ( tr ( " Recursive download of file %1 embedded in torrent %2 " , " Recursive download of test.torrent embedded in torrent test2 " ) . arg ( torrent_relpath ) . arg ( h . name ( ) ) ) ;
2010-04-05 19:03:00 +00:00
const QString torrent_fullpath = h . save_path ( ) + QDir : : separator ( ) + torrent_relpath ;
2010-03-20 11:30:11 +00:00
try {
2010-06-06 13:58:43 +00:00
boost : : intrusive_ptr < torrent_info > t = new torrent_info ( torrent_fullpath . toUtf8 ( ) . constData ( ) ) ;
2010-03-20 11:30:11 +00:00
const QString & sub_hash = misc : : toQString ( t - > info_hash ( ) ) ;
// Passing the save path along to the sub torrent file
TorrentTempData : : setSavePath ( sub_hash , h . save_path ( ) ) ;
addTorrent ( torrent_fullpath ) ;
} catch ( std : : exception & ) {
qDebug ( " Caught error loading torrent " ) ;
addConsoleMessage ( tr ( " Unable to decode %1 torrent file. " ) . arg ( torrent_fullpath ) , QString : : fromUtf8 ( " red " ) ) ;
}
}
}
}
2010-01-02 23:03:46 +00:00
// Read alerts sent by the Bittorrent session
void Bittorrent : : readAlerts ( ) {
// look at session alerts and display some infos
std : : auto_ptr < alert > a = s - > pop_alert ( ) ;
while ( a . get ( ) ) {
if ( torrent_finished_alert * p = dynamic_cast < torrent_finished_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
if ( h . is_valid ( ) ) {
emit finishedTorrent ( h ) ;
2010-03-03 17:27:25 +00:00
const QString & hash = h . hash ( ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-01-31 13:09:46 +00:00
// Remove .!qB extension if necessary
2010-01-26 08:02:15 +00:00
if ( appendqBExtension )
appendqBextensionToTorrent ( h , false ) ;
# endif
2010-01-02 23:03:46 +00:00
// Move to download directory if necessary
if ( ! defaultTempPath . isEmpty ( ) ) {
// Check if directory is different
2010-04-05 19:03:00 +00:00
const QDir current_dir ( h . save_path ( ) ) ;
const QDir save_dir ( getSavePath ( hash ) ) ;
2010-01-02 23:03:46 +00:00
if ( current_dir ! = save_dir ) {
h . move_storage ( save_dir . path ( ) ) ;
}
2009-03-08 16:26:02 +00:00
}
2010-03-03 17:27:25 +00:00
const bool was_already_seeded = TorrentPersistentData : : isSeed ( hash ) ;
2010-02-04 19:16:04 +00:00
if ( ! was_already_seeded ) {
2010-02-04 17:19:00 +00:00
h . save_resume_data ( ) ;
2010-03-20 11:30:11 +00:00
qDebug ( " Checking if the torrent contains torrent files to download " ) ;
2010-02-04 19:16:04 +00:00
// Check if there are torrent files inside
for ( int i = 0 ; i < h . get_torrent_info ( ) . num_files ( ) ; + + i ) {
2010-06-03 20:08:19 +00:00
const QString & torrent_relpath = misc : : toQStringU ( h . get_torrent_info ( ) . file_at ( i ) . path . string ( ) ) ;
2010-02-04 19:16:04 +00:00
if ( torrent_relpath . endsWith ( " .torrent " ) ) {
2010-03-20 11:30:11 +00:00
qDebug ( " Found possible recursive torrent download. " ) ;
2010-04-05 19:03:00 +00:00
const QString torrent_fullpath = h . save_path ( ) + QDir : : separator ( ) + torrent_relpath ;
2010-03-20 11:30:11 +00:00
qDebug ( " Full subtorrent path is %s " , qPrintable ( torrent_fullpath ) ) ;
2010-02-04 19:16:04 +00:00
try {
2010-06-06 13:58:43 +00:00
boost : : intrusive_ptr < torrent_info > t = new torrent_info ( torrent_fullpath . toUtf8 ( ) . constData ( ) ) ;
2010-03-20 11:30:11 +00:00
if ( t - > is_valid ( ) ) {
qDebug ( " emitting recursiveTorrentDownloadPossible() " ) ;
emit recursiveTorrentDownloadPossible ( h ) ;
break ;
}
2010-02-04 19:16:04 +00:00
} catch ( std : : exception & ) {
qDebug ( " Caught error loading torrent " ) ;
addConsoleMessage ( tr ( " Unable to decode %1 torrent file. " ) . arg ( torrent_fullpath ) , QString : : fromUtf8 ( " red " ) ) ;
}
2010-01-02 23:03:46 +00:00
}
2009-08-31 08:51:55 +00:00
}
2009-08-21 11:26:58 +00:00
}
2010-01-31 16:14:56 +00:00
// Recheck if the user asked to
2010-02-04 17:19:00 +00:00
if ( Preferences : : recheckTorrentsOnCompletion ( ) & & ! was_already_seeded ) {
2010-02-03 19:44:19 +00:00
// Remember finished state
TorrentPersistentData : : saveSeedStatus ( h ) ;
2010-01-31 16:14:56 +00:00
h . force_recheck ( ) ;
2010-02-03 19:44:19 +00:00
} else {
// Remember finished state
TorrentPersistentData : : saveSeedStatus ( h ) ;
}
2010-03-03 17:27:25 +00:00
qDebug ( " Received finished alert for %s " , qPrintable ( h . name ( ) ) ) ;
2009-08-21 11:26:58 +00:00
}
2007-08-27 09:24:22 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( save_resume_data_alert * p = dynamic_cast < save_resume_data_alert * > ( a . get ( ) ) ) {
2010-04-05 19:03:00 +00:00
const QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( p - > handle ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
2010-04-05 19:03:00 +00:00
const QString file = h . hash ( ) + " .fastresume " ;
2010-01-02 23:03:46 +00:00
// Delete old fastresume file if necessary
if ( QFile : : exists ( file ) )
QFile : : remove ( file ) ;
2010-03-03 17:27:25 +00:00
qDebug ( " Saving fastresume data in %s " , qPrintable ( file ) ) ;
2010-01-31 13:09:46 +00:00
if ( p - > resume_data ) {
2010-03-03 17:27:25 +00:00
boost : : filesystem : : ofstream out ( fs : : path ( torrentBackup . path ( ) . toLocal8Bit ( ) . constData ( ) ) / file . toLocal8Bit ( ) . constData ( ) , std : : ios_base : : binary ) ;
2010-01-02 23:03:46 +00:00
out . unsetf ( std : : ios_base : : skipws ) ;
bencode ( std : : ostream_iterator < char > ( out ) , * p - > resume_data ) ;
}
2009-09-21 19:08:33 +00:00
}
2009-08-16 03:09:20 +00:00
}
2010-06-03 20:08:19 +00:00
else if ( file_renamed_alert * p = dynamic_cast < file_renamed_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
if ( h . is_valid ( ) & & h . num_files ( ) > 1 ) {
// Check if folders were renamed
QStringList old_path_parts = misc : : toQStringU ( h . get_torrent_info ( ) . orig_files ( ) . at ( p - > index ) . path . string ( ) ) . split ( " / " ) ;
old_path_parts . removeLast ( ) ;
QString old_path = old_path_parts . join ( " / " ) ;
QStringList new_path_parts = misc : : toQStringU ( p - > name ) . split ( " / " ) ;
new_path_parts . removeLast ( ) ;
if ( old_path ! = new_path_parts . join ( " / " ) ) {
old_path = h . save_path ( ) + " / " + old_path ;
qDebug ( " Detected folder renaming, attempt to delete old folder: %s " , qPrintable ( old_path ) ) ;
QDir ( ) . rmpath ( old_path ) ;
}
}
}
2010-01-04 21:16:33 +00:00
else if ( storage_moved_alert * p = dynamic_cast < storage_moved_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
2010-04-10 14:50:11 +00:00
if ( h . is_valid ( ) ) {
2010-04-10 15:00:35 +00:00
// Attempt to remove old folder if empty
const QString & old_save_path = TorrentPersistentData : : getSavePath ( h . hash ( ) ) ;
2010-04-22 16:19:49 +00:00
const QString new_save_path = QString : : fromLocal8Bit ( p - > path . c_str ( ) ) ;
2010-04-10 15:00:35 +00:00
qDebug ( " Torrent moved from %s to %s " , qPrintable ( old_save_path ) , qPrintable ( new_save_path ) ) ;
qDebug ( " Attempting to remove %s " , qPrintable ( old_save_path ) ) ;
2010-05-01 08:29:04 +00:00
if ( old_save_path ! = defaultSavePath & & old_save_path ! = defaultTempPath )
QDir ( ) . rmdir ( old_save_path ) ;
2010-05-01 08:12:42 +00:00
if ( new_save_path ! = defaultTempPath )
TorrentPersistentData : : saveSavePath ( h . hash ( ) , new_save_path ) ;
2010-04-10 14:50:11 +00:00
emit savePathChanged ( h ) ;
2010-04-10 15:54:55 +00:00
//h.force_recheck();
2010-04-10 14:50:11 +00:00
}
2010-01-04 21:16:33 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( metadata_received_alert * p = dynamic_cast < metadata_received_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
if ( h . is_valid ( ) ) {
2010-03-03 17:27:25 +00:00
qDebug ( " Received metadata for %s " , qPrintable ( h . hash ( ) ) ) ;
2010-01-19 21:47:53 +00:00
// Save metadata
2010-04-05 19:03:00 +00:00
const QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2010-01-19 21:47:53 +00:00
if ( ! QFile : : exists ( torrentBackup . path ( ) + QDir : : separator ( ) + h . hash ( ) + QString ( " .torrent " ) ) )
h . save_torrent_file ( torrentBackup . path ( ) + QDir : : separator ( ) + h . hash ( ) + QString ( " .torrent " ) ) ;
2010-01-31 13:09:46 +00:00
// Copy the torrent file to the export folder
if ( torrentExport ) {
QDir exportPath ( Preferences : : getExportDir ( ) ) ;
if ( exportPath . exists ( ) | | exportPath . mkpath ( exportPath . absolutePath ( ) ) ) {
QString torrent_path = exportPath . absoluteFilePath ( h . name ( ) + " .torrent " ) ;
bool duplicate = false ;
if ( QFile : : exists ( torrent_path ) ) {
// Append hash to torrent name to make it unique
torrent_path = exportPath . absoluteFilePath ( h . name ( ) + " - " + h . hash ( ) + " .torrent " ) ;
duplicate = true ;
}
h . save_torrent_file ( torrent_path ) ;
if ( duplicate ) {
// Remove duplicate file if indentical
if ( misc : : sameFiles ( exportPath . absoluteFilePath ( h . name ( ) + " .torrent " ) , torrent_path ) ) {
QFile : : remove ( torrent_path ) ;
}
}
}
}
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-04-06 09:17:33 +00:00
// Append .!qB to incomplete files
if ( appendqBExtension )
appendqBextensionToTorrent ( h , true ) ;
# endif
// Truncate root folder
misc : : truncateRootFolder ( p - > handle ) ;
emit metadataReceived ( h ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_paused ( ) ) {
// XXX: Unfortunately libtorrent-rasterbar does not send a torrent_paused_alert
// and the torrent can be paused when metadata is received
emit pausedTorrent ( h ) ;
}
2010-04-06 09:17:33 +00:00
2009-08-17 08:08:51 +00:00
}
}
2010-01-02 23:03:46 +00:00
else if ( file_error_alert * p = dynamic_cast < file_error_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
2009-11-17 08:15:26 +00:00
if ( h . is_valid ( ) ) {
2010-01-02 23:03:46 +00:00
h . auto_managed ( false ) ;
std : : cerr < < " File Error: " < < p - > message ( ) . c_str ( ) < < std : : endl ;
2010-02-22 19:30:36 +00:00
addConsoleMessage ( tr ( " An I/O error occured, '%1' paused. " ) . arg ( h . name ( ) ) ) ;
addConsoleMessage ( tr ( " Reason: %1 " ) . arg ( misc : : toQString ( p - > message ( ) ) ) ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
emit fullDiskError ( h , misc : : toQString ( p - > message ( ) ) ) ;
2010-05-24 14:19:25 +00:00
//h.pause();
2010-01-02 23:03:46 +00:00
emit pausedTorrent ( h ) ;
}
2009-11-17 08:15:26 +00:00
}
2009-09-21 19:08:33 +00:00
}
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR > 14
2010-01-02 23:03:46 +00:00
else if ( file_completed_alert * p = dynamic_cast < file_completed_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
if ( appendqBExtension ) {
2010-06-03 20:08:19 +00:00
QString name = misc : : toQStringU ( h . get_torrent_info ( ) . file_at ( p - > index ) . path . string ( ) ) ;
2010-01-02 23:03:46 +00:00
if ( name . endsWith ( " .!qB " ) ) {
2010-03-13 13:45:10 +00:00
const QString old_name = name ;
2010-01-02 23:03:46 +00:00
name . chop ( 4 ) ;
2010-03-13 13:45:10 +00:00
qDebug ( " Renaming %s to %s " , qPrintable ( old_name ) , qPrintable ( name ) ) ;
2010-01-02 23:03:46 +00:00
h . rename_file ( p - > index , name ) ;
}
2009-12-18 13:36:47 +00:00
}
}
# endif
2010-03-20 21:21:42 +00:00
else if ( torrent_paused_alert * p = dynamic_cast < torrent_paused_alert * > ( a . get ( ) ) ) {
2010-03-23 21:35:47 +00:00
if ( p - > handle . is_valid ( ) ) {
p - > handle . save_resume_data ( ) ;
}
2009-08-17 08:08:51 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( tracker_error_alert * p = dynamic_cast < tracker_error_alert * > ( a . get ( ) ) ) {
// Level: fatal
2010-03-04 19:29:48 +00:00
QTorrentHandle h ( p - > handle ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
// Authentication
if ( p - > status_code ! = 401 ) {
qDebug ( " Received a tracker error for %s: %s " , p - > url . c_str ( ) , p - > msg . c_str ( ) ) ;
2010-03-03 17:27:25 +00:00
const QString & tracker_url = misc : : toQString ( p - > url ) ;
2010-01-02 23:03:46 +00:00
QHash < QString , TrackerInfos > trackers_data = trackersInfos . value ( h . hash ( ) , QHash < QString , TrackerInfos > ( ) ) ;
TrackerInfos data = trackers_data . value ( tracker_url , TrackerInfos ( tracker_url ) ) ;
data . last_message = misc : : toQString ( p - > msg ) ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2010-01-02 23:03:46 +00:00
data . verified = false ;
+ + data . fail_count ;
# endif
trackers_data . insert ( tracker_url , data ) ;
trackersInfos [ h . hash ( ) ] = trackers_data ;
} else {
emit trackerAuthenticationRequired ( h ) ;
}
}
}
else if ( tracker_reply_alert * p = dynamic_cast < tracker_reply_alert * > ( a . get ( ) ) ) {
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( p - > handle ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
2010-01-27 12:21:14 +00:00
qDebug ( " Received a tracker reply from %s (Num_peers=%d) " , p - > url . c_str ( ) , p - > num_peers ) ;
2010-01-02 23:03:46 +00:00
// Connection was successful now. Remove possible old errors
2009-11-19 15:04:43 +00:00
QHash < QString , TrackerInfos > trackers_data = trackersInfos . value ( h . hash ( ) , QHash < QString , TrackerInfos > ( ) ) ;
2010-03-03 17:27:25 +00:00
const QString & tracker_url = misc : : toQString ( p - > url ) ;
2009-11-19 15:04:43 +00:00
TrackerInfos data = trackers_data . value ( tracker_url , TrackerInfos ( tracker_url ) ) ;
2010-01-02 23:03:46 +00:00
data . last_message = " " ; // Reset error/warning message
data . num_peers = p - > num_peers ;
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2010-01-02 23:03:46 +00:00
data . fail_count = 0 ;
data . verified = true ;
2009-11-22 11:16:30 +00:00
# endif
2009-11-19 15:04:43 +00:00
trackers_data . insert ( tracker_url , data ) ;
trackersInfos [ h . hash ( ) ] = trackers_data ;
2007-08-27 09:24:22 +00:00
}
2010-01-02 23:03:46 +00:00
} else if ( tracker_warning_alert * p = dynamic_cast < tracker_warning_alert * > ( a . get ( ) ) ) {
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( p - > handle ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
// Connection was successful now but there is a warning message
QHash < QString , TrackerInfos > trackers_data = trackersInfos . value ( h . hash ( ) , QHash < QString , TrackerInfos > ( ) ) ;
2010-03-03 17:27:25 +00:00
const QString & tracker_url = misc : : toQString ( p - > url ) ;
2010-01-02 23:03:46 +00:00
TrackerInfos data = trackers_data . value ( tracker_url , TrackerInfos ( tracker_url ) ) ;
data . last_message = misc : : toQString ( p - > msg ) ; // Store warning message
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2010-01-02 23:03:46 +00:00
data . verified = true ;
data . fail_count = 0 ;
2009-11-23 07:18:28 +00:00
# endif
2010-01-02 23:03:46 +00:00
trackers_data . insert ( tracker_url , data ) ;
trackersInfos [ h . hash ( ) ] = trackers_data ;
qDebug ( " Received a tracker warning from %s: %s " , p - > url . c_str ( ) , p - > msg . c_str ( ) ) ;
}
2009-11-19 08:14:04 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( portmap_error_alert * p = dynamic_cast < portmap_error_alert * > ( a . get ( ) ) ) {
2010-03-03 19:04:34 +00:00
addConsoleMessage ( tr ( " UPnP/NAT-PMP: Port mapping failure, message: %1 " ) . arg ( misc : : toQString ( p - > message ( ) ) ) , " red " ) ;
2010-01-02 23:03:46 +00:00
//emit UPnPError(QString(p->msg().c_str()));
2008-05-15 19:48:15 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( portmap_alert * p = dynamic_cast < portmap_alert * > ( a . get ( ) ) ) {
qDebug ( " UPnP Success, msg: %s " , p - > message ( ) . c_str ( ) ) ;
2010-03-03 17:27:25 +00:00
addConsoleMessage ( tr ( " UPnP/NAT-PMP: Port mapping successful, message: %1 " ) . arg ( misc : : toQString ( p - > message ( ) ) ) , " blue " ) ;
2010-01-02 23:03:46 +00:00
//emit UPnPSuccess(QString(p->msg().c_str()));
2007-08-27 09:24:22 +00:00
}
2010-01-02 23:03:46 +00:00
else if ( peer_blocked_alert * p = dynamic_cast < peer_blocked_alert * > ( a . get ( ) ) ) {
addPeerBanMessage ( QString ( p - > ip . to_string ( ) . c_str ( ) ) , true ) ;
//emit peerBlocked(QString::fromUtf8(p->ip.to_string().c_str()));
}
else if ( peer_ban_alert * p = dynamic_cast < peer_ban_alert * > ( a . get ( ) ) ) {
addPeerBanMessage ( QString ( p - > ip . address ( ) . to_string ( ) . c_str ( ) ) , false ) ;
//emit peerBlocked(QString::fromUtf8(p->ip.to_string().c_str()));
}
else if ( fastresume_rejected_alert * p = dynamic_cast < fastresume_rejected_alert * > ( a . get ( ) ) ) {
2010-03-11 21:43:09 +00:00
const QTorrentHandle h ( p - > handle ) ;
2010-01-02 23:03:46 +00:00
if ( h . is_valid ( ) ) {
2010-03-03 17:27:25 +00:00
qDebug ( " /! \\ Fast resume failed for %s, reason: %s " , qPrintable ( h . name ( ) ) , p - > message ( ) . c_str ( ) ) ;
2010-01-02 23:03:46 +00:00
addConsoleMessage ( tr ( " Fast resume data was rejected for torrent %1, checking again... " ) . arg ( h . name ( ) ) , QString : : fromUtf8 ( " red " ) ) ;
2010-03-03 17:27:25 +00:00
addConsoleMessage ( tr ( " Reason: %1 " ) . arg ( misc : : toQString ( p - > message ( ) ) ) ) ;
2010-01-02 23:03:46 +00:00
//emit fastResumeDataRejected(h.name());
}
}
else if ( url_seed_alert * p = dynamic_cast < url_seed_alert * > ( a . get ( ) ) ) {
2010-03-03 17:27:25 +00:00
addConsoleMessage ( tr ( " Url seed lookup failed for url: %1, message: %2 " ) . arg ( misc : : toQString ( p - > url ) ) . arg ( misc : : toQString ( p - > message ( ) ) ) , QString : : fromUtf8 ( " red " ) ) ;
2010-01-02 23:03:46 +00:00
//emit urlSeedProblem(QString::fromUtf8(p->url.c_str()), QString::fromUtf8(p->msg().c_str()));
}
else if ( torrent_checked_alert * p = dynamic_cast < torrent_checked_alert * > ( a . get ( ) ) ) {
QTorrentHandle h ( p - > handle ) ;
if ( h . is_valid ( ) ) {
2010-03-03 17:27:25 +00:00
const QString & hash = h . hash ( ) ;
qDebug ( " %s have just finished checking " , qPrintable ( hash ) ) ;
2010-01-30 16:28:07 +00:00
// Save seed status
TorrentPersistentData : : saveSeedStatus ( h ) ;
2010-01-02 23:03:46 +00:00
// Move to temp directory if necessary
if ( ! h . is_seed ( ) & & ! defaultTempPath . isEmpty ( ) ) {
// Check if directory is different
2010-04-05 19:03:00 +00:00
const QDir current_dir ( h . save_path ( ) ) ;
const QDir save_dir ( getSavePath ( h . hash ( ) ) ) ;
2010-01-02 23:03:46 +00:00
if ( current_dir = = save_dir ) {
h . move_storage ( defaultTempPath ) ;
}
2009-08-06 08:58:49 +00:00
}
2010-01-02 23:03:46 +00:00
emit torrentFinishedChecking ( h ) ;
2010-01-20 18:02:26 +00:00
if ( torrentsToPausedAfterChecking . contains ( hash ) ) {
torrentsToPausedAfterChecking . removeOne ( hash ) ;
h . pause ( ) ;
emit pausedTorrent ( h ) ;
}
2009-08-06 08:58:49 +00:00
}
2007-08-17 02:03:13 +00:00
}
2010-01-02 23:03:46 +00:00
a = s - > pop_alert ( ) ;
2007-08-17 02:03:13 +00:00
}
2007-03-05 17:35:38 +00:00
}
2010-01-20 18:02:26 +00:00
void Bittorrent : : recheckTorrent ( QString hash ) {
QTorrentHandle h = getTorrentHandle ( hash ) ;
if ( h . is_valid ( ) & & h . has_metadata ( ) ) {
if ( h . is_paused ( ) ) {
if ( ! torrentsToPausedAfterChecking . contains ( h . hash ( ) ) ) {
torrentsToPausedAfterChecking < < h . hash ( ) ;
h . resume ( ) ;
}
}
h . force_recheck ( ) ;
}
}
2010-01-02 23:03:46 +00:00
QHash < QString , TrackerInfos > Bittorrent : : getTrackersInfo ( QString hash ) const {
return trackersInfos . value ( hash , QHash < QString , TrackerInfos > ( ) ) ;
}
2007-07-27 13:58:12 +00:00
2010-01-02 23:03:46 +00:00
int Bittorrent : : getListenPort ( ) const {
qDebug ( " LISTEN PORT: %d " , s - > listen_port ( ) ) ;
return s - > listen_port ( ) ;
}
2007-03-07 22:36:01 +00:00
2010-01-02 23:03:46 +00:00
session_status Bittorrent : : getSessionStatus ( ) const {
return s - > status ( ) ;
}
2007-03-07 22:36:01 +00:00
2010-04-06 09:17:33 +00:00
QString Bittorrent : : getSavePath ( QString hash , bool fromScanDir , QString filePath , QString root_folder ) {
2010-01-02 23:03:46 +00:00
QString savePath ;
if ( TorrentTempData : : hasTempData ( hash ) ) {
savePath = TorrentTempData : : getSavePath ( hash ) ;
2010-02-28 15:15:00 +00:00
if ( savePath . isEmpty ( ) ) {
2010-03-04 19:29:48 +00:00
savePath = defaultSavePath ;
2010-02-28 15:15:00 +00:00
}
2010-04-10 14:50:11 +00:00
if ( appendLabelToSavePath ) {
2010-01-02 23:03:46 +00:00
qDebug ( " appendLabelToSavePath is true " ) ;
2010-03-03 19:04:34 +00:00
const QString & label = TorrentTempData : : getLabel ( hash ) ;
2010-01-02 23:03:46 +00:00
if ( ! label . isEmpty ( ) ) {
2010-04-10 14:50:11 +00:00
savePath = misc : : updateLabelInSavePath ( defaultSavePath , savePath , " " , label ) ;
2009-12-17 20:28:30 +00:00
}
}
2010-03-03 19:04:34 +00:00
qDebug ( " getSavePath, got save_path from temp data: %s " , qPrintable ( savePath ) ) ;
2010-01-02 23:03:46 +00:00
} else {
savePath = TorrentPersistentData : : getSavePath ( hash ) ;
2010-04-22 16:19:49 +00:00
qDebug ( " SavePath got from persistant data is %s " , qPrintable ( savePath ) ) ;
2010-04-06 09:17:33 +00:00
bool append_root_folder = false ;
2010-02-28 15:15:00 +00:00
if ( savePath . isEmpty ( ) ) {
if ( fromScanDir & & m_scanFolders - > downloadInTorrentFolder ( filePath ) )
savePath = QFileInfo ( filePath ) . dir ( ) . path ( ) ;
2010-04-06 09:17:33 +00:00
else {
2010-02-28 15:15:00 +00:00
savePath = defaultSavePath ;
2010-04-06 09:17:33 +00:00
append_root_folder = true ;
}
} else {
QSettings settings ( " qBittorrent " , " qBittorrent " ) ;
if ( ! settings . value ( " ported_to_new_savepath_system " , false ) . toBool ( ) ) {
append_root_folder = true ;
}
2010-02-28 15:15:00 +00:00
}
if ( ! fromScanDir & & appendLabelToSavePath ) {
2010-03-03 19:04:34 +00:00
const QString & label = TorrentPersistentData : : getLabel ( hash ) ;
2010-01-02 23:03:46 +00:00
if ( ! label . isEmpty ( ) ) {
2010-04-22 16:19:49 +00:00
qDebug ( " Torrent label is %s " , qPrintable ( label ) ) ;
2010-04-10 14:50:11 +00:00
savePath = misc : : updateLabelInSavePath ( defaultSavePath , savePath , " " , label ) ;
2009-12-17 20:28:30 +00:00
}
}
2010-04-06 09:17:33 +00:00
if ( append_root_folder & & ! root_folder . isEmpty ( ) ) {
// Append torrent root folder to the save path
if ( ! savePath . endsWith ( QDir : : separator ( ) ) )
savePath + = QDir : : separator ( ) ;
savePath + = root_folder ;
2010-04-22 16:19:49 +00:00
qDebug ( " Torrent root folder is %s " , qPrintable ( root_folder ) ) ;
2010-04-06 09:17:33 +00:00
TorrentPersistentData : : saveSavePath ( hash , savePath ) ;
}
2010-03-03 19:04:34 +00:00
qDebug ( " getSavePath, got save_path from persistent data: %s " , qPrintable ( savePath ) ) ;
2010-01-02 23:03:46 +00:00
}
// Clean path
savePath = misc : : expandPath ( savePath ) ;
// Checking if savePath Dir exists
// create it if it is not
QDir saveDir ( savePath ) ;
if ( ! saveDir . exists ( ) ) {
if ( ! saveDir . mkpath ( saveDir . absolutePath ( ) ) ) {
2010-03-03 19:04:34 +00:00
std : : cerr < < " Couldn't create the save directory: " < < qPrintable ( saveDir . path ( ) ) < < " \n " ;
2010-01-02 23:03:46 +00:00
// XXX: Do something else?
2010-04-22 16:19:49 +00:00
} else {
qDebug ( " Created save folder at %s " , qPrintable ( saveDir . path ( ) ) ) ;
2010-01-02 23:03:46 +00:00
}
2009-12-17 20:28:30 +00:00
}
2010-01-02 23:03:46 +00:00
return savePath ;
2007-03-05 17:35:38 +00:00
}
2007-03-07 22:36:01 +00:00
2010-01-02 23:03:46 +00:00
// Take an url string to a torrent file,
// download the torrent file to a tmp location, then
// add it to download list
void Bittorrent : : downloadFromUrl ( QString url ) {
addConsoleMessage ( tr ( " Downloading '%1', please wait... " , " e.g: Downloading 'xxx.torrent', please wait... " ) . arg ( url )
2010-01-02 22:20:37 +00:00
# ifndef DISABLE_GUI
2010-01-02 23:03:46 +00:00
, QPalette : : WindowText
2010-01-02 22:20:37 +00:00
# endif
2010-01-02 23:03:46 +00:00
) ;
//emit aboutToDownloadFromUrl(url);
// Launch downloader thread
2010-03-12 20:00:51 +00:00
downloader - > downloadTorrentUrl ( url ) ;
2010-01-02 23:03:46 +00:00
}
2007-03-07 22:36:01 +00:00
2010-01-02 23:03:46 +00:00
void Bittorrent : : downloadFromURLList ( const QStringList & urls ) {
foreach ( const QString & url , urls ) {
downloadFromUrl ( url ) ;
}
2009-09-07 11:48:31 +00:00
}
2010-01-02 23:03:46 +00:00
void Bittorrent : : addMagnetSkipAddDlg ( QString uri ) {
addMagnetUri ( uri , false ) ;
}
2009-08-21 14:05:03 +00:00
2010-01-02 23:03:46 +00:00
void Bittorrent : : downloadUrlAndSkipDialog ( QString url , QString save_path ) {
//emit aboutToDownloadFromUrl(url);
2010-03-03 19:04:34 +00:00
const QUrl & qurl = QUrl : : fromEncoded ( url . toLocal8Bit ( ) ) ;
2010-01-02 23:03:46 +00:00
if ( ! save_path . isEmpty ( ) )
savepath_fromurl [ qurl ] = save_path ;
url_skippingDlg < < qurl ;
// Launch downloader thread
downloader - > downloadUrl ( url ) ;
}
2008-09-14 10:14:54 +00:00
2010-01-02 23:03:46 +00:00
// Add to Bittorrent session the downloaded torrent file
void Bittorrent : : processDownloadedFile ( QString url , QString file_path ) {
2010-03-03 19:04:34 +00:00
const int index = url_skippingDlg . indexOf ( QUrl : : fromEncoded ( url . toLocal8Bit ( ) ) ) ;
2010-01-02 23:03:46 +00:00
if ( index < 0 ) {
// Add file to torrent download list
2010-05-30 17:51:40 +00:00
# ifdef Q_WS_WIN
// Windows hack
if ( ! file_path . endsWith ( " .torrent " ) ) {
Q_ASSERT ( QFile : : exists ( file_path ) ) ;
qDebug ( " Torrent name does not end with .torrent, fixing... " ) ;
if ( QFile : : rename ( file_path , file_path + " .torrent " ) ) {
file_path + = " .torrent " ;
} else {
qDebug ( " Failed to rename torrent file! " ) ;
}
}
qDebug ( " Downloading torrent at path: %s " , qPrintable ( file_path ) ) ;
# endif
2010-01-02 23:03:46 +00:00
emit newDownloadedTorrent ( file_path , url ) ;
} else {
url_skippingDlg . removeAt ( index ) ;
addTorrent ( file_path , false , url , false ) ;
}
2008-09-14 10:14:54 +00:00
}
2007-03-07 22:36:01 +00:00
2010-01-02 23:03:46 +00:00
// Return current download rate for the BT
// session. Payload means that it only take into
// account "useful" part of the rate
float Bittorrent : : getPayloadDownloadRate ( ) const {
2010-03-03 19:04:34 +00:00
const session_status & sessionStatus = s - > status ( ) ;
2010-01-02 23:03:46 +00:00
return sessionStatus . payload_download_rate ;
}
2007-03-07 22:36:01 +00:00
2010-01-02 23:03:46 +00:00
// Return current upload rate for the BT
// session. Payload means that it only take into
// account "useful" part of the rate
float Bittorrent : : getPayloadUploadRate ( ) const {
2010-03-03 19:04:34 +00:00
const session_status & sessionStatus = s - > status ( ) ;
2010-01-02 23:03:46 +00:00
return sessionStatus . payload_upload_rate ;
}
2007-03-07 22:36:01 +00:00
2010-06-07 08:32:41 +00:00
# if LIBTORRENT_VERSION_MINOR < 15
2010-01-02 23:03:46 +00:00
// Save DHT entry to hard drive
void Bittorrent : : saveDHTEntry ( ) {
// Save DHT entry
if ( DHTEnabled ) {
try {
entry dht_state = s - > dht_state ( ) ;
2010-04-05 19:03:00 +00:00
const QString dht_path = misc : : cacheLocation ( ) + QDir : : separator ( ) + QString : : fromUtf8 ( " dht_state " ) ;
2010-03-06 21:11:47 +00:00
boost : : filesystem : : ofstream out ( dht_path . toLocal8Bit ( ) . constData ( ) , std : : ios_base : : binary ) ;
2010-01-02 23:03:46 +00:00
out . unsetf ( std : : ios_base : : skipws ) ;
bencode ( std : : ostream_iterator < char > ( out ) , dht_state ) ;
qDebug ( " DHT entry saved " ) ;
} catch ( std : : exception & e ) {
std : : cerr < < e . what ( ) < < " \n " ;
}
2007-03-07 22:36:01 +00:00
}
}
2010-04-06 16:52:17 +00:00
# endif
2009-11-06 14:30:14 +00:00
2010-01-02 23:03:46 +00:00
void Bittorrent : : applyEncryptionSettings ( pe_settings se ) {
qDebug ( " Applying encryption settings " ) ;
s - > set_pe_settings ( se ) ;
2009-08-27 03:13:39 +00:00
}
2009-11-06 14:30:14 +00:00
2010-01-02 23:03:46 +00:00
// Will fast resume torrents in
// backup directory
void Bittorrent : : startUpTorrents ( ) {
qDebug ( " Resuming unfinished torrents " ) ;
2010-04-05 19:03:00 +00:00
const QDir torrentBackup ( misc : : BTBackupLocation ( ) ) ;
2010-03-03 19:04:34 +00:00
const QStringList & known_torrents = TorrentPersistentData : : knownTorrents ( ) ;
2010-01-02 23:03:46 +00:00
// Safety measure because some people reported torrent loss since
// we switch the v1.5 way of resuming torrents on startup
QStringList filters ;
filters < < " *.torrent " ;
2010-03-03 19:04:34 +00:00
const QStringList & torrents_on_hd = torrentBackup . entryList ( filters , QDir : : Files , QDir : : Unsorted ) ;
2010-01-02 23:03:46 +00:00
foreach ( QString hash , torrents_on_hd ) {
2010-04-06 09:17:33 +00:00
qDebug ( " found torrent with hash: %s on hard disk " , qPrintable ( hash ) ) ;
2010-01-02 23:03:46 +00:00
hash . chop ( 8 ) ; // remove trailing .torrent
if ( ! known_torrents . contains ( hash ) ) {
2010-03-03 19:04:34 +00:00
std : : cerr < < " ERROR Detected!!! Adding back torrent " < < qPrintable ( hash ) < < " which got lost for some reason. " < < std : : endl ;
2009-08-17 05:14:03 +00:00
addTorrent ( torrentBackup . path ( ) + QDir : : separator ( ) + hash + " .torrent " , false , QString ( ) , true ) ;
}
2009-08-16 03:09:20 +00:00
}
2010-01-02 23:03:46 +00:00
// End of safety measure
2009-08-27 08:43:19 +00:00
2010-01-02 23:03:46 +00:00
qDebug ( " Starting up torrents " ) ;
if ( isQueueingEnabled ( ) ) {
2010-03-04 19:29:48 +00:00
priority_queue < QPair < int , QString > , vector < QPair < int , QString > > , std : : greater < QPair < int , QString > > > torrent_queue ;
2010-01-02 23:03:46 +00:00
foreach ( const QString & hash , known_torrents ) {
QString filePath ;
if ( TorrentPersistentData : : isMagnet ( hash ) ) {
filePath = TorrentPersistentData : : getMagnetUri ( hash ) ;
} else {
filePath = torrentBackup . path ( ) + QDir : : separator ( ) + hash + " .torrent " ;
2009-08-27 08:43:19 +00:00
}
2010-03-03 19:04:34 +00:00
const int prio = TorrentPersistentData : : getPriority ( hash ) ;
2010-03-04 19:29:48 +00:00
torrent_queue . push ( qMakePair ( prio , hash ) ) ;
2010-01-02 23:03:46 +00:00
}
2010-04-06 09:17:33 +00:00
qDebug ( " Priority_queue size: %ld " , ( long ) torrent_queue . size ( ) ) ;
2010-01-02 23:03:46 +00:00
// Resume downloads
2010-03-04 19:29:48 +00:00
while ( ! torrent_queue . empty ( ) ) {
2010-04-06 09:17:33 +00:00
const QString hash = torrent_queue . top ( ) . second ;
2010-03-04 19:29:48 +00:00
torrent_queue . pop ( ) ;
2010-03-03 19:04:34 +00:00
qDebug ( " Starting up torrent %s " , qPrintable ( hash ) ) ;
2010-01-02 23:03:46 +00:00
if ( TorrentPersistentData : : isMagnet ( hash ) ) {
addMagnetUri ( TorrentPersistentData : : getMagnetUri ( hash ) , true ) ;
} else {
addTorrent ( torrentBackup . path ( ) + QDir : : separator ( ) + hash + " .torrent " , false , QString ( ) , true ) ;
}
}
} else {
// Resume downloads
foreach ( const QString & hash , known_torrents ) {
2010-03-03 19:04:34 +00:00
qDebug ( " Starting up torrent %s " , qPrintable ( hash ) ) ;
2010-01-02 23:03:46 +00:00
if ( TorrentPersistentData : : isMagnet ( hash ) )
addMagnetUri ( TorrentPersistentData : : getMagnetUri ( hash ) , true ) ;
else
addTorrent ( torrentBackup . path ( ) + QDir : : separator ( ) + hash + " .torrent " , false , QString ( ) , true ) ;
2009-08-27 08:43:19 +00:00
}
}
2010-04-06 09:17:33 +00:00
QSettings settings ( " qBittorrent " , " qBittorrent " ) ;
settings . setValue ( " ported_to_new_savepath_system " , true ) ;
2010-01-02 23:03:46 +00:00
qDebug ( " Unfinished torrents resumed " ) ;
2009-08-27 08:43:19 +00:00
}