# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2016 Ryan Roden-Corrent (rcorre) # # 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 the Completer Object.""" import unittest.mock import pytest from PyQt5.QtGui import QStandardItemModel from qutebrowser.completion import completer from qutebrowser.utils import usertypes class FakeCompletionModel(QStandardItemModel): """Stub for a completion model.""" DUMB_SORT = None def __init__(self, kind, parent=None): super().__init__(parent) self.kind = kind @pytest.fixture def cmd(stubs, qtbot): """Create the statusbar command prompt the completer uses.""" cmd = stubs.FakeStatusbarCommand() qtbot.addWidget(cmd) return cmd @pytest.fixture def completer_obj(qtbot, cmd, config_stub): """Create the completer used for testing.""" config_stub.data = {'completion': {'auto-open': False}} return completer.Completer(cmd, 0) @pytest.fixture(autouse=True) def instances(monkeypatch): """Mock the instances module so get returns a fake completion model.""" # populate a model for each completion type, with a nested structure for # option and value completion instances = {kind: FakeCompletionModel(kind) for kind in usertypes.Completion} instances[usertypes.Completion.option] = { 'general': FakeCompletionModel(usertypes.Completion.option), } instances[usertypes.Completion.value] = { 'general': { 'ignore-case': FakeCompletionModel(usertypes.Completion.value), } } monkeypatch.setattr('qutebrowser.completion.completer.instances', instances) @pytest.fixture(autouse=True) def cmdutils_patch(monkeypatch, stubs): """Patch the cmdutils module to provide fake commands.""" cmds = { 'set': [usertypes.Completion.section, usertypes.Completion.option, usertypes.Completion.value], 'help': [usertypes.Completion.helptopic], 'quickmark-load': [usertypes.Completion.quickmark_by_name], 'bookmark-load': [usertypes.Completion.bookmark_by_url], 'open': [usertypes.Completion.url], 'buffer': [usertypes.Completion.tab], 'session-load': [usertypes.Completion.sessions], 'bind': [usertypes.Completion.empty, usertypes.Completion.command], 'tab-detach': None, } cmd_utils = stubs.FakeCmdUtils({ name: stubs.FakeCommand(completion=compl) for name, compl in cmds.items() }) monkeypatch.setattr('qutebrowser.completion.completer.cmdutils', cmd_utils) @pytest.mark.parametrize('txt, expected', [ (':nope|', usertypes.Completion.command), (':nope |', None), (':set |', usertypes.Completion.section), (':set gen|', usertypes.Completion.section), (':set general |', usertypes.Completion.option), (':set what |', None), (':set general ignore-case |', usertypes.Completion.value), (':set general huh |', None), (':help |', usertypes.Completion.helptopic), (':quickmark-load |', usertypes.Completion.quickmark_by_name), (':bookmark-load |', usertypes.Completion.bookmark_by_url), (':open |', usertypes.Completion.url), (':buffer |', usertypes.Completion.tab), (':session-load |', usertypes.Completion.sessions), (':bind |', usertypes.Completion.empty), (':bind |', usertypes.Completion.command), (':bind foo|', usertypes.Completion.command), (':bind | foo', usertypes.Completion.empty), (':set| general ', usertypes.Completion.command), (':|set general ', usertypes.Completion.command), (':set gene|ral ignore-case', usertypes.Completion.section), (':|', usertypes.Completion.command), (': |', usertypes.Completion.command), (':bookmark-load |', usertypes.Completion.bookmark_by_url), ('/|', None), (':open -t|', None), (':open --tab|', None), (':open -t |', usertypes.Completion.url), (':open --tab |', usertypes.Completion.url), (':open | -t', usertypes.Completion.url), (':--foo --bar |', None), (':tab-detach |', None), (':bind --mode=caret |', usertypes.Completion.command), #(':bind --mode caret |', usertypes.Completion.command), KNOWN BUG (':set -t -p |', usertypes.Completion.section), (':open -- |', None), ]) def test_update_completion(txt, expected, cmd, completer_obj, completion_widget_stub): """Test setting the completion widget's model based on command text.""" # this test uses | as a placeholder for the current cursor position cursor_pos = txt.index('|') cmd.setText(txt.replace('|', '')) cmd.setCursorPosition(cursor_pos) completer_obj.update_completion() if expected is None: assert not completion_widget_stub.set_model.called else: assert completion_widget_stub.set_model.call_count == 1 arg = completion_widget_stub.set_model.call_args[0][0] # the outer model is just for sorting; srcmodel is the completion model assert arg.srcmodel.kind == expected def test_completion_item_prev(completer_obj, cmd, completion_widget_stub, config_stub): """Test that completion_item_prev emits next_prev_item.""" cmd.setText(':') slot = unittest.mock.Mock() completer_obj.next_prev_item.connect(slot) completer_obj.completion_item_prev() slot.assert_called_with(True) def test_completion_item_next(completer_obj, cmd, completion_widget_stub, config_stub): """Test that completion_item_next emits next_prev_item.""" cmd.setText(':') slot = unittest.mock.Mock() completer_obj.next_prev_item.connect(slot) completer_obj.completion_item_next() slot.assert_called_with(False) @pytest.mark.parametrize('before, newtxt, immediate, after', [ (':foo |', 'bar', False, ':foo bar|'), (':foo |', 'bar', True, ':foo bar |'), (':foo | bar', 'baz', False, ':foo baz| bar'), (':foo | bar', 'baz', True, ':foo baz |bar'), ]) def test_change_completed_part(before, newtxt, after, immediate, completer_obj, cmd, completion_widget_stub, config_stub): """Test that change_completed_part modifies the cmd text properly.""" before_pos = before.index('|') after_pos = after.index('|') before_txt = before.replace('|', '') after_txt = after.replace('|', '') cmd.setText(before_txt) cmd.setCursorPosition(before_pos) completer_obj.update_cursor_part() completer_obj.change_completed_part(newtxt, immediate) assert cmd.text() == after_txt assert cmd.cursorPosition() == after_pos