# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2014-2015 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 . """The qutebrowser test suite contest file.""" import os import sys import collections import itertools import pytest import stubs as stubsmod from qutebrowser.config import configexc from qutebrowser.utils import objreg, usertypes @pytest.fixture(scope='session', autouse=True) def app_and_logging(qapp): """Initialize a QApplication and logging. This ensures that a QApplication is created and used by all tests. """ from log import init init() @pytest.fixture(scope='session') def stubs(): """Provide access to stub objects useful for testing.""" return stubsmod @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 @pytest.fixture(scope='session') def qnam(): """Session-wide QNetworkAccessManager.""" from PyQt5.QtNetwork import QNetworkAccessManager nam = QNetworkAccessManager() nam.setNetworkAccessible(QNetworkAccessManager.NotAccessible) return nam @pytest.fixture def webpage(qnam): """Get a new QWebPage object.""" from PyQt5.QtWebKitWidgets import QWebPage page = QWebPage() page.networkAccessManager().deleteLater() page.setNetworkAccessManager(qnam) return page @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 def fake_keyevent(key, modifiers=0, text=''): """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 return evtmock return fake_keyevent def pytest_collection_modifyitems(items): """Automatically add a 'gui' marker to all gui-related tests. 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. For example: py.test -m "not gui" # run all tests except gui tests py.test -m "gui" # run only gui tests Args: items: list of _pytest.main.Node items, where each item represents a python test that will be executed. Reference: http://pytest.org/latest/plugins.html """ for item in items: if 'qtbot' in getattr(item, 'fixturenames', ()): item.add_marker('gui') 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 for item in valid: yield TestCase(''.join(item), True) # 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) @pytest.fixture(params=_generate_cmdline_tests()) def cmdline_test(request): """Fixture which generates tests for things validating commandlines.""" # Import qutebrowser.app so all cmdutils.register decorators get run. import qutebrowser.app # pylint: disable=unused-variable return request.param class ConfigStub: """Stub for the config module. Attributes: data: The config data to return. """ def __init__(self, signal): """Constructor. Args: signal: The signal to use for self.changed. """ self.data = {} self.changed = signal def section(self, name): """Get a section from the config. Args: name: The section name to get. Return: The section as dict. """ return self.data[name] def get(self, sect, opt): """Get a value from the config.""" data = self.data[sect] try: return data[opt] except KeyError: raise configexc.NoOptionError(opt, sect) @pytest.yield_fixture def config_stub(stubs): """Fixture which provides a fake config object.""" stub = ConfigStub(stubs.FakeSignal()) objreg.register('config', stub) yield stub objreg.delete('config') def pytest_runtest_setup(item): """Add some custom markers.""" if not isinstance(item, item.Function): return if item.get_marker('posix') and os.name != 'posix': pytest.skip("Requires a POSIX os.") elif item.get_marker('windows') and os.name != 'nt': pytest.skip("Requires Windows.") elif item.get_marker('linux') and not sys.platform.startswith('linux'): pytest.skip("Requires Linux.") elif item.get_marker('osx') and sys.platform != 'darwin': pytest.skip("Requires OS X.") elif item.get_marker('not_frozen') and getattr(sys, 'frozen', False): pytest.skip("Can't be run when frozen!") elif item.get_marker('frozen') and not getattr(sys, 'frozen', False): pytest.skip("Can only run when frozen!") class MessageMock: """Helper object for message_mock. Attributes: _monkeypatch: The pytest monkeypatch fixture. MessageLevel: An enum with possible message levels. Message: A namedtuple representing a message. messages: A list of Message tuples. """ Message = collections.namedtuple('Message', ['level', 'win_id', 'text', 'immediate']) MessageLevel = usertypes.enum('Level', ('error', 'info', 'warning')) def __init__(self, monkeypatch): self._monkeypatch = monkeypatch self.messages = [] def _handle(self, level, win_id, text, immediately=False): self.messages.append(self.Message(level, win_id, text, immediately)) def _handle_error(self, *args, **kwargs): self._handle(self.MessageLevel.error, *args, **kwargs) def _handle_info(self, *args, **kwargs): self._handle(self.MessageLevel.info, *args, **kwargs) def _handle_warning(self, *args, **kwargs): self._handle(self.MessageLevel.warning, *args, **kwargs) def getmsg(self): """Get the only message in self.messages. Raises ValueError if there are multiple or no messages. """ if len(self.messages) != 1: raise ValueError("Got {} messages but expected a single " "one.".format(len(self.messages))) return self.messages[0] def patch(self, module_path): """Patch message.* in the given module (as a string).""" self._monkeypatch.setattr(module_path + '.error', self._handle_error) self._monkeypatch.setattr(module_path + '.info', self._handle_info) self._monkeypatch.setattr(module_path + '.warning', self._handle_warning) @pytest.fixture def message_mock(monkeypatch): """Fixture to get a MessageMock.""" return MessageMock(monkeypatch)