aboutsummaryrefslogtreecommitdiffstats
path: root/scripts/deploy.py
diff options
context:
space:
mode:
authorEike Ziller <eike.ziller@qt.io>2023-07-24 14:21:49 +0200
committerEike Ziller <eike.ziller@qt.io>2023-07-26 09:16:05 +0000
commitc7d0c735877f5332eab5c3f881abc161599e38d0 (patch)
tree3a715ceaf132d4481c07a28c5dfad03bc920134d /scripts/deploy.py
parent4ffbf6ec635f5392861cabea5f62898efca4665d (diff)
Build: Rename deployqt.py to deploy.py
It hasn't been about just deploying Qt for a very long time. Change-Id: I72fb070db505909500d2e68f2bafb198c3342c2b Reviewed-by: Cristian Adam <cristian.adam@qt.io>
Diffstat (limited to 'scripts/deploy.py')
-rwxr-xr-xscripts/deploy.py397
1 files changed, 397 insertions, 0 deletions
diff --git a/scripts/deploy.py b/scripts/deploy.py
new file mode 100755
index 00000000000..79aea111bdd
--- /dev/null
+++ b/scripts/deploy.py
@@ -0,0 +1,397 @@
+#!/usr/bin/env python3
+# Copyright (C) The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
+import argparse
+import collections
+import os
+import locale
+import sys
+import subprocess
+import re
+import shutil
+from glob import glob
+
+import common
+
+debug_build = False
+encoding = locale.getdefaultlocale()[1]
+
+def get_args():
+ parser = argparse.ArgumentParser(description='Deploy Qt Creator dependencies for packaging')
+ parser.add_argument('-i', '--ignore-errors', help='For backward compatibility',
+ action='store_true', default=False)
+ parser.add_argument('--elfutils-path',
+ help='Path to elfutils installation for use by perfprofiler (Windows, Linux)')
+ # TODO remove defaulting to LLVM_INSTALL_DIR when we no longer build qmake based packages
+ parser.add_argument('--llvm-path',
+ help='Path to LLVM installation',
+ default=os.environ.get('LLVM_INSTALL_DIR'))
+ parser.add_argument('qtcreator_binary', help='Path to Qt Creator binary (or the app bundle on macOS)')
+ parser.add_argument('qmake_binary', help='Path to qmake binary')
+
+ args = parser.parse_args()
+
+ args.qtcreator_binary = os.path.abspath(args.qtcreator_binary)
+ if common.is_mac_platform():
+ if not args.qtcreator_binary.lower().endswith(".app"):
+ args.qtcreator_binary = args.qtcreator_binary + ".app"
+ check = os.path.isdir
+ else:
+ check = os.path.isfile
+ if common.is_windows_platform() and not args.qtcreator_binary.lower().endswith(".exe"):
+ args.qtcreator_binary = args.qtcreator_binary + ".exe"
+
+ if not check(args.qtcreator_binary):
+ print('Cannot find Qt Creator binary.')
+ sys.exit(1)
+
+ args.qmake_binary = which(args.qmake_binary)
+ if not args.qmake_binary:
+ print('Cannot find qmake binary.')
+ sys.exit(2)
+
+ return args
+
+def usage():
+ print("Usage: %s <existing_qtcreator_binary> [qmake_path]" % os.path.basename(sys.argv[0]))
+
+def which(program):
+ def is_exe(fpath):
+ return os.path.exists(fpath) and os.access(fpath, os.X_OK)
+
+ fpath = os.path.dirname(program)
+ if fpath:
+ if is_exe(program):
+ return program
+ if common.is_windows_platform():
+ if is_exe(program + ".exe"):
+ return program + ".exe"
+ else:
+ for path in os.environ["PATH"].split(os.pathsep):
+ exe_file = os.path.join(path, program)
+ if is_exe(exe_file):
+ return exe_file
+ if common.is_windows_platform():
+ if is_exe(exe_file + ".exe"):
+ return exe_file + ".exe"
+
+ return None
+
+def is_debug(fpath):
+ # match all Qt Core dlls from Qt4, Qt5beta2 and Qt5rc1 and later
+ # which all have the number at different places
+ coredebug = re.compile(r'Qt[1-9]?Core[1-9]?d[1-9]?.dll')
+ # bootstrap exception
+ if coredebug.search(fpath):
+ return True
+ # try to use dumpbin (MSVC) or objdump (MinGW), otherwise ship all .dlls
+ if which('dumpbin'):
+ output = subprocess.check_output(['dumpbin', '/imports', fpath])
+ elif which('objdump'):
+ output = subprocess.check_output(['objdump', '-p', fpath])
+ else:
+ return debug_build
+ return coredebug.search(output.decode(encoding)) != None
+
+def is_ignored_windows_file(use_debug, basepath, filename):
+ ignore_patterns = ['.lib', '.pdb', '.exp', '.ilk']
+ if use_debug:
+ ignore_patterns.extend(['libEGL.dll', 'libGLESv2.dll'])
+ else:
+ ignore_patterns.extend(['libEGLd.dll', 'libGLESv2d.dll'])
+ for ip in ignore_patterns:
+ if filename.endswith(ip):
+ return True
+ if filename.endswith('.dll'):
+ filepath = os.path.join(basepath, filename)
+ if use_debug != is_debug(filepath):
+ return True
+ return False
+
+def ignored_qt_lib_files(path, filenames):
+ # Qt ships some unneeded object files in the qml plugins
+ # On Windows we also do not want to ship the wrong debug/release .dlls or .lib files etc
+ if not common.is_windows_platform():
+ return [fn for fn in filenames if fn.endswith('.cpp.o')]
+ return [fn for fn in filenames
+ if fn.endswith('.cpp.obj') or is_ignored_windows_file(debug_build, path, fn)]
+
+def copy_qt_libs(target_qt_prefix_path, qt_bin_dir, qt_libs_dir, qt_plugin_dir, qt_qml_dir, plugins):
+ print("copying Qt libraries...")
+
+ if common.is_windows_platform():
+ libraries = glob(os.path.join(qt_libs_dir, '*.dll'))
+ else:
+ libraries = glob(os.path.join(qt_libs_dir, '*.so.*'))
+
+ if common.is_windows_platform():
+ lib_dest = os.path.join(target_qt_prefix_path)
+ else:
+ lib_dest = os.path.join(target_qt_prefix_path, 'lib')
+
+ if not os.path.exists(lib_dest):
+ os.makedirs(lib_dest)
+
+ if common.is_windows_platform():
+ libraries = [lib for lib in libraries if not is_ignored_windows_file(debug_build, '', lib)]
+
+ for library in libraries:
+ print(library, '->', lib_dest)
+ if os.path.islink(library):
+ linkto = os.readlink(library)
+ try:
+ os.symlink(linkto, os.path.join(lib_dest, os.path.basename(library)))
+ except OSError:
+ pass
+ else:
+ shutil.copy(library, lib_dest)
+
+ print("Copying plugins:", plugins)
+ for plugin in plugins:
+ target = os.path.join(target_qt_prefix_path, 'plugins', plugin)
+ if (os.path.exists(target)):
+ shutil.rmtree(target)
+ pluginPath = os.path.join(qt_plugin_dir, plugin)
+ if (os.path.exists(pluginPath)):
+ print('{0} -> {1}'.format(pluginPath, target))
+ common.copytree(pluginPath, target, ignore=ignored_qt_lib_files, symlinks=True)
+
+ if (os.path.exists(qt_qml_dir)):
+ print("Copying qt quick 2 imports")
+ target = os.path.join(target_qt_prefix_path, 'qml')
+ if (os.path.exists(target)):
+ shutil.rmtree(target)
+ print('{0} -> {1}'.format(qt_qml_dir, target))
+ common.copytree(qt_qml_dir, target, ignore=ignored_qt_lib_files, symlinks=True)
+
+ print("Copying qtdiag")
+ bin_dest = target_qt_prefix_path if common.is_windows_platform() else os.path.join(target_qt_prefix_path, 'bin')
+ qtdiag_src = os.path.join(qt_bin_dir, 'qtdiag.exe' if common.is_windows_platform() else 'qtdiag')
+ if not os.path.exists(bin_dest):
+ os.makedirs(bin_dest)
+ shutil.copy(qtdiag_src, bin_dest)
+
+
+def add_qt_conf(target_path, qt_prefix_path):
+ qtconf_filepath = os.path.join(target_path, 'qt.conf')
+ prefix_path = os.path.relpath(qt_prefix_path, target_path).replace('\\', '/')
+ print('Creating qt.conf in "{0}":'.format(qtconf_filepath))
+ f = open(qtconf_filepath, 'w')
+ f.write('[Paths]\n')
+ f.write('Prefix={0}\n'.format(prefix_path))
+ f.write('Binaries={0}\n'.format('bin' if common.is_linux_platform() else '.'))
+ f.write('Libraries={0}\n'.format('lib' if common.is_linux_platform() else '.'))
+ f.write('Plugins=plugins\n')
+ f.write('Qml2Imports=qml\n')
+ f.close()
+
+def copy_translations(install_dir, qt_tr_dir):
+ translations = glob(os.path.join(qt_tr_dir, '*.qm'))
+ tr_dir = os.path.join(install_dir, 'share', 'qtcreator', 'translations')
+
+ print("copying translations...")
+ for translation in translations:
+ print(translation, '->', tr_dir)
+ shutil.copy(translation, tr_dir)
+
+def copyPreservingLinks(source, destination):
+ if os.path.islink(source):
+ linkto = os.readlink(source)
+ destFilePath = destination
+ if os.path.isdir(destination):
+ destFilePath = os.path.join(destination, os.path.basename(source))
+ os.symlink(linkto, destFilePath)
+ else:
+ shutil.copy(source, destination)
+
+def deploy_clang(install_dir, llvm_install_dir, chrpath_bin):
+ # contains pairs of (source, target directory)
+ deployinfo = []
+ resourcesource = os.path.join(llvm_install_dir, 'lib', 'clang')
+ if common.is_windows_platform():
+ clangbindirtarget = os.path.join(install_dir, 'bin', 'clang', 'bin')
+ if not os.path.exists(clangbindirtarget):
+ os.makedirs(clangbindirtarget)
+ clanglibdirtarget = os.path.join(install_dir, 'bin', 'clang', 'lib')
+ if not os.path.exists(clanglibdirtarget):
+ os.makedirs(clanglibdirtarget)
+ for binary in ['clangd', 'clang-tidy', 'clazy-standalone', 'clang-format']:
+ binary_filepath = os.path.join(llvm_install_dir, 'bin', binary + '.exe')
+ if os.path.exists(binary_filepath):
+ deployinfo.append((binary_filepath, clangbindirtarget))
+ resourcetarget = os.path.join(clanglibdirtarget, 'clang')
+ else:
+ # clang binaries -> clang libexec
+ clangbinary_targetdir = os.path.join(install_dir, 'libexec', 'qtcreator', 'clang', 'bin')
+ if not os.path.exists(clangbinary_targetdir):
+ os.makedirs(clangbinary_targetdir)
+ for binary in ['clangd', 'clang-tidy', 'clazy-standalone', 'clang-format']:
+ binary_filepath = os.path.join(llvm_install_dir, 'bin', binary)
+ if os.path.exists(binary_filepath):
+ deployinfo.append((binary_filepath, clangbinary_targetdir))
+ # add link target if binary is actually a symlink (to a binary in the same directory)
+ if os.path.islink(binary_filepath):
+ linktarget = os.readlink(binary_filepath)
+ deployinfo.append((os.path.join(os.path.dirname(binary_filepath), linktarget),
+ os.path.join(clangbinary_targetdir, linktarget)))
+ clanglibs_targetdir = os.path.join(install_dir, 'libexec', 'qtcreator', 'clang', 'lib')
+ # support libraries (for clazy) -> clang libexec
+ if not os.path.exists(clanglibs_targetdir):
+ os.makedirs(clanglibs_targetdir)
+ # on RHEL ClazyPlugin is in lib64
+ for lib_pattern in ['lib64/ClazyPlugin.so', 'lib/ClazyPlugin.so']:
+ for lib in glob(os.path.join(llvm_install_dir, lib_pattern)):
+ deployinfo.append((lib, clanglibs_targetdir))
+ resourcetarget = os.path.join(install_dir, 'libexec', 'qtcreator', 'clang', 'lib', 'clang')
+
+ print("copying clang...")
+ for source, target in deployinfo:
+ print(source, '->', target)
+ copyPreservingLinks(source, target)
+
+ if common.is_linux_platform():
+ # libclang was statically compiled, so there is no need for the RPATHs
+ # and they are confusing when fixing RPATHs later in the process.
+ # Also fix clazy-standalone RPATH.
+ print("fixing Clang RPATHs...")
+ for source, target in deployinfo:
+ filename = os.path.basename(source)
+ targetfilepath = target if not os.path.isdir(target) else os.path.join(target, filename)
+ if filename == 'clazy-standalone':
+ subprocess.check_call([chrpath_bin, '-r', '$ORIGIN/../lib', targetfilepath])
+ elif not os.path.islink(target):
+ targetfilepath = target if not os.path.isdir(target) else os.path.join(target, os.path.basename(source))
+ subprocess.check_call([chrpath_bin, '-d', targetfilepath])
+
+ print(resourcesource, '->', resourcetarget)
+ if (os.path.exists(resourcetarget)):
+ shutil.rmtree(resourcetarget)
+ common.copytree(resourcesource, resourcetarget, symlinks=True)
+
+def deploy_elfutils(qtc_install_dir, chrpath_bin, args):
+ if common.is_mac_platform():
+ return
+
+ libs = ['elf', 'dw']
+ version = '1'
+
+ def lib_name(name, version):
+ return ('lib' + name + '.so.' + version if common.is_linux_platform()
+ else name + '.dll')
+
+ def find_elfutils_lib_path(path):
+ for root, _, files in os.walk(path):
+ if lib_name('elf', version) in files:
+ return root
+ return path
+
+ elfutils_lib_path = find_elfutils_lib_path(os.path.join(args.elfutils_path, 'lib'))
+ if common.is_linux_platform():
+ install_path = os.path.join(qtc_install_dir, 'lib', 'elfutils')
+ backends_install_path = install_path
+ elif common.is_windows_platform():
+ install_path = os.path.join(qtc_install_dir, 'bin')
+ backends_install_path = os.path.join(qtc_install_dir, 'lib', 'elfutils')
+ libs.append('eu_compat')
+ if not os.path.exists(install_path):
+ os.makedirs(install_path)
+ if not os.path.exists(backends_install_path):
+ os.makedirs(backends_install_path)
+ # copy main libs
+ libs = [os.path.join(elfutils_lib_path, lib_name(lib, version)) for lib in libs]
+ for lib in libs:
+ print(lib, '->', install_path)
+ shutil.copy(lib, install_path)
+ # fix rpath
+ if common.is_linux_platform():
+ relative_path = os.path.relpath(backends_install_path, install_path)
+ subprocess.check_call([chrpath_bin, '-r', os.path.join('$ORIGIN', relative_path),
+ os.path.join(install_path, lib_name('dw', version))])
+ # copy backend files
+ # only non-versioned, we never dlopen the versioned ones
+ files = glob(os.path.join(elfutils_lib_path, 'elfutils', '*ebl_*.*'))
+ versioned_files = glob(os.path.join(elfutils_lib_path, 'elfutils', '*ebl_*.*-*.*.*'))
+ unversioned_files = [file for file in files if file not in versioned_files]
+ for file in unversioned_files:
+ print(file, '->', backends_install_path)
+ shutil.copy(file, backends_install_path)
+
+def deploy_mac(args):
+ (_, qt_install) = get_qt_install_info(args.qmake_binary)
+
+ env = dict(os.environ)
+ if args.llvm_path:
+ env['LLVM_INSTALL_DIR'] = args.llvm_path
+
+ script_path = os.path.dirname(os.path.realpath(__file__))
+ deployqtHelper_mac = os.path.join(script_path, 'deployqtHelper_mac.sh')
+ common.check_print_call([deployqtHelper_mac, args.qtcreator_binary, qt_install.bin,
+ qt_install.translations, qt_install.plugins, qt_install.qml],
+ env=env)
+
+def get_qt_install_info(qmake_binary):
+ qt_install_info = common.get_qt_install_info(qmake_binary)
+ QtInstallInfo = collections.namedtuple('QtInstallInfo', ['bin', 'lib', 'plugins',
+ 'qml', 'translations'])
+ return (qt_install_info,
+ QtInstallInfo(bin=qt_install_info['QT_INSTALL_BINS'],
+ lib=qt_install_info['QT_INSTALL_LIBS'],
+ plugins=qt_install_info['QT_INSTALL_PLUGINS'],
+ qml=qt_install_info['QT_INSTALL_QML'],
+ translations=qt_install_info['QT_INSTALL_TRANSLATIONS']))
+
+def main():
+ args = get_args()
+ if common.is_mac_platform():
+ deploy_mac(args)
+ return
+
+ (qt_install_info, qt_install) = get_qt_install_info(args.qmake_binary)
+
+ qtcreator_binary_path = os.path.dirname(args.qtcreator_binary)
+ install_dir = os.path.abspath(os.path.join(qtcreator_binary_path, '..'))
+ if common.is_linux_platform():
+ qt_deploy_prefix = os.path.join(install_dir, 'lib', 'Qt')
+ else:
+ qt_deploy_prefix = os.path.join(install_dir, 'bin')
+
+ chrpath_bin = None
+ if common.is_linux_platform():
+ chrpath_bin = which('chrpath')
+ if chrpath_bin == None:
+ print("Cannot find required binary 'chrpath'.")
+ sys.exit(2)
+
+ plugins = ['assetimporters', 'accessible', 'codecs', 'designer', 'iconengines', 'imageformats', 'platformthemes',
+ 'platforminputcontexts', 'platforms', 'printsupport', 'qmltooling', 'sqldrivers', 'styles',
+ 'xcbglintegrations',
+ 'wayland-decoration-client',
+ 'wayland-graphics-integration-client',
+ 'wayland-shell-integration',
+ 'tls'
+ ]
+
+ if common.is_windows_platform():
+ global debug_build
+ debug_build = is_debug(args.qtcreator_binary)
+
+ if common.is_windows_platform():
+ copy_qt_libs(qt_deploy_prefix, qt_install.bin, qt_install.bin, qt_install.plugins, qt_install.qml, plugins)
+ else:
+ copy_qt_libs(qt_deploy_prefix, qt_install.bin, qt_install.lib, qt_install.plugins, qt_install.qml, plugins)
+ copy_translations(install_dir, qt_install.translations)
+ if args.llvm_path:
+ deploy_clang(install_dir, args.llvm_path, chrpath_bin)
+
+ if args.elfutils_path:
+ deploy_elfutils(install_dir, chrpath_bin, args)
+ if not common.is_windows_platform():
+ print("fixing rpaths...")
+ common.fix_rpaths(install_dir, os.path.join(qt_deploy_prefix, 'lib'), qt_install_info, chrpath_bin)
+ add_qt_conf(os.path.join(install_dir, 'libexec', 'qtcreator'), qt_deploy_prefix) # e.g. for qml2puppet
+ add_qt_conf(os.path.join(qt_deploy_prefix, 'bin'), qt_deploy_prefix) # e.g. qtdiag
+ add_qt_conf(os.path.join(install_dir, 'bin'), qt_deploy_prefix)
+
+if __name__ == "__main__":
+ main()