Skip to content

Commit

Permalink
apidoc/python: Fix xrefs to type parameters (#372)
Browse files Browse the repository at this point in the history
Previously, if a function introduced a type parameter, any references to
that type parameter in the signature itself were correctly resolved.
However, when documenting function parameters, the type from the
signature is duplicated next to the parameter name, and any references
to type parameters in these duplicated types were not correctly
resolved.

This commit fixes that issue by applying the same resolution logic to
the duplicated types.
  • Loading branch information
jbms committed Jul 23, 2024
1 parent f5e3d7c commit 162f262
Show file tree
Hide file tree
Showing 4 changed files with 95 additions and 14 deletions.
6 changes: 5 additions & 1 deletion docs/type_param_demo.py
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,11 @@ def get(self, key: K, default: V) -> V: ...
def get(self, key: K, default: T) -> Union[V, T]: ...

def get(self, key: K, default=None):
"""Return the mapped value, or the specified default."""
"""Return the mapped value, or the specified default.
:param key: Key to retrieve.
:param default: Default value to return if key is not present.
"""
...

def __len__(self) -> int:
Expand Down
43 changes: 31 additions & 12 deletions sphinx_immaterial/apidoc/python/parameter_objects.py
Original file line number Diff line number Diff line change
Expand Up @@ -253,16 +253,29 @@ def get_objects(
PythonDomain.get_objects = get_objects # type: ignore[assignment]


def _fix_pending_xrefs_to_type_params(
type_param_symbols: dict[str, str], parent: docutils.nodes.Node
) -> None:
for xref in parent.findall(condition=sphinx.addnodes.pending_xref):
if xref["refdomain"] == "py" and xref["reftype"] in ("class", "param"):
p_symbol = type_param_symbols.get(xref["reftarget"])
if p_symbol is not None:
xref["reftarget"] = p_symbol
xref["refspecific"] = False


def _add_parameter_links_to_signature(
env: sphinx.environment.BuildEnvironment,
signode: sphinx.addnodes.desc_signature,
type_param_symbol_prefix: str,
function_param_symbol_prefix: str,
) -> Dict[str, docutils.nodes.Element]:
) -> tuple[dict[str, docutils.nodes.Element], dict[str, str]]:
"""Cross-links parameter names in signature to parameter objects.
Returns:
Map of parameter name to original (not linked) parameter node.
Tuple of:
- Map of parameter name to original (not linked) parameter node.
- Map of type parameter name to parameter object symbol.
"""
sig_param_nodes: Dict[str, docutils.nodes.Element] = {}

Expand Down Expand Up @@ -336,15 +349,13 @@ def _collect_parameters(
refnode["implicit_sig_param_ref"] = True
name_node.replace_self(refnode)

# Also cross-link references to type parameters in annotations.
for xref in signode.findall(condition=sphinx.addnodes.pending_xref):
if xref["refdomain"] == "py" and xref["reftype"] in ("class", "param"):
p_symbol = type_param_symbols.get(xref["reftarget"])
if p_symbol is not None:
xref["reftarget"] = p_symbol
xref["refspecific"] = False
if type_param_symbols:
# Also cross-link references to type parameters in annotations.
_fix_pending_xrefs_to_type_params(type_param_symbols, signode)
for parent in sig_param_nodes.values():
_fix_pending_xrefs_to_type_params(type_param_symbols, parent)

return sig_param_nodes
return sig_param_nodes, type_param_symbols


def _collate_parameter_symbols(
Expand Down Expand Up @@ -550,6 +561,8 @@ def _cross_link_parameters(
env = app.env
assert isinstance(env, sphinx.environment.BuildEnvironment)

type_param_symbols: dict[str, str] = {}

# Collect the docutils nodes corresponding to the declarations of the
# parameters in each signature, and turn the parameter names into
# cross-links to the parameter description.
Expand All @@ -559,9 +572,11 @@ def _cross_link_parameters(
# e.g. `x : int = 10` rather than just `x`.
sig_param_nodes_for_signature = []
for signode, symbol, function_symbol in zip(signodes, symbols, function_symbols):
sig_param_nodes_for_signature.append(
_add_parameter_links_to_signature(env, signode, symbol, function_symbol)
sig_param_nodes, sig_type_param_symbols = _add_parameter_links_to_signature(
env, signode, symbol, function_symbol
)
sig_param_nodes_for_signature.append(sig_param_nodes)
type_param_symbols.update(sig_type_param_symbols)

# Find all parameter descriptions in the object description body, and mark
# them as the target for cross links to that parameter. Also substitute in
Expand All @@ -576,6 +591,10 @@ def _cross_link_parameters(
noindex=noindex,
)

# Fix any remaining references to type parameters.
if type_param_symbols:
_fix_pending_xrefs_to_type_params(type_param_symbols, content)

if not noindex:
py = cast(sphinx.domains.python.PythonDomain, env.get_domain("py"))

Expand Down
31 changes: 30 additions & 1 deletion tests/python_apigen_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,15 @@ def apigen_make_app(tmp_path: pathlib.Path, make_app):

def make(extra_conf: str = "", **kwargs):
(tmp_path / "conf.py").write_text(conf + extra_conf, encoding="utf-8")
(tmp_path / "index.rst").write_text("", encoding="utf-8")
(tmp_path / "index.rst").write_text(
"""
.. python-apigen-group:: Public Members
.. python-apigen-group:: Classes
""",
encoding="utf-8",
)
return make_app(srcdir=SphinxPath(str(tmp_path)), **kwargs)

yield make
Expand Down Expand Up @@ -157,3 +165,24 @@ def test_pure_python_property(apigen_make_app):
assert member.name == "baz"
assert len(member.siblings) == 1
assert member.siblings[0].name == "bar"


@pytest.mark.skipif(
sphinx.version_info < (7, 1),
reason=f"Type parameters are not supported by Sphinx {sphinx.version_info}",
)
def test_type_params(apigen_make_app):
"""Tests that references to type parameters are all resolved."""
testmod = "python_apigen_test_modules.type_params"
app = apigen_make_app(
confoverrides=dict(
python_apigen_modules={
testmod: "api/",
},
nitpicky=True,
),
)
app.build()
print(app._status.getvalue())
print(app._warning.getvalue())
assert not app._warning.getvalue()
29 changes: 29 additions & 0 deletions tests/python_apigen_test_modules/type_params.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
from typing import TypeVar

T = TypeVar("T")


def foo(x: T) -> T:
"""Foo function.
:param x: Something or other.
"""
return x


def bar(x: T) -> T:
return x


class C:
def get(self, x: T, y: T) -> T:
"""Get function.
:param x: Something or other.
:param y: Another param.
:type y: T
"""
return x


__all__ = ["foo", "bar", "C"]

0 comments on commit 162f262

Please sign in to comment.