mirror of
https://gitlab.com/fdroid/fdroidserver.git
synced 2024-11-04 14:30:11 +01:00
update: make icon extraction less dependent on aapt
For androguard, @thezero already developed a way to get all the icons after only extracting the icon name. So this uses that for the aapt-based scans also, to make them less brittle. This should fix the problem where `fdroid update` was choosing the XML icon for apps that include one, like NewPipe. closes fdroid/fdroid-website#192
This commit is contained in:
parent
01a73071c7
commit
b2ca49b26c
@ -53,9 +53,7 @@ UNSET_VERSION_CODE = -0x100000000
|
|||||||
APK_NAME_PAT = re.compile(".*name='([a-zA-Z0-9._]*)'.*")
|
APK_NAME_PAT = re.compile(".*name='([a-zA-Z0-9._]*)'.*")
|
||||||
APK_VERCODE_PAT = re.compile(".*versionCode='([0-9]*)'.*")
|
APK_VERCODE_PAT = re.compile(".*versionCode='([0-9]*)'.*")
|
||||||
APK_VERNAME_PAT = re.compile(".*versionName='([^']*)'.*")
|
APK_VERNAME_PAT = re.compile(".*versionName='([^']*)'.*")
|
||||||
APK_LABEL_PAT = re.compile(".*label='(.*?)'(\n| [a-z]*?=).*")
|
APK_LABEL_ICON_PAT = re.compile(".*\s+label='(.*)'\s+icon='(.*)'")
|
||||||
APK_ICON_PAT = re.compile(".*application-icon-([0-9]+):'([^']+?)'.*")
|
|
||||||
APK_ICON_PAT_NODPI = re.compile(".*icon='([^']+?)'.*")
|
|
||||||
APK_SDK_VERSION_PAT = re.compile(".*'([0-9]*)'.*")
|
APK_SDK_VERSION_PAT = re.compile(".*'([0-9]*)'.*")
|
||||||
APK_PERMISSION_PAT = \
|
APK_PERMISSION_PAT = \
|
||||||
re.compile(".*(name='(?P<name>.*?)')(.*maxSdkVersion='(?P<maxSdkVersion>.*?)')?.*")
|
re.compile(".*(name='(?P<name>.*?)')(.*maxSdkVersion='(?P<maxSdkVersion>.*?)')?.*")
|
||||||
@ -1080,6 +1078,27 @@ def scan_apk(apk_file):
|
|||||||
return apk
|
return apk
|
||||||
|
|
||||||
|
|
||||||
|
def _get_apk_icons_src(apkfile, icon_name):
|
||||||
|
"""Extract the paths to the app icon in all available densities
|
||||||
|
|
||||||
|
"""
|
||||||
|
icons_src = dict()
|
||||||
|
density_re = re.compile('^res/(.*)/' + icon_name + '\.(png|xml)$')
|
||||||
|
with zipfile.ZipFile(apkfile) as zf:
|
||||||
|
for filename in zf.namelist():
|
||||||
|
m = density_re.match(filename)
|
||||||
|
if m:
|
||||||
|
folder = m.group(1).split('-')
|
||||||
|
if len(folder) > 1:
|
||||||
|
density = screen_resolutions[folder[1]]
|
||||||
|
else:
|
||||||
|
density = '160'
|
||||||
|
icons_src[density] = m.group(0)
|
||||||
|
if icons_src.get('-1') is None:
|
||||||
|
icons_src['-1'] = icons_src['160']
|
||||||
|
return icons_src
|
||||||
|
|
||||||
|
|
||||||
def scan_apk_aapt(apk, apkfile):
|
def scan_apk_aapt(apk, apkfile):
|
||||||
p = SdkToolsPopen(['aapt', 'dump', 'badging', apkfile], output=False)
|
p = SdkToolsPopen(['aapt', 'dump', 'badging', apkfile], output=False)
|
||||||
if p.returncode != 0:
|
if p.returncode != 0:
|
||||||
@ -1092,6 +1111,7 @@ def scan_apk_aapt(apk, apkfile):
|
|||||||
else:
|
else:
|
||||||
logging.error(_("Failed to get apk information, skipping {path}").format(path=apkfile))
|
logging.error(_("Failed to get apk information, skipping {path}").format(path=apkfile))
|
||||||
raise BuildException(_("Invalid APK"))
|
raise BuildException(_("Invalid APK"))
|
||||||
|
icon_name = None
|
||||||
for line in p.output.splitlines():
|
for line in p.output.splitlines():
|
||||||
if line.startswith("package:"):
|
if line.startswith("package:"):
|
||||||
try:
|
try:
|
||||||
@ -1101,25 +1121,13 @@ def scan_apk_aapt(apk, apkfile):
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
raise FDroidException("Package matching failed: " + str(e) + "\nLine was: " + line)
|
raise FDroidException("Package matching failed: " + str(e) + "\nLine was: " + line)
|
||||||
elif line.startswith("application:"):
|
elif line.startswith("application:"):
|
||||||
apk['name'] = re.match(APK_LABEL_PAT, line).group(1)
|
m = re.match(APK_LABEL_ICON_PAT, line)
|
||||||
# Keep path to non-dpi icon in case we need it
|
if m:
|
||||||
match = re.match(APK_ICON_PAT_NODPI, line)
|
apk['name'] = m.group(1)
|
||||||
if match:
|
icon_name = os.path.splitext(os.path.basename(m.group(2)))[0]
|
||||||
apk['icons_src']['-1'] = match.group(1)
|
elif not apk.get('name') and line.startswith("launchable-activity:"):
|
||||||
elif line.startswith("launchable-activity:"):
|
|
||||||
# Only use launchable-activity as fallback to application
|
# Only use launchable-activity as fallback to application
|
||||||
if not apk['name']:
|
apk['name'] = re.match(APK_LABEL_ICON_PAT, line).group(1)
|
||||||
apk['name'] = re.match(APK_LABEL_PAT, line).group(1)
|
|
||||||
if '-1' not in apk['icons_src']:
|
|
||||||
match = re.match(APK_ICON_PAT_NODPI, line)
|
|
||||||
if match:
|
|
||||||
apk['icons_src']['-1'] = match.group(1)
|
|
||||||
elif line.startswith("application-icon-"):
|
|
||||||
match = re.match(APK_ICON_PAT, line)
|
|
||||||
if match:
|
|
||||||
density = match.group(1)
|
|
||||||
path = match.group(2)
|
|
||||||
apk['icons_src'][density] = path
|
|
||||||
elif line.startswith("sdkVersion:"):
|
elif line.startswith("sdkVersion:"):
|
||||||
m = re.match(APK_SDK_VERSION_PAT, line)
|
m = re.match(APK_SDK_VERSION_PAT, line)
|
||||||
if m is None:
|
if m is None:
|
||||||
@ -1170,6 +1178,7 @@ def scan_apk_aapt(apk, apkfile):
|
|||||||
if feature.startswith("android.feature."):
|
if feature.startswith("android.feature."):
|
||||||
feature = feature[16:]
|
feature = feature[16:]
|
||||||
apk['features'].add(feature)
|
apk['features'].add(feature)
|
||||||
|
apk['icons_src'] = _get_apk_icons_src(apkfile, icon_name)
|
||||||
|
|
||||||
|
|
||||||
def scan_apk_androguard(apk, apkfile):
|
def scan_apk_androguard(apk, apkfile):
|
||||||
@ -1215,22 +1224,7 @@ def scan_apk_androguard(apk, apkfile):
|
|||||||
|
|
||||||
icon_id = int(apkobject.get_element("application", "icon").replace("@", "0x"), 16)
|
icon_id = int(apkobject.get_element("application", "icon").replace("@", "0x"), 16)
|
||||||
icon_name = arsc.get_id(apk['packageName'], icon_id)[1]
|
icon_name = arsc.get_id(apk['packageName'], icon_id)[1]
|
||||||
|
apk['icons_src'] = _get_apk_icons_src(apkfile, icon_name)
|
||||||
density_re = re.compile("^res/(.*)/" + icon_name + ".*$")
|
|
||||||
|
|
||||||
for file in apkobject.get_files():
|
|
||||||
d_re = density_re.match(file)
|
|
||||||
if d_re:
|
|
||||||
folder = d_re.group(1).split('-')
|
|
||||||
if len(folder) > 1:
|
|
||||||
resolution = folder[1]
|
|
||||||
else:
|
|
||||||
resolution = 'mdpi'
|
|
||||||
density = screen_resolutions[resolution]
|
|
||||||
apk['icons_src'][density] = d_re.group(0)
|
|
||||||
|
|
||||||
if apk['icons_src'].get('-1') is None:
|
|
||||||
apk['icons_src']['-1'] = apk['icons_src']['160']
|
|
||||||
|
|
||||||
arch_re = re.compile("^lib/(.*)/.*$")
|
arch_re = re.compile("^lib/(.*)/.*$")
|
||||||
arch = set([arch_re.match(file).group(1) for file in apkobject.get_files() if arch_re.match(file)])
|
arch = set([arch_re.match(file).group(1) for file in apkobject.get_files() if arch_re.match(file)])
|
||||||
|
Loading…
Reference in New Issue
Block a user