Skip to content

backport

backport #540

Workflow file for this run

name: backport
on:
push:
branches:
- 'main'
workflow_dispatch:
inputs:
commit_hash:
description: 'The commit hash to backport'
required: true
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
jobs:
get-backport-commits:
runs-on: ubuntu-latest
outputs:
matrix: ${{ steps.push.outputs.commits || steps.dispatch.outputs.commits }}
steps:
- if: ${{ github.event_name == 'push' }}
name: Get all commits in the push event
id: push
env:
COMMITS: ${{ toJson(github.event.commits) }}
run: |
commits_json=$(echo "$COMMITS" | jq -c '[.[].id]')
echo "commits=${commits_json}" >> $GITHUB_OUTPUT
- if: ${{ github.event_name == 'workflow_dispatch' }}
name: Get the commit from the PR
id: dispatch
run: |
commits=$(echo '${{ github.event.inputs.commit_hash }}' | jq -Rc '[.]')
echo "commits=$commits" >> $GITHUB_OUTPUT
get-backport-target-branch:
runs-on: ubuntu-latest
needs: get-backport-commits
strategy:
matrix:
commits: ${{ fromJson(needs.get-backport-commits.outputs.matrix) }}
outputs:
matrix: ${{ steps.milestones.outputs.matrix }}
latest_commit: ${{ steps.commit.outputs.latest_commit }}
commit_message: ${{ steps.commit.outputs.commit_message }}
pr_number: ${{ steps.commit.outputs.pr_number }}
latest_release: ${{ steps.commit.outputs.latest_release }}
author: ${{ steps.commit.outputs.author }}
author_email: ${{ steps.commit.outputs.author_email }}
labels: ${{ steps.commit.outputs.labels }}
steps:
- name: Checkout the revision
uses: actions/checkout@v4
with:
lfs: false
ref: ${{ matrix.commits }}
- name: Extract pr_number from commit message
id: commit
run: |
latest_commit=$(git rev-parse HEAD) # Get the recent commit hash of the current repository
echo "latest_commit=$latest_commit" >> $GITHUB_OUTPUT
commit_message=$(git show -s --format=%B $latest_commit) # Get messages from recent commit
echo "commit_message<<EOF" >> $GITHUB_OUTPUT
echo "$commit_message" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
pr_number=$(echo $commit_message | (grep -oP "\(#\d+\)" || true) | (grep -oP "\d+" || true)) # Get pr number from commit message
echo "pr_number=$pr_number" >> $GITHUB_OUTPUT
latest_release=$(cat VERSION | grep -oP "\d+\.\d+")
echo "latest_release=$latest_release" >> $GITHUB_OUTPUT
author=$(git show -s --format=%an $latest_commit)
echo "author=$author" >> $GITHUB_OUTPUT
author_email=$(git show -s --format=%ae $latest_commit)
echo "author_email=$author_email" >> $GITHUB_OUTPUT
labels=$(gh pr view $pr_number --json labels | jq -r '.labels[].name')
echo "labels<<EOF" >> $GITHUB_OUTPUT
echo "$labels" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
env:
GH_TOKEN: ${{ github.token }}
- name: Get target branches
id: milestones
if: ${{ steps.commit.outputs.pr_number }}
run: |
target_milestone=$(gh pr view ${{ steps.commit.outputs.pr_number }} --json milestone --jq .milestone.title)
milestones=$(gh api /repos/:owner/:repo/milestones --jq '.[].title')
echo "Milestones configured in the repo: $milestones"
# Remove Backlog from the backport target branch
milestones=($milestones)
for i in "${!milestones[@]}"; do
if [[ "${milestones[$i]}" == "Backlog" ]]; then
unset 'milestones[$i]'
fi
done
for i in "${!milestones[@]}"; do
if ! git ls-remote --heads | grep -q "refs/heads/${milestones[$i]}\$"; then
unset 'milestones[$i]'
fi
done
echo "Milestones with the corresponding release branch: ${milestones[@]}"
sort_milestones=($(printf "%s\n" "${milestones[@]}" | sort -r))
for i in "${!sort_milestones[@]}"; do
if [[ "${sort_milestones[$i]}" == "$target_milestone" ]]; then
target_milestones=("${sort_milestones[@]:0:$((i+1))}")
break
fi
done
pr_head=$(gh pr view ${{ steps.commit.outputs.pr_number }} --json headRepositoryOwner,headRefName --jq '.headRepositoryOwner.login + ":" + .headRefName')
STATUS_CODE=$(curl -s -o response.json -w "%{http_code}" -X GET "${{ secrets.KVSTORE_URL }}/?key=head_$pr_head" \
-H "Authorization: Bearer ${{ secrets.KVSTORE_TOKEN }}")
if [ $STATUS_CODE == "404" ]; then
target_branches=("${target_milestones[@]}")
elif [ $STATUS_CODE == "200" ]; then
pr_base=$(jq -r '.value' response.json)
target_branches=("backport/${pr_base}-to-${target_milestones[@]}")
else
echo "Failed to get the target branches"
echo "response=$(cat response.json)"
exit 1
fi
curl -X DELETE "${{ secrets.KVSTORE_URL }}/?key=head_$pr_head" \
-H "Authorization: Bearer ${{ secrets.KVSTORE_TOKEN }}"
curl -X DELETE "${{ secrets.KVSTORE_URL }}/?key=base_$pr_base" \
-H "Authorization: Bearer ${{ secrets.KVSTORE_TOKEN }}"
matrix=$(printf '%s\n' "${target_branches[@]}" | grep -v '^$' | jq -R . | jq -sc .)
echo "matrix=$matrix" >> $GITHUB_OUTPUT
env:
GH_TOKEN: ${{ github.token }}
backport:
if: ${{ needs.get-backport-target-branch.outputs.matrix != '[]' }}
runs-on: ubuntu-latest
needs: get-backport-target-branch
strategy:
matrix:
target_branch: ${{ fromJson(needs.get-backport-target-branch.outputs.matrix) }}
fail-fast: false
permissions:
pull-requests: write
steps:
- uses: actions/checkout@v4
with:
ref: ${{ matrix.target_branch }}
- name: Cherry-pick
env:
COMMIT_MESSAGE: ${{ needs.get-backport-target-branch.outputs.commit_message }}
run: |
git config user.name "${{ needs.get-backport-target-branch.outputs.author }}"
git config user.email "${{ needs.get-backport-target-branch.outputs.author_email }}"
latest_commit="${{ needs.get-backport-target-branch.outputs.latest_commit }}"
git fetch origin main "$latest_commit"
git cherry-pick --strategy=recursive --strategy-option=theirs "$latest_commit"
git commit \
--amend -m "${COMMIT_MESSAGE}" \
--trailer "Backported-from=main (${{ needs.get-backport-target-branch.outputs.latest_release }})" \
--trailer "Backported-to=${{ matrix.target_branch }}" \
--trailer "Backport-of=${{ needs.get-backport-target-branch.outputs.pr_number }}"
- name: When cherry-pick is failed
if: failure()
run: |
gh pr comment ${{ needs.get-backport-target-branch.outputs.pr_number }} -b "Backport to ${{ matrix.target_branch }} is failed. Please backport manually."
env:
GH_TOKEN: ${{ github.token }}
- id: commit_message
env:
COMMIT_MESSAGE: ${{ needs.get-backport-target-branch.outputs.commit_message }}
run: |
commit_header=$(echo "${COMMIT_MESSAGE}" | head -n 1)
echo "commit_header=$commit_header" >> $GITHUB_OUTPUT
commit_body=$(echo "${COMMIT_MESSAGE}" | awk '/^$/{p++;next} p==1')
echo "commit_body<<EOF" >> $GITHUB_OUTPUT
echo "$commit_body" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
commit_footer=$(echo "${COMMIT_MESSAGE}" | awk '/^$/{p++;next} p==2')
echo "commit_footer<<EOF" >> $GITHUB_OUTPUT
echo "$commit_footer" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
- name: Create Backport PR
id: pr
uses: peter-evans/create-pull-request@v6
with:
token: ${{ secrets.OCTODOG }}
author: "${{ needs.get-backport-target-branch.outputs.author }} <${{ needs.get-backport-target-branch.outputs.author_email }}>"
title: "${{ steps.commit_message.outputs.commit_header }}"
body: "This is an auto-generated backport PR of #${{ needs.get-backport-target-branch.pr_number }} to the ${{ matrix.target_branch }} release."
branch: "backport/${{ needs.get-backport-target-branch.outputs.pr_number }}-to-${{ matrix.target_branch }}"
base: ${{ matrix.target_branch }}
labels: |
backport
${{ needs.get-backport-target-branch.outputs.labels }}
assignees: ${{ needs.get-backport-target-branch.outputs.author }}
- id: pr_id
run: |
pr_id=$(gh api graphql -f query='
query ($pr_number: Int!, $owner: String!, $name: String!) {
repository(owner: $owner, name: $name) {
pullRequest(number: $pr_number) {
id
}
}
}
' -F pr_number=${{ steps.pr.outputs.pull-request-number }} -f owner=${{ github.repository_owner }} -f name=${{ github.event.repository.name }} | jq -r '.data.repository.pullRequest.id')
echo "pr_id=$pr_id" >> $GITHUB_OUTPUT
env:
GH_TOKEN: ${{ secrets.OCTODOG }}
- id: commit_footer
run: |
commit_footer="Co-authored-by: ${{ needs.get-backport-target-branch.outputs.author }} <${{ needs.get-backport-target-branch.outputs.author_email }}>
${{ steps.commit_message.outputs.commit_footer }}
Backported-from: main (${{ needs.get-backport-target-branch.outputs.latest_release }})
Backported-to: ${{ matrix.target_branch }}
Backport-of: ${{ needs.get-backport-target-branch.outputs.pr_number }}"
commit_footer=$(echo "$commit_footer" | grep '.') # remove empty lines
echo "commit_footer<<EOF" >> $GITHUB_OUTPUT
echo "$commit_footer" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
- name: Enable auto-merge
if: ${{ steps.pr.outputs.pull-request-number }}
run: |
gh api graphql -f query='
mutation ($pullRequestId: ID!, $mergeMethod: PullRequestMergeMethod!) {
enablePullRequestAutoMerge(input: {
pullRequestId: $pullRequestId,
mergeMethod: $mergeMethod,
commitBody: """
${{ steps.commit_message.outputs.commit_body }}
${{ steps.commit_footer.outputs.commit_footer }}
""",
commitHeadline: "${{ steps.commit_message.outputs.commit_header }} (#${{ steps.pr.outputs.pull-request-number }})"
}) {
pullRequest {
autoMergeRequest {
enabledAt
}
}
}
}
' -F pullRequestId=${{ steps.pr_id.outputs.pr_id }} -f mergeMethod="SQUASH"
env:
GH_TOKEN: ${{ secrets.OCTODOG }}