rssguard/src/librssguard/network-web/cookiejar.cpp

206 lines
5.9 KiB
C++

// For license of this file, see <project-root-folder>/LICENSE.md.
#include "network-web/cookiejar.h"
#include "3rd-party/boolinq/boolinq.h"
#include "definitions/definitions.h"
#include "miscellaneous/application.h"
#include "miscellaneous/iofactory.h"
#include "miscellaneous/settings.h"
#include <QDateTime>
#include <QDir>
#include <QNetworkCookie>
#include <QSettings>
#if defined(USE_WEBENGINE)
#include <QWebEngineCookieStore>
#include <QWebEngineProfile>
#endif
CookieJar::CookieJar(QObject* parent) : QNetworkCookieJar(parent) {
#if defined(USE_WEBENGINE)
// WebEngine does not store cookies, CookieJar does.
QWebEngineProfile::defaultProfile()
->setPersistentCookiesPolicy(QWebEngineProfile::PersistentCookiesPolicy::NoPersistentCookies);
m_webEngineCookies = QWebEngineProfile::defaultProfile()->cookieStore();
#endif
// Load all cookies and also set them into WebEngine store.
updateSettings();
loadCookies();
#if defined(USE_WEBENGINE)
// When cookies change in WebEngine, then change in main cookie jar too.
//
// Also, the synchronization between WebEngine cookie jar and main cookie jar is this:
// - On app startup, both jars are synchronized to have same cookies.
// - If cookies change in WebEngine jar, the change is propagated to main jar.
// - If cookies change in main jar, cookies are NOT propagated to WebEngine jar.
connect(m_webEngineCookies, &QWebEngineCookieStore::cookieAdded, this, [=](const QNetworkCookie& cookie) {
insertCookieInternal(cookie, false, true);
});
connect(m_webEngineCookies, &QWebEngineCookieStore::cookieRemoved, this, [=](const QNetworkCookie& cookie) {
deleteCookieInternal(cookie, false);
});
#endif
}
QList<QNetworkCookie> CookieJar::extractCookiesFromUrl(const QString& url) {
if (!url.contains(QSL(COOKIE_URL_IDENTIFIER))) {
return {};
}
int index = url.lastIndexOf(QSL(COOKIE_URL_IDENTIFIER), -1, Qt::CaseSensitivity::CaseInsensitive);
QString cookies_string = url.right(url.length() - index - QSL(COOKIE_URL_IDENTIFIER).size());
QStringList cookies_list = cookies_string.split(';');
QList<QNetworkCookie> cookies;
for (const QString& single_cookie : cookies_list) {
const QList<QNetworkCookie>& extracted_cookies = QNetworkCookie::parseCookies(single_cookie.toUtf8());
if (extracted_cookies.isEmpty()) {
continue;
}
QNetworkCookie cookie = extracted_cookies.at(0);
QDateTime date = QDateTime::currentDateTime();
date = date.addYears(30);
cookie.setExpirationDate(date);
cookies.append(cookie);
}
return cookies;
}
void CookieJar::loadCookies() {
Settings* sett = qApp->settings();
auto keys = sett->allKeys(GROUP(Cookies));
for (const QString& cookie_key : qAsConst(keys)) {
QByteArray encoded = sett->password(GROUP(Cookies), cookie_key, {}).toByteArray();
if (!encoded.isEmpty()) {
auto cookie = QNetworkCookie::parseCookies(encoded);
if (!cookie.isEmpty()) {
if (!insertCookieInternal(cookie.at(0), true, false)) {
qCriticalNN << LOGSEC_NETWORK << "Failed to load cookie" << QUOTE_W_SPACE(cookie_key) << "from settings.";
sett->remove(Cookies::ID, cookie_key);
}
}
}
}
}
void CookieJar::saveCookies() {
auto cookies = allCookies();
Settings* sett = qApp->settings();
int i = 1;
sett->beginGroup(GROUP(Cookies));
qobject_cast<QSettings*>(sett)->remove(QString());
sett->endGroup();
for (const QNetworkCookie& cookie : cookies) {
if (cookie.isSessionCookie()) {
continue;
}
sett->setPassword(GROUP(Cookies),
QSL("%1-%2").arg(QString::number(i++), QString::fromUtf8(cookie.name())),
cookie.toRawForm(QNetworkCookie::RawForm::Full));
}
}
QList<QNetworkCookie> CookieJar::cookiesForUrl(const QUrl& url) const {
return QNetworkCookieJar::cookiesForUrl(url);
}
bool CookieJar::setCookiesFromUrl(const QList<QNetworkCookie>& cookie_list, const QUrl& url) {
return QNetworkCookieJar::setCookiesFromUrl(cookie_list, url);
}
bool CookieJar::insertCookieInternal(const QNetworkCookie& cookie, bool notify_others, bool should_save) {
auto result = QNetworkCookieJar::insertCookie(cookie);
if (result) {
if (should_save) {
saveCookies();
}
#if defined(USE_WEBENGINE)
if (notify_others) {
m_webEngineCookies->setCookie(cookie);
}
#else
Q_UNUSED(notify_others)
#endif
}
return result;
}
bool CookieJar::updateCookieInternal(const QNetworkCookie& cookie, bool notify_others) {
auto result = QNetworkCookieJar::updateCookie(cookie);
if (result) {
saveCookies();
#if defined(USE_WEBENGINE)
if (notify_others) {
m_webEngineCookies->setCookie(cookie);
}
#else
Q_UNUSED(notify_others)
#endif
}
return result;
}
bool CookieJar::deleteCookieInternal(const QNetworkCookie& cookie, bool notify_others) {
auto result = QNetworkCookieJar::deleteCookie(cookie);
if (result) {
saveCookies();
#if defined(USE_WEBENGINE)
if (notify_others) {
m_webEngineCookies->deleteCookie(cookie);
}
#else
Q_UNUSED(notify_others)
#endif
}
return result;
}
bool CookieJar::insertCookie(const QNetworkCookie& cookie) {
if (m_ignoreAllCookies) {
return {};
}
else {
return insertCookieInternal(cookie, false, true);
}
}
bool CookieJar::deleteCookie(const QNetworkCookie& cookie) {
return deleteCookieInternal(cookie, false);
}
void CookieJar::updateSettings() {
m_ignoreAllCookies = qApp->settings()->value(GROUP(Network), SETTING(Network::IgnoreAllCookies)).toBool();
if (m_ignoreAllCookies) {
setAllCookies({});
qApp->settings()->remove(GROUP(Cookies));
}
}
bool CookieJar::updateCookie(const QNetworkCookie& cookie) {
return updateCookieInternal(cookie, false);
}