-
Notifications
You must be signed in to change notification settings - Fork 64
/
Copy pathbase.py
681 lines (575 loc) · 30.5 KB
/
base.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
import binascii
import copy
import json
import platform
import plistlib
import shutil
import subprocess
import textwrap
from enum import Enum
from operator import itemgetter
from pathlib import Path
from termcolor2 import c as color
from Scripts import shared, utils
class Colors(Enum):
BLUE = "\u001b[36;1m"
RED = "\u001b[31;1m"
GREEN = "\u001b[32;1m"
RESET = "\u001b[0m"
def hexswap(input_hex: str):
hex_pairs = [input_hex[i : i + 2] for i in range(0, len(input_hex), 2)]
hex_rev = hex_pairs[::-1]
hex_str = "".join(["".join(x) for x in hex_rev])
return hex_str.upper()
def read_property(input_value: bytes, places: int):
return binascii.b2a_hex(input_value).decode()[:-places]
class BaseUSBMap:
def __init__(self):
self.utils = utils.Utils(f"USBToolBox {shared.VERSION}".strip())
self.controllers = None
self.json_path = shared.current_dir / Path("usb.json")
self.settings_path = shared.current_dir / Path("settings.json")
self.settings = (
json.load(self.settings_path.open()) if self.settings_path.exists() else {"show_friendly_types": False, "use_native": False, "add_comments_to_map": True, "auto_bind_companions": True}
)
self.controllers_historical = json.load(self.json_path.open("r")) if self.json_path.exists() else None
self.monu()
@staticmethod
def is_same_controller(controller_1, controller_2):
common_keys: set[str] = set(i for i in controller_1["identifiers"] if controller_1["identifiers"][i]) & set(i for i in controller_2["identifiers"] if controller_2["identifiers"][i])
if not common_keys:
# No way to tell
return False
for key in common_keys:
if key in ["instance_id", "location_id"]:
# With Thunderbolt controllers, on hotplugs the instance ID will change, but everything else will be the same.
# On macOS, location IDs can change all the time.
continue
elif len(common_keys) == 1 and key in ["pci_revision"]:
# Don't match solely by pci_revision.
return False
elif key == "location_paths":
# Some firmwares have broken ACPI where two or more devices have the same parent and same address, and Windows does not always show all
# Evident with Thunderbolt controllers
# We will be satisified if they have at least 1 in common
if not set(controller_1["identifiers"]["location_paths"]) & set(controller_2["identifiers"]["location_paths"]):
return False
else:
if not (controller_1["identifiers"][key] is not None and controller_2["identifiers"][key] is not None and controller_1["identifiers"][key] == controller_2["identifiers"][key]):
return False
return True
@staticmethod
def get_controller_from_list(original, controller_list):
for controller in controller_list:
if BaseUSBMap.is_same_controller(original, controller):
return controller
return None
@staticmethod
def merge_properties(old, new):
if not new:
return old
if not old:
return new
if isinstance(old, list):
retval = list(old)
retval.extend(set(new) - set(old))
return retval
elif isinstance(old, dict):
retval = dict(old)
for key in new:
retval[key] = BaseUSBMap.merge_properties(retval.get(key), new[key])
return retval
else:
return new
@staticmethod
def merge_controllers(base: list, new: list):
for controller in new:
base_controller = BaseUSBMap.get_controller_from_list(controller, base)
if not base_controller:
base.append(controller)
# Don't need to merge properties because there's no base controller
continue
for key in set(controller.keys()) - set(["ports"]): # Leave merging ports to merge_ports
base_controller[key] = BaseUSBMap.merge_properties(base_controller.get(key), controller[key])
BaseUSBMap.merge_ports(base, new)
@staticmethod
def merge_ports(base: list, new: list):
for controller in new:
base_controller = BaseUSBMap.get_controller_from_list(controller, base)
for port in controller["ports"]:
base_port = ([p for p in base_controller["ports"] if p["index"] == port["index"]] or [None])[0]
if not base_port:
base_controller["ports"].append(copy.deepcopy(port))
# Don't need to merge properties because there's no base port
continue
for key in set(port.keys()) - set(["devices"]): # Leave merging devices to merge_devices
base_port[key] = BaseUSBMap.merge_properties(base_port.get(key), port[key])
base_controller["ports"].sort(key=itemgetter("index"))
BaseUSBMap.merge_devices(base, new)
@staticmethod
def recursive_merge_devices(base: list, new: list):
for i in new:
if not i or i in base:
continue
elif isinstance(i, str):
base.append(i)
elif i.get("error"):
continue
elif i["name"] not in [hub.get("name") for hub in base if isinstance(hub, dict)]:
base.append(i)
else:
old_hub = [hub for hub in base if isinstance(hub, dict) and hub.get("name") == i["name"]][0]
BaseUSBMap.recursive_merge_devices(old_hub["devices"], i["devices"])
for i in list(base):
if not i or i.get("error"):
base.remove(i)
@staticmethod
def merge_devices(base: list, new: list):
for controller in base:
new_controller = BaseUSBMap.get_controller_from_list(controller, new)
if new_controller:
for port in controller["ports"]:
BaseUSBMap.recursive_merge_devices(port["devices"], new_controller["ports"][controller["ports"].index(port)]["devices"])
def get_controllers(self):
raise NotImplementedError
def update_devices(self):
raise NotImplementedError
def controller_to_str(self, controller):
return f"{controller['name']} | {shared.USBControllerTypes(controller['class'])}"
def port_to_str(self, port):
if port["type"] is not None:
port_type = shared.USBPhysicalPortTypes(port["type"]) if self.settings["show_friendly_types"] else shared.USBPhysicalPortTypes(port["type"]).value
elif port["guessed"] is not None:
port_type = (str(shared.USBPhysicalPortTypes(port["guessed"])) if self.settings["show_friendly_types"] else str(shared.USBPhysicalPortTypes(port["guessed"]).value)) + " (guessed)"
else:
port_type = "unknown"
return f"{port['name']} | {shared.USBDeviceSpeeds(port['class'])} | " + (str(port_type) if self.settings["show_friendly_types"] else f"Type {port_type}")
def print_controllers(self, controllers, colored=False):
if not controllers:
print("Empty.")
return
for controller in controllers:
if colored:
print(color(self.controller_to_str(controller) + f" | {len(controller['ports'])} ports"))
else:
print(self.controller_to_str(controller) + f" | {len(controller['ports'])} ports")
for port in controller["ports"]:
if not colored:
print(" " + self.port_to_str(port))
elif port["devices"]:
print(" " + color(self.port_to_str(port)).green.bold)
elif (
self.get_controller_from_list(controller, self.controllers_historical)
and [i for i in self.get_controller_from_list(controller, self.controllers_historical)["ports"] if i["index"] == port["index"]][0]["devices"]
):
print(" " + color(self.port_to_str(port)).cyan.bold)
else:
print(" " + self.port_to_str(port))
if port["comment"]:
print(" " + port["comment"])
for device in port["devices"]:
self.print_devices(device)
def print_devices(self, device, indentation=" "):
if not device:
device = "Enumerating..."
if isinstance(device, str):
print(f"{indentation}- {device}")
elif device.get("error", False):
print(f"{indentation}- {device['error'] if isinstance(device['error'], str) else 'Device connected to port errored.'} Please unplug or connect a different device.")
else:
print(f"{indentation}- {device['name'].strip()} - operating at {shared.USBDeviceSpeeds(device['speed'])}")
for i in device["devices"]:
self.print_devices(i, indentation + " ")
def discover_ports(self):
self.utils.head("Port Discovery")
print()
dont_refresh = False
if not self.controllers:
print("\nGetting controllers...")
self.get_controllers()
dont_refresh = True
while True:
# os.system("cls" if os.name == "nt" else "clear")
self.utils.head("Port Discovery")
print()
if dont_refresh:
dont_refresh = False
else:
self.update_devices()
self.print_controllers(self.controllers, colored=True)
self.dump_historical()
print("\nB. Back\n")
do_quit = self.utils.grab("Waiting 5 seconds: ", timeout=5)
if str(do_quit).lower() == "b":
break
def print_historical(self):
utils.TUIMenu("Print Historical (DEBUG)", "Select an option: ", in_between=lambda: self.print_controllers(self.controllers_historical), loop=True).start()
def dump_historical(self):
if self.controllers_historical:
json.dump(self.controllers_historical, self.json_path.open("w"), indent=4, sort_keys=True)
elif self.controllers_historical == []:
self.remove_historical()
def remove_historical(self):
if self.controllers_historical or self.controllers_historical == []:
self.controllers = None
self.controllers_historical = None
self.json_path.unlink()
def dump_settings(self):
json.dump(self.settings, self.settings_path.open("w"), indent=4, sort_keys=True)
def on_quit(self):
self.dump_historical()
def print_types(self):
in_between = [f"{i}: {i.value}" for i in shared.USBPhysicalPortTypes] + [
"",
textwrap.dedent(
"""\
The difference between connector types 9 and 10 is if you reverse the plug and the devices are connected to the same ports as before, they have a switch (type 9).
If not, and they are connected to different ports, they do not have a switch (type 10)."""
),
"",
"For more information and pictures, go to https://github.com/USBToolBox/tool/blob/master/TYPES.md.",
]
utils.TUIMenu("USB Types", "Select an option: ", in_between=in_between).start()
def get_companion_port(self, port):
if not port.get("companion_info"):
return None
companion_info = port["companion_info"]
if not companion_info["hub"] or not companion_info["port"]:
return None
hub = [i for i in self.controllers_historical if i.get("hub_name") == companion_info["hub"]]
if hub:
port = [i for i in hub[0]["ports"] if i["index"] == companion_info["port"]]
if port:
return port[0]
return None
def select_ports(self):
if not self.controllers_historical:
utils.TUIMenu("Select Ports and Build Kext", "Select an option: ", in_between=["No ports! Use the discovery mode."], loop=True).start()
return
selection_index = 1
by_port = []
for controller in self.controllers_historical:
controller["selected_count"] = 0
for port in controller["ports"]:
if "selected" not in port:
port["selected"] = bool(port["devices"])
port["selected"] = port["selected"] or (bool(self.get_companion_port(port)["devices"]) if self.get_companion_port(port) else False)
controller["selected_count"] += 1 if port["selected"] else 0
port["selection_index"] = selection_index
selection_index += 1
by_port.append(port)
while True:
self.dump_historical()
for controller in self.controllers_historical:
controller["selected_count"] = sum(1 if port["selected"] else 0 for port in controller["ports"])
utils.header("Select Ports and Build Kext")
print()
for controller in self.controllers_historical:
port_count_str = f"{controller['selected_count']}/{len(controller['ports'])}"
port_count_str = color(port_count_str).red if controller["selected_count"] > 15 else color(port_count_str).green
print(self.controller_to_str(controller) + f" | {port_count_str} ports")
for port in controller["ports"]:
port_info = f"[{'#' if port['selected'] else ' '}] {port['selection_index']}.{(len(str(selection_index)) - len(str(port['selection_index'])) + 1) * ' ' }" + self.port_to_str(port)
companion = self.get_companion_port(port)
if companion:
port_info += f" | Companion to {companion['selection_index']}"
if port["selected"]:
print(color(port_info).green.bold)
else:
print(port_info)
if port["comment"]:
print(
len(f"[{'#' if port['selected'] else ' '}] {port['selection_index']}.{(len(str(selection_index)) - len(str(port['selection_index'])) + 1) * ' ' }") * " "
+ color(port["comment"]).blue.bold
)
for device in port["devices"]:
self.print_devices(device, indentation=" " + len(str(selection_index)) * " " * 2)
print()
print(f"Binding companions is currently {color('on').green if self.settings['auto_bind_companions'] else color('off').red}.\n")
print(
textwrap.dedent(
f"""\
K. Build {'USBMap' if self.settings['use_native'] else 'UTBMap'}.kext
A. Select All
N. Select None
P. Enable All Populated Ports
D. Disable All Empty Ports
T. Show Types
B. Back
- Select ports to toggle with comma-delimited lists (eg. 1,2,3,4,5)
- Change types using this formula T:1,2,3,4,5:t where t is the type
- Set custom names using this formula C:1:Name - Name = None to clear"""
)
)
output = input("Select an option: ")
if not output:
continue
elif output.upper() == "B":
break
elif output.upper() == "K":
if not self.validate_selections():
continue
self.build_kext()
continue
elif output.upper() in ("N", "A"):
for port in by_port:
port["selected"] = output.upper() == "A"
elif output.upper() == "P":
for port in by_port:
if port["devices"] or (self.get_companion_port(port)["devices"] if self.get_companion_port(port) else False):
port["selected"] = True
elif output.upper() == "D":
for port in by_port:
if not port["devices"] and not (self.get_companion_port(port)["devices"] if self.get_companion_port(port) else False):
port["selected"] = False
elif output.upper() == "T":
self.print_types()
continue
elif output[0].upper() == "T":
# We should have a type
if len(output.split(":")) != 3:
continue
try:
port_nums, port_type = output.split(":")[1:]
port_nums = port_nums.replace(" ", "").split(",")
port_type = shared.USBPhysicalPortTypes(int(port_type))
for port_num in list(port_nums):
if port_num not in port_nums:
continue
port_num = int(port_num) - 1
if port_num not in range(len(by_port)):
continue
companion = self.get_companion_port(by_port[port_num])
if self.settings["auto_bind_companions"] and companion:
companion["type"] = port_type
if str(companion["selection_index"]) in port_nums:
port_nums.remove(str(companion["selection_index"]))
by_port[port_num]["type"] = port_type
except ValueError:
continue
elif output[0].upper() == "C":
# We should have a new name
if len(output.split(":")) < 2:
continue
try:
port_nums = output.split(":")[1].replace(" ", "").split(",")
port_comment = output.split(":", 2)[2:]
for port_num in list(port_nums):
if port_num not in port_nums:
continue
port_num = int(port_num) - 1
if port_num not in range(len(by_port)):
continue
by_port[port_num]["comment"] = port_comment[0] if port_comment else None
except ValueError:
continue
else:
try:
port_nums = output.replace(" ", "").split(",")
for port_num in list(port_nums):
if port_num not in port_nums:
continue
port_num = int(port_num) - 1
if port_num not in range(len(by_port)):
continue
companion = self.get_companion_port(by_port[port_num])
if self.settings["auto_bind_companions"] and companion:
companion["selected"] = not by_port[port_num]["selected"]
if str(companion["selection_index"]) in port_nums:
port_nums.remove(str(companion["selection_index"]))
by_port[port_num]["selected"] = not by_port[port_num]["selected"]
except ValueError:
continue
def print_errors(self, errors):
if not errors:
return True
utils.TUIMenu("Selection Validation", "Select an option: ", in_between=errors, loop=True).start()
return False
def validate_selections(self):
errors = []
if not any(any(p["selected"] for p in c["ports"]) for c in self.controllers_historical):
utils.TUIMenu("Selection Validation", "Select an option: ", in_between=["No ports are selected! Select some ports."], loop=True).start()
return False
for controller in self.controllers_historical:
for port in controller["ports"]:
if not port["selected"]:
continue
if port["type"] is None and port["guessed"] is None:
errors.append(f"Port {port['selection_index']} is missing a connector type!")
return self.print_errors(errors)
def check_unique(self, identifier, predicate, controller):
if predicate(controller):
values = [identifier(c) for c in self.controllers_historical if predicate(c)]
return values.count(identifier(controller)) == 1
else:
return False
def choose_matching_key(self, controller):
if "bus_number" in controller["identifiers"]:
# M1 Macs
return {"IOPropertyMatch": {"bus-number": binascii.a2b_hex(hexswap(hex(controller["identifiers"]["bus_number"])[2:].zfill(8)))}}
elif not self.settings["use_native"] and self.check_unique(lambda c: c["identifiers"]["acpi_path"].rpartition(".")[2], lambda c: "acpi_path" in c["identifiers"], controller):
# Unique ACPI name
# Disable if using native because we don't know if it'll conflict
# TODO: Check this maybe?
shared.debug(f"Using ACPI path: {controller['identifiers']['acpi_path']}")
return {"IONameMatch": controller["identifiers"]["acpi_path"].rpartition(".")[2]}
elif "bdf" in controller["identifiers"]:
# Use bus-device-function
return {"IOPropertyMatch": {"pcidebug": ":".join([str(i) for i in controller["identifiers"]["bdf"]])}}
elif self.check_unique(lambda c: c["identifiers"]["path"], lambda c: "path" in c["identifiers"], controller):
# Use IORegistry path
return {"IOPathMatch": controller["identifiers"]["path"]}
elif self.check_unique(lambda c: c["identifiers"]["pci_id"], lambda c: "pci_id" in c["identifiers"], controller):
# Use PCI ID
pci_id: list[str] = controller["identifiers"]["pci_id"]
return {"IOPCIPrimaryMatch": f"0x{pci_id[1]}{pci_id[0]}"} | ({"IOPCISecondaryMatch": f"0x{pci_id[3]}{pci_id[2]}"} if len(pci_id) > 2 else {})
else:
raise RuntimeError("No matching key available")
def build_kext(self):
empty_controllers = [c for c in self.controllers_historical if not any(p["selected"] for p in c["ports"])]
response = None
if empty_controllers:
empty_menu = utils.TUIMenu(
"Selection Validation",
"Select an option: ",
in_between=["The following controllers have no enabled ports:", ""]
+ [controller["name"] for controller in empty_controllers]
+ ["Select whether to ignore these controllers and exclude them from the map, or disable all ports on these controllers."],
add_quit=False,
return_number=True,
)
empty_menu.add_menu_option("Ignore", key="I")
empty_menu.add_menu_option("Disable", key="D")
response = empty_menu.start()
model_identifier = None
if self.settings["use_native"]:
if platform.system() == "Darwin":
model_identifier = plistlib.loads(subprocess.run("system_profiler -detailLevel mini -xml SPHardwareDataType".split(), stdout=subprocess.PIPE, stderr=subprocess.STDOUT).stdout.strip())[
0
]["_items"][0]["machine_model"]
else:
model_menu = utils.TUIOnlyPrint(
"Enter Model Identifier",
"Enter the model identifier: ",
[
"You are seeing this as you have selected to use AppleUSBHostController. Model identifier autodetection is unavailable as you are not on macOS.",
"Please enter the model identifier of the target system below. You can find it in System Information or with 'system_profiler -detailLevel mini SPHardwareDataType'.",
],
).start()
model_identifier = model_menu.strip()
ignore = response == "I"
template = plistlib.load((shared.resource_dir / Path("Info.plist")).open("rb"))
menu = utils.TUIMenu("Building USBMap", "Select an option: ")
menu.head()
print("Generating Info.plist...")
for controller in self.controllers_historical:
if not any(i["selected"] for i in controller["ports"]) and ignore:
continue
# FIXME: ensure unique
if controller["identifiers"].get("acpi_path"):
if self.check_unique(lambda c: c["identifiers"]["acpi_path"].rpartition(".")[2], lambda c: "acpi_path" in c["identifiers"], controller):
personality_name: str = controller["identifiers"]["acpi_path"].rpartition(".")[2]
else:
personality_name: str = controller["identifiers"]["acpi_path"][1:] # Strip leading \
elif controller["identifiers"].get("bdf"):
personality_name: str = ":".join([str(i) for i in controller["identifiers"]["bdf"]])
else:
personality_name: str = controller["name"]
if self.settings["use_native"]:
personality = {
"CFBundleIdentifier": "com.apple.driver.AppleUSBHostMergeProperties",
"IOClass": "AppleUSBHostMergeProperties",
"IOProviderClass": "AppleUSBHostController",
"IOParentMatch": self.choose_matching_key(controller),
"model": model_identifier,
}
else:
personality = {
"CFBundleIdentifier": "com.dhinakg.USBToolBox.kext",
"IOClass": "USBToolBox",
"IOProviderClass": "IOPCIDevice",
"IOMatchCategory": "USBToolBox",
} | self.choose_matching_key(
controller
) # type: ignore
personality["IOProviderMergeProperties"] = {"ports": {}, "port-count": None}
port_name_index = {}
highest_index = 0
for port in controller["ports"]:
if not port["selected"]:
continue
if port["index"] > highest_index:
highest_index = port["index"]
if controller["class"] == shared.USBControllerTypes.XHCI and port["class"] == shared.USBDeviceSpeeds.SuperSpeed:
prefix = "SS"
elif controller["class"] == shared.USBControllerTypes.XHCI and port["class"] == shared.USBDeviceSpeeds.HighSpeed:
prefix = "HS"
else:
prefix = "PRT"
port_index = port_name_index.setdefault(prefix, 1)
port_name = prefix + str(port_index).zfill(4 - len(prefix))
port_name_index[prefix] += 1
personality["IOProviderMergeProperties"]["ports"][port_name] = {
"port": binascii.a2b_hex(hexswap(hex(port["index"])[2:].zfill(8))),
"UsbConnector": port["type"] or port["guessed"],
}
if self.settings["add_comments_to_map"] and port["comment"]:
personality["IOProviderMergeProperties"]["ports"][port_name]["#comment"] = port["comment"]
personality["IOProviderMergeProperties"]["port-count"] = binascii.a2b_hex(hexswap(hex(highest_index)[2:].zfill(8)))
template["IOKitPersonalities"][personality_name] = personality
if not self.settings["use_native"]:
template["OSBundleLibraries"] = {"com.dhinakg.USBToolBox.kext": "1.0.0"}
write_path = shared.current_dir / (Path("USBMap.kext") if self.settings["use_native"] else Path("UTBMap.kext"))
if write_path.exists():
print("Removing existing kext...")
shutil.rmtree(write_path)
print("Writing kext and Info.plist...")
(write_path / Path("Contents")).mkdir(parents=True)
plistlib.dump(template, (write_path / Path("Contents/Info.plist")).open("wb"), sort_keys=True)
print(f"Done. Saved to {write_path.resolve()}.\n")
menu.print_options()
menu.select()
return True
def change_settings(self):
def functionify(func):
return lambda *args, **kwargs: lambda: func(*args, **kwargs)
@functionify
def color_status(name, variable):
return f"{name}: {color('Enabled').green if self.settings[variable] else color('Disabled').red}"
@functionify
def toggle_setting(variable):
self.settings[variable] = not self.settings[variable]
def combination(name, variable):
return color_status(name, variable), toggle_setting(variable)
menu = utils.TUIMenu("Change Settings", "Toggle a setting: ", loop=True)
for i in [
["T", *combination("Show Friendly Types", "show_friendly_types"), ["Show friendly types (ie. 'USB 3 Type A') instead of numbers."]],
["N", *combination("Use Native Classes", "use_native"), ["Use native Apple classes (AppleUSBHostMergeProperties) instead of the USBToolBox kext."]],
["A", *combination("Add Comments to Map", "add_comments_to_map"), ["Add port comments inside the map."]],
[
"C",
*combination("Bind Companions", "auto_bind_companions"),
["Tie companion ports together. If one companion is enabled/disable/port type changed, the other companion will also be affected."],
],
]:
menu.add_menu_option(name=i[1], function=i[2], key=i[0], description=i[3] if len(i) == 4 else None)
menu.start()
self.dump_settings()
def monu(self):
response = None
while not (response and response == utils.TUIMenu.EXIT_MENU):
in_between = [("Saved Data: {}" + Colors.RESET.value).format(Colors.GREEN.value + "Loaded" if self.json_path.exists() else (Colors.RED.value + "None"))]
menu = utils.TUIMenu(f"USBToolBox {shared.VERSION}", "Select an option: ", in_between=in_between, top_level=True)
menu_options = [
# ["H", "Print Historical", self.print_historical],
["D", "Discover Ports", self.discover_ports],
["S", "Select Ports and Build Kext", self.select_ports],
["C", "Change Settings", self.change_settings],
]
if self.json_path.exists():
menu_options.insert(0, ["P", "Delete Saved USB Data", self.remove_historical])
for i in menu_options:
menu.add_menu_option(i[1], None, i[2], i[0])
response = menu.start()
self.on_quit()
self.utils.custom_quit()