2022-02-19 20:57:02 -06:00
|
|
|
"""Test the sphinx.config.Config class."""
|
2024-08-11 08:58:56 -05:00
|
|
|
|
2024-03-25 05:19:02 -05:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2024-01-03 12:03:07 -06:00
|
|
|
import pickle
|
2024-03-25 05:19:02 -05:00
|
|
|
from collections import Counter
|
2023-07-27 18:39:12 -05:00
|
|
|
from pathlib import Path
|
2024-04-16 20:24:56 -05:00
|
|
|
from typing import TYPE_CHECKING, Any
|
2019-03-07 08:35:36 -06:00
|
|
|
from unittest import mock
|
|
|
|
|
2018-02-19 07:39:14 -06:00
|
|
|
import pytest
|
2008-06-05 03:58:43 -05:00
|
|
|
|
2016-01-20 22:51:17 -06:00
|
|
|
import sphinx
|
2023-12-29 14:29:40 -06:00
|
|
|
from sphinx.builders.gettext import _gettext_compact_validator
|
2024-02-24 09:21:29 -06:00
|
|
|
from sphinx.config import (
|
|
|
|
ENUM,
|
|
|
|
Config,
|
|
|
|
_Opt,
|
|
|
|
check_confval_types,
|
2024-03-25 05:19:02 -05:00
|
|
|
is_serializable,
|
2024-02-24 09:21:29 -06:00
|
|
|
)
|
2024-01-03 12:03:07 -06:00
|
|
|
from sphinx.deprecation import RemovedInSphinx90Warning
|
2020-11-11 05:00:27 -06:00
|
|
|
from sphinx.errors import ConfigError, ExtensionError, VersionRequirementError
|
2008-06-05 03:58:43 -05:00
|
|
|
|
2024-03-25 05:19:02 -05:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from collections.abc import Iterable
|
2024-07-22 19:21:29 -05:00
|
|
|
from typing import TypeAlias
|
2024-03-25 05:19:02 -05:00
|
|
|
|
2024-07-22 19:21:29 -05:00
|
|
|
CircularList: TypeAlias = list[int | 'CircularList']
|
|
|
|
CircularDict: TypeAlias = dict[str, int | 'CircularDict']
|
2024-03-25 05:19:02 -05:00
|
|
|
|
|
|
|
|
|
|
|
def check_is_serializable(subject: object, *, circular: bool) -> None:
|
|
|
|
assert is_serializable(subject)
|
|
|
|
|
|
|
|
if circular:
|
2024-08-11 08:58:56 -05:00
|
|
|
|
2024-03-25 05:19:02 -05:00
|
|
|
class UselessGuard(frozenset[int]):
|
|
|
|
def __or__(self, other: object, /) -> UselessGuard:
|
|
|
|
# do nothing
|
|
|
|
return self
|
|
|
|
|
|
|
|
def union(self, *args: Iterable[object]) -> UselessGuard:
|
|
|
|
# do nothing
|
|
|
|
return self
|
|
|
|
|
|
|
|
# check that without recursive guards, a recursion error occurs
|
|
|
|
with pytest.raises(RecursionError):
|
2024-04-17 15:24:11 -05:00
|
|
|
assert is_serializable(subject, _seen=UselessGuard())
|
2024-03-25 05:19:02 -05:00
|
|
|
|
|
|
|
|
|
|
|
def test_is_serializable() -> None:
|
|
|
|
subject = [1, [2, {3, 'a'}], {'x': {'y': frozenset((4, 5))}}]
|
|
|
|
check_is_serializable(subject, circular=False)
|
|
|
|
|
|
|
|
a, b = [1], [2] # type: (CircularList, CircularList)
|
|
|
|
a.append(b)
|
|
|
|
b.append(a)
|
|
|
|
check_is_serializable(a, circular=True)
|
|
|
|
check_is_serializable(b, circular=True)
|
|
|
|
|
|
|
|
x: CircularDict = {'a': 1, 'b': {'c': 1}}
|
|
|
|
x['b'] = x
|
|
|
|
check_is_serializable(x, circular=True)
|
|
|
|
|
2008-06-05 03:58:43 -05:00
|
|
|
|
2024-01-03 12:03:07 -06:00
|
|
|
def test_config_opt_deprecated(recwarn):
|
|
|
|
opt = _Opt('default', '', ())
|
|
|
|
|
|
|
|
with pytest.warns(RemovedInSphinx90Warning):
|
|
|
|
default, rebuild, valid_types = opt
|
|
|
|
|
|
|
|
with pytest.warns(RemovedInSphinx90Warning):
|
|
|
|
_ = opt[0]
|
|
|
|
|
|
|
|
with pytest.warns(RemovedInSphinx90Warning):
|
|
|
|
_ = list(opt)
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
2024-08-11 08:58:56 -05:00
|
|
|
testroot='config',
|
|
|
|
confoverrides={
|
|
|
|
'root_doc': 'root',
|
|
|
|
'nonexisting_value': 'True',
|
|
|
|
'latex_elements.maketitle': 'blah blah blah',
|
|
|
|
'modindex_common_prefix': 'path1,path2',
|
|
|
|
},
|
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_core_config(app):
|
Merged revisions 65283,65303,65316-65317,65372-65375,65377,65380,65483-65485,65494 via svnmerge from
svn+ssh://pythondev@svn.python.org/doctools/branches/0.4.x
........
r65283 | georg.brandl | 2008-07-29 10:07:26 +0000 (Tue, 29 Jul 2008) | 2 lines
Update ez_setup.py.
........
r65303 | benjamin.peterson | 2008-07-30 12:35:34 +0000 (Wed, 30 Jul 2008) | 1 line
add a with_testapp decorator for test functions that passes the TestApp instance in a cleans up after it
........
r65316 | benjamin.peterson | 2008-07-30 23:12:07 +0000 (Wed, 30 Jul 2008) | 1 line
make the app for test_markup global to the module
........
r65317 | benjamin.peterson | 2008-07-30 23:31:29 +0000 (Wed, 30 Jul 2008) | 1 line
make TestApp.cleanup more aggressive
........
r65372 | georg.brandl | 2008-08-01 19:11:22 +0000 (Fri, 01 Aug 2008) | 2 lines
Add more tests, fix a few bugs in image handling.
........
r65373 | georg.brandl | 2008-08-01 19:28:33 +0000 (Fri, 01 Aug 2008) | 2 lines
Fix oversight.
........
r65374 | benjamin.peterson | 2008-08-01 19:36:32 +0000 (Fri, 01 Aug 2008) | 1 line
fix one broken test
........
r65375 | georg.brandl | 2008-08-01 19:41:11 +0000 (Fri, 01 Aug 2008) | 2 lines
Fix the handling of non-ASCII input in quickstart.
........
r65377 | georg.brandl | 2008-08-01 19:48:24 +0000 (Fri, 01 Aug 2008) | 2 lines
Allow REs in markup checks.
........
r65380 | georg.brandl | 2008-08-01 20:31:18 +0000 (Fri, 01 Aug 2008) | 2 lines
Don't rely on mtimes being different for changed files.
........
r65483 | georg.brandl | 2008-08-04 09:01:40 +0000 (Mon, 04 Aug 2008) | 4 lines
Add an "encoding" option to literalinclude.
Add tests for include directives.
........
r65484 | georg.brandl | 2008-08-04 09:11:17 +0000 (Mon, 04 Aug 2008) | 2 lines
Add changelog entry.
........
r65485 | georg.brandl | 2008-08-04 09:21:58 +0000 (Mon, 04 Aug 2008) | 2 lines
Fix markup.
........
r65494 | georg.brandl | 2008-08-04 16:34:59 +0000 (Mon, 04 Aug 2008) | 2 lines
Correctly use HTML file suffix in templates.
........
2008-08-04 12:01:15 -05:00
|
|
|
cfg = app.config
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# simple values
|
|
|
|
assert 'project' in cfg.__dict__
|
2008-11-23 08:22:09 -06:00
|
|
|
assert cfg.project == 'Sphinx <Tests>'
|
2008-06-05 03:58:43 -05:00
|
|
|
assert cfg.templates_path == ['_templates']
|
|
|
|
|
|
|
|
# overrides
|
2020-11-24 09:50:36 -06:00
|
|
|
assert cfg.root_doc == 'root'
|
2018-07-29 07:57:34 -05:00
|
|
|
assert cfg.latex_elements['maketitle'] == 'blah blah blah'
|
2014-01-10 15:10:29 -06:00
|
|
|
assert cfg.modindex_common_prefix == ['path1', 'path2']
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# simple default values
|
2024-01-03 20:23:45 -06:00
|
|
|
assert 'locale_dirs' in cfg.__dict__
|
2016-04-24 20:39:25 -05:00
|
|
|
assert cfg.locale_dirs == ['locales']
|
2014-09-21 10:17:02 -05:00
|
|
|
assert cfg.trim_footnote_reference_space is False
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# complex default values
|
|
|
|
assert 'html_title' not in cfg.__dict__
|
2011-01-06 13:34:37 -06:00
|
|
|
assert cfg.html_title == 'Sphinx <Tests> 0.6alpha1 documentation'
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# 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
|
Merged revisions 65138,65145-65146,65268-65273 via svnmerge from
svn+ssh://pythondev@svn.python.org/doctools/branches/0.4.x
........
r65138 | georg.brandl | 2008-07-19 15:42:35 +0200 (Sat, 19 Jul 2008) | 2 lines
#3416: fix missing parameter.
........
r65145 | georg.brandl | 2008-07-19 20:01:25 +0200 (Sat, 19 Jul 2008) | 2 lines
Now that we don't ship Jinja anymore by default the comment can go.
........
r65146 | georg.brandl | 2008-07-19 20:01:51 +0200 (Sat, 19 Jul 2008) | 2 lines
Reread documents with globbed toctrees when files are removed/added.
........
r65268 | georg.brandl | 2008-07-29 10:21:33 +0200 (Tue, 29 Jul 2008) | 2 lines
Fix by Markus Gritsch to place correct links to headings.
........
r65269 | georg.brandl | 2008-07-29 10:21:59 +0200 (Tue, 29 Jul 2008) | 2 lines
Make the writer's settings public.
........
r65270 | georg.brandl | 2008-07-29 10:22:28 +0200 (Tue, 29 Jul 2008) | 2 lines
Export test_root.
........
r65271 | georg.brandl | 2008-07-29 10:22:47 +0200 (Tue, 29 Jul 2008) | 2 lines
Add a markup test.
........
r65272 | georg.brandl | 2008-07-29 10:27:19 +0200 (Tue, 29 Jul 2008) | 2 lines
Bump version number.
........
r65273 | georg.brandl | 2008-07-29 11:05:37 +0200 (Tue, 29 Jul 2008) | 2 lines
Correct rendering of ``samp``.
........
2008-07-29 04:07:37 -05:00
|
|
|
assert 'nonexisting_value' not in cfg
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# invalid values
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(AttributeError):
|
2023-04-05 08:18:31 -05:00
|
|
|
_ = cfg._value
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(AttributeError):
|
2023-04-05 08:18:31 -05:00
|
|
|
_ = cfg.nonexisting_value
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# non-value attributes are deleted from the namespace
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(AttributeError):
|
2023-04-05 08:18:31 -05:00
|
|
|
_ = cfg.sys
|
2008-06-05 03:58:43 -05:00
|
|
|
|
|
|
|
# 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'
|
|
|
|
|
|
|
|
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_config_not_found(tmp_path):
|
2021-05-16 09:48:42 -05:00
|
|
|
with pytest.raises(ConfigError):
|
2023-07-27 18:39:12 -05:00
|
|
|
Config.read(tmp_path)
|
2021-05-16 09:48:42 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.parametrize('protocol', list(range(pickle.HIGHEST_PROTOCOL)))
|
2024-01-03 12:03:07 -06:00
|
|
|
def test_config_pickle_protocol(tmp_path, protocol: int):
|
|
|
|
config = Config()
|
|
|
|
|
|
|
|
pickled_config = pickle.loads(pickle.dumps(config, protocol))
|
|
|
|
|
2024-01-03 16:07:01 -06:00
|
|
|
assert list(config._options) == list(pickled_config._options)
|
2024-01-03 12:03:07 -06:00
|
|
|
assert repr(config) == repr(pickled_config)
|
|
|
|
|
|
|
|
|
2024-03-25 05:19:02 -05:00
|
|
|
def test_config_pickle_circular_reference_in_list():
|
|
|
|
a, b = [1], [2] # type: (CircularList, CircularList)
|
|
|
|
a.append(b)
|
|
|
|
b.append(a)
|
|
|
|
|
|
|
|
check_is_serializable(a, circular=True)
|
|
|
|
check_is_serializable(b, circular=True)
|
|
|
|
|
|
|
|
config = Config()
|
|
|
|
config.add('a', [], '', types=list)
|
|
|
|
config.add('b', [], '', types=list)
|
|
|
|
config.a, config.b = a, b
|
|
|
|
|
|
|
|
actual = pickle.loads(pickle.dumps(config))
|
|
|
|
assert isinstance(actual.a, list)
|
|
|
|
check_is_serializable(actual.a, circular=True)
|
|
|
|
|
|
|
|
assert isinstance(actual.b, list)
|
|
|
|
check_is_serializable(actual.b, circular=True)
|
|
|
|
|
|
|
|
assert actual.a[0] == 1
|
|
|
|
assert actual.a[1][0] == 2
|
|
|
|
assert actual.a[1][1][0] == 1
|
|
|
|
assert actual.a[1][1][1][0] == 2
|
|
|
|
|
|
|
|
assert actual.b[0] == 2
|
|
|
|
assert actual.b[1][0] == 1
|
|
|
|
assert actual.b[1][1][0] == 2
|
|
|
|
assert actual.b[1][1][1][0] == 1
|
|
|
|
|
|
|
|
assert len(actual.a) == 2
|
|
|
|
assert len(actual.a[1]) == 2
|
|
|
|
assert len(actual.a[1][1]) == 2
|
|
|
|
assert len(actual.a[1][1][1]) == 2
|
|
|
|
assert len(actual.a[1][1][1][1]) == 2
|
|
|
|
|
|
|
|
assert len(actual.b) == 2
|
|
|
|
assert len(actual.b[1]) == 2
|
|
|
|
assert len(actual.b[1][1]) == 2
|
|
|
|
assert len(actual.b[1][1][1]) == 2
|
|
|
|
assert len(actual.b[1][1][1][1]) == 2
|
|
|
|
|
|
|
|
def check(
|
|
|
|
u: list[list[object] | int],
|
|
|
|
v: list[list[object] | int],
|
|
|
|
*,
|
|
|
|
counter: Counter[type, int] | None = None,
|
|
|
|
guard: frozenset[int] = frozenset(),
|
|
|
|
) -> Counter[type, int]:
|
|
|
|
counter = Counter() if counter is None else counter
|
|
|
|
|
|
|
|
if id(u) in guard and id(v) in guard:
|
|
|
|
return counter
|
|
|
|
|
|
|
|
if isinstance(u, int):
|
|
|
|
assert v.__class__ is u.__class__
|
|
|
|
assert u == v
|
|
|
|
counter[type(u)] += 1
|
|
|
|
return counter
|
|
|
|
|
|
|
|
assert isinstance(u, list)
|
|
|
|
assert v.__class__ is u.__class__
|
2024-07-22 09:05:15 -05:00
|
|
|
for u_i, v_i in zip(u, v, strict=True):
|
2024-03-25 05:19:02 -05:00
|
|
|
counter[type(u)] += 1
|
|
|
|
check(u_i, v_i, counter=counter, guard=guard | {id(u), id(v)})
|
|
|
|
|
|
|
|
return counter
|
|
|
|
|
|
|
|
counter = check(actual.a, a)
|
|
|
|
# check(actual.a, a)
|
|
|
|
# check(actual.a[0], a[0]) -> ++counter[dict]
|
|
|
|
# ++counter[int] (a[0] is an int)
|
|
|
|
# check(actual.a[1], a[1]) -> ++counter[dict]
|
|
|
|
# check(actual.a[1][0], a[1][0]) -> ++counter[dict]
|
|
|
|
# ++counter[int] (a[1][0] is an int)
|
|
|
|
# check(actual.a[1][1], a[1][1]) -> ++counter[dict]
|
|
|
|
# recursive guard since a[1][1] == a
|
|
|
|
assert counter[type(a[0])] == 2
|
|
|
|
assert counter[type(a[1])] == 4
|
|
|
|
|
|
|
|
# same logic as above
|
|
|
|
counter = check(actual.b, b)
|
|
|
|
assert counter[type(b[0])] == 2
|
|
|
|
assert counter[type(b[1])] == 4
|
|
|
|
|
|
|
|
|
|
|
|
def test_config_pickle_circular_reference_in_dict():
|
|
|
|
x: CircularDict = {'a': 1, 'b': {'c': 1}}
|
|
|
|
x['b'] = x
|
|
|
|
check_is_serializable(x, circular=True)
|
|
|
|
|
|
|
|
config = Config()
|
|
|
|
config.add('x', [], '', types=dict)
|
|
|
|
config.x = x
|
|
|
|
|
|
|
|
actual = pickle.loads(pickle.dumps(config))
|
|
|
|
check_is_serializable(actual.x, circular=True)
|
|
|
|
assert isinstance(actual.x, dict)
|
|
|
|
|
|
|
|
assert actual.x['a'] == 1
|
|
|
|
assert actual.x['b']['a'] == 1
|
|
|
|
|
|
|
|
assert len(actual.x) == 2
|
|
|
|
assert len(actual.x['b']) == 2
|
|
|
|
assert len(actual.x['b']['b']) == 2
|
|
|
|
|
|
|
|
def check(
|
|
|
|
u: dict[str, dict[str, object] | int],
|
|
|
|
v: dict[str, dict[str, object] | int],
|
|
|
|
*,
|
|
|
|
counter: Counter[type, int] | None = None,
|
|
|
|
guard: frozenset[int] = frozenset(),
|
|
|
|
) -> Counter:
|
|
|
|
counter = Counter() if counter is None else counter
|
|
|
|
|
|
|
|
if id(u) in guard and id(v) in guard:
|
|
|
|
return counter
|
|
|
|
|
|
|
|
if isinstance(u, int):
|
|
|
|
assert v.__class__ is u.__class__
|
|
|
|
assert u == v
|
|
|
|
counter[type(u)] += 1
|
|
|
|
return counter
|
|
|
|
|
|
|
|
assert isinstance(u, dict)
|
|
|
|
assert v.__class__ is u.__class__
|
2024-07-22 09:05:15 -05:00
|
|
|
for u_i, v_i in zip(u, v, strict=True):
|
2024-03-25 05:19:02 -05:00
|
|
|
counter[type(u)] += 1
|
|
|
|
check(u[u_i], v[v_i], counter=counter, guard=guard | {id(u), id(v)})
|
|
|
|
return counter
|
|
|
|
|
|
|
|
counters = check(actual.x, x, counter=Counter())
|
|
|
|
# check(actual.x, x)
|
|
|
|
# check(actual.x['a'], x['a']) -> ++counter[dict]
|
|
|
|
# ++counter[int] (x['a'] is an int)
|
|
|
|
# check(actual.x['b'], x['b']) -> ++counter[dict]
|
|
|
|
# recursive guard since x['b'] == x
|
|
|
|
assert counters[type(x['a'])] == 1
|
|
|
|
assert counters[type(x['b'])] == 2
|
|
|
|
|
|
|
|
|
2018-03-23 00:26:07 -05:00
|
|
|
def test_extension_values():
|
|
|
|
config = Config()
|
2008-06-05 03:58:43 -05:00
|
|
|
|
2018-03-23 00:26:07 -05:00
|
|
|
# check standard settings
|
2020-11-24 09:50:36 -06:00
|
|
|
assert config.root_doc == 'index'
|
2008-06-05 03:58:43 -05:00
|
|
|
|
2018-03-23 00:26:07 -05:00
|
|
|
# can't override it by add_config_value()
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(ExtensionError) as excinfo:
|
2020-11-24 09:50:36 -06:00
|
|
|
config.add('root_doc', 'index', 'env', None)
|
2017-01-05 09:46:42 -06:00
|
|
|
assert 'already present' in str(excinfo.value)
|
2018-03-23 00:26:07 -05:00
|
|
|
|
|
|
|
# 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()
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(ExtensionError) as excinfo:
|
2018-03-23 00:26:07 -05:00
|
|
|
config.add('value_from_ext', [], 'env', None)
|
2017-01-05 09:46:42 -06:00
|
|
|
assert 'already present' in str(excinfo.value)
|
2010-01-12 04:45:38 -06:00
|
|
|
|
|
|
|
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_overrides():
|
2024-08-11 08:58:56 -05:00
|
|
|
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',
|
|
|
|
},
|
|
|
|
)
|
2018-03-23 00:05:04 -05:00
|
|
|
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', ())
|
|
|
|
|
|
|
|
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']
|
|
|
|
|
|
|
|
|
2019-12-09 08:53:55 -06:00
|
|
|
def test_overrides_boolean():
|
2024-08-11 08:58:56 -05:00
|
|
|
config = Config({}, {'value1': '1', 'value2': '0', 'value3': '0'})
|
2019-12-09 08:53:55 -06:00
|
|
|
config.add('value1', None, 'env', [bool])
|
|
|
|
config.add('value2', None, 'env', [bool])
|
|
|
|
config.add('value3', True, 'env', ())
|
|
|
|
|
|
|
|
assert config.value1 is True
|
|
|
|
assert config.value2 is False
|
|
|
|
assert config.value3 is False
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2024-01-03 20:23:45 -06:00
|
|
|
def test_overrides_dict_str(logger):
|
|
|
|
config = Config({}, {'spam': 'lobster'})
|
|
|
|
|
|
|
|
config.add('spam', {'ham': 'eggs'}, 'env', {dict, str})
|
|
|
|
|
|
|
|
assert config.spam == {'ham': 'eggs'}
|
|
|
|
|
|
|
|
# assert len(caplog.records) == 1
|
|
|
|
# msg = caplog.messages[0]
|
|
|
|
assert logger.method_calls
|
|
|
|
msg = str(logger.method_calls[0].args[1])
|
2024-08-11 08:58:56 -05:00
|
|
|
assert msg == (
|
|
|
|
"cannot override dictionary config setting 'spam', "
|
|
|
|
"ignoring (use 'spam.key=value' to set individual elements)"
|
|
|
|
)
|
2024-01-03 20:23:45 -06:00
|
|
|
|
|
|
|
|
|
|
|
def test_callable_defer():
|
|
|
|
config = Config()
|
|
|
|
config.add('alias', lambda c: c.master_doc, '', str)
|
|
|
|
|
|
|
|
assert config.master_doc == 'index'
|
|
|
|
assert config.alias == 'index'
|
|
|
|
|
|
|
|
config.master_doc = 'contents'
|
|
|
|
assert config.alias == 'contents'
|
|
|
|
|
|
|
|
config.master_doc = 'master_doc'
|
|
|
|
config.alias = 'spam'
|
|
|
|
assert config.alias == 'spam'
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_errors_warnings(logger, tmp_path):
|
2010-01-12 04:45:38 -06:00
|
|
|
# test the error for syntax errors in the config file
|
2023-07-27 18:39:12 -05:00
|
|
|
(tmp_path / 'conf.py').write_text('project = \n', encoding='ascii')
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(ConfigError) as excinfo:
|
2023-07-27 18:39:12 -05:00
|
|
|
Config.read(tmp_path, {}, None)
|
2017-01-05 09:46:42 -06:00
|
|
|
assert 'conf.py' in str(excinfo.value)
|
2010-01-12 04:45:38 -06:00
|
|
|
|
2010-07-31 12:47:15 -05:00
|
|
|
# test the automatic conversion of 2.x only code in configs
|
2023-07-27 18:39:12 -05:00
|
|
|
(tmp_path / 'conf.py').write_text('project = u"Jägermeister"\n', encoding='utf8')
|
|
|
|
cfg = Config.read(tmp_path, {}, None)
|
2018-12-15 08:02:28 -06:00
|
|
|
assert cfg.project == 'Jägermeister'
|
2016-12-20 08:21:30 -06:00
|
|
|
assert logger.called is False
|
2010-07-31 12:47:15 -05:00
|
|
|
|
2010-01-17 11:24:35 -06:00
|
|
|
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_errors_if_setup_is_not_callable(tmp_path, make_app):
|
2014-06-30 08:54:28 -05:00
|
|
|
# test the error to call setup() in the config file
|
2023-07-27 18:39:12 -05:00
|
|
|
(tmp_path / 'conf.py').write_text('setup = 1', encoding='utf8')
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(ConfigError) as excinfo:
|
2023-07-27 18:39:12 -05:00
|
|
|
make_app(srcdir=tmp_path)
|
2017-01-05 09:46:42 -06:00
|
|
|
assert 'callable' in str(excinfo.value)
|
2014-06-30 08:54:28 -05:00
|
|
|
|
|
|
|
|
2024-07-10 08:13:10 -05:00
|
|
|
@pytest.fixture
|
2023-07-27 18:39:12 -05:00
|
|
|
def make_app_with_empty_project(make_app, tmp_path):
|
2024-08-11 08:58:56 -05:00
|
|
|
(tmp_path / 'conf.py').touch()
|
2017-05-07 02:46:44 -05:00
|
|
|
|
|
|
|
def _make_app(*args, **kw):
|
2023-07-27 18:39:12 -05:00
|
|
|
kw.setdefault('srcdir', Path(tmp_path))
|
2017-05-07 02:46:44 -05:00
|
|
|
return make_app(*args, **kw)
|
2024-08-11 08:58:56 -05:00
|
|
|
|
2017-05-07 02:46:44 -05:00
|
|
|
return _make_app
|
|
|
|
|
|
|
|
|
2023-01-21 04:10:26 -06:00
|
|
|
@mock.patch.object(sphinx, '__display_version__', '1.6.4')
|
2017-05-07 02:46:44 -05:00
|
|
|
def test_needs_sphinx(make_app_with_empty_project):
|
|
|
|
make_app = make_app_with_empty_project
|
2016-01-20 22:51:17 -06:00
|
|
|
# micro version
|
2023-01-21 04:10:26 -06:00
|
|
|
make_app(confoverrides={'needs_sphinx': '1.6.3'}) # OK: less
|
|
|
|
make_app(confoverrides={'needs_sphinx': '1.6.4'}) # OK: equals
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(VersionRequirementError):
|
2023-01-21 04:10:26 -06:00
|
|
|
make_app(confoverrides={'needs_sphinx': '1.6.5'}) # NG: greater
|
2016-01-20 22:51:17 -06:00
|
|
|
|
|
|
|
# minor version
|
2023-01-21 04:10:26 -06:00
|
|
|
make_app(confoverrides={'needs_sphinx': '1.5'}) # OK: less
|
|
|
|
make_app(confoverrides={'needs_sphinx': '1.6'}) # OK: equals
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(VersionRequirementError):
|
2023-01-21 04:10:26 -06:00
|
|
|
make_app(confoverrides={'needs_sphinx': '1.7'}) # NG: greater
|
2016-01-20 22:51:17 -06:00
|
|
|
|
|
|
|
# major version
|
2018-07-28 06:19:30 -05:00
|
|
|
make_app(confoverrides={'needs_sphinx': '0'}) # OK: less
|
|
|
|
make_app(confoverrides={'needs_sphinx': '1'}) # OK: equals
|
2017-01-05 09:46:42 -06:00
|
|
|
with pytest.raises(VersionRequirementError):
|
2017-01-06 09:46:26 -06:00
|
|
|
make_app(confoverrides={'needs_sphinx': '2'}) # NG: greater
|
2013-02-10 00:25:45 -06:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_config_eol(logger, tmp_path):
|
2013-02-10 00:25:45 -06:00
|
|
|
# test config file's eol patterns: LF, CRLF
|
2023-07-27 18:39:12 -05:00
|
|
|
configfile = tmp_path / 'conf.py'
|
2014-05-01 08:54:09 -05:00
|
|
|
for eol in (b'\n', b'\r\n'):
|
|
|
|
configfile.write_bytes(b'project = "spam"' + eol)
|
2023-07-27 18:39:12 -05:00
|
|
|
cfg = Config.read(tmp_path, {}, None)
|
2018-12-15 08:02:28 -06:00
|
|
|
assert cfg.project == 'spam'
|
2016-12-20 08:21:30 -06:00
|
|
|
assert logger.called is False
|
2014-10-30 09:46:02 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
|
|
|
testroot='root',
|
2024-08-11 08:58:56 -05:00
|
|
|
confoverrides={'root_doc': 123, 'language': 'foo', 'primary_domain': None},
|
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_builtin_conf(app):
|
|
|
|
warnings = app.warning.getvalue()
|
2024-11-02 21:50:31 -05:00
|
|
|
assert 'root_doc' in warnings, (
|
|
|
|
'override on builtin "root_doc" should raise a type warning'
|
|
|
|
)
|
2017-01-06 09:46:26 -06:00
|
|
|
assert 'language' not in warnings, (
|
|
|
|
'explicitly permitted override on builtin "language" should NOT raise '
|
2024-08-11 08:58:56 -05:00
|
|
|
'a type warning'
|
|
|
|
)
|
2017-01-06 09:46:26 -06:00
|
|
|
assert 'primary_domain' not in warnings, (
|
2024-11-02 21:50:31 -05:00
|
|
|
'override to None on builtin "primary_domain" should NOT raise a type warning'
|
2024-08-11 08:58:56 -05:00
|
|
|
)
|
2015-09-11 02:35:46 -05:00
|
|
|
|
|
|
|
|
2018-03-23 00:05:04 -05:00
|
|
|
# example classes for type checking
|
2018-09-11 08:48:35 -05:00
|
|
|
class A:
|
2018-03-23 00:05:04 -05:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class B(A):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class C(A):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
# name, default, annotation, actual, warned
|
|
|
|
TYPECHECK_WARNINGS = [
|
2024-08-11 08:58:56 -05:00
|
|
|
('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
|
|
|
|
] # fmt: skip
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('sphinx.config.logger')
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
('name', 'default', 'annotation', 'actual', 'warned'),
|
|
|
|
TYPECHECK_WARNINGS,
|
|
|
|
)
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_check_types(logger, name, default, annotation, actual, warned):
|
|
|
|
config = Config({name: actual})
|
|
|
|
config.add(name, default, 'env', annotation or ())
|
2018-03-23 06:43:41 -05:00
|
|
|
check_confval_types(None, config)
|
2018-03-23 00:05:04 -05:00
|
|
|
assert logger.warning.called == warned
|
|
|
|
|
|
|
|
|
2018-10-02 13:20:21 -05:00
|
|
|
TYPECHECK_WARNING_MESSAGES = [
|
2024-08-11 08:58:56 -05:00
|
|
|
(
|
|
|
|
'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 `int' or `str'.",
|
|
|
|
),
|
|
|
|
(
|
|
|
|
'value1',
|
|
|
|
'string',
|
|
|
|
[str, int, tuple],
|
|
|
|
['foo', 'bar'],
|
|
|
|
"The config value `value1' has type `list'; expected `int', `str', or `tuple'.",
|
|
|
|
),
|
2018-10-02 13:20:21 -05:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
('name', 'default', 'annotation', 'actual', 'message'),
|
|
|
|
TYPECHECK_WARNING_MESSAGES,
|
|
|
|
)
|
2018-10-02 13:20:21 -05:00
|
|
|
def test_conf_warning_message(logger, name, default, annotation, actual, message):
|
|
|
|
config = Config({name: actual})
|
|
|
|
config.add(name, default, False, annotation or ())
|
|
|
|
check_confval_types(None, config)
|
2019-03-07 08:35:36 -06:00
|
|
|
assert logger.warning.called
|
2018-10-02 13:20:21 -05:00
|
|
|
assert logger.warning.call_args[0][0] == message
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_check_enum(logger):
|
|
|
|
config = Config()
|
|
|
|
config.add('value', 'default', False, ENUM('default', 'one', 'two'))
|
2018-03-23 06:43:41 -05:00
|
|
|
check_confval_types(None, config)
|
2018-03-23 00:05:04 -05:00
|
|
|
logger.warning.assert_not_called() # not warned
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_check_enum_failed(logger):
|
|
|
|
config = Config({'value': 'invalid'})
|
|
|
|
config.add('value', 'default', False, ENUM('default', 'one', 'two'))
|
2018-03-23 06:43:41 -05:00
|
|
|
check_confval_types(None, config)
|
2019-03-07 08:35:36 -06:00
|
|
|
assert logger.warning.called
|
2018-03-23 00:05:04 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_check_enum_for_list(logger):
|
|
|
|
config = Config({'value': ['one', 'two']})
|
|
|
|
config.add('value', 'default', False, ENUM('default', 'one', 'two'))
|
2018-03-23 06:43:41 -05:00
|
|
|
check_confval_types(None, config)
|
2018-03-23 00:05:04 -05:00
|
|
|
logger.warning.assert_not_called() # not warned
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2018-03-23 00:05:04 -05:00
|
|
|
def test_check_enum_for_list_failed(logger):
|
|
|
|
config = Config({'value': ['one', 'two', 'invalid']})
|
|
|
|
config.add('value', 'default', False, ENUM('default', 'one', 'two'))
|
2018-03-23 06:43:41 -05:00
|
|
|
check_confval_types(None, config)
|
2019-03-07 08:35:36 -06:00
|
|
|
assert logger.warning.called
|
2021-05-02 06:53:46 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2024-04-16 20:24:56 -05:00
|
|
|
def test_check_any(logger):
|
|
|
|
config = Config({'value': None})
|
|
|
|
config.add('value', 'default', '', Any)
|
|
|
|
check_confval_types(None, config)
|
|
|
|
logger.warning.assert_not_called() # not warned
|
|
|
|
|
|
|
|
|
2021-05-02 06:53:46 -05:00
|
|
|
nitpick_warnings = [
|
2024-08-11 08:58:56 -05:00
|
|
|
'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',
|
2021-05-02 06:53:46 -05:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2024-08-12 16:34:03 -05:00
|
|
|
@pytest.mark.sphinx('html', testroot='nitpicky-warnings')
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_nitpick_base(app):
|
2024-01-16 20:38:46 -06:00
|
|
|
app.build(force_all=True)
|
2021-05-02 06:53:46 -05:00
|
|
|
|
2024-07-23 09:35:55 -05:00
|
|
|
warning = app.warning.getvalue().strip().split('\n')
|
2024-07-22 09:05:15 -05:00
|
|
|
for actual, expected in zip(warning, nitpick_warnings, strict=True):
|
2021-05-02 06:53:46 -05:00
|
|
|
assert expected in actual
|
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
2024-08-11 08:58:56 -05:00
|
|
|
testroot='nitpicky-warnings',
|
|
|
|
confoverrides={
|
|
|
|
'nitpick_ignore': {
|
|
|
|
('py:const', 'prefix.anything.postfix'),
|
|
|
|
('py:class', 'prefix.anything'),
|
|
|
|
('py:class', 'anything.postfix'),
|
|
|
|
('js:class', 'prefix.anything.postfix'),
|
|
|
|
},
|
2023-04-06 17:11:39 -05:00
|
|
|
},
|
2024-08-11 08:58:56 -05:00
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_nitpick_ignore(app):
|
2024-01-16 20:38:46 -06:00
|
|
|
app.build(force_all=True)
|
2024-07-23 09:35:55 -05:00
|
|
|
assert not len(app.warning.getvalue().strip())
|
2021-05-02 06:54:00 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
2024-08-11 08:58:56 -05:00
|
|
|
testroot='nitpicky-warnings',
|
|
|
|
confoverrides={
|
|
|
|
'nitpick_ignore_regex': [
|
|
|
|
(r'py:.*', r'.*postfix'),
|
|
|
|
(r'.*:class', r'prefix.*'),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_nitpick_ignore_regex1(app):
|
2024-01-16 20:38:46 -06:00
|
|
|
app.build(force_all=True)
|
2024-07-23 09:35:55 -05:00
|
|
|
assert not len(app.warning.getvalue().strip())
|
2021-05-02 06:54:00 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
2024-08-11 08:58:56 -05:00
|
|
|
testroot='nitpicky-warnings',
|
|
|
|
confoverrides={
|
|
|
|
'nitpick_ignore_regex': [
|
|
|
|
(r'py:.*', r'prefix.*'),
|
|
|
|
(r'.*:class', r'.*postfix'),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_nitpick_ignore_regex2(app):
|
2024-01-16 20:38:46 -06:00
|
|
|
app.build(force_all=True)
|
2024-07-23 09:35:55 -05:00
|
|
|
assert not len(app.warning.getvalue().strip())
|
2021-05-02 06:54:00 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@pytest.mark.sphinx(
|
2024-08-12 16:34:03 -05:00
|
|
|
'html',
|
2024-08-11 08:58:56 -05:00
|
|
|
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''),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
)
|
2024-07-23 09:35:55 -05:00
|
|
|
def test_nitpick_ignore_regex_fullmatch(app):
|
2024-01-16 20:38:46 -06:00
|
|
|
app.build(force_all=True)
|
2021-05-02 06:54:00 -05:00
|
|
|
|
2024-07-23 09:35:55 -05:00
|
|
|
warning = app.warning.getvalue().strip().split('\n')
|
2024-07-22 09:05:15 -05:00
|
|
|
for actual, expected in zip(warning, nitpick_warnings, strict=True):
|
2021-05-02 06:54:00 -05:00
|
|
|
assert expected in actual
|
2022-05-27 17:50:07 -05:00
|
|
|
|
|
|
|
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_conf_py_language_none(tmp_path):
|
2022-05-27 17:50:07 -05:00
|
|
|
"""Regression test for #10474."""
|
|
|
|
# Given a conf.py file with language = None
|
2024-08-11 08:58:56 -05:00
|
|
|
(tmp_path / 'conf.py').write_text('language = None', encoding='utf-8')
|
2022-05-27 17:50:07 -05:00
|
|
|
|
|
|
|
# When we load conf.py into a Config object
|
2023-07-27 18:39:12 -05:00
|
|
|
cfg = Config.read(tmp_path, {}, None)
|
2022-05-27 17:50:07 -05:00
|
|
|
|
|
|
|
# Then the language is coerced to English
|
2024-08-11 08:58:56 -05:00
|
|
|
assert cfg.language == 'en'
|
2022-05-27 18:06:29 -05:00
|
|
|
|
|
|
|
|
2024-08-11 08:58:56 -05:00
|
|
|
@mock.patch('sphinx.config.logger')
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_conf_py_language_none_warning(logger, tmp_path):
|
2022-05-28 13:06:48 -05:00
|
|
|
"""Regression test for #10474."""
|
|
|
|
# Given a conf.py file with language = None
|
2024-08-11 08:58:56 -05:00
|
|
|
(tmp_path / 'conf.py').write_text('language = None', encoding='utf-8')
|
2022-05-28 13:06:48 -05:00
|
|
|
|
2022-05-28 13:17:46 -05:00
|
|
|
# When we load conf.py into a Config object
|
2023-07-27 18:39:12 -05:00
|
|
|
Config.read(tmp_path, {}, None)
|
2022-05-28 13:17:46 -05:00
|
|
|
|
2022-05-28 13:06:48 -05:00
|
|
|
# Then a warning is raised
|
2022-05-28 13:26:13 -05:00
|
|
|
assert logger.warning.called
|
|
|
|
assert logger.warning.call_args[0][0] == (
|
2022-05-28 13:17:46 -05:00
|
|
|
"Invalid configuration value found: 'language = None'. "
|
2024-08-11 08:58:56 -05:00
|
|
|
'Update your configuration to a valid language code. '
|
|
|
|
"Falling back to 'en' (English)."
|
|
|
|
)
|
2022-05-28 13:06:48 -05:00
|
|
|
|
|
|
|
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_conf_py_no_language(tmp_path):
|
2022-05-27 18:06:29 -05:00
|
|
|
"""Regression test for #10474."""
|
|
|
|
# Given a conf.py file with no language attribute
|
2024-08-11 08:58:56 -05:00
|
|
|
(tmp_path / 'conf.py').touch()
|
2022-05-27 18:06:29 -05:00
|
|
|
|
|
|
|
# When we load conf.py into a Config object
|
2023-07-27 18:39:12 -05:00
|
|
|
cfg = Config.read(tmp_path, {}, None)
|
2022-05-27 18:06:29 -05:00
|
|
|
|
|
|
|
# Then the language is coerced to English
|
2024-08-11 08:58:56 -05:00
|
|
|
assert cfg.language == 'en'
|
2023-04-25 05:31:57 -05:00
|
|
|
|
|
|
|
|
2023-07-27 18:39:12 -05:00
|
|
|
def test_conf_py_nitpick_ignore_list(tmp_path):
|
2023-04-25 05:31:57 -05:00
|
|
|
"""Regression test for #11355."""
|
|
|
|
# Given a conf.py file with no language attribute
|
2024-08-11 08:58:56 -05:00
|
|
|
(tmp_path / 'conf.py').touch()
|
2023-04-25 05:31:57 -05:00
|
|
|
|
|
|
|
# When we load conf.py into a Config object
|
2023-07-27 18:39:12 -05:00
|
|
|
cfg = Config.read(tmp_path, {}, None)
|
2023-04-25 05:31:57 -05:00
|
|
|
|
|
|
|
# Then the default nitpick_ignore[_regex] is an empty list
|
|
|
|
assert cfg.nitpick_ignore == []
|
|
|
|
assert cfg.nitpick_ignore_regex == []
|
2023-05-11 16:43:48 -05:00
|
|
|
|
|
|
|
|
2023-12-29 14:29:40 -06:00
|
|
|
def test_gettext_compact_command_line_true():
|
|
|
|
config = Config({}, {'gettext_compact': '1'})
|
|
|
|
config.add('gettext_compact', True, '', {bool, str})
|
|
|
|
_gettext_compact_validator(..., config)
|
|
|
|
|
|
|
|
# regression test for #8549 (-D gettext_compact=1)
|
|
|
|
assert config.gettext_compact is True
|
|
|
|
|
|
|
|
|
|
|
|
def test_gettext_compact_command_line_false():
|
|
|
|
config = Config({}, {'gettext_compact': '0'})
|
|
|
|
config.add('gettext_compact', True, '', {bool, str})
|
|
|
|
_gettext_compact_validator(..., config)
|
|
|
|
|
|
|
|
# regression test for #8549 (-D gettext_compact=0)
|
|
|
|
assert config.gettext_compact is False
|
|
|
|
|
|
|
|
|
|
|
|
def test_gettext_compact_command_line_str():
|
|
|
|
config = Config({}, {'gettext_compact': 'spam'})
|
|
|
|
config.add('gettext_compact', True, '', {bool, str})
|
|
|
|
_gettext_compact_validator(..., config)
|
|
|
|
|
|
|
|
# regression test for #8549 (-D gettext_compact=spam)
|
|
|
|
assert config.gettext_compact == 'spam'
|
2024-06-17 11:04:33 -05:00
|
|
|
|
|
|
|
|
|
|
|
def test_root_doc_and_master_doc_are_synchronized():
|
|
|
|
c = Config()
|
|
|
|
assert c.master_doc == 'index'
|
|
|
|
assert c.root_doc == c.master_doc
|
|
|
|
|
|
|
|
c = Config()
|
|
|
|
c.master_doc = '1234'
|
|
|
|
assert c.master_doc == '1234'
|
|
|
|
assert c.root_doc == c.master_doc
|
|
|
|
|
|
|
|
c = Config()
|
|
|
|
c.root_doc = '1234'
|
|
|
|
assert c.master_doc == '1234'
|
|
|
|
assert c.root_doc == c.master_doc
|