1
0
Fork 0

Refactor update_appslevels.py: use argparse, gitpython, jinja2 for pr message

This commit is contained in:
Félix Piédallu 2024-02-07 17:09:30 +01:00
parent 75ff50fe1d
commit 5138e099c2

275
update_app_levels/update_app_levels.py Normal file → Executable file
View file

@ -1,109 +1,224 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
"""
Update app catalog: commit, and create a merge request
"""
import argparse
import json import json
import os import logging
import sys
import tempfile import tempfile
import textwrap
import time import time
from collections import OrderedDict from collections import OrderedDict
from typing import Any
from pathlib import Path
import jinja2
import requests import requests
import toml import toml
from git import Repo
token = open(os.path.dirname(__file__) + "/../../.github_token").read().strip() # APPS_REPO = "YunoHost/apps"
APPS_REPO = "Salamandar/apps"
tmpdir = tempfile.mkdtemp(prefix="update_app_levels_")
os.system(f"git clone 'https://oauth2:{token}@github.com/yunohost/apps' {tmpdir}")
os.system(f"git -C {tmpdir} checkout -b update_app_levels")
# Load the app catalog and filter out the non-working ones
catalog = toml.load(open(f"{tmpdir}/apps.toml"))
# Fetch results from the CI
CI_RESULTS_URL = "https://ci-apps.yunohost.org/ci/api/results" CI_RESULTS_URL = "https://ci-apps.yunohost.org/ci/api/results"
ci_results = requests.get(CI_RESULTS_URL).json()
comment = { REPO_APPS_ROOT = Path(Repo(__file__, search_parent_directories=True).working_dir)
"major_regressions": [],
"minor_regressions": [],
"improvements": [],
"outdated": [],
"missing": [],
}
for app, infos in catalog.items(): VERBOSE = False
if infos.get("state") != "working":
continue
if app not in ci_results: def github_token() -> str | None:
comment["missing"].append(app) github_token_path = REPO_APPS_ROOT.parent / ".github_token"
continue if github_token_path.exists():
return github_token_path.open("r", encoding="utf-8").read().strip()
return None
def get_ci_results() -> dict[str, dict[str, Any]]:
return requests.get(CI_RESULTS_URL, timeout=10).json()
def ci_result_is_outdated(result) -> bool:
# 3600 * 24 * 60 = ~2 months # 3600 * 24 * 60 = ~2 months
if (int(time.time()) - ci_results[app].get("timestamp", 0)) > 3600 * 24 * 60: return (int(time.time()) - result.get("timestamp", 0)) > 3600 * 24 * 60
comment["outdated"].append(app)
continue
ci_level = ci_results[app]["level"]
current_level = infos.get("level")
if ci_level == current_level: def update_catalog(catalog, ci_results) -> dict:
continue """
elif current_level is None or ci_level > current_level: Actually change the catalog data
comment["improvements"].append((app, current_level, ci_level)) """
elif ci_level < current_level: # Re-sort the catalog keys / subkeys
if ci_level <= 4 and current_level > 4: for app, infos in catalog.items():
comment["major_regressions"].append((app, current_level, ci_level)) catalog[app] = OrderedDict(sorted(infos.items()))
catalog = OrderedDict(sorted(catalog.items()))
def app_level(app):
if app not in ci_results:
return 0
if ci_result_is_outdated(ci_results[app]):
return 0
return ci_results[app]["level"]
for app, info in catalog.items():
info["level"] = app_level(app)
return catalog
def list_changes(catalog, ci_results) -> dict[str, list[tuple[str, int, int]]]:
"""
Lists changes for a pull request
"""
changes = {
"major_regressions": [],
"minor_regressions": [],
"improvements": [],
"outdated": [],
"missing": [],
}
for app, infos in catalog.items():
if infos.get("state") != "working":
continue
if app not in ci_results:
changes["missing"].append(app)
continue
if ci_result_is_outdated(ci_results[app]):
changes["outdated"].append(app)
continue
ci_level = ci_results[app]["level"]
current_level = infos.get("level")
if ci_level == current_level:
continue
if current_level is None or ci_level > current_level:
changes["improvements"].append((app, current_level, ci_level))
continue
if ci_level < current_level:
if ci_level <= 4 < current_level:
changes["major_regressions"].append((app, current_level, ci_level))
else:
changes["minor_regressions"].append((app, current_level, ci_level))
return changes
def pretty_changes(changes: dict[str, list[tuple[str, int, int]]]) -> str:
pr_body_template = textwrap.dedent("""
{%- if changes["major_regressions"] %}
### Major regressions
{% for app in changes["major_regressions"] %}
- [ ] [{{app.0}}: {{app.1}} -> {{app.2}}](https://ci-apps.yunohost.org/ci/apps/{{app.0}}/latestjob)
{%- endfor %}
{% endif %}
{%- if changes["minor_regressions"] %}
### Minor regressions
{% for app in changes["minor_regressions"] %}
- [ ] [{{app.0}}: {{app.1}} -> {{app.2}}](https://ci-apps.yunohost.org/ci/apps/{{app.0}}/latestjob)
{%- endfor %}
{% endif %}
{%- if changes["improvements"] %}
### Improvements
{% for app in changes["improvements"] %}
- [{{app.0}}: {{app.1}} -> {{app.2}}](https://ci-apps.yunohost.org/ci/apps/{{app.0}}/latestjob)
{%- endfor %}
{% endif %}
{%- if changes["missing"] %}
### Missing
{% for app in changes["missing"] %}
- [{{app}} (See latest job if it exists)](https://ci-apps.yunohost.org/ci/apps/{{app.0}}/latestjob)
{%- endfor %}
{% endif %}
{%- if changes["outdated"] %}
### Outdated
{% for app in changes["outdated"] %}
- [ ] [{{app}} (See latest job if it exists)](https://ci-apps.yunohost.org/ci/apps/{{app.0}}/latestjob)
{%- endfor %}
{% endif %}
""")
return jinja2.Environment().from_string(pr_body_template).render(changes=changes)
def make_pull_request(pr_body: str) -> None:
pr_data = {
"title": "Update app levels according to CI results",
"body": pr_body,
"head": "update_app_levels",
"base": "master"
}
with requests.Session() as s:
s.headers.update({"Authorization": f"token {github_token()}"})
response = s.post(f"https://api.github.com/repos/{APPS_REPO}/pulls", json.dumps(pr_data))
if response.status_code == 422:
response = s.get(f"https://api.github.com/repos/{APPS_REPO}/pulls", data={"head": "update_app_levels"})
existing_url = response.json()[0]["html_url"]
logging.warning(f"A Pull Request already exists at {existing_url} !")
else: else:
comment["minor_regressions"].append((app, current_level, ci_level)) new_url = response.json()["html_url"]
logging.info(f"Opened a Pull Request at {new_url} !")
infos["level"] = ci_level response.raise_for_status()
# Also re-sort the catalog keys / subkeys
for app, infos in catalog.items():
catalog[app] = OrderedDict(sorted(infos.items()))
catalog = OrderedDict(sorted(catalog.items()))
updated_catalog = toml.dumps(catalog) def main():
updated_catalog = updated_catalog.replace(",]", " ]") parser = argparse.ArgumentParser()
open(f"{tmpdir}/apps.toml", "w").write(updated_catalog) parser.add_argument("--commit", action=argparse.BooleanOptionalAction, default=True)
parser.add_argument("--pr", action=argparse.BooleanOptionalAction, default=True)
parser.add_argument("-v", "--verbose", action=argparse.BooleanOptionalAction)
args = parser.parse_args()
os.system(f"git -C {tmpdir} commit apps.toml -m 'Update app levels according to CI results'") logging.getLogger().setLevel(logging.INFO)
os.system(f"git -C {tmpdir} push origin update_app_levels --force") global VERBOSE
os.system(f"rm -rf {tmpdir}") if args.verbose:
VERBOSE = True
logging.getLogger().setLevel(logging.DEBUG)
PR_body = "" with tempfile.TemporaryDirectory(prefix="update_app_levels_") as tmpdir:
if comment["major_regressions"]: logging.info("Cloning the repository...")
PR_body += "\n### Major regressions\n\n" apps_repo = Repo.clone_from(f"git@github.com:{APPS_REPO}", to_path=tmpdir)
for app, current_level, new_level in comment['major_regressions']:
PR_body += f"- [ ] {app} | {current_level} -> {new_level} | https://ci-apps.yunohost.org/ci/apps/{app}/latestjob\n"
if comment["minor_regressions"]:
PR_body += "\n### Minor regressions\n\n"
for app, current_level, new_level in comment['minor_regressions']:
PR_body += f"- [ ] {app} | {current_level} -> {new_level} | https://ci-apps.yunohost.org/ci/apps/{app}/latestjob\n"
if comment["improvements"]:
PR_body += "\n### Improvements\n\n"
for app, current_level, new_level in comment['improvements']:
PR_body += f"- {app} | {current_level} -> {new_level} | https://ci-apps.yunohost.org/ci/apps/{app}/latestjob\n"
if comment["missing"]:
PR_body += "\n### Missing results\n\n"
for app in comment['missing']:
PR_body += f"- {app} | https://ci-apps.yunohost.org/ci/apps/{app}/latestjob\n"
if comment["outdated"]:
PR_body += "\n### Outdated results\n\n"
for app in comment['outdated']:
PR_body += f"- [ ] {app} | https://ci-apps.yunohost.org/ci/apps/{app}/latestjob\n"
PR = {"title": "Update app levels according to CI results", # Load the app catalog and filter out the non-working ones
"body": PR_body, catalog = toml.load((Path(apps_repo.working_tree_dir) / "apps.toml").open("r", encoding="utf-8"))
"head": "update_app_levels",
"base": "master"}
with requests.Session() as s: new_branch = apps_repo.create_head("update_app_levels", apps_repo.refs.master)
s.headers.update({"Authorization": f"token {token}"}) apps_repo.head.reference = new_branch
r = s.post("https://api.github.com/repos/yunohost/apps/pulls", json.dumps(PR))
if r.status_code != 200: logging.info("Retrieving the CI results...")
print(r.text) ci_results = get_ci_results()
sys.exit(1)
# Now compute changes, then update the catalog
changes = list_changes(catalog, ci_results)
pr_body = pretty_changes(changes)
catalog = update_catalog(catalog, ci_results)
# Save the new catalog
updated_catalog = toml.dumps(catalog)
updated_catalog = updated_catalog.replace(",]", " ]")
(Path(apps_repo.working_tree_dir) / "apps.toml").open("w", encoding="utf-8").write(updated_catalog)
if args.commit:
logging.info("Committing and pushing the new catalog...")
apps_repo.index.add("apps.toml")
apps_repo.index.commit("Update app levels according to CI results")
apps_repo.remote().push(force=True)
if VERBOSE:
print(pr_body)
if args.pr:
logging.info("Opening a pull request...")
make_pull_request(pr_body)
if __name__ == "__main__":
main()