mirror of
https://github.com/onionshare/onionshare.git
synced 2025-06-20 12:34:23 -04:00
Completely rework the Mac app bundle so that it can get notarized
This commit is contained in:
parent
9d034ea5ae
commit
0b31b134ab
1 changed files with 115 additions and 110 deletions
|
@ -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()
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue