Tentative fix to include package dependencies.
This commit is contained in:
parent
3428c9fb53
commit
50e019262a
|
@ -91,11 +91,22 @@ class Build():
|
||||||
"""Calculate exclusions and other variables."""
|
"""Calculate exclusions and other variables."""
|
||||||
|
|
||||||
if self.verbose:
|
if self.verbose:
|
||||||
print("--- Preliminary system checks ---")
|
print("--- Preliminary Phase ---")
|
||||||
|
|
||||||
if isinstance(shutil.which('apt'), str):
|
if isinstance(shutil.which('apt'), str):
|
||||||
# APT is found in path. We assume we can find dependencies.
|
# APT is found in path. We assume we can find dependencies.
|
||||||
self.check_dependencies = True
|
self.check_dependencies = True
|
||||||
|
|
||||||
|
|
||||||
|
if self.verbose:
|
||||||
|
print("Updating system packages cache.")
|
||||||
|
# Updating package cache
|
||||||
|
subprocess.run(['sudo', 'apt', 'update'], check=True, stdout=subprocess.DEVNULL)
|
||||||
|
if self.verbose:
|
||||||
|
print("Ensuring apt-file is installed and updated.")
|
||||||
|
# Updating apt-file cache
|
||||||
|
subprocess.run(['sudo', 'apt', 'install', 'apt-file', '-y'], check=True, stdout=subprocess.DEVNULL)
|
||||||
|
subprocess.run(['sudo', 'apt-file', 'update'], check=True, stdout=subprocess.DEVNULL)
|
||||||
else:
|
else:
|
||||||
print("CAUTION: your system seems not to include a working version of apt.\nThis will cause the AppImage to leverage system libraries when run.")
|
print("CAUTION: your system seems not to include a working version of apt.\nThis will cause the AppImage to leverage system libraries when run.")
|
||||||
self.check_dependencies = False
|
self.check_dependencies = False
|
||||||
|
@ -237,12 +248,12 @@ class Build():
|
||||||
def build(self):
|
def build(self):
|
||||||
"""Building all the versions."""
|
"""Building all the versions."""
|
||||||
|
|
||||||
if self.verbose:
|
|
||||||
print("--- Building Phase ---")
|
|
||||||
|
|
||||||
if self.found:
|
if self.found:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
if self.verbose:
|
||||||
|
print("--- Building Phase ---")
|
||||||
|
|
||||||
# Preparation tasks
|
# Preparation tasks
|
||||||
self.appnamedir = os.path.join(self.builddir, self.appname)
|
self.appnamedir = os.path.join(self.builddir, self.appname)
|
||||||
os.makedirs(self.appnamedir, exist_ok=True)
|
os.makedirs(self.appnamedir, exist_ok=True)
|
||||||
|
@ -256,6 +267,13 @@ class Build():
|
||||||
|
|
||||||
# Build the requested version.
|
# Build the requested version.
|
||||||
self.__unpackbuild__()
|
self.__unpackbuild__()
|
||||||
|
self.__prepare_contents__()
|
||||||
|
if self.check_dependencies:
|
||||||
|
if self.verbose:
|
||||||
|
print("Searching for dependent libraries, it might take a while.")
|
||||||
|
|
||||||
|
self.__missing_dependencies__()
|
||||||
|
self.__finalize_build__()
|
||||||
|
|
||||||
|
|
||||||
def checksums(self):
|
def checksums(self):
|
||||||
|
@ -505,6 +523,7 @@ class Build():
|
||||||
subprocess.run(shlex.split(
|
subprocess.run(shlex.split(
|
||||||
f"tar xzf {self.download_path}/{archive}"), check=True)
|
f"tar xzf {self.download_path}/{archive}"), check=True)
|
||||||
|
|
||||||
|
def __prepare_contents__(self):
|
||||||
# create appimagedir
|
# create appimagedir
|
||||||
if self.verbose:
|
if self.verbose:
|
||||||
print("---- Preparing the build ----")
|
print("---- Preparing the build ----")
|
||||||
|
@ -512,6 +531,9 @@ class Build():
|
||||||
os.makedirs(self.appimagedir, exist_ok = True)
|
os.makedirs(self.appimagedir, exist_ok = True)
|
||||||
|
|
||||||
# At this point, let's decompress the deb packages
|
# At this point, let's decompress the deb packages
|
||||||
|
if self.verbose:
|
||||||
|
print("Unpacking main archives")
|
||||||
|
|
||||||
subprocess.run(shlex.split(
|
subprocess.run(shlex.split(
|
||||||
r"find .. -iname '*.deb' -exec dpkg -x {} . \;"
|
r"find .. -iname '*.deb' -exec dpkg -x {} . \;"
|
||||||
), cwd=self.appimagedir, check=True)
|
), cwd=self.appimagedir, check=True)
|
||||||
|
@ -524,6 +546,9 @@ class Build():
|
||||||
), cwd=self.appimagedir, check=True)
|
), cwd=self.appimagedir, check=True)
|
||||||
|
|
||||||
# Changing desktop file
|
# Changing desktop file
|
||||||
|
if self.verbose:
|
||||||
|
print("Preparing .desktop file.")
|
||||||
|
|
||||||
subprocess.run(shlex.split(
|
subprocess.run(shlex.split(
|
||||||
r"find . -iname startcenter.desktop -exec cp {} . \;"
|
r"find . -iname startcenter.desktop -exec cp {} . \;"
|
||||||
), cwd=self.appimagedir, check=True)
|
), cwd=self.appimagedir, check=True)
|
||||||
|
@ -533,6 +558,8 @@ class Build():
|
||||||
r"startcenter.desktop"
|
r"startcenter.desktop"
|
||||||
), cwd=self.appimagedir, check=False)
|
), cwd=self.appimagedir, check=False)
|
||||||
|
|
||||||
|
if self.verbose:
|
||||||
|
print("Preparing icon file.")
|
||||||
subprocess.run(shlex.split(
|
subprocess.run(shlex.split(
|
||||||
r"find . -name '*startcenter.png' -path '*hicolor*48x48*' " +
|
r"find . -name '*startcenter.png' -path '*hicolor*48x48*' " +
|
||||||
r"-exec cp {} . \;"
|
r"-exec cp {} . \;"
|
||||||
|
@ -542,62 +569,10 @@ class Build():
|
||||||
cmd = subprocess.run(shlex.split(
|
cmd = subprocess.run(shlex.split(
|
||||||
r"find -iname soffice.bin -print"
|
r"find -iname soffice.bin -print"
|
||||||
), cwd=self.appimagedir, check = True, capture_output=True)
|
), cwd=self.appimagedir, check = True, capture_output=True)
|
||||||
main_executable = os.path.abspath(os.path.join(
|
self.main_executable = os.path.abspath(os.path.join(
|
||||||
self.appimagedir,
|
self.appimagedir,
|
||||||
cmd.stdout.strip().decode('utf-8')))
|
cmd.stdout.strip().decode('utf-8')))
|
||||||
|
|
||||||
# If the system permits it, we leverage lddcollect
|
|
||||||
# to find the packages that contain .so dependencies in the main build.
|
|
||||||
if self.check_dependencies:
|
|
||||||
if self.verbose:
|
|
||||||
print("Checking for dependent libraries")
|
|
||||||
|
|
||||||
import lddcollect
|
|
||||||
# We first process the ELF
|
|
||||||
raw = lddcollect.process_elf(main_executable, verbose = False, dpkg = True)
|
|
||||||
|
|
||||||
# If all works as expected, we obtain a tuple of:
|
|
||||||
# (debian_packages, all_libraries, files_not_found)
|
|
||||||
(debian_packages, all_libraries, not_found) = raw
|
|
||||||
|
|
||||||
if len(debian_packages) != 0:
|
|
||||||
# We need, first, to download those packages.
|
|
||||||
debs = [ x.split(':')[0] for x in debian_packages ]
|
|
||||||
downloadpath = os.path.abspath(os.path.join(self.builddir, 'dependencies'))
|
|
||||||
os.makedirs(downloadpath)
|
|
||||||
|
|
||||||
if self.verbose:
|
|
||||||
print("Downloading missing dependencies, please wait.")
|
|
||||||
|
|
||||||
# Updating package cache
|
|
||||||
subprocess.run(['sudo', 'apt', 'update'], check=True)
|
|
||||||
# Updating apt-file cache
|
|
||||||
subprocess.run(['sudo', 'apt-file', 'update'], check=True)
|
|
||||||
|
|
||||||
# Let's try to find and install also other libraries
|
|
||||||
additional = list(dict.fromkeys([ libfinderhelper(x) for x in not_found ]))
|
|
||||||
debs.extend(additional)
|
|
||||||
|
|
||||||
# We download the missing dependencies leveraging apt
|
|
||||||
subprocess.run(shlex.split(
|
|
||||||
r"apt download " + " ".join(debs)
|
|
||||||
), cwd=downloadpath, check=True)
|
|
||||||
|
|
||||||
# then we install them inside a temporary path
|
|
||||||
temporary = os.path.abspath(os.path.join(downloadpath, 'temp'))
|
|
||||||
os.makedirs(temporary)
|
|
||||||
subprocess.run(shlex.split(
|
|
||||||
r"find " + downloadpath + r" -iname \*.deb -exec dpkg -x {} " + temporary + r" \;"
|
|
||||||
), cwd=self.builddir, check=True)
|
|
||||||
|
|
||||||
# We are finally copying the .so files in the same path as main_executable
|
|
||||||
libdirs = [ 'lib/x86_64-linux-gnu', 'usr/lib/x86_64-linux-gnu' ]
|
|
||||||
for libdir in libdirs:
|
|
||||||
fulllibdir = os.path.abspath(os.path.join(temporary, libdir))
|
|
||||||
subprocess.run(shlex.split(
|
|
||||||
f"cp -Ra {fulllibdir}/. {os.path.dirname(main_executable)}/"
|
|
||||||
), cwd=temporary, check=True)
|
|
||||||
|
|
||||||
# Find the name of the binary called in the desktop file.
|
# Find the name of the binary called in the desktop file.
|
||||||
binaryname = ''
|
binaryname = ''
|
||||||
with open(
|
with open(
|
||||||
|
@ -609,10 +584,10 @@ class Build():
|
||||||
binaryname = line.split('=')[-1].split(' ')[0]
|
binaryname = line.split('=')[-1].split(' ')[0]
|
||||||
# Esci al primo match
|
# Esci al primo match
|
||||||
break
|
break
|
||||||
|
|
||||||
#binary_exec = subprocess.run(shlex.split(r"awk 'BEGIN { FS = \"=\" } /^Exec/ { print $2; exit }' startcenter.desktop | awk '{ print $1 }'"), cwd=self.appimagedir, text=True, encoding='utf-8')
|
#binary_exec = subprocess.run(shlex.split(r"awk 'BEGIN { FS = \"=\" } /^Exec/ { print $2; exit }' startcenter.desktop | awk '{ print $1 }'"), cwd=self.appimagedir, text=True, encoding='utf-8')
|
||||||
#binaryname = binary_exec.stdout.strip("\n")
|
#binaryname = binary_exec.stdout.strip("\n")
|
||||||
|
|
||||||
# Creating a soft link so the executable in the desktop file is present
|
# Creating a soft link so the executable in the desktop file is present
|
||||||
bindir=os.path.join(self.appimagedir, 'usr', 'bin')
|
bindir=os.path.join(self.appimagedir, 'usr', 'bin')
|
||||||
os.makedirs(bindir, exist_ok = True)
|
os.makedirs(bindir, exist_ok = True)
|
||||||
|
@ -621,29 +596,93 @@ class Build():
|
||||||
r"-exec ln -sf {} ./%s \;" % binaryname
|
r"-exec ln -sf {} ./%s \;" % binaryname
|
||||||
), cwd=bindir, check=True)
|
), cwd=bindir, check=True)
|
||||||
|
|
||||||
|
def __missing_dependencies__(self):
|
||||||
|
"""Finds and copy in the appimagedir any missing libraries."""
|
||||||
|
# If the system permits it, we leverage lddcollect
|
||||||
|
# to find the packages that contain .so dependencies in the main build.
|
||||||
|
import lddcollect
|
||||||
|
# We first process the ELF
|
||||||
|
raw = lddcollect.process_elf(self.main_executable, verbose = False, dpkg = True)
|
||||||
|
|
||||||
|
# If all works as expected, we obtain a tuple of:
|
||||||
|
# (debian_packages, all_libraries, files_not_found)
|
||||||
|
(debian_packages, all_libraries, not_found) = raw
|
||||||
|
|
||||||
|
if len(debian_packages) != 0:
|
||||||
|
# Creating temporary folders
|
||||||
|
debs = [ x.split(':')[0] for x in debian_packages ]
|
||||||
|
downloadpath = os.path.abspath(os.path.join(self.builddir, 'dependencies'))
|
||||||
|
os.makedirs(downloadpath)
|
||||||
|
|
||||||
|
|
||||||
|
if self.verbose:
|
||||||
|
print("Downloading missing dependencies, please wait.")
|
||||||
|
|
||||||
|
# Let's try to find and install also other libraries
|
||||||
|
additional = list(dict.fromkeys([ Helpers.lib_to_deb(x) for x in not_found ]))
|
||||||
|
debs.extend(additional)
|
||||||
|
|
||||||
|
# It seems the download command does not download dependencies of
|
||||||
|
# the packages.
|
||||||
|
if self.verbose:
|
||||||
|
print("Constructing the dependency tree.")
|
||||||
|
|
||||||
|
for deb in debian_packages:
|
||||||
|
debs.extend(Helpers.deb_dependencies(deb))
|
||||||
|
|
||||||
|
# Re-cleaning up the dependency tree
|
||||||
|
debs = list(dict.fromkeys(debs))
|
||||||
|
|
||||||
|
# We download the missing dependencies leveraging apt
|
||||||
|
subprocess.run(shlex.split(
|
||||||
|
r"apt download " + " ".join(debs)
|
||||||
|
), cwd=downloadpath, check=True)
|
||||||
|
|
||||||
|
# then we install them inside a temporary path
|
||||||
|
temporary = os.path.abspath(os.path.join(downloadpath, 'temp'))
|
||||||
|
os.makedirs(temporary)
|
||||||
|
subprocess.run(shlex.split(
|
||||||
|
r"find " + downloadpath + r" -iname \*.deb -exec dpkg -x {} " + temporary + r" \;"
|
||||||
|
), cwd=self.builddir, check=True)
|
||||||
|
|
||||||
|
# We are finally copying the .so files in the same path as main_executable
|
||||||
|
libdirs = [ 'lib/x86_64-linux-gnu', 'usr/lib/x86_64-linux-gnu' ]
|
||||||
|
for libdir in libdirs:
|
||||||
|
fulllibdir = os.path.abspath(os.path.join(temporary, libdir))
|
||||||
|
subprocess.run(shlex.split(
|
||||||
|
f"cp -Ra {fulllibdir}/. {os.path.dirname(self.main_executable)}/"
|
||||||
|
), cwd=temporary, check=True)
|
||||||
|
|
||||||
|
if self.debug:
|
||||||
|
with open(os.path.abspath(os.builddir, 'dependencies.lst'), 'w', encoding="utf-8") as deplist:
|
||||||
|
deplist.write("\n".join(debs))
|
||||||
|
|
||||||
|
def __finalize_build__(self):
|
||||||
|
if self.verbose:
|
||||||
|
print("Finalizing build...")
|
||||||
# Cleaning up AppDir
|
# Cleaning up AppDir
|
||||||
cleanup_dirs = [ 'etc', 'lib', 'lib64', 'usr/lib', 'usr/local' ]
|
cleanup_dirs = [ 'etc', 'lib', 'lib64', 'usr/lib', 'usr/local' ]
|
||||||
for local in cleanup_dirs:
|
for local in cleanup_dirs:
|
||||||
shutil.rmtree(os.path.abspath(os.path.join(self.appimagedir, local)), ignore_errors=True)
|
shutil.rmtree(os.path.abspath(os.path.join(self.appimagedir, local)), ignore_errors=True)
|
||||||
|
|
||||||
# Download AppRun from github
|
# Download AppRun from github
|
||||||
apprunurl = r"https://github.com/AppImage/AppImageKit/releases/"
|
apprunurl = r"https://github.com/AppImage/AppImageKit/releases/"
|
||||||
apprunurl += f"download/continuous/AppRun-{self.arch}"
|
apprunurl += f"download/continuous/AppRun-{self.arch}"
|
||||||
dest = os.path.join(self.appimagedir, 'AppRun')
|
dest = os.path.join(self.appimagedir, 'AppRun')
|
||||||
self.__download__(apprunurl, dest)
|
self.__download__(apprunurl, dest)
|
||||||
os.chmod(dest, 0o755)
|
os.chmod(dest, 0o755)
|
||||||
|
|
||||||
# Dealing with extra options
|
# Dealing with extra options
|
||||||
buildopts = []
|
buildopts = []
|
||||||
if self.sign:
|
if self.sign:
|
||||||
buildopts.append('--sign')
|
buildopts.append('--sign')
|
||||||
|
|
||||||
# adding zsync build if updatable
|
# adding zsync build if updatable
|
||||||
if self.updatable:
|
if self.updatable:
|
||||||
buildopts.append(f"-u 'zsync|{self.zsyncfilename}'")
|
buildopts.append(f"-u 'zsync|{self.zsyncfilename}'")
|
||||||
|
|
||||||
buildopts_str = str.join(' ', buildopts)
|
buildopts_str = str.join(' ', buildopts)
|
||||||
|
|
||||||
# Build the number-specific build
|
# Build the number-specific build
|
||||||
if self.verbose:
|
if self.verbose:
|
||||||
print("---- Start building ----")
|
print("---- Start building ----")
|
||||||
|
@ -658,17 +697,17 @@ class Build():
|
||||||
f"{self.appimagedir}"
|
f"{self.appimagedir}"
|
||||||
), env={ "VERSION": self.appversion }, stdout=subprocess.DEVNULL,
|
), env={ "VERSION": self.appversion }, stdout=subprocess.DEVNULL,
|
||||||
stderr=subprocess.DEVNULL, check=True)
|
stderr=subprocess.DEVNULL, check=True)
|
||||||
|
|
||||||
if self.verbose:
|
if self.verbose:
|
||||||
print(f"Built AppImage version {self.appversion}")
|
print(f"Built AppImage version {self.appversion}")
|
||||||
|
|
||||||
# Cleanup phase, before new run.
|
# Cleanup phase, before new run.
|
||||||
for deb in glob.glob(self.appnamedir + '/*.deb'):
|
for deb in glob.glob(self.appnamedir + '/*.deb'):
|
||||||
os.remove(deb)
|
os.remove(deb)
|
||||||
subprocess.run(shlex.split(
|
subprocess.run(shlex.split(
|
||||||
r"find . -mindepth 1 -maxdepth 1 -type d -exec rm -rf {} \+"
|
r"find . -mindepth 1 -maxdepth 1 -type d -exec rm -rf {} \+"
|
||||||
), check=True)
|
), check=True)
|
||||||
|
|
||||||
self.built = True
|
self.built = True
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
@ -678,11 +717,31 @@ class Build():
|
||||||
shutil.rmtree(self.builddir)
|
shutil.rmtree(self.builddir)
|
||||||
|
|
||||||
|
|
||||||
def libfinderhelper(libraryname):
|
class Helpers:
|
||||||
"""Uses system tools to identify the missing package."""
|
|
||||||
|
|
||||||
libsearch = subprocess.run(shlex.split(
|
@staticmethod
|
||||||
f"sudo apt-file find -lx {libraryname}$"
|
def deb_dependencies(package_name):
|
||||||
), check=True, capture_output=True)
|
"""Returns the array of the dependencies of that package."""
|
||||||
candidate = [ x for x in libsearch.stdout.decode('utf-8').split('\n') if 'lib' in x ][0]
|
|
||||||
return candidate
|
# First pass: find dependency of that package in raw output
|
||||||
|
pass1 = subprocess.Popen(shlex.split(
|
||||||
|
f"apt-cache depends --recurse --no-recommends --no-suggests --no-conflicts --no-breaks --no-replaces --no-enhances --no-pre-depends {package_name}"
|
||||||
|
), stdout=subprocess.PIPE)
|
||||||
|
|
||||||
|
# Second pass: only grep interesting lines.
|
||||||
|
pass2 = subprocess.Popen(shlex.split(
|
||||||
|
r"grep '^\w'"
|
||||||
|
), stdin=pass1.stdout, stdout=subprocess.PIPE, encoding='utf-8')
|
||||||
|
stdout, stderr = pass2.communicate()
|
||||||
|
|
||||||
|
return stdout.strip().split("\n")
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def lib_to_deb(libraryname):
|
||||||
|
"""Uses system tools to identify the missing package."""
|
||||||
|
|
||||||
|
libsearch = subprocess.run(shlex.split(
|
||||||
|
f"sudo apt-file find -lx {libraryname}$"
|
||||||
|
), check=True, capture_output=True)
|
||||||
|
candidate = [ x for x in libsearch.stdout.decode('utf-8').split('\n') if 'lib' in x ][0]
|
||||||
|
return candidate
|
||||||
|
|
Loading…
Reference in New Issue