diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-06-05 16:20:58 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-06-05 16:20:58 +0000 |
commit | 5bb0bb4be543fd5eca41673696a62ed80d493591 (patch) | |
tree | ad2c464f140e86c7f178a6276d7ea4a93e3e6c92 /tests/test_config.py | |
parent | Adding upstream version 7.2.6. (diff) | |
download | sphinx-5bb0bb4be543fd5eca41673696a62ed80d493591.tar.xz sphinx-5bb0bb4be543fd5eca41673696a62ed80d493591.zip |
Adding upstream version 7.3.7.upstream/7.3.7
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'tests/test_config.py')
-rw-r--r-- | tests/test_config.py | 517 |
1 files changed, 0 insertions, 517 deletions
diff --git a/tests/test_config.py b/tests/test_config.py deleted file mode 100644 index 0be0a58..0000000 --- a/tests/test_config.py +++ /dev/null @@ -1,517 +0,0 @@ -"""Test the sphinx.config.Config class.""" - -import time -from pathlib import Path -from unittest import mock - -import pytest - -import sphinx -from sphinx.config import ENUM, Config, check_confval_types -from sphinx.errors import ConfigError, ExtensionError, VersionRequirementError - - -@pytest.mark.sphinx(testroot='config', confoverrides={ - 'root_doc': 'root', - 'nonexisting_value': 'True', - 'latex_elements.maketitle': 'blah blah blah', - 'modindex_common_prefix': 'path1,path2'}) -def test_core_config(app, status, warning): - cfg = app.config - - # simple values - assert 'project' in cfg.__dict__ - assert cfg.project == 'Sphinx <Tests>' - assert cfg.templates_path == ['_templates'] - - # overrides - assert cfg.root_doc == 'root' - assert cfg.latex_elements['maketitle'] == 'blah blah blah' - assert cfg.modindex_common_prefix == ['path1', 'path2'] - - # simple default values - assert 'locale_dirs' not in cfg.__dict__ - assert cfg.locale_dirs == ['locales'] - assert cfg.trim_footnote_reference_space is False - - # complex default values - assert 'html_title' not in cfg.__dict__ - assert cfg.html_title == 'Sphinx <Tests> 0.6alpha1 documentation' - - # complex default values mustn't raise - for valuename in cfg.config_values: - getattr(cfg, valuename) - - # "contains" gives True both for set and unset values - assert 'project' in cfg - assert 'html_title' in cfg - assert 'nonexisting_value' not in cfg - - # invalid values - with pytest.raises(AttributeError): - _ = cfg._value - with pytest.raises(AttributeError): - _ = cfg.nonexisting_value - - # non-value attributes are deleted from the namespace - with pytest.raises(AttributeError): - _ = cfg.sys - - # setting attributes - cfg.project = 'Foo' - assert cfg.project == 'Foo' - - # alternative access via item interface - cfg['project'] = 'Sphinx Tests' - assert cfg['project'] == cfg.project == 'Sphinx Tests' - - -def test_config_not_found(tmp_path): - with pytest.raises(ConfigError): - Config.read(tmp_path) - - -def test_extension_values(): - config = Config() - - # check standard settings - assert config.root_doc == 'index' - - # can't override it by add_config_value() - with pytest.raises(ExtensionError) as excinfo: - config.add('root_doc', 'index', 'env', None) - assert 'already present' in str(excinfo.value) - - # add a new config value - config.add('value_from_ext', [], 'env', None) - assert config.value_from_ext == [] - - # can't override it by add_config_value() - with pytest.raises(ExtensionError) as excinfo: - config.add('value_from_ext', [], 'env', None) - assert 'already present' in str(excinfo.value) - - -def test_overrides(): - config = Config({'value1': '1', 'value2': 2, 'value6': {'default': 6}}, - {'value2': 999, 'value3': '999', 'value5.attr1': 999, 'value6.attr1': 999, - 'value7': 'abc,def,ghi', 'value8': 'abc,def,ghi'}) - config.add('value1', None, 'env', ()) - config.add('value2', None, 'env', ()) - config.add('value3', 0, 'env', ()) - config.add('value4', 0, 'env', ()) - config.add('value5', {'default': 0}, 'env', ()) - config.add('value6', {'default': 0}, 'env', ()) - config.add('value7', None, 'env', ()) - config.add('value8', [], 'env', ()) - config.init_values() - - assert config.value1 == '1' - assert config.value2 == 999 - assert config.value3 == 999 - assert config.value4 == 0 - assert config.value5 == {'attr1': 999} - assert config.value6 == {'default': 6, 'attr1': 999} - assert config.value7 == 'abc,def,ghi' - assert config.value8 == ['abc', 'def', 'ghi'] - - -def test_overrides_boolean(): - config = Config({}, {'value1': '1', - 'value2': '0', - 'value3': '0'}) - config.add('value1', None, 'env', [bool]) - config.add('value2', None, 'env', [bool]) - config.add('value3', True, 'env', ()) - config.init_values() - - assert config.value1 is True - assert config.value2 is False - assert config.value3 is False - - -@mock.patch("sphinx.config.logger") -def test_errors_warnings(logger, tmp_path): - # test the error for syntax errors in the config file - (tmp_path / 'conf.py').write_text('project = \n', encoding='ascii') - with pytest.raises(ConfigError) as excinfo: - Config.read(tmp_path, {}, None) - assert 'conf.py' in str(excinfo.value) - - # test the automatic conversion of 2.x only code in configs - (tmp_path / 'conf.py').write_text('project = u"Jägermeister"\n', encoding='utf8') - cfg = Config.read(tmp_path, {}, None) - cfg.init_values() - assert cfg.project == 'Jägermeister' - assert logger.called is False - - -def test_errors_if_setup_is_not_callable(tmp_path, make_app): - # test the error to call setup() in the config file - (tmp_path / 'conf.py').write_text('setup = 1', encoding='utf8') - with pytest.raises(ConfigError) as excinfo: - make_app(srcdir=tmp_path) - assert 'callable' in str(excinfo.value) - - -@pytest.fixture() -def make_app_with_empty_project(make_app, tmp_path): - (tmp_path / 'conf.py').write_text('', encoding='utf8') - - def _make_app(*args, **kw): - kw.setdefault('srcdir', Path(tmp_path)) - return make_app(*args, **kw) - return _make_app - - -@mock.patch.object(sphinx, '__display_version__', '1.6.4') -def test_needs_sphinx(make_app_with_empty_project): - make_app = make_app_with_empty_project - # micro version - make_app(confoverrides={'needs_sphinx': '1.6.3'}) # OK: less - make_app(confoverrides={'needs_sphinx': '1.6.4'}) # OK: equals - with pytest.raises(VersionRequirementError): - make_app(confoverrides={'needs_sphinx': '1.6.5'}) # NG: greater - - # minor version - make_app(confoverrides={'needs_sphinx': '1.5'}) # OK: less - make_app(confoverrides={'needs_sphinx': '1.6'}) # OK: equals - with pytest.raises(VersionRequirementError): - make_app(confoverrides={'needs_sphinx': '1.7'}) # NG: greater - - # major version - make_app(confoverrides={'needs_sphinx': '0'}) # OK: less - make_app(confoverrides={'needs_sphinx': '1'}) # OK: equals - with pytest.raises(VersionRequirementError): - make_app(confoverrides={'needs_sphinx': '2'}) # NG: greater - - -@mock.patch("sphinx.config.logger") -def test_config_eol(logger, tmp_path): - # test config file's eol patterns: LF, CRLF - configfile = tmp_path / 'conf.py' - for eol in (b'\n', b'\r\n'): - configfile.write_bytes(b'project = "spam"' + eol) - cfg = Config.read(tmp_path, {}, None) - cfg.init_values() - assert cfg.project == 'spam' - assert logger.called is False - - -@pytest.mark.sphinx(confoverrides={'root_doc': 123, - 'language': 'foo', - 'primary_domain': None}) -def test_builtin_conf(app, status, warning): - warnings = warning.getvalue() - assert 'root_doc' in warnings, ( - 'override on builtin "root_doc" should raise a type warning') - assert 'language' not in warnings, ( - 'explicitly permitted override on builtin "language" should NOT raise ' - 'a type warning') - assert 'primary_domain' not in warnings, ( - 'override to None on builtin "primary_domain" should NOT raise a type ' - 'warning') - - -# example classes for type checking -class A: - pass - - -class B(A): - pass - - -class C(A): - pass - - -# name, default, annotation, actual, warned -TYPECHECK_WARNINGS = [ - ('value1', 'string', None, 123, True), # wrong type - ('value2', lambda _: [], None, 123, True), # lambda with wrong type - ('value3', lambda _: [], None, [], False), # lambda with correct type - ('value4', 100, None, True, True), # child type - ('value5', False, None, True, False), # parent type - ('value6', [], None, (), True), # other sequence type - ('value7', 'string', [list], ['foo'], False), # explicit type annotation - ('value8', B(), None, C(), False), # sibling type - ('value9', None, None, 'foo', False), # no default or no annotations - ('value10', None, None, 123, False), # no default or no annotations - ('value11', None, [str], 'bar', False), # str - ('value12', 'string', None, 'bar', False), # str -] - - -@mock.patch("sphinx.config.logger") -@pytest.mark.parametrize(('name', 'default', 'annotation', 'actual', 'warned'), TYPECHECK_WARNINGS) -def test_check_types(logger, name, default, annotation, actual, warned): - config = Config({name: actual}) - config.add(name, default, 'env', annotation or ()) - config.init_values() - check_confval_types(None, config) - assert logger.warning.called == warned - - -TYPECHECK_WARNING_MESSAGES = [ - ('value1', 'string', [str], ['foo', 'bar'], - "The config value `value1' has type `list'; expected `str'."), - ('value1', 'string', [str, int], ['foo', 'bar'], - "The config value `value1' has type `list'; expected `str' or `int'."), - ('value1', 'string', [str, int, tuple], ['foo', 'bar'], - "The config value `value1' has type `list'; expected `str', `int', or `tuple'."), -] - - -@mock.patch("sphinx.config.logger") -@pytest.mark.parametrize(('name', 'default', 'annotation', 'actual', 'message'), TYPECHECK_WARNING_MESSAGES) -def test_conf_warning_message(logger, name, default, annotation, actual, message): - config = Config({name: actual}) - config.add(name, default, False, annotation or ()) - config.init_values() - check_confval_types(None, config) - assert logger.warning.called - assert logger.warning.call_args[0][0] == message - - -@mock.patch("sphinx.config.logger") -def test_check_enum(logger): - config = Config() - config.add('value', 'default', False, ENUM('default', 'one', 'two')) - config.init_values() - check_confval_types(None, config) - logger.warning.assert_not_called() # not warned - - -@mock.patch("sphinx.config.logger") -def test_check_enum_failed(logger): - config = Config({'value': 'invalid'}) - config.add('value', 'default', False, ENUM('default', 'one', 'two')) - config.init_values() - check_confval_types(None, config) - assert logger.warning.called - - -@mock.patch("sphinx.config.logger") -def test_check_enum_for_list(logger): - config = Config({'value': ['one', 'two']}) - config.add('value', 'default', False, ENUM('default', 'one', 'two')) - config.init_values() - check_confval_types(None, config) - logger.warning.assert_not_called() # not warned - - -@mock.patch("sphinx.config.logger") -def test_check_enum_for_list_failed(logger): - config = Config({'value': ['one', 'two', 'invalid']}) - config.add('value', 'default', False, ENUM('default', 'one', 'two')) - config.init_values() - check_confval_types(None, config) - assert logger.warning.called - - -nitpick_warnings = [ - "WARNING: py:const reference target not found: prefix.anything.postfix", - "WARNING: py:class reference target not found: prefix.anything", - "WARNING: py:class reference target not found: anything.postfix", - "WARNING: js:class reference target not found: prefix.anything.postfix", -] - - -@pytest.mark.sphinx(testroot='nitpicky-warnings') -def test_nitpick_base(app, status, warning): - app.builder.build_all() - - warning = warning.getvalue().strip().split('\n') - assert len(warning) == len(nitpick_warnings) - for actual, expected in zip(warning, nitpick_warnings): - assert expected in actual - - -@pytest.mark.sphinx(testroot='nitpicky-warnings', confoverrides={ - 'nitpick_ignore': { - ('py:const', 'prefix.anything.postfix'), - ('py:class', 'prefix.anything'), - ('py:class', 'anything.postfix'), - ('js:class', 'prefix.anything.postfix'), - }, -}) -def test_nitpick_ignore(app, status, warning): - app.builder.build_all() - assert not len(warning.getvalue().strip()) - - -@pytest.mark.sphinx(testroot='nitpicky-warnings', confoverrides={ - 'nitpick_ignore_regex': [ - (r'py:.*', r'.*postfix'), - (r'.*:class', r'prefix.*'), - ], -}) -def test_nitpick_ignore_regex1(app, status, warning): - app.builder.build_all() - assert not len(warning.getvalue().strip()) - - -@pytest.mark.sphinx(testroot='nitpicky-warnings', confoverrides={ - 'nitpick_ignore_regex': [ - (r'py:.*', r'prefix.*'), - (r'.*:class', r'.*postfix'), - ], -}) -def test_nitpick_ignore_regex2(app, status, warning): - app.builder.build_all() - assert not len(warning.getvalue().strip()) - - -@pytest.mark.sphinx(testroot='nitpicky-warnings', confoverrides={ - 'nitpick_ignore_regex': [ - # None of these should match - (r'py:', r'.*'), - (r':class', r'.*'), - (r'', r'.*'), - (r'.*', r'anything'), - (r'.*', r'prefix'), - (r'.*', r'postfix'), - (r'.*', r''), - ], -}) -def test_nitpick_ignore_regex_fullmatch(app, status, warning): - app.builder.build_all() - - warning = warning.getvalue().strip().split('\n') - assert len(warning) == len(nitpick_warnings) - for actual, expected in zip(warning, nitpick_warnings): - assert expected in actual - - -def test_conf_py_language_none(tmp_path): - """Regression test for #10474.""" - - # Given a conf.py file with language = None - (tmp_path / 'conf.py').write_text("language = None", encoding='utf-8') - - # When we load conf.py into a Config object - cfg = Config.read(tmp_path, {}, None) - cfg.init_values() - - # Then the language is coerced to English - assert cfg.language == "en" - - -@mock.patch("sphinx.config.logger") -def test_conf_py_language_none_warning(logger, tmp_path): - """Regression test for #10474.""" - - # Given a conf.py file with language = None - (tmp_path / 'conf.py').write_text("language = None", encoding='utf-8') - - # When we load conf.py into a Config object - Config.read(tmp_path, {}, None) - - # Then a warning is raised - assert logger.warning.called - assert logger.warning.call_args[0][0] == ( - "Invalid configuration value found: 'language = None'. " - "Update your configuration to a valid language code. " - "Falling back to 'en' (English).") - - -def test_conf_py_no_language(tmp_path): - """Regression test for #10474.""" - - # Given a conf.py file with no language attribute - (tmp_path / 'conf.py').write_text("", encoding='utf-8') - - # When we load conf.py into a Config object - cfg = Config.read(tmp_path, {}, None) - cfg.init_values() - - # Then the language is coerced to English - assert cfg.language == "en" - - -def test_conf_py_nitpick_ignore_list(tmp_path): - """Regression test for #11355.""" - - # Given a conf.py file with no language attribute - (tmp_path / 'conf.py').write_text("", encoding='utf-8') - - # When we load conf.py into a Config object - cfg = Config.read(tmp_path, {}, None) - cfg.init_values() - - # Then the default nitpick_ignore[_regex] is an empty list - assert cfg.nitpick_ignore == [] - assert cfg.nitpick_ignore_regex == [] - - -@pytest.fixture(params=[ - # test with SOURCE_DATE_EPOCH unset: no modification - None, - # test with SOURCE_DATE_EPOCH set: copyright year should be updated - 1293840000, - 1293839999, -]) -def source_date_year(request, monkeypatch): - sde = request.param - with monkeypatch.context() as m: - if sde: - m.setenv('SOURCE_DATE_EPOCH', str(sde)) - yield time.gmtime(sde).tm_year - else: - m.delenv('SOURCE_DATE_EPOCH', raising=False) - yield None - - -@pytest.mark.sphinx(testroot='copyright-multiline') -def test_multi_line_copyright(source_date_year, app, monkeypatch): - app.builder.build_all() - - content = (app.outdir / 'index.html').read_text(encoding='utf-8') - - if source_date_year is None: - # check the copyright footer line by line (empty lines ignored) - assert ' © Copyright 2006.<br/>\n' in content - assert ' © Copyright 2006-2009, Alice.<br/>\n' in content - assert ' © Copyright 2010-2013, Bob.<br/>\n' in content - assert ' © Copyright 2014-2017, Charlie.<br/>\n' in content - assert ' © Copyright 2018-2021, David.<br/>\n' in content - assert ' © Copyright 2022-2025, Eve.' in content - - # check the raw copyright footer block (empty lines included) - assert ( - ' © Copyright 2006.<br/>\n' - ' \n' - ' © Copyright 2006-2009, Alice.<br/>\n' - ' \n' - ' © Copyright 2010-2013, Bob.<br/>\n' - ' \n' - ' © Copyright 2014-2017, Charlie.<br/>\n' - ' \n' - ' © Copyright 2018-2021, David.<br/>\n' - ' \n' - ' © Copyright 2022-2025, Eve.' - ) in content - else: - # check the copyright footer line by line (empty lines ignored) - assert f' © Copyright {source_date_year}.<br/>\n' in content - assert f' © Copyright 2006-{source_date_year}, Alice.<br/>\n' in content - assert f' © Copyright 2010-{source_date_year}, Bob.<br/>\n' in content - assert f' © Copyright 2014-{source_date_year}, Charlie.<br/>\n' in content - assert f' © Copyright 2018-{source_date_year}, David.<br/>\n' in content - assert f' © Copyright 2022-{source_date_year}, Eve.' in content - - # check the raw copyright footer block (empty lines included) - assert ( - f' © Copyright {source_date_year}.<br/>\n' - f' \n' - f' © Copyright 2006-{source_date_year}, Alice.<br/>\n' - f' \n' - f' © Copyright 2010-{source_date_year}, Bob.<br/>\n' - f' \n' - f' © Copyright 2014-{source_date_year}, Charlie.<br/>\n' - f' \n' - f' © Copyright 2018-{source_date_year}, David.<br/>\n' - f' \n' - f' © Copyright 2022-{source_date_year}, Eve.' - ) in content |