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
2014-04-28 19:36:28 -04:00
import subprocess, sys, re, os, shutil, stat, os.path, time
2011-11-02 14:58:50 +01:00
from argparse import ArgumentParser
2020-11-08 14:43:58 +08:00
from pathlib import Path
from string import Template
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)
2011-11-02 14:58:50 +01:00
qt_conf="""[Paths]
2014-03-16 16:12:52 -04:00
Translations=Resources
Plugins=PlugIns
2011-11-02 14:58:50 +01:00
"""
2012-01-19 21:45:49 +01:00
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 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")
ap.add_argument("-dmg", nargs="?", const="", metavar="basename", help="create a .dmg disk image; if basename is not specified, a camel-cased version of the app name is used")
ap.add_argument("-fancy", nargs=1, metavar="plist", default=[], help="make a fancy looking disk image using the given plist file with instructions; requires -dmg to work")
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.")
2015-12-10 21:49:27 +00:00
ap.add_argument("-volname", nargs=1, metavar="volname", default=[], help="custom volume name for dmg")
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]
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)
app_bundle_name = os.path.splitext(os.path.basename(app_bundle))[0]
# ------------------------------------------------
if len(config.fancy) == 1:
p = config.fancy[0]
2020-11-09 11:10:39 +08:00
if verbose:
2019-07-16 10:42:33 +08:00
print("Fancy: Loading \"{}\"...".format(p))
2011-11-02 14:58:50 +01:00
if not os.path.exists(p):
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: Could not find fancy disk image plist at \"{}\"\n".format(p))
2011-11-02 14:58:50 +01:00
sys.exit(1)
try:
2020-11-04 09:59:36 +01:00
with open(p, 'rb') as fp:
fancy = plistlib.load(fp, fmt=plistlib.FMT_XML)
2011-11-02 14:58:50 +01:00
except:
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: Could not parse fancy disk image plist at \"{}\"\n".format(p))
2011-11-02 14:58:50 +01:00
sys.exit(1)
try:
2016-03-20 17:51:52 +00:00
assert "window_bounds" not in fancy or (isinstance(fancy["window_bounds"], list) and len(fancy["window_bounds"]) == 4)
assert "background_picture" not in fancy or isinstance(fancy["background_picture"], str)
assert "icon_size" not in fancy or isinstance(fancy["icon_size"], int)
assert "applications_symlink" not in fancy or isinstance(fancy["applications_symlink"], bool)
if "items_position" in fancy:
2011-11-02 14:58:50 +01:00
assert isinstance(fancy["items_position"], dict)
2016-03-20 17:51:52 +00:00
for key, value in fancy["items_position"].items():
2011-11-02 14:58:50 +01:00
assert isinstance(value, list) and len(value) == 2 and isinstance(value[0], int) and isinstance(value[1], int)
except:
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: Bad format of fancy disk image plist at \"{}\"\n".format(p))
2011-11-02 14:58:50 +01:00
sys.exit(1)
2016-03-20 17:51:52 +00:00
if "background_picture" in fancy:
2011-11-02 14:58:50 +01:00
bp = fancy["background_picture"]
2020-11-09 11:10:39 +08:00
if verbose:
2019-07-16 10:42:33 +08:00
print("Fancy: Resolving background picture \"{}\"...".format(bp))
2011-11-02 14:58:50 +01:00
if not os.path.exists(bp):
bp = os.path.join(os.path.dirname(p), bp)
if not os.path.exists(bp):
2020-11-09 11:10:39 +08:00
sys.stderr.write("Error: Could not find background picture at \"{}\" or \"{}\"\n".format(fancy["background_picture"], bp))
2011-11-02 14:58:50 +01:00
sys.exit(1)
else:
fancy["background_picture"] = bp
else:
fancy = None
# ------------------------------------------------
if os.path.exists("dist"):
2020-11-09 11:10:39 +08:00
print("+ Removing old dist folder +")
2011-11-02 14:58:50 +01:00
shutil.rmtree("dist")
# ------------------------------------------------
2015-12-10 21:49:27 +00:00
if len(config.volname) == 1:
volname = config.volname[0]
else:
volname = app_bundle_name
# ------------------------------------------------
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
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
# ------------------------------------------------
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)
2011-11-02 14:58:50 +01:00
2020-11-09 11:10:39 +08:00
if fancy is None:
print("+ Creating .dmg disk image +")
else:
print("+ Preparing .dmg disk image +")
2011-11-02 14:58:50 +01:00
if config.dmg != "":
dmg_name = config.dmg
else:
spl = app_bundle_name.split(" ")
dmg_name = spl[0] + "".join(p.capitalize() for p in spl[1:])
if fancy is None:
try:
2015-12-10 21:49:27 +00:00
runHDIUtil("create", dmg_name, srcfolder="dist", format="UDBZ", volname=volname, ov=True)
2011-11-02 14:58:50 +01:00
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
else:
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Determining size of \"dist\"...")
2011-11-02 14:58:50 +01:00
size = 0
for path, dirs, files in os.walk("dist"):
for file in files:
size += os.path.getsize(os.path.join(path, file))
2015-01-19 19:08:05 -05:00
size += int(size * 0.15)
2011-11-02 14:58:50 +01:00
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Creating temp image for modification...")
2011-11-02 14:58:50 +01:00
try:
2015-12-10 21:49:27 +00:00
runHDIUtil("create", dmg_name + ".temp", srcfolder="dist", format="UDRW", size=size, volname=volname, ov=True)
2011-11-02 14:58:50 +01:00
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print("Attaching temp image...")
2011-11-02 14:58:50 +01:00
try:
output = runHDIUtil("attach", dmg_name + ".temp", readwrite=True, noverify=True, noautoopen=True, capture_stdout=True)
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
2019-02-01 17:00:02 -08:00
m = re.search(r"/Volumes/(.+$)", output)
2011-11-02 14:58:50 +01:00
disk_root = m.group(0)
disk_name = m.group(1)
2020-11-09 11:10:39 +08:00
print("+ Applying fancy settings +")
2011-11-02 14:58:50 +01:00
2016-03-20 17:51:52 +00:00
if "background_picture" in fancy:
2015-01-19 19:08:05 -05:00
bg_path = os.path.join(disk_root, ".background", os.path.basename(fancy["background_picture"]))
os.mkdir(os.path.dirname(bg_path))
2020-11-09 11:10:39 +08:00
if verbose:
2016-03-20 17:51:52 +00:00
print(fancy["background_picture"], "->", bg_path)
2011-11-02 14:58:50 +01:00
shutil.copy2(fancy["background_picture"], bg_path)
else:
bg_path = None
if fancy.get("applications_symlink", False):
os.symlink("/Applications", os.path.join(disk_root, "Applications"))
2013-01-14 11:52:15 -05:00
# The Python appscript package broke with OSX 10.8 and isn't being fixed.
# So we now build up an AppleScript string and use the osascript command
# to make the .dmg file pretty:
appscript = Template( """
on run argv
tell application "Finder"
tell disk "$disk"
open
set current view of container window to icon view
set toolbar visible of container window to false
set statusbar visible of container window to false
set the bounds of container window to {$window_bounds}
set theViewOptions to the icon view options of container window
set arrangement of theViewOptions to not arranged
set icon size of theViewOptions to $icon_size
$background_commands
$items_positions
close -- close/reopen works around a bug...
open
update without registering applications
delay 5
eject
end tell
end tell
end run
""")
itemscript = Template('set position of item "${item}" of container window to {${position}}')
items_positions = []
2016-03-20 17:51:52 +00:00
if "items_position" in fancy:
for name, position in fancy["items_position"].items():
2013-01-14 11:52:15 -05:00
params = { "item" : name, "position" : ",".join([str(p) for p in position]) }
items_positions.append(itemscript.substitute(params))
params = {
2015-12-10 21:49:27 +00:00
"disk" : volname,
2013-01-14 11:52:15 -05:00
"window_bounds" : "300,300,800,620",
"icon_size" : "96",
"background_commands" : "",
"items_positions" : "\n ".join(items_positions)
}
2016-03-20 17:51:52 +00:00
if "window_bounds" in fancy:
2016-12-22 08:02:15 -08:00
params["window_bounds"] = ",".join([str(p) for p in fancy["window_bounds"]])
2016-03-20 17:51:52 +00:00
if "icon_size" in fancy:
2013-01-14 11:52:15 -05:00
params["icon_size"] = str(fancy["icon_size"])
2011-11-02 14:58:50 +01:00
if bg_path is not None:
2013-01-14 11:52:15 -05:00
# Set background file, then call SetFile to make it invisible.
# (note: making it invisible first makes set background picture fail)
2015-01-19 19:08:05 -05:00
bgscript = Template("""set background picture of theViewOptions to file ".background:$bgpic"
do shell script "SetFile -a V /Volumes/$disk/.background/$bgpic" """)
2013-01-14 11:52:15 -05:00
params["background_commands"] = bgscript.substitute({"bgpic" : os.path.basename(bg_path), "disk" : params["disk"]})
s = appscript.substitute(params)
2020-11-09 11:10:39 +08:00
print("Running AppleScript:")
print(s)
2013-01-14 11:52:15 -05:00
p = subprocess.Popen(['osascript', '-'], stdin=subprocess.PIPE)
2016-06-03 16:36:26 -04:00
p.communicate(input=s.encode('utf-8'))
2013-01-14 11:52:15 -05:00
if p.returncode:
print("Error running osascript.")
2020-11-09 11:10:39 +08:00
print("+ Finalizing .dmg disk image +")
time.sleep(5)
2011-11-02 14:58:50 +01:00
try:
runHDIUtil("convert", dmg_name + ".temp", format="UDBZ", o=dmg_name + ".dmg", ov=True)
except subprocess.CalledProcessError as e:
sys.exit(e.returncode)
os.unlink(dmg_name + ".temp.dmg")
# ------------------------------------------------
2020-11-09 11:10:39 +08:00
print("+ Done +")
2011-11-02 14:58:50 +01:00
sys.exit(0)