# SPDX-FileCopyrightText: Florian Bruhin (The Compiler) # # SPDX-License-Identifier: GPL-3.0-or-later """Tests for qutebrowser.commands.argparser.""" import inspect import enum import pytest from qutebrowser.qt.core import QUrl from qutebrowser.commands import argparser, cmdexc class Enum(enum.Enum): foo = enum.auto() foo_bar = enum.auto() class TestArgumentParser: @pytest.fixture def parser(self): return argparser.ArgumentParser('foo') def test_name(self, parser): assert parser.name == 'foo' def test_exit(self, parser): parser.add_argument('--help', action='help') with pytest.raises(argparser.ArgumentParserExit) as excinfo: parser.parse_args(['--help']) assert excinfo.value.status == 0 def test_error(self, parser): with pytest.raises(argparser.ArgumentParserError, match="Unrecognized arguments: --foo"): parser.parse_args(['--foo']) def test_help(self, parser, tabbed_browser_stubs): parser.add_argument('--help', action=argparser.HelpAction, nargs=0) with pytest.raises(argparser.ArgumentParserExit): parser.parse_args(['--help']) expected_url = QUrl('qute://help/commands.html#foo') assert tabbed_browser_stubs[1].loaded_url == expected_url @pytest.mark.parametrize('types, value, expected', [ ([Enum], 'foo', Enum.foo), ([Enum], 'foo-bar', Enum.foo_bar), ([int], '2', 2), ([int, str], 'foo', 'foo'), ]) @pytest.mark.parametrize('multi', [True, False]) def test_type_conv_valid(types, value, expected, multi): param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) if multi: assert argparser.multitype_conv(param, types, value) == expected elif len(types) == 1: assert argparser.type_conv(param, types[0], value) == expected @pytest.mark.parametrize('typ, value', [ (Enum, 'blubb'), (Enum, 'foo_bar'), (int, '2.5'), (int, 'foo'), ]) @pytest.mark.parametrize('multi', [True, False]) def test_type_conv_invalid(typ, value, multi): param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) if multi: msg = 'foo: Invalid value {}'.format(value) elif typ is Enum: msg = ('foo: Invalid value {} - expected one of: foo, ' 'foo-bar'.format(value)) else: msg = 'foo: Invalid {} value {}'.format(typ.__name__, value) with pytest.raises(cmdexc.ArgumentTypeError, match=msg): if multi: argparser.multitype_conv(param, [typ], value) else: argparser.type_conv(param, typ, value) def test_multitype_conv_invalid_type(): """Test using an invalid type with a multitype converter.""" param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) with pytest.raises(ValueError, match="foo: Unknown type None!"): argparser.multitype_conv(param, [None], '') @pytest.mark.parametrize('value, typ', [(None, None), (42, int)]) def test_conv_default_param(value, typ): """The default value should always be a valid choice.""" def func(foo=value): pass param = inspect.signature(func).parameters['foo'] assert argparser.type_conv(param, typ, value, str_choices=['val']) == value def test_conv_str_type(): """Using a str literal as type used to mean exactly that's a valid value. This got replaced by @cmdutils.argument(..., choices=...), so we make sure no string annotations are there anymore. """ param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) with pytest.raises(TypeError, match='foo: Legacy string type!'): argparser.type_conv(param, 'val', None) def test_conv_str_choices_valid(): """Calling str type with str_choices and valid value.""" param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) converted = argparser.type_conv(param, str, 'val1', str_choices=['val1', 'val2']) assert converted == 'val1' def test_conv_str_choices_invalid(): """Calling str type with str_choices and invalid value.""" param = inspect.Parameter('foo', inspect.Parameter.POSITIONAL_ONLY) with pytest.raises(cmdexc.ArgumentTypeError, match='foo: Invalid value ' 'val3 - expected one of: val1, val2'): argparser.type_conv(param, str, 'val3', str_choices=['val1', 'val2'])