/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2017 Vladimir Golovnev * * 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 "rss_autodownloader.h" #include #include #include #include #include #include #include #include #include #include #include #include "../bittorrent/magneturi.h" #include "../bittorrent/session.h" #include "../asyncfilestorage.h" #include "../global.h" #include "../logger.h" #include "../profile.h" #include "../settingsstorage.h" #include "../tristatebool.h" #include "../utils/fs.h" #include "rss_article.h" #include "rss_autodownloadrule.h" #include "rss_feed.h" #include "rss_folder.h" #include "rss_session.h" struct ProcessingJob { QString feedURL; QVariantHash articleData; }; const QString ConfFolderName(QStringLiteral("rss")); const QString RulesFileName(QStringLiteral("download_rules.json")); const QString SettingsKey_ProcessingEnabled(QStringLiteral("RSS/AutoDownloader/EnableProcessing")); const QString SettingsKey_SmartEpisodeFilter(QStringLiteral("RSS/AutoDownloader/SmartEpisodeFilter")); const QString SettingsKey_DownloadRepacks(QStringLiteral("RSS/AutoDownloader/DownloadRepacks")); namespace { QVector rulesFromJSON(const QByteArray &jsonData) { QJsonParseError jsonError; QJsonDocument jsonDoc = QJsonDocument::fromJson(jsonData, &jsonError); if (jsonError.error != QJsonParseError::NoError) throw RSS::ParsingError(jsonError.errorString()); if (!jsonDoc.isObject()) throw RSS::ParsingError(RSS::AutoDownloader::tr("Invalid data format.")); const QJsonObject jsonObj {jsonDoc.object()}; QVector rules; for (auto it = jsonObj.begin(); it != jsonObj.end(); ++it) { const QJsonValue jsonVal {it.value()}; if (!jsonVal.isObject()) throw RSS::ParsingError(RSS::AutoDownloader::tr("Invalid data format.")); rules.append(RSS::AutoDownloadRule::fromJsonObject(jsonVal.toObject(), it.key())); } return rules; } } using namespace RSS; QPointer AutoDownloader::m_instance = nullptr; QString computeSmartFilterRegex(const QStringList &filters) { return QString("(?:_|\\b)(?:%1)(?:_|\\b)").arg(filters.join(QString(")|(?:"))); } AutoDownloader::AutoDownloader() : m_processingEnabled(SettingsStorage::instance()->loadValue(SettingsKey_ProcessingEnabled, false).toBool()) , m_processingTimer(new QTimer(this)) , m_ioThread(new QThread(this)) { Q_ASSERT(!m_instance); // only one instance is allowed m_instance = this; m_fileStorage = new AsyncFileStorage( Utils::Fs::expandPathAbs(specialFolderLocation(SpecialFolder::Config) + ConfFolderName)); if (!m_fileStorage) throw std::runtime_error("Directory for RSS AutoDownloader data is unavailable."); m_fileStorage->moveToThread(m_ioThread); connect(m_ioThread, &QThread::finished, m_fileStorage, &AsyncFileStorage::deleteLater); connect(m_fileStorage, &AsyncFileStorage::failed, [](const QString &fileName, const QString &errorString) { LogMsg(tr("Couldn't save RSS AutoDownloader data in %1. Error: %2") .arg(fileName, errorString), Log::CRITICAL); }); m_ioThread->start(); connect(BitTorrent::Session::instance(), &BitTorrent::Session::downloadFromUrlFinished , this, &AutoDownloader::handleTorrentDownloadFinished); connect(BitTorrent::Session::instance(), &BitTorrent::Session::downloadFromUrlFailed , this, &AutoDownloader::handleTorrentDownloadFailed); // initialise the smart episode regex const QString regex = computeSmartFilterRegex(smartEpisodeFilters()); m_smartEpisodeRegex = QRegularExpression(regex, QRegularExpression::CaseInsensitiveOption | QRegularExpression::ExtendedPatternSyntaxOption | QRegularExpression::UseUnicodePropertiesOption); load(); m_processingTimer->setSingleShot(true); connect(m_processingTimer, &QTimer::timeout, this, &AutoDownloader::process); if (m_processingEnabled) startProcessing(); } AutoDownloader::~AutoDownloader() { store(); m_ioThread->quit(); m_ioThread->wait(); } AutoDownloader *AutoDownloader::instance() { return m_instance; } bool AutoDownloader::hasRule(const QString &ruleName) const { return m_rules.contains(ruleName); } AutoDownloadRule AutoDownloader::ruleByName(const QString &ruleName) const { return m_rules.value(ruleName, AutoDownloadRule("Unknown Rule")); } QList AutoDownloader::rules() const { return m_rules.values(); } void AutoDownloader::insertRule(const AutoDownloadRule &rule) { if (!hasRule(rule.name())) { // Insert new rule setRule_impl(rule); m_dirty = true; store(); emit ruleAdded(rule.name()); resetProcessingQueue(); } else if (ruleByName(rule.name()) != rule) { // Update existing rule setRule_impl(rule); m_dirty = true; storeDeferred(); emit ruleChanged(rule.name()); resetProcessingQueue(); } } bool AutoDownloader::renameRule(const QString &ruleName, const QString &newRuleName) { if (!hasRule(ruleName)) return false; if (hasRule(newRuleName)) return false; m_rules.insert(newRuleName, m_rules.take(ruleName)); m_dirty = true; store(); emit ruleRenamed(newRuleName, ruleName); return true; } void AutoDownloader::removeRule(const QString &ruleName) { if (m_rules.contains(ruleName)) { emit ruleAboutToBeRemoved(ruleName); m_rules.remove(ruleName); m_dirty = true; store(); } } QByteArray AutoDownloader::exportRules(AutoDownloader::RulesFileFormat format) const { switch (format) { case RulesFileFormat::Legacy: return exportRulesToLegacyFormat(); default: return exportRulesToJSONFormat(); } } void AutoDownloader::importRules(const QByteArray &data, AutoDownloader::RulesFileFormat format) { switch (format) { case RulesFileFormat::Legacy: importRulesFromLegacyFormat(data); break; default: importRulesFromJSONFormat(data); } } QByteArray AutoDownloader::exportRulesToJSONFormat() const { QJsonObject jsonObj; for (const auto &rule : copyAsConst(rules())) jsonObj.insert(rule.name(), rule.toJsonObject()); return QJsonDocument(jsonObj).toJson(); } void AutoDownloader::importRulesFromJSONFormat(const QByteArray &data) { for (const auto &rule : copyAsConst(rulesFromJSON(data))) insertRule(rule); } QByteArray AutoDownloader::exportRulesToLegacyFormat() const { QVariantHash dict; for (const auto &rule : copyAsConst(rules())) dict[rule.name()] = rule.toLegacyDict(); QByteArray data; QDataStream out(&data, QIODevice::WriteOnly); out.setVersion(QDataStream::Qt_4_5); out << dict; return data; } void AutoDownloader::importRulesFromLegacyFormat(const QByteArray &data) { QDataStream in(data); in.setVersion(QDataStream::Qt_4_5); QVariantHash dict; in >> dict; if (in.status() != QDataStream::Ok) throw ParsingError(tr("Invalid data format")); for (const QVariant &val : qAsConst(dict)) insertRule(AutoDownloadRule::fromLegacyDict(val.toHash())); } QStringList AutoDownloader::smartEpisodeFilters() const { const QVariant filtersSetting = SettingsStorage::instance()->loadValue(SettingsKey_SmartEpisodeFilter); if (filtersSetting.isNull()) { QStringList filters = { "s(\\d+)e(\\d+)", // Format 1: s01e01 "(\\d+)x(\\d+)", // Format 2: 01x01 "(\\d{4}[.\\-]\\d{1,2}[.\\-]\\d{1,2})", // Format 3: 2017.01.01 "(\\d{1,2}[.\\-]\\d{1,2}[.\\-]\\d{4})" // Format 4: 01.01.2017 }; return filters; } return filtersSetting.toStringList(); } QRegularExpression AutoDownloader::smartEpisodeRegex() const { return m_smartEpisodeRegex; } void AutoDownloader::setSmartEpisodeFilters(const QStringList &filters) { SettingsStorage::instance()->storeValue(SettingsKey_SmartEpisodeFilter, filters); const QString regex = computeSmartFilterRegex(filters); m_smartEpisodeRegex.setPattern(regex); } bool AutoDownloader::downloadRepacks() const { return SettingsStorage::instance()->loadValue(SettingsKey_DownloadRepacks, true).toBool(); } void AutoDownloader::setDownloadRepacks(const bool downloadRepacks) { SettingsStorage::instance()->storeValue(SettingsKey_DownloadRepacks, downloadRepacks); } void AutoDownloader::process() { if (m_processingQueue.isEmpty()) return; // processing was disabled processJob(m_processingQueue.takeFirst()); if (!m_processingQueue.isEmpty()) // Schedule to process the next torrent (if any) m_processingTimer->start(); } void AutoDownloader::handleTorrentDownloadFinished(const QString &url) { auto job = m_waitingJobs.take(url); if (!job) return; if (Feed *feed = Session::instance()->feedByURL(job->feedURL)) if (Article *article = feed->articleByGUID(job->articleData.value(Article::KeyId).toString())) article->markAsRead(); } void AutoDownloader::handleTorrentDownloadFailed(const QString &url) { m_waitingJobs.remove(url); // TODO: Re-schedule job here. } void AutoDownloader::handleNewArticle(Article *article) { if (!article->isRead() && !article->torrentUrl().isEmpty()) addJobForArticle(article); } void AutoDownloader::setRule_impl(const AutoDownloadRule &rule) { m_rules.insert(rule.name(), rule); } void AutoDownloader::addJobForArticle(Article *article) { const QString torrentURL = article->torrentUrl(); if (m_waitingJobs.contains(torrentURL)) return; QSharedPointer job(new ProcessingJob); job->feedURL = article->feed()->url(); job->articleData = article->data(); m_processingQueue.append(job); if (!m_processingTimer->isActive()) m_processingTimer->start(); } void AutoDownloader::processJob(const QSharedPointer &job) { for (AutoDownloadRule &rule: m_rules) { if (!rule.isEnabled()) continue; if (!rule.feedURLs().contains(job->feedURL)) continue; if (!rule.accepts(job->articleData)) continue; m_dirty = true; storeDeferred(); BitTorrent::AddTorrentParams params; params.savePath = rule.savePath(); params.category = rule.assignedCategory(); params.addPaused = rule.addPaused(); if (!rule.savePath().isEmpty()) params.useAutoTMM = TriStateBool::False; auto torrentURL = job->articleData.value(Article::KeyTorrentURL).toString(); BitTorrent::Session::instance()->addTorrent(torrentURL, params); if (BitTorrent::MagnetUri(torrentURL).isValid()) { if (Feed *feed = Session::instance()->feedByURL(job->feedURL)) { if (Article *article = feed->articleByGUID(job->articleData.value(Article::KeyId).toString())) article->markAsRead(); } } else { // waiting for torrent file downloading // normalize URL string via QUrl since DownloadManager do it m_waitingJobs.insert(QUrl(torrentURL).toString(), job); } return; } } void AutoDownloader::load() { QFile rulesFile(m_fileStorage->storageDir().absoluteFilePath(RulesFileName)); if (!rulesFile.exists()) loadRulesLegacy(); else if (rulesFile.open(QFile::ReadOnly)) loadRules(rulesFile.readAll()); else LogMsg(tr("Couldn't read RSS AutoDownloader rules from %1. Error: %2") .arg(rulesFile.fileName(), rulesFile.errorString()), Log::CRITICAL); } void AutoDownloader::loadRules(const QByteArray &data) { try { const auto rules = rulesFromJSON(data); for (const auto &rule : rules) setRule_impl(rule); } catch (const ParsingError &error) { LogMsg(tr("Couldn't load RSS AutoDownloader rules. Reason: %1") .arg(error.message()), Log::CRITICAL); } } void AutoDownloader::loadRulesLegacy() { SettingsPtr settings = Profile::instance().applicationSettings(QStringLiteral("qBittorrent-rss")); QVariantHash rules = settings->value(QStringLiteral("download_rules")).toHash(); foreach (const QVariant &ruleVar, rules) { auto rule = AutoDownloadRule::fromLegacyDict(ruleVar.toHash()); if (!rule.name().isEmpty()) insertRule(rule); } } void AutoDownloader::store() { if (!m_dirty) return; m_dirty = false; m_savingTimer.stop(); QJsonObject jsonObj; foreach (auto rule, m_rules) jsonObj.insert(rule.name(), rule.toJsonObject()); m_fileStorage->store(RulesFileName, QJsonDocument(jsonObj).toJson()); } void AutoDownloader::storeDeferred() { if (!m_savingTimer.isActive()) m_savingTimer.start(5 * 1000, this); } bool AutoDownloader::isProcessingEnabled() const { return m_processingEnabled; } void AutoDownloader::resetProcessingQueue() { m_processingQueue.clear(); if (!m_processingEnabled) return; foreach (Article *article, Session::instance()->rootFolder()->articles()) { if (!article->isRead() && !article->torrentUrl().isEmpty()) addJobForArticle(article); } } void AutoDownloader::startProcessing() { resetProcessingQueue(); connect(Session::instance()->rootFolder(), &Folder::newArticle, this, &AutoDownloader::handleNewArticle); } void AutoDownloader::setProcessingEnabled(bool enabled) { if (m_processingEnabled != enabled) { m_processingEnabled = enabled; SettingsStorage::instance()->storeValue(SettingsKey_ProcessingEnabled, m_processingEnabled); if (m_processingEnabled) { startProcessing(); } else { m_processingQueue.clear(); disconnect(Session::instance()->rootFolder(), &Folder::newArticle, this, &AutoDownloader::handleNewArticle); } emit processingStateChanged(m_processingEnabled); } } void AutoDownloader::timerEvent(QTimerEvent *event) { Q_UNUSED(event); store(); } ParsingError::ParsingError(const QString &message) : std::runtime_error(message.toUtf8().data()) { } QString ParsingError::message() const { return what(); }