/* * Copyright (C) by Daniel Molkentin * * 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. */ #include "cookiejar.h" #include "configfile.h" #include #include #include #include #include #include namespace OCC { Q_LOGGING_CATEGORY(lcCookieJar, "nextcloud.sync.cookiejar", QtInfoMsg) namespace { const unsigned int JAR_VERSION = 23; } QDataStream &operator<<(QDataStream &stream, const QList &list) { stream << JAR_VERSION; stream << quint32(list.size()); for (const auto &cookie : list) stream << cookie.toRawForm(); return stream; } QDataStream &operator>>(QDataStream &stream, QList &list) { list.clear(); quint32 version = 0; stream >> version; if (version != JAR_VERSION) return stream; quint32 count = 0; stream >> count; for (quint32 i = 0; i < count; ++i) { QByteArray value; stream >> value; QList newCookies = QNetworkCookie::parseCookies(value); if (newCookies.count() == 0 && value.length() != 0) { qCWarning(lcCookieJar) << "CookieJar: Unable to parse saved cookie:" << value; } for (int j = 0; j < newCookies.count(); ++j) list.append(newCookies.at(j)); if (stream.atEnd()) break; } return stream; } CookieJar::CookieJar(QObject *parent) : QNetworkCookieJar(parent) { } CookieJar::~CookieJar() = default; bool CookieJar::setCookiesFromUrl(const QList &cookieList, const QUrl &url) { if (QNetworkCookieJar::setCookiesFromUrl(cookieList, url)) { Q_EMIT newCookiesForUrl(cookieList, url); return true; } return false; } QList CookieJar::cookiesForUrl(const QUrl &url) const { QList cookies = QNetworkCookieJar::cookiesForUrl(url); qCDebug(lcCookieJar) << url << "requests:" << cookies; return cookies; } void CookieJar::clearSessionCookies() { setAllCookies(removeExpired(allCookies())); } bool CookieJar::save(const QString &fileName) { const QFileInfo info(fileName); if (!info.dir().exists()) { info.dir().mkpath("."); } qCDebug(lcCookieJar) << fileName; QFile file(fileName); if (!file.open(QIODevice::WriteOnly)) { return false; } QDataStream stream(&file); stream << removeExpired(allCookies()); file.close(); return true; } bool CookieJar::restore(const QString &fileName) { const QFileInfo info(fileName); if (!info.exists()) { return false; } QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) { return false; } QDataStream stream(&file); QList list; stream >> list; setAllCookies(removeExpired(list)); file.close(); return true; } QList CookieJar::removeExpired(const QList &cookies) { QList updatedList; foreach (const QNetworkCookie &cookie, cookies) { if (cookie.expirationDate() > QDateTime::currentDateTimeUtc() && !cookie.isSessionCookie()) { updatedList << cookie; } } return updatedList; } } // namespace OCC