# Copyright Kevin Deldycke <kevin@deldycke.com> and contributors.
#
# This program 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 2
# of the License, or (at your option) any later version.
#
# This program 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 this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
"""Sphinx rendering of CLI based on Click Extra.
.. seealso::
These directives are based on `Pallets' Sphinx Themes
<https://github.com/pallets/pallets-sphinx-themes/blob/main/src/pallets_sphinx_themes/themes/click/domain.py>`_,
`released under a BSD-3-Clause license
<https://github.com/pallets/pallets-sphinx-themes/blob/main/LICENSE.txt>`_.
Compared to the latter, it:
- Add support for MyST syntax.
- Adds rendering of ANSI codes in CLI results.
- Has better error handling and reporting which helps you pinpoint the failing
code in your documentation.
- Removes the ``println`` function which was used to explicitly print a blank
line. This is no longer needed as it is now handled natively.
"""
from __future__ import annotations
import ast
import contextlib
import inspect
import re
import shlex
import subprocess
import sys
import tempfile
from functools import cached_property, partial
import click
from click.testing import CliRunner, EchoingStdin
from docutils import nodes
from docutils.statemachine import StringList
from sphinx.directives import SphinxDirective, directives
from sphinx.directives.code import CodeBlock
from sphinx.util import logging
from ..color import forced_color
from ._base import (
StatelessDomain,
compile_directive,
directive_source,
make_cleanup,
)
TYPE_CHECKING = False
if TYPE_CHECKING:
from collections.abc import Iterable
from typing import ClassVar, Literal
from sphinx.util.typing import OptionSpec
logger = logging.getLogger(__name__)
RST_INDENT = " " * 3
"""The indentation used for rST code blocks lines."""
_CLIRUNNER_HAS_CAPTURE = "capture" in inspect.signature(CliRunner.__init__).parameters
"""Whether Click's :class:`~click.testing.CliRunner` accepts the ``capture`` keyword.
Added in Click 8.4 to select the stream-capture strategy (``"sys"`` or ``"fd"``).
Absent in earlier releases, where the runner's capture behavior is fixed.
"""
[docs]
class TerminatedEchoingStdin(EchoingStdin):
"""Like ``click.testing.EchoingStdin`` but adds a visible
``^D`` in place of the EOT character (``\x04``).
:meth:`ClickRunner.invoke` adds ``\x04`` when ``terminate_input=True``.
"""
def _echo(self, rv: bytes) -> bytes:
eof = rv[-1] == b"\x04"[0]
if eof:
rv = rv[:-1]
if not self._paused:
self._output.write(rv)
if eof:
self._output.write(b"^D\n")
return rv
[docs]
@contextlib.contextmanager
def patch_subprocess():
"""Patch subprocess to work better with :meth:`ClickRunner.invoke`.
``subprocess.call`` output is redirected to ``click.echo`` so it
shows up in the example output.
.. caution::
The replacement is installed on the ``subprocess`` module itself
(not thread-local), so for the duration of the ``with`` block any
other code in the same process that calls ``subprocess.call`` sees
the patched version. With ``parallel_read_safe = True`` declared
on :class:`ClickDomain`, a parallel reader running concurrently
on a different document gets the patched ``subprocess.call`` too.
The redirection is benign (output goes to ``click.echo``) but the
race is real, and the parallel-safe claim is weaker than it
looks for documents that themselves shell out via
``subprocess.call``.
"""
old_call = subprocess.call
def dummy_call(*args, **kwargs):
with tempfile.TemporaryFile("wb+") as f:
kwargs["stdout"] = f
kwargs["stderr"] = f
rv = subprocess.Popen(*args, **kwargs).wait()
f.seek(0)
click.echo(f.read().decode("utf-8", "replace").rstrip())
return rv
subprocess.call = dummy_call
try:
yield
finally:
subprocess.call = old_call
[docs]
class ClickRunner(CliRunner):
"""A sub-class of :class:`click.testing.CliRunner` for Sphinx directive execution.
Produces unfiltered ANSI codes so that the ``Directive`` sub-classes below can
render colors in the HTML output. Because Click Extra executes the documented
command here, :meth:`invoke` forces color across both color systems a CLI might use:
``color=True`` covers Click's (``should_strip_ansi``), and
:func:`~click_extra.color.forced_color` sets ``FORCE_COLOR`` for Rich's (which
``rich-click`` uses and ``color=True`` never reaches). The MkDocs plugin shares the
latter lever but cannot pass ``color=True``, since it patches a renderer it never
executes.
On Click 8.4+ the runner defaults to ``capture="fd"`` on Unix (overridable through
the ``click_extra_run_capture`` ``conf.py`` value) so a documented command that
writes through ``sys.stdout.fileno()`` is captured and rendered, instead of aborting
the build with :exc:`io.UnsupportedOperation`. On Windows, where fd-backed streams
are not supported, the default falls back to ``capture="sys"``.
"""
def __init__(self, capture: Literal["sys", "fd"] | None = None) -> None:
# capture="fd" backs the captured streams with a real file descriptor so a
# documented command calling sys.stdout.fileno() renders instead of crashing
# the build. It is the default (the click_extra_run_capture conf.py value
# selects it), safe at doc-build time unlike under the pytest stream
# duplication that got it reverted as a Click default (pallets/click#3391).
# Click < 8.4 lacks the parameter and needs none (8.3.3+ exposed a fileno by
# default; < 8.3.3 never did), so omitting it is correct.
# Windows does not support fd-backed streams (no Unix file descriptors), so
# fall back to "sys" when the caller has not pinned a mode explicitly.
if _CLIRUNNER_HAS_CAPTURE:
default_capture: Literal["sys", "fd"] = (
"sys" if sys.platform == "win32" else "fd"
)
super().__init__(echo_stdin=True, capture=capture or default_capture)
else:
super().__init__(echo_stdin=True)
self.namespace = {"click": click, "__file__": "dummy.py"}
[docs]
@contextlib.contextmanager
def isolation(self, *args, **kwargs):
iso = super().isolation(*args, **kwargs)
with iso as streams:
try:
buffer = sys.stdin.buffer
except AttributeError:
buffer = sys.stdin
# FIXME: We need to replace EchoingStdin with our custom
# class that outputs "^D". At this point we know sys.stdin
# has been patched so it's safe to reassign the class.
# Remove this once EchoingStdin is overridable.
buffer.__class__ = TerminatedEchoingStdin
yield streams
[docs]
def invoke( # type: ignore[override]
self,
cli,
args=None,
prog_name=None,
input=None,
terminate_input=False,
env=None,
_output_lines=None,
**extra,
) -> click.testing.Result:
"""Like ``CliRunner.invoke`` but displays what the user
would enter in the terminal for env vars, command arguments, and
prompts.
:param terminate_input: Whether to display ``^D`` after a list of
input.
:param _output_lines: A list used internally to collect lines to
be displayed.
"""
output_lines = _output_lines if _output_lines is not None else []
if env:
for key, value in sorted(env.items()):
value = shlex.quote(value)
output_lines.append(f"$ export {key}={value}")
args = args or []
if prog_name is None:
prog_name = cli.name.replace("_", "-")
output_lines.append(f"$ {prog_name} {shlex.join(args)}".rstrip())
# remove "python" from command
prog_name = prog_name.rsplit(" ", 1)[-1]
if isinstance(input, (tuple, list)):
input = "\n".join(input) + "\n"
if terminate_input:
input += "\x04"
# ``color=True`` keeps ANSI in Click's color system: it flips
# ``should_strip_ansi``, which CliRunner otherwise leaves stripping on its
# non-TTY buffer. But rich-click renders through Rich's Console, a separate
# system that ignores ``should_strip_ansi`` and only honors ``FORCE_COLOR``, so
# ``forced_color()`` sets that too. Together they cover both color systems a
# documented CLI might use.
with forced_color():
result = super().invoke(
cli=cli,
args=args,
input=input,
env=env,
prog_name=prog_name,
color=True,
**extra,
)
output_lines.extend(result.output.splitlines())
return result
[docs]
def execute_source(self, directive: SphinxDirective) -> None:
"""Execute the given code, adding it to the runner's namespace."""
code = compile_directive(directive)
with patch_subprocess():
exec(code, self.namespace) # noqa: S102
[docs]
def run_cli(self, directive: SphinxDirective) -> list[str]:
"""Execute the given ``source_code``.
Returns a simulation of terminal execution, including a mix of input, output,
prompts and tracebacks.
The execution context is augmented, so you can refer directly to these
functions in the provided ``source_code``:
- :meth:`invoke()`: which is the same as :meth:`ClickRunner.invoke`
- ``isolated_filesystem()``: A context manager that changes to a temporary
directory while executing the block.
If any local variable in the provided ``source_code`` conflicts with these
functions, a :class:`RuntimeError` is raised to help you pinpoint the issue.
"""
source_code, location = directive_source(directive)
buffer: list[str] = []
# Functions available as local variables when executing the code.
local_vars = {
"invoke": partial(self.invoke, _output_lines=buffer),
"isolated_filesystem": self.isolated_filesystem,
}
# Check for local variable conflicts.
tree = ast.parse(source_code, location)
for node in ast.walk(tree):
if (
isinstance(node, ast.Name)
and isinstance(node.ctx, ast.Store)
and node.id in local_vars
):
# Get the source lines for better error reporting.
source_lines = source_code.splitlines()
# Get the line number relative to the source code.
python_lineno = node.lineno
python_line = source_lines[python_lineno - 1]
# Compute the absolute line number in the document.
if directive.is_myst_syntax:
# In MyST, the content offset is the position of the first line
# of the source code, relative to the directive itself.
doc_lineno = (
directive.lineno + directive.content_offset + python_lineno
)
# XXX MyST absolute error line reporting is broken in some
# situations, see:
# https://github.com/executablebooks/MyST-Parser/pull/1048
else:
# In rST, the content offset is the absolute position at which
# the source code starts in the document.
doc_lineno = directive.content_offset + python_lineno
raise RuntimeError(
f"Local variable {node.id!r} at "
f"{location}:{directive.name}:{doc_lineno} conflicts with "
f"the one automatically provided by the {directive.name} "
"directive.\n"
f"Line: {python_line}"
)
code = compile_directive(directive)
exec(code, self.namespace, local_vars) # noqa: S102
return buffer
def _resolve_run_capture(
configured: Literal["sys", "fd"],
) -> Literal["sys", "fd"]:
"""Degrade the configured stream-capture mode to one the platform supports.
The ``click_extra_run_capture`` ``conf.py`` value is a build-time
*preference*. ``"fd"`` backs the captured streams with a real file descriptor
so a command writing through ``sys.stdout.fileno()`` renders (see
:class:`ClickRunner`), but Windows has no Unix file descriptors and Click
rejects ``capture="fd"`` there. Degrade ``"fd"`` to ``"sys"`` on Windows so
the documentation build proceeds: such fileno-writing commands simply do not
render, instead of aborting the whole build.
A direct ``ClickRunner(capture="fd")`` call still honors the explicit pin (and
raises on Windows); only the config-derived preference degrades here.
"""
if configured == "fd" and sys.platform == "win32":
return "sys"
return configured
[docs]
class ClickDirective(SphinxDirective):
has_content = True
required_arguments = 0
optional_arguments = 1
"""The optional argument overrides the default Pygments language to use."""
final_argument_whitespace = False
option_spec: ClassVar[OptionSpec] = CodeBlock.option_spec | {
"language": directives.unchanged_required,
"show-source": directives.flag,
"hide-source": directives.flag,
"show-results": directives.flag,
"hide-results": directives.flag,
"emphasize-result-lines": CodeBlock.option_spec["emphasize-lines"],
# TODO: Add a show-prompts and hide-prompts options?
}
"""Options supported by this directive.
Support the `same options
<https://github.com/sphinx-doc/sphinx/blob/cc7c6f4/sphinx/directives/code.py#L108-L117>`_
as ``sphinx.directives.code.CodeBlock``, and some specific to Click
directives.
The standard ``emphasize-lines`` option applies to the source block only. Use
``emphasize-result-lines`` to highlight specific lines in the captured output
block, with the same syntax (like ``:emphasize-result-lines: 1,3-5``).
"""
default_language: str
"""Default highlighting language to use to render the code block.
`All Pygments' languages short names <https://pygments.org/languages/>`_ are
recognized.
"""
show_source_by_default: bool = True
"""Whether to render the source code of the example in the code block."""
show_results_by_default: bool = True
"""Whether to render the results of the example in the code block."""
runner_method: str
"""The name of the method to call on the :class:`ClickRunner` instance."""
runner_attr: ClassVar[str] = "click_runner"
"""Name of the attribute holding the runner on the doctree.
Subclasses (like ``PythonDirective``)
override this so the Click and Python runners don't collide on the same
document.
"""
runner_factory: ClassVar[type] = None # type: ignore[assignment]
"""Class to instantiate for the per-document runner.
Defaults to :class:`ClickRunner` in :class:`ClickDirective` (set after the
class definition to break the forward reference).
"""
@property
def runner(self):
"""Get or create the per-document runner.
Creates one runner per document, keyed by :attr:`runner_attr`.
"""
runner = getattr(self.state.document, self.runner_attr, None)
if runner is None:
runner = self.runner_factory(
capture=_resolve_run_capture(self.env.config.click_extra_run_capture)
)
setattr(self.state.document, self.runner_attr, runner)
return runner
[docs]
@cached_property
def language(self) -> str:
"""Short name of the Pygments lexer used to highlight the code block.
Returns, in order of precedence, the language specified in the `:language:`
directive options, the first argument of the directive (if any), or the default
set in the directive class.
"""
if "language" in self.options:
return self.options["language"] # type: ignore[no-any-return]
if self.arguments:
return str(self.arguments[0])
return self.default_language
[docs]
def code_block_options(self, target: str = "source") -> list[str]:
"""Render the options supported by Sphinx' native ``code-block`` directive.
``target`` selects which block these options will be attached to:
``"source"`` for the directive's input source code, ``"results"`` for the
captured output. ``emphasize-lines`` routes to the source block;
``emphasize-result-lines`` is rewritten as ``emphasize-lines`` on the
results block, so authors can highlight different lines in each.
"""
options = []
for option_id in CodeBlock.option_spec:
if option_id == "emphasize-lines":
if target == "source" and "emphasize-lines" in self.options:
options.append(f":emphasize-lines: {self.options[option_id]}")
elif target == "results" and "emphasize-result-lines" in self.options:
options.append(
f":emphasize-lines: {self.options['emphasize-result-lines']}"
)
continue
if option_id in self.options:
value = self.options[option_id]
line = f":{option_id}:"
if value:
line += f" {value}"
options.append(line)
return options
[docs]
@cached_property
def show_source(self) -> bool:
"""Whether to show the source code of the example in the code block.
The last occurrence of either ``show-source`` or ``hide-source`` options
wins. If neither is set, the default is taken from ``show_source_by_default``.
"""
show_source = self.show_source_by_default
for option_id in self.options:
if option_id == "show-source":
show_source = True
elif option_id == "hide-source":
show_source = False
return show_source
[docs]
@cached_property
def show_results(self) -> bool:
"""Whether to show the results of running the example in the code block.
The last occurrence of either ``show-results`` or ``hide-results`` options
wins. If neither is set, the default is taken from ``show_results_by_default``.
"""
show_results = self.show_results_by_default
for option_id in self.options:
if option_id == "show-results":
show_results = True
elif option_id == "hide-results":
show_results = False
return show_results
[docs]
@cached_property
def is_myst_syntax(self) -> bool:
"""Check if the current directive is written with MyST syntax."""
return bool(self.state.__module__.split(".", 1)[0] == "myst_parser")
[docs]
def render_code_block(
self,
lines: Iterable[str],
language: str,
target: str = "source",
) -> list[str]:
"""Render the code block with the source code or results.
``target`` is forwarded to :meth:`code_block_options` so the
``emphasize-lines`` / ``emphasize-result-lines`` split routes the right
highlighting to each block.
"""
block: list[str] = []
if not lines:
return block
# Initiate the code block with with its MyST or rST syntax.
code_directive = "```{code-block}" if self.is_myst_syntax else ".. code-block::"
block.append(f"{code_directive} {language}")
# Re-attach each option to the code block.
# Indent the line in rST code block.
block.extend(
line if self.is_myst_syntax else RST_INDENT + line
for line in self.code_block_options(target)
)
# Both rST and MyST need a blank line before the body of the block else the
# first line will be interpreted as a directive option or argument.
block.append("")
block.extend(
line if self.is_myst_syntax else RST_INDENT + line for line in lines
)
# In MyST, we need to close the code block.
if self.is_myst_syntax:
block.append("```")
return block
[docs]
def run(self) -> list[nodes.Node]:
assert hasattr(self.runner, self.runner_method), (
f"{self.runner!r} does not have a method named {self.runner_method!r}."
)
runner_func = getattr(self.runner, self.runner_method)
results = runner_func(self)
# If neither source code nor results are requested, we don't render anything.
if not self.show_source and not self.show_results:
return []
lines = []
if self.show_source:
language = self.language
# If we are running a CLI, we force rendering the source code as a
# Python code block.
if self.runner_method == "run_cli":
language = SourceDirective.default_language
lines.extend(self.render_code_block(self.content, language, "source"))
if self.show_results:
lines.extend(self.render_code_block(results, self.language, "results"))
# Convert code block lines to a Docutils node tree.
# The section element is the main unit of hierarchy for Docutils documents.
section = nodes.section()
source_file, _line_number = self.get_source_info()
self.state.nested_parse(
StringList(lines, source_file),
# XXX Check that the offset here is properly computed in both rST and MyST.
self.content_offset,
section,
)
return section.children
[docs]
class SourceDirective(ClickDirective):
"""Directive to declare a Click CLI source code.
This directive is used to declare a Click CLI example in the
documentation. It renders the source code of the example in a
Python code block.
"""
default_language = "python"
show_source_by_default = True
show_results_by_default = False
runner_method = "execute_source"
[docs]
class RunDirective(ClickDirective):
"""Directive to run a Click CLI example.
This directive is used to run a Click CLI example in the
documentation. It renders the results of running the example in a
shell session code block supporting ANSI colors.
"""
default_language = "ansi-shell-session"
show_source_by_default = False
show_results_by_default = True
runner_method = "run_cli"
ClickDirective.runner_factory = ClickRunner
[docs]
class TreeDirective(ClickDirective):
"""Render a complete CLI reference for a Click command and all its subcommands.
Walks the Click command tree at build time and emits, in MyST syntax:
- A GFM summary table linking each command to its section anchor.
- A heading + ``click:run`` ``--help`` block for the root command.
- One heading + ``click:run`` ``--help`` block per subcommand, nested by
depth.
Designed to replace per-project hand-rolled generators (like repomatic's
``docs_update.py::cli_reference()``) with a single declarative directive
that walks the live command tree on every build.
The required argument is a Python expression evaluated in the per-document
runner namespace; it must yield a :class:`click.Command`. The optional
directive body is Python preamble exec'd in the same namespace before
evaluation, so authors may either import the CLI in a prior
``click:source :hide-source:`` block or inline the import here.
.. note::
Currently MyST-only. Use the directive in a ``.md`` document with
``myst_parser`` enabled.
"""
has_content = True
required_arguments = 1
optional_arguments = 0
final_argument_whitespace = False
option_spec: ClassVar[OptionSpec] = {
"max-depth": directives.positive_int,
"heading-offset": directives.nonnegative_int,
"anchor-prefix": directives.unchanged,
"label-prefix": directives.unchanged,
"root-label": directives.unchanged,
"no-table": directives.flag,
"no-root": directives.flag,
}
"""Recognized directive options.
``max-depth`` caps the recursion into nested :class:`click.Group` commands
(default: ``10``). ``heading-offset`` shifts all generated headings down
by N levels. When unset, the directive reads
``state.memo.section_level`` and uses the surrounding section depth so
the root nests one level below the enclosing section: inside the
document's ``h1`` title this yields ``1`` (root at ``h2``); inside an
``h3`` section it yields ``3`` (root at ``h4``). Override only when the
auto-detected depth is wrong for the page layout. ``anchor-prefix`` and
``label-prefix`` override the slug and display prefix used for anchors
and labels; both default to the CLI's :attr:`click.Command.name`.
``root-label`` sets the heading text for the root help block
(default: ``"Help screen"``). ``no-table`` skips the summary table;
``no-root`` skips the root ``--help`` block.
"""
# The runner_attr, runner property and is_myst_syntax cached-property are
# inherited unchanged from ClickDirective. Sharing the "click_runner"
# attribute means a click:source that ran earlier on the same document has
# already populated the namespace with the CLI variable this directive
# resolves.
@staticmethod
def _slug(value: str) -> str:
"""Lower-case + non-alphanumeric β ``-``, mirroring docutils' ``make_id``."""
return re.sub(r"[^a-z0-9]+", "-", value.lower()).strip("-")
def _surrounding_section_depth(self) -> int:
"""Return the heading level of the section wrapping this directive.
Drives the default :attr:`heading-offset` so generated headings nest
correctly under the surrounding section, regardless of how deep in
the document the directive is placed. A value of ``1`` means the
directive sits inside the document's top-level ``h1`` section (the
next legal heading is ``h2``); ``3`` means it sits inside an ``h3``
section (next legal heading is ``h4``).
Read from ``state.memo.section_level``, which docutils' ``RSTState``
and MyST's ``MockState`` both populate. Falls back to ``1`` if the
attribute is unavailable (preserves the historical default).
"""
try:
level = self.state.memo.section_level
except AttributeError:
return 1
return max(int(level), 1)
def _walk(
self,
root: click.Command,
max_depth: int,
) -> list[tuple[list[str], click.Command]]:
"""Depth-first traversal of the command tree, sorted alphabetically.
Returns ``(path, command)`` tuples where ``path`` is the list of
subcommand names from the root (exclusive) down to ``command``. The
root itself is not included; callers that want a root entry add it
separately (see :meth:`run`).
"""
entries: list[tuple[list[str], click.Command]] = []
def recurse(cmd: click.Command, path: list[str], depth: int) -> None:
if not isinstance(cmd, click.Group) or depth >= max_depth:
return
for name in sorted(cmd.commands):
sub_path = [*path, name]
entries.append((sub_path, cmd.commands[name]))
recurse(cmd.commands[name], sub_path, depth + 1)
recurse(root, [], 0)
return entries
[docs]
def run(self) -> list[nodes.Node]:
# Hard errors (RuntimeError, not self.error()) so the build fails
# fast: a partially rendered reference page hides bugs in the CLI
# tree the directive was meant to document.
if not self.is_myst_syntax:
raise RuntimeError(
"click:tree currently only supports MyST syntax. "
"Place the directive in a .md document with myst_parser enabled.",
)
# Execute the optional body in the runner namespace so callers can
# inline `from mypkg.cli import mycli` instead of seeding the
# namespace with a separate `click:source :hide-source:` block.
if self.content:
self.runner.execute_source(self)
cli_expr = self.arguments[0].strip()
try:
cli = eval(cli_expr, self.runner.namespace)
except Exception as exc:
raise RuntimeError(
f"click:tree: failed to evaluate {cli_expr!r}: {exc}",
) from exc
if not isinstance(cli, click.Command):
raise TypeError(
f"click:tree: {cli_expr!r} did not yield a click.Command "
f"(got {type(cli).__name__}).",
)
max_depth = self.options.get("max-depth", 10)
# Without an explicit override, nest the generated headings one
# level below the surrounding section so the document outline stays
# consistent regardless of where the directive is placed. At the
# document's top level this resolves to the historical default of 1
# (root rendered at h2 under a document title at h1).
heading_offset = self.options.get(
"heading-offset",
self._surrounding_section_depth(),
)
label_prefix = self.options.get("label-prefix") or cli.name or cli_expr
anchor_prefix = self.options.get("anchor-prefix") or self._slug(label_prefix)
root_label = self.options.get("root-label", "Help screen")
include_table = "no-table" not in self.options
include_root = "no-root" not in self.options
entries = self._walk(cli, max_depth)
# Local import to avoid a circular import: click_extra.table is part
# of the same package and pulls in optional rendering deps.
from ..table import TableFormat, render_table
lines: list[str] = []
# Summary table.
if include_table:
rows: list[list[str]] = []
if include_root:
desc = (cli.get_short_help_str() or "").rstrip(".")
rows.append([f"[`{label_prefix}`](#{anchor_prefix})", desc])
for path, cmd in entries:
label = f"{label_prefix} {' '.join(path)}".strip()
anchor = "-".join([anchor_prefix, *(self._slug(p) for p in path)])
desc = (cmd.get_short_help_str() or "").rstrip(".")
rows.append([f"[`{label}`](#{anchor})", desc])
if rows:
lines.append(
render_table(
rows,
headers=["Command", "Description"],
table_format=TableFormat.GITHUB,
),
)
lines.append("")
# Root help block. Placed at the same heading level as top-level
# commands so subcommands always nest one level deeper than their
# parent, matching the repomatic convention.
if include_root:
heading = "#" * (heading_offset + 1)
lines.append(f"({anchor_prefix})=")
lines.append(f"{heading} {root_label}")
lines.append("")
lines.append("```{click:run}")
lines.append(f"invoke({cli_expr}, args=['--help'])")
lines.append("```")
lines.append("")
# Per-command sections.
for path, _cmd in entries:
heading = "#" * (heading_offset + len(path))
anchor = "-".join([anchor_prefix, *(self._slug(p) for p in path)])
label = f"{label_prefix} {' '.join(path)}".strip()
args_repr = ", ".join(repr(a) for a in [*path, "--help"])
lines.append(f"({anchor})=")
lines.append(f"{heading} `{label}`")
lines.append("")
lines.append("```{click:run}")
lines.append(f"invoke({cli_expr}, args=[{args_repr}])")
lines.append("```")
lines.append("")
# Hand the generated MyST source back to the parser. Nested directives
# (`{click:run}`) execute during this pass and share the runner
# namespace, so the CLI variable resolves inside each generated block.
section = nodes.section()
source_file, _ = self.get_source_info()
self.state.nested_parse(
StringList(lines, source_file),
self.content_offset,
section,
)
return section.children
[docs]
class ClickDomain(StatelessDomain):
"""Setup new directives under the same ``click`` namespace:
- ``click:source`` which renders a Click CLI source code
- ``click:run`` which renders the results of running a Click CLI
- ``click:tree`` which walks a Click command tree and renders the full
``--help`` reference for every subcommand, with a summary table on top
"""
name = "click"
label = "Click"
directives: ClassVar[dict] = {
"source": SourceDirective,
"run": RunDirective,
"tree": TreeDirective,
}
cleanup_runner = make_cleanup("click_runner")
"""Drop the :class:`ClickRunner` from the doctree once the document is read."""