# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2017-2021 Florian Bruhin (The Compiler) # This file is part of qutebrowser. # # qutebrowser 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 3 of the License, or # (at your option) any later version. # # qutebrowser 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. # # You should have received a copy of the GNU General Public License # along with qutebrowser. If not, see . """Tests for qutebrowser.browser.greasemonkey.""" import logging import textwrap import pytest import py.path # pylint: disable=no-name-in-module from PyQt5.QtCore import QUrl from qutebrowser.utils import usertypes, version from qutebrowser.browser import greasemonkey from qutebrowser.misc import objects test_gm_script = r""" // ==UserScript== // @name qutebrowser test userscript // @namespace invalid.org // @include http://localhost:*/data/title.html // @match http://*.trolol.com/* // @exclude https://badhost.xxx/* // @run-at document-start // ==/UserScript== console.log("Script is running."); """ pytestmark = [ pytest.mark.usefixtures('data_tmpdir'), pytest.mark.usefixtures('config_tmpdir') ] def _save_script(script_text, filename): # pylint: disable=no-member file_path = py.path.local(greasemonkey._scripts_dirs()[0]) / filename # pylint: enable=no-member file_path.write_text(script_text, encoding='utf-8', ensure=True) def test_all(): """Test that a script gets read from file, parsed and returned.""" _save_script(test_gm_script, 'test.user.js') gm_manager = greasemonkey.GreasemonkeyManager() assert (gm_manager.all_scripts()[0].name == "qutebrowser test userscript") @pytest.mark.parametrize("url, expected_matches", [ # included ('http://trolol.com/', 1), # neither included nor excluded ('http://aaaaaaaaaa.com/', 0), # excluded ('https://badhost.xxx/', 0), ]) def test_get_scripts_by_url(url, expected_matches): """Check Greasemonkey include/exclude rules work.""" _save_script(test_gm_script, 'test.user.js') gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.scripts_for(QUrl(url)) assert len(scripts.start + scripts.end + scripts.idle) == expected_matches @pytest.mark.parametrize("url, expected_matches", [ # included ('https://github.com/qutebrowser/qutebrowser/', 1), # neither included nor excluded ('http://aaaaaaaaaa.com/', 0), # excluded takes priority ('http://github.com/foo', 0), ]) def test_regex_includes_scripts_for(url, expected_matches): """Ensure our GM @*clude support supports regular expressions.""" gh_dark_example = textwrap.dedent(r""" // ==UserScript== // @include /^https?://((gist|guides|help|raw|status|developer)\.)?github\.com/((?!generated_pages\/preview).)*$/ // @exclude /https?://github\.com/foo/ // @run-at document-start // ==/UserScript== """) _save_script(gh_dark_example, 'test.user.js') gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.scripts_for(QUrl(url)) assert len(scripts.start + scripts.end + scripts.idle) == expected_matches def test_no_metadata(caplog): """Run on all sites at document-end is the default.""" _save_script("var nothing = true;\n", 'nothing.user.js') with caplog.at_level(logging.WARNING): gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.scripts_for(QUrl('http://notamatch.invalid/')) assert len(scripts.start + scripts.end + scripts.idle) == 1 assert len(scripts.end) == 1 def test_no_name(): """Ensure that GreaseMonkeyScripts must have a name.""" msg = "@name key required or pass filename to init." with pytest.raises(ValueError, match=msg): greasemonkey.GreasemonkeyScript([("something", "else")], "") def test_no_name_with_fallback(): """Ensure that script's name can fallback to the provided filename.""" script = greasemonkey.GreasemonkeyScript( [("something", "else")], "", filename=r"C:\COM1") assert script assert script.name == r"C:\COM1" @pytest.mark.parametrize('properties, inc_counter, expected', [ ([("name", "gorilla")], False, "GM-gorilla"), ([("namespace", "apes"), ("name", "gorilla")], False, "GM-apes/gorilla"), ([("name", "gorilla")], True, "GM-gorilla-2"), ([("namespace", "apes"), ("name", "gorilla")], True, "GM-apes/gorilla-2"), ]) def test_full_name(properties, inc_counter, expected): script = greasemonkey.GreasemonkeyScript(properties, code="") if inc_counter: script.dedup_suffix += 1 assert script.full_name() == expected def test_bad_scheme(caplog): """qute:// isn't in the list of allowed schemes.""" _save_script("var nothing = true;\n", 'nothing.user.js') with caplog.at_level(logging.WARNING): gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.scripts_for(QUrl('qute://settings')) assert len(scripts.start + scripts.end + scripts.idle) == 0 def test_load_emits_signal(qtbot): gm_manager = greasemonkey.GreasemonkeyManager() with qtbot.wait_signal(gm_manager.scripts_reloaded): gm_manager.load_scripts() def test_utf8_bom(): """Make sure UTF-8 BOMs are stripped from scripts. If we don't strip them, we'll have a BOM in the middle of the file, causing QtWebEngine to not catch the "// ==UserScript==" line. """ script = textwrap.dedent(""" \N{BYTE ORDER MARK}// ==UserScript== // @name qutebrowser test userscript // ==/UserScript== """.lstrip('\n')) _save_script(script, 'bom.user.js') gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.all_scripts() assert len(scripts) == 1 script = scripts[0] assert '// ==UserScript==' in script.code().splitlines() class TestForceDocumentEnd: def _get_script(self, *, namespace, name): source = textwrap.dedent(""" // ==UserScript== // @namespace {} // @name {} // ==/UserScript== """.format(namespace, name)) _save_script(source, 'force.user.js') gm_manager = greasemonkey.GreasemonkeyManager() scripts = gm_manager.all_scripts() assert len(scripts) == 1 return scripts[0] @pytest.mark.parametrize('namespace, name, force', [ ('http://userstyles.org', 'foobar', True), ('https://github.com/ParticleCore', 'Iridium', True), ('https://github.com/ParticleCore', 'Foo', False), ('https://example.org', 'Iridium', False), ]) def test_matching(self, monkeypatch, namespace, name, force): """Test matching based on namespace/name.""" monkeypatch.setattr(objects, 'backend', usertypes.Backend.QtWebEngine) script = self._get_script(namespace=namespace, name=name) assert script.needs_document_end_workaround() == force def test_required_scripts_are_included(download_stub, tmp_path): test_require_script = textwrap.dedent(""" // ==UserScript== // @name qutebrowser test userscript // @namespace invalid.org // @include http://localhost:*/data/title.html // @match http://trolol* // @exclude https://badhost.xxx/* // @run-at document-start // @require http://localhost/test.js // ==/UserScript== console.log("Script is running."); """) _save_script(test_require_script, 'requiring.user.js') (tmp_path / 'test.js').write_text('REQUIRED SCRIPT', encoding='UTF-8') gm_manager = greasemonkey.GreasemonkeyManager() assert len(gm_manager._in_progress_dls) == 1 for download in gm_manager._in_progress_dls: download.finished.emit() scripts = gm_manager.all_scripts() assert len(scripts) == 1 assert "REQUIRED SCRIPT" in scripts[0].code() # Additionally check that the base script is still being parsed correctly assert "Script is running." in scripts[0].code() assert scripts[0].excludes def test_window_isolation(js_tester, request): """Check that greasemonkey scripts get a shadowed global scope.""" # Change something in the global scope setup_script = "window.$ = 'global'" # Greasemonkey script to report back on its scope. test_gm_script = greasemonkey.GreasemonkeyScript.parse( textwrap.dedent(""" // ==UserScript== // @name scopetest // ==/UserScript== // Check the thing the page set is set to the expected type result.push(window.$); result.push($); // Now overwrite it window.$ = 'shadowed'; // And check everything is how the script would expect it to be // after just writing to the "global" scope result.push(window.$); result.push($); """) ) # The compiled source of that scripts with some additional setup # bookending it. test_script = "\n".join([ """ const result = []; """, test_gm_script.code(), """ // Now check that the actual global scope has // not been overwritten result.push(window.$); result.push($); // And return our findings result; """ ]) # What we expect the script to report back. expected = ["global", "global", "shadowed", "shadowed", "global", "global"] # The JSCore in 602.1 doesn't fully support Proxy. xfail = False if (js_tester.tab.backend == usertypes.Backend.QtWebKit and version.qWebKitVersion() == '602.1'): expected[-1] = 'shadowed' expected[-2] = 'shadowed' xfail = True js_tester.run(setup_script) js_tester.run(test_script, expected=expected) if xfail: pytest.xfail("Broken on WebKit 602.1") def test_shared_window_proxy(js_tester): """Check that all scripts have access to the same window proxy.""" # Greasemonkey script to add a property to the window proxy. test_script_a = greasemonkey.GreasemonkeyScript.parse( textwrap.dedent(""" // ==UserScript== // @name a // ==/UserScript== // Set a value from script a window.$ = 'test'; """) ).code() # Greasemonkey script to retrieve a property from the window proxy. test_script_b = greasemonkey.GreasemonkeyScript.parse( textwrap.dedent(""" // ==UserScript== // @name b // ==/UserScript== // Check that the value is accessible from script b return [window.$, $]; """) ).code() js_tester.run(test_script_a) js_tester.run(test_script_b, expected=["test", "test"])