forked from kubernetes/release
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrelnotes
executable file
·536 lines (468 loc) · 17.4 KB
/
relnotes
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
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
#!/bin/bash
#
# Copyright 2016 The Kubernetes Authors All rights reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# Set PROGram name
PROG=${0##*/}
########################################################################
#+
#+ NAME
#+ $PROG - Generate release notes for K8S releases
#+
#+ SYNOPSIS
#+ $PROG [--quiet] [[starttag..]endtag] [--htmlize-md] [--full]
#+ [--release-tars=/path/to/release-tars]
#+ [--github-token=<token>] [--branch=<branch>]
#+ [--markdown-file=<file>] [--html-file=<file>]
#+ [--release-bucket=<gs bucket>] [--preview]
#+ $PROG [--helpshort|--usage|-?]
#+ $PROG [--help|-man]
#+
#+ DESCRIPTION
#+ $PROG scans 'git log' for 'merge pull's and collects and displays
#+ release notes based on release-note-* labels from PR titles and PR
#+ release-note blocks (in the body).
#+
#+ By default, $PROG produces release notes for the last github release
#+ to the HEAD of current branch and uses a standard git range otherwise.
#+ You may omit the end of the range to terminate the range at the HEAD
#+ of the current branch.
#+
#+ The default output is pure markdown and unless [--markdown-file=] is
#+ specified, the output file is in $TMPDIR/$PROG-release-notes.md.
#+ If [--html-file=] is set, $PROG will also produce a pure html version
#+ of the notes at that location.
#+
#+ If [--quiet] is not specified, the output to stdout will always be
#+ only the markdown version of the output.
#+
#+ [--branch=] is used to specify a branch other than the current one.
#+
#+ Other options detailed in EXAMPLES below.
#+
#+ OPTIONS
#+ --quiet - Don't display the notes when done
#+ --htmlize-md - Output markdown with html for PRs and
#+ contributors (for use in CHANGELOG-x.y.md)
#+ --full - Force 'full' release format to show all
#+ sections of release notes. (This is the
#+ *default* for new branch X.Y.0 notes)
#+ --release-tars= - directory of tars to sha256 sum for display
#+ --markdown-file= - Specify an alt file to use to store notes
#+ --html-file= - Produce a html version of the notes
#+ --release-bucket= - Specify gs bucket to point to in
#+ generated notes (informational only)
#+ --preview - Report additional branch statistics (used for
#+ reporting outside of releases)
#+ --github-token= - Must be specified if GITHUB_TOKEN not set
#+ --branch= - Specify a branch other than the current one
#+ [--help | -man] - Display man page for this script
#+ [--usage | -?] - Display in-line usage
#+
#+ EXAMPLES
#+ $PROG - Notes for last release to HEAD
#+ on current branch
#+ $PROG v1.1.4.. - Notes for v1.1.4 to HEAD on current branch
#+ $PROG v1.1.4..v1.1.7 - Notes for v1.1.4..v1.1.7
#+ $PROG v1.1.7 - Notes for last release
#+ on current branch to v1.1.7
#+
#+ FILES
#+ $TMPDIR/$PROG-release-htmls.md
#+ $TMPDIR/$PROG-release-htmls.html
#+
#+ SEE ALSO
#+ common.sh - Base function definitions
#+ gitlib.sh - git-related function definitions
#+ https://stedolan.github.io/jq - JSON CLI
#+
#+ BUGS/TODO
#+
########################################################################
# If NO ARGUMENTS should return *usage*, uncomment the following line:
#usage=${1-yes}
source $(dirname $(readlink -ne $BASH_SOURCE))/lib/common.sh
source $TOOL_LIB_PATH/gitlib.sh
# Validate command-line
common::argc_validate 1
[[ -n $FLAGS_release_tars && ! -d $FLAGS_release_tars ]] \
&& common::exit 1 "--release-tars=$FLAGS_release_tars doesn't exist! Exiting..."
###############################################################################
# FUNCTIONS
###############################################################################
###############################################################################
# Get titles from a list of PRs
# @param prs - A space separated list of PRs to extract
#
extract_pr_title () {
local prs="$*"
local pr
local content
local body
local author
local pull_json
for pr in $prs; do
pull_json="$($GHCURL $K8S_GITHUB_API/pulls/$pr)"
[[ -z "$pull_json" ]] && return 1
body="$(echo "$pull_json" |jq -r '.body' |tr -d '\r')"
# Look for a body release note first and default to title
# * indent lines >1
# Try to account for and adjust user-entered formatting
# * This is somewhat complicated by the fact that we convert this to
# html, for things like email, group posts, dashboards, and there
# is a disconnect between pandoc and github's markdown for lists.
# https://github.com/jgm/pandoc/issues/2210
content=$(echo "$body" |\
sed -n '/^``` *release-note/,/^```/{/^```/!p;/^```$/q}' |\
sed -e '/^$/d' -e '2,$s/^\( *\* \)/ \1/g' \
-e '2,$s/^\( *[^ \*]\)/ * \1/g')
# if the release-note block is empty (unchanged from the template), use title
if [[ -z "$content" ]]; then
content=$(echo "$pull_json" | jq -r '.title')
fi
author=$(echo "$pull_json" | jq -r '.user.login')
content=$(echo "$content" |sed -e '1s/^ *\** */* /g' \
-e "1s/$/ (#$pr, @$author)/g")
logecho -r "$content"
done
}
###############################################################################
# Get merged PRs that match a specified label
# @param label - A label(string) to use for searching github PRs
get_prs_by_label () {
local label="$1"
local matching_prs
local -a prs
local total
local current_page
local num_pages
local prs_per_page
prs_per_page=100
matching_prs="$($GHCURL ${K8S_GITHUB_SEARCHAPI}label:${label}&per_page=1)"
total="$(echo -n $matching_prs | jq -r '.total_count')"
# Calculate number of pages, rounding up
num_pages=$(((total + prs_per_page - 1) / prs_per_page ))
for current_page in `seq 1 $num_pages`; do
prs+=($($GHCURL "${K8S_GITHUB_SEARCHAPI}label:${label}&page=$current_page" | jq -r '.items[] | (.number | tostring)'))
done
echo "${prs[@]}"
}
# Create a markdown dtable for the specified tarballs on GCS
# @param heading - level-3 heading to print before the table
# @param @ - local path to tarballs to list
#
create_downloads_table () {
local heading=$1
shift
local url_prefix
if [[ "$release_bucket" == "kubernetes-release" ]]; then
url_prefix="https://dl.k8s.io"
else
url_prefix="https://storage.googleapis.com/$release_bucket/release"
fi
[[ -n "$heading" ]] && echo "### $heading"
echo
echo "filename | sha256 hash"
echo "-------- | -----------"
for file in $@; do
echo "[${file##*/}]($url_prefix/$release_tag/${file##*/}) | \`$(common::sha $file 256)\`"
done
echo
}
###############################################################################
# Create the release note markdown body
# @param release_tars - A directory containing tarballs to link to on GCS
# @param start_tag - The start tag of range
# @param release_tag - The release tag of range
#
create_body () {
local release_tars=$1
local start_tag=$2
local release_tag=$3
local release_bucket=${FLAGS_release_bucket:-"kubernetes-release"}
local title
((FLAGS_preview)) && title="Branch "
# Show a more useful header if release_tag == HEAD
if [[ "$release_tag" == "HEAD" ]]; then
title+=$CURRENT_BRANCH
else
title+=$release_tag
fi
((FLAGS_preview)) && echo "**Release Note Preview - generated on $(date)**"
echo
echo "# $title"
echo
echo "[Documentation](https://docs.k8s.io) &" \
"[Examples](https://releases.k8s.io/$CURRENT_BRANCH/examples)"
echo
if [[ -n $release_tars ]]; then
echo "## Downloads for $title"
echo
create_downloads_table "" $release_tars/kubernetes{,-src}.tar.gz
create_downloads_table "Client Binaries" ${release_tars}/kubernetes-client*.tar.gz
create_downloads_table "Server Binaries" ${release_tars}/kubernetes-server*.tar.gz
create_downloads_table "Node Binaries" ${release_tars}/kubernetes-node*.tar.gz
fi
cat $PR_NOTES
}
###############################################################################
# CI job status
# Uses global CURRENT_BRANCH
ci_job_status () {
local content
local red=${TPUT[RED]}
local green=${TPUT[GREEN]}
local off=${TPUT[OFF]}
local extra_flag
if ((FLAGS_htmlize_md)); then
red="<FONT COLOR=RED>"
green="<FONT COLOR=GREEN>"
off="</FONT>"
fi
# If working on a release branch assume --official for the
# purpose of displaying find_green_build output
if [[ $CURRENT_BRANCH =~ release- ]]; then
extra_flag="--official"
else
# For master branch, limit the analysis to 30 primary ci jobs.
# This is necessary due to the recently expanded blocking test list for
# master. The expanded test list is often unable to find a complete
# passing set and find_green_build runs unbounded for hours.
extra_flag="--limit=30"
fi
# State of tree
echo
echo "## State of $CURRENT_BRANCH branch"
if content=$(find_green_build -v $extra_flag $CURRENT_BRANCH); then
echo "${green}GOOD TO GO!$off"
else
echo "${red}NOT READY$off"
fi
echo
echo "### Details"
echo '```'
echo "$content"
echo '```'
}
###############################################################################
# Scan PRs for release-note-* labels and generate markdown for the actual
# release notes section of the report
# Uses global LAST_RELEASE CURRENT_BRANCH
generate_notes () {
local branch_head
local range
local start_tag
local release_tag
local labels
local body
local tempcss=$TMPDIR/$PROG-ca.$$
local tempfile=$TMPDIR/$PROG-file-1.$$
local anchor
local changelog_file
local -a normal_prs
local -a action_prs
local -a notes_normal
local -a notes_action
local -a prs
branch_head=$(git rev-parse refs/remotes/origin/$CURRENT_BRANCH 2>/dev/null)
# If ${LAST_RELEASE[$CURRENT_BRANCH]} is unset attempt to get the last
# release from the parent branch and then master
: ${LAST_RELEASE[$CURRENT_BRANCH]:=${LAST_RELEASE[${CURRENT_BRANCH%.*}]}}
: ${LAST_RELEASE[$CURRENT_BRANCH]:=${LAST_RELEASE[master]}}
# Default
range="${POSITIONAL_ARGV[0]:-"${LAST_RELEASE[$CURRENT_BRANCH]}..$branch_head"}"
if [[ "${POSITIONAL_ARGV[0]}" =~ ([v0-9.]*-*(alpha|beta|rc)*\.*[0-9]*)\.\.([v0-9.]*-*(alpha|beta|rc)*\.*[0-9]*)$ ]]; then
start_tag=${BASH_REMATCH[1]}
release_tag=${BASH_REMATCH[3]}
else
start_tag="${LAST_RELEASE[$CURRENT_BRANCH]}"
release_tag=${POSITIONAL_ARGV[0]}
fi
# Describe/tag $branch_head if no release_tag set
: ${release_tag:=$(git describe $branch_head)}
if [[ -z "$start_tag" ]]; then
common::exit 1 "Unable to set beginning of range automatically." \
"Specify on the command-line. Exiting..."
fi
range="$start_tag..$release_tag"
if [[ $release_tag =~ ${VER_REGEX[release]} ]]; then
changelog_file="CHANGELOG-${BASH_REMATCH[1]}.${BASH_REMATCH[2]}.md"
else
logecho "Unable to set CHANGELOG file!"
return 1
fi
# If range is unterminated, finish it with $branch_head
[[ $range =~ \.\.$ ]] && range+=$branch_head
# Validate range
if ! git rev-parse $range &>/dev/null; then
logecho
logecho "Invalid tags/range $range !"
return 1
fi
# Deref all the PRs back to master, paying special attention to
# automated cherrypicks that could have multiple sources
while read line; do
if [[ "$line" =~ automated-cherry-pick-of-(#[0-9]+-){1,} ]]; then
prs+=($(echo "${BASH_REMATCH[0]}" | egrep -o "#[0-9]*" |tr -d '#'))
else
prs+=($(echo "$line" | awk '{gsub(/#/,"");print $4 }'))
fi
done < <(git log $range --format="%s" --grep="Merge pull")
logecho
echo "Scanning action required PR labels on the $CURRENT_BRANCH branch..."
action_prs=($(get_prs_by_label release-note-breaking-change))
action_prs+=($(get_prs_by_label release-note-action-required))
echo "Scanning release-note PR label on the $CURRENT_BRANCH branch..."
normal_prs=($(get_prs_by_label release-note))
for pr in ${prs[*]}; do
if [[ " ${action_prs[@]} " =~ " ${pr} " ]]; then
notes_action+=("$pr")
elif [[ " ${normal_prs[@]} " =~ " ${pr} " ]]; then
notes_normal+=("$pr")
fi
done
logecho
logecho "Generating release notes..."
# Bootstrap notes for major (new branch) releases
if ((FLAGS_full)) || [[ $release_tag =~ ${VER_REGEX[dotzero]} ]]; then
# Check for draft and use it if available
logecho "Checking if draft release notes exist for $release_tag..."
$GHCURL \
$K8S_GITHUB_RAW_ORG/features/master/$CURRENT_BRANCH/release-notes-draft.md > $tempfile
if [[ -s $tempfile ]]; then
logecho "Draft found - using for release notes..."
cat $tempfile >> $PR_NOTES
logrun rm -f $tempfile
else
logecho "No draft found - creating generic template..."
cat <<EOF+ >> $PR_NOTES
## Major Themes
* TBD
## Other notable improvements
* TBD
## Known Issues
* TBD
## Provider-specific Notes
* TBD
EOF+
fi
# Aggregate all previous releases in series
echo
echo "### Previous Releases Included in $release_tag"
while read anchor; do
echo "$anchor"
done< <(git show master:$changelog_file | egrep "^- \[${release_tag}-") \
>> $PR_NOTES
else
echo "## Changelog since $start_tag" >> $PR_NOTES
echo >> $PR_NOTES
if [[ -n "${notes_action[*]}" ]]; then
echo "### Action Required" >> $PR_NOTES
echo >> $PR_NOTES
extract_pr_title "${notes_action[*]}" >> $PR_NOTES \
|| common::exit 1 "$FAILED: github rate limiting."
echo >> $PR_NOTES
fi
if [[ -n "${notes_normal[*]}" ]]; then
echo "### Other notable changes" >> $PR_NOTES
echo >> $PR_NOTES
extract_pr_title "${notes_normal[*]}" >> $PR_NOTES \
|| common::exit 1 "$FAILED: github rate limiting."
fi
if [[ -z "${notes_normal[*]}" && -z "${notes_action[*]}" ]]; then
logecho
logecho "**No notable changes for this release**" >> $PR_NOTES
logecho
fi
fi
echo >> $PR_NOTES
logecho "Preparing layout..."
create_body ${FLAGS_release_tars:-""} $start_tag ${release_tag:-HEAD} \
> $RELEASE_NOTES_MD
if ((FLAGS_preview)); then
# Pending PRS
logecho "Adding pending PR status..."
(
echo "-------"
echo "## PENDING PRs on the $CURRENT_BRANCH branch"
gitlib::pending_prs $CURRENT_BRANCH
) >> $RELEASE_NOTES_MD
fi
if ((FLAGS_htmlize_md)) && [[ ! $release_tag =~ ${VER_REGEX[dotzero]} ]]; then
# Make users and PRs linkable
# Also, expand anchors (needed for email announce())
sed -i -e "s,#\([0-9]\{5\,\}\),[#\1]($K8S_GITHUB_URL/pull/\1),g" \
-e "s,\(#v[0-9]\{3\}-\),$K8S_GITHUB_URL/blob/master/$changelog_file\1,g" \
-e "s,@\([a-zA-Z0-9-]*\),[@\1](https://github.com/\1),g" \
$RELEASE_NOTES_MD
fi
if ((FLAGS_preview)); then
# We do this after htmlizing because we don't want to update the
# issues in the block of this section
logecho "Adding CI job status (this may take a while)..."
ci_job_status >> $RELEASE_NOTES_MD
fi
if [[ -n "$RELEASE_NOTES_HTML" ]]; then
echo "<style type=text/css>" \
"table,th,tr,td {border: 1px solid gray;" \
"border-collapse: collapse;padding: 5px;}" \
"</style>" > $tempcss
pandoc -H $tempcss --from markdown_github --to html \
$RELEASE_NOTES_MD > $RELEASE_NOTES_HTML
# Remove temp file
logrun rm -f $tempcss
fi
}
##############################################################################
# CONSTANTS
##############################################################################
CURRENT_BRANCH=${FLAGS_branch:-$(gitlib::current_branch)} \
|| common::exit 1
PR_NOTES=$TMPDIR/$PROG-$CURRENT_BRANCH-prnotes
# Initialize new PR_NOTES for session
>$PR_NOTES
RELEASE_NOTES_MD=$(common::absolute_path \
${FLAGS_markdown_file:-$TMPDIR/$PROG-$CURRENT_BRANCH.md})
RELEASE_NOTES_HTML=$(common::absolute_path $FLAGS_html_file)
ANNOUNCEMENT_TEXT=$TMPDIR/$PROG-announcement
###############################################################################
# MAIN
###############################################################################
# Initialize and save up to 10 (rotated logs)
MYLOG=$TMPDIR/$PROG.log
common::logfileinit $MYLOG 10
# BEGIN script
common::timestamp begin
# Check token
gitlib::github_api_token
# Check for packages
common::check_packages jq pandoc
# Build LAST_RELEASE dictionary
gitlib::last_releases
generate_notes || common::exit 1
logecho
if ((FLAGS_quiet)); then
logecho -n "Notes written to $RELEASE_NOTES_MD"
if [[ -f $RELEASE_NOTES_HTML ]]; then
logecho " and $RELEASE_NOTES_HTML"
else
logecho
fi
else
logecho -r "$HR"
cat $RELEASE_NOTES_MD
logecho -r "$HR"
fi
common::timestamp end