2023-03-27 17:49:48 +02:00
import time
2023-03-21 00:24:52 +01:00
import hashlib
2023-03-13 17:40:35 +01:00
import re
import sys
import requests
import toml
import os
2023-03-27 18:41:38 +02:00
import glob
2023-04-04 18:32:12 +02:00
from datetime import datetime
2023-03-13 17:40:35 +01:00
2023-03-28 19:13:50 +02:00
STRATEGIES = [ " latest_github_release " , " latest_github_tag " , " latest_github_commit " ]
2023-03-27 17:49:48 +02:00
2023-04-04 16:25:12 +02:00
if " --commit-and-create-PR " not in sys . argv :
2023-03-28 19:13:50 +02:00
dry_run = True
else :
dry_run = False
2023-03-13 17:40:35 +01:00
2023-04-04 18:15:35 +02:00
args = [ arg for arg in sys . argv [ 1 : ] if arg != " --commit-and-create-PR " ]
if len ( args ) :
auth = None
else :
2023-03-28 19:28:11 +02:00
GITHUB_LOGIN = (
open ( os . path . dirname ( __file__ ) + " /../../.github_login " ) . read ( ) . strip ( )
)
GITHUB_TOKEN = (
open ( os . path . dirname ( __file__ ) + " /../../.github_token " ) . read ( ) . strip ( )
)
GITHUB_EMAIL = (
open ( os . path . dirname ( __file__ ) + " /../../.github_email " ) . read ( ) . strip ( )
)
2023-03-13 17:40:35 +01:00
2023-03-28 19:13:50 +02:00
from github import Github , InputGitAuthor
2023-03-28 19:28:11 +02:00
2023-04-04 18:15:35 +02:00
auth = ( GITHUB_LOGIN , GITHUB_TOKEN )
2023-03-28 19:13:50 +02:00
github = Github ( GITHUB_TOKEN )
author = InputGitAuthor ( GITHUB_LOGIN , GITHUB_EMAIL )
2023-03-13 17:40:35 +01:00
2023-03-27 18:41:38 +02:00
def apps_to_run_auto_update_for ( ) :
catalog = toml . load ( open ( os . path . dirname ( __file__ ) + " /../../apps.toml " ) )
2023-03-28 00:42:18 +02:00
apps_flagged_as_working_and_on_yunohost_apps_org = [
app
for app , infos in catalog . items ( )
if infos [ " state " ] == " working "
and " /github.com/yunohost-apps " in infos [ " url " ] . lower ( )
]
2023-03-27 18:41:38 +02:00
2023-03-28 00:42:18 +02:00
manifest_tomls = glob . glob (
os . path . dirname ( __file__ ) + " /../../.apps_cache/*/manifest.toml "
)
2023-03-27 18:41:38 +02:00
apps_with_manifest_toml = [ path . split ( " / " ) [ - 2 ] for path in manifest_tomls ]
2023-03-28 00:42:18 +02:00
relevant_apps = list (
sorted (
set ( apps_flagged_as_working_and_on_yunohost_apps_org )
& set ( apps_with_manifest_toml )
)
)
2023-03-27 18:41:38 +02:00
out = [ ]
for app in relevant_apps :
2023-03-28 00:42:18 +02:00
manifest = toml . load (
os . path . dirname ( __file__ ) + f " /../../.apps_cache/ { app } /manifest.toml "
)
2023-03-27 18:41:38 +02:00
sources = manifest . get ( " resources " , { } ) . get ( " sources " , { } )
if any ( " autoupdate " in source for source in sources . values ( ) ) :
out . append ( app )
return out
2023-03-28 19:13:50 +02:00
def filter_and_get_latest_tag ( tags , app_id ) :
2023-03-13 17:40:35 +01:00
filter_keywords = [ " start " , " rc " , " beta " , " alpha " ]
tags = [ t for t in tags if not any ( keyword in t for keyword in filter_keywords ) ]
2023-03-28 19:13:50 +02:00
tag_dict = { }
2023-03-13 17:40:35 +01:00
for t in tags :
2023-03-28 19:13:50 +02:00
t_to_check = t
if t . startswith ( app_id + " - " ) :
t_to_check = t . split ( " - " , 1 ) [ - 1 ]
2023-06-19 21:19:06 +02:00
# Boring special case for dokuwiki...
elif t . startswith ( " release- " ) :
t_to_check = t . split ( " - " , 1 ) [ - 1 ] . replace ( " - " , " . " )
2023-03-13 17:40:35 +01:00
2023-03-28 19:13:50 +02:00
if not re . match ( r " ^v?[ \ d \ .]* \ d$ " , t_to_check ) :
print ( f " Ignoring tag { t_to_check } , doesn ' t look like a version number " )
else :
tag_dict [ t ] = tag_to_int_tuple ( t_to_check )
tags = sorted ( list ( tag_dict . keys ( ) ) , key = tag_dict . get )
2023-04-04 18:15:35 +02:00
2023-03-28 19:28:11 +02:00
return tags [ - 1 ] , " . " . join ( [ str ( i ) for i in tag_dict [ tags [ - 1 ] ] ] )
2023-03-13 17:40:35 +01:00
def tag_to_int_tuple ( tag ) :
2023-06-19 21:05:23 +02:00
tag = tag . strip ( " v " ) . strip ( " . " )
2023-03-13 17:40:35 +01:00
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 )
2023-03-27 17:49:48 +02:00
def sha256_of_remote_file ( url ) :
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 :
print ( f " Failed to compute sha256 for { url } : { e } " )
return None
2023-03-28 00:42:18 +02:00
class AppAutoUpdater :
2023-04-04 16:25:12 +02:00
def __init__ ( self , app_id , app_id_is_local_app_dir = False ) :
2023-03-27 17:49:48 +02:00
2023-04-04 16:25:12 +02:00
if app_id_is_local_app_dir :
2023-03-28 19:13:50 +02:00
if not os . path . exists ( app_id + " /manifest.toml " ) :
2023-03-28 19:28:11 +02:00
raise Exception ( " manifest.toml doesnt exists? " )
2023-03-28 19:13:50 +02:00
# app_id is in fact a path
manifest = toml . load ( open ( app_id + " /manifest.toml " ) )
2023-03-13 17:40:35 +01:00
2023-03-28 19:13:50 +02:00
else :
# We actually want to look at the manifest on the "testing" (or default) branch
self . repo = github . get_repo ( f " Yunohost-Apps/ { app_id } _ynh " )
# Determine base branch, either `testing` or default branch
try :
self . base_branch = self . repo . get_branch ( " testing " ) . name
except :
self . base_branch = self . repo . default_branch
contents = self . repo . get_contents ( " manifest.toml " , ref = self . base_branch )
self . manifest_raw = contents . decoded_content . decode ( )
self . manifest_raw_sha = contents . sha
manifest = toml . loads ( self . manifest_raw )
self . app_id = manifest [ " id " ]
2023-03-13 17:40:35 +01:00
self . current_version = manifest [ " version " ] . split ( " ~ " ) [ 0 ]
self . sources = manifest . get ( " resources " , { } ) . get ( " sources " )
if not self . sources :
raise Exception ( " There ' s no resources.sources in manifest.toml ? " )
2023-03-28 19:13:50 +02:00
self . main_upstream = manifest . get ( " upstream " , { } ) . get ( " code " )
2023-03-13 17:40:35 +01:00
def run ( self ) :
2023-03-27 17:49:48 +02:00
todos = { }
2023-03-13 17:40:35 +01:00
for source , infos in self . sources . items ( ) :
if " autoupdate " not in infos :
continue
strategy = infos . get ( " autoupdate " , { } ) . get ( " strategy " )
if strategy not in STRATEGIES :
2023-03-28 00:42:18 +02:00
raise Exception (
f " Unknown strategy to autoupdate { source } , expected one of { STRATEGIES } , got { strategy } "
)
2023-03-13 17:40:35 +01:00
asset = infos . get ( " autoupdate " , { } ) . get ( " asset " , " tarball " )
2023-04-04 18:32:12 +02:00
print ( f " \n Checking { source } ... " )
2023-03-13 17:40:35 +01:00
2023-03-28 00:42:18 +02:00
new_version , new_asset_urls = self . get_latest_version_and_asset (
2023-03-28 19:13:50 +02:00
strategy , asset , infos , source
2023-03-28 00:42:18 +02:00
)
2023-03-13 17:40:35 +01:00
2023-03-21 00:24:52 +01:00
if source == " main " :
2023-03-28 19:13:50 +02:00
print ( f " Current version in manifest: { self . current_version } " )
2023-04-04 18:32:12 +02:00
print ( f " Newest version on upstream: { new_version } " )
2023-07-24 21:09:13 +02:00
# Maybe new version is older than current version
# Which can happen for example if we manually release a RC,
# which is ignored by this script
# Though we wrap this in a try/except pass, because don't want to miserably crash
# if the tag can't properly be converted to int tuple ...
try :
if tag_to_int_tuple ( self . current_version ) > tag_to_int_tuple ( new_version ) :
print ( " Up to date (current version appears more recent than newest version found) " )
continue
except :
pass
2023-04-04 18:55:05 +02:00
if self . current_version == new_version :
print ( " Up to date " )
continue
2023-07-24 20:59:46 +02:00
if ( isinstance ( new_asset_urls , dict ) and isinstance ( infos . get ( " url " ) , str ) ) \
or ( isinstance ( new_asset_urls , str ) and not isinstance ( infos . get ( " url " ) , str ) ) :
raise Exception ( f " It looks like there ' s an inconsistency between the old asset list and the new ones ... one is arch-specific, the other is not ... Did you forget to define arch-specific regexes ? ... New asset url is/are : { new_asset_urls } " )
2023-04-04 18:32:12 +02:00
if isinstance ( new_asset_urls , str ) and infos [ " url " ] == new_asset_urls :
print ( f " URL for asset { source } is up to date " )
continue
elif isinstance ( new_asset_urls , dict ) and new_asset_urls == {
k : infos [ k ] [ " url " ] for k in new_asset_urls . keys ( )
} :
print ( f " URLs for asset { source } are up to date " )
continue
2023-03-21 00:24:52 +01:00
else :
2023-04-04 18:32:12 +02:00
print ( f " Update needed for { source } " )
todos [ source ] = {
" new_asset_urls " : new_asset_urls ,
" old_assets " : infos ,
}
if source == " main " :
todos [ source ] [ " new_version " ] = new_version
2023-03-21 00:24:52 +01:00
2023-03-28 19:13:50 +02:00
if dry_run or not todos :
2023-04-04 18:15:35 +02:00
return bool ( todos )
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
if " main " in todos :
new_version = todos [ " main " ] [ " new_version " ]
message = f " Upgrade to v { new_version } "
new_branch = f " ci-auto-update- { new_version } "
else :
message = " Upgrade sources "
new_branch = " ci-auto-update-sources "
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
try :
# Get the commit base for the new branch, and create it
commit_sha = self . repo . get_branch ( self . base_branch ) . commit . sha
self . repo . create_git_ref ( ref = f " refs/heads/ { new_branch } " , sha = commit_sha )
except :
2023-04-05 17:07:36 +02:00
print ( " ... Branch already exists, skipping " )
return False
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
manifest_new = self . manifest_raw
for source , infos in todos . items ( ) :
2023-03-28 00:42:18 +02:00
manifest_new = self . replace_version_and_asset_in_manifest (
manifest_new ,
infos . get ( " new_version " ) ,
infos [ " new_asset_urls " ] ,
infos [ " old_assets " ] ,
is_main = source == " main " ,
)
self . repo . update_file (
" manifest.toml " ,
message = message ,
content = manifest_new ,
sha = self . manifest_raw_sha ,
branch = new_branch ,
author = author ,
)
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
# Wait a bit to preserve the API rate limit
time . sleep ( 1.5 )
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
# Open the PR
2023-03-28 00:42:18 +02:00
pr = self . repo . create_pull (
title = message , body = message , head = new_branch , base = self . base_branch
)
2023-03-21 00:24:52 +01:00
2023-03-27 17:49:48 +02:00
print ( " Created PR " + self . repo . full_name + " updated with PR # " + str ( pr . id ) )
2023-03-21 00:24:52 +01:00
2023-04-04 18:15:35 +02:00
return bool ( todos )
2023-03-28 19:13:50 +02:00
def get_latest_version_and_asset ( self , strategy , asset , infos , source ) :
2023-04-04 18:44:43 +02:00
upstream = infos . get ( " autoupdate " , { } ) . get ( " upstream " , self . main_upstream ) . strip ( " / " )
2023-03-13 17:40:35 +01:00
if " github " in strategy :
2023-03-28 19:13:50 +02:00
assert upstream and upstream . startswith (
2023-03-28 00:42:18 +02:00
" https://github.com/ "
2023-03-28 19:13:50 +02:00
) , f " When using strategy { strategy } , having a defined upstream code repo on github.com is required "
2023-03-28 19:28:11 +02:00
upstream_repo = upstream . replace ( " https://github.com/ " , " " ) . strip ( " / " )
2023-03-28 00:42:18 +02:00
assert (
2023-03-28 19:13:50 +02:00
len ( upstream_repo . split ( " / " ) ) == 2
) , f " ' { upstream } ' doesn ' t seem to be a github repository ? "
2023-03-13 17:40:35 +01:00
if strategy == " latest_github_release " :
2023-03-28 19:13:50 +02:00
releases = self . github_api ( f " repos/ { upstream_repo } /releases " )
2023-03-28 00:42:18 +02:00
tags = [
release [ " tag_name " ]
for release in releases
if not release [ " draft " ] and not release [ " prerelease " ]
]
2023-03-28 19:28:11 +02:00
latest_version_orig , latest_version = filter_and_get_latest_tag (
tags , self . app_id
)
2023-03-13 17:40:35 +01:00
if asset == " tarball " :
2023-03-28 00:42:18 +02:00
latest_tarball = (
2023-03-28 19:13:50 +02:00
f " { upstream } /archive/refs/tags/ { latest_version_orig } .tar.gz "
2023-03-28 00:42:18 +02:00
)
2023-03-28 19:13:50 +02:00
return latest_version , latest_tarball
2023-03-13 17:40:35 +01:00
# FIXME
else :
2023-03-28 00:42:18 +02:00
latest_release = [
release
for release in releases
2023-03-28 19:13:50 +02:00
if release [ " tag_name " ] == latest_version_orig
2023-03-28 00:42:18 +02:00
] [ 0 ]
latest_assets = {
a [ " name " ] : a [ " browser_download_url " ]
for a in latest_release [ " assets " ]
if not a [ " name " ] . endswith ( " .md5 " )
}
2023-07-24 20:43:30 +02:00
latest_release_html_url = latest_release [ " html_url " ]
2023-03-13 17:40:35 +01:00
if isinstance ( asset , str ) :
2023-03-28 00:42:18 +02:00
matching_assets_urls = [
url
for name , url in latest_assets . items ( )
if re . match ( asset , name )
]
2023-03-13 17:40:35 +01:00
if not matching_assets_urls :
2023-03-28 00:42:18 +02:00
raise Exception (
2023-07-24 20:43:30 +02:00
f " No assets matching regex ' { asset } ' for release { latest_version } among { list ( latest_assets . keys ( ) ) } . Full release details on { latest_release_html_url } "
2023-03-28 00:42:18 +02:00
)
2023-03-13 17:40:35 +01:00
elif len ( matching_assets_urls ) > 1 :
2023-03-28 00:42:18 +02:00
raise Exception (
2023-07-24 20:43:30 +02:00
f " Too many assets matching regex ' { asset } ' for release { latest_version } : { matching_assets_urls } . Full release details on { latest_release_html_url } "
2023-03-28 00:42:18 +02:00
)
2023-03-28 19:13:50 +02:00
return latest_version , matching_assets_urls [ 0 ]
2023-03-13 17:40:35 +01:00
elif isinstance ( asset , dict ) :
matching_assets_dicts = { }
for asset_name , asset_regex in asset . items ( ) :
2023-03-28 00:42:18 +02:00
matching_assets_urls = [
url
for name , url in latest_assets . items ( )
if re . match ( asset_regex , name )
]
2023-03-13 17:40:35 +01:00
if not matching_assets_urls :
2023-03-28 00:42:18 +02:00
raise Exception (
2023-07-24 20:43:30 +02:00
f " No assets matching regex ' { asset_regex } ' for release { latest_version } among { list ( latest_assets . keys ( ) ) } . Full release details on { latest_release_html_url } "
2023-03-28 00:42:18 +02:00
)
2023-03-13 17:40:35 +01:00
elif len ( matching_assets_urls ) > 1 :
2023-03-28 00:42:18 +02:00
raise Exception (
2023-07-24 20:43:30 +02:00
f " Too many assets matching regex ' { asset } ' for release { latest_version } : { matching_assets_urls } . Full release details on { latest_release_html_url } "
2023-03-28 00:42:18 +02:00
)
2023-03-13 17:40:35 +01:00
matching_assets_dicts [ asset_name ] = matching_assets_urls [ 0 ]
return latest_version . strip ( " v " ) , matching_assets_dicts
elif strategy == " latest_github_tag " :
if asset != " tarball " :
2023-03-28 00:42:18 +02:00
raise Exception (
" For the latest_github_tag strategy, only asset = ' tarball ' is supported "
)
2023-03-28 19:13:50 +02:00
tags = self . github_api ( f " repos/ { upstream_repo } /tags " )
2023-03-28 19:28:11 +02:00
latest_version_orig , latest_version = filter_and_get_latest_tag (
[ t [ " name " ] for t in tags ] , self . app_id
2023-03-28 00:42:18 +02:00
)
2023-04-04 19:15:04 +02:00
latest_tarball = f " { upstream } /archive/refs/tags/ { latest_version_orig } .tar.gz "
2023-03-28 19:13:50 +02:00
return latest_version , latest_tarball
elif strategy == " latest_github_commit " :
if asset != " tarball " :
raise Exception (
" For the latest_github_release strategy, only asset = ' tarball ' is supported "
)
commits = self . github_api ( f " repos/ { upstream_repo } /commits " )
latest_commit = commits [ 0 ]
latest_tarball = f " https://github.com/ { upstream_repo } /archive/ { latest_commit [ ' sha ' ] } .tar.gz "
# Let's have the version as something like "2023.01.23"
2023-04-04 18:41:30 +02:00
latest_commit_date = datetime . strptime ( latest_commit [ " commit " ] [ " author " ] [ " date " ] [ : 10 ] , " % Y- % m- %d " )
version_format = infos . get ( " autoupdate " , { } ) . get ( " force_version " , " % Y. % m. %d " )
2023-04-04 18:32:12 +02:00
latest_version = latest_commit_date . strftime ( version_format )
2023-03-28 19:13:50 +02:00
return latest_version , latest_tarball
def github_api ( self , uri ) :
2023-04-04 16:25:12 +02:00
r = requests . get ( f " https://api.github.com/ { uri } " , auth = auth )
2023-03-13 17:40:35 +01:00
assert r . status_code == 200 , r
return r . json ( )
2023-03-28 00:42:18 +02:00
def replace_version_and_asset_in_manifest (
self , content , new_version , new_assets_urls , current_assets , is_main
) :
2023-03-27 17:49:48 +02:00
if isinstance ( new_assets_urls , str ) :
sha256 = sha256_of_remote_file ( new_assets_urls )
elif isinstance ( new_assets_urls , dict ) :
2023-03-28 00:42:18 +02:00
sha256 = {
url : sha256_of_remote_file ( url ) for url in new_assets_urls . values ( )
}
2023-03-27 17:49:48 +02:00
if is_main :
2023-03-28 00:42:18 +02:00
2023-03-27 17:49:48 +02:00
def repl ( m ) :
return m . group ( 1 ) + new_version + m . group ( 3 )
2023-03-28 00:42:18 +02:00
content = re . sub (
r " ( \ s*version \ s*= \ s*[ \" \ ' ])([ \ d \ .]+)( \ ~ynh \ d+[ \" \ ' ]) " , repl , content
)
2023-03-27 17:49:48 +02:00
if isinstance ( new_assets_urls , str ) :
content = content . replace ( current_assets [ " url " ] , new_assets_urls )
content = content . replace ( current_assets [ " sha256 " ] , sha256 )
elif isinstance ( new_assets_urls , dict ) :
for key , url in new_assets_urls . items ( ) :
content = content . replace ( current_assets [ key ] [ " url " ] , url )
content = content . replace ( current_assets [ key ] [ " sha256 " ] , sha256 [ url ] )
return content
2023-03-27 18:41:38 +02:00
# Progress bar helper, stolen from https://stackoverflow.com/a/34482761
def progressbar ( it , prefix = " " , size = 60 , file = sys . stdout ) :
it = list ( it )
count = len ( it )
2023-03-28 00:42:18 +02:00
2023-03-27 18:41:38 +02:00
def show ( j , name = " " ) :
name + = " "
2023-03-28 00:42:18 +02:00
x = int ( size * j / count )
file . write (
2023-04-04 18:15:35 +02:00
" \n %s [ %s %s ] %i / %i %s \n " % ( prefix , " # " * x , " . " * ( size - x ) , j , count , name )
2023-03-28 00:42:18 +02:00
)
2023-03-27 18:41:38 +02:00
file . flush ( )
2023-03-28 00:42:18 +02:00
2023-03-27 18:41:38 +02:00
show ( 0 )
for i , item in enumerate ( it ) :
2023-03-28 00:42:18 +02:00
show ( i + 1 , item )
2023-04-04 18:15:35 +02:00
yield item
2023-03-27 18:41:38 +02:00
file . write ( " \n " )
file . flush ( )
2023-03-21 00:24:52 +01:00
2023-03-13 17:40:35 +01:00
2023-07-24 20:43:30 +02:00
def paste_on_haste ( data ) :
# 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...
# and doesn't have the proper front-end LE cert in this context
SERVER_URL = " http://paste.yunohost.org "
TIMEOUT = 3
try :
url = SERVER_URL + " /documents "
response = requests . post ( url , data = data . encode ( ' utf-8 ' ) , timeout = TIMEOUT )
response . raise_for_status ( )
dockey = response . json ( ) [ ' key ' ]
return SERVER_URL + " /raw/ " + dockey
except requests . exceptions . RequestException as e :
print ( " \033 [31mError: {} \033 [0m " . format ( e ) )
sys . exit ( 1 )
2023-03-13 17:40:35 +01:00
if __name__ == " __main__ " :
2023-04-04 16:25:12 +02:00
args = [ arg for arg in sys . argv [ 1 : ] if arg != " --commit-and-create-PR " ]
if len ( args ) :
AppAutoUpdater ( args [ 0 ] , app_id_is_local_app_dir = True ) . run ( )
2023-03-28 19:13:50 +02:00
else :
2023-04-04 18:15:35 +02:00
apps_failed = [ ]
2023-07-24 20:43:30 +02:00
apps_failed_details = { }
2023-04-04 18:15:35 +02:00
apps_updated = [ ]
2023-03-28 19:13:50 +02:00
for app in progressbar ( apps_to_run_auto_update_for ( ) , " Checking: " , 40 ) :
2023-04-04 18:15:35 +02:00
try :
updated = AppAutoUpdater ( app ) . run ( )
except Exception as e :
apps_failed . append ( app )
import traceback
2023-07-24 20:43:30 +02:00
t = traceback . format_exc ( )
apps_failed_details [ app ] = t
print ( t )
2023-04-04 18:15:35 +02:00
else :
if updated :
apps_updated . append ( app )
if apps_failed :
print ( f " Apps failed: { ' , ' . join ( apps_failed ) } " )
if os . path . exists ( " /usr/bin/sendxmpppy " ) :
2023-07-24 20:43:30 +02:00
paste = ' \n ========= \n ' . join ( [ app + " \n ------- \n " + trace + " \n \n " for app , trace in apps_failed_details . items ( ) ] )
paste_url = paste_on_haste ( paste )
os . system ( f " /usr/bin/sendxmpppy ' Failed to run the source auto-update for : { ' , ' . join ( apps_failed ) } . Please run manually the `autoupdate_app_sources.py` script on these apps to debug what is happening! Debug log : { paste_url } ' " )
2023-04-04 18:15:35 +02:00
if apps_updated :
print ( f " Apps updated: { ' , ' . join ( apps_updated ) } " )