-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathrelease.py
112 lines (91 loc) · 3.53 KB
/
release.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
#!/usr/bin/env python3
import subprocess
DEFAULT_NOTES = 'release-notes.md.tmpl'
class SemanticVersion:
"""
A class that handles semantic version. Instantiate with a valid string version.
"""
def __init__(self, major: int, minor: int, patch: int):
self.major = major
self.minor = minor
self.patch = patch
def __str__(self):
return f"v{self.major}.{self.minor}.{self.patch}"
def bump_major(self):
self.major += 1
def bump_minor(self):
self.minor += 1
def bump_patch(self):
self.patch += 1
@classmethod
def from_string(cls, version: str):
try:
major, minor, patch = map(int, version.lstrip('v').split('.', 2))
return cls(major, minor, patch)
except Exception as err:
raise Exception(f"Error parsing '{version}' as semantic version: {err}")
def extract_tag():
tmp = subprocess.run(["git", "describe", "--tags", "--exact"], capture_output=True)
if tmp.returncode == 0:
raise Exception(f"There is a tag pointing to current commit: {tmp.stdout.decode().rstrip()}")
latest_tag = subprocess.run(["git", "describe", "--tags", "--abbrev=0"], capture_output=True)
return latest_tag.stdout.rstrip().decode()
def mk_release(ver: SemanticVersion, notes_file: str, dry_run: bool):
notes_params = ['-F', notes_file] if notes_file else ['-F', DEFAULT_NOTES]
cmd = ["gh", "release", 'create', '-n', ''] + notes_params + ['-t', f"Release: {ver}", f"{ver}"]
if dry_run:
print(f"DRY RUN: would create release with version: {ver}")
print(f"DRY RUN: would run: {cmd}")
else:
subprocess.run(cmd)
print("")
print('A new release + tag were created on github. Please make sure to pull the changes locally '
'and verify that docker image is building.')
if not notes_file:
print('')
print('No notes file provided. Opening release in web browser to edit...')
input('Print ENTER to exit ')
subprocess.run(['gh', 'release', 'view', '-w', str(ver)])
def run(args):
tag = extract_tag()
ver = SemanticVersion.from_string(tag)
match args.level:
case 'major':
ver.bump_major()
case 'minor':
ver.bump_minor()
case 'patch':
ver.bump_patch()
case _:
raise Exception("No level supplied")
mk_release(ver, args.notes, args.dry)
if __name__ == "__main__":
import sys
import argparse
parser = argparse.ArgumentParser(
prog="Release",
description='Bump version (tag) according to \'level\' and create github release. '
f"Copy and edit '{DEFAULT_NOTES}' to add release notes, otherwise we will open a browser to "
'manually edit.',
epilog="IMPORTANT: This script assumes you have both 'git' and 'gh' "
"configured and running correctly on your machine",
)
parser.add_argument('level', choices=['major', 'minor', 'patch'])
parser.add_argument(
'--dry-run',
help='Do not create release + tag on github, Just print the version that would have been created',
action=argparse.BooleanOptionalAction,
dest='dry',
default=False
)
parser.add_argument(
'-F', '--notes-file',
help='Markdown file containing release notes',
dest='notes'
)
args = parser.parse_args()
try:
run(args)
except Exception as err:
print(err, file=sys.stderr)
sys.exit(1)