Coverage for tsfpga/tools/sphinx_doc.py: 0%

76 statements  

« prev     ^ index     » next       coverage.py v7.6.10, created at 2025-01-21 20:51 +0000

1# -------------------------------------------------------------------------------------------------- 

2# Copyright (c) Lukas Vik. All rights reserved. 

3# 

4# This file is part of the tsfpga project, a project platform for modern FPGA development. 

5# https://tsfpga.com 

6# https://github.com/tsfpga/tsfpga 

7# -------------------------------------------------------------------------------------------------- 

8# A set of methods for building sphinx docs. Should be reusable between projects. 

9# -------------------------------------------------------------------------------------------------- 

10 

11# Standard libraries 

12import sys 

13from datetime import datetime 

14from pathlib import Path 

15from subprocess import CalledProcessError 

16from typing import Iterable, Optional 

17 

18# Third party libraries 

19from git.repo import Repo 

20from packaging.version import Version, parse 

21 

22# First party libraries 

23from tsfpga.system_utils import read_file, run_command 

24 

25 

26def generate_release_notes( 

27 repo_root: Path, release_notes_directory: Path, project_name: str 

28) -> str: 

29 """ 

30 Generate release notes in RST format based on a directory full of release note files. 

31 Will match each file to a git tag. 

32 

33 Arguments: 

34 repo_root: Git commands will be executed here. 

35 release_notes_directory: Location of release notes files. 

36 project_name: Name of project will be used for the GitHub link. 

37 

38 Return: 

39 RST code with release notes. 

40 """ 

41 rst = "" 

42 

43 for release, previous_release_git_tag in _get_release_notes_files( 

44 repo_root=repo_root, release_notes_directory=release_notes_directory 

45 ): 

46 heading = f"{release.version} ({release.date})" 

47 rst += heading + "\n" 

48 rst += "-" * len(heading) + "\n" 

49 rst += "\n" 

50 if previous_release_git_tag is not None: 

51 diff_url = ( 

52 f"https://github.com/{project_name}/{project_name}/compare/" 

53 f"{previous_release_git_tag}...{release.git_tag}" 

54 ) 

55 rst += f"`Changes since previous release <{diff_url}>`__\n" 

56 rst += "\n" 

57 rst += read_file(release.release_notes_file) 

58 rst += "\n" 

59 

60 return rst 

61 

62 

63def _get_release_notes_files( 

64 repo_root: Path, release_notes_directory: Path 

65) -> Iterable[tuple["Release", Optional[str]]]: 

66 """ 

67 Iterate the release notes. 

68 """ 

69 unreleased_notes_file = release_notes_directory / "unreleased.rst" 

70 

71 release_notes = [] 

72 

73 # Get all versioned release notes files and sort them in order newest -> oldest 

74 for release_notes_file in release_notes_directory.glob("*.rst"): 

75 if not release_notes_file == unreleased_notes_file: 

76 release_notes.append(release_notes_file) 

77 

78 # Sort by parsing the version number in the file name. Newest to oldest. 

79 def sort_key(path: Path) -> Version: 

80 return parse(path.stem) 

81 

82 release_notes.sort(key=sort_key, reverse=True) 

83 

84 # The "Unreleased" shall be first 

85 release_notes.insert(0, unreleased_notes_file) 

86 

87 repo = Repo(repo_root) 

88 releases = [ 

89 Release(repo=repo, release_notes_file=release_notes_file) 

90 for release_notes_file in release_notes 

91 ] 

92 

93 for idx, release in enumerate(releases): 

94 if idx == len(releases) - 1: 

95 previous_release_git_tag = None 

96 else: 

97 previous_release_git_tag = releases[idx + 1].git_tag 

98 

99 yield release, previous_release_git_tag 

100 

101 

102class Release: 

103 """ 

104 Used to represent a release. 

105 """ 

106 

107 def __init__(self, repo: Repo, release_notes_file: Path) -> None: 

108 self.release_notes_file = release_notes_file 

109 

110 version = release_notes_file.stem 

111 if version == "unreleased": 

112 self.version = "Unreleased" 

113 self.git_tag = "main" 

114 self.date = "YYYY-MM-DD" 

115 else: 

116 self.version = version 

117 self.git_tag = "v" + self.version 

118 self.date = self.get_git_date_from_tag(repo=repo, tag=self.git_tag) 

119 

120 @staticmethod 

121 def get_git_date_from_tag(repo: Repo, tag: str) -> str: 

122 """ 

123 Get a formatted date string, gathered from git log based on tag name. 

124 """ 

125 timestamp = repo.tag(f"refs/tags/{tag}").commit.committed_date 

126 time = datetime.fromtimestamp(timestamp) 

127 return f"{time.day} {time:%B} {time.year}".lower() 

128 

129 

130def build_sphinx(build_path: Path, output_path: Path) -> None: 

131 """ 

132 Execute sphinx on command line to build HTML documentation. 

133 

134 Arguments: 

135 build_path: The location that contains ``conf.py`` and ``index.rst``. 

136 output_path: Where to place the generated HTML. 

137 """ 

138 # Since we set the working directory when making the system call, paths must be absolute. 

139 build_path = build_path.resolve() 

140 output_path = output_path.resolve() 

141 

142 print("Building Sphinx documentation...") 

143 

144 cmd = [ 

145 sys.executable, 

146 "-m", 

147 "sphinx", 

148 # Turn warnings into errors. 

149 "-W", 

150 # Show full traceback upon error. 

151 "-T", 

152 str(build_path), 

153 str(output_path), 

154 ] 

155 try: 

156 run_command(cmd, cwd=build_path, capture_output=True) 

157 except CalledProcessError as exception: 

158 print(f"ERROR: Sphinx build command failed with exit code {exception.returncode}.") 

159 if exception.stdout: 

160 print("-" * 80) 

161 print("ERROR: Command STDOUT:\n") 

162 print(exception.stdout) 

163 if exception.stderr: 

164 print("-" * 80) 

165 print("ERROR: Command STDERR:\n") 

166 print(exception.stderr) 

167 raise exception 

168 

169 index_html = output_path / "index.html" 

170 assert index_html.exists(), index_html 

171 print(f"Documentation build done. Open with:\n firefox {index_html} &")