Merge pull request #2027 from Salamandar/fix_autoupdater
Fix autoupdater
This commit is contained in:
commit
64e1467143
1 changed files with 62 additions and 31 deletions
|
@ -4,6 +4,7 @@ import argparse
|
||||||
import hashlib
|
import hashlib
|
||||||
import multiprocessing
|
import multiprocessing
|
||||||
import logging
|
import logging
|
||||||
|
from enum import Enum
|
||||||
from typing import Any, Optional, Union
|
from typing import Any, Optional, Union
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
|
@ -163,6 +164,13 @@ class LocalOrRemoteRepo:
|
||||||
return next(pull.html_url for pull in self.repo.get_pulls(head=branch))
|
return next(pull.html_url for pull in self.repo.get_pulls(head=branch))
|
||||||
|
|
||||||
|
|
||||||
|
class State(Enum):
|
||||||
|
up_to_date = 0
|
||||||
|
already = 1
|
||||||
|
created = 2
|
||||||
|
failure = 3
|
||||||
|
|
||||||
|
|
||||||
class AppAutoUpdater:
|
class AppAutoUpdater:
|
||||||
def __init__(self, app_id: Union[str, Path]) -> None:
|
def __init__(self, app_id: Union[str, Path]) -> None:
|
||||||
self.repo = LocalOrRemoteRepo(app_id)
|
self.repo = LocalOrRemoteRepo(app_id)
|
||||||
|
@ -178,11 +186,10 @@ class AppAutoUpdater:
|
||||||
|
|
||||||
self.main_upstream = self.manifest.get("upstream", {}).get("code")
|
self.main_upstream = self.manifest.get("upstream", {}).get("code")
|
||||||
|
|
||||||
def run(self, edit: bool = False, commit: bool = False, pr: bool = False
|
def run(self, edit: bool = False, commit: bool = False, pr: bool = False) -> tuple[State, str, str, str]:
|
||||||
) -> Union[bool, tuple[Optional[str], Optional[str], Optional[str]]]:
|
state = State.up_to_date
|
||||||
has_updates = False
|
main_version = ""
|
||||||
main_version = None
|
pr_url = ""
|
||||||
pr_url = None
|
|
||||||
|
|
||||||
# Default message
|
# Default message
|
||||||
pr_title = commit_msg = "Upgrade sources"
|
pr_title = commit_msg = "Upgrade sources"
|
||||||
|
@ -192,7 +199,8 @@ class AppAutoUpdater:
|
||||||
update = self.get_source_update(source, infos)
|
update = self.get_source_update(source, infos)
|
||||||
if update is None:
|
if update is None:
|
||||||
continue
|
continue
|
||||||
has_updates = True
|
# We assume we'll create a PR
|
||||||
|
state = State.created
|
||||||
version, assets, msg = update
|
version, assets, msg = update
|
||||||
|
|
||||||
if source == "main":
|
if source == "main":
|
||||||
|
@ -206,8 +214,8 @@ class AppAutoUpdater:
|
||||||
self.repo.manifest_raw, version, assets, infos, is_main=source == "main",
|
self.repo.manifest_raw, version, assets, infos, is_main=source == "main",
|
||||||
)
|
)
|
||||||
|
|
||||||
if not has_updates:
|
if state == State.up_to_date:
|
||||||
return False
|
return (State.up_to_date, "", "", "")
|
||||||
|
|
||||||
if edit:
|
if edit:
|
||||||
self.repo.edit_manifest(self.repo.manifest_raw)
|
self.repo.edit_manifest(self.repo.manifest_raw)
|
||||||
|
@ -218,13 +226,14 @@ class AppAutoUpdater:
|
||||||
if commit:
|
if commit:
|
||||||
self.repo.commit(commit_msg)
|
self.repo.commit(commit_msg)
|
||||||
if pr:
|
if pr:
|
||||||
pr_url = self.repo.create_pr(branch_name, pr_title, commit_msg)
|
pr_url = self.repo.create_pr(branch_name, pr_title, commit_msg) or ""
|
||||||
except github.GithubException as e:
|
except github.GithubException as e:
|
||||||
if e.status == 422 or e.status == 409:
|
if e.status == 422 or e.status == 409:
|
||||||
pr_url = f"already existing pr: {self.repo.get_pr(branch_name)}"
|
state = State.already
|
||||||
|
pr_url = self.repo.get_pr(branch_name)
|
||||||
else:
|
else:
|
||||||
raise
|
raise
|
||||||
return self.current_version, main_version, pr_url
|
return (state, self.current_version, main_version, pr_url)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def filter_and_get_latest_tag(tags: list[str], app_id: str) -> tuple[str, str]:
|
def filter_and_get_latest_tag(tags: list[str], app_id: str) -> tuple[str, str]:
|
||||||
|
@ -276,7 +285,8 @@ class AppAutoUpdater:
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
raise RuntimeError(f"Failed to compute sha256 for {url} : {e}") from e
|
raise RuntimeError(f"Failed to compute sha256 for {url} : {e}") from e
|
||||||
|
|
||||||
def get_source_update(self, name: str, infos: dict[str, Any]) -> Optional[tuple[str, Union[str, dict[str, str]], str]]:
|
def get_source_update(self, name: str, infos: dict[str, Any]
|
||||||
|
) -> Optional[tuple[str, Union[str, dict[str, str]], str]]:
|
||||||
if "autoupdate" not in infos:
|
if "autoupdate" not in infos:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
@ -485,28 +495,29 @@ class StdoutSwitch:
|
||||||
sys.stdout = self.save_stdout
|
sys.stdout = self.save_stdout
|
||||||
|
|
||||||
|
|
||||||
def run_autoupdate_for_multiprocessing(data) -> Optional[tuple[bool, str, Any]]:
|
def run_autoupdate_for_multiprocessing(data) -> tuple[str, tuple[State, str, str, str]]:
|
||||||
app, edit, commit, pr = data
|
app, edit, commit, pr = data
|
||||||
stdoutswitch = StdoutSwitch()
|
stdoutswitch = StdoutSwitch()
|
||||||
try:
|
try:
|
||||||
result = AppAutoUpdater(app).run(edit=edit, commit=commit, pr=pr)
|
result = AppAutoUpdater(app).run(edit=edit, commit=commit, pr=pr)
|
||||||
if result is False:
|
return (app, result)
|
||||||
return None
|
|
||||||
return True, app, result
|
|
||||||
except Exception:
|
except Exception:
|
||||||
log_str = stdoutswitch.reset()
|
log_str = stdoutswitch.reset()
|
||||||
import traceback
|
import traceback
|
||||||
t = traceback.format_exc()
|
t = traceback.format_exc()
|
||||||
return False, app, f"{log_str}\n{t}"
|
return (app, (State.failure, log_str, str(t), ""))
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
def main() -> None:
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument("apps", nargs="*", type=Path,
|
parser.add_argument("apps", nargs="*", type=Path,
|
||||||
help="If not passed, the script will run on the catalog. Github keys required.")
|
help="If not passed, the script will run on the catalog. Github keys required.")
|
||||||
parser.add_argument("--edit", action=argparse.BooleanOptionalAction, help="Edit the local files", default=True)
|
parser.add_argument("--edit", action=argparse.BooleanOptionalAction, default=True,
|
||||||
parser.add_argument("--commit", action=argparse.BooleanOptionalAction, help="Create a commit with the changes")
|
help="Edit the local files")
|
||||||
parser.add_argument("--pr", action=argparse.BooleanOptionalAction, help="Create a pull request with the changes")
|
parser.add_argument("--commit", action=argparse.BooleanOptionalAction, default=False,
|
||||||
|
help="Create a commit with the changes")
|
||||||
|
parser.add_argument("--pr", action=argparse.BooleanOptionalAction, default=False,
|
||||||
|
help="Create a pull request with the changes")
|
||||||
parser.add_argument("--paste", action="store_true")
|
parser.add_argument("--paste", action="store_true")
|
||||||
parser.add_argument("-j", "--processes", type=int, default=multiprocessing.cpu_count())
|
parser.add_argument("-j", "--processes", type=int, default=multiprocessing.cpu_count())
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
@ -522,23 +533,43 @@ def main() -> None:
|
||||||
|
|
||||||
# Handle apps or no apps
|
# Handle apps or no apps
|
||||||
apps = list(args.apps) if args.apps else apps_to_run_auto_update_for()
|
apps = list(args.apps) if args.apps else apps_to_run_auto_update_for()
|
||||||
apps_failed = {}
|
apps_already = {} # for which a PR already exists
|
||||||
apps_updated = {}
|
apps_updated = {}
|
||||||
|
apps_failed = {}
|
||||||
|
|
||||||
with multiprocessing.Pool(processes=args.processes) as pool:
|
with multiprocessing.Pool(processes=args.processes) as pool:
|
||||||
tasks = pool.imap(run_autoupdate_for_multiprocessing,
|
tasks = pool.imap(run_autoupdate_for_multiprocessing,
|
||||||
((app, args.edit, args.commit, args.pr) for app in apps))
|
((app, args.edit, args.commit, args.pr) for app in apps))
|
||||||
for result in tqdm.tqdm(tasks, total=len(apps), ascii=" ·#"):
|
for app, result in tqdm.tqdm(tasks, total=len(apps), ascii=" ·#"):
|
||||||
if result is None:
|
state, current_version, main_version, pr_url = result
|
||||||
|
if state == State.up_to_date:
|
||||||
continue
|
continue
|
||||||
is_ok, app, info = result
|
if state == State.already:
|
||||||
if is_ok:
|
apps_already[app] = (current_version, main_version, pr_url)
|
||||||
apps_updated[app] = info
|
if state == State.created:
|
||||||
else:
|
apps_updated[app] = (current_version, main_version, pr_url)
|
||||||
apps_failed[app] = info
|
if state == State.failure:
|
||||||
pass
|
apps_failed[app] = current_version, main_version # actually stores logs
|
||||||
|
|
||||||
result_message = ""
|
result_message = ""
|
||||||
|
if apps_already:
|
||||||
|
result_message += f"\n{'=' * 80}\nApps already with an update PR:"
|
||||||
|
for app, info in apps_already.items():
|
||||||
|
result_message += f"\n- {app}"
|
||||||
|
if isinstance(info, tuple):
|
||||||
|
result_message += f" ({info[0]} -> {info[1]})"
|
||||||
|
if info[2] is not None:
|
||||||
|
result_message += f" see {info[2]}"
|
||||||
|
|
||||||
|
if apps_updated:
|
||||||
|
result_message += f"\n{'=' * 80}\nApps updated:"
|
||||||
|
for app, info in apps_updated.items():
|
||||||
|
result_message += f"\n- {app}"
|
||||||
|
if isinstance(info, tuple):
|
||||||
|
result_message += f" ({info[0]} -> {info[1]})"
|
||||||
|
if info[2] is not None:
|
||||||
|
result_message += f" see {info[2]}"
|
||||||
|
|
||||||
if apps_updated:
|
if apps_updated:
|
||||||
result_message += f"\n{'=' * 80}\nApps updated:"
|
result_message += f"\n{'=' * 80}\nApps updated:"
|
||||||
for app, info in apps_updated.items():
|
for app, info in apps_updated.items():
|
||||||
|
@ -550,8 +581,8 @@ def main() -> None:
|
||||||
|
|
||||||
if apps_failed:
|
if apps_failed:
|
||||||
result_message += f"\n{'=' * 80}\nApps failed:"
|
result_message += f"\n{'=' * 80}\nApps failed:"
|
||||||
for app, info in apps_failed.items():
|
for app, logs in apps_failed.items():
|
||||||
result_message += f"\n{'='*40}\n{app}\n{'-'*40}\n{info}\n\n"
|
result_message += f"\n{'='*40}\n{app}\n{'-'*40}\n{logs[0]}\n{logs[1]}\n\n"
|
||||||
|
|
||||||
if apps_failed and args.paste:
|
if apps_failed and args.paste:
|
||||||
paste_url = paste_on_haste(result_message)
|
paste_url = paste_on_haste(result_message)
|
||||||
|
|
Loading…
Reference in a new issue