/sphinx/domains/changeset.py

https://github.com/sphinx-doc/sphinx · Python · 155 lines · 110 code · 29 blank · 16 comment · 14 complexity · 27dfc6c6db34b65858da08e0c02059f7 MD5 · raw file

  1. """
  2. sphinx.domains.changeset
  3. ~~~~~~~~~~~~~~~~~~~~~~~~
  4. The changeset domain.
  5. :copyright: Copyright 2007-2020 by the Sphinx team, see AUTHORS.
  6. :license: BSD, see LICENSE for details.
  7. """
  8. from collections import namedtuple
  9. from typing import Any, Dict, List
  10. from typing import cast
  11. from docutils import nodes
  12. from docutils.nodes import Node
  13. from sphinx import addnodes
  14. from sphinx.domains import Domain
  15. from sphinx.locale import _
  16. from sphinx.util.docutils import SphinxDirective
  17. if False:
  18. # For type annotation
  19. from sphinx.application import Sphinx
  20. from sphinx.environment import BuildEnvironment
  21. versionlabels = {
  22. 'versionadded': _('New in version %s'),
  23. 'versionchanged': _('Changed in version %s'),
  24. 'deprecated': _('Deprecated since version %s'),
  25. }
  26. versionlabel_classes = {
  27. 'versionadded': 'added',
  28. 'versionchanged': 'changed',
  29. 'deprecated': 'deprecated',
  30. }
  31. # TODO: move to typing.NamedTuple after dropping py35 support (see #5958)
  32. ChangeSet = namedtuple('ChangeSet',
  33. ['type', 'docname', 'lineno', 'module', 'descname', 'content'])
  34. class VersionChange(SphinxDirective):
  35. """
  36. Directive to describe a change/addition/deprecation in a specific version.
  37. """
  38. has_content = True
  39. required_arguments = 1
  40. optional_arguments = 1
  41. final_argument_whitespace = True
  42. option_spec = {} # type: Dict
  43. def run(self) -> List[Node]:
  44. node = addnodes.versionmodified()
  45. node.document = self.state.document
  46. self.set_source_info(node)
  47. node['type'] = self.name
  48. node['version'] = self.arguments[0]
  49. text = versionlabels[self.name] % self.arguments[0]
  50. if len(self.arguments) == 2:
  51. inodes, messages = self.state.inline_text(self.arguments[1],
  52. self.lineno + 1)
  53. para = nodes.paragraph(self.arguments[1], '', *inodes, translatable=False)
  54. self.set_source_info(para)
  55. node.append(para)
  56. else:
  57. messages = []
  58. if self.content:
  59. self.state.nested_parse(self.content, self.content_offset, node)
  60. classes = ['versionmodified', versionlabel_classes[self.name]]
  61. if len(node):
  62. if isinstance(node[0], nodes.paragraph) and node[0].rawsource:
  63. content = nodes.inline(node[0].rawsource, translatable=True)
  64. content.source = node[0].source
  65. content.line = node[0].line
  66. content += node[0].children
  67. node[0].replace_self(nodes.paragraph('', '', content, translatable=False))
  68. para = cast(nodes.paragraph, node[0])
  69. para.insert(0, nodes.inline('', '%s: ' % text, classes=classes))
  70. else:
  71. para = nodes.paragraph('', '',
  72. nodes.inline('', '%s.' % text,
  73. classes=classes),
  74. translatable=False)
  75. node.append(para)
  76. domain = cast(ChangeSetDomain, self.env.get_domain('changeset'))
  77. domain.note_changeset(node)
  78. ret = [node] # type: List[Node]
  79. ret += messages
  80. return ret
  81. class ChangeSetDomain(Domain):
  82. """Domain for changesets."""
  83. name = 'changeset'
  84. label = 'changeset'
  85. initial_data = {
  86. 'changes': {}, # version -> list of ChangeSet
  87. } # type: Dict
  88. @property
  89. def changesets(self) -> Dict[str, List[ChangeSet]]:
  90. return self.data.setdefault('changes', {}) # version -> list of ChangeSet
  91. def note_changeset(self, node: addnodes.versionmodified) -> None:
  92. version = node['version']
  93. module = self.env.ref_context.get('py:module')
  94. objname = self.env.temp_data.get('object')
  95. changeset = ChangeSet(node['type'], self.env.docname, node.line,
  96. module, objname, node.astext())
  97. self.changesets.setdefault(version, []).append(changeset)
  98. def clear_doc(self, docname: str) -> None:
  99. for version, changes in self.changesets.items():
  100. for changeset in changes[:]:
  101. if changeset.docname == docname:
  102. changes.remove(changeset)
  103. def merge_domaindata(self, docnames: List[str], otherdata: Dict) -> None:
  104. # XXX duplicates?
  105. for version, otherchanges in otherdata['changes'].items():
  106. changes = self.changesets.setdefault(version, [])
  107. for changeset in otherchanges:
  108. if changeset.docname in docnames:
  109. changes.append(changeset)
  110. def process_doc(self, env: "BuildEnvironment", docname: str, document: nodes.document) -> None: # NOQA
  111. pass # nothing to do here. All changesets are registered on calling directive.
  112. def get_changesets_for(self, version: str) -> List[ChangeSet]:
  113. return self.changesets.get(version, [])
  114. def setup(app: "Sphinx") -> Dict[str, Any]:
  115. app.add_domain(ChangeSetDomain)
  116. app.add_directive('deprecated', VersionChange)
  117. app.add_directive('versionadded', VersionChange)
  118. app.add_directive('versionchanged', VersionChange)
  119. return {
  120. 'version': 'builtin',
  121. 'env_version': 1,
  122. 'parallel_read_safe': True,
  123. 'parallel_write_safe': True,
  124. }