sphinxext.py 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. # Copyright 2018 Red Hat, Inc.
  2. # All Rights Reserved.
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License"); you may
  5. # not use this file except in compliance with the License. You may obtain
  6. # a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. # License for the specific language governing permissions and limitations
  14. # under the License.
  15. import os.path
  16. from six.moves import configparser
  17. from sphinx.util import logging
  18. import pbr.version
  19. _project = None
  20. logger = logging.getLogger(__name__)
  21. def _find_setup_cfg(srcdir):
  22. """Find the 'setup.cfg' file, if it exists.
  23. This assumes we're using 'doc/source' for documentation, but also allows
  24. for single level 'doc' paths.
  25. """
  26. # TODO(stephenfin): Are we sure that this will always exist, e.g. for
  27. # an sdist or wheel? Perhaps we should check for 'PKG-INFO' or
  28. # 'METADATA' files, a la 'pbr.packaging._get_version_from_pkg_metadata'
  29. for path in [
  30. os.path.join(srcdir, os.pardir, 'setup.cfg'),
  31. os.path.join(srcdir, os.pardir, os.pardir, 'setup.cfg')]:
  32. if os.path.exists(path):
  33. return path
  34. return None
  35. def _get_project_name(srcdir):
  36. """Return string name of project name, or None.
  37. This extracts metadata from 'setup.cfg'. We don't rely on
  38. distutils/setuptools as we don't want to actually install the package
  39. simply to build docs.
  40. """
  41. global _project
  42. if _project is None:
  43. parser = configparser.ConfigParser()
  44. path = _find_setup_cfg(srcdir)
  45. if not path or not parser.read(path):
  46. logger.info('Could not find a setup.cfg to extract project name '
  47. 'from')
  48. return None
  49. try:
  50. # for project name we use the name in setup.cfg, but if the
  51. # length is longer then 32 we use summary. Otherwise thAe
  52. # menu rendering looks brolen
  53. project = parser.get('metadata', 'name')
  54. if len(project.split()) == 1 and len(project) > 32:
  55. project = parser.get('metadata', 'summary')
  56. except configparser.Error:
  57. logger.info('Could not extract project metadata from setup.cfg')
  58. return None
  59. _project = project
  60. return _project
  61. def _builder_inited(app):
  62. # TODO(stephenfin): Once Sphinx 1.8 is released, we should move the below
  63. # to a 'config-inited' handler
  64. project_name = _get_project_name(app.srcdir)
  65. try:
  66. version_info = pbr.version.VersionInfo(project_name)
  67. except Exception:
  68. version_info = None
  69. if version_info and not app.config.version and not app.config.release:
  70. app.config.version = version_info.canonical_version_string()
  71. app.config.release = version_info.version_string_with_vcs()
  72. def setup(app):
  73. app.connect('builder-inited', _builder_inited)
  74. return {
  75. 'parallel_read_safe': True,
  76. 'parallel_write_safe': True,
  77. }