Completely rework the Mac app bundle so that it can get notarized

This commit is contained in:
Micah Lee 2022-01-16 11:25:15 -08:00
parent 9d034ea5ae
commit 0b31b134ab

View file

@ -14,7 +14,7 @@ root = os.path.dirname(
def run(cmd, cwd=None, error_ok=False): def run(cmd, cwd=None, error_ok=False):
print(cmd) print(f"{cmd} # cwd={cwd}")
subprocess.run(cmd, cwd=cwd, check=True) subprocess.run(cmd, cwd=cwd, check=True)
@ -37,22 +37,23 @@ def codesign(path, entitlements, identity):
str(entitlements), str(entitlements),
"--timestamp", "--timestamp",
"--deep", "--deep",
str(path),
"--force", "--force",
"--options", "--options",
"runtime", "runtime,library",
str(path),
] ]
) )
def main(): def main():
desktop_dir = os.path.join(root, "desktop") desktop_dir = f"{root}/desktop"
app_dir = f"{desktop_dir}/build/OnionShare.app"
print("○ Clean up from last build") print("○ Clean up from last build")
if os.path.exists(os.path.join(desktop_dir, "build")): if os.path.exists(f"{desktop_dir}/build"):
shutil.rmtree(os.path.join(desktop_dir, "build")) shutil.rmtree(f"{desktop_dir}/build")
if os.path.exists(os.path.join(desktop_dir, "dist")): if os.path.exists(f"{desktop_dir}/dist"):
shutil.rmtree(os.path.join(desktop_dir, "dist")) shutil.rmtree(f"{desktop_dir}/dist")
print("○ Building binaries") print("○ Building binaries")
run( run(
@ -63,41 +64,9 @@ def main():
], ],
desktop_dir, desktop_dir,
) )
before_size = get_size( before_size = get_size(f"{app_dir}")
os.path.join(desktop_dir, "build", "OnionShare.app")
)
print("○ Delete unused PySide2 stuff to save space") print("○ Delete unused Qt Frameworks")
for dirname in [
"PySide2/Designer.app",
"PySide2/examples",
"PySide2/glue",
"PySide2/Qt/qml",
"shiboken2/files.dir",
]:
shutil.rmtree(
os.path.join(
desktop_dir,
"build",
"OnionShare.app",
"Contents",
"MacOS",
"lib",
dirname,
)
)
shutil.rmtree(
os.path.join(
desktop_dir,
"build",
"OnionShare.app",
"Contents",
"MacOS",
"lib",
"shiboken2",
"docs",
)
)
for framework in [ for framework in [
"Qt3DAnimation", "Qt3DAnimation",
"Qt3DCore", "Qt3DCore",
@ -173,109 +142,144 @@ def main():
"QtXmlPatterns", "QtXmlPatterns",
]: ]:
shutil.rmtree( shutil.rmtree(
os.path.join( f"{app_dir}/Contents/MacOS/lib/PySide2/Qt/lib/{framework}.framework"
desktop_dir,
"build",
"OnionShare.app",
"Contents",
"MacOS",
"lib",
"PySide2",
"Qt",
"lib",
f"{framework}.framework",
)
) )
try: try:
os.remove( os.remove(
os.path.join( f"{app_dir}/Contents/MacOS/lib/PySide2/{framework}.abi3.so"
desktop_dir,
"build",
"OnionShare.app",
"Contents",
"MacOS",
"lib",
"PySide2",
f"{framework}.abi3.so",
)
) )
os.remove( os.remove(
os.path.join( f"{app_dir}/Contents/MacOS/lib/PySide2/{framework}.pyi"
desktop_dir,
"build",
"OnionShare.app",
"Contents",
"MacOS",
"lib",
"PySide2",
f"{framework}.pyi",
)
) )
except FileNotFoundError: except FileNotFoundError:
pass pass
after_size = get_size( print("○ Move files around so Apple will notarize")
os.path.join(desktop_dir, "build", "OnionShare.app") # https://github.com/marcelotduarte/cx_Freeze/issues/594
# https://gist.github.com/TechnicalPirate/259a9c24878fcad948452cb148af2a2c#file-custom_bdist_mac-py-L415
# Move lib from MacOS into Resources
os.rename(
f"{app_dir}/Contents/MacOS/lib",
f"{app_dir}/Contents/Resources/lib",
) )
run(
["ln", "-s", "../Resources/lib"],
cwd=f"{app_dir}/Contents/MacOS",
)
# Move frameworks from Resources/lib into Frameworks
os.makedirs(f"{app_dir}/Contents/Frameworks", exist_ok=True)
for framework_filename in glob.glob(
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib/Qt*.framework"
):
basename = os.path.basename(framework_filename)
os.rename(framework_filename, f"{app_dir}/Contents/Frameworks/{basename}")
run(
["ln", "-s", f"../../../../../Frameworks/{basename}"],
cwd=f"{app_dir}/Contents/Resources/lib/PySide2/Qt/lib",
)
if os.path.exists(f"{app_dir}/Contents/Frameworks/{basename}/Resources"):
os.rename(
f"{app_dir}/Contents/Frameworks/{basename}/Resources",
f"{app_dir}/Contents/Frameworks/{basename}/Versions/5/Resources"
)
run(
["ln", "-s", "Versions/5/Resources"],
cwd=f"{app_dir}/Contents/Frameworks/{basename}",
)
run(
["ln", "-s", "5", "Current"],
cwd=f"{app_dir}/Contents/Frameworks/{basename}/Versions",
)
# Move Qt plugins
os.rename(
f"{app_dir}/Contents/Resources/lib/PySide2/Qt/plugins",
f"{app_dir}/Contents/Frameworks/plugins",
)
run(
["ln", "-s", "../../../../Frameworks/plugins"],
cwd=f"{app_dir}/Contents/Resources/lib/PySide2/Qt",
)
print("○ Delete more unused PySide2 stuff to save space")
for filename in [
"PySide2/Designer.app",
"PySide2/examples",
"PySide2/glue",
"PySide2/include",
"PySide2/pyside2-lupdate",
"PySide2/Qt/qml",
"PySide2/libpyside2.abi3.5.15.dylib",
"PySide2/Qt/lib/QtRepParser.framework",
"PySide2/Qt/lib/QtUiPlugin.framework",
"PySide2/Qt/lib/QtWebEngineCore.framework/Helpers",
"shiboken2/libshiboken2.abi3.5.15.dylib",
"shiboken2/docs",
"PySide2/rcc",
"PySide2/uic",
]:
if os.path.isdir(filename):
shutil.rmtree(
f"{app_dir}/Contents/Resources/lib/{filename}"
)
elif os.path.isfile(filename):
os.remove(
f"{app_dir}/Contents/Resources/lib/{filename}"
)
after_size = get_size(f"{app_dir}")
freed_bytes = before_size - after_size freed_bytes = before_size - after_size
freed_mb = int(freed_bytes / 1024 / 1024) freed_mb = int(freed_bytes / 1024 / 1024)
print(f"○ Freed {freed_mb} mb") print(f"○ Freed {freed_mb} mb")
print("○ Sign app bundle") print("○ Sign app bundle")
identity_name_application = "Developer ID Application: Micah Lee (N9B95FDWH4)" identity_name_application = "Developer ID Application: Micah Lee (N9B95FDWH4)"
entitlements_plist_path = os.path.join(desktop_dir, "package", "Entitlements.plist") entitlements_plist_path = f"{desktop_dir}/package/Entitlements.plist"
for path in itertools.chain( for path in itertools.chain(
glob.glob( glob.glob(f"{app_dir}/Contents/Resources/lib/**/*.so", recursive=True),
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/**/*.dylib",
recursive=True,
),
glob.glob(
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/**/*.so", recursive=True
),
[ [
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/onionshare", f"{app_dir}/Contents/Frameworks/QtCore.framework/Versions/5/QtCore",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/onionshare-cli", f"{app_dir}/Contents/Frameworks/QtDBus.framework/Versions/5/QtDBus",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/Qt/lib/QtCore.framework/Versions/5/QtCore", f"{app_dir}/Contents/Frameworks/QtGui.framework/Versions/5/QtGui",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/Qt/lib/QtDBus.framework/Versions/5/QtDBus", f"{app_dir}/Contents/Frameworks/QtMacExtras.framework/Versions/5/QtMacExtras",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/Qt/lib/QtGui.framework/Versions/5/QtGui", f"{app_dir}/Contents/Frameworks/QtWidgets.framework/Versions/5/QtWidgets",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/Qt/lib/QtMacExtras.framework/Versions/5/QtMacExtras", f"{app_dir}/Contents/Resources/lib/Python",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/Qt/lib/QtWidgets.framework/Versions/5/QtWidgets", f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/meek-client",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/pyside2-lupdate", f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/obfs4proxy",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/rcc", f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/snowflake-client",
f"{desktop_dir}/build/OnionShare.app/Contents/MacOS/lib/PySide2/uic", f"{app_dir}/Contents/Resources/lib/onionshare/resources/tor/tor",
f"{app_dir}/Contents/MacOS/onionshare",
f"{app_dir}/Contents/MacOS/onionshare-cli",
f"{app_dir}",
], ],
): ):
codesign(path, entitlements_plist_path, identity_name_application) codesign(path, entitlements_plist_path, identity_name_application)
codesign(
f"{desktop_dir}/build/OnionShare.app", print(f"○ Signed app bundle: {app_dir}")
entitlements_plist_path,
identity_name_application,
)
print(f"○ Signed app bundle: {desktop_dir}/build/OnionShare.app")
if not os.path.exists("/usr/local/bin/create-dmg"): if not os.path.exists("/usr/local/bin/create-dmg"):
print("○ Error: create-dmg is not installed") print("○ Error: create-dmg is not installed")
return return
print("○ Create DMG") print("○ Create DMG")
version_filename = os.path.join( version_filename = f"{root}/cli/onionshare_cli/resources/version.txt"
root, "cli", "onionshare_cli", "resources", "version.txt"
)
with open(version_filename) as f: with open(version_filename) as f:
version = f.read().strip() version = f.read().strip()
os.makedirs(os.path.join(desktop_dir, "dist"), exist_ok=True) os.makedirs(f"{desktop_dir}/dist", exist_ok=True)
dmg_path = os.path.join(desktop_dir, "dist", f"OnionShare-{version}.dmg") dmg_path = f"{desktop_dir}/dist/OnionShare-{version}.dmg"
run( run(
[ [
"create-dmg", "create-dmg",
"--volname", "--volname",
"OnionShare", "OnionShare",
"--volicon", "--volicon",
os.path.join( f"{desktop_dir}/onionshare/resources/onionshare.icns",
desktop_dir, "onionshare", "resources", "onionshare.icns"
),
"--window-size", "--window-size",
"400", "400",
"200", "200",
@ -291,7 +295,7 @@ def main():
"300", "300",
"70", "70",
dmg_path, dmg_path,
f"{desktop_dir}/build/OnionShare.app", f"{app_dir}",
"--identity", "--identity",
identity_name_application, identity_name_application,
] ]
@ -299,5 +303,6 @@ def main():
print(f"○ Finished building DMG: {dmg_path}") print(f"○ Finished building DMG: {dmg_path}")
if __name__ == "__main__": if __name__ == "__main__":
main() main()