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
|
|
|
|
2016-06-30 14:02:30 +02:00
|
|
|
import re
|
2015-06-28 22:58:48 +02:00
|
|
|
import os
|
|
|
|
import sys
|
2015-10-01 21:34:00 +02:00
|
|
|
import warnings
|
2016-06-30 14:02:30 +02:00
|
|
|
import operator
|
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
|
|
|
from helpers import logfail
|
|
|
|
from helpers.logfail import fail_on_logging
|
|
|
|
from helpers.messagemock import message_mock
|
2016-02-04 07:10:12 +01:00
|
|
|
from helpers.fixtures import * # pylint: disable=wildcard-import
|
2015-04-09 00:14:06 +02:00
|
|
|
|
2016-02-04 07:10:12 +01:00
|
|
|
from PyQt5.QtCore import PYQT_VERSION
|
2015-08-30 23:11:23 +02:00
|
|
|
|
2016-06-30 14:02:30 +02:00
|
|
|
from qutebrowser.utils import qtutils
|
|
|
|
|
2015-04-09 00:14:06 +02:00
|
|
|
|
2015-12-15 18:09:52 +01:00
|
|
|
# Set hypothesis settings
|
2016-01-10 21:57:06 +01:00
|
|
|
hypothesis.settings.register_profile('default',
|
|
|
|
hypothesis.settings(strict=True))
|
|
|
|
hypothesis.settings.load_profile('default')
|
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 = [
|
2016-02-27 02:45:34 +01:00
|
|
|
('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"),
|
|
|
|
('ci', 'CI' not in os.environ, "Only runs on CI."),
|
2015-10-02 09:26:33 +02:00
|
|
|
]
|
|
|
|
|
2016-02-27 02:45:34 +01:00
|
|
|
for searched_marker, condition, default_reason in markers:
|
2015-10-02 09:26:33 +02:00
|
|
|
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
|
|
|
|
|
2016-02-27 02:45:34 +01:00
|
|
|
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)
|
2015-10-02 09:26:33 +02:00
|
|
|
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:
|
|
|
|
|
2016-08-22 07:41:10 +02:00
|
|
|
pytest -m "not gui" # run all tests except gui tests
|
|
|
|
pytest -m "gui" # run only gui tests
|
2015-04-09 00:14:06 +02:00
|
|
|
|
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-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]
|
2016-05-29 18:20:00 +02:00
|
|
|
if module_root_dir == 'end2end':
|
|
|
|
item.add_marker(pytest.mark.end2end)
|
2015-08-31 21:26:30 +02:00
|
|
|
|
2015-10-02 09:26:33 +02:00
|
|
|
_apply_platform_markers(item)
|
2016-01-14 07:53:00 +01:00
|
|
|
if item.get_marker('xfail_norun'):
|
|
|
|
item.add_marker(pytest.mark.xfail(run=False))
|
2016-06-09 11:17:18 +02:00
|
|
|
if item.get_marker('flaky_once'):
|
|
|
|
item.add_marker(pytest.mark.flaky(reruns=1))
|
2015-08-18 20:43:42 +02:00
|
|
|
|
|
|
|
|
2015-11-02 20:32:15 +01:00
|
|
|
def pytest_ignore_collect(path):
|
2016-08-05 14:36:32 +02:00
|
|
|
"""Ignore BDD tests if we're unable to run them."""
|
2016-08-19 17:17:08 +02:00
|
|
|
skip_bdd = hasattr(sys, 'frozen')
|
2015-11-02 20:32:15 +01:00
|
|
|
rel_path = path.relto(os.path.dirname(__file__))
|
2016-08-05 14:36:32 +02:00
|
|
|
return rel_path == os.path.join('end2end', 'features') and skip_bdd
|
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
|
|
|
|
|
|
|
|
2016-08-22 06:50:00 +02:00
|
|
|
@pytest.fixture(scope='function', autouse=True)
|
|
|
|
def bug_workaround():
|
|
|
|
# WORKAROUND for https://github.com/pytest-dev/pytest/issues/1832
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2015-10-07 23:05:39 +02:00
|
|
|
def pytest_addoption(parser):
|
2015-11-18 20:01:40 +01:00
|
|
|
parser.addoption('--qute-delay', action='store', default=0, type=int,
|
|
|
|
help="Delay between qutebrowser commands.")
|
2016-02-11 08:01:29 +01:00
|
|
|
parser.addoption('--qute-profile-subprocs', action='store_true',
|
|
|
|
default=False, help="Run cProfile for subprocesses.")
|
2016-07-11 17:24:03 +02:00
|
|
|
parser.addoption('--qute-bdd-webengine', action='store_true',
|
|
|
|
help='Use QtWebEngine for BDD tests')
|
2015-10-07 23:05:39 +02:00
|
|
|
|
|
|
|
|
2016-02-16 20:36:23 +01:00
|
|
|
@pytest.fixture(scope='session', autouse=True)
|
2016-04-06 08:27:43 +02:00
|
|
|
def check_display(request):
|
2016-02-16 20:36:23 +01:00
|
|
|
if (not request.config.getoption('--no-xvfb') and
|
|
|
|
'QUTE_BUILDBOT' in os.environ and
|
|
|
|
request.config.xvfb is not None):
|
|
|
|
raise Exception("Xvfb is running on buildbot!")
|
2016-02-04 06:43:14 +01:00
|
|
|
|
2016-04-06 08:27:43 +02:00
|
|
|
if sys.platform == 'linux' and not os.environ.get('DISPLAY', ''):
|
|
|
|
raise Exception("No display and no Xvfb available!")
|
|
|
|
|
2016-02-04 06:43:14 +01:00
|
|
|
|
|
|
|
@pytest.hookimpl(tryfirst=True, hookwrapper=True)
|
|
|
|
def pytest_runtest_makereport(item, call):
|
|
|
|
"""Make test information available in fixtures.
|
|
|
|
|
|
|
|
See http://pytest.org/latest/example/simple.html#making-test-result-information-available-in-fixtures
|
|
|
|
"""
|
|
|
|
outcome = yield
|
|
|
|
rep = outcome.get_result()
|
|
|
|
setattr(item, "rep_" + rep.when, rep)
|
2016-05-29 21:47:46 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.hookimpl(hookwrapper=True)
|
|
|
|
def pytest_sessionfinish(exitstatus):
|
|
|
|
"""Create a file to tell run_pytest.py how pytest exited."""
|
|
|
|
outcome = yield
|
|
|
|
outcome.get_result()
|
2016-05-29 22:21:35 +02:00
|
|
|
|
|
|
|
cache_dir = os.path.join(os.path.abspath(os.path.dirname(__file__)),
|
|
|
|
'..', '.cache')
|
|
|
|
try:
|
|
|
|
os.mkdir(cache_dir)
|
|
|
|
except FileExistsError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
status_file = os.path.join(cache_dir, 'pytest_status')
|
2016-05-29 21:47:46 +02:00
|
|
|
with open(status_file, 'w', encoding='ascii') as f:
|
|
|
|
f.write(str(exitstatus))
|
2016-06-30 14:02:30 +02:00
|
|
|
|
|
|
|
|
2016-06-30 17:40:54 +02:00
|
|
|
if not getattr(sys, 'frozen', False):
|
2016-08-18 17:41:34 +02:00
|
|
|
def _get_version_tag(tag):
|
2016-06-30 17:42:05 +02:00
|
|
|
"""Handle tags like pyqt>=5.3.1 for BDD tests.
|
|
|
|
|
|
|
|
This transforms e.g. pyqt>=5.3.1 into an appropriate @pytest.mark.skip
|
|
|
|
marker, and falls back to pytest-bdd's implementation for all other
|
|
|
|
casesinto an appropriate @pytest.mark.skip marker, and falls back to
|
|
|
|
"""
|
2016-06-30 17:40:54 +02:00
|
|
|
version_re = re.compile(r"""
|
|
|
|
(?P<package>qt|pyqt)
|
|
|
|
(?P<operator>==|>|>=|<|<=|!=)
|
|
|
|
(?P<version>\d+\.\d+\.\d+)
|
|
|
|
""", re.VERBOSE)
|
|
|
|
|
|
|
|
match = version_re.match(tag)
|
|
|
|
if not match:
|
|
|
|
return None
|
|
|
|
|
|
|
|
operators = {
|
|
|
|
'==': operator.eq,
|
|
|
|
'>': operator.gt,
|
|
|
|
'<': operator.lt,
|
|
|
|
'>=': operator.ge,
|
|
|
|
'<=': operator.le,
|
|
|
|
'!=': operator.ne,
|
|
|
|
}
|
|
|
|
|
|
|
|
package = match.group('package')
|
|
|
|
op = operators[match.group('operator')]
|
|
|
|
version = match.group('version')
|
|
|
|
|
|
|
|
if package == 'qt':
|
2016-08-18 17:41:34 +02:00
|
|
|
return pytest.mark.skipif(qtutils.version_check(version, op),
|
2016-06-30 17:40:54 +02:00
|
|
|
reason='Needs ' + tag)
|
|
|
|
elif package == 'pyqt':
|
|
|
|
major, minor, patch = [int(e) for e in version.split('.')]
|
|
|
|
hex_version = (major << 16) | (minor << 8) | patch
|
2016-08-18 17:41:34 +02:00
|
|
|
return pytest.mark.skipif(not op(PYQT_VERSION, hex_version),
|
2016-06-30 17:40:54 +02:00
|
|
|
reason='Needs ' + tag)
|
|
|
|
else:
|
|
|
|
raise ValueError("Invalid package {!r}".format(package))
|
|
|
|
|
2016-08-18 17:41:34 +02:00
|
|
|
def _get_qtwebengine_tag(tag):
|
2016-08-18 20:33:53 +02:00
|
|
|
"""Handle a @qtwebengine_* tag."""
|
|
|
|
pytest_marks = {
|
|
|
|
'qtwebengine_todo': pytest.mark.qtwebengine_todo,
|
|
|
|
'qtwebengine_skip': pytest.mark.qtwebengine_skip,
|
|
|
|
}
|
|
|
|
if not any(tag.startswith(t + ':') for t in pytest_marks):
|
2016-08-18 17:41:34 +02:00
|
|
|
return None
|
2016-08-18 20:33:53 +02:00
|
|
|
name, desc = tag.split(':', maxsplit=1)
|
|
|
|
return pytest_marks[name](desc)
|
2016-08-18 17:41:34 +02:00
|
|
|
|
|
|
|
def pytest_bdd_apply_tag(tag, function):
|
|
|
|
"""Handle custom tags for BDD tests.
|
|
|
|
|
|
|
|
This tries various functions, and if none knows how to handle this tag,
|
|
|
|
it returns None so it falls back to pytest-bdd's implementation.
|
|
|
|
"""
|
|
|
|
funcs = [_get_version_tag, _get_qtwebengine_tag]
|
|
|
|
for func in funcs:
|
|
|
|
mark = func(tag)
|
|
|
|
if mark is not None:
|
|
|
|
mark(function)
|
|
|
|
return True
|
|
|
|
return None
|