Falkon Develop
Cross-platform Qt-based web browser
gm_script.cpp
Go to the documentation of this file.
1/* ============================================================
2* GreaseMonkey plugin for Falkon
3* Copyright (C) 2012-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 "gm_script.h"
19#include "gm_manager.h"
20#include "gm_downloader.h"
21
22#include "delayedfilewatcher.h"
23#include "mainapplication.h"
24#include "webpage.h"
25#include "networkmanager.h"
26
27#include <QFile>
28#include <QTextStream>
29#include <QStringList>
30#include <QWebEngineScript>
31#include <QCryptographicHash>
32#include <QNetworkReply>
33#include <QNetworkRequest>
34
35GM_Script::GM_Script(GM_Manager* manager, const QString &filePath)
36 : QObject(manager)
37 , m_manager(manager)
38 , m_fileWatcher(new DelayedFileWatcher(this))
39 , m_namespace(QSL("GreaseMonkeyNS"))
40 , m_startAt(DocumentEnd)
41 , m_noframes(false)
42 , m_fileName(filePath)
43 , m_enabled(true)
44 , m_valid(false)
45 , m_updating(false)
46{
47 parseScript();
48
49 connect(m_fileWatcher, &DelayedFileWatcher::delayedFileChanged, this, &GM_Script::watchedFileChanged);
50}
51
53{
54 return m_valid;
55}
56
57QString GM_Script::name() const
58{
59 return m_name;
60}
61
62QString GM_Script::nameSpace() const
63{
64 return m_namespace;
65}
66
67QString GM_Script::fullName() const
68{
69 return QSL("%1/%2").arg(m_namespace, m_name);
70}
71
73{
74 return m_description;
75}
76
77QString GM_Script::version() const
78{
79 return m_version;
80}
81
82QIcon GM_Script::icon() const
83{
84 return m_icon;
85}
86
88{
89 return m_iconUrl;
90}
91
93{
94 return m_downloadUrl;
95}
96
98{
99 return m_updateUrl;
100}
101
103{
104 return m_startAt;
105}
106
108{
109 return m_noframes;
110}
111
113{
114 return m_valid && m_enabled;
115}
116
117void GM_Script::setEnabled(bool enable)
118{
119 m_enabled = enable;
120}
121
122QStringList GM_Script::include() const
123{
124 return m_include;
125}
126
127QStringList GM_Script::exclude() const
128{
129 return m_exclude;
130}
131
132QStringList GM_Script::require() const
133{
134 return m_require;
135}
136
137QString GM_Script::fileName() const
138{
139 return m_fileName;
140}
141
142QWebEngineScript GM_Script::webScript() const
143{
144 QWebEngineScript script;
145 script.setSourceCode(QSL("%1\n%2").arg(m_manager->bootstrapScript(), m_script));
146 script.setName(fullName());
147 script.setWorldId(WebPage::SafeJsWorld);
148 script.setRunsOnSubFrames(!m_noframes);
149 return script;
150}
151
153{
154 return m_updating;
155}
156
158{
159 if (!m_downloadUrl.isValid() || m_updating)
160 return;
161
162 m_updating = true;
163 Q_EMIT updatingChanged(m_updating);
164
165 auto *downloader = new GM_Downloader(m_downloadUrl, m_manager);
166 downloader->updateScript(m_fileName);
167 connect(downloader, &GM_Downloader::finished, this, [this]() {
168 m_updating = false;
169 Q_EMIT updatingChanged(m_updating);
170 });
171 connect(downloader, &GM_Downloader::error, this, [this]() {
172 m_updating = false;
173 Q_EMIT updatingChanged(m_updating);
174 });
175 downloadRequires();
176}
177
178void GM_Script::watchedFileChanged(const QString &file)
179{
180 if (m_fileName == file) {
181 reloadScript();
182 }
183}
184
185void GM_Script::parseScript()
186{
187 m_name.clear();
188 m_namespace = QSL("GreaseMonkeyNS");
189 m_description.clear();
190 m_version.clear();
191 m_include.clear();
192 m_exclude.clear();
193 m_require.clear();
194 m_icon = QIcon();
195 m_iconUrl.clear();
196 m_downloadUrl.clear();
197 m_updateUrl.clear();
198 m_startAt = DocumentEnd;
199 m_noframes = false;
200 m_script.clear();
201 m_enabled = true;
202 m_valid = false;
203
204 QFile file(m_fileName);
205 if (!file.open(QFile::ReadOnly)) {
206 qWarning() << "GreaseMonkey: Cannot open file for reading" << m_fileName;
207 return;
208 }
209
210 if (!m_fileWatcher->files().contains(m_fileName)) {
211 m_fileWatcher->addPath(m_fileName);
212 }
213
214 const QByteArray fileData = file.readAll();
215
216 bool inMetadata = false;
217
218 QTextStream stream(fileData);
219 QString line;
220 while (stream.readLineInto(&line)) {
221 if (line.startsWith(QL1S("// ==UserScript=="))) {
222 inMetadata = true;
223 }
224 if (line.startsWith(QL1S("// ==/UserScript=="))) {
225 break;
226 }
227 if (!inMetadata) {
228 continue;
229 }
230
231 if (!line.startsWith(QLatin1String("// @"))) {
232 continue;
233 }
234
235 line = line.mid(3).replace(QLatin1Char('\t'), QLatin1Char(' '));
236 int index = line.indexOf(QLatin1Char(' '));
237
238 const QString key = line.left(index).trimmed();
239 const QString value = index > 0 ? line.mid(index + 1).trimmed() : QString();
240
241 if (key.isEmpty()) {
242 continue;
243 }
244
245 if (key == QLatin1String("@name")) {
246 m_name = value;
247 }
248 else if (key == QLatin1String("@namespace")) {
249 m_namespace = value;
250 }
251 else if (key == QLatin1String("@description")) {
252 m_description = value;
253 }
254 else if (key == QLatin1String("@version")) {
255 m_version = value;
256 }
257 else if (key == QLatin1String("@updateURL")) {
258 m_updateUrl = QUrl(value);
259 }
260 else if (key == QLatin1String("@downloadURL")) {
261 m_downloadUrl = QUrl(value);
262 }
263 else if (key == QLatin1String("@include") || key == QLatin1String("@match")) {
264 m_include.append(value);
265 }
266 else if (key == QLatin1String("@exclude") || key == QLatin1String("@exclude_match")) {
267 m_exclude.append(value);
268 }
269 else if (key == QLatin1String("@require")) {
270 m_require.append(value);
271 }
272 else if (key == QLatin1String("@run-at")) {
273 if (value == QLatin1String("document-end")) {
274 m_startAt = DocumentEnd;
275 }
276 else if (value == QLatin1String("document-start")) {
277 m_startAt = DocumentStart;
278 }
279 else if (value == QLatin1String("document-idle")) {
280 m_startAt = DocumentIdle;
281 }
282 }
283 else if (key == QL1S("@icon")) {
284 m_iconUrl = QUrl(value);
285 }
286 else if (key == QL1S("@noframes")) {
287 m_noframes = true;
288 }
289 }
290
291 if (!inMetadata) {
292 qWarning() << "GreaseMonkey: File does not contain metadata block" << m_fileName;
293 return;
294 }
295
296 m_iconUrl = m_downloadUrl.resolved(m_iconUrl);
297
298 if (m_include.isEmpty()) {
299 m_include.append(QSL("*"));
300 }
301
302 const QString nspace = QString::fromLatin1(QCryptographicHash::hash(fullName().toUtf8(), QCryptographicHash::Md4).toHex());
303 const QString gmValues = m_manager->valuesScript().arg(nspace);
304 m_script = QSL("(function(){%1\n%2\n%3\n})();").arg(gmValues, m_manager->requireScripts(m_require), QString::fromUtf8(fileData));
305 m_valid = true;
306
307 downloadIcon();
308 downloadRequires();
309}
310
311void GM_Script::reloadScript()
312{
313 parseScript();
314
315 m_manager->removeScript(this, false);
316 m_manager->addScript(this);
317
318 Q_EMIT scriptChanged();
319}
320
321void GM_Script::downloadIcon()
322{
323 if (m_iconUrl.isValid()) {
324 QNetworkReply *reply = mApp->networkManager()->get(QNetworkRequest(m_iconUrl));
325 connect(reply, &QNetworkReply::finished, this, [=]() {
326 reply->deleteLater();
327 if (reply->error() == QNetworkReply::NoError) {
328 m_icon = QPixmap::fromImage(QImage::fromData(reply->readAll()));
329 }
330 });
331 }
332}
333
334void GM_Script::downloadRequires()
335{
336 for (const QString &url : std::as_const(m_require)) {
337 if (m_manager->requireScripts({url}).isEmpty()) {
338 auto *downloader = new GM_Downloader(QUrl(url), m_manager, GM_Downloader::DownloadRequireScript);
339 connect(downloader, &GM_Downloader::finished, this, &GM_Script::reloadScript);
340 }
341 }
342}
void delayedFileChanged(const QString &path)
void finished(const QString &fileName)
QString requireScripts(const QStringList &urlList) const
Definition: gm_manager.cpp:98
bool removeScript(GM_Script *script, bool removeFile=true)
Definition: gm_manager.cpp:206
bool addScript(GM_Script *script)
Definition: gm_manager.cpp:190
QString bootstrapScript() const
Definition: gm_manager.cpp:126
QString valuesScript() const
Definition: gm_manager.cpp:131
QString nameSpace() const
Definition: gm_script.cpp:62
QStringList exclude() const
Definition: gm_script.cpp:127
QUrl iconUrl() const
Definition: gm_script.cpp:87
@ DocumentIdle
Definition: gm_script.h:37
@ DocumentStart
Definition: gm_script.h:37
@ DocumentEnd
Definition: gm_script.h:37
void updateScript()
Definition: gm_script.cpp:157
QStringList include() const
Definition: gm_script.cpp:122
QIcon icon() const
Definition: gm_script.cpp:82
QString name() const
Definition: gm_script.cpp:57
QStringList require() const
Definition: gm_script.cpp:132
GM_Script(GM_Manager *manager, const QString &filePath)
Definition: gm_script.cpp:35
QWebEngineScript webScript() const
Definition: gm_script.cpp:142
QUrl downloadUrl() const
Definition: gm_script.cpp:92
bool isValid() const
Definition: gm_script.cpp:52
void scriptChanged()
void setEnabled(bool enable)
Definition: gm_script.cpp:117
bool isEnabled() const
Definition: gm_script.cpp:112
StartAt startAt() const
Definition: gm_script.cpp:102
QString fullName() const
Definition: gm_script.cpp:67
QString fileName() const
Definition: gm_script.cpp:137
void updatingChanged(bool updating)
bool isUpdating()
Definition: gm_script.cpp:152
QString version() const
Definition: gm_script.cpp:77
QUrl updateUrl() const
Definition: gm_script.cpp:97
bool noFrames() const
Definition: gm_script.cpp:107
QString description() const
Definition: gm_script.cpp:72
@ SafeJsWorld
Definition: webpage.h:45
#define mApp
int value(const QColor &c)
Definition: colors.cpp:238
#define QL1S(x)
Definition: qzcommon.h:44
#define QSL(x)
Definition: qzcommon.h:40