|  | #!/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} | 
|  | } |