kasts/src/fetcher.cpp
Bart De Vries f861f4e802 Add podcast subscription and episode play state synchronization
This implements the gpodder API from scratch.  It turned out that
libmygpo-qt has several critical bugs, and there's no response to pull
requests upstream.  So using that library was not an option.

The implementation into kasts consists of the following:
- Can sync with gpodder.net or with a nextcloud server that has the
  nextcloud-gpodder app installed.  (This app is mostly API compatible
  with gpodder.)
- Passwords are stored using qtkeychain.  If the keychain is
  unavailable it will fallback to file.
- It syncs podcast subscriptions and episode play positions, including
  marking episodes as played. Episodes that have a non-zero play
  position will be added to the queue automatically.
- It will check for a metered connection before syncing.  This is
  coupled to the allowMeteredFeedUpdates setting.
- Full synchronization can be performed either manually (from the
  settings page) or through automatic triggers: on startup and/or on
  feed refresh.
- There is an additional possibility to trigger quick upload-only syncs
  to make sure that the local changes are immediately uploaded to the
  server (if the connection allows).  This will trigger when
  subscriptions are added or removed, when the pause/play button is
  toggled or an episode is marked as played.
- This implements a few safeguards to avoid having multiple feed URLS
  pointing to the same underlying feed (e.g. http vs https).  This
  solves part of #17

Solves #13
2021-10-29 18:47:55 +02:00

230 lines
7.0 KiB
C++

/**
* SPDX-FileCopyrightText: 2020 Tobias Fella <fella@posteo.de>
* SPDX-FileCopyrightText: 2021 Bart De Vries <bart@mogwai.be>
*
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
*/
#include "fetcher.h"
#include "fetcherlogging.h"
#include <KLocalizedString>
#include <QDateTime>
#include <QDebug>
#include <QDir>
#include <QDomElement>
#include <QFile>
#include <QFileInfo>
#include <QMultiMap>
#include <QNetworkAccessManager>
#include <QNetworkReply>
#include <QTextDocumentFragment>
#include <QTime>
#include <Syndication/Syndication>
#include "database.h"
#include "enclosure.h"
#include "fetchfeedsjob.h"
#include "kasts-version.h"
#include "models/errorlogmodel.h"
#include "settingsmanager.h"
#include "storagemanager.h"
#include "sync/sync.h"
#include <solidextras/networkstatus.h>
Fetcher::Fetcher()
{
connect(this, &Fetcher::error, &ErrorLogModel::instance(), &ErrorLogModel::monitorErrorMessages);
m_updateProgress = -1;
m_updateTotal = -1;
m_updating = false;
manager = new QNetworkAccessManager(this);
manager->setRedirectPolicy(QNetworkRequest::NoLessSafeRedirectPolicy);
manager->setStrictTransportSecurityEnabled(true);
manager->enableStrictTransportSecurityStore(true);
}
void Fetcher::fetch(const QString &url)
{
QStringList urls(url);
fetch(urls);
}
void Fetcher::fetchAll()
{
if (Sync::instance().syncEnabled() && SettingsManager::self()->syncWhenUpdatingFeeds()) {
Sync::instance().doRegularSync(true);
} else {
QStringList urls;
QSqlQuery query;
query.prepare(QStringLiteral("SELECT url FROM Feeds;"));
Database::instance().execute(query);
while (query.next()) {
urls += query.value(0).toString();
}
if (urls.count() > 0) {
fetch(urls);
}
}
}
void Fetcher::fetch(const QStringList &urls)
{
if (m_updating)
return; // update is already running, do nothing
m_updating = true;
m_updateProgress = 0;
m_updateTotal = urls.count();
Q_EMIT updatingChanged(m_updating);
Q_EMIT updateProgressChanged(m_updateProgress);
Q_EMIT updateTotalChanged(m_updateTotal);
qCDebug(kastsFetcher) << "Create fetchFeedsJob";
FetchFeedsJob *fetchFeedsJob = new FetchFeedsJob(urls, this);
connect(this, &Fetcher::cancelFetching, fetchFeedsJob, &FetchFeedsJob::abort);
connect(fetchFeedsJob, &FetchFeedsJob::processedAmountChanged, this, [this](KJob *job, KJob::Unit unit, qulonglong amount) {
qCDebug(kastsFetcher) << "FetchFeedsJob::processedAmountChanged:" << amount;
Q_UNUSED(job);
Q_ASSERT(unit == KJob::Unit::Items);
m_updateProgress = amount;
Q_EMIT updateProgressChanged(m_updateProgress);
});
connect(fetchFeedsJob, &FetchFeedsJob::result, this, [this, fetchFeedsJob]() {
qCDebug(kastsFetcher) << "result slot of FetchFeedsJob";
if (fetchFeedsJob->error() && !fetchFeedsJob->aborted()) {
Q_EMIT error(Error::Type::FeedUpdate, QString(), QString(), fetchFeedsJob->error(), fetchFeedsJob->errorString(), QString());
}
if (m_updating) {
m_updating = false;
Q_EMIT updatingChanged(m_updating);
}
});
fetchFeedsJob->start();
qCDebug(kastsFetcher) << "end of Fetcher::fetch";
}
QString Fetcher::image(const QString &url)
{
if (url.isEmpty()) {
return QLatin1String("no-image");
}
// if image is already cached, then return the path
QString path = StorageManager::instance().imagePath(url);
if (QFileInfo::exists(path)) {
if (QFileInfo(path).size() != 0) {
return QUrl::fromLocalFile(path).toString();
}
}
// avoid restarting an image download if it's already running
if (m_ongoingImageDownloads.contains(url)) {
return QLatin1String("fetching");
}
// if image has not yet been cached, then check for network connectivity if
// possible; and download the image
SolidExtras::NetworkStatus networkStatus;
if (networkStatus.connectivity() == SolidExtras::NetworkStatus::No
|| (networkStatus.metered() == SolidExtras::NetworkStatus::Yes && !SettingsManager::self()->allowMeteredImageDownloads())) {
return QLatin1String("no-image");
}
m_ongoingImageDownloads.insert(url);
QNetworkRequest request((QUrl(url)));
request.setTransferTimeout();
QNetworkReply *reply = get(request);
connect(reply, &QNetworkReply::finished, this, [=]() {
if (reply->isOpen() && !reply->error()) {
QByteArray data = reply->readAll();
QFile file(path);
file.open(QIODevice::WriteOnly);
file.write(data);
file.close();
Q_EMIT downloadFinished(url);
}
m_ongoingImageDownloads.remove(url);
reply->deleteLater();
});
return QLatin1String("fetching");
}
QNetworkReply *Fetcher::download(const QString &url, const QString &filePath) const
{
QNetworkRequest request((QUrl(url)));
request.setTransferTimeout();
QFile *file = new QFile(filePath);
if (file->exists() && file->size() > 0) {
// try to resume download
int resumedAt = file->size();
qCDebug(kastsFetcher) << "Resuming download at" << resumedAt << "bytes";
QByteArray rangeHeaderValue = QByteArray("bytes=") + QByteArray::number(resumedAt) + QByteArray("-");
request.setRawHeader(QByteArray("Range"), rangeHeaderValue);
file->open(QIODevice::WriteOnly | QIODevice::Append);
} else {
qCDebug(kastsFetcher) << "Starting new download";
file->open(QIODevice::WriteOnly);
}
QNetworkReply *reply = get(request);
connect(reply, &QNetworkReply::readyRead, this, [=]() {
if (reply->isOpen() && file) {
QByteArray data = reply->readAll();
file->write(data);
}
});
connect(reply, &QNetworkReply::finished, this, [=]() {
if (reply->isOpen() && file) {
QByteArray data = reply->readAll();
file->write(data);
file->close();
Q_EMIT downloadFinished(url);
}
// clean up; close file if still open in case something has gone wrong
if (file) {
if (file->isOpen()) {
file->close();
}
delete file;
}
reply->deleteLater();
});
return reply;
}
QNetworkReply *Fetcher::get(QNetworkRequest &request) const
{
setHeader(request);
return manager->get(request);
}
QNetworkReply *Fetcher::post(QNetworkRequest &request, const QByteArray &data) const
{
setHeader(request);
request.setHeader(QNetworkRequest::ContentTypeHeader, QLatin1String("application/json"));
return manager->post(request, data);
}
QNetworkReply *Fetcher::head(QNetworkRequest &request) const
{
setHeader(request);
return manager->head(request);
}
void Fetcher::setHeader(QNetworkRequest &request) const
{
request.setRawHeader(QByteArray("User-Agent"), QByteArray("Kasts/") + QByteArray(KASTS_VERSION_STRING) + QByteArray("; Syndication"));
}