forked from SignTools/SignTools-CI
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sign.py
executable file
·1280 lines (1081 loc) · 49.4 KB
/
sign.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
#!/usr/bin/env python3
import copy
import os
import re
import sys
import time
import traceback
from subprocess import PIPE, Popen
import subprocess
from typing import Dict, List, NamedTuple, Set, Tuple, IO, Any, Optional, Mapping, Union
from pathlib import Path
import plistlib
import shutil
import random
import string
import tempfile
import json
from multiprocessing.pool import ThreadPool
secret_url = os.path.expandvars("$SECRET_URL").strip().rstrip("/")
secret_key = os.path.expandvars("$SECRET_KEY")
StrPath = Union[str, Path]
def safe_glob(input: Path, pattern: str):
for f in sorted(input.glob(pattern)):
if not f.name.startswith("._") and f.name not in [".DS_Store", ".AppleDouble", "__MACOSX"]:
yield f
def decode_clean(b: bytes):
return "" if not b else b.decode("utf-8").strip()
def run_process(
*cmd: str,
capture: bool = True,
check: bool = True,
env: Optional[Mapping[str, str]] = None,
cwd: Optional[str] = None,
timeout: Optional[float] = None,
):
try:
result = subprocess.run(cmd, capture_output=capture, check=check, env=env, cwd=cwd, timeout=timeout)
except (subprocess.CalledProcessError, subprocess.TimeoutExpired) as e:
raise (
Exception(
{
"stdout": decode_clean(e.stdout),
"stderr": decode_clean(e.stderr),
}
)
) from e
return result
def run_process_async(
*cmd: str,
env: Optional[Mapping[str, str]] = None,
cwd: Optional[str] = None,
):
return subprocess.Popen(cmd, env=env, cwd=cwd, stdout=PIPE, stderr=PIPE)
def rand_str(len: int, seed: Any = None):
old_state: object = None
if seed is not None:
old_state = random.getstate()
random.seed(seed)
result = "".join(random.choices(string.ascii_lowercase + string.digits, k=len))
if old_state is not None:
random.setstate(old_state)
return result
def read_file(file_path: StrPath):
with open(file_path) as f:
return f.read()
def extract_zip(archive: Path, dest_dir: Path):
return run_process("unzip", "-o", str(archive), "-d", str(dest_dir))
def archive_zip(content_dir: Path, dest_file: Path):
return run_process("zip", "-r", str(dest_file.resolve()), ".", cwd=str(content_dir))
def print_object(obj: Any):
print(json.dumps(obj, indent=4, sort_keys=True, default=str))
def plutil_convert(plist: Path):
return run_process("plutil", "-convert", "xml1", "-o", "-", str(plist), capture=True).stdout
def plist_load(plist: Path):
return plistlib.loads(plutil_convert(plist))
def plist_loads(plist: str) -> Any:
with tempfile.NamedTemporaryFile(suffix=".plist", mode="w") as f:
f.write(plist)
f.flush()
return plist_load(Path(f.name))
def plist_dump(data: Any, f: IO[bytes]):
return plistlib.dump(data, f)
def network_init():
return run_process("npm", "install", cwd="node-utils")
def node_upload(file: Path, endpoint: str, capture: bool = True):
return run_process("node", "node-utils/upload.js", str(file), endpoint, secret_key, capture=capture)
def node_download(download_url: str, output_file: Path, capture: bool = True):
return run_process(
"node",
"node-utils/download.js",
download_url,
secret_key,
str(output_file),
capture=capture,
)
def curl_with_auth(
url: str,
form_data: List[Tuple[str, str]] = [],
output: Optional[Path] = None,
check: bool = True,
capture: bool = True,
):
args = map(lambda x: ["-F", f"{x[0]}={x[1]}"], form_data)
args = [item for sublist in args for item in sublist]
if output:
args.extend(["-o", str(output)])
return run_process(
"curl",
*["-S", "-f", "-L", "-H"],
f"Authorization: Bearer {secret_key}",
*args,
url,
check=check,
capture=capture,
)
def security_get_keychain_list():
return map(
lambda x: x.strip('"'),
decode_clean(run_process("security", "list-keychains", "-d", "user").stdout).split(),
)
def security_remove_keychain(keychain: str):
keychains = security_get_keychain_list()
keychains = filter(lambda x: keychain not in x, keychains)
run_process("security", "list-keychains", "-d", "user", "-s", *keychains)
run_process("security", "delete-keychain", keychain)
def security_import(cert: Path, cert_pass: str, keychain: str) -> List[str]:
password = "1234"
keychains = [*security_get_keychain_list(), keychain]
run_process("security", "create-keychain", "-p", password, keychain),
run_process("security", "unlock-keychain", "-p", password, keychain),
run_process("security", "set-keychain-settings", keychain),
run_process("security", "list-keychains", "-d", "user", "-s", *keychains),
run_process("security", "import", str(cert), "-P", cert_pass, "-A", "-k", keychain),
run_process(
"security",
*["set-key-partition-list", "-S", "apple-tool:,apple:,codesign:", "-s", "-k"],
password,
keychain,
),
identity: str = decode_clean(run_process("security", "find-identity", "-p", "appleID", "-v", keychain).stdout)
return [line.strip('"') for line in re.findall('".*"', identity)]
def extract_tar(archive: Path, dest_dir: Path):
return run_process("tar", "-x", "-f", str(archive), "-C" + str(dest_dir))
def extract_deb(app_bin_name: str, app_bundle_id: str, archive: Path, dest_dir: Path):
with tempfile.TemporaryDirectory() as temp_dir_str:
temp_dir = Path(temp_dir_str)
run_process("ar", "x", str(archive.resolve()), cwd=str(temp_dir))
with tempfile.TemporaryDirectory() as temp_dir2_str:
temp_dir2 = Path(temp_dir2_str)
extract_tar(next(safe_glob(temp_dir, "data.tar*")), temp_dir2)
for file in safe_glob(temp_dir2, "**/*"):
if file.is_symlink():
target = file.resolve()
if target.is_absolute():
target = temp_dir2.joinpath(str(target)[1:])
os.unlink(file)
if target.is_dir():
shutil.copytree(target, file)
else:
shutil.copy2(target, file)
rootless_dir = temp_dir2 / "var" / "jb"
if rootless_dir.is_dir():
temp_dir2 = rootless_dir
for glob in [
"Library/Application Support/*/*.bundle",
"Library/Application Support/*", # *.bundle, [email protected]
"Library/Frameworks/*.framework",
"usr/lib/*.framework",
]:
for file in safe_glob(temp_dir2, glob):
# skip empty directories
if file.is_dir() and next(safe_glob(file, "*"), None) is None:
continue
move_merge_replace(file, dest_dir)
for glob in [
"Library/MobileSubstrate/DynamicLibraries/*.dylib",
"usr/lib/*.dylib",
]:
for file in safe_glob(temp_dir2, glob):
if not file.is_file():
continue
file_plist = file.parent.joinpath(file.stem + ".plist")
if file_plist.exists():
info = plist_load(file_plist)
if "Filter" in info:
ok = False
if "Bundles" in info["Filter"] and app_bundle_id in info["Filter"]["Bundles"]:
ok = True
elif "Executables" in info["Filter"] and app_bin_name in info["Filter"]["Executables"]:
ok = True
if not ok:
continue
move_merge_replace(file, dest_dir)
def move_merge_replace(src: Path, dest_dir: Path):
dest = dest_dir.joinpath(src.name)
if src == dest:
return
dest_dir.mkdir(exist_ok=True, parents=True)
if src.is_dir():
shutil.copytree(src, dest, dirs_exist_ok=True)
shutil.rmtree(src)
else:
shutil.copy2(src, dest)
os.remove(src)
def file_is_type(file: Path, type: str):
return type in decode_clean(run_process("file", str(file)).stdout)
def get_otool_imports(binary: Path):
output = decode_clean(run_process("otool", "-L", str(binary)).stdout).splitlines()[1:]
matches = [re.search(r"(.+)\s\(.+\)", line.strip()) for line in output]
results = [match.group(1) for match in matches if match]
if len(output) != len(results):
raise Exception("Failed to parse imports", {"output": output, "parsed": results})
return results
def install_name_change(binary: Path, old: Path, new: Path):
print("Re-linking", binary, old, new)
return run_process("install_name_tool", "-change", str(old), str(new), str(binary))
def insert_dylib(binary: Path, path: Path):
return run_process("./insert_dylib", "--inplace", "--no-strip-codesig", str(path), str(binary))
def get_binary_map(dir: Path):
return {file.name: file for file in safe_glob(dir, "**/*") if file_is_type(file, "Mach-O")}
def codesign_async(identity: str, component: Path, entitlements: Optional[Path] = None):
cmd = ["codesign", "--continue", "-f", "--no-strict", "-s", identity]
if entitlements:
cmd.extend(["--entitlements", str(entitlements)])
return run_process_async(*cmd, str(component))
def clean_dev_portal_name(name: str):
return re.sub("[^0-9a-zA-Z]+", " ", name).strip()
def fastlane_auth(account_name: str, account_pass: str, team_id: str):
my_env = os.environ.copy()
my_env["FASTLANE_USER"] = account_name
my_env["FASTLANE_PASSWORD"] = account_pass
my_env["FASTLANE_TEAM_ID"] = team_id
auth_pipe = subprocess.Popen(
# enable copy to clipboard so we're not interactively prompted
["fastlane", "spaceauth", "--copy_to_clipboard"],
stdin=PIPE,
stdout=PIPE,
stderr=PIPE,
env=my_env,
)
start_time = time.time()
while True:
if time.time() - start_time > 60:
raise Exception("Operation timed out")
else:
result = auth_pipe.poll()
if result == 0:
print("Logged in!")
break
elif result is not None:
raise Exception(f"Error logging in, got result: {result}")
account_2fa_file = Path("account_2fa.txt")
result = curl_with_auth(
f"{secret_url}/jobs/{job_id}/2fa",
output=account_2fa_file,
check=False,
)
if result.returncode == 0:
account_2fa = read_file(account_2fa_file)
auth_pipe.communicate((account_2fa + "\n").encode())
time.sleep(1)
def fastlane_register_app_extras(
my_env: Dict[Any, Any],
bundle_id: str,
extra_type: str,
extra_prefix: str,
matchable_entitlements: List[str],
entitlements: Dict[Any, Any],
):
matched_ids: Set[str] = set()
for k, v in entitlements.items():
if k in matchable_entitlements:
if type(v) is list:
matched_ids.update(v)
elif type(v) is str:
matched_ids.add(v)
else:
raise Exception(f"Unknown value type for {v}: {type(v)}")
# ensure all ids are prefixed correctly or registration will fail
# some matchable entitlements are incorrectly prefixed with team id
matched_ids = set(
id if id.startswith(extra_prefix) else extra_prefix + id[id.index(".") + 1 :] for id in matched_ids
)
jobs: List[Popen[bytes]] = []
for id in matched_ids:
jobs.append(
run_process_async(
"fastlane",
"produce",
extra_type,
"--skip_itc",
"-g",
id,
"-n",
clean_dev_portal_name(f"ST {id}"),
env=my_env,
)
)
for pipe in jobs:
if pipe.poll() is None:
pipe.wait()
popen_check(pipe)
run_process(
"fastlane",
"produce",
f"associate_{extra_type}",
"--skip_itc",
"--app_identifier",
bundle_id,
*matched_ids,
env=my_env,
)
def fastlane_register_app(
account_name: str, account_pass: str, team_id: str, bundle_id: str, entitlements: Dict[Any, Any]
):
my_env = os.environ.copy()
my_env["FASTLANE_USER"] = account_name
my_env["FASTLANE_PASSWORD"] = account_pass
my_env["FASTLANE_TEAM_ID"] = team_id
# no-op if already exists
run_process(
"fastlane",
"produce",
"create",
"--skip_itc",
"--app_identifier",
bundle_id,
"--app-name",
clean_dev_portal_name(f"ST {bundle_id}"),
env=my_env,
)
supported_services = [
"--push-notification",
"--health-kit",
"--home-kit",
"--wireless-accessory",
"--inter-app-audio",
"--extended-virtual-address-space",
"--multipath",
"--network-extension",
"--personal-vpn",
"--access-wifi",
"--nfc-tag-reading",
"--siri-kit",
"--associated-domains",
"--icloud",
"--app-group",
]
# clear any previous services
run_process(
"fastlane",
"produce",
"disable_services",
"--skip_itc",
"--app_identifier",
bundle_id,
*supported_services,
env=my_env,
)
icloud_entitlements = [
"com.apple.developer.icloud-container-development-container-identifiers",
"com.apple.developer.icloud-container-identifiers",
"com.apple.developer.ubiquity-container-identifiers",
"com.apple.developer.ubiquity-kvstore-identifier",
]
group_entitlements = ["com.apple.security.application-groups"]
entitlement_map: Dict[str, Tuple[str, ...]] = {
"aps-environment": tuple(["--push-notification"]), # iOS
"com.apple.developer.aps-environment": tuple(["--push-notification"]), # macOS
"com.apple.developer.healthkit": tuple(["--health-kit"]),
"com.apple.developer.homekit": tuple(["--home-kit"]),
"com.apple.external-accessory.wireless-configuration": tuple(["--wireless-accessory"]),
"inter-app-audio": tuple(["--inter-app-audio"]),
"com.apple.developer.kernel.extended-virtual-addressing": tuple(["--extended-virtual-address-space"]),
"com.apple.developer.networking.multipath": tuple(["--multipath"]),
"com.apple.developer.networking.networkextension": tuple(["--network-extension"]),
"com.apple.developer.networking.vpn.api": tuple(["--personal-vpn"]),
"com.apple.developer.networking.wifi-info": tuple(["--access-wifi"]),
"com.apple.developer.nfc.readersession.formats": tuple(["--nfc-tag-reading"]),
"com.apple.developer.siri": tuple(["--siri-kit"]),
"com.apple.developer.associated-domains": tuple(["--associated-domains"]),
}
for k in icloud_entitlements:
entitlement_map[k] = tuple(["--icloud", "xcode6_compatible"])
for k in group_entitlements:
entitlement_map[k] = tuple(["--app-group"])
service_flags = set(entitlement_map[f] for f in entitlements.keys() if f in entitlement_map)
service_flags = [item for sublist in service_flags for item in sublist]
print("Enabling services:", service_flags)
run_process(
"fastlane",
"produce",
"enable_services",
"--skip_itc",
"--app_identifier",
bundle_id,
*service_flags,
env=my_env,
)
app_extras = [("cloud_container", "iCloud.", icloud_entitlements), ("group", "group.", group_entitlements)]
with ThreadPool(len(app_extras)) as p:
p.starmap(
lambda extra_type, extra_prefix, matchable_entitlements: fastlane_register_app_extras(
my_env, bundle_id, extra_type, extra_prefix, matchable_entitlements, entitlements
),
app_extras,
)
def fastlane_get_prov_profile(
account_name: str, account_pass: str, team_id: str, bundle_id: str, prov_type: str, platform: str, out_file: Path
):
my_env = os.environ.copy()
my_env["FASTLANE_USER"] = account_name
my_env["FASTLANE_PASSWORD"] = account_pass
my_env["FASTLANE_TEAM_ID"] = team_id
with tempfile.TemporaryDirectory() as tmpdir_str:
run_process(
"fastlane",
"sigh",
"renew",
"--app_identifier",
bundle_id,
"--provisioning_name",
clean_dev_portal_name(f"ST {bundle_id} {prov_type}"),
"--force",
"--skip_install",
"--include_mac_in_profiles",
"--platform",
platform,
"--" + prov_type,
"--output_path",
tmpdir_str,
"--filename",
"prov.mobileprovision",
env=my_env,
)
shutil.copy2(Path(tmpdir_str).joinpath("prov.mobileprovision"), out_file)
def codesign_dump_entitlements(component: Path) -> Dict[Any, Any]:
entitlements_str = decode_clean(
run_process("codesign", "--no-strict", "-d", "--entitlements", ":-", str(component)).stdout
)
return plist_loads(entitlements_str)
def binary_replace(pattern: str, f: Path):
if not f.exists() or not f.is_file():
raise Exception(f, "does not exist or is a directory")
return run_process("perl", "-p", "-i", "-e", pattern, str(f))
def security_dump_prov(f: Path):
return decode_clean(run_process("security", "cms", "-D", "-i", str(f)).stdout)
def dump_prov(prov_file: Path) -> Dict[Any, Any]:
s = security_dump_prov(prov_file)
return plist_loads(s)
def dump_prov_entitlements(prov_file: Path) -> Dict[Any, Any]:
return dump_prov(prov_file)["Entitlements"]
def popen_check(pipe: Popen[bytes]):
if pipe.returncode != 0:
data = {"message": f"{pipe.args} failed with status code {pipe.returncode}"}
if pipe.stdout:
data["stdout"] = decode_clean(pipe.stdout.read())
if pipe.stderr:
data["stderr"] = decode_clean(pipe.stderr.read())
raise Exception(data)
def inject_tweaks(ipa_dir: Path, tweaks_dir: Path):
main_app = get_main_app_path(ipa_dir)
main_info_plist = get_info_plist_path(main_app)
info = plist_load(main_info_plist)
app_bundle_id = info["CFBundleIdentifier"]
app_bundle_exe = info["CFBundleExecutable"]
is_mac_app = main_info_plist.parent.name == "Contents"
if is_mac_app:
base_dir = main_info_plist.parent
app_bin = base_dir.joinpath("MacOS", app_bundle_exe)
base_load_path = Path("@executable_path").joinpath("..")
else:
base_dir = main_app
app_bin = base_dir.joinpath(app_bundle_exe)
base_load_path = Path("@executable_path")
with tempfile.TemporaryDirectory() as temp_dir_str:
temp_dir = Path(temp_dir_str)
for tweak in safe_glob(tweaks_dir, "*"):
print("Processing", tweak.name)
if tweak.suffix == ".zip":
extract_zip(tweak, temp_dir)
elif tweak.suffix == ".tar":
extract_tar(tweak, temp_dir)
elif tweak.suffix == ".deb":
extract_deb(app_bin.name, app_bundle_id, tweak, temp_dir)
else:
move_merge_replace(tweak, temp_dir)
# move files if we know where they need to go
move_map = {"Frameworks": ["*.framework", "*.dylib"], "PlugIns": ["*.appex"]}
for dest_dir, globs in move_map.items():
for glob in globs:
for file in safe_glob(temp_dir, glob):
move_merge_replace(file, temp_dir.joinpath(dest_dir))
# NOTE: https://iphonedev.wiki/index.php/Cydia_Substrate
# hooking with "MSHookFunction" does not work in a jailed environment using any of the libs
# libsubstrate will silently fail and continue, while the rest will crash the app
# if you're a tweak developer, use fishhook instead, though it only works on public symbols
support_libs = {
# Path("./libhooker"): ["libhooker.dylib", "libblackjack.dylib"],
# Path("./libsubstitute"): ["libsubstitute.dylib", "libsubstitute.0.dylib"],
Path("./libsubstrate"): ["libsubstrate.dylib", "CydiaSubstrate"],
}
aliases = {
"libsubstitute.0.dylib": "libsubstitute.dylib",
"CydiaSubstrate": "libsubstrate.dylib",
}
binary_map = get_binary_map(temp_dir)
# inject any user libs
for binary_path in binary_map.values():
binary_rel = binary_path.relative_to(temp_dir)
if (len(binary_rel.parts) == 2 and binary_rel.parent.name == "Frameworks") or (
len(binary_rel.parts) == 3
and binary_rel.parent.suffix == ".framework"
and binary_rel.parent.parent.name == "Frameworks"
):
binary_fixed = base_load_path.joinpath(binary_rel)
print("Injecting", binary_path, binary_fixed)
insert_dylib(app_bin, binary_fixed)
# detect any references to support libs and install missing files
for binary_path in binary_map.values():
for link in get_otool_imports(binary_path):
link_path = Path(link)
for lib_dir, lib_names in support_libs.items():
if link_path.name not in lib_names:
continue
print("Detected", lib_dir.name)
for lib_src in safe_glob(lib_dir, "*"):
lib_dest = temp_dir.joinpath("Frameworks").joinpath(lib_src.name)
if not lib_dest.exists():
print(f"Installing {lib_src.name} to {lib_dest}")
lib_dest.parent.mkdir(exist_ok=True, parents=True)
shutil.copy2(lib_src, lib_dest)
# refresh the binary map with any new libs from previous step
binary_map = get_binary_map(temp_dir)
# re-link any dependencies
for binary_path in binary_map.values():
for link in get_otool_imports(binary_path):
link_path = Path(link)
link_name = aliases[link_path.name] if link_path.name in aliases else link_path.name
if link_name in binary_map:
link_fixed = base_load_path.joinpath(binary_map[link_name].relative_to(temp_dir))
print("Re-linking", binary_path, link_path, link_fixed)
install_name_change(binary_path, link_path, link_fixed)
for file in safe_glob(temp_dir, "*"):
move_merge_replace(file, base_dir)
class SignOpts(NamedTuple):
app_dir: Path
common_name: str
team_id: str
account_name: str
account_pass: str
prov_file: Optional[Path]
bundle_id: Optional[str]
bundle_name: Optional[str]
patch_debug: bool
patch_all_devices: bool
patch_mac: bool
patch_file_sharing: bool
encode_ids: bool
patch_ids: bool
force_original_id: bool
class RemapDef(NamedTuple):
entitlements: List[str]
prefix: str
prefix_only: bool
is_list: bool
class ComponentData(NamedTuple):
old_bundle_id: str
bundle_id: str
entitlements: Dict[Any, Any]
info_plist: Path
def get_info_plist_path(app_dir: Path):
return min(list(safe_glob(app_dir, "**/Info.plist")), key=lambda p: len(str(p)))
def get_main_app_path(app_dir: Path):
return min(list(safe_glob(app_dir, "**/*.app")), key=lambda p: len(str(p)))
class Signer:
opts: SignOpts
main_bundle_id: str
old_main_bundle_id: str
mappings: Dict[str, str]
removed_entitlements: Set[str]
is_distribution: bool
components: List[Path]
is_mac_app: bool
def gen_id(self, input_id: str):
"""
Encodes the provided id into a different but constant id that
has the same length and is unique based on the team id.
"""
if not input_id.strip():
return input_id
if not self.opts.encode_ids:
return input_id
new_parts = map(lambda x: rand_str(len(x), x + self.opts.team_id), input_id.split("."))
result = ".".join(new_parts)
return result
def __get_application_identifier_key(self):
return "com.apple.application-identifier" if self.is_mac_app else "application-identifier"
def __get_aps_environment_key(self):
return "com.apple.developer.aps-environment" if self.is_mac_app else "aps-environment"
def __init__(self, opts: SignOpts):
self.opts = opts
main_app = get_main_app_path(opts.app_dir)
main_info_plist = get_info_plist_path(main_app)
main_info: Dict[Any, Any] = plist_load(main_info_plist)
self.old_main_bundle_id = main_info["CFBundleIdentifier"]
self.is_distribution = "Distribution" in opts.common_name
self.is_mac_app = main_info_plist.parent.name == "Contents"
if self.is_distribution and self.is_mac_app:
raise Exception(
"Cannot use distribution certificate for macOS as the platform does not support adhoc provisioning profiles."
)
self.mappings: Dict[str, str] = {}
self.removed_entitlements = set()
if opts.prov_file:
if opts.bundle_id is None:
print("Using original bundle id")
self.main_bundle_id = self.old_main_bundle_id
elif opts.bundle_id == "":
print("Using provisioning profile's application id")
prov_app_id = dump_prov_entitlements(opts.prov_file)[self.__get_application_identifier_key()]
self.main_bundle_id = prov_app_id[prov_app_id.find(".") + 1 :]
if self.main_bundle_id == "*":
print("Provisioning profile is wildcard, using original bundle id")
self.main_bundle_id = self.old_main_bundle_id
else:
print("Using custom bundle id")
self.main_bundle_id = opts.bundle_id
else:
if opts.bundle_id:
print("Using custom bundle id")
self.main_bundle_id = opts.bundle_id
elif opts.encode_ids:
print("Using encoded original bundle id")
self.main_bundle_id = self.gen_id(self.old_main_bundle_id)
if not self.opts.force_original_id and self.old_main_bundle_id != self.main_bundle_id:
self.mappings[self.old_main_bundle_id] = self.main_bundle_id
else:
print("Using original bundle id")
self.main_bundle_id = self.old_main_bundle_id
if opts.bundle_name:
print(f"Setting CFBundleDisplayName to {opts.bundle_name}")
main_info["CFBundleDisplayName"] = opts.bundle_name
if self.opts.patch_all_devices:
if self.is_mac_app:
# https://developer.apple.com/documentation/bundleresources/information_property_list/lsminimumsystemversion
main_info["LSMinimumSystemVersion"] = "10.0"
else:
# https://developer.apple.com/documentation/bundleresources/information_property_list/minimumosversion
main_info["MinimumOSVersion"] = "3.0"
with open("bundle_id.txt", "w") as f:
if opts.force_original_id:
f.write(self.old_main_bundle_id)
else:
f.write(self.main_bundle_id)
with main_info_plist.open("wb") as f:
plist_dump(main_info, f)
for watch_name in ["com.apple.WatchPlaceholder", "Watch"]:
watch_dir = main_app.joinpath(watch_name)
if watch_dir.exists():
print(f"Removing {watch_name} directory")
shutil.rmtree(watch_dir)
component_exts = ["*.app", "*.appex", "*.framework", "*.dylib", "PlugIns/*.bundle"]
# make sure components are ordered depth-first, otherwise signing will overlap and become invalid
self.components = [item for e in component_exts for item in safe_glob(main_app, "**/" + e)][::-1]
self.components.append(main_app)
def __sign_secondary(self, component: Path, tmpdir: Path):
# entitlements of frameworks, etc. don't matter, so leave them (potentially) invalid
print("Signing with original entitlements")
return codesign_async(self.opts.common_name, component)
def __sign_primary(self, component: Path, tmpdir: Path, data: ComponentData):
info = plist_load(data.info_plist)
if self.opts.force_original_id:
print("Keeping original CFBundleIdentifier")
info["CFBundleIdentifier"] = data.old_bundle_id
else:
print(f"Setting CFBundleIdentifier to {data.bundle_id}")
info["CFBundleIdentifier"] = data.bundle_id
if self.opts.patch_debug:
data.entitlements["get-task-allow"] = True
print("Enabled app debugging")
else:
data.entitlements.pop("get-task-allow", False)
print("Disabled app debugging")
if not self.is_mac_app:
if self.opts.patch_all_devices:
print("Force enabling support for all devices")
info.pop("UISupportedDevices", False)
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/iPhoneOSKeys.html
info["UIDeviceFamily"] = [1, 2, 3, 4] # iOS, iPadOS, tvOS, watchOS
if self.opts.patch_mac:
info.pop("UIRequiresFullScreen", False)
for device in ["ipad", "iphone", "ipod"]:
info.pop("UISupportedInterfaceOrientations~" + device, False)
info["UISupportedInterfaceOrientations"] = [
"UIInterfaceOrientationPortrait",
"UIInterfaceOrientationPortraitUpsideDown",
"UIInterfaceOrientationLandscapeLeft",
"UIInterfaceOrientationLandscapeRight",
]
if self.opts.patch_file_sharing:
print("Force enabling file sharing")
info["UIFileSharingEnabled"] = True
info["UISupportsDocumentBrowser"] = True
with data.info_plist.open("wb") as f:
plist_dump(info, f)
print("Signing with entitlements:")
print_object(data.entitlements)
# iOS : MyApp.app/embedded.mobileprovision
# macOS : MyApp.app/Contents/embedded.provisionprofile
embedded_prov = data.info_plist.parent.joinpath(
"embedded.provisionprofile" if self.is_mac_app else "embedded.mobileprovision"
)
if self.opts.prov_file is not None:
shutil.copy2(self.opts.prov_file, embedded_prov)
else:
print("Registering component with Apple...")
fastlane_register_app(
self.opts.account_name, self.opts.account_pass, self.opts.team_id, data.bundle_id, data.entitlements
)
print("Generating provisioning profile...")
prov_type = "adhoc" if self.is_distribution else "development"
platform = "macos" if self.is_mac_app else "ios"
fastlane_get_prov_profile(
self.opts.account_name,
self.opts.account_pass,
self.opts.team_id,
data.bundle_id,
prov_type,
platform,
embedded_prov,
)
entitlements_plist = Path(tmpdir).joinpath("entitlements.plist")
with open(entitlements_plist, "wb") as f:
plist_dump(data.entitlements, f)
print("Signing component...")
return codesign_async(self.opts.common_name, component, entitlements_plist)
def __prepare_primary(
self,
component: Path,
workdir: Path,
):
info_plist = get_info_plist_path(component)
info: Dict[Any, Any] = plist_load(info_plist)
old_bundle_id = info["CFBundleIdentifier"]
# create bundle id by suffixing the existing main bundle id with the original suffix
bundle_id = f"{self.main_bundle_id}{old_bundle_id[len(self.old_main_bundle_id):]}"
if not self.opts.force_original_id and old_bundle_id != bundle_id:
if len(old_bundle_id) != len(bundle_id):
print(
f"WARNING: Component's bundle id '{bundle_id}' is different length from the original bundle id '{old_bundle_id}'.",
"The signed app may crash!",
)
else:
self.mappings[old_bundle_id] = bundle_id
old_entitlements: Dict[Any, Any]
try:
old_entitlements = codesign_dump_entitlements(component)
except:
print("Failed to dump entitlements, using empty")
old_entitlements = {}
print("Original entitlements:")
print_object(old_entitlements)
old_team_id: Optional[str] = old_entitlements.get("com.apple.developer.team-identifier", None)
if not old_team_id:
print("Failed to read old team id")
elif old_team_id != self.opts.team_id:
if len(old_team_id) != len(self.opts.team_id):
print("WARNING: Team ID length mismatch:", old_team_id, self.opts.team_id)
else:
self.mappings[old_team_id] = self.opts.team_id
# before 2011 this was known as 'bundle seed id' and could be set freely
# now it is always equal to team id, but some old apps haven't updated
old_app_id_prefix: Optional[str] = old_entitlements.get(self.__get_application_identifier_key(), "").split(
"."
)[0]
if not old_app_id_prefix:
old_app_id_prefix = None
print("Failed to read old app id prefix")
elif old_app_id_prefix != self.opts.team_id:
if len(old_app_id_prefix) != len(self.opts.team_id):
print("WARNING: App ID Prefix length mismatch:", old_app_id_prefix, self.opts.team_id)
else:
self.mappings[old_app_id_prefix] = self.opts.team_id
if self.opts.prov_file is not None:
# This may cause issues with wildcard entitlements, since they are valid in the provisioning
# profile, but not when applied to a binary. For example:
# com.apple.developer.icloud-services = *
# Ideally, all such cases should be manually replaced.
entitlements = dump_prov_entitlements(self.opts.prov_file)
prov_app_id = entitlements[self.__get_application_identifier_key()]
component_app_id = f"{self.opts.team_id}.{bundle_id}"
wildcard_app_id = f"{self.opts.team_id}.*"
# if the prov file has wildcard app id, expand it, or it would be invalid
if prov_app_id == wildcard_app_id:
entitlements[self.__get_application_identifier_key()] = component_app_id
elif prov_app_id != component_app_id:
print(
f"WARNING: Provisioning profile's app id '{prov_app_id}' does not match component's app id '{component_app_id}'.",
"Using provisioning profile's app id - the component will run, but some functions such as file importing will not work!",
sep="\n",
)
keychain: Optional[List[str]] = entitlements.get("keychain-access-groups", None)
old_keychain: Optional[List[str]] = old_entitlements.get("keychain-access-groups", None)
if old_keychain is None:
entitlements.pop("keychain-access-groups", None)
# if the prov file has wildcard keychain group, expand it, or all signed apps will use the same keychain
elif keychain and any(item == wildcard_app_id for item in keychain):
keychain.clear()
for item in old_keychain:
keychain.append(f"{self.opts.team_id}.{item[item.index('.')+1:]}")
else:
supported_entitlements = [
self.__get_application_identifier_key(),
"com.apple.developer.team-identifier",
"com.apple.developer.healthkit",
"com.apple.developer.healthkit.access",
"com.apple.developer.homekit",
"com.apple.external-accessory.wireless-configuration",
"com.apple.security.application-groups",
"inter-app-audio",
"get-task-allow",
"keychain-access-groups",
self.__get_aps_environment_key(),
"com.apple.developer.icloud-container-development-container-identifiers",
"com.apple.developer.icloud-container-environment",
"com.apple.developer.icloud-container-identifiers",
"com.apple.developer.icloud-services",
"com.apple.developer.kernel.extended-virtual-addressing",
"com.apple.developer.networking.multipath",
"com.apple.developer.networking.networkextension",
"com.apple.developer.networking.vpn.api",
"com.apple.developer.networking.wifi-info",
"com.apple.developer.nfc.readersession.formats",
"com.apple.developer.siri",
"com.apple.developer.ubiquity-container-identifiers",
"com.apple.developer.ubiquity-kvstore-identifier",