2017-12-12 14:47:24 -05:00
#!/usr/bin/env python3
2011-11-02 14:58:50 +01:00
#
# Copyright (C) 2011 Patrick "p2k" Schneider <me@p2k-network.org>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
2020-11-09 11:18:14 +08:00
import plistlib
2020-11-09 17:47:21 +08:00
import subprocess, sys, re, os, shutil, stat, os.path
2011-11-02 14:58:50 +01:00
from argparse import ArgumentParser
2020-11-09 17:47:21 +08:00
from ds_store import DSStore
from mac_alias import Alias
2020-11-08 14:43:58 +08:00
from pathlib import Path
2019-07-16 10:40:31 +08:00
from typing import List, Optional
2011-11-02 14:58:50 +01:00
2012-01-19 21:45:49 +01:00
# This is ported from the original macdeployqt with modifications
class FrameworkInfo(object):
def __init__(self):
self.frameworkDirectory = ""
self.frameworkName = ""
self.frameworkPath = ""
self.binaryDirectory = ""
self.binaryName = ""
self.binaryPath = ""
self.version = ""
self.installName = ""
self.deployedInstallName = ""
self.sourceFilePath = ""
self.destinationDirectory = ""
self.sourceResourcesDirectory = ""
2014-05-30 19:14:01 -04:00
self.sourceVersionContentsDirectory = ""
self.sourceContentsDirectory = ""
2012-01-19 21:45:49 +01:00
self.destinationResourcesDirectory = ""
2014-05-30 19:14:01 -04:00
self.destinationVersionContentsDirectory = ""
2012-01-19 21:45:49 +01:00
def __eq__(self, other):
if self.__class__ == other.__class__:
return self.__dict__ == other.__dict__
else:
return False
def __str__(self):
2019-07-16 10:42:33 +08:00
return """ Framework name: {}
Framework directory: {}
Framework path: {}
Binary name: {}
Binary directory: {}
Binary path: {}
Version: {}
Install name: {}
Deployed install name: {}
Source file Path: {}
Deployed Directory (relative to bundle): {}
""".format(self.frameworkName,
2012-01-19 21:45:49 +01:00
self.frameworkDirectory,
self.frameworkPath,
self.binaryName,
self.binaryDirectory,
self.binaryPath,
self.version,
self.installName,
self.deployedInstallName,
self.sourceFilePath,
self.destinationDirectory)
def isDylib(self):
return self.frameworkName.endswith(".dylib")
def isQtFramework(self):
if self.isDylib():
return self.frameworkName.startswith("libQt")
else:
return self.frameworkName.startswith("Qt")
reOLine = re.compile(r'^(.+) \(compatibility version [0-9.]+, current version [0-9.]+\)$')
bundleFrameworkDirectory = "Contents/Frameworks"
bundleBinaryDirectory = "Contents/MacOS"
@classmethod
2019-07-16 10:40:31 +08:00
def fromOtoolLibraryLine(cls, line: str) -> Optional['FrameworkInfo']:
2012-01-19 21:45:49 +01:00
# Note: line must be trimmed
if line == "":
return None
# Don't deploy system libraries (exception for libQtuitools and libQtlucene).
if line.startswith("/System/Library/") or line.startswith("@executable_path") or (line.startswith("/usr/lib/") and "libQt" not in line):
return None
m = cls.reOLine.match(line)
if m is None:
raise RuntimeError("otool line could not be parsed: " + line)
path = m.group(1)
info = cls()
info.sourceFilePath = path
info.installName = path
if path.endswith(".dylib"):
dirname, filename = os.path.split(path)
info.frameworkName = filename
info.frameworkDirectory = dirname
info.frameworkPath = path
info.binaryDirectory = dirname
info.binaryName = filename
info.binaryPath = path
info.version = "-"
info.installName = path
info.deployedInstallName = "@executable_path/../Frameworks/" + info.binaryName
info.sourceFilePath = path
info.destinationDirectory = cls.bundleFrameworkDirectory
else:
parts = path.split("/")
i = 0
# Search for the .framework directory
for part in parts:
if part.endswith(".framework"):
break
i += 1
if i == len(parts):
raise RuntimeError("Could not find .framework or .dylib in otool line: " + line)
info.frameworkName = parts[i]
info.frameworkDirectory = "/".join(parts[:i])
info.frameworkPath = os.path.join(info.frameworkDirectory, info.frameworkName)
info.binaryName = parts[i+3]
info.binaryDirectory = "/".join(parts[i+1:i+3])
info.binaryPath = os.path.join(info.binaryDirectory, info.binaryName)
info.version = parts[i+2]
info.deployedInstallName = "@executable_path/../Frameworks/" + os.path.join(info.frameworkName, info.binaryPath)
info.destinationDirectory = os.path.join(cls.bundleFrameworkDirectory, info.frameworkName, info.binaryDirectory)
info.sourceResourcesDirectory = os.path.join(info.frameworkPath, "Resources")
2014-05-30 19:14:01 -04:00
info.sourceContentsDirectory = os.path.join(info.frameworkPath, "Contents")
info.sourceVersionContentsDirectory = os.path.join(info.frameworkPath, "Versions", info.version, "Contents")
2012-01-19 21:45:49 +01:00
info.destinationResourcesDirectory = os.path.join(cls.bundleFrameworkDirectory, info.frameworkName, "Resources")
2014-05-30 19:14:01 -04:00
info.destinationVersionContentsDirectory = os.path.join(cls.bundleFrameworkDirectory, info.frameworkName, "Versions", info.version, "Contents")
2012-01-19 21:45:49 +01:00
return info
class ApplicationBundleInfo(object):
2019-07-16 10:40:31 +08:00
def __init__(self, path: str):
2012-01-19 21:45:49 +01:00
self.path = path
2015-05-07 10:12:27 +02:00
appName = "Bitcoin-Qt"
2012-01-19 21:45:49 +01:00
self.binaryPath = os.path.join(path, "Contents", "MacOS", appName)
if not os.path.exists(self.binaryPath):
raise RuntimeError("Could not find bundle binary for " + path)
self.resourcesPath = os.path.join(path, "Contents", "Resources")
self.pluginPath = os.path.join(path, "Contents", "PlugIns")
class DeploymentInfo(object):
def __init__(self):
self.qtPath = None
self.pluginPath = None
self.deployedFrameworks = []
2019-07-16 10:40:31 +08:00
def detectQtPath(self, frameworkDirectory: str):
2012-01-19 21:45:49 +01:00
parentDir = os.path.dirname(frameworkDirectory)
if os.path.exists(os.path.join(parentDir, "translations")):
# Classic layout, e.g. "/usr/local/Trolltech/Qt-4.x.x"
self.qtPath = parentDir
2012-11-21 19:38:56 +00:00
else:
self.qtPath = os.getenv("QTDIR", None)
2012-01-19 21:45:49 +01:00
if self.qtPath is not None:
pluginPath = os.path.join(self.qtPath, "plugins")
if os.path.exists(pluginPath):
self.pluginPath = pluginPath
2019-07-16 10:40:31 +08:00
def usesFramework(self, name: str) -> bool:
2019-07-16 10:42:33 +08:00
nameDot = "{}.".format(name)
libNameDot = "lib{}.".format(name)
2012-01-19 21:45:49 +01:00
for framework in self.deployedFrameworks:
if framework.endswith(".framework"):
if framework.startswith(nameDot):
return True
elif framework.endswith(".dylib"):
if framework.startswith(libNameDot):
return True
return False
2019-07-16 10:40:31 +08:00
def getFrameworks(binaryPath: str, verbose: int) -> List[FrameworkInfo]:
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Inspecting with otool: " + binaryPath)
2013-12-06 18:08:53 -05:00
otoolbin=os.getenv("OTOOL", "otool")
2017-12-12 14:47:24 -05:00
otool = subprocess.Popen([otoolbin, "-L", binaryPath], stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True)
2012-01-19 21:45:49 +01:00
o_stdout, o_stderr = otool.communicate()
if otool.returncode != 0:
2020-11-09 11:10:39 +08:00
sys.stderr.write(o_stderr)
sys.stderr.flush()
raise RuntimeError("otool failed with return code {}".format(otool.returncode))
2016-04-02 16:45:26 +02:00
2017-12-12 14:47:24 -05:00
otoolLines = o_stdout.split("\n")
2012-01-19 21:45:49 +01:00
otoolLines.pop(0) # First line is the inspected binary
if ".framework" in binaryPath or binaryPath.endswith(".dylib"):
otoolLines.pop(0) # Frameworks and dylibs list themselves as a dependency.
libraries = []
for line in otoolLines:
2014-07-09 20:50:30 -03:00
line = line.replace("@loader_path", os.path.dirname(binaryPath))
2012-01-19 21:45:49 +01:00
info = FrameworkInfo.fromOtoolLibraryLine(line.strip())
if info is not None:
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Found framework:")
print(info)
2012-01-19 21:45:49 +01:00
libraries.append(info)
return libraries
2019-07-16 10:40:31 +08:00
def runInstallNameTool(action: str, *args):
2013-12-06 18:08:53 -05:00
installnametoolbin=os.getenv("INSTALLNAMETOOL", "install_name_tool")
subprocess.check_call([installnametoolbin, "-"+action] + list(args))
2012-01-19 21:45:49 +01:00
2019-07-16 10:40:31 +08:00
def changeInstallName(oldName: str, newName: str, binaryPath: str, verbose: int):
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Using install_name_tool:")
print(" in", binaryPath)
print(" change reference", oldName)
print(" to", newName)
2012-01-19 21:45:49 +01:00
runInstallNameTool("change", oldName, newName, binaryPath)
2019-07-16 10:40:31 +08:00
def changeIdentification(id: str, binaryPath: str, verbose: int):
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Using install_name_tool:")
print(" change identification in", binaryPath)
print(" to", id)
2012-01-19 21:45:49 +01:00
runInstallNameTool("id", id, binaryPath)
2019-07-16 10:40:31 +08:00
def runStrip(binaryPath: str, verbose: int):
2013-12-06 18:08:53 -05:00
stripbin=os.getenv("STRIP", "strip")
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Using strip:")
print(" stripped", binaryPath)
2013-12-06 18:08:53 -05:00
subprocess.check_call([stripbin, "-x", binaryPath])
2012-01-19 21:45:49 +01:00
2019-07-16 10:40:31 +08:00
def copyFramework(framework: FrameworkInfo, path: str, verbose: int) -> Optional[str]:
2012-11-21 19:38:56 +00:00
if framework.sourceFilePath.startswith("Qt"):
#standard place for Nokia Qt installer's frameworks
fromPath = "/Library/Frameworks/" + framework.sourceFilePath
else:
fromPath = framework.sourceFilePath
2012-01-19 21:45:49 +01:00
toDir = os.path.join(path, framework.destinationDirectory)
toPath = os.path.join(toDir, framework.binaryName)
if not os.path.exists(fromPath):
raise RuntimeError("No file at " + fromPath)
if os.path.exists(toPath):
return None # Already there
if not os.path.exists(toDir):
os.makedirs(toDir)
shutil.copy2(fromPath, toPath)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Copied:", fromPath)
print(" to:", toPath)
2012-02-06 09:32:35 -05:00
permissions = os.stat(toPath)
if not permissions.st_mode & stat.S_IWRITE:
os.chmod(toPath, permissions.st_mode | stat.S_IWRITE)
2012-01-19 21:45:49 +01:00
if not framework.isDylib(): # Copy resources for real frameworks
2014-05-30 19:14:01 -04:00
2014-09-29 22:03:11 -04:00
linkfrom = os.path.join(path, "Contents","Frameworks", framework.frameworkName, "Versions", "Current")
linkto = framework.version
2014-05-30 19:14:01 -04:00
if not os.path.exists(linkfrom):
os.symlink(linkto, linkfrom)
2020-11-09 11:10:39 +08:00
print("Linked:", linkfrom, "->", linkto)
2012-01-19 21:45:49 +01:00
fromResourcesDir = framework.sourceResourcesDirectory
if os.path.exists(fromResourcesDir):
toResourcesDir = os.path.join(path, framework.destinationResourcesDirectory)
2014-09-29 12:09:46 -04:00
shutil.copytree(fromResourcesDir, toResourcesDir, symlinks=True)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Copied resources:", fromResourcesDir)
print(" to:", toResourcesDir)
2014-05-30 19:14:01 -04:00
fromContentsDir = framework.sourceVersionContentsDirectory
if not os.path.exists(fromContentsDir):
fromContentsDir = framework.sourceContentsDirectory
if os.path.exists(fromContentsDir):
toContentsDir = os.path.join(path, framework.destinationVersionContentsDirectory)
2014-09-29 12:09:46 -04:00
shutil.copytree(fromContentsDir, toContentsDir, symlinks=True)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Copied Contents:", fromContentsDir)
print(" to:", toContentsDir)
2012-01-19 21:45:49 +01:00
elif framework.frameworkName.startswith("libQtGui"): # Copy qt_menu.nib (applies to non-framework layout)
qtMenuNibSourcePath = os.path.join(framework.frameworkDirectory, "Resources", "qt_menu.nib")
qtMenuNibDestinationPath = os.path.join(path, "Contents", "Resources", "qt_menu.nib")
if os.path.exists(qtMenuNibSourcePath) and not os.path.exists(qtMenuNibDestinationPath):
2014-09-29 12:09:46 -04:00
shutil.copytree(qtMenuNibSourcePath, qtMenuNibDestinationPath, symlinks=True)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Copied for libQtGui:", qtMenuNibSourcePath)
print(" to:", qtMenuNibDestinationPath)
2012-01-19 21:45:49 +01:00
return toPath
2019-07-16 10:40:31 +08:00
def deployFrameworks(frameworks: List[FrameworkInfo], bundlePath: str, binaryPath: str, strip: bool, verbose: int, deploymentInfo: Optional[DeploymentInfo] = None) -> DeploymentInfo:
2012-01-19 21:45:49 +01:00
if deploymentInfo is None:
deploymentInfo = DeploymentInfo()
while len(frameworks) > 0:
framework = frameworks.pop(0)
deploymentInfo.deployedFrameworks.append(framework.frameworkName)
2020-11-09 11:10:39 +08:00
print("Processing", framework.frameworkName, "...")
2012-01-19 21:45:49 +01:00
# Get the Qt path from one of the Qt frameworks
if deploymentInfo.qtPath is None and framework.isQtFramework():
deploymentInfo.detectQtPath(framework.frameworkDirectory)
2014-07-09 20:50:30 -03:00
if framework.installName.startswith("@executable_path") or framework.installName.startswith(bundlePath):
2020-11-09 11:10:39 +08:00
print(framework.frameworkName, "already deployed, skipping.")
2012-01-19 21:45:49 +01:00
continue
# install_name_tool the new id into the binary
changeInstallName(framework.installName, framework.deployedInstallName, binaryPath, verbose)
2017-01-29 18:19:55 +01:00
# Copy framework to app bundle.
2012-01-19 21:45:49 +01:00
deployedBinaryPath = copyFramework(framework, bundlePath, verbose)
# Skip the rest if already was deployed.
if deployedBinaryPath is None:
continue
if strip:
runStrip(deployedBinaryPath, verbose)
# install_name_tool it a new id.
changeIdentification(framework.deployedInstallName, deployedBinaryPath, verbose)
# Check for framework dependencies
dependencies = getFrameworks(deployedBinaryPath, verbose)
for dependency in dependencies:
changeInstallName(dependency.installName, dependency.deployedInstallName, deployedBinaryPath, verbose)
# Deploy framework if necessary.
if dependency.frameworkName not in deploymentInfo.deployedFrameworks and dependency not in frameworks:
frameworks.append(dependency)
return deploymentInfo
2019-07-16 10:40:31 +08:00
def deployFrameworksForAppBundle(applicationBundle: ApplicationBundleInfo, strip: bool, verbose: int) -> DeploymentInfo:
2012-01-19 21:45:49 +01:00
frameworks = getFrameworks(applicationBundle.binaryPath, verbose)
2020-11-09 11:10:39 +08:00
if len(frameworks) == 0:
2019-07-16 10:42:33 +08:00
print("Warning: Could not find any external frameworks to deploy in {}.".format(applicationBundle.path))
2012-01-19 21:45:49 +01:00
return DeploymentInfo()
else:
return deployFrameworks(frameworks, applicationBundle.path, applicationBundle.binaryPath, strip, verbose)
2019-07-16 10:40:31 +08:00
def deployPlugins(appBundleInfo: ApplicationBundleInfo, deploymentInfo: DeploymentInfo, strip: bool, verbose: int):
2012-01-19 21:45:49 +01:00
# Lookup available plugins, exclude unneeded
plugins = []
2013-12-06 18:08:53 -05:00
if deploymentInfo.pluginPath is None:
return
2012-01-19 21:45:49 +01:00
for dirpath, dirnames, filenames in os.walk(deploymentInfo.pluginPath):
pluginDirectory = os.path.relpath(dirpath, deploymentInfo.pluginPath)
if pluginDirectory == "designer":
# Skip designer plugins
continue
2019-07-16 10:48:19 +08:00
elif pluginDirectory == "printsupport":
# Skip printsupport plugins
continue
elif pluginDirectory == "imageformats":
# Skip imageformats plugins
continue
2012-01-19 21:45:49 +01:00
elif pluginDirectory == "sqldrivers":
# Deploy the sql plugins only if QtSql is in use
if not deploymentInfo.usesFramework("QtSql"):
continue
elif pluginDirectory == "script":
# Deploy the script plugins only if QtScript is in use
if not deploymentInfo.usesFramework("QtScript"):
continue
2014-10-01 19:22:20 -04:00
elif pluginDirectory == "qmltooling" or pluginDirectory == "qml1tooling":
2012-01-19 21:45:49 +01:00
# Deploy the qml plugins only if QtDeclarative is in use
if not deploymentInfo.usesFramework("QtDeclarative"):
continue
elif pluginDirectory == "bearer":
# Deploy the bearer plugins only if QtNetwork is in use
if not deploymentInfo.usesFramework("QtNetwork"):
continue
2014-10-01 19:22:20 -04:00
elif pluginDirectory == "position":
# Deploy the position plugins only if QtPositioning is in use
if not deploymentInfo.usesFramework("QtPositioning"):
continue
elif pluginDirectory == "sensors" or pluginDirectory == "sensorgestures":
# Deploy the sensor plugins only if QtSensors is in use
if not deploymentInfo.usesFramework("QtSensors"):
continue
elif pluginDirectory == "audio" or pluginDirectory == "playlistformats":
# Deploy the audio plugins only if QtMultimedia is in use
if not deploymentInfo.usesFramework("QtMultimedia"):
continue
elif pluginDirectory == "mediaservice":
# Deploy the mediaservice plugins only if QtMultimediaWidgets is in use
if not deploymentInfo.usesFramework("QtMultimediaWidgets"):
continue
2019-07-16 10:48:19 +08:00
elif pluginDirectory == "canbus":
# Deploy the canbus plugins only if QtSerialBus is in use
if not deploymentInfo.usesFramework("QtSerialBus"):
continue
elif pluginDirectory == "webview":
# Deploy the webview plugins only if QtWebView is in use
if not deploymentInfo.usesFramework("QtWebView"):
continue
elif pluginDirectory == "gamepads":
# Deploy the webview plugins only if QtGamepad is in use
if not deploymentInfo.usesFramework("QtGamepad"):
continue
elif pluginDirectory == "geoservices":
# Deploy the webview plugins only if QtLocation is in use
if not deploymentInfo.usesFramework("QtLocation"):
continue
elif pluginDirectory == "texttospeech":
# Deploy the texttospeech plugins only if QtTextToSpeech is in use
if not deploymentInfo.usesFramework("QtTextToSpeech"):
continue
elif pluginDirectory == "virtualkeyboard":
# Deploy the virtualkeyboard plugins only if QtVirtualKeyboard is in use
if not deploymentInfo.usesFramework("QtVirtualKeyboard"):
continue
elif pluginDirectory == "sceneparsers":
# Deploy the virtualkeyboard plugins only if Qt3DCore is in use
if not deploymentInfo.usesFramework("Qt3DCore"):
continue
elif pluginDirectory == "renderplugins":
# Deploy the renderplugins plugins only if Qt3DCore is in use
if not deploymentInfo.usesFramework("Qt3DCore"):
continue
elif pluginDirectory == "geometryloaders":
# Deploy the geometryloaders plugins only if Qt3DCore is in use
if not deploymentInfo.usesFramework("Qt3DCore"):
continue
2014-10-01 19:22:20 -04:00
2012-01-19 21:45:49 +01:00
for pluginName in filenames:
pluginPath = os.path.join(pluginDirectory, pluginName)
if pluginName.endswith("_debug.dylib"):
# Skip debug plugins
continue
elif pluginPath == "imageformats/libqsvg.dylib" or pluginPath == "iconengines/libqsvgicon.dylib":
# Deploy the svg plugins only if QtSvg is in use
if not deploymentInfo.usesFramework("QtSvg"):
continue
elif pluginPath == "accessible/libqtaccessiblecompatwidgets.dylib":
# Deploy accessibility for Qt3Support only if the Qt3Support is in use
if not deploymentInfo.usesFramework("Qt3Support"):
continue
elif pluginPath == "graphicssystems/libqglgraphicssystem.dylib":
# Deploy the opengl graphicssystem plugin only if QtOpenGL is in use
if not deploymentInfo.usesFramework("QtOpenGL"):
continue
2014-10-01 19:22:20 -04:00
elif pluginPath == "accessible/libqtaccessiblequick.dylib":
# Deploy the accessible qtquick plugin only if QtQuick is in use
if not deploymentInfo.usesFramework("QtQuick"):
continue
2019-07-16 10:48:19 +08:00
elif pluginPath == "platforminputcontexts/libqtvirtualkeyboardplugin.dylib":
# Deploy the virtualkeyboardplugin plugin only if QtVirtualKeyboard is in use
if not deploymentInfo.usesFramework("QtVirtualKeyboard"):
continue
2014-10-01 19:22:20 -04:00
2012-01-19 21:45:49 +01:00
plugins.append((pluginDirectory, pluginName))
for pluginDirectory, pluginName in plugins:
2020-11-09 11:10:39 +08:00
print("Processing plugin", os.path.join(pluginDirectory, pluginName), "...")
2012-01-19 21:45:49 +01:00
sourcePath = os.path.join(deploymentInfo.pluginPath, pluginDirectory, pluginName)
destinationDirectory = os.path.join(appBundleInfo.pluginPath, pluginDirectory)
if not os.path.exists(destinationDirectory):
os.makedirs(destinationDirectory)
destinationPath = os.path.join(destinationDirectory, pluginName)
shutil.copy2(sourcePath, destinationPath)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Copied:", sourcePath)
print(" to:", destinationPath)
2012-01-19 21:45:49 +01:00
if strip:
runStrip(destinationPath, verbose)
dependencies = getFrameworks(destinationPath, verbose)
for dependency in dependencies:
changeInstallName(dependency.installName, dependency.deployedInstallName, destinationPath, verbose)
# Deploy framework if necessary.
if dependency.frameworkName not in deploymentInfo.deployedFrameworks:
deployFrameworks([dependency], appBundleInfo.path, destinationPath, strip, verbose, deploymentInfo)
ap = ArgumentParser(description="""Improved version of macdeployqt.
2011-11-02 14:58:50 +01:00
Outputs a ready-to-deploy app in a folder "dist" and optionally wraps it in a .dmg file.
Note, that the "dist" folder will be deleted before deploying on each run.
2020-11-09 10:51:20 +08:00
Optionally, Qt translation files (.qm) can be added to the bundle.""")
2011-11-02 14:58:50 +01:00
ap.add_argument("app_bundle", nargs=1, metavar="app-bundle", help="application bundle to be deployed")
2020-11-09 17:47:21 +08:00
ap.add_argument("appname", nargs=1, metavar="appname", help="name of the app being deployed")
2020-11-09 11:10:39 +08:00
ap.add_argument("-verbose", nargs="?", const=True, help="Output additional debugging information")
2011-11-02 14:58:50 +01:00
ap.add_argument("-no-plugins", dest="plugins", action="store_false", default=True, help="skip plugin deployment")
ap.add_argument("-no-strip", dest="strip", action="store_false", default=True, help="don't run 'strip' on the binaries")
2020-11-09 17:47:21 +08:00
ap.add_argument("-dmg", nargs="?", const="", metavar="basename", help="create a .dmg disk image")
2020-11-08 14:43:58 +08:00
ap.add_argument("-translations-dir", nargs=1, metavar="path", default=None, help="Path to Qt's translations. Base translations will automatically be added to the bundle's resources.")
2011-11-02 14:58:50 +01:00
config = ap.parse_args()
2020-11-09 11:10:39 +08:00
verbose = config.verbose
2011-11-02 14:58:50 +01:00
# ------------------------------------------------
app_bundle = config.app_bundle[0]
2020-11-11 13:16:27 +08:00
appname = config.appname[0]
2011-11-02 14:58:50 +01:00
if not os.path.exists(app_bundle):
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: Could not find app bundle \"{}\"\n".format(app_bundle))
2011-11-02 14:58:50 +01:00
sys.exit(1)
# ------------------------------------------------
if os.path.exists("dist"):
2020-11-11 13:16:27 +08:00
print("+ Removing existing dist folder +")
2011-11-02 14:58:50 +01:00
shutil.rmtree("dist")
2020-11-11 13:16:27 +08:00
if os.path.exists(appname + ".dmg"):
print("+ Removing existing DMG +")
os.unlink(appname + ".dmg")
2015-12-10 21:49:27 +00:00
# ------------------------------------------------
2015-06-01 15:42:34 +02:00
target = os.path.join("dist", "Bitcoin-Qt.app")
2011-11-02 14:58:50 +01:00
2020-11-09 11:10:39 +08:00
print("+ Copying source bundle +")
if verbose:
2016-03-20 17:51:52 +00:00
print(app_bundle, "->", target)
2011-11-02 14:58:50 +01:00
os.mkdir("dist")
2014-09-29 12:09:46 -04:00
shutil.copytree(app_bundle, target, symlinks=True)
2011-11-02 14:58:50 +01:00
2012-01-19 21:45:49 +01:00
applicationBundle = ApplicationBundleInfo(target)
2011-11-02 14:58:50 +01:00
2012-01-19 21:45:49 +01:00
# ------------------------------------------------
2011-11-02 14:58:50 +01:00
2020-11-09 11:10:39 +08:00
print("+ Deploying frameworks +")
2011-11-02 14:58:50 +01:00
2012-01-19 21:45:49 +01:00
try:
deploymentInfo = deployFrameworksForAppBundle(applicationBundle, config.strip, verbose)
if deploymentInfo.qtPath is None:
deploymentInfo.qtPath = os.getenv("QTDIR", None)
if deploymentInfo.qtPath is None:
2020-11-09 11:10:39 +08:00
sys.stderr.write("Warning: Could not detect Qt's path, skipping plugin deployment!\n")
2012-01-19 21:45:49 +01:00
config.plugins = False
except RuntimeError as e:
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: {}\n".format(str(e)))
2014-05-15 03:27:35 -03:00
sys.exit(1)
2011-11-02 14:58:50 +01:00
# ------------------------------------------------
2012-01-19 21:45:49 +01:00
if config.plugins:
2020-11-09 11:10:39 +08:00
print("+ Deploying plugins +")
2012-01-19 21:45:49 +01:00
try:
deployPlugins(applicationBundle, deploymentInfo, config.strip, verbose)
except RuntimeError as e:
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: {}\n".format(str(e)))
2014-05-15 03:27:35 -03:00
sys.exit(1)
2012-01-19 21:45:49 +01:00
# ------------------------------------------------
2020-11-08 14:43:58 +08:00
if config.translations_dir:
if not Path(config.translations_dir[0]).exists():
sys.stderr.write("Error: Could not find translation dir \"{}\"\n".format(config.translations_dir[0]))
sys.exit(1)
2020-11-09 11:10:39 +08:00
print("+ Adding Qt translations +")
2020-11-08 14:43:58 +08:00
translations = Path(config.translations_dir[0])
regex = re.compile('qt_[a-z]*(.qm|_[A-Z]*.qm)')
lang_files = [x for x in translations.iterdir() if regex.match(x.name)]
for file in lang_files:
2020-11-09 11:10:39 +08:00
if verbose:
2020-11-08 14:43:58 +08:00
print(file.as_posix(), "->", os.path.join(applicationBundle.resourcesPath, file.name))
shutil.copy2(file.as_posix(), os.path.join(applicationBundle.resourcesPath, file.name))
2012-01-19 21:45:49 +01:00
# ------------------------------------------------
2020-11-09 11:10:39 +08:00
print("+ Installing qt.conf +")
2011-11-02 14:58:50 +01:00
2020-11-09 17:49:11 +08:00
qt_conf="""[Paths]
Translations=Resources
Plugins=PlugIns
"""
2017-04-25 09:14:57 +01:00
with open(os.path.join(applicationBundle.resourcesPath, "qt.conf"), "wb") as f:
f.write(qt_conf.encode())
2011-11-02 14:58:50 +01:00
# ------------------------------------------------
2020-11-09 17:47:21 +08:00
print("+ Generating .DS_Store +")
output_file = os.path.join("dist", ".DS_Store")
ds = DSStore.open(output_file, 'w+')
ds['.']['bwsp'] = {
'WindowBounds': '{{300, 280}, {500, 343}}',
'PreviewPaneVisibility': False,
}
icvp = {
'gridOffsetX': 0.0,
'textSize': 12.0,
'viewOptionsVersion': 1,
'backgroundImageAlias': b'\x00\x00\x00\x00\x02\x1e\x00\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xd1\x94\\\xb0H+\x00\x05\x00\x00\x00\x98\x0fbackground.tiff\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x99\xd19\xb0\xf8\x00\x00\x00\x00\x00\x00\x00\x00\xff\xff\xff\xff\x00\x00\r\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0b.background\x00\x00\x10\x00\x08\x00\x00\xd1\x94\\\xb0\x00\x00\x00\x11\x00\x08\x00\x00\xd19\xb0\xf8\x00\x00\x00\x01\x00\x04\x00\x00\x00\x98\x00\x0e\x00 \x00\x0f\x00b\x00a\x00c\x00k\x00g\x00r\x00o\x00u\x00n\x00d\x00.\x00t\x00i\x00f\x00f\x00\x0f\x00\x02\x00\x00\x00\x12\x00\x1c/.background/background.tiff\x00\x14\x01\x06\x00\x00\x00\x00\x01\x06\x00\x02\x00\x00\x0cMacintosh HD\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xce\x97\xab\xc3H+\x00\x00\x01\x88[\x88\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02u\xab\x8d\xd1\x94\\\xb0devrddsk\xff\xff\xff\xff\x00\x00\t \x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x07bitcoin\x00\x00\x10\x00\x08\x00\x00\xce\x97\xab\xc3\x00\x00\x00\x11\x00\x08\x00\x00\xd1\x94\\\xb0\x00\x00\x00\x01\x00\x14\x01\x88[\x88\x00\x16\xa9\t\x00\x08\xfaR\x00\x08\xfaQ\x00\x02d\x8e\x00\x0e\x00\x02\x00\x00\x00\x0f\x00\x1a\x00\x0c\x00M\x00a\x00c\x00i\x00n\x00t\x00o\x00s\x00h\x00 \x00H\x00D\x00\x13\x00\x01/\x00\x00\x15\x00\x02\x00\x14\xff\xff\x00\x00\xff\xff\x00\x00',
'backgroundColorBlue': 1.0,
'iconSize': 96.0,
'backgroundColorGreen': 1.0,
'arrangeBy': 'none',
'showIconPreview': True,
'gridSpacing': 100.0,
'gridOffsetY': 0.0,
'showItemInfo': False,
'labelOnBottom': True,
'backgroundType': 2,
'backgroundColorRed': 1.0
}
alias = Alias().from_bytes(icvp['backgroundImageAlias'])
alias.volume.name = appname
alias.volume.posix_path = '/Volumes/' + appname
icvp['backgroundImageAlias'] = alias.to_bytes()
ds['.']['icvp'] = icvp
ds['.']['vSrn'] = ('long', 1)
ds['Applications']['Iloc'] = (370, 156)
ds['Bitcoin-Qt.app']['Iloc'] = (128, 156)
ds.flush()
ds.close()
# ------------------------------------------------
2011-11-02 14:58:50 +01:00
if config.dmg is not None:
2013-05-27 19:55:01 -04:00
2019-07-16 10:40:31 +08:00
def runHDIUtil(verb: str, image_basename: str, **kwargs) -> int:
2011-11-02 14:58:50 +01:00
hdiutil_args = ["hdiutil", verb, image_basename + ".dmg"]
2016-03-20 17:51:52 +00:00
if "capture_stdout" in kwargs:
2011-11-02 14:58:50 +01:00
del kwargs["capture_stdout"]
run = subprocess.check_output
else:
2020-11-09 11:10:39 +08:00
if verbose:
2011-11-02 14:58:50 +01:00
hdiutil_args.append("-verbose")
run = subprocess.check_call
2016-03-20 17:51:52 +00:00
for key, value in kwargs.items():
2011-11-02 14:58:50 +01:00
hdiutil_args.append("-" + key)
2019-07-16 10:44:38 +08:00
if value is not True:
2011-11-02 14:58:50 +01:00
hdiutil_args.append(str(value))
2017-12-12 14:47:24 -05:00
return run(hdiutil_args, universal_newlines=True)
2020-11-09 17:47:21 +08:00
print("+ Preparing .dmg disk image +")
if verbose:
print("Determining size of \"dist\"...")
size = 0
for path, dirs, files in os.walk("dist"):
for file in files:
size += os.path.getsize(os.path.join(path, file))
size += int(size * 0.15)
if verbose:
print("Creating temp image for modification...")
try:
runHDIUtil("create", appname + ".temp", srcfolder="dist", format="UDRW", size=size, volname=appname, ov=True)
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
if verbose:
print("Attaching temp image...")
try:
output = runHDIUtil("attach", appname + ".temp", readwrite=True, noverify=True, noautoopen=True, capture_stdout=True)
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
m = re.search(r"/Volumes/(.+$)", output)
disk_root = m.group(0)
print("+ Applying fancy settings +")
bg_path = os.path.join(disk_root, ".background", os.path.basename('background.tiff'))
os.mkdir(os.path.dirname(bg_path))
if verbose:
print('background.tiff', "->", bg_path)
shutil.copy2('background.tiff', bg_path)
os.symlink("/Applications", os.path.join(disk_root, "Applications"))
print("+ Finalizing .dmg disk image +")
subprocess.run(["hdiutil", "detach", "/Volumes/{}".format(appname)], universal_newlines=True)
try:
runHDIUtil("convert", appname + ".temp", format="UDBZ", o=appname + ".dmg", ov=True)
except subprocess.CalledProcessError as e:
print(e)
sys.exit(e.returncode)
os.unlink(appname + ".temp.dmg")
2011-11-02 14:58:50 +01:00
# ------------------------------------------------
2020-11-09 11:10:39 +08:00
print("+ Done +")
2011-11-02 14:58:50 +01:00
sys.exit(0)