clan-infra/pkgs/clan-merge/clan_merge/__init__.py
DavHau e0a849d383
All checks were successful
build / test (push) Successful in 10s
clan-merge: improve PR debug messages
2023-08-07 10:40:37 +02:00

142 lines
4.1 KiB
Python

import argparse
import json
import urllib.request
from os import environ
from typing import Optional
def load_token() -> str:
GITEA_TOKEN_FILE = environ.get("GITEA_TOKEN_FILE", default=None)
assert GITEA_TOKEN_FILE is not None
with open(GITEA_TOKEN_FILE, "r") as f:
return f.read().strip()
def pr_message(pr: dict) -> str:
return f"PR {pr['number']} in repo {pr['base']['repo']['name']} from user {pr['user']['login']}: {pr['title']}"
def is_ci_green(pr: dict) -> bool:
print(f"Checking CI status for {pr_message(pr)}")
repo = pr["base"]["repo"]["name"]
url = (
"https://git.clan.lol/api/v1/repos/clan/"
+ repo
+ "/commits/"
+ pr["head"]["sha"]
+ "/status"
)
response = urllib.request.urlopen(url)
data = json.loads(response.read())
# check for all commit statuses to have status "success"
if not data["statuses"]:
print(f"No CI status for {pr_message(pr)}")
return False
for status in data["statuses"]:
if status["status"] != "success":
return False
return True
def decide_merge(pr: dict, allowed_users: list[str], bot_name: str) -> bool:
assignees = pr["assignees"] if pr["assignees"] else []
if (
pr["user"]["login"] in allowed_users
and pr["mergeable"] is True
and not pr["title"].startswith("WIP:")
and pr["state"] == "open"
# check if bot is assigned
and any(assignee["login"] == bot_name for assignee in assignees)
and is_ci_green(pr)
):
return True
return False
# python equivalent for: curl -X 'GET' https://git.clan.lol/api/v1/repos/clan/{repo}/pulls
def list_prs(repo: str) -> list:
url = "https://git.clan.lol/api/v1/repos/clan/" + repo + "/pulls"
response = urllib.request.urlopen(url)
data = json.loads(response.read())
return data
def list_prs_to_merge(prs: list, allowed_users: list[str], bot_name: str) -> list:
prs_to_merge = []
for pr in prs:
if decide_merge(pr, allowed_users, bot_name) is True:
prs_to_merge.append(pr)
return prs_to_merge
def parse_args() -> argparse.Namespace:
parser = argparse.ArgumentParser(description="Merge PRs on clan.lol")
# parse a list of allowed users
parser.add_argument(
"--allowed-users",
nargs="+",
help="list of users allowed to merge",
required=True,
)
# option for bot-name
parser.add_argument(
"--bot-name",
help="name of the bot",
required=True,
)
# parse list of repository names for which to merge PRs
parser.add_argument(
"--repos",
nargs="+",
help="list of repositories for which to merge PRs",
required=True,
)
# option for dry run
parser.add_argument(
"--dry-run",
action="store_true",
help="dry run",
)
return parser.parse_args()
def clan_merge(
args: Optional[argparse.Namespace] = None, gitea_token: Optional[str] = None
) -> None:
if gitea_token is None:
gitea_token = load_token()
if args is None:
args = parse_args()
allowed_users = args.allowed_users
repos = args.repos
dry_run = args.dry_run
bot_name = args.bot_name
for repo in repos:
prs = list_prs(repo)
prs_to_merge = list_prs_to_merge(prs, allowed_users, bot_name)
for pr in prs_to_merge:
url = (
"https://git.clan.lol/api/v1/repos/clan/"
+ repo
+ "/pulls/"
+ str(pr["number"])
+ "/merge"
+ f"?token={gitea_token}"
)
if dry_run is True:
print(f"Would merge {pr_message(pr)}")
else:
print(f"Merging {pr_message(pr)}")
data = dict(
Do="merge",
)
data_encoded = json.dumps(data).encode("utf8")
req = urllib.request.Request(
url, data=data_encoded, headers={"Content-Type": "application/json"}
)
urllib.request.urlopen(req)
if __name__ == "__main__":
clan_merge()