/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2015 Vladimir Golovnev * Copyright (C) 2006 Christophe Dumez * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. */ #ifndef BITTORRENT_SESSION_H #define BITTORRENT_SESSION_H #include #include #include #include #include #if LIBTORRENT_VERSION_NUM < 10100 #include #endif #include #include #include #include #include #include #include "base/settingvalue.h" #include "base/tristatebool.h" #include "base/types.h" #include "torrentinfo.h" namespace libtorrent { class session; struct torrent_handle; class entry; struct add_torrent_params; struct pe_settings; #if LIBTORRENT_VERSION_NUM < 10100 struct session_settings; #else struct settings_pack; #endif struct session_status; class alert; struct torrent_alert; struct state_update_alert; struct stats_alert; struct add_torrent_alert; struct torrent_checked_alert; struct torrent_finished_alert; struct torrent_removed_alert; struct torrent_deleted_alert; struct torrent_delete_failed_alert; struct torrent_paused_alert; struct torrent_resumed_alert; struct save_resume_data_alert; struct save_resume_data_failed_alert; struct file_renamed_alert; struct storage_moved_alert; struct storage_moved_failed_alert; struct metadata_received_alert; struct file_error_alert; struct file_completed_alert; struct tracker_error_alert; struct tracker_reply_alert; struct tracker_warning_alert; struct portmap_error_alert; struct portmap_alert; struct peer_blocked_alert; struct peer_ban_alert; struct fastresume_rejected_alert; struct url_seed_alert; struct listen_succeeded_alert; struct listen_failed_alert; struct external_ip_alert; } class QThread; class QTimer; class QStringList; class QString; class QUrl; template class QList; class FilterParserThread; class BandwidthScheduler; class Statistics; class ResumeDataSavingManager; enum MaxRatioAction { Pause, Remove }; enum TorrentExportFolder { Regular, Finished }; namespace BitTorrent { class InfoHash; class CacheStatus; class SessionStatus; class TorrentHandle; class Tracker; class MagnetUri; class TrackerEntry; struct AddTorrentData; struct AddTorrentParams { QString name; QString category; QString savePath; bool disableTempPath = false; // e.g. for imported torrents bool sequential = false; TriStateBool addForced; TriStateBool addPaused; QVector filePriorities; // used if TorrentInfo is set bool ignoreShareRatio = false; bool skipChecking = false; }; struct TorrentStatusReport { uint nbDownloading = 0; uint nbSeeding = 0; uint nbCompleted = 0; uint nbActive = 0; uint nbInactive = 0; uint nbPaused = 0; uint nbResumed = 0; uint nbErrored = 0; }; class Session : public QObject { Q_OBJECT Q_DISABLE_COPY(Session) public: static void initInstance(); static void freeInstance(); static Session *instance(); QString defaultSavePath() const; void setDefaultSavePath(QString path); QString tempPath() const; void setTempPath(QString path); bool isTempPathEnabled() const; void setTempPathEnabled(bool enabled); QString torrentTempPath(const InfoHash &hash) const; static bool isValidCategoryName(const QString &name); // returns category itself and all top level categories static QStringList expandCategory(const QString &category); QStringList categories() const; QString categorySavePath(const QString &categoryName) const; bool addCategory(const QString &name, const QString &savePath = ""); bool editCategory(const QString &name, const QString &savePath); bool removeCategory(const QString &name); bool isSubcategoriesEnabled() const; void setSubcategoriesEnabled(bool value); // Torrent Management Mode subsystem (TMM) // // Each torrent can be either in Manual mode or in Automatic mode // In Manual Mode various torrent properties are set explicitly(eg save path) // In Automatic Mode various torrent properties are set implicitly(eg save path) // based on the associated category. // In Automatic Mode torrent save path can be changed in following cases: // 1. Default save path changed // 2. Torrent category save path changed // 3. Torrent category changed // (unless otherwise is specified) bool isAutoTMMDisabledByDefault() const; void setAutoTMMDisabledByDefault(bool value); bool isDisableAutoTMMWhenCategoryChanged() const; void setDisableAutoTMMWhenCategoryChanged(bool value); bool isDisableAutoTMMWhenDefaultSavePathChanged() const; void setDisableAutoTMMWhenDefaultSavePathChanged(bool value); bool isDisableAutoTMMWhenCategorySavePathChanged() const; void setDisableAutoTMMWhenCategorySavePathChanged(bool value); qreal globalMaxRatio() const; void setGlobalMaxRatio(qreal ratio); bool isDHTEnabled() const; void setDHTEnabled(bool enabled); bool isLSDEnabled() const; void setLSDEnabled(bool enabled); bool isPeXEnabled() const; void setPeXEnabled(bool enabled); bool isTrackerExchangeEnabled() const; void setTrackerExchangeEnabled(bool enabled); bool isAddTorrentPaused() const; void setAddTorrentPaused(bool value); bool isTrackerEnabled() const; void setTrackerEnabled(bool enabled); bool isAppendExtensionEnabled() const; void setAppendExtensionEnabled(bool enabled); uint refreshInterval() const; void setRefreshInterval(uint value); bool isPreallocationEnabled() const; void setPreallocationEnabled(bool enabled); QString torrentExportDirectory() const; void setTorrentExportDirectory(QString path); QString finishedTorrentExportDirectory() const; void setFinishedTorrentExportDirectory(QString path); int globalDownloadSpeedLimit() const; void setGlobalDownloadSpeedLimit(int limit); int globalUploadSpeedLimit() const; void setGlobalUploadSpeedLimit(int limit); int altGlobalDownloadSpeedLimit() const; void setAltGlobalDownloadSpeedLimit(int limit); int altGlobalUploadSpeedLimit() const; void setAltGlobalUploadSpeedLimit(int limit); int downloadSpeedLimit() const; void setDownloadSpeedLimit(int limit); int uploadSpeedLimit() const; void setUploadSpeedLimit(int limit); bool isAltGlobalSpeedLimitEnabled() const; void setAltGlobalSpeedLimitEnabled(bool enabled); bool isBandwidthSchedulerEnabled() const; void setBandwidthSchedulerEnabled(bool enabled); uint saveResumeDataInterval() const; void setSaveResumeDataInterval(uint value); int port() const; void setPort(int port); bool useRandomPort() const; void setUseRandomPort(bool value); QString networkInterface() const; void setNetworkInterface(const QString &interface); QString networkInterfaceName() const; void setNetworkInterfaceName(const QString &name); QString networkInterfaceAddress() const; void setNetworkInterfaceAddress(const QString &address); bool isIPv6Enabled() const; void setIPv6Enabled(bool enabled); int encryption() const; void setEncryption(int state); bool isForceProxyEnabled() const; void setForceProxyEnabled(bool enabled); bool isProxyPeerConnectionsEnabled() const; void setProxyPeerConnectionsEnabled(bool enabled); bool isAddTrackersEnabled() const; void setAddTrackersEnabled(bool enabled); QString additionalTrackers() const; void setAdditionalTrackers(const QString &trackers); bool isIPFilteringEnabled() const; void setIPFilteringEnabled(bool enabled); QString IPFilterFile() const; void setIPFilterFile(QString path); bool announceToAllTrackers() const; void setAnnounceToAllTrackers(bool val); uint diskCacheSize() const; void setDiskCacheSize(uint size); uint diskCacheTTL() const; void setDiskCacheTTL(uint ttl); bool useOSCache() const; void setUseOSCache(bool use); bool isAnonymousModeEnabled() const; void setAnonymousModeEnabled(bool enabled); bool isQueueingSystemEnabled() const; void setQueueingSystemEnabled(bool enabled); bool ignoreSlowTorrentsForQueueing() const; void setIgnoreSlowTorrentsForQueueing(bool ignore); uint outgoingPortsMin() const; void setOutgoingPortsMin(uint min); uint outgoingPortsMax() const; void setOutgoingPortsMax(uint max); bool ignoreLimitsOnLAN() const; void setIgnoreLimitsOnLAN(bool ignore); bool includeOverheadInLimits() const; void setIncludeOverheadInLimits(bool include); QString announceIP() const; void setAnnounceIP(const QString &ip); bool isSuperSeedingEnabled() const; void setSuperSeedingEnabled(bool enabled); int maxConnections() const; void setMaxConnections(int max); int maxHalfOpenConnections() const; void setMaxHalfOpenConnections(int max); int maxConnectionsPerTorrent() const; void setMaxConnectionsPerTorrent(int max); int maxUploads() const; void setMaxUploads(int max); int maxUploadsPerTorrent() const; void setMaxUploadsPerTorrent(int max); int maxActiveDownloads() const; void setMaxActiveDownloads(int max); int maxActiveUploads() const; void setMaxActiveUploads(int max); int maxActiveTorrents() const; void setMaxActiveTorrents(int max); bool isUTPEnabled() const; void setUTPEnabled(bool enabled); bool isUTPRateLimited() const; void setUTPRateLimited(bool limited); bool isTrackerFilteringEnabled() const; void setTrackerFilteringEnabled(bool enabled); TorrentHandle *findTorrent(const InfoHash &hash) const; QHash torrents() const; TorrentStatusReport torrentStatusReport() const; bool hasActiveTorrents() const; bool hasUnfinishedTorrents() const; SessionStatus status() const; CacheStatus cacheStatus() const; quint64 getAlltimeDL() const; quint64 getAlltimeUL() const; bool isListening() const; MaxRatioAction maxRatioAction() const; void setMaxRatioAction(MaxRatioAction act); void banIP(const QString &ip); bool isKnownTorrent(const InfoHash &hash) const; bool addTorrent(QString source, const AddTorrentParams ¶ms = AddTorrentParams()); bool addTorrent(const TorrentInfo &torrentInfo, const AddTorrentParams ¶ms = AddTorrentParams()); bool deleteTorrent(const QString &hash, bool deleteLocalFiles = false); bool loadMetadata(const MagnetUri &magnetUri); bool cancelLoadMetadata(const InfoHash &hash); void recursiveTorrentDownload(const InfoHash &hash); void increaseTorrentsPriority(const QStringList &hashes); void decreaseTorrentsPriority(const QStringList &hashes); void topTorrentsPriority(const QStringList &hashes); void bottomTorrentsPriority(const QStringList &hashes); // TorrentHandle interface void handleTorrentRatioLimitChanged(TorrentHandle *const torrent); void handleTorrentSavePathChanged(TorrentHandle *const torrent); void handleTorrentCategoryChanged(TorrentHandle *const torrent, const QString &oldCategory); void handleTorrentSavingModeChanged(TorrentHandle *const torrent); void handleTorrentMetadataReceived(TorrentHandle *const torrent); void handleTorrentPaused(TorrentHandle *const torrent); void handleTorrentResumed(TorrentHandle *const torrent); void handleTorrentChecked(TorrentHandle *const torrent); void handleTorrentFinished(TorrentHandle *const torrent); void handleTorrentTrackersAdded(TorrentHandle *const torrent, const QList &newTrackers); void handleTorrentTrackersRemoved(TorrentHandle *const torrent, const QList &deletedTrackers); void handleTorrentTrackersChanged(TorrentHandle *const torrent); void handleTorrentUrlSeedsAdded(TorrentHandle *const torrent, const QList &newUrlSeeds); void handleTorrentUrlSeedsRemoved(TorrentHandle *const torrent, const QList &urlSeeds); void handleTorrentResumeDataReady(TorrentHandle *const torrent, const libtorrent::entry &data); void handleTorrentResumeDataFailed(TorrentHandle *const torrent); void handleTorrentTrackerReply(TorrentHandle *const torrent, const QString &trackerUrl); void handleTorrentTrackerWarning(TorrentHandle *const torrent, const QString &trackerUrl); void handleTorrentTrackerError(TorrentHandle *const torrent, const QString &trackerUrl); void handleTorrentTrackerAuthenticationRequired(TorrentHandle *const torrent, const QString &trackerUrl); signals: void torrentsUpdated(); void addTorrentFailed(const QString &error); void torrentAdded(BitTorrent::TorrentHandle *const torrent); void torrentNew(BitTorrent::TorrentHandle *const torrent); void torrentAboutToBeRemoved(BitTorrent::TorrentHandle *const torrent); void torrentPaused(BitTorrent::TorrentHandle *const torrent); void torrentResumed(BitTorrent::TorrentHandle *const torrent); void torrentFinished(BitTorrent::TorrentHandle *const torrent); void torrentFinishedChecking(BitTorrent::TorrentHandle *const torrent); void torrentSavePathChanged(BitTorrent::TorrentHandle *const torrent); void torrentCategoryChanged(BitTorrent::TorrentHandle *const torrent, const QString &oldCategory); void torrentSavingModeChanged(BitTorrent::TorrentHandle *const torrent); void allTorrentsFinished(); void metadataLoaded(const BitTorrent::TorrentInfo &info); void torrentMetadataLoaded(BitTorrent::TorrentHandle *const torrent); void fullDiskError(BitTorrent::TorrentHandle *const torrent, const QString &msg); void trackerSuccess(BitTorrent::TorrentHandle *const torrent, const QString &tracker); void trackerWarning(BitTorrent::TorrentHandle *const torrent, const QString &tracker); void trackerError(BitTorrent::TorrentHandle *const torrent, const QString &tracker); void trackerAuthenticationRequired(BitTorrent::TorrentHandle *const torrent); void recursiveTorrentDownloadPossible(BitTorrent::TorrentHandle *const torrent); void speedLimitModeChanged(bool alternative); void IPFilterParsed(bool error, int ruleCount); void trackersAdded(BitTorrent::TorrentHandle *const torrent, const QList &trackers); void trackersRemoved(BitTorrent::TorrentHandle *const torrent, const QList &trackers); void trackersChanged(BitTorrent::TorrentHandle *const torrent); void trackerlessStateChanged(BitTorrent::TorrentHandle *const torrent, bool trackerless); void downloadFromUrlFailed(const QString &url, const QString &reason); void downloadFromUrlFinished(const QString &url); void categoryAdded(const QString &categoryName); void categoryRemoved(const QString &categoryName); void subcategoriesSupportChanged(); private slots: void configureDeferred(); void readAlerts(); void refresh(); void processBigRatios(); void generateResumeData(bool final = false); void handleIPFilterParsed(int ruleCount); void handleIPFilterError(); void handleDownloadFinished(const QString &url, const QString &filePath); void handleDownloadFailed(const QString &url, const QString &reason); void handleRedirectedToMagnet(const QString &url, const QString &magnetUri); void switchToAlternativeMode(bool alternative); // Session reconfiguration triggers void networkOnlineStateChanged(const bool online); void networkConfigurationChange(const QNetworkConfiguration&); private: explicit Session(QObject *parent = 0); ~Session(); bool hasPerTorrentRatioLimit() const; void initResumeFolder(); // Session configuration Q_INVOKABLE void configure(); #if LIBTORRENT_VERSION_NUM < 10100 void configure(libtorrent::session_settings &sessionSettings); void adjustLimits(libtorrent::session_settings &sessionSettings); #else void configure(libtorrent::settings_pack &settingsPack); void adjustLimits(libtorrent::settings_pack &settingsPack); #endif void adjustLimits(); void processBannedIPs(); const QStringList getListeningIPs(); void configureListeningInterface(); void changeSpeedLimitMode_impl(bool alternative); void enableTracker(bool enable); void enableBandwidthScheduler(); void populateAdditionalTrackers(); void enableIPFilter(); void disableIPFilter(); void startUpTorrents(); bool addTorrent_impl(AddTorrentData addData, const MagnetUri &magnetUri, TorrentInfo torrentInfo = TorrentInfo(), const QByteArray &fastresumeData = QByteArray()); bool findIncompleteFiles(TorrentInfo &torrentInfo, QString &savePath) const; void updateRatioTimer(); void exportTorrentFile(TorrentHandle *const torrent, TorrentExportFolder folder = TorrentExportFolder::Regular); void saveTorrentResumeData(TorrentHandle *const torrent); void handleAlert(libtorrent::alert *a); void dispatchTorrentAlert(libtorrent::alert *a); void handleAddTorrentAlert(libtorrent::add_torrent_alert *p); void handleStateUpdateAlert(libtorrent::state_update_alert *p); void handleMetadataReceivedAlert(libtorrent::metadata_received_alert *p); void handleFileErrorAlert(libtorrent::file_error_alert *p); void handleTorrentRemovedAlert(libtorrent::torrent_removed_alert *p); void handleTorrentDeletedAlert(libtorrent::torrent_deleted_alert *p); void handleTorrentDeleteFailedAlert(libtorrent::torrent_delete_failed_alert *p); void handlePortmapWarningAlert(libtorrent::portmap_error_alert *p); void handlePortmapAlert(libtorrent::portmap_alert *p); void handlePeerBlockedAlert(libtorrent::peer_blocked_alert *p); void handlePeerBanAlert(libtorrent::peer_ban_alert *p); void handleUrlSeedAlert(libtorrent::url_seed_alert *p); void handleListenSucceededAlert(libtorrent::listen_succeeded_alert *p); void handleListenFailedAlert(libtorrent::listen_failed_alert *p); void handleExternalIPAlert(libtorrent::external_ip_alert *p); void createTorrentHandle(const libtorrent::torrent_handle &nativeHandle); void saveResumeData(); #if LIBTORRENT_VERSION_NUM < 10100 void dispatchAlerts(std::auto_ptr alertPtr); #endif void getPendingAlerts(std::vector &out, ulong time = 0); // BitTorrent libtorrent::session *m_nativeSession; bool m_deferredConfigureScheduled; bool m_IPFilteringChanged; #if LIBTORRENT_VERSION_NUM >= 10100 bool m_listenInterfaceChanged; // optimization #endif CachedSettingValue m_isDHTEnabled; CachedSettingValue m_isLSDEnabled; CachedSettingValue m_isPeXEnabled; CachedSettingValue m_isTrackerExchangeEnabled; CachedSettingValue m_isIPFilteringEnabled; CachedSettingValue m_isTrackerFilteringEnabled; CachedSettingValue m_IPFilterFile; CachedSettingValue m_announceToAllTrackers; CachedSettingValue m_diskCacheSize; CachedSettingValue m_diskCacheTTL; CachedSettingValue m_useOSCache; CachedSettingValue m_isAnonymousModeEnabled; CachedSettingValue m_isQueueingEnabled; CachedSettingValue m_maxActiveDownloads; CachedSettingValue m_maxActiveUploads; CachedSettingValue m_maxActiveTorrents; CachedSettingValue m_ignoreSlowTorrentsForQueueing; CachedSettingValue m_outgoingPortsMin; CachedSettingValue m_outgoingPortsMax; CachedSettingValue m_ignoreLimitsOnLAN; CachedSettingValue m_includeOverheadInLimits; CachedSettingValue m_announceIP; CachedSettingValue m_isSuperSeedingEnabled; CachedSettingValue m_maxConnections; CachedSettingValue m_maxHalfOpenConnections; CachedSettingValue m_maxUploads; CachedSettingValue m_maxConnectionsPerTorrent; CachedSettingValue m_maxUploadsPerTorrent; CachedSettingValue m_isUTPEnabled; CachedSettingValue m_isUTPRateLimited; CachedSettingValue m_isAddTrackersEnabled; CachedSettingValue m_additionalTrackers; CachedSettingValue m_globalMaxRatio; CachedSettingValue m_isAddTorrentPaused; CachedSettingValue m_isAppendExtensionEnabled; CachedSettingValue m_refreshInterval; CachedSettingValue m_isPreallocationEnabled; CachedSettingValue m_torrentExportDirectory; CachedSettingValue m_finishedTorrentExportDirectory; CachedSettingValue m_globalDownloadSpeedLimit; CachedSettingValue m_globalUploadSpeedLimit; CachedSettingValue m_altGlobalDownloadSpeedLimit; CachedSettingValue m_altGlobalUploadSpeedLimit; CachedSettingValue m_isAltGlobalSpeedLimitEnabled; CachedSettingValue m_isBandwidthSchedulerEnabled; CachedSettingValue m_saveResumeDataInterval; CachedSettingValue m_port; CachedSettingValue m_useRandomPort; CachedSettingValue m_networkInterface; CachedSettingValue m_networkInterfaceName; CachedSettingValue m_networkInterfaceAddress; CachedSettingValue m_isIPv6Enabled; CachedSettingValue m_encryption; CachedSettingValue m_isForceProxyEnabled; CachedSettingValue m_isProxyPeerConnectionsEnabled; CachedSettingValue m_storedCategories; CachedSettingValue m_maxRatioAction; CachedSettingValue m_defaultSavePath; CachedSettingValue m_tempPath; CachedSettingValue m_isSubcategoriesEnabled; CachedSettingValue m_isTempPathEnabled; CachedSettingValue m_isAutoTMMDisabledByDefault; CachedSettingValue m_isDisableAutoTMMWhenCategoryChanged; CachedSettingValue m_isDisableAutoTMMWhenDefaultSavePathChanged; CachedSettingValue m_isDisableAutoTMMWhenCategorySavePathChanged; CachedSettingValue m_isTrackerEnabled; CachedSettingValue m_bannedIPs; // Order is important. These need to be declared after their CachedSettingsValue // counterparts, because they use them for initialization in the constructor // initialization list. const bool m_wasPexEnabled; const bool m_wasTrackerExchangeEnabled; int m_numResumeData; int m_extraLimit; QList m_additionalTrackerList; QString m_resumeFolderPath; QFile m_resumeFolderLock; QHash m_savePathsToRemove; bool m_useProxy; QTimer *m_refreshTimer; QTimer *m_bigRatioTimer; QTimer *m_resumeDataTimer; Statistics *m_statistics; // IP filtering QPointer m_filterParser; QPointer m_bwScheduler; // Tracker QPointer m_tracker; // fastresume data writing thread QThread *m_ioThread; ResumeDataSavingManager *m_resumeDataSavingManager; QHash m_loadedMetadata; QHash m_torrents; QHash m_addingTorrents; QHash m_downloadedTorrents; TorrentStatusReport m_torrentStatusReport; QStringMap m_categories; #if LIBTORRENT_VERSION_NUM < 10100 QMutex m_alertsMutex; QWaitCondition m_alertsWaitCondition; std::vector m_alerts; #endif QNetworkConfigurationManager m_networkManager; mutable QReadWriteLock m_lock; static Session *m_instance; }; } #endif // BITTORRENT_SESSION_H