|  | # D-Bus XML documentation extension | 
|  | # | 
|  | # Copyright (C) 2021, Red Hat Inc. | 
|  | # | 
|  | # SPDX-License-Identifier: LGPL-2.1-or-later | 
|  | # | 
|  | # Author: Marc-André Lureau <marcandre.lureau@redhat.com> | 
|  | """dbus-doc is a Sphinx extension that provides documentation from D-Bus XML.""" | 
|  |  | 
|  | import os | 
|  | import re | 
|  | from typing import ( | 
|  | TYPE_CHECKING, | 
|  | Any, | 
|  | Callable, | 
|  | Dict, | 
|  | Iterator, | 
|  | List, | 
|  | Optional, | 
|  | Sequence, | 
|  | Set, | 
|  | Tuple, | 
|  | Type, | 
|  | TypeVar, | 
|  | Union, | 
|  | ) | 
|  |  | 
|  | import sphinx | 
|  | from docutils import nodes | 
|  | from docutils.nodes import Element, Node | 
|  | from docutils.parsers.rst import Directive, directives | 
|  | from docutils.parsers.rst.states import RSTState | 
|  | from docutils.statemachine import StringList, ViewList | 
|  | from sphinx.application import Sphinx | 
|  | from sphinx.errors import ExtensionError | 
|  | from sphinx.util import logging | 
|  | from sphinx.util.docstrings import prepare_docstring | 
|  | from sphinx.util.docutils import SphinxDirective, switch_source_input | 
|  | from sphinx.util.nodes import nested_parse_with_titles | 
|  |  | 
|  | import dbusdomain | 
|  | from dbusparser import parse_dbus_xml | 
|  |  | 
|  | logger = logging.getLogger(__name__) | 
|  |  | 
|  | __version__ = "1.0" | 
|  |  | 
|  |  | 
|  | class DBusDoc: | 
|  | def __init__(self, sphinx_directive, dbusfile): | 
|  | self._cur_doc = None | 
|  | self._sphinx_directive = sphinx_directive | 
|  | self._dbusfile = dbusfile | 
|  | self._top_node = nodes.section() | 
|  | self.result = StringList() | 
|  | self.indent = "" | 
|  |  | 
|  | def add_line(self, line: str, *lineno: int) -> None: | 
|  | """Append one line of generated reST to the output.""" | 
|  | if line.strip():  # not a blank line | 
|  | self.result.append(self.indent + line, self._dbusfile, *lineno) | 
|  | else: | 
|  | self.result.append("", self._dbusfile, *lineno) | 
|  |  | 
|  | def add_method(self, method): | 
|  | self.add_line(f".. dbus:method:: {method.name}") | 
|  | self.add_line("") | 
|  | self.indent += "   " | 
|  | for arg in method.in_args: | 
|  | self.add_line(f":arg {arg.signature} {arg.name}: {arg.doc_string}") | 
|  | for arg in method.out_args: | 
|  | self.add_line(f":ret {arg.signature} {arg.name}: {arg.doc_string}") | 
|  | self.add_line("") | 
|  | for line in prepare_docstring("\n" + method.doc_string): | 
|  | self.add_line(line) | 
|  | self.indent = self.indent[:-3] | 
|  |  | 
|  | def add_signal(self, signal): | 
|  | self.add_line(f".. dbus:signal:: {signal.name}") | 
|  | self.add_line("") | 
|  | self.indent += "   " | 
|  | for arg in signal.args: | 
|  | self.add_line(f":arg {arg.signature} {arg.name}: {arg.doc_string}") | 
|  | self.add_line("") | 
|  | for line in prepare_docstring("\n" + signal.doc_string): | 
|  | self.add_line(line) | 
|  | self.indent = self.indent[:-3] | 
|  |  | 
|  | def add_property(self, prop): | 
|  | self.add_line(f".. dbus:property:: {prop.name}") | 
|  | self.indent += "   " | 
|  | self.add_line(f":type: {prop.signature}") | 
|  | access = {"read": "readonly", "write": "writeonly", "readwrite": "readwrite"}[ | 
|  | prop.access | 
|  | ] | 
|  | self.add_line(f":{access}:") | 
|  | if prop.emits_changed_signal: | 
|  | self.add_line(f":emits-changed: yes") | 
|  | self.add_line("") | 
|  | for line in prepare_docstring("\n" + prop.doc_string): | 
|  | self.add_line(line) | 
|  | self.indent = self.indent[:-3] | 
|  |  | 
|  | def add_interface(self, iface): | 
|  | self.add_line(f".. dbus:interface:: {iface.name}") | 
|  | self.add_line("") | 
|  | self.indent += "   " | 
|  | for line in prepare_docstring("\n" + iface.doc_string): | 
|  | self.add_line(line) | 
|  | for method in iface.methods: | 
|  | self.add_method(method) | 
|  | for sig in iface.signals: | 
|  | self.add_signal(sig) | 
|  | for prop in iface.properties: | 
|  | self.add_property(prop) | 
|  | self.indent = self.indent[:-3] | 
|  |  | 
|  |  | 
|  | def parse_generated_content(state: RSTState, content: StringList) -> List[Node]: | 
|  | """Parse a generated content by Documenter.""" | 
|  | with switch_source_input(state, content): | 
|  | node = nodes.paragraph() | 
|  | node.document = state.document | 
|  | state.nested_parse(content, 0, node) | 
|  |  | 
|  | return node.children | 
|  |  | 
|  |  | 
|  | class DBusDocDirective(SphinxDirective): | 
|  | """Extract documentation from the specified D-Bus XML file""" | 
|  |  | 
|  | has_content = True | 
|  | required_arguments = 1 | 
|  | optional_arguments = 0 | 
|  | final_argument_whitespace = True | 
|  |  | 
|  | def run(self): | 
|  | reporter = self.state.document.reporter | 
|  |  | 
|  | try: | 
|  | source, lineno = reporter.get_source_and_line(self.lineno)  # type: ignore | 
|  | except AttributeError: | 
|  | source, lineno = (None, None) | 
|  |  | 
|  | logger.debug("[dbusdoc] %s:%s: input:\n%s", source, lineno, self.block_text) | 
|  |  | 
|  | env = self.state.document.settings.env | 
|  | dbusfile = env.config.qapidoc_srctree + "/" + self.arguments[0] | 
|  | with open(dbusfile, "rb") as f: | 
|  | xml_data = f.read() | 
|  | xml = parse_dbus_xml(xml_data) | 
|  | doc = DBusDoc(self, dbusfile) | 
|  | for iface in xml: | 
|  | doc.add_interface(iface) | 
|  |  | 
|  | result = parse_generated_content(self.state, doc.result) | 
|  | return result | 
|  |  | 
|  |  | 
|  | def setup(app: Sphinx) -> Dict[str, Any]: | 
|  | """Register dbus-doc directive with Sphinx""" | 
|  | app.add_config_value("dbusdoc_srctree", None, "env") | 
|  | app.add_directive("dbus-doc", DBusDocDirective) | 
|  | dbusdomain.setup(app) | 
|  |  | 
|  | return dict(version=__version__, parallel_read_safe=True, parallel_write_safe=True) |