-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathversion_check.py
executable file
·163 lines (127 loc) · 6.29 KB
/
version_check.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
#!/usr/bin/env python3
import sys
import argparse
import fnmatch
from typing import Optional, Dict
import mistune
import CppHeaderParser
import semver
import git
parser = argparse.ArgumentParser(description='''Check that the versioning of the repo has been correctly changed.
Intended to run in a PR CI''')
parser.add_argument('branchspec', default='origin/develop', nargs='?',
help='Git branch specification for previous file versions (default: %(default)s)')
def die(*args, error_code=1, **kwargs):
print('ERROR: ', end='', file=sys.stderr)
print(*args, **kwargs, file=sys.stderr)
exit(error_code)
class SemVerWithVPrefix(semver.VersionInfo):
@classmethod
def parse(cls, version):
if version[0] != 'V':
raise ValueError(f"{version}: not a valid semantic version tag. Must start with 'v' or 'V'")
self = super(SemVerWithVPrefix, cls).parse(version[1:])
return self
def __str__(self):
return 'V' + super(SemVerWithVPrefix, self).__str__()
def only_changed_non_fw_files(refspec: str) -> bool:
print('Checking if the files changed impact the firmware...')
repo = git.Repo('.')
index = repo.index
origin_develop_commit = repo.commit(refspec)
# index vs working copy (unstaged files)
changed_but_not_staged = [item.a_path for item in index.diff(None)]
# index vs current HEAD tree (staged files)
changed_and_staged = [item.a_path for item in index.diff('HEAD')]
# origin vs current HEAD tree (files changed in the branch)
branch_files_changed = [item.a_path for item in origin_develop_commit.diff('HEAD')]
print(f'Unstaged changes: {changed_but_not_staged}')
print(f'Staged changes: {changed_and_staged}')
print(f'Changed in branch: {branch_files_changed}')
all_changed_filenames = changed_but_not_staged + changed_and_staged + branch_files_changed
print(f'Total changed files: {all_changed_filenames}')
non_fw_globs = [
# non-build Python scripts (pre/post Platformio scripts still affect the build process)
'matrix_build*.py', 'version_check.py', 'scripts/MeadeCommandParser.py',
# prose text files
'*.md', 'LICENSE', '.mailmap',
# build system
'.github/*.yml', 'requirements_*.txt',
# misc tooling files
'.gitignore', '.git-blame-ignore-revs',
]
for changed_filename in all_changed_filenames:
if not any(fnmatch.fnmatch(changed_filename, non_fw_glob) for non_fw_glob in non_fw_globs):
# If the changed file isn't a non-FW file (==> is a FW file)
# then we need to do the full version check
print(f"Changed file {changed_filename} will affect firmware ({non_fw_globs})")
return False
return True
def get_header_defines(header_contents: str) -> Dict[str, str]:
defines_detail = CppHeaderParser.CppHeader(header_contents, argType='string').defines_detail
return dict(d['value'].split(' ', maxsplit=1) for d in defines_detail)
def find_child_text(node: dict) -> Optional[str]:
if 'children' in node.keys():
if node['children'] and isinstance(node['children'], list):
return find_child_text(node['children'][0])
elif 'text' in node.keys():
return node['text']
return None
def get_header_version(version_header_contents: str) -> semver.VersionInfo:
version_defines = get_header_defines(version_header_contents)
try:
version_str = next(v.strip('"') for k, v in version_defines.items() if 'version' in k.lower())
except StopIteration:
die(f'Could not find version define in\n{version_header_contents}: {version_defines}')
try:
version = SemVerWithVPrefix.parse(version_str)
except ValueError as e:
die(f'SemVer error: {e}')
print(f'Version is {version}')
return version
def get_changelog_version(changelog_markdown_path: str) -> str:
markdown = mistune.create_markdown(renderer=mistune.AstRenderer())
with open(changelog_markdown_path, 'r') as f:
changelog_lines = f.read()
changelog_ast = markdown(changelog_lines)
first_heading_has_list = (changelog_ast[1]['type'] == 'list')
if not first_heading_has_list:
die(f'Latest changelog entry does not include a _list_ of what has changed')
first_heading_str = find_child_text(changelog_ast[0])
if not first_heading_str:
die(f'Could not find the latest changelog heading: {changelog_ast[0]}')
print(f'Latest changelog heading: {first_heading_str}')
return first_heading_str
def get_previous_header_version(version_header_path: str, refspec: str) -> semver.VersionInfo:
repo = git.Repo('.')
previous_header_contents = repo.git.show(f'{refspec}:{version_header_path}')
print(f'Checking previous header version from {refspec}:{version_header_path}')
previous_version = get_header_version(previous_header_contents)
print(f'Read previous header version: {previous_version}')
return previous_version
def main():
if only_changed_non_fw_files(args.branchspec):
print('No FW files changed, not checking version.')
return
version_header_path = './Version.h'
# Get the current version from the header
print(f'Checking current header version from {version_header_path}')
with open(version_header_path, 'r') as fp:
version_header_contents = fp.read()
header_version = get_header_version(version_header_contents)
# Get the previous version (using git)
previous_header_version = get_previous_header_version(version_header_path, args.branchspec)
# Check that the version has been increased
if header_version <= previous_header_version:
die(f'Header version must be greater than the previous version {header_version} <= {previous_header_version}')
changelog_markdown_path = './Changelog.md'
# Get a string from the changelog, should have a version in the latest entry
changelog_version = get_changelog_version(changelog_markdown_path)
# Check that the header and the changelog match
if str(header_version).lower() not in changelog_version.lower():
die(f'Header version ("{header_version}" in {version_header_path}) does \
not match changelog version ("{changelog_version}" in {changelog_markdown_path})')
if __name__ == '__main__':
args = parser.parse_args()
main()
print('Finished version check!')