2021-04-23 12:02:25 +03:00
|
|
|
/*
|
|
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
|
|
* Copyright (C) 2021 Vladimir Golovnev <glassez@yandex.ru>
|
|
|
|
* Copyright (C) 2010 Christian Kandeler, Christophe Dumez <chris@qbittorrent.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* In addition, as a special exception, the copyright holders give permission to
|
|
|
|
* link this program with the OpenSSL project's "OpenSSL" library (or with
|
|
|
|
* modified versions of it that use the same license as the "OpenSSL" library),
|
|
|
|
* and distribute the linked executables. You must obey the GNU General Public
|
|
|
|
* License in all respects for all of the code used other than "OpenSSL". If you
|
|
|
|
* modify file(s), you may extend this exception to your version of the file(s),
|
|
|
|
* but you are not obligated to do so. If you do not wish to do so, delete this
|
|
|
|
* exception statement from your version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "torrentfileswatcher.h"
|
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
|
|
|
|
#include <QtGlobal>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QDirIterator>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QFileSystemWatcher>
|
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
|
|
|
#include <QJsonValue>
|
|
|
|
#include <QSet>
|
|
|
|
#include <QThread>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <QVariant>
|
|
|
|
|
|
|
|
#include "base/algorithm.h"
|
|
|
|
#include "base/bittorrent/magneturi.h"
|
|
|
|
#include "base/bittorrent/torrentcontentlayout.h"
|
|
|
|
#include "base/bittorrent/session.h"
|
|
|
|
#include "base/bittorrent/torrent.h"
|
|
|
|
#include "base/bittorrent/torrentinfo.h"
|
|
|
|
#include "base/exceptions.h"
|
|
|
|
#include "base/global.h"
|
|
|
|
#include "base/logger.h"
|
|
|
|
#include "base/profile.h"
|
|
|
|
#include "base/settingsstorage.h"
|
|
|
|
#include "base/tagset.h"
|
|
|
|
#include "base/utils/fs.h"
|
2021-09-05 12:29:32 +08:00
|
|
|
#include "base/utils/io.h"
|
2021-04-23 12:02:25 +03:00
|
|
|
#include "base/utils/string.h"
|
|
|
|
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
|
2022-06-22 16:36:10 +08:00
|
|
|
const std::chrono::seconds WATCH_INTERVAL {10};
|
2021-04-23 12:02:25 +03:00
|
|
|
const int MAX_FAILED_RETRIES = 5;
|
2022-03-26 11:53:50 +08:00
|
|
|
const QString CONF_FILE_NAME = u"watched_folders.json"_qs;
|
|
|
|
|
|
|
|
const QString OPTION_ADDTORRENTPARAMS = u"add_torrent_params"_qs;
|
|
|
|
const QString OPTION_RECURSIVE = u"recursive"_qs;
|
|
|
|
|
|
|
|
const QString PARAM_CATEGORY = u"category"_qs;
|
|
|
|
const QString PARAM_TAGS = u"tags"_qs;
|
|
|
|
const QString PARAM_SAVEPATH = u"save_path"_qs;
|
|
|
|
const QString PARAM_USEDOWNLOADPATH = u"use_download_path"_qs;
|
|
|
|
const QString PARAM_DOWNLOADPATH = u"download_path"_qs;
|
|
|
|
const QString PARAM_OPERATINGMODE = u"operating_mode"_qs;
|
|
|
|
const QString PARAM_STOPPED = u"stopped"_qs;
|
|
|
|
const QString PARAM_SKIPCHECKING = u"skip_checking"_qs;
|
|
|
|
const QString PARAM_CONTENTLAYOUT = u"content_layout"_qs;
|
|
|
|
const QString PARAM_AUTOTMM = u"use_auto_tmm"_qs;
|
|
|
|
const QString PARAM_UPLOADLIMIT = u"upload_limit"_qs;
|
|
|
|
const QString PARAM_DOWNLOADLIMIT = u"download_limit"_qs;
|
|
|
|
const QString PARAM_SEEDINGTIMELIMIT = u"seeding_time_limit"_qs;
|
|
|
|
const QString PARAM_RATIOLIMIT = u"ratio_limit"_qs;
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
TagSet parseTagSet(const QJsonArray &jsonArr)
|
|
|
|
{
|
|
|
|
TagSet tags;
|
|
|
|
for (const QJsonValue &jsonVal : jsonArr)
|
|
|
|
tags.insert(jsonVal.toString());
|
|
|
|
|
|
|
|
return tags;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonArray serializeTagSet(const TagSet &tags)
|
|
|
|
{
|
|
|
|
QJsonArray arr;
|
|
|
|
for (const QString &tag : tags)
|
|
|
|
arr.append(tag);
|
|
|
|
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<bool> getOptionalBool(const QJsonObject &jsonObj, const QString &key)
|
|
|
|
{
|
|
|
|
const QJsonValue jsonVal = jsonObj.value(key);
|
|
|
|
if (jsonVal.isUndefined() || jsonVal.isNull())
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return jsonVal.toBool();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Enum>
|
|
|
|
std::optional<Enum> getOptionalEnum(const QJsonObject &jsonObj, const QString &key)
|
|
|
|
{
|
|
|
|
const QJsonValue jsonVal = jsonObj.value(key);
|
|
|
|
if (jsonVal.isUndefined() || jsonVal.isNull())
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return Utils::String::toEnum<Enum>(jsonVal.toString(), {});
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Enum>
|
|
|
|
Enum getEnum(const QJsonObject &jsonObj, const QString &key)
|
|
|
|
{
|
|
|
|
const QJsonValue jsonVal = jsonObj.value(key);
|
|
|
|
return Utils::String::toEnum<Enum>(jsonVal.toString(), {});
|
|
|
|
}
|
|
|
|
|
|
|
|
BitTorrent::AddTorrentParams parseAddTorrentParams(const QJsonObject &jsonObj)
|
|
|
|
{
|
|
|
|
BitTorrent::AddTorrentParams params;
|
|
|
|
params.category = jsonObj.value(PARAM_CATEGORY).toString();
|
|
|
|
params.tags = parseTagSet(jsonObj.value(PARAM_TAGS).toArray());
|
2022-02-08 06:03:48 +03:00
|
|
|
params.savePath = Path(jsonObj.value(PARAM_SAVEPATH).toString());
|
2021-05-20 10:36:44 +03:00
|
|
|
params.useDownloadPath = getOptionalBool(jsonObj, PARAM_USEDOWNLOADPATH);
|
2022-02-08 06:03:48 +03:00
|
|
|
params.downloadPath = Path(jsonObj.value(PARAM_DOWNLOADPATH).toString());
|
2021-04-23 12:02:25 +03:00
|
|
|
params.addForced = (getEnum<BitTorrent::TorrentOperatingMode>(jsonObj, PARAM_OPERATINGMODE) == BitTorrent::TorrentOperatingMode::Forced);
|
|
|
|
params.addPaused = getOptionalBool(jsonObj, PARAM_STOPPED);
|
2021-09-09 07:00:51 +03:00
|
|
|
params.skipChecking = jsonObj.value(PARAM_SKIPCHECKING).toBool();
|
2021-04-23 12:02:25 +03:00
|
|
|
params.contentLayout = getOptionalEnum<BitTorrent::TorrentContentLayout>(jsonObj, PARAM_CONTENTLAYOUT);
|
|
|
|
params.useAutoTMM = getOptionalBool(jsonObj, PARAM_AUTOTMM);
|
|
|
|
params.uploadLimit = jsonObj.value(PARAM_UPLOADLIMIT).toInt(-1);
|
|
|
|
params.downloadLimit = jsonObj.value(PARAM_DOWNLOADLIMIT).toInt(-1);
|
|
|
|
params.seedingTimeLimit = jsonObj.value(PARAM_SEEDINGTIMELIMIT).toInt(BitTorrent::Torrent::USE_GLOBAL_SEEDING_TIME);
|
|
|
|
params.ratioLimit = jsonObj.value(PARAM_RATIOLIMIT).toDouble(BitTorrent::Torrent::USE_GLOBAL_RATIO);
|
|
|
|
|
|
|
|
return params;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject serializeAddTorrentParams(const BitTorrent::AddTorrentParams ¶ms)
|
|
|
|
{
|
|
|
|
QJsonObject jsonObj {
|
|
|
|
{PARAM_CATEGORY, params.category},
|
|
|
|
{PARAM_TAGS, serializeTagSet(params.tags)},
|
2022-02-08 06:03:48 +03:00
|
|
|
{PARAM_SAVEPATH, params.savePath.data()},
|
|
|
|
{PARAM_DOWNLOADPATH, params.downloadPath.data()},
|
2021-04-23 12:02:25 +03:00
|
|
|
{PARAM_OPERATINGMODE, Utils::String::fromEnum(params.addForced
|
|
|
|
? BitTorrent::TorrentOperatingMode::Forced : BitTorrent::TorrentOperatingMode::AutoManaged)},
|
2021-09-09 07:00:51 +03:00
|
|
|
{PARAM_SKIPCHECKING, params.skipChecking},
|
2021-04-23 12:02:25 +03:00
|
|
|
{PARAM_UPLOADLIMIT, params.uploadLimit},
|
|
|
|
{PARAM_DOWNLOADLIMIT, params.downloadLimit},
|
|
|
|
{PARAM_SEEDINGTIMELIMIT, params.seedingTimeLimit},
|
|
|
|
{PARAM_RATIOLIMIT, params.ratioLimit}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (params.addPaused)
|
|
|
|
jsonObj[PARAM_STOPPED] = *params.addPaused;
|
|
|
|
if (params.contentLayout)
|
|
|
|
jsonObj[PARAM_CONTENTLAYOUT] = Utils::String::fromEnum(*params.contentLayout);
|
|
|
|
if (params.useAutoTMM)
|
|
|
|
jsonObj[PARAM_AUTOTMM] = *params.useAutoTMM;
|
2021-05-20 10:36:44 +03:00
|
|
|
if (params.useDownloadPath)
|
|
|
|
jsonObj[PARAM_USEDOWNLOADPATH] = *params.useDownloadPath;
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
return jsonObj;
|
|
|
|
}
|
|
|
|
|
|
|
|
TorrentFilesWatcher::WatchedFolderOptions parseWatchedFolderOptions(const QJsonObject &jsonObj)
|
|
|
|
{
|
|
|
|
TorrentFilesWatcher::WatchedFolderOptions options;
|
|
|
|
options.addTorrentParams = parseAddTorrentParams(jsonObj.value(OPTION_ADDTORRENTPARAMS).toObject());
|
|
|
|
options.recursive = jsonObj.value(OPTION_RECURSIVE).toBool();
|
|
|
|
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject serializeWatchedFolderOptions(const TorrentFilesWatcher::WatchedFolderOptions &options)
|
|
|
|
{
|
|
|
|
return {
|
|
|
|
{OPTION_ADDTORRENTPARAMS, serializeAddTorrentParams(options.addTorrentParams)},
|
|
|
|
{OPTION_RECURSIVE, options.recursive}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class TorrentFilesWatcher::Worker final : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
2021-06-29 14:45:23 +08:00
|
|
|
Q_DISABLE_COPY_MOVE(Worker)
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
public:
|
|
|
|
Worker();
|
|
|
|
|
|
|
|
public slots:
|
2022-02-08 06:03:48 +03:00
|
|
|
void setWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options);
|
|
|
|
void removeWatchedFolder(const Path &path);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
signals:
|
|
|
|
void magnetFound(const BitTorrent::MagnetUri &magnetURI, const BitTorrent::AddTorrentParams &addTorrentParams);
|
|
|
|
void torrentFound(const BitTorrent::TorrentInfo &torrentInfo, const BitTorrent::AddTorrentParams &addTorrentParams);
|
|
|
|
|
|
|
|
private:
|
|
|
|
void onTimeout();
|
2022-02-08 06:03:48 +03:00
|
|
|
void scheduleWatchedFolderProcessing(const Path &path);
|
|
|
|
void processWatchedFolder(const Path &path);
|
|
|
|
void processFolder(const Path &path, const Path &watchedFolderPath, const TorrentFilesWatcher::WatchedFolderOptions &options);
|
2021-04-23 12:02:25 +03:00
|
|
|
void processFailedTorrents();
|
2022-02-08 06:03:48 +03:00
|
|
|
void addWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options);
|
|
|
|
void updateWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
QFileSystemWatcher *m_watcher = nullptr;
|
|
|
|
QTimer *m_watchTimer = nullptr;
|
2022-02-08 06:03:48 +03:00
|
|
|
QHash<Path, TorrentFilesWatcher::WatchedFolderOptions> m_watchedFolders;
|
|
|
|
QSet<Path> m_watchedByTimeoutFolders;
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
// Failed torrents
|
|
|
|
QTimer *m_retryTorrentTimer = nullptr;
|
2022-02-08 06:03:48 +03:00
|
|
|
QHash<Path, QHash<Path, int>> m_failedTorrents;
|
2021-04-23 12:02:25 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
TorrentFilesWatcher *TorrentFilesWatcher::m_instance = nullptr;
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::initInstance()
|
|
|
|
{
|
|
|
|
if (!m_instance)
|
|
|
|
m_instance = new TorrentFilesWatcher;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::freeInstance()
|
|
|
|
{
|
|
|
|
delete m_instance;
|
|
|
|
m_instance = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
TorrentFilesWatcher *TorrentFilesWatcher::instance()
|
|
|
|
{
|
|
|
|
return m_instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
TorrentFilesWatcher::TorrentFilesWatcher(QObject *parent)
|
|
|
|
: QObject {parent}
|
2022-12-08 08:37:14 +03:00
|
|
|
, m_ioThread {new QThread}
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-07-04 12:48:21 +03:00
|
|
|
const auto *btSession = BitTorrent::Session::instance();
|
|
|
|
if (btSession->isRestored())
|
|
|
|
initWorker();
|
|
|
|
else
|
|
|
|
connect(btSession, &BitTorrent::Session::restored, this, &TorrentFilesWatcher::initWorker);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
load();
|
|
|
|
}
|
|
|
|
|
2022-07-04 12:48:21 +03:00
|
|
|
void TorrentFilesWatcher::initWorker()
|
|
|
|
{
|
|
|
|
Q_ASSERT(!m_asyncWorker);
|
|
|
|
|
|
|
|
m_asyncWorker = new TorrentFilesWatcher::Worker;
|
|
|
|
|
|
|
|
connect(m_asyncWorker, &TorrentFilesWatcher::Worker::magnetFound, this, &TorrentFilesWatcher::onMagnetFound);
|
|
|
|
connect(m_asyncWorker, &TorrentFilesWatcher::Worker::torrentFound, this, &TorrentFilesWatcher::onTorrentFound);
|
|
|
|
|
2022-12-08 08:37:14 +03:00
|
|
|
m_asyncWorker->moveToThread(m_ioThread.get());
|
|
|
|
connect(m_ioThread.get(), &QThread::finished, m_asyncWorker, &QObject::deleteLater);
|
2022-07-04 12:48:21 +03:00
|
|
|
m_ioThread->start();
|
|
|
|
|
|
|
|
for (auto it = m_watchedFolders.cbegin(); it != m_watchedFolders.cend(); ++it)
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(m_asyncWorker, [this, path = it.key(), options = it.value()]()
|
|
|
|
{
|
|
|
|
m_asyncWorker->setWatchedFolder(path, options);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-23 12:02:25 +03:00
|
|
|
void TorrentFilesWatcher::load()
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
QFile confFile {(specialFolderLocation(SpecialFolder::Config) / Path(CONF_FILE_NAME)).data()};
|
2021-04-23 12:02:25 +03:00
|
|
|
if (!confFile.exists())
|
|
|
|
{
|
|
|
|
loadLegacy();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!confFile.open(QFile::ReadOnly))
|
|
|
|
{
|
|
|
|
LogMsg(tr("Couldn't load Watched Folders configuration from %1. Error: %2")
|
|
|
|
.arg(confFile.fileName(), confFile.errorString()), Log::WARNING);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonParseError jsonError;
|
|
|
|
const QJsonDocument jsonDoc = QJsonDocument::fromJson(confFile.readAll(), &jsonError);
|
|
|
|
if (jsonError.error != QJsonParseError::NoError)
|
|
|
|
{
|
2021-06-27 13:01:03 +04:00
|
|
|
LogMsg(tr("Couldn't parse Watched Folders configuration from %1. Error: %2")
|
2021-04-23 12:02:25 +03:00
|
|
|
.arg(confFile.fileName(), jsonError.errorString()), Log::WARNING);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!jsonDoc.isObject())
|
|
|
|
{
|
|
|
|
LogMsg(tr("Couldn't load Watched Folders configuration from %1. Invalid data format.")
|
|
|
|
.arg(confFile.fileName()), Log::WARNING);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QJsonObject jsonObj = jsonDoc.object();
|
|
|
|
for (auto it = jsonObj.constBegin(); it != jsonObj.constEnd(); ++it)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path watchedFolder {it.key()};
|
2021-04-23 12:02:25 +03:00
|
|
|
const WatchedFolderOptions options = parseWatchedFolderOptions(it.value().toObject());
|
|
|
|
try
|
|
|
|
{
|
|
|
|
doSetWatchedFolder(watchedFolder, options);
|
|
|
|
}
|
|
|
|
catch (const InvalidArgument &err)
|
|
|
|
{
|
|
|
|
LogMsg(err.message(), Log::WARNING);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::loadLegacy()
|
|
|
|
{
|
2022-03-04 14:24:14 +08:00
|
|
|
const auto dirs = SettingsStorage::instance()->loadValue<QVariantHash>(u"Preferences/Downloads/ScanDirsV2"_qs);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
for (auto it = dirs.cbegin(); it != dirs.cend(); ++it)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path watchedFolder {it.key()};
|
2021-04-23 12:02:25 +03:00
|
|
|
BitTorrent::AddTorrentParams params;
|
2022-02-08 06:03:48 +03:00
|
|
|
if (it.value().type() == QVariant::Int)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
if (it.value().toInt() == 0)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
params.savePath = watchedFolder;
|
|
|
|
params.useAutoTMM = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path customSavePath {it.value().toString()};
|
2021-04-23 12:02:25 +03:00
|
|
|
params.savePath = customSavePath;
|
|
|
|
params.useAutoTMM = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
doSetWatchedFolder(watchedFolder, {params, false});
|
|
|
|
}
|
|
|
|
catch (const InvalidArgument &err)
|
|
|
|
{
|
|
|
|
LogMsg(err.message(), Log::WARNING);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
store();
|
2022-03-04 14:24:14 +08:00
|
|
|
SettingsStorage::instance()->removeValue(u"Preferences/Downloads/ScanDirsV2"_qs);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::store() const
|
|
|
|
{
|
|
|
|
QJsonObject jsonObj;
|
|
|
|
for (auto it = m_watchedFolders.cbegin(); it != m_watchedFolders.cend(); ++it)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path &watchedFolder = it.key();
|
2021-04-23 12:02:25 +03:00
|
|
|
const WatchedFolderOptions &options = it.value();
|
2022-02-08 06:03:48 +03:00
|
|
|
jsonObj[watchedFolder.data()] = serializeWatchedFolderOptions(options);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path path = specialFolderLocation(SpecialFolder::Config) / Path(CONF_FILE_NAME);
|
2021-04-23 12:02:25 +03:00
|
|
|
const QByteArray data = QJsonDocument(jsonObj).toJson();
|
2021-09-05 12:29:32 +08:00
|
|
|
const nonstd::expected<void, QString> result = Utils::IO::saveToFile(path, data);
|
|
|
|
if (!result)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
LogMsg(tr("Couldn't store Watched Folders configuration to %1. Error: %2")
|
2022-02-08 06:03:48 +03:00
|
|
|
.arg(path.toString(), result.error()), Log::WARNING);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
QHash<Path, TorrentFilesWatcher::WatchedFolderOptions> TorrentFilesWatcher::folders() const
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
return m_watchedFolders;
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::setWatchedFolder(const Path &path, const WatchedFolderOptions &options)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
doSetWatchedFolder(path, options);
|
|
|
|
store();
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::doSetWatchedFolder(const Path &path, const WatchedFolderOptions &options)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
if (path.isEmpty())
|
|
|
|
throw InvalidArgument(tr("Watched folder Path cannot be empty."));
|
|
|
|
|
|
|
|
if (path.isRelative())
|
|
|
|
throw InvalidArgument(tr("Watched folder Path cannot be relative."));
|
|
|
|
|
|
|
|
m_watchedFolders[path] = options;
|
2021-04-23 12:02:25 +03:00
|
|
|
|
2022-07-04 12:48:21 +03:00
|
|
|
if (m_asyncWorker)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-07-04 12:48:21 +03:00
|
|
|
QMetaObject::invokeMethod(m_asyncWorker, [this, path, options]()
|
|
|
|
{
|
|
|
|
m_asyncWorker->setWatchedFolder(path, options);
|
|
|
|
});
|
|
|
|
}
|
2021-04-23 12:02:25 +03:00
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
emit watchedFolderSet(path, options);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::removeWatchedFolder(const Path &path)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
if (m_watchedFolders.remove(path))
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-07-04 12:48:21 +03:00
|
|
|
if (m_asyncWorker)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-07-04 12:48:21 +03:00
|
|
|
QMetaObject::invokeMethod(m_asyncWorker, [this, path]()
|
|
|
|
{
|
|
|
|
m_asyncWorker->removeWatchedFolder(path);
|
|
|
|
});
|
|
|
|
}
|
2021-04-23 12:02:25 +03:00
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
emit watchedFolderRemoved(path);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
store();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::onMagnetFound(const BitTorrent::MagnetUri &magnetURI
|
|
|
|
, const BitTorrent::AddTorrentParams &addTorrentParams)
|
|
|
|
{
|
|
|
|
BitTorrent::Session::instance()->addTorrent(magnetURI, addTorrentParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::onTorrentFound(const BitTorrent::TorrentInfo &torrentInfo
|
|
|
|
, const BitTorrent::AddTorrentParams &addTorrentParams)
|
|
|
|
{
|
|
|
|
BitTorrent::Session::instance()->addTorrent(torrentInfo, addTorrentParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
TorrentFilesWatcher::Worker::Worker()
|
|
|
|
: m_watcher {new QFileSystemWatcher(this)}
|
|
|
|
, m_watchTimer {new QTimer(this)}
|
|
|
|
, m_retryTorrentTimer {new QTimer(this)}
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
connect(m_watcher, &QFileSystemWatcher::directoryChanged, this, [this](const QString &path)
|
|
|
|
{
|
|
|
|
scheduleWatchedFolderProcessing(Path(path));
|
|
|
|
});
|
2021-04-23 12:02:25 +03:00
|
|
|
connect(m_watchTimer, &QTimer::timeout, this, &Worker::onTimeout);
|
|
|
|
|
|
|
|
connect(m_retryTorrentTimer, &QTimer::timeout, this, &Worker::processFailedTorrents);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::Worker::onTimeout()
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
for (const Path &path : asConst(m_watchedByTimeoutFolders))
|
2021-04-23 12:02:25 +03:00
|
|
|
processWatchedFolder(path);
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::setWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
if (m_watchedFolders.contains(path))
|
|
|
|
updateWatchedFolder(path, options);
|
|
|
|
else
|
|
|
|
addWatchedFolder(path, options);
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::removeWatchedFolder(const Path &path)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
m_watchedFolders.remove(path);
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
m_watcher->removePath(path.data());
|
2021-04-23 12:02:25 +03:00
|
|
|
m_watchedByTimeoutFolders.remove(path);
|
|
|
|
if (m_watchedByTimeoutFolders.isEmpty())
|
|
|
|
m_watchTimer->stop();
|
|
|
|
|
|
|
|
m_failedTorrents.remove(path);
|
|
|
|
if (m_failedTorrents.isEmpty())
|
|
|
|
m_retryTorrentTimer->stop();
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::scheduleWatchedFolderProcessing(const Path &path)
|
2021-08-07 12:33:29 +03:00
|
|
|
{
|
2022-06-22 16:36:10 +08:00
|
|
|
QTimer::singleShot(2s, this, [this, path]()
|
2021-08-07 12:33:29 +03:00
|
|
|
{
|
|
|
|
processWatchedFolder(path);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::processWatchedFolder(const Path &path)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
const TorrentFilesWatcher::WatchedFolderOptions options = m_watchedFolders.value(path);
|
|
|
|
processFolder(path, path, options);
|
|
|
|
|
|
|
|
if (!m_failedTorrents.empty() && !m_retryTorrentTimer->isActive())
|
|
|
|
m_retryTorrentTimer->start(WATCH_INTERVAL);
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::processFolder(const Path &path, const Path &watchedFolderPath
|
2021-04-23 12:02:25 +03:00
|
|
|
, const TorrentFilesWatcher::WatchedFolderOptions &options)
|
|
|
|
{
|
2022-03-04 14:24:14 +08:00
|
|
|
QDirIterator dirIter {path.data(), {u"*.torrent"_qs, u"*.magnet"_qs}, QDir::Files};
|
2021-04-23 12:02:25 +03:00
|
|
|
while (dirIter.hasNext())
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path filePath {dirIter.next()};
|
2021-04-23 12:02:25 +03:00
|
|
|
BitTorrent::AddTorrentParams addTorrentParams = options.addTorrentParams;
|
|
|
|
if (path != watchedFolderPath)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path subdirPath = watchedFolderPath.relativePathOf(path);
|
2022-01-11 07:25:30 +03:00
|
|
|
const bool useAutoTMM = addTorrentParams.useAutoTMM.value_or(!BitTorrent::Session::instance()->isAutoTMMDisabledByDefault());
|
|
|
|
if (useAutoTMM)
|
2021-10-22 05:51:11 +03:00
|
|
|
{
|
|
|
|
addTorrentParams.category = addTorrentParams.category.isEmpty()
|
2022-03-26 11:53:50 +08:00
|
|
|
? subdirPath.data() : (addTorrentParams.category + u'/' + subdirPath.data());
|
2021-10-22 05:51:11 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
addTorrentParams.savePath = addTorrentParams.savePath / subdirPath;
|
2021-10-22 05:51:11 +03:00
|
|
|
}
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
2022-03-26 11:53:50 +08:00
|
|
|
if (filePath.hasExtension(u".magnet"_qs))
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
QFile file {filePath.data()};
|
2021-04-23 12:02:25 +03:00
|
|
|
if (file.open(QIODevice::ReadOnly | QIODevice::Text))
|
|
|
|
{
|
2022-06-09 11:18:41 +08:00
|
|
|
while (!file.atEnd())
|
|
|
|
{
|
|
|
|
const auto line = QString::fromLatin1(file.readLine()).trimmed();
|
|
|
|
emit magnetFound(BitTorrent::MagnetUri(line), addTorrentParams);
|
|
|
|
}
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
file.close();
|
2022-02-08 06:03:48 +03:00
|
|
|
Utils::Fs::removeFile(filePath);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LogMsg(tr("Failed to open magnet file: %1").arg(file.errorString()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-10-06 21:45:37 +03:00
|
|
|
const nonstd::expected<BitTorrent::TorrentInfo, QString> result = BitTorrent::TorrentInfo::loadFromFile(filePath);
|
|
|
|
if (result)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2021-10-06 21:45:37 +03:00
|
|
|
emit torrentFound(result.value(), addTorrentParams);
|
2022-02-08 06:03:48 +03:00
|
|
|
Utils::Fs::removeFile(filePath);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!m_failedTorrents.value(path).contains(filePath))
|
|
|
|
{
|
|
|
|
m_failedTorrents[path][filePath] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.recursive)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
QDirIterator dirIter {path.data(), (QDir::Dirs | QDir::NoDot | QDir::NoDotDot)};
|
2021-04-23 12:02:25 +03:00
|
|
|
while (dirIter.hasNext())
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path folderPath {dirIter.next()};
|
2021-04-23 12:02:25 +03:00
|
|
|
// Skip processing of subdirectory that is explicitly set as watched folder
|
|
|
|
if (!m_watchedFolders.contains(folderPath))
|
|
|
|
processFolder(folderPath, watchedFolderPath, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentFilesWatcher::Worker::processFailedTorrents()
|
|
|
|
{
|
|
|
|
// Check which torrents are still partial
|
2022-02-08 06:03:48 +03:00
|
|
|
Algorithm::removeIf(m_failedTorrents, [this](const Path &watchedFolderPath, QHash<Path, int> &partialTorrents)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
const TorrentFilesWatcher::WatchedFolderOptions options = m_watchedFolders.value(watchedFolderPath);
|
2022-02-08 06:03:48 +03:00
|
|
|
Algorithm::removeIf(partialTorrents, [this, &watchedFolderPath, &options](const Path &torrentPath, int &value)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
if (!torrentPath.exists())
|
2021-04-23 12:02:25 +03:00
|
|
|
return true;
|
|
|
|
|
2021-10-06 21:45:37 +03:00
|
|
|
const nonstd::expected<BitTorrent::TorrentInfo, QString> result = BitTorrent::TorrentInfo::loadFromFile(torrentPath);
|
|
|
|
if (result)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
BitTorrent::AddTorrentParams addTorrentParams = options.addTorrentParams;
|
2022-02-08 06:03:48 +03:00
|
|
|
if (torrentPath != watchedFolderPath)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const Path subdirPath = watchedFolderPath.relativePathOf(torrentPath);
|
2022-01-11 07:25:30 +03:00
|
|
|
const bool useAutoTMM = addTorrentParams.useAutoTMM.value_or(!BitTorrent::Session::instance()->isAutoTMMDisabledByDefault());
|
|
|
|
if (useAutoTMM)
|
2021-10-22 05:51:11 +03:00
|
|
|
{
|
|
|
|
addTorrentParams.category = addTorrentParams.category.isEmpty()
|
2022-03-26 11:53:50 +08:00
|
|
|
? subdirPath.data() : (addTorrentParams.category + u'/' + subdirPath.data());
|
2021-10-22 05:51:11 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
addTorrentParams.savePath = addTorrentParams.savePath / subdirPath;
|
2021-10-22 05:51:11 +03:00
|
|
|
}
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
2021-10-06 21:45:37 +03:00
|
|
|
emit torrentFound(result.value(), addTorrentParams);
|
2022-02-08 06:03:48 +03:00
|
|
|
Utils::Fs::removeFile(torrentPath);
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value >= MAX_FAILED_RETRIES)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
LogMsg(tr("Rejecting failed torrent file: %1").arg(torrentPath.toString()));
|
2022-03-23 23:56:47 +08:00
|
|
|
Utils::Fs::renameFile(torrentPath, (torrentPath + u".qbt_rejected"));
|
2021-04-23 12:02:25 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
++value;
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (partialTorrents.isEmpty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
// Stop the partial timer if necessary
|
|
|
|
if (m_failedTorrents.empty())
|
|
|
|
m_retryTorrentTimer->stop();
|
|
|
|
else
|
|
|
|
m_retryTorrentTimer->start(WATCH_INTERVAL);
|
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::addWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
// Check if the `path` points to a network file system or not
|
2021-06-28 14:48:17 +03:00
|
|
|
if (Utils::Fs::isNetworkFileSystem(path) || options.recursive)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
m_watchedByTimeoutFolders.insert(path);
|
|
|
|
if (!m_watchTimer->isActive())
|
|
|
|
m_watchTimer->start(WATCH_INTERVAL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
m_watcher->addPath(path.data());
|
2021-08-07 12:33:29 +03:00
|
|
|
scheduleWatchedFolderProcessing(path);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
m_watchedFolders[path] = options;
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
LogMsg(tr("Watching folder: \"%1\"").arg(path.toString()));
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
void TorrentFilesWatcher::Worker::updateWatchedFolder(const Path &path, const TorrentFilesWatcher::WatchedFolderOptions &options)
|
2021-04-23 12:02:25 +03:00
|
|
|
{
|
|
|
|
const bool recursiveModeChanged = (m_watchedFolders[path].recursive != options.recursive);
|
|
|
|
if (recursiveModeChanged && !Utils::Fs::isNetworkFileSystem(path))
|
|
|
|
{
|
|
|
|
if (options.recursive)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
m_watcher->removePath(path.data());
|
2021-04-23 12:02:25 +03:00
|
|
|
|
|
|
|
m_watchedByTimeoutFolders.insert(path);
|
|
|
|
if (!m_watchTimer->isActive())
|
|
|
|
m_watchTimer->start(WATCH_INTERVAL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_watchedByTimeoutFolders.remove(path);
|
|
|
|
if (m_watchedByTimeoutFolders.isEmpty())
|
|
|
|
m_watchTimer->stop();
|
|
|
|
|
2022-02-08 06:03:48 +03:00
|
|
|
m_watcher->addPath(path.data());
|
2021-08-07 12:33:29 +03:00
|
|
|
scheduleWatchedFolderProcessing(path);
|
2021-04-23 12:02:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_watchedFolders[path] = options;
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "torrentfileswatcher.moc"
|