main.py 8.6 KB

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