-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrelease-notes.py
executable file
·172 lines (155 loc) · 5.22 KB
/
release-notes.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
164
165
166
167
168
169
170
171
172
#!/usr/bin/env python3
import csv
import json
import os
import re
import requests
import subprocess
import sys
from collections import OrderedDict
# Usage: ./release-notes.py [repo] [previous release tag] [new release tag]
# You need to have the source code `git clone`d here
SOURCES_BASE_DIR = os.environ['SOURCES_BASE_DIR']
SOURCE_DIR = SOURCES_BASE_DIR.rstrip('/') + '/{repo}'
# Generate a token at https://github.com/settings/tokens
GITHUB_API_TOKEN = os.environ['GITHUB_API_TOKEN']
try:
sys.argv.remove('--markdown')
except ValueError:
markdown = False
else:
markdown = True
repo = sys.argv[1]
old_tag = sys.argv[2]
this_tag = sys.argv[3]
base_branch = None
#base_branch = 'master' # consider only PRs on this branch
commit_prs_endpoint = (
f'https://api.github.com/repos/kobotoolbox/{repo}/commits/{{commit}}/pulls'
)
headers = {'Accept': 'application/vnd.github.groot-preview+json'}
headers['Authorization'] = f'Token {GITHUB_API_TOKEN}'
substitutions = (
# Regular expressions based on Zulip's linkifiers
(
r'(?P<text>(?P<org>[a-zA-Z0-9_-]+)/(?P<repo>[a-zA-Z0-9_-]+)#(?P<id>[0-9]+))',
r'[\g<text>](https://github.com/\g<org>/\g<repo>/issues/\g<id>)',
),
(
r'(?P<text>(?P<repo>[a-zA-Z0-9_-]+)#(?P<id>[0-9]+))',
r'[\g<text>](https://github.com/kobotoolbox/\g<repo>/issues/\g<id>)',
),
(
r'\B(?P<text>#(?P<id>[0-9]+))',
r'[\g<text>](https://github.com/kobotoolbox/{repo}/issues/\g<id>)'.format(repo=repo),
),
)
latest_commit = (
subprocess.check_output(
['git', 'show', '--no-patch', '--pretty=format:%H on %ai'],
cwd=SOURCE_DIR.format(repo=repo),
)
.decode('utf-8')
.strip()
)
commits = (
subprocess.check_output(
['git', 'log', '--merges', '--pretty=format:%H', f'^{old_tag}', this_tag],
#['git', 'log', '--pretty=format:%H', f'^{old_tag}', this_tag],
cwd=SOURCE_DIR.format(repo=repo),
)
.decode('utf-8')
.split('\n')
)
if not commits[0] and len(commits) == 1:
sys.stderr.write('No commits were found. Bye!\n')
sys.exit(0)
print(f'<!-- as of commit {latest_commit} -->')
prs = OrderedDict()
for commit in commits:
resp = requests.get(
commit_prs_endpoint.format(commit=commit),
headers=headers,
)
resp.raise_for_status()
interesting_pr = None
for pr in resp.json():
if not pr['merged_at']:
continue
if base_branch and pr['base']['ref'] != base_branch:
continue
if interesting_pr:
sys.stderr.write(f'!!! Multiple interesting PRs for {commit}\n')
break
interesting_pr = pr
if interesting_pr:
prs[pr['number']] = interesting_pr
sys.stderr.write(f"Found PR {pr['number']} for merge {commit}\n")
else:
commit_description = subprocess.check_output(
['git', 'show', '--oneline', '--no-patch', commit],
cwd=SOURCE_DIR.format(repo=repo),
).decode('utf-8').strip()
sys.stderr.write(f"!!! No PR found for merge {commit_description}\n")
def condense_spaces(s): return re.sub(' +', ' ', s)
def remove_outer_blanks(l):
start = 0
end = len(l)
for i in l:
if i == '' and start < end:
start += 1
else:
break
for i in reversed(l):
if i == '' and end > start:
end -= 1
else:
break
return l[start:end]
def write_row(row, csv_writer=csv.writer(sys.stdout)):
if markdown:
print('|', ' | '.join(row), '|')
else:
csv_writer.writerow(row)
# disable "related issues" for now
# print('| PR | Description | Related Issues |')
# print('| - | - | - |')
write_row(['PR', 'Description'])
if markdown:
write_row(['-', '-'])
for number, details in prs.items():
row = []
row.append(f"[{repo}#{number}]({details['html_url']})")
description_lines = []
related_issues_lines = []
reading_state = None
body = details['body'] or ''
for line in body.replace('\r', '').split('\n'):
line = line.strip()
if line.startswith('## '):
reading_state = None
if reading_state is None:
standardized_line = condense_spaces(line).lower()
if standardized_line == '## description':
reading_state = 'description'
elif standardized_line == '## related issues':
reading_state = 'related issues'
continue
for pattern, repl in substitutions:
line, sub_count = re.subn(pattern, repl, line)
if sub_count:
# Process only the first matching pattern
break
if reading_state == 'description':
description_lines.append(line)
elif reading_state == 'related issues':
related_issues_lines.append(line)
description_lines = remove_outer_blanks(description_lines)
# always include the title; see
# https://chat.kobotoolbox.org/#narrow/stream/4-KoBo-Dev/topic/Change.20logs/near/9770
description_lines.insert(0, details['title'])
row.append('<br>'.join(description_lines))
# disable "related issues" for now
# row.append('<br>'.join(remove_outer_blanks(related_issues_lines)))
write_row(row)
#import IPython; IPython.embed()