|
|
|
# -*- mode: python -*-
|
|
|
|
|
|
|
|
from PyInstaller.utils.hooks import collect_data_files, collect_submodules, collect_dynamic_libs
|
|
|
|
|
|
|
|
import sys, os
|
|
|
|
|
|
|
|
PACKAGE='Electrum'
|
|
|
|
PYPKG='electrum'
|
|
|
|
MAIN_SCRIPT='run_electrum'
|
|
|
|
ICONS_FILE=PYPKG + '/gui/icons/electrum.icns'
|
|
|
|
APP_SIGN = os.environ.get('APP_SIGN', '')
|
|
|
|
|
|
|
|
def fail(*msg):
|
|
|
|
RED='\033[0;31m'
|
|
|
|
NC='\033[0m' # No Color
|
|
|
|
print("\r🗯 {}ERROR:{}".format(RED, NC), *msg)
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
def codesign(identity, binary):
|
|
|
|
d = os.path.dirname(binary)
|
|
|
|
saved_dir=None
|
|
|
|
if d:
|
|
|
|
# switch to directory of the binary so codesign verbose messages don't include long path
|
|
|
|
saved_dir = os.path.abspath(os.path.curdir)
|
|
|
|
os.chdir(d)
|
|
|
|
binary = os.path.basename(binary)
|
|
|
|
os.system("codesign -v -f -s '{}' '{}'".format(identity, binary))==0 or fail("Could not code sign " + binary)
|
|
|
|
if saved_dir:
|
|
|
|
os.chdir(saved_dir)
|
|
|
|
|
|
|
|
def monkey_patch_pyinstaller_for_codesigning(identity):
|
|
|
|
# Monkey-patch PyInstaller so that we app-sign all binaries *after* they are modified by PyInstaller
|
|
|
|
# If we app-sign before that point, the signature will be invalid because PyInstaller modifies
|
|
|
|
# @loader_path in the Mach-O loader table.
|
|
|
|
try:
|
|
|
|
import PyInstaller.depend.dylib
|
|
|
|
_saved_func = PyInstaller.depend.dylib.mac_set_relative_dylib_deps
|
|
|
|
except (ImportError, NameError, AttributeError):
|
|
|
|
# Hmm. Likely wrong PyInstaller version.
|
|
|
|
fail("Could not monkey-patch PyInstaller for code signing. Please ensure that you are using PyInstaller 3.4.")
|
|
|
|
_signed = set()
|
|
|
|
def my_func(fn, distname):
|
|
|
|
_saved_func(fn, distname)
|
|
|
|
if (fn, distname) not in _signed:
|
|
|
|
codesign(identity, fn)
|
|
|
|
_signed.add((fn,distname)) # remember we signed it so we don't sign again
|
|
|
|
PyInstaller.depend.dylib.mac_set_relative_dylib_deps = my_func
|
|
|
|
|
|
|
|
|
|
|
|
for i, x in enumerate(sys.argv):
|
|
|
|
if x == '--name':
|
|
|
|
VERSION = sys.argv[i+1]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
raise Exception('no version')
|
|
|
|
|
|
|
|
electrum = os.path.abspath(".") + "/"
|
|
|
|
block_cipher = None
|
|
|
|
|
|
|
|
# see https://github.com/pyinstaller/pyinstaller/issues/2005
|
|
|
|
hiddenimports = []
|
|
|
|
hiddenimports += collect_submodules('trezorlib')
|
|
|
|
hiddenimports += collect_submodules('safetlib')
|
|
|
|
hiddenimports += collect_submodules('btchip')
|
|
|
|
hiddenimports += collect_submodules('keepkeylib')
|
|
|
|
hiddenimports += collect_submodules('websocket')
|
|
|
|
hiddenimports += collect_submodules('ckcc')
|
|
|
|
|
|
|
|
# safetlib imports PyQt5.Qt. We use a local updated copy of pinmatrix.py until they
|
|
|
|
# release a new version that includes https://github.com/archos-safe-t/python-safet/commit/b1eab3dba4c04fdfc1fcf17b66662c28c5f2380e
|
|
|
|
hiddenimports.remove('safetlib.qt.pinmatrix')
|
|
|
|
|
|
|
|
|
|
|
|
datas = [
|
|
|
|
(electrum + PYPKG + '/*.json', PYPKG),
|
|
|
|
(electrum + PYPKG + '/wordlist/english.txt', PYPKG + '/wordlist'),
|
|
|
|
(electrum + PYPKG + '/locale', PYPKG + '/locale'),
|
|
|
|
(electrum + PYPKG + '/plugins', PYPKG + '/plugins'),
|
|
|
|
(electrum + PYPKG + '/gui/icons', PYPKG + '/gui/icons'),
|
|
|
|
]
|
|
|
|
datas += collect_data_files('trezorlib')
|
|
|
|
datas += collect_data_files('safetlib')
|
|
|
|
datas += collect_data_files('btchip')
|
|
|
|
datas += collect_data_files('keepkeylib')
|
|
|
|
datas += collect_data_files('ckcc')
|
|
|
|
|
|
|
|
# Add the QR Scanner helper app
|
|
|
|
datas += [(electrum + "contrib/osx/CalinsQRReader/build/Release/CalinsQRReader.app", "./contrib/osx/CalinsQRReader/build/Release/CalinsQRReader.app")]
|
|
|
|
|
|
|
|
# Add libusb so Trezor and Safe-T mini will work
|
|
|
|
binaries = [(electrum + "contrib/osx/libusb-1.0.dylib", ".")]
|
|
|
|
binaries += [(electrum + "contrib/osx/libsecp256k1.0.dylib", ".")]
|
|
|
|
|
|
|
|
# Workaround for "Retro Look":
|
|
|
|
binaries += [b for b in collect_dynamic_libs('PyQt5') if 'macstyle' in b[0]]
|
|
|
|
|
|
|
|
# We don't put these files in to actually include them in the script but to make the Analysis method scan them for imports
|
|
|
|
a = Analysis([electrum+ MAIN_SCRIPT,
|
|
|
|
electrum+'electrum/gui/qt/main_window.py',
|
|
|
|
electrum+'electrum/gui/text.py',
|
|
|
|
electrum+'electrum/util.py',
|
|
|
|
electrum+'electrum/wallet.py',
|
|
|
|
electrum+'electrum/simple_config.py',
|
|
|
|
electrum+'electrum/bitcoin.py',
|
|
|
|
electrum+'electrum/dnssec.py',
|
|
|
|
electrum+'electrum/commands.py',
|
|
|
|
electrum+'electrum/plugins/cosigner_pool/qt.py',
|
|
|
|
electrum+'electrum/plugins/email_requests/qt.py',
|
|
|
|
electrum+'electrum/plugins/trezor/qt.py',
|
|
|
|
electrum+'electrum/plugins/safe_t/client.py',
|
|
|
|
electrum+'electrum/plugins/safe_t/qt.py',
|
|
|
|
electrum+'electrum/plugins/keepkey/qt.py',
|
|
|
|
electrum+'electrum/plugins/ledger/qt.py',
|
|
|
|
electrum+'electrum/plugins/coldcard/qt.py',
|
|
|
|
],
|
|
|
|
binaries=binaries,
|
|
|
|
datas=datas,
|
|
|
|
hiddenimports=hiddenimports,
|
|
|
|
hookspath=[])
|
|
|
|
|
|
|
|
# http://stackoverflow.com/questions/19055089/pyinstaller-onefile-warning-pyconfig-h-when-importing-scipy-or-scipy-signal
|
|
|
|
for d in a.datas:
|
|
|
|
if 'pyconfig' in d[0]:
|
|
|
|
a.datas.remove(d)
|
|
|
|
break
|
|
|
|
|
|
|
|
# Strip out parts of Qt that we never use. Reduces binary size by tens of MBs. see #4815
|
|
|
|
qt_bins2remove=('qtweb', 'qt3d', 'qtgame', 'qtdesigner', 'qtquick', 'qtlocation', 'qttest', 'qtxml')
|
|
|
|
print("Removing Qt binaries:", *qt_bins2remove)
|
|
|
|
for x in a.binaries.copy():
|
|
|
|
for r in qt_bins2remove:
|
|
|
|
if x[0].lower().startswith(r):
|
|
|
|
a.binaries.remove(x)
|
|
|
|
print('----> Removed x =', x)
|
|
|
|
|
|
|
|
# If code signing, monkey-patch in a code signing step to pyinstaller. See: https://github.com/spesmilo/electrum/issues/4994
|
|
|
|
if APP_SIGN:
|
|
|
|
monkey_patch_pyinstaller_for_codesigning(APP_SIGN)
|
|
|
|
|
|
|
|
pyz = PYZ(a.pure, a.zipped_data, cipher=block_cipher)
|
|
|
|
|
|
|
|
exe = EXE(pyz,
|
|
|
|
a.scripts,
|
|
|
|
a.binaries,
|
|
|
|
a.datas,
|
|
|
|
name=PACKAGE,
|
|
|
|
debug=False,
|
|
|
|
strip=False,
|
|
|
|
upx=True,
|
|
|
|
icon=electrum+ICONS_FILE,
|
|
|
|
console=False)
|
|
|
|
|
|
|
|
app = BUNDLE(exe,
|
|
|
|
version = VERSION,
|
|
|
|
name=PACKAGE + '.app',
|
|
|
|
icon=electrum+ICONS_FILE,
|
|
|
|
bundle_identifier=None,
|
|
|
|
info_plist={
|
|
|
|
'NSHighResolutionCapable': 'True',
|
|
|
|
'NSSupportsAutomaticGraphicsSwitching': 'True'
|
|
|
|
}
|
|
|
|
)
|