konqueror/add-kwallet-support-to-webengine.patch

1257 lines
44 KiB
Diff
Raw Normal View History

From 56ea2a32d2f1d9f4e6c97b27080cf8144a932ee3 Mon Sep 17 00:00:00 2001
From: Stefano Crocco <stefano.crocco@alice.it>
Date: Sun, 14 Jan 2018 11:25:17 +0100
Subject: Add back KWallet support to Konqueror
Summary:
Added back the ability to store user names and passwords to KWallet and automatically fill them when loading a page.
I copied the files `kwebwallet.cpp` and `kwebwallet.h` from the `KDEWebKit` framework and modified it to adapt to `QWebEngine`. In particular, I needed to modify the way form data is passed from javascript to C++ serializing it using JSON and read it back using `QJsonDocument`. Also, since `QWebEnginePage` doesn't allow to access frames from the C++ side, I needed to remove all the code dealing with recurisve searches and implement it in javascript.
As the old code, this intentionally doesn't work with input elements with the `autocomplete` attribute set to `false`.
I've not modified the copyright information in the files I copied from `KDEWebKit` as I don't know which is the correct way to do so: I must make clear my files are based on the work on other people but at the same time that I've changed them: how do I do this?
Test Plan: I tried to access several sites requiring a login in, asking Konqueror to save the password, then logging out and visiting the page again. Except when the `autocomplete` attribute was `false` (checked by looking at the html code), Konqueror always saved and restored login information correctly.
Reviewers: dfaure
Reviewed By: dfaure
Differential Revision: https://phabricator.kde.org/D10178
---
webenginepart/src/CMakeLists.txt | 4 +
webenginepart/src/webenginepage.cpp | 14 +-
webenginepart/src/webenginepage.h | 7 +-
webenginepart/src/webenginepart.cpp | 62 +--
webenginepart/src/webenginepart.h | 5 +
webenginepart/src/webenginepartfactory.cpp | 1 +
webenginepart/src/webenginewallet.cpp | 599 +++++++++++++++++++++++++++++
webenginepart/src/webenginewallet.h | 305 +++++++++++++++
8 files changed, 964 insertions(+), 33 deletions(-)
create mode 100644 webenginepart/src/webenginewallet.cpp
create mode 100644 webenginepart/src/webenginewallet.h
diff --git a/webenginepart/src/CMakeLists.txt b/webenginepart/src/CMakeLists.txt
index bfac067..fffce88 100644
--- a/webenginepart/src/CMakeLists.txt
+++ b/webenginepart/src/CMakeLists.txt
@@ -1,3 +1,5 @@
+find_package(KF5 ${KF5_MIN_VERSION} REQUIRED COMPONENTS Wallet)
+
include_directories(${CMAKE_CURRENT_SOURCE_DIR} ${CMAKE_BINARY_DIR})
set(kwebenginepartlib_LIB_SRCS
@@ -8,6 +10,7 @@ set(kwebenginepartlib_LIB_SRCS
websslinfo.cpp
webhistoryinterface.cpp
webenginepartdownloadmanager.cpp
+ webenginewallet.cpp
settings/webenginesettings.cpp
settings/webengine_filter.cpp
ui/searchbar.cpp
@@ -27,6 +30,7 @@ target_link_libraries(kwebenginepartlib Qt5::Core Qt5::DBus Qt5::Gui Qt5::Widget
KF5::Parts
KF5::SonnetCore
KF5::WindowSystem # for KUserTimestamp
+ KF5::Wallet
)
target_include_directories(kwebenginepartlib PUBLIC
diff --git a/webenginepart/src/webenginepage.cpp b/webenginepart/src/webenginepage.cpp
index 212e189..615e8f7 100644
--- a/webenginepart/src/webenginepage.cpp
+++ b/webenginepart/src/webenginepage.cpp
@@ -27,6 +27,7 @@
#include "webengineview.h"
#include "settings/webenginesettings.h"
#include "webenginepartdownloadmanager.h"
+#include "webenginewallet.h"
#include <QWebEngineSettings>
#include <QWebEngineProfile>
@@ -72,7 +73,8 @@ WebEnginePage::WebEnginePage(WebEnginePart *part, QWidget *parent)
m_kioErrorCode(0),
m_ignoreError(false),
m_part(part),
- m_passwdServerClient(new KPasswdServerClient)
+ m_passwdServerClient(new KPasswdServerClient),
+ m_wallet(Q_NULLPTR)
{
if (view())
WebEngineSettings::self()->computeFontSizes(view()->logicalDpiY());
@@ -94,6 +96,7 @@ WebEnginePage::WebEnginePage(WebEnginePart *part, QWidget *parent)
this->profile()->setHttpUserAgent(this->profile()->httpUserAgent() + " Konqueror (WebEnginePart)");
}
WebEnginePartDownloadManager::instance()->addPage(this);
+ m_wallet = new WebEngineWallet(this, parent ? parent->window()->winId() : 0);
}
WebEnginePage::~WebEnginePage()
@@ -204,8 +207,9 @@ bool WebEnginePage::acceptNavigationRequest(const QUrl& url, NavigationType type
bool inPageRequest = true;
switch (type) {
case QWebEnginePage::NavigationTypeFormSubmitted:
- //if (!checkFormData(request))
- // return false;
+ if (!checkFormData(url))
+ return false;
+ m_wallet->saveFormData(this);
break;
#if 0
case QWebEnginePage::NavigationTypeFormResubmitted:
@@ -548,9 +552,9 @@ bool WebEnginePage::checkLinkSecurity(const QNetworkRequest &req, NavigationType
return true;
}
-bool WebEnginePage::checkFormData(const QNetworkRequest &req) const
+bool WebEnginePage::checkFormData(const QUrl &url) const
{
- const QString scheme (req.url().scheme());
+ const QString scheme (url.scheme());
if (m_sslInfo.isValid() &&
!scheme.compare(QL1S("https")) && !scheme.compare(QL1S("mailto")) &&
diff --git a/webenginepart/src/webenginepage.h b/webenginepart/src/webenginepage.h
index 8fa0386..cb3d5ed 100644
--- a/webenginepart/src/webenginepage.h
+++ b/webenginepart/src/webenginepage.h
@@ -40,6 +40,7 @@ class WebSslInfo;
class WebEnginePart;
class QWebEngineDownloadItem;
class KPasswdServerClient;
+class WebEngineWallet;
class WebEnginePage : public QWebEnginePage
{
@@ -64,6 +65,8 @@ public:
void download(const QUrl &url, bool newWindow = false);
+ WebEngineWallet* wallet() const {return m_wallet;}
+
Q_SIGNALS:
/**
* This signal is emitted whenever a user cancels/aborts a load resource
@@ -107,7 +110,7 @@ protected Q_SLOTS:
private:
bool checkLinkSecurity(const QNetworkRequest& req, NavigationType type) const;
- bool checkFormData(const QNetworkRequest& req) const;
+ bool checkFormData(const QUrl& url) const;
bool handleMailToUrl (const QUrl& , NavigationType type) const;
void setPageJScriptPolicy(const QUrl& url);
@@ -121,6 +124,7 @@ private:
QPointer<WebEnginePart> m_part;
QScopedPointer<KPasswdServerClient> m_passwdServerClient;
+ WebEngineWallet *m_wallet;
};
@@ -158,6 +162,7 @@ private:
KParts::WindowArgs m_windowArgs;
WebWindowType m_type;
bool m_createNewWindow;
+ WebEngineWallet* m_wallet;
};
#endif // WEBENGINEPAGE_H
diff --git a/webenginepart/src/webenginepart.cpp b/webenginepart/src/webenginepart.cpp
index 10bbffc..66edce7 100644
--- a/webenginepart/src/webenginepart.cpp
+++ b/webenginepart/src/webenginepart.cpp
@@ -34,6 +34,7 @@
#include "webenginepage.h"
#include "websslinfo.h"
#include "webhistoryinterface.h"
+#include "webenginewallet.h"
#include "ui/searchbar.h"
#include "ui/passwordbar.h"
@@ -80,7 +81,8 @@ WebEnginePart::WebEnginePart(QWidget *parentWidget, QObject *parent,
m_statusBarWalletLabel(0),
m_searchBar(0),
m_passwordBar(0),
- m_featurePermissionBar(0)
+ m_featurePermissionBar(0),
+ m_wallet(Q_NULLPTR)
{
KAboutData about = KAboutData(QStringLiteral("webenginepart"),
i18nc("Program Name", "WebEnginePart"),
@@ -168,6 +170,7 @@ WebEnginePart::WebEnginePart(QWidget *parentWidget, QObject *parent,
// Load plugins once we are fully ready
loadPlugins();
+ setWallet(page()->wallet());
}
WebEnginePart::~WebEnginePart()
@@ -311,17 +314,25 @@ void WebEnginePart::connectWebEnginePageSignals(WebEnginePage* page)
m_browserExtension->setIconUrl(url);
}
});
+}
-#if 0
- KWebWallet *wallet = page->wallet();
- if (wallet) {
- connect(wallet, SIGNAL(saveFormDataRequested(QString,QUrl)),
- this, SLOT(slotSaveFormDataRequested(QString,QUrl)));
- connect(wallet, SIGNAL(fillFormRequestCompleted(bool)),
- this, SLOT(slotFillFormRequestCompleted(bool)));
- connect(wallet, SIGNAL(walletClosed()), this, SLOT(slotWalletClosed()));
+void WebEnginePart::setWallet(WebEngineWallet* wallet)
+{
+ if(m_wallet){
+ disconnect(m_wallet, &WebEngineWallet::saveFormDataRequested,
+ this, &WebEnginePart::slotSaveFormDataRequested);
+ disconnect(m_wallet, &WebEngineWallet::fillFormRequestCompleted,
+ this, &WebEnginePart::slotFillFormRequestCompleted);
+ disconnect(m_wallet, &WebEngineWallet::walletClosed, this, &WebEnginePart::slotWalletClosed);
+ }
+ m_wallet = wallet;
+ if (m_wallet) {
+ connect(m_wallet, &WebEngineWallet::saveFormDataRequested,
+ this, &WebEnginePart::slotSaveFormDataRequested);
+ connect(m_wallet, &WebEngineWallet::fillFormRequestCompleted,
+ this, &WebEnginePart::slotFillFormRequestCompleted);
+ connect(m_wallet, &WebEngineWallet::walletClosed, this, &WebEnginePart::slotWalletClosed);
}
-#endif
}
bool WebEnginePart::openUrl(const QUrl &_u)
@@ -451,15 +462,15 @@ void WebEnginePart::slotLoadFinished (bool ok)
}
if (!Utils::isBlankUrl(url())) {
m_hasCachedFormData = false;
-
if (WebEngineSettings::self()->isNonPasswordStorableSite(url().host())) {
addWalletStatusBarIcon();
- } else {
+ }
+ else {
// Attempt to fill the web form...
-// KWebWallet *webWallet = page() ? page()->wallet() : 0;
-// if (webWallet) {
-// webWallet->fillFormData(frame, false);
-// }
+ WebEngineWallet *wallet = page() ? page()->wallet() : 0;
+ if (wallet){
+ wallet->fillFormData(page());
+ }
}
}
@@ -766,10 +777,10 @@ void WebEnginePart::slotDeleteNonPasswordStorableSite()
void WebEnginePart::slotRemoveCachedPasswords()
{
- if (!page()) // || !page()->wallet())
+ if (!page() || !page()->wallet())
return;
-// page()->wallet()->removeFormData(page()->mainFrame(), true);
+ page()->wallet()->removeFormData(page());
m_hasCachedFormData = false;
}
@@ -822,8 +833,8 @@ void WebEnginePart::slotShowFeaturePermissionBar(QWebEnginePage::Feature feature
this, SLOT(slotFeaturePermissionGranted(QWebEnginePage::Feature)));
connect(m_featurePermissionBar, SIGNAL(permissionDenied(QWebEnginePage::Feature)),
this, SLOT(slotFeaturePermissionDenied(QWebEnginePage::Feature)));
-// connect(m_passwordBar, SIGNAL(done()),
-// this, SLOT(slotSaveFormDataDone()));
+ connect(m_passwordBar, SIGNAL(done()),
+ this, SLOT(slotSaveFormDataDone()));
QBoxLayout* lay = qobject_cast<QBoxLayout*>(widget()->layout());
if (lay)
lay->insertWidget(0, m_featurePermissionBar);
@@ -862,19 +873,16 @@ void WebEnginePart::slotSaveFormDataRequested (const QString& key, const QUrl& u
if (!m_passwordBar) {
m_passwordBar = new PasswordBar(widget());
-#if 0
- KWebWallet* wallet = page()->wallet();
- if (!wallet) {
- kWarning() << "No wallet instance found! This should never happen!";
+ if (!m_wallet) {
+ qDebug() << "No m_wallet instance found! This should never happen!";
return;
}
connect(m_passwordBar, SIGNAL(saveFormDataAccepted(QString)),
- wallet, SLOT(acceptSaveFormDataRequest(QString)));
+ m_wallet, SLOT(acceptSaveFormDataRequest(QString)));
connect(m_passwordBar, SIGNAL(saveFormDataRejected(QString)),
- wallet, SLOT(rejectSaveFormDataRequest(QString)));
+ m_wallet, SLOT(rejectSaveFormDataRequest(QString)));
connect(m_passwordBar, SIGNAL(done()),
this, SLOT(slotSaveFormDataDone()));
-#endif
}
Q_ASSERT(m_passwordBar);
diff --git a/webenginepart/src/webenginepart.h b/webenginepart/src/webenginepart.h
index 6889e6d..91afa42 100644
--- a/webenginepart/src/webenginepart.h
+++ b/webenginepart/src/webenginepart.h
@@ -42,6 +42,7 @@ class PasswordBar;
class FeaturePermissionBar;
class KUrlLabel;
class WebEngineBrowserExtension;
+class WebEngineWallet;
/**
* A KPart wrapper for the QtWebEngine's browser rendering engine.
@@ -98,6 +99,9 @@ public:
void connectWebEnginePageSignals(WebEnginePage* page);
void slotShowFeaturePermissionBar(QWebEnginePage::Feature);
+
+ void setWallet(WebEngineWallet* wallet);
+
protected:
/**
* Re-implemented for internal reasons. API remains unaffected.
@@ -160,6 +164,7 @@ private:
WebEngineBrowserExtension* m_browserExtension;
KParts::StatusBarExtension* m_statusBarExtension;
WebEngineView* m_webView;
+ WebEngineWallet* m_wallet;
};
#endif // WEBENGINEPART_H
diff --git a/webenginepart/src/webenginepartfactory.cpp b/webenginepart/src/webenginepartfactory.cpp
index 04853bd..7a877b4 100644
--- a/webenginepart/src/webenginepartfactory.cpp
+++ b/webenginepart/src/webenginepartfactory.cpp
@@ -24,6 +24,7 @@
#include <QWidget>
+
WebEngineFactory::~WebEngineFactory()
{
// kDebug() << this;
diff --git a/webenginepart/src/webenginewallet.cpp b/webenginepart/src/webenginewallet.cpp
new file mode 100644
index 0000000..f501b27
--- /dev/null
+++ b/webenginepart/src/webenginewallet.cpp
@@ -0,0 +1,599 @@
+/*
+ * This file is part of the KDE project.
+ *
+ * Copyright (C) 2009 Dawit Alemayehu <adawit@kde.org>
+ * Copyright (C) 2018 Stefano Crocco <stefano.crocco@alice.it>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public License
+ * along with this library; see the file COPYING.LIB. If not, write to
+ * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include "webenginewallet.h"
+#include "webenginepage.h"
+
+#include <KWallet>
+#include <QDebug>
+
+#include <QtCore/QSet>
+#include <QtCore/QHash>
+#include <QtCore/QFile>
+#include <QtCore/QPointer>
+#include <QtCore/QScopedPointer>
+#include <QJsonDocument>
+#include <qwindowdefs.h>
+
+#define QL1S(x) QLatin1String(x)
+#define QL1C(x) QLatin1Char(x)
+
+// Javascript used to extract/set data from <form> elements.
+static const char s_fillableFormElementExtractorJs[] = "(function(){ "
+" function findFormsRecursive(wnd, existingList, path){"
+" findFormsInFrame(wnd, existingList, path);"
+" frameList = wnd.frames;"
+" for(var i = 0; i < frameList.length; ++i) {"
+" var newPath = path.concat(i);"
+" findFormsRecursive(frameList[i], existingList, newPath);"
+" }"
+" }"
+" function findFormsInFrame(frm, existingList, path){"
+" var url = frm.location;"
+" var formList;"
+" try{ formList = frm.document.forms; } "
+" catch(e){"
+" return;"
+" }"
+" if (formList.length > 0) { "
+" for (var i = 0; i < formList.length; ++i) { "
+" var inputList = formList[i].elements; "
+" if (inputList.length < 1) { "
+" continue; "
+" } "
+" var formObject = new Object; "
+" formObject.url = url;"
+" formObject.name = formList[i].name; "
+" if (typeof(formObject.name) != 'string') { "
+" formObject.name = String(formList[i].id); "
+" } "
+" formObject.index = i; "
+" formObject.elements = new Array; "
+" for (var j = 0; j < inputList.length; ++j) { "
+" if (inputList[j].type != 'text' && inputList[j].type != 'email' && inputList[j].type != 'password') { "
+" continue; "
+" } "
+" if (inputList[j].disabled || inputList[j].autocomplete == 'off') { "
+" continue; "
+" } "
+" var element = new Object; "
+" element.name = inputList[j].name; "
+" if (typeof(element.name) != 'string' ) { "
+" element.name = String(inputList[j].id); "
+" } "
+" element.value = String(inputList[j].value); "
+" element.type = String(inputList[j].type); "
+" element.readonly = Boolean(inputList[j].readOnly); "
+" formObject.elements.push(element); "
+" } "
+" if (formObject.elements.length > 0) { "
+" formObject.framePath = path;"
+" console.log(JSON.stringify(formObject));"
+" existingList.push(JSON.stringify(formObject)); "
+" } "
+" } "
+" } "
+" }"
+" var forms = new Array;"
+" findFormsRecursive(window, forms, []);"
+" return forms;"
+"})()";
+
+//javascript used to fill a single form element
+static const char s_javascriptFillInputFragment[] = "var frm = window;"
+" for(var i=0; i < [%1].length; ++i) frm=frm.frames[i];"
+" if (frm.document.forms['%2'] && frm.document.forms['%2'].elements['%3']){"
+" frm.document.forms['%2'].elements['%3'].value='%4';\n"
+" }";
+
+
+/**
+ * Creates key used to store and retrieve form data.
+ *
+ */
+static QString walletKey(WebEngineWallet::WebForm form)
+{
+ QString key = form.url.toString(QUrl::RemoveQuery | QUrl::RemoveFragment);
+ key += QL1C('#');
+ key += form.name;
+ return key;
+}
+
+static QUrl urlForFrame(const QUrl &frameUrl, const QUrl &pageUrl)
+{
+ return (frameUrl.isEmpty() || frameUrl.isRelative() ? pageUrl.resolved(frameUrl) : frameUrl);
+}
+
+class WebEngineWallet::WebEngineWalletPrivate
+{
+public:
+ struct FormsData {
+ QPointer<WebEnginePage> page;
+ WebEngineWallet::WebFormList forms;
+ };
+
+ typedef std::function<void(const WebEngineWallet::WebFormList &)> WebWalletCallback;
+
+ WebEngineWalletPrivate(WebEngineWallet *parent);
+
+ void withFormData(WebEnginePage *page, WebWalletCallback callback, bool fillform = true, bool ignorepasswd = false);
+ WebFormList parseFormData(const QVariant &result, const QUrl &url, bool fillform = true, bool ignorepasswd = false);
+ void performFormDataParsing(const QVariant &result, const QUrl &url, WebWalletCallback callback, bool fillform, bool ignorepasswd);
+ void fillDataFromCache(WebEngineWallet::WebFormList &formList);
+ void saveDataToCache(const QString &key);
+ void removeDataFromCache(const WebFormList &formList);
+ void openWallet();
+
+ // Private slots...
+ void _k_openWalletDone(bool);
+ void _k_walletClosed();
+
+ WId wid;
+ WebEngineWallet *q;
+ QScopedPointer<KWallet::Wallet> wallet;
+ WebEngineWallet::WebFormList pendingRemoveRequests;
+ QHash<QUrl, FormsData> pendingFillRequests;
+ QHash<QString, WebEngineWallet::WebFormList> pendingSaveRequests;
+ QSet<QUrl> confirmSaveRequestOverwrites;
+};
+
+WebEngineWallet::WebEngineWalletPrivate::WebEngineWalletPrivate(WebEngineWallet *parent)
+ : wid(0), q(parent)
+{
+}
+
+WebEngineWallet::WebFormList WebEngineWallet::WebEngineWalletPrivate::parseFormData(const QVariant &result, const QUrl &url, bool fillform, bool ignorepasswd)
+{
+ const QVariantList results(result.toList());
+ WebEngineWallet::WebFormList list;
+ Q_FOREACH (const QVariant &formVariant, results) {
+ QJsonDocument doc = QJsonDocument::fromJson(formVariant.toString().toUtf8());
+ const QVariantMap map = doc.toVariant().toMap();
+ WebEngineWallet::WebForm form;
+ form.url = urlForFrame(QUrl(map[QL1S("url")].toString()), url);
+ form.name = map[QL1S("name")].toString();
+ form.index = map[QL1S("index")].toString();
+ form.framePath = map["framePath"].toStringList().join(",");
+ bool formHasPasswords = false;
+ const QVariantList elements = map[QL1S("elements")].toList();
+ QVector<WebEngineWallet::WebForm::WebField> inputFields;
+ Q_FOREACH (const QVariant &element, elements) {
+ QVariantMap elementMap(element.toMap());
+ const QString name(elementMap[QL1S("name")].toString());
+ const QString value(ignorepasswd ? QString() : elementMap[QL1S("value")].toString());
+ if (name.isEmpty()) {
+ continue;
+ }
+ if (fillform && elementMap[QL1S("readonly")].toBool()) {
+ continue;
+ }
+ if (elementMap[QL1S("type")].toString().compare(QL1S("password"), Qt::CaseInsensitive) == 0) {
+ if (!fillform && value.isEmpty()) {
+ continue;
+ }
+ formHasPasswords = true;
+ }
+ inputFields.append(qMakePair(name, value));
+ }
+ // Only add the input fields on form save requests...
+ if (formHasPasswords || fillform) {
+ form.fields = inputFields;
+ }
+
+ // Add the form to the list if we are saving it or it has cached data.
+ if ((fillform && q->hasCachedFormData(form)) || (!fillform && !form.fields.isEmpty())) {
+ list << form;
+ }
+ }
+ return list;
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::withFormData(WebEnginePage* page, WebWalletCallback callback, bool fillform, bool ignorepasswd)
+{
+ Q_ASSERT(page);
+ QUrl url = page->url();
+ auto internalCallback = [this, url, fillform, ignorepasswd, callback](const QVariant &result){
+ WebFormList res = parseFormData(result, url, fillform, ignorepasswd);
+ callback(res);
+ };
+ page->runJavaScript(QL1S(s_fillableFormElementExtractorJs), internalCallback);
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::fillDataFromCache(WebEngineWallet::WebFormList &formList)
+{
+ if (!wallet) {
+ qWarning() << "Unable to retrieve form data from wallet";
+ return;
+ }
+
+ QString lastKey;
+ QMap<QString, QString> cachedValues;
+ QMutableVectorIterator <WebForm> formIt(formList);
+
+ while (formIt.hasNext()) {
+ WebEngineWallet::WebForm &form = formIt.next();
+ const QString key(walletKey(form));
+ if (key != lastKey && wallet->readMap(key, cachedValues) != 0) {
+ qWarning() << "Unable to read form data for key:" << key;
+ continue;
+ }
+
+ for (int i = 0, count = form.fields.count(); i < count; ++i) {
+ form.fields[i].second = cachedValues.value(form.fields[i].first);
+ }
+ lastKey = key;
+ }
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::saveDataToCache(const QString &key)
+{
+ // Make sure the specified keys exists before acting on it. See BR# 270209.
+ if (!pendingSaveRequests.contains(key)) {
+ return;
+ }
+
+ bool success = false;
+ const QUrl url = pendingSaveRequests.value(key).first().url;
+
+ if (wallet) {
+ int count = 0;
+ const WebEngineWallet::WebFormList list = pendingSaveRequests.value(key);
+ QVectorIterator<WebEngineWallet::WebForm> formIt(list);
+
+ while (formIt.hasNext()) {
+ QMap<QString, QString> values, storedValues;
+ const WebEngineWallet::WebForm form = formIt.next();
+ const QString accessKey = walletKey(form);
+ if (confirmSaveRequestOverwrites.contains(url)) {
+ confirmSaveRequestOverwrites.remove(url);
+ const int status = wallet->readMap(accessKey, storedValues);
+ if (status == 0 && storedValues.count()) {
+ QVectorIterator<WebEngineWallet::WebForm::WebField> fieldIt(form.fields);
+ while (fieldIt.hasNext()) {
+ const WebEngineWallet::WebForm::WebField field = fieldIt.next();
+ if (storedValues.contains(field.first) &&
+ storedValues.value(field.first) != field.second) {
+ emit q->saveFormDataRequested(key, url);
+ return;
+ }
+ }
+ // If we got here it means the new credential is exactly
+ // the same as the one already cached ; so skip the
+ // re-saving part...
+ success = true;
+ continue;
+ }
+ }
+ QVectorIterator<WebEngineWallet::WebForm::WebField> fieldIt(form.fields);
+ while (fieldIt.hasNext()) {
+ const WebEngineWallet::WebForm::WebField field = fieldIt.next();
+ values.insert(field.first, field.second);
+ }
+
+ if (wallet->writeMap(accessKey, values) == 0) {
+ count++;
+ } else {
+ qWarning() << "Unable to write form data to wallet";
+ }
+ }
+
+ if (list.isEmpty() || count > 0) {
+ success = true;
+ }
+
+ pendingSaveRequests.remove(key);
+ } else {
+ qWarning() << "NULL Wallet instance!";
+ }
+
+ emit q->saveFormDataCompleted(url, success);
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::openWallet()
+{
+ if (!wallet.isNull()) {
+ return;
+ }
+
+ wallet.reset(KWallet::Wallet::openWallet(KWallet::Wallet::NetworkWallet(),
+ wid, KWallet::Wallet::Asynchronous));
+
+ if (wallet.isNull()) {
+ return;
+ }
+
+ connect(wallet.data(), SIGNAL(walletOpened(bool)), q, SLOT(_k_openWalletDone(bool)));
+ connect(wallet.data(), SIGNAL(walletClosed()), q, SLOT(_k_walletClosed()));
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::removeDataFromCache(const WebFormList &formList)
+{
+ if (!wallet) {
+ qWarning() << "NULL Wallet instance!";
+ return;
+ }
+
+ QVectorIterator<WebForm> formIt(formList);
+ while (formIt.hasNext()) {
+ wallet->removeEntry(walletKey(formIt.next()));
+ }
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::_k_openWalletDone(bool ok)
+{
+ Q_ASSERT(wallet);
+
+ if (ok &&
+ (wallet->hasFolder(KWallet::Wallet::FormDataFolder()) ||
+ wallet->createFolder(KWallet::Wallet::FormDataFolder())) &&
+ wallet->setFolder(KWallet::Wallet::FormDataFolder())) {
+
+ // Do pending fill requests...
+ if (!pendingFillRequests.isEmpty()) {
+ QList<QUrl> urlList;
+ QMutableHashIterator<QUrl, FormsData> requestIt(pendingFillRequests);
+ while (requestIt.hasNext()) {
+ requestIt.next();
+ WebEngineWallet::WebFormList list = requestIt.value().forms;
+ fillDataFromCache(list);
+ q->fillWebForm(requestIt.key(), list);
+ }
+
+ pendingFillRequests.clear();
+ }
+
+ // Do pending save requests...
+ if (!pendingSaveRequests.isEmpty()) {
+ QListIterator<QString> keysIt(pendingSaveRequests.keys());
+ while (keysIt.hasNext()) {
+ saveDataToCache(keysIt.next());
+ }
+ }
+
+ // Do pending remove requests...
+ if (!pendingRemoveRequests.isEmpty()) {
+ removeDataFromCache(pendingRemoveRequests);
+ pendingRemoveRequests.clear();
+ }
+ } else {
+ // Delete the wallet if opening the wallet failed or we were unable
+ // to change to the folder we wanted to change to.
+ delete wallet.take();
+ }
+}
+
+void WebEngineWallet::WebEngineWalletPrivate::_k_walletClosed()
+{
+ if (wallet) {
+ wallet.take()->deleteLater();
+ }
+
+ emit q->walletClosed();
+}
+
+WebEngineWallet::WebEngineWallet(QObject *parent, WId wid)
+ : QObject(parent), d(new WebEngineWalletPrivate(this))
+{
+ d->wid = wid;
+}
+
+WebEngineWallet::~WebEngineWallet()
+{
+ delete d;
+}
+
+void WebEngineWallet::fillFormDataCallback(WebEnginePage* page, const WebEngineWallet::WebFormList& formsList)
+{
+ QList<QUrl> urlList;
+ if (!formsList.isEmpty()) {
+ const QUrl url(page->url());
+ if (d->pendingFillRequests.contains(url)) {
+ qWarning() << "Duplicate request rejected!";
+ } else {
+ WebEngineWalletPrivate::FormsData data;
+ data.page = QPointer<WebEnginePage>(page);
+ data.forms << formsList;
+ d->pendingFillRequests.insert(url, data);
+ urlList << url;
+ }
+ }
+
+ if (!urlList.isEmpty()) {
+ fillFormDataFromCache(urlList);
+ }
+
+}
+
+void WebEngineWallet::fillFormData(WebEnginePage *page)
+{
+ if (!page) return;
+ auto callback = [this, page](const WebFormList &forms){
+ fillFormDataCallback(page, forms);
+ };
+ d->withFormData(page, callback);
+}
+
+static void createSaveKeyFor(WebEnginePage *page, QString *key)
+{
+ QUrl pageUrl(page->url());
+ pageUrl.setPassword(QString());
+
+ QString keyStr = pageUrl.toString();
+
+ *key = QString::number(qHash(keyStr), 16);
+}
+
+void WebEngineWallet::saveFormData(WebEnginePage *page, bool ignorePasswordFields)
+{
+ if (!page) {
+ return;
+ }
+
+ QString key;
+ createSaveKeyFor(page, &key);
+ if (d->pendingSaveRequests.contains(key)) {
+ return;
+ }
+
+ QUrl url = page->url();
+ auto callback = [this, key, url](const WebFormList &list){saveFormDataCallback(key, url, list);};
+ d->withFormData(page, callback, false, ignorePasswordFields);
+}
+
+void WebEngineWallet::saveFormDataCallback(const QString &key, const QUrl& url, const WebEngineWallet::WebFormList& formsList)
+{
+
+ if (formsList.isEmpty()) {
+ return;
+ }
+
+ WebFormList list(formsList);
+
+ d->pendingSaveRequests.insert(key, list);
+
+ QMutableVectorIterator<WebForm> it(list);
+ while (it.hasNext()) {
+ const WebForm form(it.next());
+ if (hasCachedFormData(form)) {
+ it.remove();
+ }
+ }
+
+ if (list.isEmpty()) {
+ d->confirmSaveRequestOverwrites.insert(url);
+ saveFormDataToCache(key);
+ return;
+ }
+
+ emit saveFormDataRequested(key, url);
+}
+
+void WebEngineWallet::removeFormData(WebEnginePage *page)
+{
+ if (page) {
+ auto callback = [this](const WebFormList &list){removeFormDataFromCache(list);};
+ d->withFormData(page, callback);
+ }
+}
+
+void WebEngineWallet::removeFormDataCallback(const WebFormList& list)
+{
+ removeFormDataFromCache(list);
+}
+
+
+void WebEngineWallet::removeFormData(const WebFormList &forms)
+{
+ d->pendingRemoveRequests << forms;
+ removeFormDataFromCache(forms);
+}
+
+void WebEngineWallet::acceptSaveFormDataRequest(const QString &key)
+{
+ saveFormDataToCache(key);
+}
+
+void WebEngineWallet::rejectSaveFormDataRequest(const QString &key)
+{
+ d->pendingSaveRequests.remove(key);
+}
+
+void WebEngineWallet::fillWebForm(const QUrl &url, const WebEngineWallet::WebFormList &forms)
+{
+ QPointer<WebEnginePage> page = d->pendingFillRequests.value(url).page;
+ if (!page) {
+ return;
+ }
+
+ QString script;
+ bool wasFilled = false;
+
+ Q_FOREACH (const WebEngineWallet::WebForm &form, forms) {
+ Q_FOREACH (const WebEngineWallet::WebForm::WebField &field, form.fields) {
+ QString value = field.second;
+ value.replace(QL1C('\\'), QL1S("\\\\"));
+ script+= QString(s_javascriptFillInputFragment)
+ .arg(form.framePath)
+ .arg((form.name.isEmpty() ? form.index : form.name))
+ .arg(field.first).arg(value);
+ }
+ }
+ if (!script.isEmpty()) {
+ wasFilled = true;
+ auto callback = [wasFilled, this](const QVariant &){emit fillFormRequestCompleted(wasFilled);};
+ page.data()->runJavaScript(script, callback);
+ }
+}
+
+WebEngineWallet::WebFormList WebEngineWallet::formsToFill(const QUrl &url) const
+{
+ return d->pendingFillRequests.value(url).forms;
+}
+
+WebEngineWallet::WebFormList WebEngineWallet::formsToSave(const QString &key) const
+{
+ return d->pendingSaveRequests.value(key);
+}
+
+bool WebEngineWallet::hasCachedFormData(const WebForm &form) const
+{
+ return !KWallet::Wallet::keyDoesNotExist(KWallet::Wallet::NetworkWallet(),
+ KWallet::Wallet::FormDataFolder(),
+ walletKey(form));
+}
+
+void WebEngineWallet::fillFormDataFromCache(const QList<QUrl> &urlList)
+{
+ if (d->wallet) {
+ QListIterator<QUrl> urlIt(urlList);
+ while (urlIt.hasNext()) {
+ const QUrl url = urlIt.next();
+ WebFormList list = formsToFill(url);
+ d->fillDataFromCache(list);
+ fillWebForm(url, list);
+ }
+ d->pendingFillRequests.clear();
+ }
+ d->openWallet();
+}
+
+void WebEngineWallet::saveFormDataToCache(const QString &key)
+{
+ if (d->wallet) {
+ d->saveDataToCache(key);
+ return;
+ }
+ d->openWallet();
+}
+
+void WebEngineWallet::removeFormDataFromCache(const WebFormList &forms)
+{
+ if (d->wallet) {
+ d->removeDataFromCache(forms);
+ d->pendingRemoveRequests.clear();
+ return;
+ }
+ d->openWallet();
+}
+
+#include "moc_webenginewallet.cpp"
diff --git a/webenginepart/src/webenginewallet.h b/webenginepart/src/webenginewallet.h
new file mode 100644
index 0000000..2c3d65b
--- /dev/null
+++ b/webenginepart/src/webenginewallet.h
@@ -0,0 +1,305 @@
+/*
+ * This file is part of the KDE project.
+ *
+ * Copyright (C) 2009 Dawit Alemayehu <adawit@kde.org>
+ * Copyright (C) 2018 Stefano Crocco <stefano.crocco@alice.it>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public License
+ * along with this library; see the file COPYING.LIB. If not, write to
+ * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+ *
+ */
+#ifndef WEBENGINEWALLET_H
+#define WEBENGINEWALLET_H
+
+#include <QtCore/QObject>
+#include <QtCore/QString>
+#include <QtCore/QList>
+#include <QtCore/QPair>
+#include <QUrl>
+#include <QWidget>
+#include <QtCore/QtGlobal>
+
+class WebEnginePage;
+
+class WebEngineWallet : public QObject
+{
+ Q_OBJECT
+
+public:
+
+ /**
+ * Holds data from a HTML &lt;form&gt; element.
+ */
+ struct WebForm {
+ /**
+ * A typedef for storing the name and value attributes of HTML &lt;input&gt;
+ * elements.
+ */
+ typedef QPair<QString, QString> WebField;
+
+ /** The URL the form was found at. */
+ QUrl url;
+ /** The name attribute of the form. */
+ QString name;
+ /** The position of the form on the web page, relative to other forms. */
+ QString index;
+ /** The path of the frame the form belongs to relative to the toplevel window (in the javascript sense).
+ *
+ * This is stored as a string containing a javascript array (it is passed as is to javascript code, so no need to store it in C++ format
+ */
+ QString framePath;
+ /** The name and value attributes of each input element in the form. */
+ QVector<WebField> fields;
+ };
+
+ /**
+ * A list of web forms
+ */
+ typedef QVector<WebForm> WebFormList;
+
+ /**
+ * Constructs a WebEngineWebWallet
+ *
+ * @p parent is usually the WebEnginePage this wallet is being used for.
+ *
+ * The @p wid parameter is used to tell the KWallet manager which window
+ * is requesting access to the wallet.
+ *
+ * @param parent the owner of this wallet
+ * @param wid the window ID of the window the web page will be
+ * embedded in
+ */
+ explicit WebEngineWallet(QObject *parent = nullptr, WId wid = 0);
+
+ /**
+ * Destructor
+ */
+ virtual ~WebEngineWallet();
+
+ /**
+ * Attempts to save the form data from @p page and its children frames.
+ *
+ * You must connect to the @ref saveFormDataRequested signal and call either
+ * @ref rejectSaveFormDataRequest or @ref acceptSaveFormDataRequest signals
+ * in order to complete the save request. Otherwise, you request will simply
+ * be ignored.
+ *
+ * Note that this function is asynchronous, as it requires running javascript code
+ * on the page using @ref QWebEnginePage::runJavaScript. This function only requests
+ * for the form data to be saved when @ref QWebEnginePage::runJavaScript finishes.
+ * The actual saving is done by @ref saveFormDataCallback
+ */
+ void saveFormData(WebEnginePage *page, bool ignorePasswordFields = false);
+
+ /**
+ * Attempts to fill forms contained in @p page with cached data.
+ *
+ * Note that this function is asynchronous, as it requires running javascript code
+ * on the page using @ref QWebEnginePage::runJavaScript. This function only requests
+ * for the form data to be filled when @ref QWebEnginePage::runJavaScript finishes.
+ * The actual filling is done by @ref fillFormDataCallback
+ */
+ void fillFormData(WebEnginePage *page);
+
+ /**
+ * Removes the form data specified by @p forms from the persistent storage.
+ *
+ * Note that this function will remove all cached data for forms found in @p page.
+ *
+ * Note that this function is asynchronous, as it requires running javascript code
+ * on the page using @ref QWebEnginePage::runJavaScript. This function only requests
+ * for the form data to be removed when @ref QWebEnginePage::runJavaScript finishes.
+ * The actual removing is done by @ref removeFormDataCallback
+ */
+ void removeFormData(WebEnginePage *page);
+
+ /**
+ * Removes the form data specified by @p forms from the persistent storage.
+ *
+ * @see formsWithCachedData
+ */
+ void removeFormData(const WebFormList &forms);
+
+public Q_SLOTS:
+ /**
+ * Accepts the save form data request associated with @p key.
+ *
+ * The @p key parameter is the one sent through the @ref saveFormDataRequested
+ * signal.
+ *
+ * You must always call this function or @ref rejectSaveFormDataRequest in
+ * order to complete the save form data request. Otherwise, the request will
+ * simply be ignored.
+ *
+ * @see saveFormDataRequested.
+ */
+ void acceptSaveFormDataRequest(const QString &key);
+
+ /**
+ * Rejects the save form data request associated with @p key.
+ *
+ * The @p key parameter is the one sent through the @ref saveFormDataRequested
+ * signal.
+ *
+ * @see saveFormDataRequested.
+ */
+ void rejectSaveFormDataRequest(const QString &key);
+
+Q_SIGNALS:
+ /**
+ * This signal is emitted whenever a save form data request is received.
+ *
+ * Unless you connect to this signal and and call @ref acceptSaveFormDataRequest
+ * or @ref rejectSaveFormDataRequest slots, the save form data requested through
+ * @ref saveFormData will simply be ignored.
+ *
+ * @p key is a value that uniquely identifies the save request and @p url
+ * is the address for which the form data is being saved.
+ *
+ * @see acceptSaveFormDataRequest
+ * @see rejectSaveFormDataRequest
+ */
+ void saveFormDataRequested(const QString &key, const QUrl &url);
+
+ /**
+ * This signal is emitted whenever a save form data request is completed.
+ *
+ * @p ok will be set to true if the save form data request for @p url was
+ * completed successfully.
+ *
+ * @see saveFormDataRequested
+ */
+ void saveFormDataCompleted(const QUrl &url, bool ok);
+
+ /**
+ * This signal is emitted whenever a fill form data request is completed.
+ *
+ * @p ok will be set to true if any forms were successfully filled with
+ * cached data from the persistent storage.
+ *
+ * @see fillFormData
+ * @since 4.5
+ */
+ void fillFormRequestCompleted(bool ok);
+
+ /**
+ * This signal is emitted whenever the current wallet is closed.
+ */
+ void walletClosed();
+
+protected:
+ /**
+ * Returns a list of forms for @p url that are waiting to be filled.
+ *
+ * This function returns an empty list if there is no pending requests
+ * for filling forms associated with @p url.
+ */
+ WebFormList formsToFill(const QUrl &url) const;
+
+ /**
+ * Returns a list of for @p key that are waiting to be saved.
+ *
+ * This function returns an empty list if there are no pending requests
+ * for saving forms associated with @p key.
+ */
+ WebFormList formsToSave(const QString &key) const;
+
+ /**
+ * Returns forms to be removed from persistent storage.
+ */
+ WebFormList formsToDelete() const;
+
+ /**
+ * Returns true when there is data associated with @p form in the
+ * persistent storage.
+ */
+ bool hasCachedFormData(const WebForm &form) const;
+
+ /**
+ * Fills the web forms in frame that point to @p url with data from @p forms.
+ *
+ * @see fillFormDataFromCache.
+ */
+ void fillWebForm(const QUrl &url, const WebFormList &forms);
+
+ /**
+ * Fills form data from persistent storage.
+ *
+ * If you reimplement this function, call @ref formsToFill to obtain
+ * the list of forms pending to be filled. Once you fill the list with
+ * the cached data from the persistent storage, you must call @p fillWebForm
+ * to fill out the actual web forms.
+ *
+ * @see formsToFill
+ */
+ void fillFormDataFromCache(const QList<QUrl> &list);
+
+ /**
+ * Stores form data associated with @p key to a persistent storage.
+ *
+ * If you reimplement this function, call @ref formsToSave to obtain the
+ * list of form data pending to be saved to persistent storage.
+ *
+ *@see formsToSave
+ */
+ void saveFormDataToCache(const QString &key);
+
+ /**
+ * Removes all cached form data associated with @p forms from persistent storage.
+ *
+ * If you reimplement this function, call @ref formsToDelete to obtain the
+ * list of form data pending to be removed from persistent storage.
+ *
+ *@see formsToDelete
+ */
+ void removeFormDataFromCache(const WebFormList &forms);
+
+private:
+
+ /**
+ * Callback used by @ref fillFormData to insert form data
+ *
+ * This function is called as a callback from @ref fillFormData after
+ * @ref QWebEnginePage::runJavaScript has finished
+ */
+ void fillFormDataCallback(WebEnginePage *page, const WebFormList &formsList);
+
+ /**
+ * Callback used by @ref saveFormData to save data
+ *
+ * This function is called as a callback from @ref saveFormData after
+ * @ref QWebEnginePage::runJavaScript has finished
+ */
+ void saveFormDataCallback(const QString &key, const QUrl &url, const WebFormList &formslist);
+
+ /**
+ * Callback used by @ref removeFormData to remove data
+ *
+ * This function is called as a callback from @ref removeFormData after
+ * @ref QWebEnginePage::runJavaScript has finished
+ */
+ void removeFormDataCallback(const WebFormList &list);
+
+private:
+ class WebEngineWalletPrivate;
+ friend class WebEngineWalletPrivate;
+ WebEngineWalletPrivate *const d;
+
+ Q_PRIVATE_SLOT(d, void _k_openWalletDone(bool))
+ Q_PRIVATE_SLOT(d, void _k_walletClosed())
+};
+
+
+#endif // WEBENGINEWALLET_H
--
cgit v0.11.2