2014-12-15 22:06:11 +01:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
2017-05-09 21:37:03 +02:00
|
|
|
# Copyright 2014-2017 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
2014-12-15 22:06:11 +01: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/>.
|
|
|
|
|
|
|
|
"""Tests for qutebrowser.config.config."""
|
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
import copy
|
2017-09-15 11:31:55 +02:00
|
|
|
import types
|
2017-10-03 14:12:29 +02:00
|
|
|
import unittest.mock
|
2017-07-02 17:12:31 +02:00
|
|
|
|
2017-07-02 16:05:04 +02:00
|
|
|
import pytest
|
2017-10-02 07:06:05 +02:00
|
|
|
from PyQt5.QtCore import QObject
|
2017-07-03 12:12:55 +02:00
|
|
|
from PyQt5.QtGui import QColor
|
2017-07-02 16:05:04 +02:00
|
|
|
|
2017-10-03 13:05:01 +02:00
|
|
|
from qutebrowser.config import config, configdata, configexc, configfiles
|
2017-10-02 07:06:05 +02:00
|
|
|
from qutebrowser.utils import usertypes
|
2017-07-03 13:58:19 +02:00
|
|
|
from qutebrowser.misc import objects
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def configdata_init():
|
|
|
|
"""Initialize configdata if needed."""
|
|
|
|
if configdata.DATA is None:
|
|
|
|
configdata.init()
|
|
|
|
|
|
|
|
|
|
|
|
class TestChangeFilter:
|
|
|
|
|
2017-07-03 13:20:10 +02:00
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def cleanup_globals(self, monkeypatch):
|
2017-09-22 14:08:06 +02:00
|
|
|
"""Make sure config.change_filters is cleaned up."""
|
|
|
|
monkeypatch.setattr(config, 'change_filters', [])
|
2017-07-03 13:20:10 +02:00
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
@pytest.mark.parametrize('option', ['foobar', 'tab', 'tabss', 'tabs.'])
|
|
|
|
def test_unknown_option(self, option):
|
|
|
|
cf = config.change_filter(option)
|
|
|
|
with pytest.raises(configexc.NoOptionError):
|
|
|
|
cf.validate()
|
2017-07-02 16:05:04 +02:00
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
@pytest.mark.parametrize('option', ['confirm_quit', 'tabs', 'tabs.show'])
|
|
|
|
def test_validate(self, option):
|
|
|
|
cf = config.change_filter(option)
|
|
|
|
cf.validate()
|
2017-09-22 14:08:06 +02:00
|
|
|
assert cf in config.change_filters
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
@pytest.mark.parametrize('method', [True, False])
|
|
|
|
@pytest.mark.parametrize('option, changed, matches', [
|
|
|
|
('confirm_quit', 'confirm_quit', True),
|
|
|
|
('tabs', 'tabs.show', True),
|
|
|
|
('tabs.show', 'tabs.show', True),
|
|
|
|
('tabs', None, True),
|
|
|
|
('tabs', 'colors.tabs.bar.bg', False),
|
|
|
|
])
|
|
|
|
def test_call(self, method, option, changed, matches):
|
|
|
|
was_called = False
|
|
|
|
if method:
|
|
|
|
|
|
|
|
class Foo:
|
|
|
|
|
|
|
|
@config.change_filter(option)
|
|
|
|
def meth(self):
|
|
|
|
nonlocal was_called
|
|
|
|
was_called = True
|
|
|
|
|
|
|
|
foo = Foo()
|
2017-07-04 15:33:58 +02:00
|
|
|
foo.meth(changed) # pylint: disable=too-many-function-args
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
else:
|
2017-07-02 16:05:04 +02:00
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
@config.change_filter(option, function=True)
|
|
|
|
def func():
|
|
|
|
nonlocal was_called
|
|
|
|
was_called = True
|
|
|
|
|
2017-07-04 15:33:58 +02:00
|
|
|
func(changed) # pylint: disable=too-many-function-args
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
assert was_called == matches
|
|
|
|
|
|
|
|
|
|
|
|
class TestKeyConfig:
|
|
|
|
|
2017-07-02 22:10:28 +02:00
|
|
|
@pytest.fixture
|
|
|
|
def no_bindings(self):
|
|
|
|
"""Get a dict with no bindings."""
|
|
|
|
return {'normal': {}}
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('key, expected', [
|
|
|
|
('A', 'A'),
|
|
|
|
('<Ctrl-X>', '<ctrl+x>'),
|
|
|
|
])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_prepare_valid(self, key_config_stub, key, expected):
|
2017-07-02 22:10:28 +02:00
|
|
|
"""Make sure prepare normalizes the key."""
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub._prepare(key, 'normal') == expected
|
2017-07-02 22:10:28 +02:00
|
|
|
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_prepare_invalid(self, key_config_stub):
|
2017-07-02 22:10:28 +02:00
|
|
|
"""Make sure prepare checks the mode."""
|
|
|
|
with pytest.raises(configexc.KeybindingError):
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub._prepare('x', 'abnormal')
|
2017-07-02 22:10:28 +02:00
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
@pytest.mark.parametrize('commands, expected', [
|
|
|
|
# Unbinding default key
|
2017-07-02 21:07:38 +02:00
|
|
|
({'a': None}, {'b': 'message-info bar'}),
|
2017-07-02 17:12:31 +02:00
|
|
|
# Additional binding
|
2017-07-02 21:07:38 +02:00
|
|
|
({'c': 'message-info baz'},
|
2017-07-04 15:33:58 +02:00
|
|
|
{'a': 'message-info foo', 'b': 'message-info bar',
|
|
|
|
'c': 'message-info baz'}),
|
2017-07-02 17:12:31 +02:00
|
|
|
# Unbinding unknown key
|
2017-07-02 21:07:38 +02:00
|
|
|
({'x': None}, {'a': 'message-info foo', 'b': 'message-info bar'}),
|
2017-07-02 17:12:31 +02:00
|
|
|
])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_get_bindings_for_and_get_command(self, key_config_stub,
|
|
|
|
config_stub,
|
2017-07-02 22:10:28 +02:00
|
|
|
commands, expected):
|
2017-07-02 21:07:38 +02:00
|
|
|
orig_default_bindings = {'normal': {'a': 'message-info foo',
|
|
|
|
'b': 'message-info bar'},
|
|
|
|
'insert': {},
|
|
|
|
'hint': {},
|
|
|
|
'passthrough': {},
|
|
|
|
'command': {},
|
|
|
|
'prompt': {},
|
|
|
|
'caret': {},
|
|
|
|
'register': {}}
|
2017-07-02 17:12:31 +02:00
|
|
|
config_stub.val.bindings.default = copy.deepcopy(orig_default_bindings)
|
|
|
|
config_stub.val.bindings.commands = {'normal': commands}
|
2017-10-05 11:09:07 +02:00
|
|
|
bindings = key_config_stub.get_bindings_for('normal')
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
# Make sure the code creates a copy and doesn't modify the setting
|
|
|
|
assert config_stub.val.bindings.default == orig_default_bindings
|
|
|
|
assert bindings == expected
|
2017-07-02 22:10:28 +02:00
|
|
|
for key, command in expected.items():
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub.get_command(key, 'normal') == command
|
2017-07-02 22:10:28 +02:00
|
|
|
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_get_command_unbound(self, key_config_stub, config_stub,
|
|
|
|
no_bindings):
|
2017-07-02 22:10:28 +02:00
|
|
|
config_stub.val.bindings.default = no_bindings
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub.get_command('foobar', 'normal') is None
|
2017-07-02 17:12:31 +02:00
|
|
|
|
|
|
|
@pytest.mark.parametrize('bindings, expected', [
|
|
|
|
# Simple
|
2017-07-02 21:07:38 +02:00
|
|
|
({'a': 'message-info foo', 'b': 'message-info bar'},
|
|
|
|
{'message-info foo': ['a'], 'message-info bar': ['b']}),
|
2017-07-02 17:12:31 +02:00
|
|
|
# Multiple bindings
|
2017-07-02 21:07:38 +02:00
|
|
|
({'a': 'message-info foo', 'b': 'message-info foo'},
|
|
|
|
{'message-info foo': ['b', 'a']}),
|
|
|
|
# With special keys (should be listed last and normalized)
|
|
|
|
({'a': 'message-info foo', '<Escape>': 'message-info foo'},
|
|
|
|
{'message-info foo': ['a', '<escape>']}),
|
2017-07-02 17:12:31 +02:00
|
|
|
# Chained command
|
2017-07-02 21:07:38 +02:00
|
|
|
({'a': 'message-info foo ;; message-info bar'},
|
|
|
|
{'message-info foo': ['a'], 'message-info bar': ['a']}),
|
2017-07-02 17:12:31 +02:00
|
|
|
])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_get_reverse_bindings_for(self, key_config_stub, config_stub,
|
|
|
|
no_bindings, bindings, expected):
|
2017-07-02 22:10:28 +02:00
|
|
|
config_stub.val.bindings.default = no_bindings
|
2017-07-02 17:12:31 +02:00
|
|
|
config_stub.val.bindings.commands = {'normal': bindings}
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub.get_reverse_bindings_for('normal') == expected
|
2017-07-02 17:12:31 +02:00
|
|
|
|
2017-07-03 09:09:50 +02:00
|
|
|
@pytest.mark.parametrize('key', ['a', '<Ctrl-X>', 'b'])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_bind_duplicate(self, key_config_stub, config_stub, key):
|
2017-07-03 09:09:50 +02:00
|
|
|
config_stub.val.bindings.default = {'normal': {'a': 'nop',
|
|
|
|
'<Ctrl+x>': 'nop'}}
|
|
|
|
config_stub.val.bindings.commands = {'normal': {'b': 'nop'}}
|
2017-10-05 11:09:07 +02:00
|
|
|
key_config_stub.bind(key, 'message-info foo', mode='normal')
|
|
|
|
assert key_config_stub.get_command(key, 'normal') == 'message-info foo'
|
2017-07-02 22:10:28 +02:00
|
|
|
|
|
|
|
@pytest.mark.parametrize('mode', ['normal', 'caret'])
|
2017-09-21 22:28:51 +02:00
|
|
|
@pytest.mark.parametrize('command', [
|
|
|
|
'message-info foo',
|
|
|
|
'nop ;; wq', # https://github.com/qutebrowser/qutebrowser/issues/3002
|
|
|
|
])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_bind(self, key_config_stub, config_stub, qtbot, no_bindings,
|
2017-09-21 22:28:51 +02:00
|
|
|
mode, command):
|
2017-07-02 22:10:28 +02:00
|
|
|
config_stub.val.bindings.default = no_bindings
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
|
|
|
|
|
|
|
with qtbot.wait_signal(config_stub.changed):
|
2017-10-05 11:09:07 +02:00
|
|
|
key_config_stub.bind('a', command, mode=mode)
|
2017-07-02 22:10:28 +02:00
|
|
|
|
|
|
|
assert config_stub.val.bindings.commands[mode]['a'] == command
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub.get_bindings_for(mode)['a'] == command
|
|
|
|
assert key_config_stub.get_command('a', mode) == command
|
2017-07-02 22:10:28 +02:00
|
|
|
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_bind_mode_changing(self, key_config_stub, config_stub,
|
|
|
|
no_bindings):
|
2017-09-21 22:28:51 +02:00
|
|
|
"""Make sure we can bind to a command which changes the mode.
|
|
|
|
|
|
|
|
https://github.com/qutebrowser/qutebrowser/issues/2989
|
|
|
|
"""
|
|
|
|
config_stub.val.bindings.default = no_bindings
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
2017-10-05 11:09:07 +02:00
|
|
|
key_config_stub.bind('a', 'set-cmd-text :nop ;; rl-beginning-of-line',
|
|
|
|
mode='normal')
|
2017-09-21 22:28:51 +02:00
|
|
|
|
2017-10-08 21:48:48 +02:00
|
|
|
def test_bind_default(self, key_config_stub, config_stub):
|
|
|
|
"""Bind a key to its default."""
|
|
|
|
default_cmd = 'message-info default'
|
|
|
|
bound_cmd = 'message-info bound'
|
|
|
|
config_stub.val.bindings.default = {'normal': {'a': default_cmd}}
|
|
|
|
config_stub.val.bindings.commands = {'normal': {'a': bound_cmd}}
|
|
|
|
assert key_config_stub.get_command('a', mode='normal') == bound_cmd
|
|
|
|
|
|
|
|
key_config_stub.bind_default('a', mode='normal')
|
|
|
|
|
|
|
|
assert key_config_stub.get_command('a', mode='normal') == default_cmd
|
|
|
|
|
|
|
|
def test_bind_default_unbound(self, key_config_stub, config_stub,
|
|
|
|
no_bindings):
|
|
|
|
"""Try binding a key to default which is not bound."""
|
|
|
|
config_stub.val.bindings.default = no_bindings
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
|
|
|
with pytest.raises(configexc.KeybindingError,
|
|
|
|
match="Can't find binding 'foobar' in normal mode"):
|
|
|
|
key_config_stub.bind_default('foobar', mode='normal')
|
|
|
|
|
2017-07-02 22:10:28 +02:00
|
|
|
@pytest.mark.parametrize('key, normalized', [
|
2017-07-03 09:09:50 +02:00
|
|
|
('a', 'a'), # default bindings
|
|
|
|
('b', 'b'), # custom bindings
|
2017-07-02 22:10:28 +02:00
|
|
|
('<Ctrl-X>', '<ctrl+x>')
|
2017-07-02 17:12:31 +02:00
|
|
|
])
|
2017-07-03 13:20:10 +02:00
|
|
|
@pytest.mark.parametrize('mode', ['normal', 'caret', 'prompt'])
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_unbind(self, key_config_stub, config_stub, qtbot,
|
|
|
|
key, normalized, mode):
|
2017-07-03 13:20:10 +02:00
|
|
|
default_bindings = {
|
2017-07-02 22:10:28 +02:00
|
|
|
'normal': {'a': 'nop', '<ctrl+x>': 'nop'},
|
|
|
|
'caret': {'a': 'nop', '<ctrl+x>': 'nop'},
|
2017-07-03 13:20:10 +02:00
|
|
|
# prompt: a mode which isn't in bindings.commands yet
|
|
|
|
'prompt': {'a': 'nop', 'b': 'nop', '<ctrl+x>': 'nop'},
|
2017-07-02 22:10:28 +02:00
|
|
|
}
|
2017-07-03 13:20:10 +02:00
|
|
|
old_default_bindings = copy.deepcopy(default_bindings)
|
|
|
|
config_stub.val.bindings.default = default_bindings
|
2017-07-03 09:09:50 +02:00
|
|
|
config_stub.val.bindings.commands = {
|
|
|
|
'normal': {'b': 'nop'},
|
|
|
|
'caret': {'b': 'nop'},
|
|
|
|
}
|
2017-07-02 17:12:31 +02:00
|
|
|
|
2017-07-02 22:10:28 +02:00
|
|
|
with qtbot.wait_signal(config_stub.changed):
|
2017-10-05 11:09:07 +02:00
|
|
|
key_config_stub.unbind(key, mode=mode)
|
2017-07-02 22:10:28 +02:00
|
|
|
|
2017-10-05 11:09:07 +02:00
|
|
|
assert key_config_stub.get_command(key, mode) is None
|
2017-07-02 22:10:28 +02:00
|
|
|
|
|
|
|
mode_bindings = config_stub.val.bindings.commands[mode]
|
2017-07-03 13:20:10 +02:00
|
|
|
if key == 'b' and mode != 'prompt':
|
2017-07-03 09:09:50 +02:00
|
|
|
# Custom binding
|
|
|
|
assert normalized not in mode_bindings
|
|
|
|
else:
|
2017-07-02 22:10:28 +02:00
|
|
|
default_bindings = config_stub.val.bindings.default
|
2017-07-03 13:20:10 +02:00
|
|
|
assert default_bindings[mode] == old_default_bindings[mode]
|
2017-07-02 22:10:28 +02:00
|
|
|
assert mode_bindings[normalized] is None
|
|
|
|
|
2017-10-05 11:09:07 +02:00
|
|
|
def test_unbind_unbound(self, key_config_stub, config_stub, no_bindings):
|
2017-07-02 22:10:28 +02:00
|
|
|
"""Try unbinding a key which is not bound."""
|
|
|
|
config_stub.val.bindings.default = no_bindings
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
|
|
|
with pytest.raises(configexc.KeybindingError,
|
|
|
|
match="Can't find binding 'foobar' in normal mode"):
|
2017-10-05 11:09:07 +02:00
|
|
|
key_config_stub.unbind('foobar', mode='normal')
|
2017-07-02 17:12:31 +02:00
|
|
|
|
2017-11-04 15:15:58 +01:00
|
|
|
def test_unbound_twice(self, key_config_stub, config_stub, no_bindings):
|
|
|
|
"""Try unbinding an already-unbound default key.
|
|
|
|
|
|
|
|
For custom-bound keys (in bindings.commands), it's okay to display an
|
|
|
|
error, as this isn't something you'd do in e.g a config.py anyways.
|
|
|
|
|
|
|
|
https://github.com/qutebrowser/qutebrowser/issues/3162
|
|
|
|
"""
|
|
|
|
config_stub.val.bindings.default = {'normal': {'a': 'nop'}}
|
|
|
|
config_stub.val.bindings.commands = no_bindings
|
|
|
|
|
|
|
|
key_config_stub.unbind('a')
|
|
|
|
assert key_config_stub.get_command('a', mode='normal') is None
|
|
|
|
key_config_stub.unbind('a')
|
|
|
|
assert key_config_stub.get_command('a', mode='normal') is None
|
|
|
|
|
2017-11-08 15:08:36 +01:00
|
|
|
def test_empty_command(self, key_config_stub):
|
|
|
|
"""Try binding a key to an empty command."""
|
|
|
|
message = "Can't add binding 'x' with empty command in normal mode"
|
|
|
|
with pytest.raises(configexc.KeybindingError, match=message):
|
|
|
|
key_config_stub.bind('x', ' ', mode='normal')
|
|
|
|
|
2017-07-02 17:12:31 +02:00
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
class TestConfig:
|
|
|
|
|
|
|
|
@pytest.fixture
|
2017-10-03 13:05:01 +02:00
|
|
|
def conf(self, config_tmpdir):
|
|
|
|
yaml_config = configfiles.YamlConfig()
|
2017-07-03 12:42:33 +02:00
|
|
|
return config.Config(yaml_config)
|
|
|
|
|
2017-10-03 14:12:29 +02:00
|
|
|
def test_init_save_manager(self, conf, fake_save_manager):
|
|
|
|
conf.init_save_manager(fake_save_manager)
|
|
|
|
fake_save_manager.add_saveable.assert_called_once_with(
|
|
|
|
'yaml-config', unittest.mock.ANY, unittest.mock.ANY)
|
|
|
|
|
2017-07-03 13:58:19 +02:00
|
|
|
def test_set_value(self, qtbot, conf, caplog):
|
|
|
|
opt = conf.get_opt('tabs.show')
|
2017-07-03 12:42:33 +02:00
|
|
|
with qtbot.wait_signal(conf.changed) as blocker:
|
2017-07-03 13:58:19 +02:00
|
|
|
conf._set_value(opt, 'never')
|
|
|
|
|
|
|
|
assert blocker.args == ['tabs.show']
|
2017-07-03 12:42:33 +02:00
|
|
|
assert len(caplog.records) == 1
|
2017-07-03 13:58:19 +02:00
|
|
|
expected_message = 'Config option changed: tabs.show = never'
|
|
|
|
assert caplog.records[0].message == expected_message
|
2017-07-03 12:42:33 +02:00
|
|
|
|
2017-09-19 07:08:07 +02:00
|
|
|
def test_set_value_no_backend(self, monkeypatch, conf):
|
|
|
|
"""Make sure setting values when the backend is still unknown works."""
|
|
|
|
monkeypatch.setattr(config.objects, 'backend', objects.NoBackend())
|
|
|
|
opt = conf.get_opt('tabs.show')
|
|
|
|
conf._set_value(opt, 'never')
|
|
|
|
assert conf._values['tabs.show'] == 'never'
|
|
|
|
|
2017-10-03 12:44:22 +02:00
|
|
|
@pytest.mark.parametrize('save_yaml', [True, False])
|
|
|
|
def test_unset(self, conf, qtbot, save_yaml):
|
|
|
|
name = 'tabs.show'
|
|
|
|
conf.set_obj(name, 'never', save_yaml=True)
|
|
|
|
assert conf.get(name) == 'never'
|
|
|
|
|
|
|
|
with qtbot.wait_signal(conf.changed):
|
|
|
|
conf.unset(name, save_yaml=save_yaml)
|
|
|
|
|
|
|
|
assert conf.get(name) == 'always'
|
|
|
|
if save_yaml:
|
|
|
|
assert name not in conf._yaml
|
|
|
|
else:
|
|
|
|
assert conf._yaml[name] == 'never'
|
|
|
|
|
|
|
|
def test_unset_never_set(self, conf, qtbot):
|
|
|
|
name = 'tabs.show'
|
|
|
|
assert conf.get(name) == 'always'
|
|
|
|
|
|
|
|
with qtbot.assert_not_emitted(conf.changed):
|
|
|
|
conf.unset(name)
|
|
|
|
|
|
|
|
assert conf.get(name) == 'always'
|
|
|
|
|
|
|
|
def test_unset_unknown(self, conf):
|
|
|
|
with pytest.raises(configexc.NoOptionError):
|
|
|
|
conf.unset('tabs')
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('save_yaml', [True, False])
|
|
|
|
def test_clear(self, conf, qtbot, save_yaml):
|
|
|
|
name1 = 'tabs.show'
|
|
|
|
name2 = 'content.plugins'
|
|
|
|
conf.set_obj(name1, 'never', save_yaml=True)
|
|
|
|
conf.set_obj(name2, True, save_yaml=True)
|
|
|
|
assert conf._values[name1] == 'never'
|
|
|
|
assert conf._values[name2] is True
|
|
|
|
|
|
|
|
with qtbot.waitSignals([conf.changed, conf.changed]) as blocker:
|
|
|
|
conf.clear(save_yaml=save_yaml)
|
|
|
|
|
2017-10-04 09:02:34 +02:00
|
|
|
options = {e.args[0] for e in blocker.all_signals_and_args}
|
|
|
|
assert options == {name1, name2}
|
2017-10-03 12:44:22 +02:00
|
|
|
|
|
|
|
if save_yaml:
|
|
|
|
assert name1 not in conf._yaml
|
|
|
|
assert name2 not in conf._yaml
|
|
|
|
else:
|
|
|
|
assert conf._yaml[name1] == 'never'
|
|
|
|
assert conf._yaml[name2] is True
|
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
def test_read_yaml(self, conf):
|
2017-09-19 17:26:03 +02:00
|
|
|
conf._yaml['content.plugins'] = True
|
2017-07-03 12:42:33 +02:00
|
|
|
conf.read_yaml()
|
|
|
|
assert conf._values['content.plugins'] is True
|
|
|
|
|
|
|
|
def test_get_opt_valid(self, conf):
|
|
|
|
assert conf.get_opt('tabs.show') == configdata.DATA['tabs.show']
|
|
|
|
|
|
|
|
def test_get_opt_invalid(self, conf):
|
|
|
|
with pytest.raises(configexc.NoOptionError):
|
|
|
|
conf.get_opt('tabs')
|
|
|
|
|
|
|
|
def test_get(self, conf):
|
|
|
|
"""Test conf.get() with a QColor (where get/get_obj is different)."""
|
|
|
|
assert conf.get('colors.completion.fg') == QColor('white')
|
|
|
|
|
2017-07-03 15:13:38 +02:00
|
|
|
@pytest.mark.parametrize('value', [{}, {'normal': {'a': 'nop'}}])
|
|
|
|
def test_get_bindings(self, config_stub, conf, value):
|
|
|
|
"""Test conf.get() with bindings which have missing keys."""
|
|
|
|
config_stub.val.aliases = {}
|
|
|
|
conf._values['bindings.commands'] = value
|
|
|
|
assert conf.get('bindings.commands')['prompt'] == {}
|
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
def test_get_mutable(self, conf):
|
|
|
|
"""Make sure we don't observe everything for mutations."""
|
|
|
|
conf.get('content.headers.custom')
|
|
|
|
assert not conf._mutables
|
|
|
|
|
|
|
|
def test_get_obj_simple(self, conf):
|
|
|
|
assert conf.get_obj('colors.completion.fg') == 'white'
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('option', ['content.headers.custom',
|
2017-07-03 15:13:38 +02:00
|
|
|
'keyhint.blacklist',
|
|
|
|
'bindings.commands'])
|
2017-07-03 12:42:33 +02:00
|
|
|
@pytest.mark.parametrize('mutable', [True, False])
|
|
|
|
@pytest.mark.parametrize('mutated', [True, False])
|
2017-07-03 15:13:38 +02:00
|
|
|
def test_get_obj_mutable(self, conf, config_stub, qtbot, caplog,
|
|
|
|
option, mutable, mutated):
|
2017-07-03 12:42:33 +02:00
|
|
|
"""Make sure mutables are handled correctly.
|
|
|
|
|
|
|
|
When we get a mutable object from the config, some invariants should be
|
|
|
|
true:
|
2017-07-04 15:33:58 +02:00
|
|
|
- The object we get from the config is always a copy, i.e. mutating
|
|
|
|
it doesn't change the internal value (or default) stored in the
|
|
|
|
config.
|
2017-07-03 12:42:33 +02:00
|
|
|
- If we mutate the object (mutated=True) and the config watches for
|
|
|
|
mutables (mutable=True), it should notice that the object changed.
|
|
|
|
- With mutable=False, we should always get the old object back.
|
|
|
|
|
|
|
|
We try this with a dict (content.headers.custom) and a list
|
|
|
|
(keyhint.blacklist).
|
|
|
|
"""
|
|
|
|
# Setting new value
|
|
|
|
obj = conf.get_obj(option, mutable=mutable)
|
|
|
|
with qtbot.assert_not_emitted(conf.changed):
|
|
|
|
if option == 'content.headers.custom':
|
|
|
|
old = {}
|
|
|
|
new = {}
|
|
|
|
assert obj == old
|
|
|
|
if mutated:
|
|
|
|
obj['X-Answer'] = '42'
|
|
|
|
if mutable:
|
|
|
|
new = {'X-Answer': '42'}
|
|
|
|
assert obj == new
|
2017-07-03 15:13:38 +02:00
|
|
|
elif option == 'keyhint.blacklist':
|
2017-07-03 12:42:33 +02:00
|
|
|
old = []
|
|
|
|
new = []
|
|
|
|
assert obj == old
|
|
|
|
if mutated:
|
|
|
|
obj.append('foo')
|
|
|
|
if mutable:
|
|
|
|
new = ['foo']
|
|
|
|
assert obj == new
|
2017-07-03 15:13:38 +02:00
|
|
|
else:
|
|
|
|
assert option == 'bindings.commands'
|
|
|
|
config_stub.val.aliases = {}
|
|
|
|
old = {}
|
|
|
|
new = {}
|
|
|
|
assert obj == old
|
|
|
|
if mutated:
|
|
|
|
obj['prompt'] = {}
|
|
|
|
obj['prompt']['foobar'] = 'nop'
|
|
|
|
if mutable:
|
|
|
|
new = {'prompt': {'foobar': 'nop'}}
|
|
|
|
assert obj == new
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
if mutable:
|
2017-09-19 23:26:02 +02:00
|
|
|
assert conf._mutables[option] == (old, new)
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
if mutable and mutated:
|
|
|
|
# Now let's update
|
|
|
|
with qtbot.wait_signal(conf.changed):
|
|
|
|
conf.update_mutables()
|
|
|
|
|
|
|
|
expected_log = '{} was mutated, updating'.format(option)
|
|
|
|
assert caplog.records[-2].message == expected_log
|
|
|
|
else:
|
|
|
|
with qtbot.assert_not_emitted(conf.changed):
|
|
|
|
conf.update_mutables()
|
|
|
|
|
|
|
|
assert not conf._mutables
|
|
|
|
assert conf.get_obj(option) == new
|
|
|
|
|
2017-10-03 14:12:29 +02:00
|
|
|
def test_get_mutable_twice(self, conf):
|
|
|
|
"""Get a mutable value twice."""
|
|
|
|
option = 'content.headers.custom'
|
|
|
|
obj = conf.get_obj(option, mutable=True)
|
|
|
|
obj['X-Foo'] = 'fooval'
|
|
|
|
obj2 = conf.get_obj(option, mutable=True)
|
|
|
|
obj2['X-Bar'] = 'barval'
|
|
|
|
|
|
|
|
conf.update_mutables()
|
|
|
|
|
|
|
|
expected = {'X-Foo': 'fooval', 'X-Bar': 'barval'}
|
|
|
|
assert conf.get_obj(option) == expected
|
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
def test_get_obj_unknown_mutable(self, conf):
|
|
|
|
"""Make sure we don't have unknown mutable types."""
|
|
|
|
conf._values['aliases'] = set() # This would never happen
|
|
|
|
with pytest.raises(AssertionError):
|
|
|
|
conf.get_obj('aliases')
|
|
|
|
|
|
|
|
def test_get_str(self, conf):
|
|
|
|
assert conf.get_str('content.plugins') == 'false'
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('save_yaml', [True, False])
|
|
|
|
@pytest.mark.parametrize('method, value', [
|
|
|
|
('set_obj', True),
|
|
|
|
('set_str', 'true'),
|
|
|
|
])
|
|
|
|
def test_set_valid(self, conf, qtbot, save_yaml, method, value):
|
|
|
|
option = 'content.plugins'
|
|
|
|
meth = getattr(conf, method)
|
|
|
|
with qtbot.wait_signal(conf.changed):
|
|
|
|
meth(option, value, save_yaml=save_yaml)
|
|
|
|
assert conf._values[option] is True
|
|
|
|
if save_yaml:
|
2017-09-19 17:26:03 +02:00
|
|
|
assert conf._yaml[option] is True
|
2017-07-03 12:42:33 +02:00
|
|
|
else:
|
2017-09-19 17:26:03 +02:00
|
|
|
assert option not in conf._yaml
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
@pytest.mark.parametrize('method', ['set_obj', 'set_str'])
|
|
|
|
def test_set_invalid(self, conf, qtbot, method):
|
|
|
|
meth = getattr(conf, method)
|
|
|
|
with pytest.raises(configexc.ValidationError):
|
|
|
|
with qtbot.assert_not_emitted(conf.changed):
|
|
|
|
meth('content.plugins', '42')
|
|
|
|
assert 'content.plugins' not in conf._values
|
|
|
|
|
2017-07-03 13:58:19 +02:00
|
|
|
@pytest.mark.parametrize('method', ['set_obj', 'set_str'])
|
|
|
|
def test_set_wrong_backend(self, conf, qtbot, monkeypatch, method):
|
|
|
|
monkeypatch.setattr(objects, 'backend', usertypes.Backend.QtWebEngine)
|
|
|
|
meth = getattr(conf, method)
|
|
|
|
with pytest.raises(configexc.BackendError):
|
|
|
|
with qtbot.assert_not_emitted(conf.changed):
|
|
|
|
meth('content.cookies.accept', 'all')
|
|
|
|
assert 'content.cookies.accept' not in conf._values
|
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
def test_dump_userconfig(self, conf):
|
|
|
|
conf.set_obj('content.plugins', True)
|
|
|
|
conf.set_obj('content.headers.custom', {'X-Foo': 'bar'})
|
2017-07-04 07:41:40 +02:00
|
|
|
lines = ['content.headers.custom = {"X-Foo": "bar"}',
|
|
|
|
'content.plugins = true']
|
2017-07-03 12:42:33 +02:00
|
|
|
assert conf.dump_userconfig().splitlines() == lines
|
|
|
|
|
|
|
|
def test_dump_userconfig_default(self, conf):
|
|
|
|
assert conf.dump_userconfig() == '<Default configuration>'
|
|
|
|
|
|
|
|
|
|
|
|
class TestContainer:
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def container(self, config_stub):
|
|
|
|
return config.ConfigContainer(config_stub)
|
|
|
|
|
|
|
|
def test_getattr_invalid_private(self, container):
|
|
|
|
"""Make sure an invalid _attribute doesn't try getting a container."""
|
|
|
|
with pytest.raises(AttributeError):
|
2017-07-04 15:33:58 +02:00
|
|
|
container._foo # pylint: disable=pointless-statement
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
def test_getattr_prefix(self, container):
|
|
|
|
new_container = container.tabs
|
|
|
|
assert new_container._prefix == 'tabs'
|
|
|
|
new_container = new_container.favicons
|
|
|
|
assert new_container._prefix == 'tabs.favicons'
|
|
|
|
|
2017-09-14 22:47:06 +02:00
|
|
|
@pytest.mark.parametrize('configapi, expected', [
|
|
|
|
(object(), 'rgb'),
|
|
|
|
(None, QColor.Rgb),
|
2017-09-14 17:31:14 +02:00
|
|
|
])
|
2017-09-14 22:47:06 +02:00
|
|
|
def test_getattr_option(self, container, configapi, expected):
|
|
|
|
container._configapi = configapi
|
2017-09-14 17:31:14 +02:00
|
|
|
# Use an option with a to_py() so we can check the conversion.
|
|
|
|
assert container.colors.downloads.system.fg == expected
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
def test_getattr_invalid(self, container):
|
|
|
|
with pytest.raises(configexc.NoOptionError) as excinfo:
|
2017-07-04 15:33:58 +02:00
|
|
|
container.tabs.foobar # pylint: disable=pointless-statement
|
2017-07-03 12:42:33 +02:00
|
|
|
assert excinfo.value.option == 'tabs.foobar'
|
|
|
|
|
|
|
|
def test_setattr_option(self, config_stub, container):
|
|
|
|
container.content.cookies.store = False
|
|
|
|
assert config_stub._values['content.cookies.store'] is False
|
|
|
|
|
2017-09-15 11:31:55 +02:00
|
|
|
def test_confapi_errors(self, container):
|
|
|
|
configapi = types.SimpleNamespace(errors=[])
|
|
|
|
container._configapi = configapi
|
|
|
|
container.tabs.foobar # pylint: disable=pointless-statement
|
|
|
|
|
|
|
|
assert len(configapi.errors) == 1
|
|
|
|
error = configapi.errors[0]
|
|
|
|
assert error.text == "While getting 'tabs.foobar'"
|
|
|
|
assert str(error.exception) == "No option 'tabs.foobar'"
|
|
|
|
|
2017-07-03 12:42:33 +02:00
|
|
|
|
|
|
|
class StyleObj(QObject):
|
|
|
|
|
|
|
|
def __init__(self, stylesheet=None, parent=None):
|
|
|
|
super().__init__(parent)
|
|
|
|
if stylesheet is not None:
|
2017-11-24 16:25:01 +01:00
|
|
|
self.STYLESHEET = stylesheet # noqa: N801,N806 pylint: disable=invalid-name
|
2017-07-03 12:42:33 +02:00
|
|
|
self.rendered_stylesheet = None
|
|
|
|
|
|
|
|
def setStyleSheet(self, stylesheet):
|
|
|
|
self.rendered_stylesheet = stylesheet
|
|
|
|
|
|
|
|
|
|
|
|
def test_get_stylesheet(config_stub):
|
|
|
|
config_stub.val.colors.hints.fg = 'magenta'
|
|
|
|
observer = config.StyleSheetObserver(
|
2017-11-11 04:39:10 +01:00
|
|
|
StyleObj(), stylesheet="{{ conf.colors.hints.fg }}", update=False)
|
2017-07-03 12:42:33 +02:00
|
|
|
assert observer._get_stylesheet() == 'magenta'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('delete', [True, False])
|
|
|
|
@pytest.mark.parametrize('stylesheet_param', [True, False])
|
|
|
|
@pytest.mark.parametrize('update', [True, False])
|
|
|
|
def test_set_register_stylesheet(delete, stylesheet_param, update, qtbot,
|
|
|
|
config_stub, caplog):
|
|
|
|
config_stub.val.colors.hints.fg = 'magenta'
|
|
|
|
stylesheet = "{{ conf.colors.hints.fg }}"
|
|
|
|
|
|
|
|
with caplog.at_level(9): # VDEBUG
|
|
|
|
if stylesheet_param:
|
|
|
|
obj = StyleObj()
|
|
|
|
config.set_register_stylesheet(obj, stylesheet=stylesheet,
|
|
|
|
update=update)
|
|
|
|
else:
|
|
|
|
obj = StyleObj(stylesheet)
|
|
|
|
config.set_register_stylesheet(obj, update=update)
|
|
|
|
|
|
|
|
assert caplog.records[-1].message == 'stylesheet for StyleObj: magenta'
|
|
|
|
|
|
|
|
assert obj.rendered_stylesheet == 'magenta'
|
|
|
|
|
|
|
|
if delete:
|
|
|
|
with qtbot.waitSignal(obj.destroyed):
|
|
|
|
obj.deleteLater()
|
|
|
|
|
|
|
|
config_stub.val.colors.hints.fg = 'yellow'
|
|
|
|
|
|
|
|
if delete or not update:
|
|
|
|
expected = 'magenta'
|
|
|
|
else:
|
|
|
|
expected = 'yellow'
|
|
|
|
|
|
|
|
assert obj.rendered_stylesheet == expected
|