Use cached current time when parse RSS feed

PR #21959.
This commit is contained in:
Vladimir Golovnev 2024-12-07 11:10:53 +03:00 committed by GitHub
parent a180162405
commit 200f7fc628
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 16 additions and 17 deletions

View file

@ -1,6 +1,6 @@
/* /*
* Bittorrent Client using Qt and libtorrent. * Bittorrent Client using Qt and libtorrent.
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru> * Copyright (C) 2015-2024 Vladimir Golovnev <glassez@yandex.ru>
* Copyright (C) 2012 Christophe Dumez <chris@qbittorrent.org> * Copyright (C) 2012 Christophe Dumez <chris@qbittorrent.org>
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
@ -29,11 +29,8 @@
#include "rss_parser.h" #include "rss_parser.h"
#include <QDateTime>
#include <QDebug> #include <QDebug>
#include <QGlobalStatic>
#include <QHash> #include <QHash>
#include <QMetaObject>
#include <QRegularExpression> #include <QRegularExpression>
#include <QStringList> #include <QStringList>
#include <QTimeZone> #include <QTimeZone>
@ -360,7 +357,7 @@ namespace
}; };
// Ported to Qt from KDElibs4 // Ported to Qt from KDElibs4
QDateTime parseDate(const QString &string) QDateTime parseDate(const QString &string, const QDateTime &fallbackDate)
{ {
const char16_t shortDay[][4] = const char16_t shortDay[][4] =
{ {
@ -383,7 +380,7 @@ namespace
const QString str = string.trimmed(); const QString str = string.trimmed();
if (str.isEmpty()) if (str.isEmpty())
return QDateTime::currentDateTime(); return fallbackDate;
int nyear = 6; // indexes within string to values int nyear = 6; // indexes within string to values
int nmonth = 4; int nmonth = 4;
@ -403,14 +400,14 @@ namespace
const bool h1 = (parts[3] == u"-"); const bool h1 = (parts[3] == u"-");
const bool h2 = (parts[5] == u"-"); const bool h2 = (parts[5] == u"-");
if (h1 != h2) if (h1 != h2)
return QDateTime::currentDateTime(); return fallbackDate;
} }
else else
{ {
// Check for the obsolete form "Wdy Mon DD HH:MM:SS YYYY" // Check for the obsolete form "Wdy Mon DD HH:MM:SS YYYY"
rx = QRegularExpression {u"^([A-Z][a-z]+)\\s+(\\S+)\\s+(\\d\\d)\\s+(\\d\\d):(\\d\\d):(\\d\\d)\\s+(\\d\\d\\d\\d)$"_s}; rx = QRegularExpression {u"^([A-Z][a-z]+)\\s+(\\S+)\\s+(\\d\\d)\\s+(\\d\\d):(\\d\\d):(\\d\\d)\\s+(\\d\\d\\d\\d)$"_s};
if (str.indexOf(rx, 0, &rxMatch) != 0) if (str.indexOf(rx, 0, &rxMatch) != 0)
return QDateTime::currentDateTime(); return fallbackDate;
nyear = 7; nyear = 7;
nmonth = 2; nmonth = 2;
@ -428,14 +425,14 @@ namespace
const int hour = parts[nhour].toInt(&ok[2]); const int hour = parts[nhour].toInt(&ok[2]);
const int minute = parts[nmin].toInt(&ok[3]); const int minute = parts[nmin].toInt(&ok[3]);
if (!ok[0] || !ok[1] || !ok[2] || !ok[3]) if (!ok[0] || !ok[1] || !ok[2] || !ok[3])
return QDateTime::currentDateTime(); return fallbackDate;
int second = 0; int second = 0;
if (!parts[nsec].isEmpty()) if (!parts[nsec].isEmpty())
{ {
second = parts[nsec].toInt(&ok[0]); second = parts[nsec].toInt(&ok[0]);
if (!ok[0]) if (!ok[0])
return QDateTime::currentDateTime(); return fallbackDate;
} }
const bool leapSecond = (second == 60); const bool leapSecond = (second == 60);
@ -519,21 +516,21 @@ namespace
const QDate qDate(year, month + 1, day); // convert date, and check for out-of-range const QDate qDate(year, month + 1, day); // convert date, and check for out-of-range
if (!qDate.isValid()) if (!qDate.isValid())
return QDateTime::currentDateTime(); return fallbackDate;
const QTime qTime(hour, minute, second); const QTime qTime(hour, minute, second);
QDateTime result(qDate, qTime, QTimeZone::UTC); QDateTime result(qDate, qTime, QTimeZone::UTC);
if (offset) if (offset)
result = result.addSecs(-offset); result = result.addSecs(-offset);
if (!result.isValid()) if (!result.isValid())
return QDateTime::currentDateTime(); // invalid date/time return fallbackDate; // invalid date/time
if (leapSecond) if (leapSecond)
{ {
// Validate a leap second time. Leap seconds are inserted after 23:59:59 UTC. // Validate a leap second time. Leap seconds are inserted after 23:59:59 UTC.
// Convert the time to UTC and check that it is 00:00:00. // Convert the time to UTC and check that it is 00:00:00.
if ((hour*3600 + minute*60 + 60 - offset + 86400*5) % 86400) // (max abs(offset) is 100 hours) if ((hour*3600 + minute*60 + 60 - offset + 86400*5) % 86400) // (max abs(offset) is 100 hours)
return QDateTime::currentDateTime(); // the time isn't the last second of the day return fallbackDate; // the time isn't the last second of the day
} }
return result; return result;
@ -551,6 +548,7 @@ RSS::Private::Parser::Parser(const QString &lastBuildDate)
void RSS::Private::Parser::parse(const QByteArray &feedData) void RSS::Private::Parser::parse(const QByteArray &feedData)
{ {
QXmlStreamReader xml {feedData}; QXmlStreamReader xml {feedData};
m_fallbackDate = QDateTime::currentDateTime();
XmlStreamEntityResolver resolver; XmlStreamEntityResolver resolver;
xml.setEntityResolver(&resolver); xml.setEntityResolver(&resolver);
bool foundChannel = false; bool foundChannel = false;
@ -642,7 +640,7 @@ void RSS::Private::Parser::parseRssArticle(QXmlStreamReader &xml)
} }
else if (name == u"pubDate") else if (name == u"pubDate")
{ {
article[Article::KeyDate] = parseDate(xml.readElementText().trimmed()); article[Article::KeyDate] = parseDate(xml.readElementText().trimmed(), m_fallbackDate);
} }
else if (name == u"author") else if (name == u"author")
{ {
@ -700,7 +698,6 @@ void RSS::Private::Parser::parseRSSChannel(QXmlStreamReader &xml)
void RSS::Private::Parser::parseAtomArticle(QXmlStreamReader &xml) void RSS::Private::Parser::parseAtomArticle(QXmlStreamReader &xml)
{ {
const auto currentDateTime = QDateTime::currentDateTime();
QVariantHash article; QVariantHash article;
bool doubleContent = false; bool doubleContent = false;
@ -757,7 +754,7 @@ void RSS::Private::Parser::parseAtomArticle(QXmlStreamReader &xml)
{ {
// ATOM uses standard compliant date, don't do fancy stuff // ATOM uses standard compliant date, don't do fancy stuff
const QDateTime articleDate = QDateTime::fromString(xml.readElementText().trimmed(), Qt::ISODate); const QDateTime articleDate = QDateTime::fromString(xml.readElementText().trimmed(), Qt::ISODate);
article[Article::KeyDate] = (articleDate.isValid() ? articleDate : currentDateTime); article[Article::KeyDate] = (articleDate.isValid() ? articleDate : m_fallbackDate);
} }
else if (name == u"author") else if (name == u"author")
{ {

View file

@ -1,6 +1,6 @@
/* /*
* Bittorrent Client using Qt and libtorrent. * Bittorrent Client using Qt and libtorrent.
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru> * Copyright (C) 2015-2024 Vladimir Golovnev <glassez@yandex.ru>
* Copyright (C) 2012 Christophe Dumez <chris@qbittorrent.org> * Copyright (C) 2012 Christophe Dumez <chris@qbittorrent.org>
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
@ -29,6 +29,7 @@
#pragma once #pragma once
#include <QDateTime>
#include <QList> #include <QList>
#include <QObject> #include <QObject>
#include <QSet> #include <QSet>
@ -66,6 +67,7 @@ namespace RSS::Private
void parseAtomChannel(QXmlStreamReader &xml); void parseAtomChannel(QXmlStreamReader &xml);
void addArticle(QVariantHash article); void addArticle(QVariantHash article);
QDateTime m_fallbackDate;
QString m_baseUrl; QString m_baseUrl;
ParsingResult m_result; ParsingResult m_result;
QSet<QString> m_articleIDs; QSet<QString> m_articleIDs;