Coverage for src/mkdocstrings_handlers/shell/_internal/handler.py: 47.06%
51 statements
« prev ^ index » next coverage.py v7.7.1, created at 2025-03-26 22:03 +0100
« prev ^ index » next coverage.py v7.7.1, created at 2025-03-26 22:03 +0100
1# This module implements a handler for shell scripts and shell libraries.
3from __future__ import annotations
5from pathlib import Path
6from typing import TYPE_CHECKING, Any, ClassVar
8from mkdocs.exceptions import PluginError
9from mkdocstrings import BaseHandler, CollectionError, CollectorItem, get_logger
10from shellman import DocFile
11from shellman.templates.filters import FILTERS
13from mkdocstrings_handlers.shell._internal.config import ShellConfig, ShellOptions
15if TYPE_CHECKING:
16 from collections.abc import Mapping, MutableMapping
18 from mkdocs.config.defaults import MkDocsConfig
19 from mkdocstrings import HandlerOptions
22_logger = get_logger(__name__)
25class ShellHandler(BaseHandler):
26 """The Shell handler class."""
28 name: ClassVar[str] = "shell"
29 """The handler's name."""
31 domain: ClassVar[str] = "shell"
32 """The cross-documentation domain/language for this handler."""
34 enable_inventory: ClassVar[bool] = False
35 """Whether this handler is interested in enabling the creation of the `objects.inv` Sphinx inventory file."""
37 fallback_theme: ClassVar[str] = "material"
38 """The theme to fallback to."""
40 def __init__(self, config: ShellConfig, base_dir: Path, **kwargs: Any) -> None:
41 super().__init__(**kwargs)
42 self.config = config
43 """The handler configuration."""
44 self.base_dir = base_dir
45 """The base directory for the handler."""
46 self.global_options = config.options
47 """The global options for the handler."""
49 def get_options(self, local_options: Mapping[str, Any]) -> HandlerOptions:
50 """Get the combined (global and local) options."""
51 extra = {**self.global_options.get("extra", {}), **local_options.get("extra", {})}
52 options = {**self.global_options, **local_options, "extra": extra}
53 try:
54 return ShellOptions.from_data(**options)
55 except Exception as error:
56 raise PluginError(f"Invalid options: {error}") from error
58 def collect(self, identifier: str, options: ShellOptions) -> CollectorItem: # noqa: ARG002
59 """Collect data from a shell script/library."""
60 script_path = self.base_dir / identifier
61 try:
62 return DocFile(str(script_path))
63 except FileNotFoundError as error:
64 raise CollectionError(f"Could not find script '{script_path}'") from error
66 def render(self, data: CollectorItem, options: ShellOptions) -> str:
67 """Render the collected data."""
68 heading_level = options.heading_level
69 template = self.env.get_template("script.html.jinja")
70 return template.render(
71 config=options,
72 filename=data.filename,
73 script=data.sections,
74 heading_level=heading_level,
75 )
77 def update_env(self, config: MkDocsConfig) -> None: # noqa: ARG002
78 """Update the Jinja environment."""
79 self.env.trim_blocks = True
80 self.env.lstrip_blocks = True
81 self.env.keep_trailing_newline = False
82 self.env.filters.update(FILTERS)
85def get_handler(
86 *,
87 handler_config: MutableMapping[str, Any],
88 tool_config: MkDocsConfig,
89 **kwargs: Any,
90) -> ShellHandler:
91 """Simply return an instance of `ShellHandler`.
93 Parameters:
94 handler_config: The handler configuration.
95 tool_config: The tool (SSG) configuration.
96 **kwargs: Keyword arguments for the base handler constructor.
98 Returns:
99 An instance of the handler.
100 """
101 base_dir = Path(tool_config.config_file_path or "./mkdocs.yml").parent
102 return ShellHandler(config=ShellConfig(**handler_config), base_dir=base_dir, **kwargs)