2018-04-17 14:12:08 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
# Copyright (C) 2018 Freie Universitat Berlin
|
|
|
|
# Copyright (C) 2018 Inria
|
|
|
|
#
|
|
|
|
# This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
# General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
# directory for more details.
|
|
|
|
#
|
|
|
|
# @author Koen Zandberg <koen@bergzand.net>
|
|
|
|
|
2020-07-04 16:06:37 +02:00
|
|
|
"""Script used to backport PRs."""
|
|
|
|
|
2018-04-17 14:12:08 +02:00
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import sys
|
|
|
|
import shutil
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
import git
|
|
|
|
from agithub.GitHub import GitHub
|
|
|
|
|
|
|
|
ORG = "RIOT-OS"
|
|
|
|
REPO = "RIOT"
|
|
|
|
GITHUBTOKEN_FILE = ".riotgithubtoken"
|
|
|
|
WORKTREE_SUBDIR = "backport_temp"
|
|
|
|
|
|
|
|
RELEASE_PREFIX = ""
|
|
|
|
RELEASE_SUFFIX = "-branch"
|
|
|
|
|
2021-09-28 15:10:13 +02:00
|
|
|
LABELS_REMOVE = ["Process: needs backport", "Reviewed: "]
|
|
|
|
LABELS_ADD = ["Process: release backport"]
|
2018-04-17 14:12:08 +02:00
|
|
|
|
2021-09-28 15:10:13 +02:00
|
|
|
BACKPORT_BRANCH = "backport/{release}/{origbranch}"
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
|
2020-07-04 16:06:37 +02:00
|
|
|
def _get_labels(pull_request):
|
2019-04-24 11:47:22 +02:00
|
|
|
"""
|
|
|
|
>>> _get_labels({'labels': [{'name': 'test'}, {'name': 'abcd'}]})
|
|
|
|
['Process: release backport', 'abcd', 'test']
|
2020-07-04 16:06:37 +02:00
|
|
|
>>> _get_labels({'labels': [{'name': 'Reviewed: what'}, \
|
|
|
|
{'name': 'Reviewed: 3-testing'}]})
|
2019-04-24 11:47:49 +02:00
|
|
|
['Process: release backport']
|
2019-04-24 11:47:22 +02:00
|
|
|
>>> _get_labels({'labels': [{'name': 'Process: release backport'}]})
|
|
|
|
['Process: release backport']
|
|
|
|
>>> _get_labels({'labels': [{'name': 'Process: needs backport'}]})
|
|
|
|
['Process: release backport']
|
|
|
|
"""
|
2021-09-28 15:10:13 +02:00
|
|
|
labels = set(
|
|
|
|
label["name"]
|
|
|
|
for label in pull_request["labels"]
|
|
|
|
if all(not label["name"].startswith(remove) for remove in LABELS_REMOVE)
|
|
|
|
)
|
2018-04-17 14:12:08 +02:00
|
|
|
labels.update(LABELS_ADD)
|
|
|
|
return sorted(list(labels))
|
|
|
|
|
|
|
|
|
2021-09-28 15:10:13 +02:00
|
|
|
def _branch_name_strip(branch_name, prefix=RELEASE_PREFIX, suffix=RELEASE_SUFFIX):
|
2018-04-17 14:12:08 +02:00
|
|
|
"""Strip suffix and prefix.
|
|
|
|
|
|
|
|
>>> _branch_name_strip('2018.10-branch')
|
|
|
|
'2018.10'
|
|
|
|
"""
|
2021-09-28 15:10:13 +02:00
|
|
|
if branch_name.startswith(prefix) and branch_name.endswith(suffix):
|
2018-04-17 14:12:08 +02:00
|
|
|
if prefix:
|
|
|
|
branch_name = branch_name.split(prefix, maxsplit=1)[0]
|
|
|
|
if suffix:
|
|
|
|
branch_name = branch_name.rsplit(suffix, maxsplit=1)[0]
|
|
|
|
return branch_name
|
|
|
|
|
|
|
|
|
|
|
|
def _get_latest_release(branches):
|
2020-07-04 16:07:11 +02:00
|
|
|
"""Get latest release from a list of branches.
|
|
|
|
|
|
|
|
>>> _get_latest_release([{'name': '2018.10-branch'}, \
|
|
|
|
{'name': '2020.10-branch'}])
|
|
|
|
('2020.10', '2020.10-branch')
|
|
|
|
>>> _get_latest_release([{'name': '2020.01-branch'}, \
|
|
|
|
{'name': '2020.04-branch'}])
|
|
|
|
('2020.04', '2020.04-branch')
|
|
|
|
>>> _get_latest_release([{'name': 'non-release-branch'}, \
|
|
|
|
{'name': '2020.04-branch'}])
|
|
|
|
('2020.04', '2020.04-branch')
|
|
|
|
"""
|
2018-04-17 14:12:08 +02:00
|
|
|
version_latest = 0
|
2021-09-28 15:10:13 +02:00
|
|
|
release_fullname = ""
|
|
|
|
release_short = ""
|
2018-04-17 14:12:08 +02:00
|
|
|
for branch in branches:
|
2021-09-28 15:10:13 +02:00
|
|
|
branch_name = _branch_name_strip(branch["name"])
|
2018-04-17 14:12:08 +02:00
|
|
|
branch_num = 0
|
|
|
|
try:
|
2021-09-28 15:10:13 +02:00
|
|
|
branch_num = int("".join(branch_name.split(".")))
|
2018-04-17 14:12:08 +02:00
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
if branch_num > version_latest:
|
|
|
|
version_latest = branch_num
|
|
|
|
release_short = branch_name
|
2021-09-28 15:10:13 +02:00
|
|
|
release_fullname = branch["name"]
|
2018-04-17 14:12:08 +02:00
|
|
|
return (release_short, release_fullname)
|
|
|
|
|
|
|
|
|
2019-04-17 12:33:00 +02:00
|
|
|
def _find_remote(repo, user, repo_name):
|
2018-04-17 14:12:08 +02:00
|
|
|
for remote in repo.remotes:
|
2021-09-28 15:10:13 +02:00
|
|
|
if remote.url.endswith(f"{user}/{repo_name}.git") or remote.url.endswith(
|
|
|
|
f"{user}/{repo_name}"
|
|
|
|
):
|
2018-04-17 14:12:08 +02:00
|
|
|
return remote
|
2021-09-28 15:10:13 +02:00
|
|
|
raise ValueError(f"Could not find remote with URL ending in {user}/{repo_name}.git")
|
2019-04-17 12:33:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _get_upstream(repo):
|
|
|
|
return _find_remote(repo, ORG, REPO)
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _delete_worktree(repo, workdir):
|
|
|
|
shutil.rmtree(workdir)
|
2021-09-28 15:10:13 +02:00
|
|
|
repo.git.worktree("prune")
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2020-07-04 16:06:37 +02:00
|
|
|
# pylint:disable=too-many-locals,too-many-branches,too-many-statements
|
|
|
|
"""Main function of this script."""
|
2021-09-28 15:10:13 +02:00
|
|
|
keyfile = os.path.join(os.environ["HOME"], GITHUBTOKEN_FILE)
|
2018-04-17 14:12:08 +02:00
|
|
|
parser = argparse.ArgumentParser()
|
2021-09-28 15:10:13 +02:00
|
|
|
parser.add_argument(
|
|
|
|
"-k",
|
|
|
|
"--keyfile",
|
|
|
|
type=argparse.FileType("r"),
|
|
|
|
default=keyfile,
|
|
|
|
help="File containing github token",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-c",
|
|
|
|
"--comment",
|
|
|
|
action="store_true",
|
|
|
|
help="Put a comment with a reference under" "the original PR",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-n",
|
|
|
|
"--noop",
|
|
|
|
action="store_true",
|
|
|
|
help="Limited noop mode, creates branch, but doesn't" "push and create the PR",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-r",
|
|
|
|
"--release-branch",
|
|
|
|
type=str,
|
|
|
|
help="Base the backport on this branch, " "default is the latest",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--backport-branch-fmt",
|
|
|
|
type=str,
|
|
|
|
default=BACKPORT_BRANCH,
|
|
|
|
help="Backport branch format. "
|
|
|
|
"Fields '{release}' and '{origbranch} will be "
|
|
|
|
"replaced by the release name and remote branch "
|
|
|
|
"name.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-d",
|
|
|
|
"--gitdir",
|
|
|
|
type=str,
|
|
|
|
default=os.getcwd(),
|
|
|
|
help="Base git repo to work from",
|
|
|
|
)
|
2018-04-17 14:12:08 +02:00
|
|
|
parser.add_argument("PR", type=int, help="Pull request number to backport")
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
gittoken = args.keyfile.read().strip()
|
2020-07-04 16:06:37 +02:00
|
|
|
github_api = GitHub(token=gittoken)
|
2018-04-17 14:12:08 +02:00
|
|
|
# TODO: exception handling
|
2020-07-04 16:06:37 +02:00
|
|
|
status, user = github_api.user.get()
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 200:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'Could not retrieve user: {user["message"]}')
|
2020-07-04 16:06:37 +02:00
|
|
|
sys.exit(1)
|
2020-03-03 17:04:58 +01:00
|
|
|
# Token-scope-check: Is the token is powerful enough to complete
|
|
|
|
# the Backport?
|
2020-07-04 16:06:37 +02:00
|
|
|
response_headers = dict(github_api.getheaders())
|
2020-03-03 17:04:58 +01:00
|
|
|
# agithub documentation says it's lower case header field-names but
|
|
|
|
# at this moment it's not
|
2021-09-28 15:10:13 +02:00
|
|
|
if "X-OAuth-Scopes" in response_headers:
|
|
|
|
scopes = response_headers["X-OAuth-Scopes"]
|
2020-03-03 17:04:58 +01:00
|
|
|
else:
|
2021-09-28 15:10:13 +02:00
|
|
|
scopes = response_headers["x-oauth-scopes"]
|
|
|
|
scopes_list = [x.strip() for x in scopes.split(",")]
|
|
|
|
if not ("public_repo" in scopes_list or "repo" in scopes_list):
|
|
|
|
print(
|
|
|
|
"missing public_repo scope from token settings."
|
|
|
|
" Please add it on the GitHub webinterface"
|
|
|
|
)
|
2020-07-04 16:06:37 +02:00
|
|
|
sys.exit(1)
|
2021-09-28 15:10:13 +02:00
|
|
|
username = user["login"]
|
2020-07-04 16:06:37 +02:00
|
|
|
status, pulldata = github_api.repos[ORG][REPO].pulls[args.PR].get()
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 200:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'Commit #{args.PR} not found: {pulldata["message"]}')
|
2018-04-17 14:12:08 +02:00
|
|
|
sys.exit(2)
|
2021-09-28 15:10:13 +02:00
|
|
|
if not pulldata["merged"]:
|
2018-04-17 14:12:08 +02:00
|
|
|
print("Original PR not yet merged")
|
2020-07-04 16:06:37 +02:00
|
|
|
sys.exit(0)
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'Fetching for commit: #{args.PR}: {pulldata["title"]}')
|
2021-09-28 15:10:13 +02:00
|
|
|
orig_branch = pulldata["head"]["ref"]
|
2020-07-04 16:06:37 +02:00
|
|
|
status, commits = github_api.repos[ORG][REPO].pulls[args.PR].commits.get()
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 200:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'No commits found for #{args.PR}: {commits["message"]}')
|
2018-04-17 14:12:08 +02:00
|
|
|
sys.exit(3)
|
|
|
|
for commit in commits:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'found {commit["sha"]} : {commit["commit"]["message"]}')
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
# Find latest release branch
|
|
|
|
if args.release_branch:
|
|
|
|
release_fullname = args.release_branch
|
|
|
|
release_shortname = _branch_name_strip(args.release_branch)
|
|
|
|
else:
|
2023-10-18 14:12:45 +02:00
|
|
|
status, branches = github_api.repos[ORG][REPO].branches.get(per_page=100)
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 200:
|
2021-09-28 15:10:13 +02:00
|
|
|
print(
|
|
|
|
f"Could not retrieve branches for {ORG}/{REPO}: "
|
|
|
|
f'{branches["message"]}'
|
|
|
|
)
|
2018-04-17 14:12:08 +02:00
|
|
|
sys.exit(4)
|
|
|
|
release_shortname, release_fullname = _get_latest_release(branches)
|
|
|
|
if not release_fullname:
|
|
|
|
print("No release branch found, exiting")
|
|
|
|
sys.exit(5)
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"Backport based on branch {release_fullname}")
|
2018-04-17 14:12:08 +02:00
|
|
|
|
2022-11-10 09:32:49 +01:00
|
|
|
repo = git.Repo(args.gitdir, search_parent_directories=True)
|
2018-04-17 14:12:08 +02:00
|
|
|
# Fetch current upstream
|
|
|
|
upstream_remote = _get_upstream(repo)
|
|
|
|
if not upstream_remote:
|
|
|
|
print("No upstream remote found, can't fetch")
|
2020-07-04 16:06:37 +02:00
|
|
|
sys.exit(6)
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"Fetching {upstream_remote} remote")
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
upstream_remote.fetch()
|
|
|
|
# Build topic branch in temp dir
|
2021-09-28 15:10:13 +02:00
|
|
|
new_branch = args.backport_branch_fmt.format(
|
|
|
|
release=release_shortname, origbranch=orig_branch
|
|
|
|
)
|
2019-04-17 12:44:21 +02:00
|
|
|
if new_branch in repo.branches:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"ERROR: Branch {new_branch} already exists")
|
2019-04-17 12:44:21 +02:00
|
|
|
sys.exit(1)
|
2022-11-11 09:28:38 +01:00
|
|
|
worktree_dir = os.path.join(repo.working_dir, WORKTREE_SUBDIR)
|
2021-09-28 15:10:13 +02:00
|
|
|
repo.git.worktree(
|
|
|
|
"add",
|
|
|
|
"-b",
|
|
|
|
new_branch,
|
|
|
|
WORKTREE_SUBDIR,
|
|
|
|
f"{upstream_remote}/{release_fullname}",
|
|
|
|
)
|
2019-04-24 15:22:40 +02:00
|
|
|
# transform branch name into Head object for later configuring
|
|
|
|
new_branch = repo.branches[new_branch]
|
2019-04-17 12:11:53 +02:00
|
|
|
try:
|
|
|
|
bp_repo = git.Repo(worktree_dir)
|
|
|
|
# Apply commits
|
|
|
|
for commit in commits:
|
2021-09-28 15:10:13 +02:00
|
|
|
bp_repo.git.cherry_pick("-x", commit["sha"])
|
2019-04-17 12:11:53 +02:00
|
|
|
# Push to github
|
2019-04-17 12:42:02 +02:00
|
|
|
origin = _find_remote(repo, username, REPO)
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"Pushing branch {new_branch} to {origin}")
|
2019-04-17 12:11:53 +02:00
|
|
|
if not args.noop:
|
2021-09-17 10:53:26 +02:00
|
|
|
push_info = origin.push(f"{new_branch}:{new_branch}")
|
2019-04-24 12:32:54 +02:00
|
|
|
new_branch.set_tracking_branch(push_info[0].remote_ref)
|
2019-04-17 12:11:53 +02:00
|
|
|
except Exception as exc:
|
|
|
|
# Delete worktree
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"Pruning temporary workdir at {worktree_dir}")
|
2019-04-17 12:11:53 +02:00
|
|
|
_delete_worktree(repo, worktree_dir)
|
|
|
|
# also delete branch created by worktree; this is only possible after
|
|
|
|
# the worktree was deleted
|
|
|
|
repo.delete_head(new_branch)
|
|
|
|
raise exc
|
2023-02-01 16:00:28 +01:00
|
|
|
|
|
|
|
# Delete worktree
|
|
|
|
print(f"Pruning temporary workdir at {worktree_dir}")
|
|
|
|
_delete_worktree(repo, worktree_dir)
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
labels = _get_labels(pulldata)
|
2021-09-28 15:10:13 +02:00
|
|
|
merger = pulldata["merged_by"]["login"]
|
2018-04-17 14:12:08 +02:00
|
|
|
if not args.noop:
|
|
|
|
# Open new PR on github
|
2020-07-04 16:06:37 +02:00
|
|
|
pull_request = {
|
2021-09-28 15:10:13 +02:00
|
|
|
"title": f'{pulldata["title"]} [backport {release_shortname}]',
|
|
|
|
"head": f"{username}:{new_branch}",
|
|
|
|
"base": release_fullname,
|
|
|
|
"body": f'# Backport of #{args.PR}\n\n{pulldata["body"]}',
|
|
|
|
"maintainer_can_modify": True,
|
2018-04-17 14:12:08 +02:00
|
|
|
}
|
2021-09-28 15:10:13 +02:00
|
|
|
status, new_pr = github_api.repos[ORG][REPO].pulls.post(body=pull_request)
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 201:
|
2021-09-28 15:10:13 +02:00
|
|
|
print(
|
|
|
|
f'Error creating the new pr: "{new_pr["message"]}". '
|
|
|
|
'Is "Public Repo" access enabled for the token?'
|
|
|
|
)
|
|
|
|
pr_number = new_pr["number"]
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f"Create PR number #{pr_number} for backport")
|
2020-07-04 16:06:37 +02:00
|
|
|
github_api.repos[ORG][REPO].issues[pr_number].labels.post(body=labels)
|
2018-04-17 14:12:08 +02:00
|
|
|
review_request = {"reviewers": [merger]}
|
2021-09-28 15:10:13 +02:00
|
|
|
github_api.repos[ORG][REPO].pulls[pr_number].requested_reviewers.post(
|
|
|
|
body=review_request
|
|
|
|
)
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
# Put commit under old PR
|
|
|
|
if args.comment and not args.noop:
|
2021-09-17 10:53:26 +02:00
|
|
|
comment = {"body": f"Backport provided in #{pr_number}"}
|
2021-09-28 15:10:13 +02:00
|
|
|
status, res = (
|
|
|
|
github_api.repos[ORG][REPO].issues[args.PR].comments.post(body=comment)
|
|
|
|
)
|
2018-04-17 14:12:08 +02:00
|
|
|
if status != 201:
|
2021-09-17 10:53:26 +02:00
|
|
|
print(f'Something went wrong adding the comment: {res["message"]}')
|
|
|
|
print(f"Added comment to #{args.PR}")
|
2018-04-17 14:12:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|