k8s.io/kubernetes@v1.31.0-alpha.0.0.20240520171757-56147500dadc/hack/cherry_pick_pull.sh (about)

     1  #!/usr/bin/env bash
     2  
     3  # Copyright 2015 The Kubernetes Authors.
     4  #
     5  # Licensed under the Apache License, Version 2.0 (the "License");
     6  # you may not use this file except in compliance with the License.
     7  # You may obtain a copy of the License at
     8  #
     9  #     http://www.apache.org/licenses/LICENSE-2.0
    10  #
    11  # Unless required by applicable law or agreed to in writing, software
    12  # distributed under the License is distributed on an "AS IS" BASIS,
    13  # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    14  # See the License for the specific language governing permissions and
    15  # limitations under the License.
    16  
    17  # Usage Instructions: https://git.k8s.io/community/contributors/devel/sig-release/cherry-picks.md
    18  
    19  # Checkout a PR from GitHub. (Yes, this is sitting in a Git tree. How
    20  # meta.) Assumes you care about pulls from remote "upstream" and
    21  # checks them out to a branch named:
    22  #  automated-cherry-pick-of-<pr>-<target branch>-<timestamp>
    23  
    24  set -o errexit
    25  set -o nounset
    26  set -o pipefail
    27  
    28  REPO_ROOT="$(git rev-parse --show-toplevel)"
    29  declare -r REPO_ROOT
    30  cd "${REPO_ROOT}"
    31  
    32  STARTINGBRANCH=$(git symbolic-ref --short HEAD)
    33  declare -r STARTINGBRANCH
    34  declare -r REBASEMAGIC="${REPO_ROOT}/.git/rebase-apply"
    35  DRY_RUN=${DRY_RUN:-""}
    36  REGENERATE_DOCS=${REGENERATE_DOCS:-""}
    37  UPSTREAM_REMOTE=${UPSTREAM_REMOTE:-upstream}
    38  FORK_REMOTE=${FORK_REMOTE:-origin}
    39  MAIN_REPO_ORG=${MAIN_REPO_ORG:-$(git remote get-url "$UPSTREAM_REMOTE" | awk '{gsub(/http[s]:\/\/|git@/,"")}1' | awk -F'[@:./]' 'NR==1{print $3}')}
    40  MAIN_REPO_NAME=${MAIN_REPO_NAME:-$(git remote get-url "$UPSTREAM_REMOTE" | awk '{gsub(/http[s]:\/\/|git@/,"")}1' | awk -F'[@:./]' 'NR==1{print $4}')}
    41  
    42  if [[ -z ${GITHUB_USER:-} ]]; then
    43    echo "Please export GITHUB_USER=<your-user> (or GH organization, if that's where your fork lives)"
    44    exit 1
    45  fi
    46  
    47  if ! command -v gh > /dev/null; then
    48    echo "Can't find 'gh' tool in PATH, please install from https://github.com/cli/cli"
    49    exit 1
    50  fi
    51  
    52  if [[ "$#" -lt 2 ]]; then
    53    echo "${0} <remote branch> <pr-number>...: cherry pick one or more <pr> onto <remote branch> and leave instructions for proposing pull request"
    54    echo
    55    echo "  Checks out <remote branch> and handles the cherry-pick of <pr> (possibly multiple) for you."
    56    echo "  Examples:"
    57    echo "    $0 upstream/release-3.14 12345        # Cherry-picks PR 12345 onto upstream/release-3.14 and proposes that as a PR."
    58    echo "    $0 upstream/release-3.14 12345 56789  # Cherry-picks PR 12345, then 56789 and proposes the combination as a single PR."
    59    echo
    60    echo "  Set the DRY_RUN environment var to skip git push and creating PR."
    61    echo "  This is useful for creating patches to a release branch without making a PR."
    62    echo "  When DRY_RUN is set the script will leave you in a branch containing the commits you cherry-picked."
    63    echo
    64    echo "  Set the REGENERATE_DOCS environment var to regenerate documentation for the target branch after picking the specified commits."
    65    echo "  This is useful when picking commits containing changes to API documentation."
    66    echo
    67    echo "  Set UPSTREAM_REMOTE (default: upstream) and FORK_REMOTE (default: origin)"
    68    echo "  to override the default remote names to what you have locally."
    69    echo
    70    echo "  For merge process info, see https://git.k8s.io/community/contributors/devel/sig-release/cherry-picks.md"
    71    exit 2
    72  fi
    73  
    74  # Checks if you are logged in. Will error/bail if you are not.
    75  gh auth status
    76  
    77  if git_status=$(git status --porcelain --untracked=no 2>/dev/null) && [[ -n "${git_status}" ]]; then
    78    echo "!!! Dirty tree. Clean up and try again."
    79    exit 1
    80  fi
    81  
    82  if [[ -e "${REBASEMAGIC}" ]]; then
    83    echo "!!! 'git rebase' or 'git am' in progress. Clean up and try again."
    84    exit 1
    85  fi
    86  
    87  declare -r BRANCH="$1"
    88  shift 1
    89  declare -r PULLS=( "$@" )
    90  
    91  function join { local IFS="$1"; shift; echo "$*"; }
    92  PULLDASH=$(join - "${PULLS[@]/#/#}") # Generates something like "#12345-#56789"
    93  declare -r PULLDASH
    94  PULLSUBJ=$(join " " "${PULLS[@]/#/#}") # Generates something like "#12345 #56789"
    95  declare -r PULLSUBJ
    96  
    97  echo "+++ Updating remotes..."
    98  git remote update "${UPSTREAM_REMOTE}" "${FORK_REMOTE}"
    99  
   100  if ! git log -n1 --format=%H "${BRANCH}" >/dev/null 2>&1; then
   101    echo "!!! '${BRANCH}' not found. The second argument should be something like ${UPSTREAM_REMOTE}/release-0.21."
   102    echo "    (In particular, it needs to be a valid, existing remote branch that I can 'git checkout'.)"
   103    exit 1
   104  fi
   105  
   106  NEWBRANCHREQ="automated-cherry-pick-of-${PULLDASH}" # "Required" portion for tools.
   107  declare -r NEWBRANCHREQ
   108  NEWBRANCH="$(echo "${NEWBRANCHREQ}-${BRANCH}" | sed 's/\//-/g')"
   109  declare -r NEWBRANCH
   110  NEWBRANCHUNIQ="${NEWBRANCH}-$(date +%s)"
   111  declare -r NEWBRANCHUNIQ
   112  echo "+++ Creating local branch ${NEWBRANCHUNIQ}"
   113  
   114  cleanbranch=""
   115  gitamcleanup=false
   116  function return_to_kansas {
   117    if [[ "${gitamcleanup}" == "true" ]]; then
   118      echo
   119      echo "+++ Aborting in-progress git am."
   120      git am --abort >/dev/null 2>&1 || true
   121    fi
   122  
   123    # return to the starting branch and delete the PR text file
   124    if [[ -z "${DRY_RUN}" ]]; then
   125      echo
   126      echo "+++ Returning you to the ${STARTINGBRANCH} branch and cleaning up."
   127      git checkout -f "${STARTINGBRANCH}" >/dev/null 2>&1 || true
   128      if [[ -n "${cleanbranch}" ]]; then
   129        git branch -D "${cleanbranch}" >/dev/null 2>&1 || true
   130      fi
   131    fi
   132  }
   133  trap return_to_kansas EXIT
   134  
   135  SUBJECTS=()
   136  function make-a-pr() {
   137    local rel
   138    rel="$(basename "${BRANCH}")"
   139    echo
   140    echo "+++ Creating a pull request on GitHub at ${GITHUB_USER}:${NEWBRANCH}"
   141  
   142    local numandtitle
   143    numandtitle=$(printf '%s\n' "${SUBJECTS[@]}")
   144    prtext=$(cat <<EOF
   145  Cherry pick of ${PULLSUBJ} on ${rel}.
   146  
   147  ${numandtitle}
   148  
   149  For details on the cherry pick process, see the [cherry pick requests](https://git.k8s.io/community/contributors/devel/sig-release/cherry-picks.md) page.
   150  
   151  \`\`\`release-note
   152  
   153  \`\`\`
   154  EOF
   155  )
   156  
   157    gh pr create --title="Automated cherry pick of ${numandtitle}" --body="${prtext}" --head "${GITHUB_USER}:${NEWBRANCH}" --base "${rel}" --repo="${MAIN_REPO_ORG}/${MAIN_REPO_NAME}"
   158  }
   159  
   160  git checkout -b "${NEWBRANCHUNIQ}" "${BRANCH}"
   161  cleanbranch="${NEWBRANCHUNIQ}"
   162  
   163  gitamcleanup=true
   164  for pull in "${PULLS[@]}"; do
   165    echo "+++ Downloading patch to /tmp/${pull}.patch (in case you need to do this again)"
   166  
   167    curl -o "/tmp/${pull}.patch" -sSL "https://github.com/${MAIN_REPO_ORG}/${MAIN_REPO_NAME}/pull/${pull}.patch"
   168    echo
   169    echo "+++ About to attempt cherry pick of PR. To reattempt:"
   170    echo "  $ git am -3 /tmp/${pull}.patch"
   171    echo
   172    git am -3 "/tmp/${pull}.patch" || {
   173      conflicts=false
   174      while unmerged=$(git status --porcelain | grep ^U) && [[ -n ${unmerged} ]] \
   175        || [[ -e "${REBASEMAGIC}" ]]; do
   176        conflicts=true # <-- We should have detected conflicts once
   177        echo
   178        echo "+++ Conflicts detected:"
   179        echo
   180        (git status --porcelain | grep ^U) || echo "!!! None. Did you git am --continue?"
   181        echo
   182        echo "+++ Please resolve the conflicts in another window (and remember to 'git add / git am --continue')"
   183        read -p "+++ Proceed (anything other than 'y' aborts the cherry-pick)? [y/n] " -r
   184        echo
   185        if ! [[ "${REPLY}" =~ ^[yY]$ ]]; then
   186          echo "Aborting." >&2
   187          exit 1
   188        fi
   189      done
   190  
   191      if [[ "${conflicts}" != "true" ]]; then
   192        echo "!!! git am failed, likely because of an in-progress 'git am' or 'git rebase'"
   193        exit 1
   194      fi
   195    }
   196  
   197    # set the subject
   198    subject=$(grep -m 1 "^Subject" "/tmp/${pull}.patch" | sed -e 's/Subject: \[PATCH//g' | sed 's/.*] //')
   199    SUBJECTS+=("#${pull}: ${subject}")
   200  
   201    # remove the patch file from /tmp
   202    rm -f "/tmp/${pull}.patch"
   203  done
   204  gitamcleanup=false
   205  
   206  # Re-generate docs (if needed)
   207  if [[ -n "${REGENERATE_DOCS}" ]]; then
   208    echo
   209    echo "Regenerating docs..."
   210    if ! hack/generate-docs.sh; then
   211      echo
   212      echo "hack/generate-docs.sh FAILED to complete."
   213      exit 1
   214    fi
   215  fi
   216  
   217  if [[ -n "${DRY_RUN}" ]]; then
   218    echo "!!! Skipping git push and PR creation because you set DRY_RUN."
   219    echo "To return to the branch you were in when you invoked this script:"
   220    echo
   221    echo "  git checkout ${STARTINGBRANCH}"
   222    echo
   223    echo "To delete this branch:"
   224    echo
   225    echo "  git branch -D ${NEWBRANCHUNIQ}"
   226    exit 0
   227  fi
   228  
   229  if git remote -v | grep ^"${FORK_REMOTE}" | grep "${MAIN_REPO_ORG}/${MAIN_REPO_NAME}.git"; then
   230    echo "!!! You have ${FORK_REMOTE} configured as your ${MAIN_REPO_ORG}/${MAIN_REPO_NAME}.git"
   231    echo "This isn't normal. Leaving you with push instructions:"
   232    echo
   233    echo "+++ First manually push the branch this script created:"
   234    echo
   235    echo "  git push REMOTE ${NEWBRANCHUNIQ}:${NEWBRANCH}"
   236    echo
   237    echo "where REMOTE is your personal fork (maybe ${UPSTREAM_REMOTE}? Consider swapping those.)."
   238    echo "OR consider setting UPSTREAM_REMOTE and FORK_REMOTE to different values."
   239    echo
   240    make-a-pr
   241    cleanbranch=""
   242    exit 0
   243  fi
   244  
   245  echo
   246  echo "+++ I'm about to do the following to push to GitHub (and I'm assuming ${FORK_REMOTE} is your personal fork):"
   247  echo
   248  echo "  git push ${FORK_REMOTE} ${NEWBRANCHUNIQ}:${NEWBRANCH}"
   249  echo
   250  read -p "+++ Proceed (anything other than 'y' aborts the cherry-pick)? [y/n] " -r
   251  if ! [[ "${REPLY}" =~ ^[yY]$ ]]; then
   252    echo "Aborting." >&2
   253    exit 1
   254  fi
   255  
   256  git push "${FORK_REMOTE}" -f "${NEWBRANCHUNIQ}:${NEWBRANCH}"
   257  make-a-pr