| #!/bin/bash |
| |
| # Copyright 2015 The Bazel 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 -eu |
| |
| # Generate the release notes from the git history. |
| |
| SCRIPT_DIR=$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd) |
| source ${SCRIPT_DIR}/common.sh |
| |
| # It uses the RELNOTES tag in the history to knows the important changes to |
| # report: |
| # RELNOTES: indicates a change important the user. |
| # RELNOTES[NEW]: introduces a new feature. |
| # RELNOTES[INC]: indicates an incompatible change. |
| # The previous releases base is detected using the CHANGELOG file from the |
| # repository. |
| RELNOTES_TYPES=("INC" "NEW" "") |
| RELNOTES_DESC=("Incompatible changes" "New features" "Important changes") |
| |
| # Get the baseline version and cherry-picks of the previous release |
| # Parameter: $1 is the path to the changelog file |
| # Output: "${BASELINE} ${CHERRYPICKS}" |
| # BASELINE is the hash of the baseline commit of the latest release |
| # CHERRYPICKS is the list of hash of cherry-picked commits of the latest release |
| # return 1 if there is no initial release |
| function __get_last_release() { |
| local changelog=$1 |
| [ -f "$changelog" ] || return 1 # No changelog = initial release |
| local BASELINE_LINE=$(grep -m 1 -n '^Baseline: ' "$changelog") || return 1 |
| [ -n "${BASELINE_LINE}" ] || return 1 # No baseline = initial release |
| local BASELINE_LINENB=$(echo "${BASELINE_LINE}" | cut -d ":" -f 1) |
| BASELINE=$(echo "${BASELINE_LINE}" | cut -d " " -f 2) |
| local CHERRYPICK_LINE=$(($BASELINE_LINENB + 3)) |
| # grep -B999 looks for all lines before the empty line and after that we |
| # restrict to only lines with the cherry picked hash then finally we cut |
| # the hash. |
| local CHERRY_PICKS=$(tail -n +${CHERRYPICK_LINE} "$changelog" \ |
| | grep -m 1 "^$" -B999 \ |
| | grep -E '^ \+ [a-z0-9]+:' \ |
| | cut -d ":" -f 1 | cut -d "+" -f 2) |
| echo $BASELINE $CHERRY_PICKS |
| return 0 |
| } |
| |
| # Now get the list of commit with a RELNOTES since latest release baseline ($1) |
| # discarding cherry_picks ($2..) and rollbacks. The returned list of commits is |
| # from the oldest to the newest |
| function __get_release_notes_commits() { |
| local baseline=$1 |
| shift |
| local cherry_picks="$@" |
| local rollback_commits=$(git log --oneline -E --grep='^Rollback of commit [a-z0-9]+.$' ${baseline}.. \ |
| | grep -E '^[a-z0-9]+ Rollback of commit [a-z0-9]+.$' || true) |
| local rollback_hashes=$(echo "$rollback_commits" | cut -d " " -f 1) |
| local rolledback_hashes=$(echo "$rollback_commits" | cut -d " " -f 5 | sed -E 's/^(.......).*$/\1/') |
| local exclude_hashes=$(echo DUMMY $cherry_picks $rollback_hashes $rolledback_hashes | xargs echo | sed 's/ /|/g') |
| git log --reverse --pretty=format:%H ${baseline}.. -E --grep='^RELNOTES(\[[^\]+\])?:' \ |
| | grep -Ev "^(${exclude_hashes})" || true |
| } |
| |
| # Extract the release note from a commit hash ($1). It extracts |
| # the RELNOTES([??]): lines. A new empty line ends the relnotes tag. |
| # It adds the relnotes, if not "None" ("None.") or "n/a" ("n/a.") to |
| # the correct array: |
| # RELNOTES_INC for incompatible changes |
| # RELNOTES_NEW for new features changes |
| # RELNOTES for other changes |
| function __extract_release_note() { |
| local find_relnote_awk_script=" |
| BEGIN { in_relnote = 0 } |
| /^$/ { in_relnote = 0 } |
| /^PiperOrigin-RevId:.*$/ { in_relnote = 0 } |
| /^RELNOTES(\[[^\]]+\])?:/ { in_relnote = 1 } |
| { if (in_relnote) { print } }" |
| local relnote="$(git show -s $1 --pretty=format:%B | awk "${find_relnote_awk_script}")" |
| local regex="^RELNOTES(\[([a-zA-Z]*)\])?:[[:space:]]*([^[:space:]].*[^[:space:]])[[:space:]]*$" |
| if [[ "$relnote" =~ $regex ]]; then |
| local relnote_kind=${BASH_REMATCH[2]} |
| local relnote_text="${BASH_REMATCH[3]}" |
| if [[ ! "$(echo $relnote_text | awk '{print tolower($0)}')" =~ ^((none|n/a|no)(\.( .*)?)?|\.)$ ]]; then |
| eval "RELNOTES_${relnote_kind}+=(\"\${relnote_text}\")" |
| fi |
| fi |
| } |
| |
| # Build release notes arrays from a list of commits ($@) and return the release |
| # note in an array of array. |
| function __generate_release_notes() { |
| for i in "${RELNOTES_TYPES[@]}"; do |
| eval "RELNOTES_${i}=()" |
| done |
| for commit in $@; do |
| __extract_release_note "${commit}" |
| done |
| } |
| |
| # Returns the list of release notes in arguments into a list of points in |
| # a markdown list. The release notes are wrapped to 70 characters so it |
| # displays nicely in a git history. |
| function __format_release_notes() { |
| local i |
| for (( i=1; $i <= $#; i=$i+1 )); do |
| local relnote="${!i}" |
| local lines=$(echo "$relnote" | wrap_text 66) # wrap to 70 counting the 4 leading spaces. |
| echo " - $lines" | head -1 |
| echo "$lines" | tail -n +2 | sed 's/^/ /' |
| done |
| } |
| |
| # Create the release notes since commit $1 ($2...${[#]} are the cherry-picks, |
| # so the commits to ignore. |
| function __release_notes() { |
| local last_release=$1 |
| local i |
| local commits=$(__get_release_notes_commits $last_release) |
| local length="${#RELNOTES_TYPES[@]}" |
| __generate_release_notes "$commits" |
| for (( i=0; $i < $length; i=$i+1 )); do |
| local relnotes_title="${RELNOTES_DESC[$i]}" |
| local relnotes_type=${RELNOTES_TYPES[$i]} |
| local relnotes="RELNOTES_${relnotes_type}[@]" |
| local nb_relnotes=$(eval "echo \${#$relnotes}") |
| if (( "${nb_relnotes}" > 0 )); then |
| echo "${relnotes_title}:" |
| echo |
| __format_release_notes "${!relnotes}" |
| echo |
| fi |
| done |
| |
| # Add a list of contributors to thank. |
| # Stages: |
| # 1. Get the list of authors from the last release til now, both name and |
| # email. |
| # 2. Sort and uniqify. |
| # 3. Remove googlers. (This is why the email is needed) |
| # 4. Cut the email address, leaving only the name. |
| # 5-n. Remove trailing spaces and newlines, substituting with a comman and a |
| # space, removing any trailing spaces again. |
| local external_authors=$(git log $last_release..HEAD --format="%aN <%aE>" \ |
| | sort \ |
| | uniq \ |
| | grep -v "google.com" \ |
| | cut -d'<' -f 1 \ |
| | sed -e 's/[[:space:]]$//' \ |
| | tr '\n' ',' \ |
| | sed -e 's/,$/\n/' \ |
| | sed -e 's/,/, /g') |
| echo "This release contains contributions from many people at Google, as well as ${external_authors}." |
| } |
| |
| # A wrapper around all the previous function, using the CHANGELOG.md |
| # file in $1 to compute the last release commit hash. |
| function create_release_notes() { |
| local last_release=$(__get_last_release "$1") || \ |
| { echo "Initial release."; return 0; } |
| [ -n "${last_release}" ] || { echo "Initial release."; return 0; } |
| __release_notes ${last_release} |
| } |