Falkon Develop
Cross-platform Qt-based web browser
networkmanager.cpp
Go to the documentation of this file.
1/* ============================================================
2* Falkon - Qt web browser
3* Copyright (C) 2010-2018 David Rosca <nowrep@gmail.com>
4*
5* This program is free software: you can redistribute it and/or modify
6* it under the terms of the GNU General Public License as published by
7* the Free Software Foundation, either version 3 of the License, or
8* (at your option) any later version.
9*
10* This program is distributed in the hope that it will be useful,
11* but WITHOUT ANY WARRANTY; without even the implied warranty of
12* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13* GNU General Public License for more details.
14*
15* You should have received a copy of the GNU General Public License
16* along with this program. If not, see <http://www.gnu.org/licenses/>.
17* ============================================================ */
18#include "networkmanager.h"
19#include "autofill.h"
20#include "qztools.h"
21#include "settings.h"
22#include "cookiejar.h"
23#include "acceptlanguage.h"
24#include "mainapplication.h"
25#include "passwordmanager.h"
26#include "sslerrordialog.h"
30#include "webpage.h"
31
32#include <QLabel>
33#include <QDialog>
34#include <QLineEdit>
35#include <QCheckBox>
36#include <QFormLayout>
37#include <QAuthenticator>
38#include <QDialogButtonBox>
39#include <QNetworkReply>
40#include <QNetworkProxy>
41#include <QWebEngineProfile>
42#include <QtWebEngineWidgetsVersion>
43
44#include <QWebEngineUrlScheme>
45
47 : QNetworkAccessManager(parent)
48{
49 // Create scheme handlers
50 m_extensionScheme = new ExtensionSchemeManager();
51
52 mApp->webProfile()->installUrlSchemeHandler(QByteArrayLiteral("falkon"), new FalkonSchemeHandler());
53 mApp->webProfile()->installUrlSchemeHandler(QByteArrayLiteral("extension"), m_extensionScheme);
55 WebPage::addSupportedScheme(QSL("extension"));
56
57 // Create url interceptor
58 m_urlInterceptor = new NetworkUrlInterceptor(this);
59 mApp->webProfile()->setUrlRequestInterceptor(m_urlInterceptor);
60
61 // Create cookie jar
62 mApp->cookieJar();
63
64 connect(this, &QNetworkAccessManager::authenticationRequired, this, [this](QNetworkReply *reply, QAuthenticator *auth) {
65 authentication(reply->url(), auth);
66 });
67
68 connect(this, &QNetworkAccessManager::proxyAuthenticationRequired, this, [this](const QNetworkProxy &proxy, QAuthenticator *auth) {
69 proxyAuthentication(proxy.hostName(), auth);
70 });
71}
72
73bool NetworkManager::certificateError(QWebEngineCertificateError &error, QWidget *parent)
74{
75 const QString &host = error.url().host();
76 const auto errorType = error.type();
77
78 if (m_rejectedSslErrors.contains(host) && m_rejectedSslErrors.value(host) == errorType) {
79 return false;
80 }
81
82 if ((m_ignoredSslErrors.contains(host) && m_ignoredSslErrors.value(host) == errorType)
83 || m_ignoredSslHosts.contains(host)) {
84 return true;
85 }
86
87 // Defer loading the URL until the user prompt has completed.
88 if (error.isOverridable())
89 error.defer();
90
91 QString title = tr("SSL Certificate Error!");
92 QString text1 = tr("The page you are trying to access has the following errors in the SSL certificate:");
93 QString text2 = tr("Would you like to make an exception for this certificate?");
94
95 const auto errorDescription = error.description();
96 QString message = QSL("<b>%1</b><p>%2</p><ul><li>%3</li></ul><p>%4</p>").arg(title, text1, errorDescription, text2);
97
98 SslErrorDialog dialog(parent);
99 dialog.setText(message);
100 dialog.exec();
101
102 switch (dialog.result()) {
104 m_ignoredSslHosts.append(host);
105 saveIgnoredSslHosts();
106 return true;
107
109 m_ignoredSslErrors[host] = errorType;
110 return true;
111
113 m_rejectedSslErrors[host] = errorType;
114 return false;
115
116 default:
117 return false;
118 }
119}
120
121void NetworkManager::authentication(const QUrl &url, QAuthenticator *auth, QWidget *parent)
122{
123 auto* dialog = new QDialog(parent);
124 dialog->setWindowTitle(tr("Authorization required"));
125
126 auto* formLa = new QFormLayout(dialog);
127
128 auto* label = new QLabel(dialog);
129 auto* userLab = new QLabel(dialog);
130 auto* passLab = new QLabel(dialog);
131 userLab->setText(tr("Username: "));
132 passLab->setText(tr("Password: "));
133
134 auto* user = new QLineEdit(dialog);
135 auto* pass = new QLineEdit(dialog);
136 pass->setEchoMode(QLineEdit::Password);
137 auto* save = new QCheckBox(dialog);
138 save->setText(tr("Save username and password for this site"));
139
140 auto* box = new QDialogButtonBox(dialog);
141 box->addButton(QDialogButtonBox::Ok);
142 box->addButton(QDialogButtonBox::Cancel);
143 connect(box, &QDialogButtonBox::rejected, dialog, &QDialog::reject);
144 connect(box, &QDialogButtonBox::accepted, dialog, &QDialog::accept);
145
146 label->setText(tr("A username and password are being requested by %1. "
147 "The site says: \"%2\"").arg(url.host(), auth->realm().toHtmlEscaped()));
148
149 formLa->addRow(label);
150 formLa->addRow(userLab, user);
151 formLa->addRow(passLab, pass);
152 formLa->addRow(save);
153 formLa->addWidget(box);
154
155 AutoFill* fill = mApp->autoFill();
156 QString storedUser;
157 QString storedPassword;
158 bool shouldUpdateEntry = false;
159
160 if (fill->isStored(url)) {
161 const QVector<PasswordEntry> &data = fill->getFormData(url);
162 if (!data.isEmpty()) {
163 save->setChecked(true);
164 shouldUpdateEntry = true;
165 storedUser = data.at(0).username;
166 storedPassword = data.at(0).password;
167 user->setText(storedUser);
168 pass->setText(storedPassword);
169 }
170 }
171
172 // Do not save when private browsing is enabled
173 if (mApp->isPrivate()) {
174 save->setVisible(false);
175 }
176
177 if (dialog->exec() != QDialog::Accepted) {
178 *auth = QAuthenticator();
179 delete dialog;
180 return;
181 }
182
183 auth->setUser(user->text());
184 auth->setPassword(pass->text());
185
186 if (save->isChecked()) {
187 if (shouldUpdateEntry) {
188 if (storedUser != user->text() || storedPassword != pass->text()) {
189 fill->updateEntry(url, user->text(), pass->text());
190 }
191 }
192 else {
193 fill->addEntry(url, user->text(), pass->text());
194 }
195 }
196
197 delete dialog;
198}
199
200void NetworkManager::proxyAuthentication(const QString &proxyHost, QAuthenticator *auth, QWidget *parent)
201{
202 const QNetworkProxy proxy = QNetworkProxy::applicationProxy();
203 if (!proxy.user().isEmpty() && !proxy.password().isEmpty()) {
204 auth->setUser(proxy.user());
205 auth->setPassword(proxy.password());
206 return;
207 }
208
209 auto* dialog = new QDialog(parent);
210 dialog->setWindowTitle(tr("Proxy authorization required"));
211
212 auto* formLa = new QFormLayout(dialog);
213
214 auto* label = new QLabel(dialog);
215 auto* userLab = new QLabel(dialog);
216 auto* passLab = new QLabel(dialog);
217 userLab->setText(tr("Username: "));
218 passLab->setText(tr("Password: "));
219
220 auto* user = new QLineEdit(dialog);
221 auto* pass = new QLineEdit(dialog);
222 pass->setEchoMode(QLineEdit::Password);
223
224 auto* box = new QDialogButtonBox(dialog);
225 box->addButton(QDialogButtonBox::Ok);
226 box->addButton(QDialogButtonBox::Cancel);
227 connect(box, &QDialogButtonBox::rejected, dialog, &QDialog::reject);
228 connect(box, &QDialogButtonBox::accepted, dialog, &QDialog::accept);
229
230 label->setText(tr("A username and password are being requested by proxy %1. ").arg(proxyHost));
231 formLa->addRow(label);
232 formLa->addRow(userLab, user);
233 formLa->addRow(passLab, pass);
234 formLa->addWidget(box);
235
236 if (dialog->exec() != QDialog::Accepted) {
237 *auth = QAuthenticator();
238 delete dialog;
239 return;
240 }
241
242 auth->setUser(user->text());
243 auth->setPassword(pass->text());
244
245 delete dialog;
246}
247
249{
250 m_urlInterceptor->installUrlInterceptor(interceptor);
251}
252
254{
255 m_urlInterceptor->removeUrlInterceptor(interceptor);
256}
257
259{
260 m_extensionScheme->registerHandler(name, handler);
261}
262
264{
265 m_extensionScheme->unregisterHandler(handler);
266}
267
269{
270 Settings settings;
271 settings.beginGroup(QSL("Language"));
272 QStringList langs = settings.value(QSL("acceptLanguage"), AcceptLanguage::defaultLanguage()).toStringList();
273 settings.endGroup();
274 mApp->webProfile()->setHttpAcceptLanguage(QString::fromLatin1(AcceptLanguage::generateHeader(langs)));
275
276 QNetworkProxy proxy;
277 settings.beginGroup(QSL("Web-Proxy"));
278 const int proxyType = settings.value(QSL("ProxyType"), 2).toInt();
279 proxy.setHostName(settings.value(QSL("HostName"), QString()).toString());
280 proxy.setPort(settings.value(QSL("Port"), 8080).toInt());
281 proxy.setUser(settings.value(QSL("Username"), QString()).toString());
282 proxy.setPassword(settings.value(QSL("Password"), QString()).toString());
283 settings.endGroup();
284
285 if (proxyType == 0) {
286 proxy.setType(QNetworkProxy::NoProxy);
287 } else if (proxyType == 3) {
288 proxy.setType(QNetworkProxy::HttpProxy);
289 } else if (proxyType == 4) {
290 proxy.setType(QNetworkProxy::Socks5Proxy);
291 }
292
293 if (proxyType == 2) {
294 QNetworkProxy::setApplicationProxy(QNetworkProxy());
295 QNetworkProxyFactory::setUseSystemConfiguration(true);
296 } else {
297 QNetworkProxy::setApplicationProxy(proxy);
298 QNetworkProxyFactory::setUseSystemConfiguration(false);
299 }
300
301 m_urlInterceptor->loadSettings();
302
303 settings.beginGroup(QSL("Web-Browser-Settings"));
304 m_ignoredSslHosts = settings.value(QSL("IgnoredSslHosts"), QStringList()).toStringList();
305 settings.endGroup();
306}
307
308void NetworkManager::saveIgnoredSslHosts()
309{
310 Settings settings;
311 settings.beginGroup(QSL("Web-Browser-Settings"));
312 settings.setValue(QSL("IgnoredSslHosts"), m_ignoredSslHosts);
313 settings.endGroup();
314}
315
317{
318 mApp->webProfile()->setUrlRequestInterceptor(nullptr);
319 saveIgnoredSslHosts();
320}
321
322// static
324{
325 QWebEngineUrlScheme falkonScheme("falkon");
326 falkonScheme.setFlags(QWebEngineUrlScheme::SecureScheme | QWebEngineUrlScheme::ContentSecurityPolicyIgnored);
327 falkonScheme.setSyntax(QWebEngineUrlScheme::Syntax::Path);
328 QWebEngineUrlScheme::registerScheme(falkonScheme);
329 QWebEngineUrlScheme extensionScheme("extension");
330 extensionScheme.setFlags(QWebEngineUrlScheme::SecureScheme | QWebEngineUrlScheme::ContentSecurityPolicyIgnored);
331 extensionScheme.setSyntax(QWebEngineUrlScheme::Syntax::Path);
332 QWebEngineUrlScheme::registerScheme(extensionScheme);
333}
334
335QNetworkReply *NetworkManager::createRequest(QNetworkAccessManager::Operation op, const QNetworkRequest &request, QIODevice *outgoingData)
336{
337 QNetworkRequest req = request;
338 /* TODO Use HTTP2 */
339
340 return QNetworkAccessManager::createRequest(op, req, outgoingData);
341}
static QByteArray generateHeader(const QStringList &langs)
static QStringList defaultLanguage()
void addEntry(const QUrl &url, const QString &name, const QString &pass)
Definition: autofill.cpp:128
void updateEntry(const QUrl &url, const QString &name, const QString &pass)
Definition: autofill.cpp:151
QVector< PasswordEntry > getFormData(const QUrl &url)
Definition: autofill.cpp:112
bool isStored(const QUrl &url)
Definition: autofill.cpp:67
void unregisterHandler(ExtensionSchemeHandler *handler)
void registerHandler(const QString &name, ExtensionSchemeHandler *handler)
void authentication(const QUrl &url, QAuthenticator *auth, QWidget *parent=nullptr)
NetworkManager(QObject *parent=nullptr)
void registerExtensionSchemeHandler(const QString &name, ExtensionSchemeHandler *handler)
void unregisterExtensionSchemeHandler(ExtensionSchemeHandler *handler)
void proxyAuthentication(const QString &proxyHost, QAuthenticator *auth, QWidget *parent=nullptr)
static void registerSchemes()
QNetworkReply * createRequest(Operation op, const QNetworkRequest &request, QIODevice *outgoingData) override
bool certificateError(QWebEngineCertificateError &error, QWidget *parent=nullptr)
void removeUrlInterceptor(UrlInterceptor *interceptor)
void installUrlInterceptor(UrlInterceptor *interceptor)
void installUrlInterceptor(UrlInterceptor *interceptor)
void removeUrlInterceptor(UrlInterceptor *interceptor)
void beginGroup(const QString &prefix)
Definition: settings.cpp:79
void endGroup()
Definition: settings.cpp:84
QVariant value(const QString &key, const QVariant &defaultValue=QVariant())
Definition: settings.cpp:74
void setValue(const QString &key, const QVariant &defaultValue=QVariant())
Definition: settings.cpp:69
void setText(const QString &text)
static void addSupportedScheme(const QString &scheme)
Definition: webpage.cpp:240
#define mApp
#define QSL(x)
Definition: qzcommon.h:40