main.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. # -*- coding: utf-8 -*-
  2. import itertools
  3. import argparse
  4. import json
  5. import logging
  6. import os
  7. import pathlib
  8. import re
  9. import string
  10. import subprocess
  11. import sys
  12. import tempfile
  13. from sphinx import config as sphinx_config
  14. from sphinx import project as sphinx_project
  15. from . import sphinx
  16. from . import git
  17. def main(argv=None):
  18. if not argv:
  19. argv = sys.argv[1:]
  20. parser = argparse.ArgumentParser()
  21. parser.add_argument("sourcedir", help="path to documentation source files")
  22. parser.add_argument("outputdir", help="path to output directory")
  23. parser.add_argument(
  24. "filenames",
  25. nargs="*",
  26. help="a list of specific files to rebuild. Ignored if -a is specified",
  27. )
  28. parser.add_argument(
  29. "-c",
  30. metavar="PATH",
  31. dest="confdir",
  32. help=(
  33. "path where configuration file (conf.py) is located "
  34. "(default: same as SOURCEDIR)"
  35. ),
  36. )
  37. parser.add_argument(
  38. "-C",
  39. action="store_true",
  40. dest="noconfig",
  41. help="use no config file at all, only -D options",
  42. )
  43. parser.add_argument(
  44. "-D",
  45. metavar="setting=value",
  46. action="append",
  47. dest="define",
  48. default=[],
  49. help="override a setting in configuration file",
  50. )
  51. parser.add_argument(
  52. "--dump-metadata",
  53. action="store_true",
  54. help="dump generated metadata and exit",
  55. )
  56. args, argv = parser.parse_known_args(argv)
  57. if args.noconfig:
  58. return 1
  59. sourcedir_absolute = os.path.abspath(args.sourcedir)
  60. confdir_absolute = (
  61. os.path.abspath(args.confdir)
  62. if args.confdir is not None
  63. else sourcedir_absolute
  64. )
  65. # Conf-overrides
  66. confoverrides = {}
  67. for d in args.define:
  68. key, _, value = d.partition("=")
  69. confoverrides[key] = value
  70. # Parse config
  71. config = sphinx_config.Config.read(confdir_absolute, confoverrides)
  72. config.add("smv_tag_whitelist", sphinx.DEFAULT_TAG_WHITELIST, "html", str)
  73. config.add(
  74. "smv_branch_whitelist", sphinx.DEFAULT_TAG_WHITELIST, "html", str
  75. )
  76. config.add(
  77. "smv_remote_whitelist", sphinx.DEFAULT_REMOTE_WHITELIST, "html", str
  78. )
  79. config.add(
  80. "smv_released_pattern", sphinx.DEFAULT_RELEASED_PATTERN, "html", str
  81. )
  82. config.add(
  83. "smv_outputdir_format", sphinx.DEFAULT_OUTPUTDIR_FORMAT, "html", str
  84. )
  85. config.add("smv_prefer_remote_refs", False, "html", bool)
  86. config.pre_init_values()
  87. config.init_values()
  88. # Get relative paths to root of git repository
  89. gitroot = pathlib.Path(".").resolve()
  90. sourcedir = os.path.relpath(sourcedir_absolute, str(gitroot))
  91. if args.confdir:
  92. confdir = os.path.relpath(confdir_absolute, str(gitroot))
  93. else:
  94. confdir = sourcedir
  95. conffile = os.path.join(confdir, "conf.py")
  96. # Get git references
  97. gitrefs = git.get_refs(
  98. str(gitroot),
  99. config.smv_tag_whitelist,
  100. config.smv_branch_whitelist,
  101. config.smv_remote_whitelist,
  102. files=(sourcedir, conffile),
  103. )
  104. # Order git refs
  105. if config.smv_prefer_remote_refs:
  106. gitrefs = sorted(gitrefs, key=lambda x: (not x.is_remote, *x))
  107. else:
  108. gitrefs = sorted(gitrefs, key=lambda x: (x.is_remote, *x))
  109. logger = logging.getLogger(__name__)
  110. with tempfile.TemporaryDirectory() as tmp:
  111. # Generate Metadata
  112. metadata = {}
  113. outputdirs = set()
  114. for gitref in gitrefs:
  115. # Clone Git repo
  116. repopath = os.path.join(tmp, gitref.commit)
  117. try:
  118. git.copy_tree(gitroot.as_uri(), repopath, gitref)
  119. except (OSError, subprocess.CalledProcessError):
  120. logger.error(
  121. "Failed to copy git tree for %s to %s",
  122. gitref.refname,
  123. repopath,
  124. )
  125. continue
  126. # Find config
  127. confpath = os.path.join(repopath, confdir)
  128. try:
  129. current_config = sphinx_config.Config.read(
  130. confpath, confoverrides,
  131. )
  132. except (OSError, sphinx_config.ConfigError):
  133. logger.error(
  134. "Failed load config for %s from %s",
  135. gitref.refname,
  136. confpath,
  137. )
  138. continue
  139. current_config.pre_init_values()
  140. current_config.init_values()
  141. # Ensure that there are not duplicate output dirs
  142. outputdir = config.smv_outputdir_format.format(
  143. ref=gitref, config=current_config,
  144. )
  145. if outputdir in outputdirs:
  146. logger.warning(
  147. "outputdir '%s' for %s conflicts with other versions",
  148. outputdir,
  149. gitref.refname,
  150. )
  151. continue
  152. outputdirs.add(outputdir)
  153. # Get List of files
  154. source_suffixes = current_config.source_suffix
  155. if isinstance(source_suffixes, str):
  156. source_suffixes = [current_config.source_suffix]
  157. current_sourcedir = os.path.join(repopath, sourcedir)
  158. project = sphinx_project.Project(
  159. current_sourcedir, source_suffixes
  160. )
  161. metadata[gitref.name] = {
  162. "name": gitref.name,
  163. "version": current_config.version,
  164. "release": current_config.release,
  165. "is_released": bool(
  166. re.match(config.smv_released_pattern, gitref.refname)
  167. ),
  168. "source": gitref.source,
  169. "creatordate": gitref.creatordate.strftime(sphinx.DATE_FMT),
  170. "sourcedir": current_sourcedir,
  171. "outputdir": os.path.join(
  172. os.path.abspath(args.outputdir), outputdir
  173. ),
  174. "confdir": confpath,
  175. "docnames": list(project.discover()),
  176. }
  177. if args.dump_metadata:
  178. print(json.dumps(metadata, indent=2))
  179. return 0
  180. if not metadata:
  181. logger.error("No matching refs found!")
  182. return 2
  183. # Write Metadata
  184. metadata_path = os.path.abspath(os.path.join(tmp, "versions.json"))
  185. with open(metadata_path, mode="w") as fp:
  186. json.dump(metadata, fp, indent=2)
  187. # Run Sphinx
  188. argv.extend(["-D", "smv_metadata_path={}".format(metadata_path)])
  189. for version_name, data in metadata.items():
  190. os.makedirs(data["outputdir"], exist_ok=True)
  191. defines = itertools.chain(
  192. *(
  193. ("-D", string.Template(d).safe_substitute(data))
  194. for d in args.define
  195. )
  196. )
  197. current_argv = argv.copy()
  198. current_argv.extend(
  199. [
  200. *defines,
  201. "-D",
  202. "smv_current_version={}".format(version_name),
  203. "-c",
  204. data["confdir"],
  205. data["sourcedir"],
  206. data["outputdir"],
  207. *args.filenames,
  208. ]
  209. )
  210. logger.debug("Running sphinx-build with args: %r", current_argv)
  211. cmd = (sys.executable, "-m", "sphinx", *current_argv)
  212. subprocess.check_call(cmd)
  213. return 0