2017-03-07 16:10:42 +03:00
|
|
|
/*
|
|
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
|
|
* Copyright (C) 2017 Vladimir Golovnev <glassez@yandex.ru>
|
|
|
|
* Copyright (C) 2010 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 "rss_autodownloadrule.h"
|
|
|
|
|
2019-01-11 16:05:57 +08:00
|
|
|
#include <algorithm>
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
#include <QDebug>
|
|
|
|
#include <QHash>
|
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonObject>
|
|
|
|
#include <QRegularExpression>
|
|
|
|
#include <QSharedData>
|
|
|
|
#include <QString>
|
|
|
|
#include <QStringList>
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
#include "../global.h"
|
2017-03-07 16:10:42 +03:00
|
|
|
#include "../preferences.h"
|
|
|
|
#include "../tristatebool.h"
|
|
|
|
#include "../utils/fs.h"
|
|
|
|
#include "../utils/string.h"
|
|
|
|
#include "rss_article.h"
|
2018-01-27 13:40:00 +00:00
|
|
|
#include "rss_autodownloader.h"
|
2018-04-14 22:53:45 +03:00
|
|
|
#include "rss_feed.h"
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
TriStateBool jsonValueToTriStateBool(const QJsonValue &jsonVal)
|
|
|
|
{
|
|
|
|
if (jsonVal.isBool())
|
|
|
|
return TriStateBool(jsonVal.toBool());
|
|
|
|
|
|
|
|
if (!jsonVal.isNull())
|
|
|
|
qDebug() << Q_FUNC_INFO << "Incorrect value" << jsonVal.toVariant();
|
|
|
|
|
|
|
|
return TriStateBool::Undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonValue triStateBoolToJsonValue(const TriStateBool &triStateBool)
|
|
|
|
{
|
|
|
|
switch (static_cast<int>(triStateBool)) {
|
2017-11-22 20:25:12 +03:00
|
|
|
case 0: return false;
|
|
|
|
case 1: return true;
|
2019-02-14 19:16:42 +02:00
|
|
|
default: return {};
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
}
|
2017-11-22 20:25:12 +03:00
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
TriStateBool addPausedLegacyToTriStateBool(const int val)
|
2017-11-22 20:25:12 +03:00
|
|
|
{
|
|
|
|
switch (val) {
|
|
|
|
case 1: return TriStateBool::True; // always
|
|
|
|
case 2: return TriStateBool::False; // never
|
|
|
|
default: return TriStateBool::Undefined; // default
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int triStateBoolToAddPausedLegacy(const TriStateBool &triStateBool)
|
|
|
|
{
|
|
|
|
switch (static_cast<int>(triStateBool)) {
|
|
|
|
case 0: return 2; // never
|
|
|
|
case 1: return 1; // always
|
|
|
|
default: return 0; // default
|
|
|
|
}
|
|
|
|
}
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
const QString Str_Name(QStringLiteral("name"));
|
|
|
|
const QString Str_Enabled(QStringLiteral("enabled"));
|
|
|
|
const QString Str_UseRegex(QStringLiteral("useRegex"));
|
|
|
|
const QString Str_MustContain(QStringLiteral("mustContain"));
|
|
|
|
const QString Str_MustNotContain(QStringLiteral("mustNotContain"));
|
|
|
|
const QString Str_EpisodeFilter(QStringLiteral("episodeFilter"));
|
|
|
|
const QString Str_AffectedFeeds(QStringLiteral("affectedFeeds"));
|
|
|
|
const QString Str_SavePath(QStringLiteral("savePath"));
|
|
|
|
const QString Str_AssignedCategory(QStringLiteral("assignedCategory"));
|
|
|
|
const QString Str_LastMatch(QStringLiteral("lastMatch"));
|
|
|
|
const QString Str_IgnoreDays(QStringLiteral("ignoreDays"));
|
|
|
|
const QString Str_AddPaused(QStringLiteral("addPaused"));
|
2016-05-22 14:59:31 +01:00
|
|
|
const QString Str_SmartFilter(QStringLiteral("smartFilter"));
|
|
|
|
const QString Str_PreviouslyMatched(QStringLiteral("previouslyMatchedEpisodes"));
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
namespace RSS
|
|
|
|
{
|
2018-04-14 22:53:45 +03:00
|
|
|
struct AutoDownloadRuleData : public QSharedData
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
|
|
|
QString name;
|
|
|
|
bool enabled = true;
|
|
|
|
|
|
|
|
QStringList mustContain;
|
|
|
|
QStringList mustNotContain;
|
|
|
|
QString episodeFilter;
|
|
|
|
QStringList feedURLs;
|
|
|
|
bool useRegex = false;
|
|
|
|
int ignoreDays = 0;
|
|
|
|
QDateTime lastMatch;
|
|
|
|
|
|
|
|
QString savePath;
|
|
|
|
QString category;
|
|
|
|
TriStateBool addPaused = TriStateBool::Undefined;
|
|
|
|
|
2016-05-22 14:59:31 +01:00
|
|
|
bool smartFilter = false;
|
|
|
|
QStringList previouslyMatchedEpisodes;
|
|
|
|
|
2018-12-09 13:14:06 +00:00
|
|
|
mutable QStringList lastComputedEpisodes;
|
2017-03-07 16:10:42 +03:00
|
|
|
mutable QHash<QString, QRegularExpression> cachedRegexes;
|
|
|
|
|
|
|
|
bool operator==(const AutoDownloadRuleData &other) const
|
|
|
|
{
|
|
|
|
return (name == other.name)
|
|
|
|
&& (enabled == other.enabled)
|
|
|
|
&& (mustContain == other.mustContain)
|
|
|
|
&& (mustNotContain == other.mustNotContain)
|
|
|
|
&& (episodeFilter == other.episodeFilter)
|
|
|
|
&& (feedURLs == other.feedURLs)
|
|
|
|
&& (useRegex == other.useRegex)
|
|
|
|
&& (ignoreDays == other.ignoreDays)
|
|
|
|
&& (lastMatch == other.lastMatch)
|
|
|
|
&& (savePath == other.savePath)
|
|
|
|
&& (category == other.category)
|
2016-05-22 14:59:31 +01:00
|
|
|
&& (addPaused == other.addPaused)
|
|
|
|
&& (smartFilter == other.smartFilter);
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
using namespace RSS;
|
|
|
|
|
2016-05-22 14:59:31 +01:00
|
|
|
QString computeEpisodeName(const QString &article)
|
|
|
|
{
|
2018-01-27 13:40:00 +00:00
|
|
|
const QRegularExpression episodeRegex = AutoDownloader::instance()->smartEpisodeRegex();
|
|
|
|
const QRegularExpressionMatch match = episodeRegex.match(article);
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-01-27 13:40:00 +00:00
|
|
|
// See if we can extract an season/episode number or date from the title
|
|
|
|
if (!match.hasMatch())
|
2019-02-14 19:16:42 +02:00
|
|
|
return {};
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-01-27 13:40:00 +00:00
|
|
|
QStringList ret;
|
|
|
|
for (int i = 1; i <= match.lastCapturedIndex(); ++i) {
|
2019-02-21 23:31:43 +02:00
|
|
|
const QString cap = match.captured(i);
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-01-27 13:40:00 +00:00
|
|
|
if (cap.isEmpty())
|
|
|
|
continue;
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-01-27 13:40:00 +00:00
|
|
|
bool isInt = false;
|
2019-02-21 23:31:43 +02:00
|
|
|
const int x = cap.toInt(&isInt);
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-01-27 13:40:00 +00:00
|
|
|
ret.append(isInt ? QString::number(x) : cap);
|
2016-05-22 14:59:31 +01:00
|
|
|
}
|
2018-01-27 13:40:00 +00:00
|
|
|
return ret.join('x');
|
2016-05-22 14:59:31 +01:00
|
|
|
}
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
AutoDownloadRule::AutoDownloadRule(const QString &name)
|
|
|
|
: m_dataPtr(new AutoDownloadRuleData)
|
|
|
|
{
|
|
|
|
setName(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoDownloadRule::AutoDownloadRule(const AutoDownloadRule &other)
|
|
|
|
: m_dataPtr(other.m_dataPtr)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoDownloadRule::~AutoDownloadRule() {}
|
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
QRegularExpression AutoDownloadRule::cachedRegex(const QString &expression, const bool isRegex) const
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
|
|
|
// Use a cache of regexes so we don't have to continually recompile - big performance increase.
|
|
|
|
// The cache is cleared whenever the regex/wildcard, must or must not contain fields or
|
|
|
|
// episode filter are modified.
|
|
|
|
Q_ASSERT(!expression.isEmpty());
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
QRegularExpression ®ex = m_dataPtr->cachedRegexes[expression];
|
|
|
|
if (regex.pattern().isEmpty()) {
|
|
|
|
regex = QRegularExpression {
|
|
|
|
(isRegex ? expression : Utils::String::wildcardToRegex(expression))
|
|
|
|
, QRegularExpression::CaseInsensitiveOption};
|
|
|
|
}
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
return regex;
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
bool AutoDownloadRule::matchesExpression(const QString &articleTitle, const QString &expression) const
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
2018-05-17 21:34:50 +03:00
|
|
|
const QRegularExpression whitespace {"\\s+"};
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
if (expression.isEmpty()) {
|
|
|
|
// A regex of the form "expr|" will always match, so do the same for wildcards
|
|
|
|
return true;
|
|
|
|
}
|
2018-05-17 21:34:50 +03:00
|
|
|
|
|
|
|
if (m_dataPtr->useRegex) {
|
2019-02-21 23:31:43 +02:00
|
|
|
const QRegularExpression reg(cachedRegex(expression));
|
2017-03-07 16:10:42 +03:00
|
|
|
return reg.match(articleTitle).hasMatch();
|
|
|
|
}
|
2018-05-17 21:34:50 +03:00
|
|
|
|
|
|
|
// Only match if every wildcard token (separated by spaces) is present in the article name.
|
|
|
|
// Order of wildcard tokens is unimportant (if order is important, they should have used *).
|
|
|
|
const QStringList wildcards {expression.split(whitespace, QString::SplitBehavior::SkipEmptyParts)};
|
|
|
|
for (const QString &wildcard : wildcards) {
|
|
|
|
const QRegularExpression reg {cachedRegex(wildcard, false)};
|
|
|
|
if (!reg.match(articleTitle).hasMatch())
|
|
|
|
return false;
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
bool AutoDownloadRule::matchesMustContainExpression(const QString &articleTitle) const
|
|
|
|
{
|
|
|
|
if (m_dataPtr->mustContain.empty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Each expression is either a regex, or a set of wildcards separated by whitespace.
|
|
|
|
// Accept if any complete expression matches.
|
2019-01-11 16:05:57 +08:00
|
|
|
return std::any_of(m_dataPtr->mustContain.cbegin(), m_dataPtr->mustContain.cend(), [this, &articleTitle](const QString &expression)
|
|
|
|
{
|
2018-05-17 21:34:50 +03:00
|
|
|
// A regex of the form "expr|" will always match, so do the same for wildcards
|
2019-01-11 16:05:57 +08:00
|
|
|
return matchesExpression(articleTitle, expression);
|
|
|
|
});
|
2018-05-17 21:34:50 +03:00
|
|
|
}
|
|
|
|
|
2018-11-06 17:49:17 +02:00
|
|
|
bool AutoDownloadRule::matchesMustNotContainExpression(const QString &articleTitle) const
|
2018-05-17 21:34:50 +03:00
|
|
|
{
|
|
|
|
if (m_dataPtr->mustNotContain.empty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Each expression is either a regex, or a set of wildcards separated by whitespace.
|
|
|
|
// Reject if any complete expression matches.
|
2019-01-11 16:05:57 +08:00
|
|
|
return std::none_of(m_dataPtr->mustNotContain.cbegin(), m_dataPtr->mustNotContain.cend(), [this, &articleTitle](const QString &expression)
|
|
|
|
{
|
2018-05-17 21:34:50 +03:00
|
|
|
// A regex of the form "expr|" will always match, so do the same for wildcards
|
2019-01-11 16:05:57 +08:00
|
|
|
return matchesExpression(articleTitle, expression);
|
|
|
|
});
|
2018-05-17 21:34:50 +03:00
|
|
|
}
|
|
|
|
|
2018-11-06 17:49:17 +02:00
|
|
|
bool AutoDownloadRule::matchesEpisodeFilterExpression(const QString &articleTitle) const
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
2016-05-22 14:59:31 +01:00
|
|
|
// Reset the lastComputedEpisode, we don't want to leak it between matches
|
2018-12-09 13:14:06 +00:00
|
|
|
m_dataPtr->lastComputedEpisodes.clear();
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
if (m_dataPtr->episodeFilter.isEmpty())
|
|
|
|
return true;
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
const QRegularExpression filterRegex {cachedRegex("(^\\d{1,4})x(.*;$)")};
|
|
|
|
const QRegularExpressionMatch matcher {filterRegex.match(m_dataPtr->episodeFilter)};
|
|
|
|
if (!matcher.hasMatch())
|
|
|
|
return false;
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
const QString season {matcher.captured(1)};
|
|
|
|
const QStringList episodes {matcher.captured(2).split(';')};
|
|
|
|
const int seasonOurs {season.toInt()};
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
for (QString episode : episodes) {
|
|
|
|
if (episode.isEmpty())
|
|
|
|
continue;
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
// We need to trim leading zeroes, but if it's all zeros then we want episode zero.
|
|
|
|
while ((episode.size() > 1) && episode.startsWith('0'))
|
|
|
|
episode = episode.right(episode.size() - 1);
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
if (episode.indexOf('-') != -1) { // Range detected
|
|
|
|
const QString partialPattern1 {"\\bs0?(\\d{1,4})[ -_\\.]?e(0?\\d{1,4})(?:\\D|\\b)"};
|
|
|
|
const QString partialPattern2 {"\\b(\\d{1,4})x(0?\\d{1,4})(?:\\D|\\b)"};
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
// Extract partial match from article and compare as digits
|
|
|
|
QRegularExpressionMatch matcher = cachedRegex(partialPattern1).match(articleTitle);
|
|
|
|
bool matched = matcher.hasMatch();
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
if (!matched) {
|
|
|
|
matcher = cachedRegex(partialPattern2).match(articleTitle);
|
|
|
|
matched = matcher.hasMatch();
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
if (matched) {
|
|
|
|
const int seasonTheirs {matcher.captured(1).toInt()};
|
|
|
|
const int episodeTheirs {matcher.captured(2).toInt()};
|
2017-03-07 16:10:42 +03:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
if (episode.endsWith('-')) { // Infinite range
|
|
|
|
const int episodeOurs {episode.leftRef(episode.size() - 1).toInt()};
|
|
|
|
if (((seasonTheirs == seasonOurs) && (episodeTheirs >= episodeOurs)) || (seasonTheirs > seasonOurs))
|
|
|
|
return true;
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
else { // Normal range
|
2018-05-17 21:34:50 +03:00
|
|
|
const QStringList range {episode.split('-')};
|
2017-03-07 16:10:42 +03:00
|
|
|
Q_ASSERT(range.size() == 2);
|
|
|
|
if (range.first().toInt() > range.last().toInt())
|
|
|
|
continue; // Ignore this subrule completely
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
const int episodeOursFirst {range.first().toInt()};
|
|
|
|
const int episodeOursLast {range.last().toInt()};
|
|
|
|
if ((seasonTheirs == seasonOurs) && ((episodeOursFirst <= episodeTheirs) && (episodeOursLast >= episodeTheirs)))
|
|
|
|
return true;
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-17 21:34:50 +03:00
|
|
|
else { // Single number
|
|
|
|
const QString expStr {QString("\\b(?:s0?%1[ -_\\.]?e0?%2|%1x0?%2)(?:\\D|\\b)").arg(season, episode)};
|
|
|
|
if (cachedRegex(expStr).match(articleTitle).hasMatch())
|
|
|
|
return true;
|
|
|
|
}
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-11-06 17:49:17 +02:00
|
|
|
bool AutoDownloadRule::matchesSmartEpisodeFilter(const QString &articleTitle) const
|
2018-05-17 21:34:50 +03:00
|
|
|
{
|
|
|
|
if (!useSmartFilter())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
const QString episodeStr = computeEpisodeName(articleTitle);
|
|
|
|
if (episodeStr.isEmpty())
|
|
|
|
return true;
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2018-05-17 21:34:50 +03:00
|
|
|
// See if this episode has been downloaded before
|
|
|
|
const bool previouslyMatched = m_dataPtr->previouslyMatchedEpisodes.contains(episodeStr);
|
2018-12-09 13:14:06 +00:00
|
|
|
if (previouslyMatched) {
|
|
|
|
if (!AutoDownloader::instance()->downloadRepacks())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Now see if we've downloaded this particular repack/proper combination
|
|
|
|
const bool isRepack = articleTitle.contains("REPACK", Qt::CaseInsensitive);
|
|
|
|
const bool isProper = articleTitle.contains("PROPER", Qt::CaseInsensitive);
|
|
|
|
|
|
|
|
if (!isRepack && !isProper)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const QString fullEpisodeStr = QString("%1%2%3").arg(episodeStr,
|
|
|
|
isRepack ? "-REPACK" : "",
|
|
|
|
isProper ? "-PROPER" : "");
|
|
|
|
const bool previouslyMatchedFull = m_dataPtr->previouslyMatchedEpisodes.contains(fullEpisodeStr);
|
|
|
|
if (previouslyMatchedFull)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_dataPtr->lastComputedEpisodes.append(fullEpisodeStr);
|
|
|
|
|
|
|
|
// If this is a REPACK and PROPER download, add the individual entries to the list
|
|
|
|
// so we don't download those
|
|
|
|
if (isRepack && isProper) {
|
|
|
|
m_dataPtr->lastComputedEpisodes.append(QString("%1-REPACK").arg(episodeStr));
|
|
|
|
m_dataPtr->lastComputedEpisodes.append(QString("%1-PROPER").arg(episodeStr));
|
|
|
|
}
|
|
|
|
}
|
2018-01-27 13:40:00 +00:00
|
|
|
|
2018-12-09 13:14:06 +00:00
|
|
|
m_dataPtr->lastComputedEpisodes.append(episodeStr);
|
2018-05-17 21:34:50 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-18 14:41:52 +03:00
|
|
|
bool AutoDownloadRule::matches(const QVariantHash &articleData) const
|
2018-05-17 21:34:50 +03:00
|
|
|
{
|
2018-05-18 14:41:52 +03:00
|
|
|
const QDateTime articleDate {articleData[Article::KeyDate].toDateTime()};
|
|
|
|
if (ignoreDays() > 0) {
|
|
|
|
if (lastMatch().isValid() && (articleDate < lastMatch().addDays(ignoreDays())))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString articleTitle {articleData[Article::KeyTitle].toString()};
|
2018-05-17 21:34:50 +03:00
|
|
|
if (!matchesMustContainExpression(articleTitle))
|
|
|
|
return false;
|
|
|
|
if (!matchesMustNotContainExpression(articleTitle))
|
|
|
|
return false;
|
|
|
|
if (!matchesEpisodeFilterExpression(articleTitle))
|
|
|
|
return false;
|
|
|
|
if (!matchesSmartEpisodeFilter(articleTitle))
|
|
|
|
return false;
|
2016-05-22 14:59:31 +01:00
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-18 14:41:52 +03:00
|
|
|
bool AutoDownloadRule::accepts(const QVariantHash &articleData)
|
|
|
|
{
|
|
|
|
if (!matches(articleData))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
setLastMatch(articleData[Article::KeyDate].toDateTime());
|
|
|
|
|
2018-12-09 13:14:06 +00:00
|
|
|
// If there's a matched episode string, add that to the previously matched list
|
|
|
|
if (!m_dataPtr->lastComputedEpisodes.isEmpty()) {
|
|
|
|
m_dataPtr->previouslyMatchedEpisodes.append(m_dataPtr->lastComputedEpisodes);
|
|
|
|
m_dataPtr->lastComputedEpisodes.clear();
|
2018-05-18 14:41:52 +03:00
|
|
|
}
|
2018-05-21 21:24:02 +03:00
|
|
|
|
|
|
|
return true;
|
2018-05-18 14:41:52 +03:00
|
|
|
}
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
AutoDownloadRule &AutoDownloadRule::operator=(const AutoDownloadRule &other)
|
|
|
|
{
|
|
|
|
m_dataPtr = other.m_dataPtr;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AutoDownloadRule::operator==(const AutoDownloadRule &other) const
|
|
|
|
{
|
|
|
|
return (m_dataPtr == other.m_dataPtr) // optimization
|
|
|
|
|| (*m_dataPtr == *other.m_dataPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AutoDownloadRule::operator!=(const AutoDownloadRule &other) const
|
|
|
|
{
|
|
|
|
return !operator==(other);
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject AutoDownloadRule::toJsonObject() const
|
|
|
|
{
|
|
|
|
return {{Str_Enabled, isEnabled()}
|
|
|
|
, {Str_UseRegex, useRegex()}
|
|
|
|
, {Str_MustContain, mustContain()}
|
|
|
|
, {Str_MustNotContain, mustNotContain()}
|
|
|
|
, {Str_EpisodeFilter, episodeFilter()}
|
|
|
|
, {Str_AffectedFeeds, QJsonArray::fromStringList(feedURLs())}
|
|
|
|
, {Str_SavePath, savePath()}
|
|
|
|
, {Str_AssignedCategory, assignedCategory()}
|
|
|
|
, {Str_LastMatch, lastMatch().toString(Qt::RFC2822Date)}
|
|
|
|
, {Str_IgnoreDays, ignoreDays()}
|
2016-05-22 14:59:31 +01:00
|
|
|
, {Str_AddPaused, triStateBoolToJsonValue(addPaused())}
|
|
|
|
, {Str_SmartFilter, useSmartFilter()}
|
|
|
|
, {Str_PreviouslyMatched, QJsonArray::fromStringList(previouslyMatchedEpisodes())}};
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
AutoDownloadRule AutoDownloadRule::fromJsonObject(const QJsonObject &jsonObj, const QString &name)
|
|
|
|
{
|
|
|
|
AutoDownloadRule rule(name.isEmpty() ? jsonObj.value(Str_Name).toString() : name);
|
|
|
|
|
|
|
|
rule.setUseRegex(jsonObj.value(Str_UseRegex).toBool(false));
|
|
|
|
rule.setMustContain(jsonObj.value(Str_MustContain).toString());
|
|
|
|
rule.setMustNotContain(jsonObj.value(Str_MustNotContain).toString());
|
|
|
|
rule.setEpisodeFilter(jsonObj.value(Str_EpisodeFilter).toString());
|
|
|
|
rule.setEnabled(jsonObj.value(Str_Enabled).toBool(true));
|
|
|
|
rule.setSavePath(jsonObj.value(Str_SavePath).toString());
|
|
|
|
rule.setCategory(jsonObj.value(Str_AssignedCategory).toString());
|
|
|
|
rule.setAddPaused(jsonValueToTriStateBool(jsonObj.value(Str_AddPaused)));
|
|
|
|
rule.setLastMatch(QDateTime::fromString(jsonObj.value(Str_LastMatch).toString(), Qt::RFC2822Date));
|
|
|
|
rule.setIgnoreDays(jsonObj.value(Str_IgnoreDays).toInt());
|
2016-05-22 14:59:31 +01:00
|
|
|
rule.setUseSmartFilter(jsonObj.value(Str_SmartFilter).toBool(false));
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
const QJsonValue feedsVal = jsonObj.value(Str_AffectedFeeds);
|
|
|
|
QStringList feedURLs;
|
|
|
|
if (feedsVal.isString())
|
|
|
|
feedURLs << feedsVal.toString();
|
2018-11-27 22:15:04 +02:00
|
|
|
else for (const QJsonValue &urlVal : asConst(feedsVal.toArray()))
|
2017-03-07 16:10:42 +03:00
|
|
|
feedURLs << urlVal.toString();
|
|
|
|
rule.setFeedURLs(feedURLs);
|
|
|
|
|
2016-05-22 14:59:31 +01:00
|
|
|
const QJsonValue previouslyMatchedVal = jsonObj.value(Str_PreviouslyMatched);
|
|
|
|
QStringList previouslyMatched;
|
|
|
|
if (previouslyMatchedVal.isString()) {
|
|
|
|
previouslyMatched << previouslyMatchedVal.toString();
|
|
|
|
}
|
|
|
|
else {
|
2018-11-27 22:15:04 +02:00
|
|
|
for (const QJsonValue &val : asConst(previouslyMatchedVal.toArray()))
|
2016-05-22 14:59:31 +01:00
|
|
|
previouslyMatched << val.toString();
|
|
|
|
}
|
|
|
|
rule.setPreviouslyMatchedEpisodes(previouslyMatched);
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
return rule;
|
|
|
|
}
|
|
|
|
|
2017-11-22 20:25:12 +03:00
|
|
|
QVariantHash AutoDownloadRule::toLegacyDict() const
|
|
|
|
{
|
|
|
|
return {{"name", name()},
|
|
|
|
{"must_contain", mustContain()},
|
|
|
|
{"must_not_contain", mustNotContain()},
|
|
|
|
{"save_path", savePath()},
|
|
|
|
{"affected_feeds", feedURLs()},
|
|
|
|
{"enabled", isEnabled()},
|
|
|
|
{"category_assigned", assignedCategory()},
|
|
|
|
{"use_regex", useRegex()},
|
|
|
|
{"add_paused", triStateBoolToAddPausedLegacy(addPaused())},
|
|
|
|
{"episode_filter", episodeFilter()},
|
|
|
|
{"last_match", lastMatch()},
|
|
|
|
{"ignore_days", ignoreDays()}};
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoDownloadRule AutoDownloadRule::fromLegacyDict(const QVariantHash &dict)
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
2017-11-22 20:25:12 +03:00
|
|
|
AutoDownloadRule rule(dict.value("name").toString());
|
|
|
|
|
|
|
|
rule.setUseRegex(dict.value("use_regex", false).toBool());
|
|
|
|
rule.setMustContain(dict.value("must_contain").toString());
|
|
|
|
rule.setMustNotContain(dict.value("must_not_contain").toString());
|
|
|
|
rule.setEpisodeFilter(dict.value("episode_filter").toString());
|
|
|
|
rule.setFeedURLs(dict.value("affected_feeds").toStringList());
|
|
|
|
rule.setEnabled(dict.value("enabled", false).toBool());
|
|
|
|
rule.setSavePath(dict.value("save_path").toString());
|
|
|
|
rule.setCategory(dict.value("category_assigned").toString());
|
|
|
|
rule.setAddPaused(addPausedLegacyToTriStateBool(dict.value("add_paused").toInt()));
|
|
|
|
rule.setLastMatch(dict.value("last_match").toDateTime());
|
|
|
|
rule.setIgnoreDays(dict.value("ignore_days").toInt());
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
return rule;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setMustContain(const QString &tokens)
|
|
|
|
{
|
|
|
|
m_dataPtr->cachedRegexes.clear();
|
|
|
|
|
|
|
|
if (m_dataPtr->useRegex)
|
|
|
|
m_dataPtr->mustContain = QStringList() << tokens;
|
|
|
|
else
|
2018-07-21 13:28:13 +08:00
|
|
|
m_dataPtr->mustContain = tokens.split('|');
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
// Check for single empty string - if so, no condition
|
|
|
|
if ((m_dataPtr->mustContain.size() == 1) && m_dataPtr->mustContain[0].isEmpty())
|
|
|
|
m_dataPtr->mustContain.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setMustNotContain(const QString &tokens)
|
|
|
|
{
|
|
|
|
m_dataPtr->cachedRegexes.clear();
|
|
|
|
|
|
|
|
if (m_dataPtr->useRegex)
|
|
|
|
m_dataPtr->mustNotContain = QStringList() << tokens;
|
|
|
|
else
|
2018-07-21 13:28:13 +08:00
|
|
|
m_dataPtr->mustNotContain = tokens.split('|');
|
2017-03-07 16:10:42 +03:00
|
|
|
|
|
|
|
// Check for single empty string - if so, no condition
|
|
|
|
if ((m_dataPtr->mustNotContain.size() == 1) && m_dataPtr->mustNotContain[0].isEmpty())
|
|
|
|
m_dataPtr->mustNotContain.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList AutoDownloadRule::feedURLs() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->feedURLs;
|
|
|
|
}
|
|
|
|
|
2017-05-09 13:55:24 +08:00
|
|
|
void AutoDownloadRule::setFeedURLs(const QStringList &urls)
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
2017-05-09 13:55:24 +08:00
|
|
|
m_dataPtr->feedURLs = urls;
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString AutoDownloadRule::name() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setName(const QString &name)
|
|
|
|
{
|
|
|
|
m_dataPtr->name = name;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AutoDownloadRule::savePath() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->savePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setSavePath(const QString &savePath)
|
|
|
|
{
|
2019-06-16 20:14:15 +03:00
|
|
|
m_dataPtr->savePath = Utils::Fs::toUniformPath(savePath);
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
TriStateBool AutoDownloadRule::addPaused() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->addPaused;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setAddPaused(const TriStateBool &addPaused)
|
|
|
|
{
|
|
|
|
m_dataPtr->addPaused = addPaused;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AutoDownloadRule::assignedCategory() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->category;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setCategory(const QString &category)
|
|
|
|
{
|
|
|
|
m_dataPtr->category = category;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AutoDownloadRule::isEnabled() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->enabled;
|
|
|
|
}
|
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
void AutoDownloadRule::setEnabled(const bool enable)
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
|
|
|
m_dataPtr->enabled = enable;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDateTime AutoDownloadRule::lastMatch() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->lastMatch;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setLastMatch(const QDateTime &lastMatch)
|
|
|
|
{
|
|
|
|
m_dataPtr->lastMatch = lastMatch;
|
|
|
|
}
|
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
void AutoDownloadRule::setIgnoreDays(const int d)
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
|
|
|
m_dataPtr->ignoreDays = d;
|
|
|
|
}
|
|
|
|
|
|
|
|
int AutoDownloadRule::ignoreDays() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->ignoreDays;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AutoDownloadRule::mustContain() const
|
|
|
|
{
|
2018-07-21 13:28:13 +08:00
|
|
|
return m_dataPtr->mustContain.join('|');
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString AutoDownloadRule::mustNotContain() const
|
|
|
|
{
|
2018-07-21 13:28:13 +08:00
|
|
|
return m_dataPtr->mustNotContain.join('|');
|
2017-03-07 16:10:42 +03:00
|
|
|
}
|
|
|
|
|
2016-05-22 14:59:31 +01:00
|
|
|
bool AutoDownloadRule::useSmartFilter() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->smartFilter;
|
|
|
|
}
|
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
void AutoDownloadRule::setUseSmartFilter(const bool enabled)
|
2016-05-22 14:59:31 +01:00
|
|
|
{
|
|
|
|
m_dataPtr->smartFilter = enabled;
|
|
|
|
}
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
bool AutoDownloadRule::useRegex() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->useRegex;
|
|
|
|
}
|
|
|
|
|
2019-02-21 23:31:43 +02:00
|
|
|
void AutoDownloadRule::setUseRegex(const bool enabled)
|
2017-03-07 16:10:42 +03:00
|
|
|
{
|
|
|
|
m_dataPtr->useRegex = enabled;
|
|
|
|
m_dataPtr->cachedRegexes.clear();
|
|
|
|
}
|
|
|
|
|
2016-05-22 14:59:31 +01:00
|
|
|
QStringList AutoDownloadRule::previouslyMatchedEpisodes() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->previouslyMatchedEpisodes;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setPreviouslyMatchedEpisodes(const QStringList &previouslyMatchedEpisodes)
|
|
|
|
{
|
|
|
|
m_dataPtr->previouslyMatchedEpisodes = previouslyMatchedEpisodes;
|
|
|
|
}
|
|
|
|
|
2017-03-07 16:10:42 +03:00
|
|
|
QString AutoDownloadRule::episodeFilter() const
|
|
|
|
{
|
|
|
|
return m_dataPtr->episodeFilter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDownloadRule::setEpisodeFilter(const QString &e)
|
|
|
|
{
|
|
|
|
m_dataPtr->episodeFilter = e;
|
|
|
|
m_dataPtr->cachedRegexes.clear();
|
|
|
|
}
|