2015-04-03 00:05:20 +02:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
|
2016-01-04 07:12:39 +01:00
|
|
|
# Copyright 2014-2016 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
2015-04-03 00:05:20 +02:00
|
|
|
#
|
|
|
|
# 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 <http://www.gnu.org/licenses/>.
|
|
|
|
|
2015-12-01 20:58:08 +01:00
|
|
|
# pylint: disable=unused-import
|
2015-08-18 20:43:42 +02:00
|
|
|
|
2015-11-23 11:17:26 +01:00
|
|
|
"""The qutebrowser test suite conftest file."""
|
2015-04-03 00:05:20 +02:00
|
|
|
|
2015-06-28 22:58:48 +02:00
|
|
|
import os
|
|
|
|
import sys
|
2015-04-13 07:47:09 +02:00
|
|
|
import collections
|
|
|
|
import itertools
|
2015-08-09 11:49:27 +02:00
|
|
|
import logging
|
2015-09-09 19:25:34 +02:00
|
|
|
import textwrap
|
2015-10-01 21:34:00 +02:00
|
|
|
import warnings
|
2015-04-13 07:47:09 +02:00
|
|
|
|
2015-04-03 00:05:20 +02:00
|
|
|
import pytest
|
2015-12-15 18:09:52 +01:00
|
|
|
import hypothesis
|
2015-04-03 00:05:20 +02:00
|
|
|
|
2015-12-01 22:45:59 +01:00
|
|
|
import helpers.stubs as stubsmod
|
|
|
|
from helpers import logfail
|
|
|
|
from helpers.logfail import fail_on_logging
|
|
|
|
from helpers.messagemock import message_mock
|
2015-07-20 11:23:34 +02:00
|
|
|
from qutebrowser.config import config
|
2015-08-18 20:43:42 +02:00
|
|
|
from qutebrowser.utils import objreg
|
2015-04-09 00:14:06 +02:00
|
|
|
|
2016-01-04 07:09:00 +01:00
|
|
|
from PyQt5.QtCore import QEvent
|
2015-08-30 23:11:23 +02:00
|
|
|
from PyQt5.QtNetwork import QNetworkCookieJar
|
2015-10-07 16:54:46 +02:00
|
|
|
import xvfbwrapper
|
2015-08-30 23:11:23 +02:00
|
|
|
|
2015-04-09 00:14:06 +02:00
|
|
|
|
2015-12-15 18:09:52 +01:00
|
|
|
# Set hypothesis settings
|
2015-12-15 18:57:14 +01:00
|
|
|
hypothesis.Settings.default.strict = True # pylint: disable=no-member
|
2015-12-15 18:09:52 +01:00
|
|
|
|
|
|
|
|
2015-10-02 09:26:33 +02:00
|
|
|
def _apply_platform_markers(item):
|
|
|
|
"""Apply a skip marker to a given item."""
|
|
|
|
markers = [
|
|
|
|
('posix', os.name != 'posix', "Requires a POSIX os"),
|
|
|
|
('windows', os.name != 'nt', "Requires Windows"),
|
|
|
|
('linux', not sys.platform.startswith('linux'), "Requires Linux"),
|
|
|
|
('osx', sys.platform != 'darwin', "Requires OS X"),
|
|
|
|
('not_osx', sys.platform == 'darwin', "Skipped on OS X"),
|
|
|
|
('not_frozen', getattr(sys, 'frozen', False),
|
|
|
|
"Can't be run when frozen"),
|
|
|
|
('frozen', not getattr(sys, 'frozen', False),
|
|
|
|
"Can only run when frozen"),
|
2015-11-24 16:17:07 +01:00
|
|
|
('not_xvfb', item.config.xvfb_display is not None,
|
|
|
|
"Can't be run with Xvfb."),
|
2016-01-06 07:14:30 +01:00
|
|
|
('skip', True, "Always skipped."),
|
2015-10-02 09:26:33 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
for searched_marker, condition, default_reason in markers:
|
|
|
|
marker = item.get_marker(searched_marker)
|
2015-10-02 10:39:21 +02:00
|
|
|
if not marker or not condition:
|
2015-10-02 09:26:33 +02:00
|
|
|
continue
|
|
|
|
|
|
|
|
if 'reason' in marker.kwargs:
|
|
|
|
reason = '{}: {}'.format(default_reason,
|
|
|
|
marker.kwargs['reason'])
|
|
|
|
del marker.kwargs['reason']
|
|
|
|
else:
|
|
|
|
reason = default_reason + '.'
|
|
|
|
skipif_marker = pytest.mark.skipif(condition, *marker.args,
|
|
|
|
reason=reason, **marker.kwargs)
|
|
|
|
item.add_marker(skipif_marker)
|
|
|
|
|
|
|
|
|
2015-04-09 00:14:06 +02:00
|
|
|
def pytest_collection_modifyitems(items):
|
2015-10-02 09:26:33 +02:00
|
|
|
"""Handle custom markers.
|
|
|
|
|
|
|
|
pytest hook called after collection has been performed.
|
|
|
|
|
|
|
|
Adds a marker named "gui" which can be used to filter gui tests from the
|
|
|
|
command line.
|
2015-04-09 06:42:34 +02:00
|
|
|
|
2015-04-09 00:14:06 +02:00
|
|
|
For example:
|
|
|
|
|
|
|
|
py.test -m "not gui" # run all tests except gui tests
|
|
|
|
py.test -m "gui" # run only gui tests
|
|
|
|
|
2015-10-02 09:26:33 +02:00
|
|
|
It also handles the platform specific markers by translating them to skipif
|
|
|
|
markers.
|
|
|
|
|
2015-04-09 00:14:06 +02:00
|
|
|
Args:
|
|
|
|
items: list of _pytest.main.Node items, where each item represents
|
2015-04-09 06:42:34 +02:00
|
|
|
a python test that will be executed.
|
2015-04-09 00:14:06 +02:00
|
|
|
|
|
|
|
Reference:
|
|
|
|
http://pytest.org/latest/plugins.html
|
|
|
|
"""
|
|
|
|
for item in items:
|
2015-08-09 20:47:50 +02:00
|
|
|
if 'qapp' in getattr(item, 'fixturenames', ()):
|
2015-04-09 00:16:45 +02:00
|
|
|
item.add_marker('gui')
|
2015-08-09 20:47:50 +02:00
|
|
|
if sys.platform == 'linux' and not os.environ.get('DISPLAY', ''):
|
2015-09-28 22:19:33 +02:00
|
|
|
if ('CI' in os.environ and
|
|
|
|
not os.environ.get('QUTE_NO_DISPLAY_OK', '')):
|
2015-08-09 20:47:50 +02:00
|
|
|
raise Exception("No display available on CI!")
|
|
|
|
skip_marker = pytest.mark.skipif(
|
|
|
|
True, reason="No DISPLAY available")
|
|
|
|
item.add_marker(skip_marker)
|
2015-04-13 07:47:09 +02:00
|
|
|
|
2015-08-31 21:26:30 +02:00
|
|
|
if hasattr(item, 'module'):
|
|
|
|
module_path = os.path.relpath(
|
|
|
|
item.module.__file__,
|
|
|
|
os.path.commonprefix([__file__, item.module.__file__]))
|
|
|
|
|
|
|
|
module_root_dir = os.path.split(module_path)[0]
|
|
|
|
if module_root_dir == 'integration':
|
|
|
|
item.add_marker(pytest.mark.integration)
|
|
|
|
|
2015-10-02 09:26:33 +02:00
|
|
|
_apply_platform_markers(item)
|
2015-08-18 20:43:42 +02:00
|
|
|
|
|
|
|
|
2015-11-02 20:32:15 +01:00
|
|
|
def pytest_ignore_collect(path):
|
|
|
|
"""Ignore BDD tests during collection if frozen."""
|
|
|
|
rel_path = path.relto(os.path.dirname(__file__))
|
2015-11-13 22:27:41 +01:00
|
|
|
return (rel_path == os.path.join('integration', 'features') and
|
|
|
|
hasattr(sys, 'frozen'))
|
2015-11-02 20:32:15 +01:00
|
|
|
|
|
|
|
|
2015-09-28 21:50:04 +02:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def qapp(qapp):
|
2015-09-08 20:02:53 +02:00
|
|
|
"""Change the name of the QApplication instance."""
|
2015-09-08 22:14:39 +02:00
|
|
|
qapp.setApplicationName('qute_test')
|
2015-09-28 21:50:04 +02:00
|
|
|
return qapp
|
2015-09-08 20:02:53 +02:00
|
|
|
|
|
|
|
|
2015-08-24 08:00:16 +02:00
|
|
|
class WinRegistryHelper:
|
|
|
|
|
|
|
|
"""Helper class for win_registry."""
|
|
|
|
|
|
|
|
FakeWindow = collections.namedtuple('FakeWindow', ['registry'])
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self._ids = []
|
|
|
|
|
|
|
|
def add_window(self, win_id):
|
|
|
|
assert win_id not in objreg.window_registry
|
|
|
|
registry = objreg.ObjectRegistry()
|
|
|
|
window = self.FakeWindow(registry)
|
|
|
|
objreg.window_registry[win_id] = window
|
|
|
|
self._ids.append(win_id)
|
|
|
|
|
|
|
|
def cleanup(self):
|
|
|
|
for win_id in self._ids:
|
|
|
|
del objreg.window_registry[win_id]
|
2015-08-18 20:43:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture
|
|
|
|
def win_registry():
|
2015-08-23 21:55:12 +02:00
|
|
|
"""Fixture providing a window registry for win_id 0 and 1."""
|
2015-08-24 08:00:16 +02:00
|
|
|
helper = WinRegistryHelper()
|
|
|
|
helper.add_window(0)
|
|
|
|
yield helper
|
|
|
|
helper.cleanup()
|
2015-08-18 20:43:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture
|
|
|
|
def tab_registry(win_registry):
|
|
|
|
"""Fixture providing a tab registry for win_id 0."""
|
|
|
|
registry = objreg.ObjectRegistry()
|
|
|
|
objreg.register('tab-registry', registry, scope='window', window=0)
|
|
|
|
yield registry
|
|
|
|
objreg.delete('tab-registry', scope='window', window=0)
|
|
|
|
|
|
|
|
|
2015-04-13 07:47:09 +02:00
|
|
|
def _generate_cmdline_tests():
|
|
|
|
"""Generate testcases for test_split_binding."""
|
|
|
|
# pylint: disable=invalid-name
|
|
|
|
TestCase = collections.namedtuple('TestCase', 'cmd, valid')
|
|
|
|
separators = [';;', ' ;; ', ';; ', ' ;;']
|
|
|
|
invalid = ['foo', '']
|
|
|
|
valid = ['leave-mode', 'hint all']
|
|
|
|
# Valid command only -> valid
|
|
|
|
for item in valid:
|
|
|
|
yield TestCase(''.join(item), True)
|
|
|
|
# Invalid command only -> invalid
|
2015-11-27 16:17:07 +01:00
|
|
|
for item in invalid:
|
|
|
|
yield TestCase(''.join(item), False)
|
2015-04-13 07:47:09 +02:00
|
|
|
# Invalid command combined with invalid command -> invalid
|
|
|
|
for item in itertools.product(invalid, separators, invalid):
|
|
|
|
yield TestCase(''.join(item), False)
|
|
|
|
# Valid command combined with valid command -> valid
|
|
|
|
for item in itertools.product(valid, separators, valid):
|
|
|
|
yield TestCase(''.join(item), True)
|
|
|
|
# Valid command combined with invalid command -> invalid
|
|
|
|
for item in itertools.product(valid, separators, invalid):
|
|
|
|
yield TestCase(''.join(item), False)
|
|
|
|
# Invalid command combined with valid command -> invalid
|
|
|
|
for item in itertools.product(invalid, separators, valid):
|
|
|
|
yield TestCase(''.join(item), False)
|
|
|
|
# Command with no_cmd_split combined with an "invalid" command -> valid
|
|
|
|
for item in itertools.product(['bind x open'], separators, invalid):
|
|
|
|
yield TestCase(''.join(item), True)
|
|
|
|
|
|
|
|
|
2015-11-27 16:25:25 +01:00
|
|
|
@pytest.fixture(params=_generate_cmdline_tests(), ids=lambda e: e.cmd)
|
2015-04-13 07:47:09 +02:00
|
|
|
def cmdline_test(request):
|
|
|
|
"""Fixture which generates tests for things validating commandlines."""
|
|
|
|
# Import qutebrowser.app so all cmdutils.register decorators get run.
|
2015-12-01 20:55:38 +01:00
|
|
|
import qutebrowser.app
|
2015-04-13 07:47:09 +02:00
|
|
|
return request.param
|
2015-05-07 07:58:22 +02:00
|
|
|
|
|
|
|
|
2015-07-20 11:23:34 +02:00
|
|
|
@pytest.yield_fixture
|
|
|
|
def config_stub(stubs):
|
|
|
|
"""Fixture which provides a fake config object."""
|
2015-08-19 19:37:19 +02:00
|
|
|
stub = stubs.ConfigStub()
|
2015-07-20 11:23:34 +02:00
|
|
|
objreg.register('config', stub)
|
|
|
|
yield stub
|
|
|
|
objreg.delete('config')
|
2015-05-07 07:58:22 +02:00
|
|
|
|
|
|
|
|
2015-07-20 11:23:34 +02:00
|
|
|
@pytest.yield_fixture
|
|
|
|
def default_config():
|
|
|
|
"""Fixture that provides and registers an empty default config object."""
|
|
|
|
config_obj = config.ConfigManager(configdir=None, fname=None, relaxed=True)
|
|
|
|
objreg.register('config', config_obj)
|
|
|
|
yield config_obj
|
|
|
|
objreg.delete('config')
|
2015-05-07 07:58:22 +02:00
|
|
|
|
|
|
|
|
2015-07-20 11:23:34 +02:00
|
|
|
@pytest.yield_fixture
|
|
|
|
def key_config_stub(stubs):
|
|
|
|
"""Fixture which provides a fake key config object."""
|
|
|
|
stub = stubs.KeyConfigStub()
|
|
|
|
objreg.register('key-config', stub)
|
|
|
|
yield stub
|
|
|
|
objreg.delete('key-config')
|
2015-05-07 07:58:22 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture
|
2015-07-20 11:23:34 +02:00
|
|
|
def host_blocker_stub(stubs):
|
|
|
|
"""Fixture which provides a fake host blocker object."""
|
|
|
|
stub = stubs.HostBlockerStub()
|
|
|
|
objreg.register('host-blocker', stub)
|
2015-05-07 07:58:22 +02:00
|
|
|
yield stub
|
2015-07-20 11:23:34 +02:00
|
|
|
objreg.delete('host-blocker')
|
2015-06-28 22:58:48 +02:00
|
|
|
|
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def stubs():
|
|
|
|
"""Provide access to stub objects useful for testing."""
|
|
|
|
return stubsmod
|
2015-07-06 17:10:57 +02:00
|
|
|
|
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def unicode_encode_err():
|
|
|
|
"""Provide a fake UnicodeEncodeError exception."""
|
|
|
|
return UnicodeEncodeError('ascii', # codec
|
|
|
|
'', # object
|
|
|
|
0, # start
|
|
|
|
2, # end
|
|
|
|
'fake exception') # reason
|
2015-07-06 17:10:57 +02:00
|
|
|
|
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def qnam(qapp):
|
|
|
|
"""Session-wide QNetworkAccessManager."""
|
|
|
|
from PyQt5.QtNetwork import QNetworkAccessManager
|
|
|
|
nam = QNetworkAccessManager()
|
|
|
|
nam.setNetworkAccessible(QNetworkAccessManager.NotAccessible)
|
|
|
|
return nam
|
2015-07-06 17:10:57 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2015-08-18 20:43:42 +02:00
|
|
|
def webpage(qnam):
|
|
|
|
"""Get a new QWebPage object."""
|
|
|
|
from PyQt5.QtWebKitWidgets import QWebPage
|
2015-07-20 11:23:34 +02:00
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
page = QWebPage()
|
|
|
|
page.networkAccessManager().deleteLater()
|
|
|
|
page.setNetworkAccessManager(qnam)
|
|
|
|
return page
|
2015-08-18 08:02:25 +02:00
|
|
|
|
|
|
|
|
2015-08-23 18:17:22 +02:00
|
|
|
@pytest.fixture
|
|
|
|
def webview(qtbot, webpage):
|
|
|
|
"""Get a new QWebView object."""
|
|
|
|
from PyQt5.QtWebKitWidgets import QWebView
|
|
|
|
|
|
|
|
view = QWebView()
|
|
|
|
qtbot.add_widget(view)
|
|
|
|
|
|
|
|
view.page().deleteLater()
|
|
|
|
view.setPage(webpage)
|
|
|
|
|
|
|
|
view.resize(640, 480)
|
|
|
|
return view
|
|
|
|
|
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
@pytest.fixture
|
|
|
|
def webframe(webpage):
|
|
|
|
"""Convenience fixture to get a mainFrame of a QWebPage."""
|
|
|
|
return webpage.mainFrame()
|
2015-08-18 08:02:25 +02:00
|
|
|
|
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
@pytest.fixture
|
|
|
|
def fake_keyevent_factory():
|
|
|
|
"""Fixture that when called will return a mock instance of a QKeyEvent."""
|
|
|
|
from unittest import mock
|
|
|
|
from PyQt5.QtGui import QKeyEvent
|
2015-08-18 08:02:25 +02:00
|
|
|
|
2016-01-04 07:09:00 +01:00
|
|
|
def fake_keyevent(key, modifiers=0, text='', typ=QEvent.KeyPress):
|
2015-08-18 20:43:42 +02:00
|
|
|
"""Generate a new fake QKeyPressEvent."""
|
|
|
|
evtmock = mock.create_autospec(QKeyEvent, instance=True)
|
|
|
|
evtmock.key.return_value = key
|
|
|
|
evtmock.modifiers.return_value = modifiers
|
|
|
|
evtmock.text.return_value = text
|
2016-01-04 07:09:00 +01:00
|
|
|
evtmock.type.return_value = typ
|
2015-08-18 20:43:42 +02:00
|
|
|
return evtmock
|
2015-08-18 08:02:25 +02:00
|
|
|
|
2015-08-18 20:43:42 +02:00
|
|
|
return fake_keyevent
|
2015-08-30 23:11:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture
|
|
|
|
def cookiejar_and_cache(stubs):
|
|
|
|
"""Fixture providing a fake cookie jar and cache."""
|
|
|
|
jar = QNetworkCookieJar()
|
|
|
|
cache = stubs.FakeNetworkCache()
|
|
|
|
objreg.register('cookie-jar', jar)
|
|
|
|
objreg.register('cache', cache)
|
|
|
|
yield
|
|
|
|
objreg.delete('cookie-jar')
|
|
|
|
objreg.delete('cache')
|
2015-09-09 19:25:34 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def py_proc():
|
|
|
|
"""Get a python executable and args list which executes the given code."""
|
2015-12-21 09:46:30 +01:00
|
|
|
if getattr(sys, 'frozen', False):
|
|
|
|
pytest.skip("Can't be run when frozen")
|
2015-09-09 19:25:34 +02:00
|
|
|
def func(code):
|
|
|
|
return (sys.executable, ['-c', textwrap.dedent(code.strip('\n'))])
|
|
|
|
return func
|
2015-10-01 21:34:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture(autouse=True)
|
|
|
|
def fail_tests_on_warnings():
|
|
|
|
warnings.simplefilter('error')
|
|
|
|
yield
|
|
|
|
warnings.resetwarnings()
|
2015-10-07 16:54:46 +02:00
|
|
|
|
|
|
|
|
2015-10-07 23:05:39 +02:00
|
|
|
def pytest_addoption(parser):
|
|
|
|
parser.addoption('--no-xvfb', action='store_true', default=False,
|
|
|
|
help='Disable xvfb in tests.')
|
2015-11-18 20:01:40 +01:00
|
|
|
parser.addoption('--qute-delay', action='store', default=0, type=int,
|
|
|
|
help="Delay between qutebrowser commands.")
|
2015-10-07 23:05:39 +02:00
|
|
|
|
|
|
|
|
2015-10-07 16:54:46 +02:00
|
|
|
def pytest_configure(config):
|
|
|
|
"""Start Xvfb if we're on Linux, not on a CI and Xvfb is available.
|
|
|
|
|
|
|
|
This is a lot nicer than having windows popping up.
|
|
|
|
"""
|
|
|
|
config.xvfb_display = None
|
2015-10-08 06:36:38 +02:00
|
|
|
if os.environ.get('DISPLAY', None) == '':
|
|
|
|
# xvfbwrapper doesn't handle DISPLAY="" correctly
|
|
|
|
del os.environ['DISPLAY']
|
2015-10-07 23:05:39 +02:00
|
|
|
if sys.platform.startswith('linux') and not config.getoption('--no-xvfb'):
|
|
|
|
assert 'QUTE_BUILDBOT' not in os.environ
|
2015-10-07 16:54:46 +02:00
|
|
|
try:
|
|
|
|
disp = xvfbwrapper.Xvfb(width=800, height=600, colordepth=16)
|
|
|
|
disp.start()
|
|
|
|
except FileNotFoundError:
|
|
|
|
# We run without Xvfb if it's unavailable.
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
config.xvfb_display = disp
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_unconfigure(config):
|
|
|
|
if config.xvfb_display is not None:
|
|
|
|
config.xvfb_display.stop()
|