finish autoupdater rework
This commit is contained in:
parent
878ea4640a
commit
a791da9c3e
1 changed files with 174 additions and 109 deletions
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
import hashlib
|
import hashlib
|
||||||
|
import multiprocessing
|
||||||
import logging
|
import logging
|
||||||
from typing import Any
|
from typing import Any
|
||||||
import re
|
import re
|
||||||
|
@ -14,7 +15,6 @@ from datetime import datetime
|
||||||
import requests
|
import requests
|
||||||
import toml
|
import toml
|
||||||
import tqdm
|
import tqdm
|
||||||
from tqdm.contrib.logging import logging_redirect_tqdm
|
|
||||||
import github
|
import github
|
||||||
|
|
||||||
# add apps/tools to sys.path
|
# add apps/tools to sys.path
|
||||||
|
@ -76,56 +76,6 @@ def apps_to_run_auto_update_for():
|
||||||
return relevant_apps
|
return relevant_apps
|
||||||
|
|
||||||
|
|
||||||
def filter_and_get_latest_tag(tags: list[str], app_id: str) -> tuple[str, str]:
|
|
||||||
def version_numbers(tag: str) -> list[int] | None:
|
|
||||||
filter_keywords = ["start", "rc", "beta", "alpha"]
|
|
||||||
if any(keyword in tag for keyword in filter_keywords):
|
|
||||||
logging.debug(f"Tag {tag} contains filtered keyword from {filter_keywords}.")
|
|
||||||
return None
|
|
||||||
|
|
||||||
t_to_check = tag
|
|
||||||
if tag.startswith(app_id + "-"):
|
|
||||||
t_to_check = tag.split("-", 1)[-1]
|
|
||||||
# Boring special case for dokuwiki...
|
|
||||||
elif tag.startswith("release-"):
|
|
||||||
t_to_check = tag.split("-", 1)[-1].replace("-", ".")
|
|
||||||
|
|
||||||
if re.match(r"^v?[\d\.]*\-?\d$", t_to_check):
|
|
||||||
return list(tag_to_int_tuple(t_to_check))
|
|
||||||
print(f"Ignoring tag {t_to_check}, doesn't look like a version number")
|
|
||||||
return None
|
|
||||||
|
|
||||||
# sorted will sort by keys
|
|
||||||
tags_dict: dict[list[int] | None, str] = dict(sorted({
|
|
||||||
version_numbers(tag): tag for tag in tags
|
|
||||||
}.items()))
|
|
||||||
tags_dict.pop(None, None)
|
|
||||||
if not tags_dict:
|
|
||||||
raise RuntimeError("No tags were found after sanity filtering!")
|
|
||||||
the_tag_list, the_tag = next(iter(tags_dict.items()))
|
|
||||||
assert the_tag_list is not None
|
|
||||||
return the_tag, ".".join(str(i) for i in the_tag_list)
|
|
||||||
|
|
||||||
|
|
||||||
def tag_to_int_tuple(tag) -> tuple[int, ...]:
|
|
||||||
tag = tag.strip("v").replace("-", ".").strip(".")
|
|
||||||
int_tuple = tag.split(".")
|
|
||||||
assert all(i.isdigit() for i in int_tuple), f"Cant convert {tag} to int tuple :/"
|
|
||||||
return tuple(int(i) for i in int_tuple)
|
|
||||||
|
|
||||||
|
|
||||||
def sha256_of_remote_file(url: str) -> str:
|
|
||||||
print(f"Computing sha256sum for {url} ...")
|
|
||||||
try:
|
|
||||||
r = requests.get(url, stream=True)
|
|
||||||
m = hashlib.sha256()
|
|
||||||
for data in r.iter_content(8192):
|
|
||||||
m.update(data)
|
|
||||||
return m.hexdigest()
|
|
||||||
except Exception as e:
|
|
||||||
raise RuntimeError(f"Failed to compute sha256 for {url} : {e}") from e
|
|
||||||
|
|
||||||
|
|
||||||
class LocalOrRemoteRepo:
|
class LocalOrRemoteRepo:
|
||||||
def __init__(self, app: str | Path) -> None:
|
def __init__(self, app: str | Path) -> None:
|
||||||
self.local = False
|
self.local = False
|
||||||
|
@ -181,16 +131,20 @@ class LocalOrRemoteRepo:
|
||||||
author=author,
|
author=author,
|
||||||
)
|
)
|
||||||
|
|
||||||
def new_branch(self, name: str):
|
def new_branch(self, name: str) -> bool:
|
||||||
if self.local:
|
if self.local:
|
||||||
logging.warning("Can't create branches for local repositories")
|
logging.warning("Can't create branches for local repositories")
|
||||||
return
|
return False
|
||||||
if self.remote:
|
if self.remote:
|
||||||
self.pr_branch = name
|
self.pr_branch = name
|
||||||
commit_sha = self.repo.get_branch(self.base_branch).commit.sha
|
commit_sha = self.repo.get_branch(self.base_branch).commit.sha
|
||||||
|
if self.pr_branch in [branch.name for branch in self.repo.get_branches()]:
|
||||||
|
return False
|
||||||
self.repo.create_git_ref(ref=f"refs/heads/{name}", sha=commit_sha)
|
self.repo.create_git_ref(ref=f"refs/heads/{name}", sha=commit_sha)
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
def create_pr(self, branch: str, title: str, message: str):
|
def create_pr(self, branch: str, title: str, message: str) -> str | None:
|
||||||
if self.local:
|
if self.local:
|
||||||
logging.warning("Can't create pull requests for local repositories")
|
logging.warning("Can't create pull requests for local repositories")
|
||||||
return
|
return
|
||||||
|
@ -199,7 +153,10 @@ class LocalOrRemoteRepo:
|
||||||
pr = self.repo.create_pull(
|
pr = self.repo.create_pull(
|
||||||
title=title, body=message, head=branch, base=self.base_branch
|
title=title, body=message, head=branch, base=self.base_branch
|
||||||
)
|
)
|
||||||
print("Created PR " + self.repo.full_name + " updated with PR #" + str(pr.id))
|
return pr.url
|
||||||
|
|
||||||
|
def get_pr(self, branch: str) -> str:
|
||||||
|
return next(pull.html_url for pull in self.repo.get_pulls(head=branch))
|
||||||
|
|
||||||
|
|
||||||
class AppAutoUpdater:
|
class AppAutoUpdater:
|
||||||
|
@ -217,8 +174,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) -> bool:
|
def run(self, edit: bool = False, commit: bool = False, pr: bool = False) -> bool | tuple[str | None, str | None, str | None]:
|
||||||
has_updates = False
|
has_updates = False
|
||||||
|
main_version = None
|
||||||
|
pr_url = None
|
||||||
|
|
||||||
# Default message
|
# Default message
|
||||||
pr_title = commit_msg = "Upgrade sources"
|
pr_title = commit_msg = "Upgrade sources"
|
||||||
|
@ -226,13 +185,13 @@ class AppAutoUpdater:
|
||||||
|
|
||||||
for source, infos in self.sources.items():
|
for source, infos in self.sources.items():
|
||||||
update = self.get_source_update(source, infos)
|
update = self.get_source_update(source, infos)
|
||||||
print(update)
|
|
||||||
if update is None:
|
if update is None:
|
||||||
continue
|
continue
|
||||||
has_updates = True
|
has_updates = True
|
||||||
version, assets, msg = update
|
version, assets, msg = update
|
||||||
|
|
||||||
if source == "main":
|
if source == "main":
|
||||||
|
main_version = version
|
||||||
branch_name = f"ci-auto-update-{version}"
|
branch_name = f"ci-auto-update-{version}"
|
||||||
pr_title = commit_msg = f"Upgrade to v{version}"
|
pr_title = commit_msg = f"Upgrade to v{version}"
|
||||||
if msg:
|
if msg:
|
||||||
|
@ -242,16 +201,76 @@ 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:
|
||||||
|
return False
|
||||||
|
|
||||||
if edit:
|
if edit:
|
||||||
self.repo.edit_manifest(self.repo.manifest_raw)
|
self.repo.edit_manifest(self.repo.manifest_raw)
|
||||||
if pr:
|
|
||||||
self.repo.new_branch(branch_name)
|
|
||||||
if commit:
|
|
||||||
self.repo.commit(commit_msg)
|
|
||||||
if pr:
|
|
||||||
self.repo.create_pr(branch_name, pr_title, commit_msg)
|
|
||||||
|
|
||||||
return has_updates
|
try:
|
||||||
|
if pr:
|
||||||
|
self.repo.new_branch(branch_name)
|
||||||
|
if commit:
|
||||||
|
self.repo.commit(commit_msg)
|
||||||
|
if pr:
|
||||||
|
pr_url = self.repo.create_pr(branch_name, pr_title, commit_msg)
|
||||||
|
except github.GithubException as e:
|
||||||
|
if e.status == 422 or e.status == 409:
|
||||||
|
pr_url = f"already existing pr: {self.repo.get_pr(branch_name)}"
|
||||||
|
else:
|
||||||
|
raise
|
||||||
|
return self.current_version, main_version, pr_url
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def filter_and_get_latest_tag(tags: list[str], app_id: str) -> tuple[str, str]:
|
||||||
|
def version_numbers(tag: str) -> tuple[int, ...] | None:
|
||||||
|
filter_keywords = ["start", "rc", "beta", "alpha"]
|
||||||
|
if any(keyword in tag for keyword in filter_keywords):
|
||||||
|
logging.debug(f"Tag {tag} contains filtered keyword from {filter_keywords}.")
|
||||||
|
return None
|
||||||
|
|
||||||
|
t_to_check = tag
|
||||||
|
if tag.startswith(app_id + "-"):
|
||||||
|
t_to_check = tag.split("-", 1)[-1]
|
||||||
|
# Boring special case for dokuwiki...
|
||||||
|
elif tag.startswith("release-"):
|
||||||
|
t_to_check = tag.split("-", 1)[-1].replace("-", ".")
|
||||||
|
|
||||||
|
if re.match(r"^v?[\d\.]*\-?\d$", t_to_check):
|
||||||
|
return AppAutoUpdater.tag_to_int_tuple(t_to_check)
|
||||||
|
print(f"Ignoring tag {t_to_check}, doesn't look like a version number")
|
||||||
|
return None
|
||||||
|
|
||||||
|
# sorted will sort by keys
|
||||||
|
tags_dict = {version_numbers(tag): tag for tag in tags}
|
||||||
|
tags_dict.pop(None, None)
|
||||||
|
# reverse=True will set the last release as first element
|
||||||
|
tags_dict = dict(sorted(tags_dict.items(), reverse=True))
|
||||||
|
if not tags_dict:
|
||||||
|
raise RuntimeError("No tags were found after sanity filtering!")
|
||||||
|
the_tag_list, the_tag = next(iter(tags_dict.items()))
|
||||||
|
assert the_tag_list is not None
|
||||||
|
return the_tag, ".".join(str(i) for i in the_tag_list)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def tag_to_int_tuple(tag: str) -> tuple[int, ...]:
|
||||||
|
tag = tag.strip("v").replace("-", ".").strip(".")
|
||||||
|
int_tuple = tag.split(".")
|
||||||
|
assert all(i.isdigit() for i in int_tuple), f"Cant convert {tag} to int tuple :/"
|
||||||
|
return tuple(int(i) for i in int_tuple)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def sha256_of_remote_file(url: str) -> str:
|
||||||
|
print(f"Computing sha256sum for {url} ...")
|
||||||
|
return ""
|
||||||
|
try:
|
||||||
|
r = requests.get(url, stream=True)
|
||||||
|
m = hashlib.sha256()
|
||||||
|
for data in r.iter_content(8192):
|
||||||
|
m.update(data)
|
||||||
|
return m.hexdigest()
|
||||||
|
except Exception as e:
|
||||||
|
raise RuntimeError(f"Failed to compute sha256 for {url} : {e}") from e
|
||||||
|
|
||||||
def get_source_update(self, name: str, infos: dict[str, Any]) -> tuple[str, str | dict[str, str], str] | None:
|
def get_source_update(self, name: str, infos: dict[str, Any]) -> tuple[str, str | dict[str, str], str] | None:
|
||||||
if "autoupdate" not in infos:
|
if "autoupdate" not in infos:
|
||||||
|
@ -281,7 +300,7 @@ class AppAutoUpdater:
|
||||||
print("Up to date")
|
print("Up to date")
|
||||||
return None
|
return None
|
||||||
try:
|
try:
|
||||||
if tag_to_int_tuple(self.current_version) > tag_to_int_tuple(new_version):
|
if self.tag_to_int_tuple(self.current_version) > self.tag_to_int_tuple(new_version):
|
||||||
print("Up to date (current version appears more recent than newest version found)")
|
print("Up to date (current version appears more recent than newest version found)")
|
||||||
return None
|
return None
|
||||||
except (AssertionError, ValueError):
|
except (AssertionError, ValueError):
|
||||||
|
@ -310,7 +329,7 @@ class AppAutoUpdater:
|
||||||
name: url for name, url in assets.items() if re.match(regex, name)
|
name: url for name, url in assets.items() if re.match(regex, name)
|
||||||
}
|
}
|
||||||
if not matching_assets:
|
if not matching_assets:
|
||||||
raise RuntimeError(f"No assets matching regex '{regex}'")
|
raise RuntimeError(f"No assets matching regex '{regex}' in {list(assets.keys())}")
|
||||||
if len(matching_assets) > 1:
|
if len(matching_assets) > 1:
|
||||||
raise RuntimeError(f"Too many assets matching regex '{regex}': {matching_assets}")
|
raise RuntimeError(f"Too many assets matching regex '{regex}': {matching_assets}")
|
||||||
return next(iter(matching_assets.items()))
|
return next(iter(matching_assets.items()))
|
||||||
|
@ -336,7 +355,7 @@ class AppAutoUpdater:
|
||||||
for release in api.releases()
|
for release in api.releases()
|
||||||
if not release["draft"] and not release["prerelease"]
|
if not release["draft"] and not release["prerelease"]
|
||||||
}
|
}
|
||||||
latest_version_orig, latest_version = filter_and_get_latest_tag(list(releases.keys()), self.app_id)
|
latest_version_orig, latest_version = self.filter_and_get_latest_tag(list(releases.keys()), self.app_id)
|
||||||
latest_release = releases[latest_version_orig]
|
latest_release = releases[latest_version_orig]
|
||||||
latest_assets = {
|
latest_assets = {
|
||||||
a["name"]: a["browser_download_url"]
|
a["name"]: a["browser_download_url"]
|
||||||
|
@ -375,7 +394,7 @@ class AppAutoUpdater:
|
||||||
if asset != "tarball":
|
if asset != "tarball":
|
||||||
raise ValueError("For the latest tag strategies, only asset = 'tarball' is supported")
|
raise ValueError("For the latest tag strategies, only asset = 'tarball' is supported")
|
||||||
tags = [t["name"] for t in api.tags()]
|
tags = [t["name"] for t in api.tags()]
|
||||||
latest_version_orig, latest_version = filter_and_get_latest_tag(tags, self.app_id)
|
latest_version_orig, latest_version = self.filter_and_get_latest_tag(tags, self.app_id)
|
||||||
latest_tarball = api.url_for_ref(latest_version_orig, RefType.tags)
|
latest_tarball = api.url_for_ref(latest_version_orig, RefType.tags)
|
||||||
return latest_version, latest_tarball, ""
|
return latest_version, latest_tarball, ""
|
||||||
|
|
||||||
|
@ -397,14 +416,14 @@ class AppAutoUpdater:
|
||||||
if isinstance(new_assets_urls, str):
|
if isinstance(new_assets_urls, str):
|
||||||
replacements = [
|
replacements = [
|
||||||
(current_assets["url"], new_assets_urls),
|
(current_assets["url"], new_assets_urls),
|
||||||
(current_assets["sha256"], sha256_of_remote_file(new_assets_urls)),
|
(current_assets["sha256"], self.sha256_of_remote_file(new_assets_urls)),
|
||||||
]
|
]
|
||||||
if isinstance(new_assets_urls, dict):
|
if isinstance(new_assets_urls, dict):
|
||||||
replacements = [
|
replacements = [
|
||||||
repl
|
repl
|
||||||
for key, url in new_assets_urls.items() for repl in (
|
for key, url in new_assets_urls.items() for repl in (
|
||||||
(current_assets[key]["url"], url),
|
(current_assets[key]["url"], url),
|
||||||
(current_assets[key]["sha256"], sha256_of_remote_file(url))
|
(current_assets[key]["sha256"], self.sha256_of_remote_file(url))
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -423,26 +442,66 @@ def paste_on_haste(data):
|
||||||
# NB: we hardcode this here and can't use the yunopaste command
|
# NB: we hardcode this here and can't use the yunopaste command
|
||||||
# because this script runs on the same machine than haste is hosted on...
|
# because this script runs on the same machine than haste is hosted on...
|
||||||
# and doesn't have the proper front-end LE cert in this context
|
# and doesn't have the proper front-end LE cert in this context
|
||||||
SERVER_URL = "http://paste.yunohost.org"
|
SERVER_HOST = "http://paste.yunohost.org"
|
||||||
TIMEOUT = 3
|
TIMEOUT = 3
|
||||||
try:
|
try:
|
||||||
url = SERVER_URL + "/documents"
|
url = f"{SERVER_HOST}/documents"
|
||||||
response = requests.post(url, data=data.encode("utf-8"), timeout=TIMEOUT)
|
response = requests.post(url, data=data.encode("utf-8"), timeout=TIMEOUT)
|
||||||
response.raise_for_status()
|
response.raise_for_status()
|
||||||
dockey = response.json()["key"]
|
dockey = response.json()["key"]
|
||||||
return SERVER_URL + "/raw/" + dockey
|
return f"{SERVER_HOST}/raw/{dockey}"
|
||||||
except requests.exceptions.RequestException as e:
|
except requests.exceptions.RequestException as e:
|
||||||
logging.error("\033[31mError: {}\033[0m".format(e))
|
logging.error("\033[31mError: {}\033[0m".format(e))
|
||||||
sys.exit(1)
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
class StdoutSwitch:
|
||||||
|
|
||||||
|
class DummyFile:
|
||||||
|
def __init__(self):
|
||||||
|
self.result = ""
|
||||||
|
|
||||||
|
def write(self, x):
|
||||||
|
self.result += x
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
self.save_stdout = sys.stdout
|
||||||
|
sys.stdout = self.DummyFile()
|
||||||
|
|
||||||
|
def reset(self) -> str:
|
||||||
|
result = ""
|
||||||
|
if isinstance(sys.stdout, self.DummyFile):
|
||||||
|
result = sys.stdout.result
|
||||||
|
sys.stdout = self.save_stdout
|
||||||
|
return result
|
||||||
|
|
||||||
|
def __exit__(self) -> None:
|
||||||
|
sys.stdout = self.save_stdout
|
||||||
|
|
||||||
|
|
||||||
|
def run_autoupdate_for_multiprocessing(data) -> tuple[bool, str, Any] | None:
|
||||||
|
app, edit, commit, pr = data
|
||||||
|
# stdoutswitch = StdoutSwitch()
|
||||||
|
try:
|
||||||
|
result = AppAutoUpdater(app).run(edit=edit, commit=commit, pr=pr)
|
||||||
|
if result is not False:
|
||||||
|
return True, app, result
|
||||||
|
except Exception:
|
||||||
|
# result = stdoutswitch.reset()
|
||||||
|
import traceback
|
||||||
|
t = traceback.format_exc()
|
||||||
|
return False, app, f"{result}\n{t}"
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
def main() -> None:
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument("app_dir", nargs="?", type=Path)
|
parser.add_argument("apps", nargs="*", type=Path,
|
||||||
|
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, help="Edit the local files", default=True)
|
||||||
parser.add_argument("--commit", action=argparse.BooleanOptionalAction, help="Create a commit with the changes")
|
parser.add_argument("--commit", action=argparse.BooleanOptionalAction, help="Create a commit with the changes")
|
||||||
parser.add_argument("--pr", action=argparse.BooleanOptionalAction, help="Create a pull request with the changes")
|
parser.add_argument("--pr", action=argparse.BooleanOptionalAction, 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())
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
if args.commit and not args.edit:
|
if args.commit and not args.edit:
|
||||||
|
@ -450,43 +509,49 @@ def main() -> None:
|
||||||
if args.pr and not args.commit:
|
if args.pr and not args.commit:
|
||||||
parser.error("--pr requires --commit")
|
parser.error("--pr requires --commit")
|
||||||
|
|
||||||
if args.app_dir:
|
# Handle apps or no apps
|
||||||
AppAutoUpdater(args.app_dir).run(edit=args.edit, commit=args.commit, pr=args.pr)
|
apps = list(args.apps) if args.apps else ["mobilizon"] # apps_to_run_auto_update_for()
|
||||||
else:
|
apps_failed = {}
|
||||||
apps_failed = {}
|
apps_updated = {}
|
||||||
apps_updated = []
|
|
||||||
|
|
||||||
with logging_redirect_tqdm():
|
with multiprocessing.Pool(processes=args.processes) as pool:
|
||||||
for app in tqdm.tqdm(apps_to_run_auto_update_for(), ascii=" ·#"):
|
tasks = pool.imap(run_autoupdate_for_multiprocessing,
|
||||||
try:
|
((app, args.edit, args.commit, args.pr) for app in apps))
|
||||||
if AppAutoUpdater(app).run(edit=args.edit, commit=args.commit, pr=args.pr):
|
for result in tqdm.tqdm(tasks, total=len(apps), ascii=" ·#"):
|
||||||
apps_updated.append(app)
|
if result is None:
|
||||||
except Exception:
|
continue
|
||||||
import traceback
|
is_ok, app, info = result
|
||||||
|
if is_ok:
|
||||||
t = traceback.format_exc()
|
apps_updated[app] = info
|
||||||
apps_failed[app] = t
|
|
||||||
logging.error(t)
|
|
||||||
|
|
||||||
if apps_failed:
|
|
||||||
error_log = "\n=========\n".join(
|
|
||||||
[
|
|
||||||
f"{app}\n-------\n{trace}\n\n"
|
|
||||||
for app, trace in apps_failed.items()
|
|
||||||
]
|
|
||||||
)
|
|
||||||
if args.paste:
|
|
||||||
paste_url = paste_on_haste(error_log)
|
|
||||||
logging.error(textwrap.dedent(f"""
|
|
||||||
Failed to run the source auto-update for: {', '.join(apps_failed.keys())}
|
|
||||||
Please run manually the `autoupdate_app_sources.py` script on these apps to debug what is happening!
|
|
||||||
See the debug log here: {paste_url}"
|
|
||||||
"""))
|
|
||||||
else:
|
else:
|
||||||
print(error_log)
|
apps_failed[app] = info
|
||||||
|
pass
|
||||||
|
|
||||||
if apps_updated:
|
result_message = ""
|
||||||
print(f"Apps updated: {', '.join(apps_updated)}")
|
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):
|
||||||
|
print(info)
|
||||||
|
result_message += f" ({info[0]} -> {info[1]})"
|
||||||
|
if info[2] is not None:
|
||||||
|
result_message += f" see {info[2]}"
|
||||||
|
|
||||||
|
if apps_failed:
|
||||||
|
result_message += f"\n{'=' * 80}\nApps failed:"
|
||||||
|
for app, info in apps_failed.items():
|
||||||
|
result_message += f"\n{'='*40}\n{app}\n{'-'*40}\n{info}\n\n"
|
||||||
|
|
||||||
|
if apps_failed and args.paste:
|
||||||
|
paste_url = paste_on_haste(result_message)
|
||||||
|
logging.error(textwrap.dedent(f"""
|
||||||
|
Failed to run the source auto-update for: {', '.join(apps_failed.keys())}
|
||||||
|
Please run manually the `autoupdate_app_sources.py` script on these apps to debug what is happening!
|
||||||
|
See the debug log here: {paste_url}"
|
||||||
|
"""))
|
||||||
|
|
||||||
|
print(result_message)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|
Loading…
Reference in a new issue