123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166 |
- # 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)
|