anki/qt/bundle/mac/notarize.py
Damien Elmes 95dbf30fb9 updates to the build process and binary bundles
All platforms:

- rename scripts/ to tools/: Bazelisk expects to find its wrapper script
(used by the Mac changes below) in tools/. Rather than have a separate
scripts/ and tools/, it's simpler to just move everything into tools/.
- wheel outputs and binary bundles now go into .bazel/out/dist. While
not technically Bazel build products, doing it this way ensures they get
cleaned up when 'bazel clean' is run, and it keeps them out of the source
folder.
- update to the latest Bazel

Windows changes:

- bazel.bat has been removed, and tools\setup-env.bat has been added.
Other scripts like .\run.bat will automatically call it to set up the
environment.
- because Bazel is now on the path, you can 'bazel test ...' from any
folder, instead of having to do \anki\bazel.
- the bat files can handle being called from any working directory,
so things like running "\anki\tools\python" from c:\ will work.
- build installer as part of bundling process

Mac changes:

- `arch -arch x86_64 bazel ...` will now automatically use a different
build root, so that it is cheap to switch back and forth between archs
on a new Mac.
- tools/run-qt* will now automatically use Rosetta
- disable jemalloc in Mac x86 build for now, as it won't build under
Rosetta (perhaps due to its build scripts using $host_cpu instead of
$target_cpu)
- create app bundle as part of bundling process

Linux changes:

- remove arm64 orjson workaround in Linux bundle, as without a
readily-available, relatively distro-agonstic PyQt/Qt build
we can use, the arm64 Linux bundle is of very limited usefulness.
- update Docker files for release build
- include fcitx5 in both the qt5 and qt6 bundles
- create tarballs as part of the bundling process
2022-02-10 19:23:07 +10:00

149 lines
3.5 KiB
Python

#!/usr/bin/env python
# Copyright: Ankitects Pty Ltd and contributors
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
import os
import re
import subprocess
import sys
import time
from pathlib import Path
USERNAME = os.getenv("NOTARIZE_USER")
PASSWORD = os.getenv("NOTARIZE_PASSWORD")
BUNDLE_ID = "net.ankiweb.dtop"
def upload(base_dir: Path, uuid_path: Path) -> None:
print("--- Prepare notarization zip")
app_dir = base_dir / "Anki.app"
zip_path = app_dir.with_suffix(".zip")
subprocess.run(["ditto", "-c", "-k", "--keepParent", app_dir, zip_path])
print("--- Upload for notarization")
try:
output = subprocess.check_output(
[
"xcrun",
"altool",
"--notarize-app",
"--primary-bundle-id",
BUNDLE_ID,
"--username",
USERNAME,
"--password",
PASSWORD,
"--file",
zip_path,
],
stderr=subprocess.STDOUT,
encoding="utf8",
)
except subprocess.CalledProcessError as e:
print("error uploading:", e.output)
sys.exit(1)
uuid = None
for line in output.splitlines():
m = re.search(r"RequestUUID = (.*)", line)
if m:
uuid = m.group(1)
if not uuid:
print("no uuid found - upload output:")
print(output)
sys.exit(1)
open(uuid_path, "w").write(uuid)
zip_path.unlink()
def _extract_status(output):
for line in output.splitlines():
m = re.search(r"Status: (.*)", line)
if m:
return m.group(1)
return None
def wait_for_success(uuid, wait=True):
while True:
print("checking status...", end="", flush=True)
try:
output = subprocess.check_output(
[
"xcrun",
"altool",
"--notarization-info",
uuid,
"--username",
USERNAME,
"--password",
PASSWORD,
],
stderr=subprocess.STDOUT,
encoding="utf8",
)
except subprocess.CalledProcessError as e:
print("error checking status:")
print(e.output)
sys.exit(1)
status = _extract_status(output)
if status is None:
print("could not extract status:")
print(output)
sys.exit(1)
if status == "invalid":
print("notarization failed:")
print(output)
sys.exit(1)
if status == "success":
print("success!")
print(output)
return
print(status)
if not wait:
return
time.sleep(30)
def staple(app_path):
try:
subprocess.check_call(
[
"xcrun",
"stapler",
"staple",
app_path,
]
)
except subprocess.CalledProcessError as e:
print("error stapling:")
print(e.output)
sys.exit(1)
cmd = sys.argv[1]
base_dir = Path(sys.argv[2])
uuid_path = base_dir / "uuid"
if cmd == "upload":
upload(base_dir, uuid_path)
elif cmd == "status":
uuid = open(uuid_path).read()
wait_for_success(uuid, False)
elif cmd == "staple":
uuid = open(uuid_path).read()
wait_for_success(uuid)
staple(base_dir / "Anki.app")
uuid_path.unlink()