diff --git a/CHANGELOG.md b/CHANGELOG.md index 962ac59f..4f31540d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -26,6 +26,7 @@ This project uses the changelog in accordance with [keepchangelog](http://keepac - Added support for timestamped comments in CLI via `rem`, `;`, `%` or `#` (@doegox) - Fixed watchdog trigger during `hw factory_reset` (@doegox) - Added PyInstaller support for CLI client (@augustozanellato) + - Parallelize mfkey32v2 processes called from CLI (@p-l-) ## [v2.0.0][2023-09-26] - Added `hw slot nick delete` and DELETE_SLOT_TAG_NICK (@doegox) diff --git a/software/script/chameleon_cli_unit.py b/software/script/chameleon_cli_unit.py index b551bb9c..b3c7e203 100644 --- a/software/script/chameleon_cli_unit.py +++ b/software/script/chameleon_cli_unit.py @@ -10,6 +10,7 @@ import serial.tools.list_ports import threading import struct +from multiprocessing import Pool, cpu_count from typing import Union from pathlib import Path from platform import uname @@ -1022,6 +1023,32 @@ def res_value(self, src_blk, src_type, src_key, dst_blk, dst_type, dst_key): print(f" - {CR}Restore fail.{C0}") +_KEY = re.compile("[a-fA-F0-9]{12}", flags=re.MULTILINE) + + +def _run_mfkey32v2(items): + output_str = subprocess.run( + [ + "mfkey32v2.exe" if sys.platform == "win32" else "./mfkey32v2", + items[0]["uid"], + items[0]["nt"], + items[0]["nr"], + items[0]["ar"], + items[1]["nt"], + items[1]["nr"], + items[1]["ar"], + ], + cwd=default_cwd, + capture_output=True, + check=True, + encoding="ascii", + ).stdout + sea_obj = _KEY.search(output_str) + if sea_obj is not None: + return sea_obj[0] + return None + + @hf_mf.command('elog') class HFMFELog(DeviceRequiredUnit): detection_log_size = 18 @@ -1044,30 +1071,19 @@ def decrypt_by_list(self, rs: list): msg3 = " key(s) found" n = 1 keys = set() - for i in range(len(rs)): - item0 = rs[i] - for j in range(i + 1, len(rs)): - item1 = rs[j] + with Pool(cpu_count()) as pool: + for key in pool.imap( + _run_mfkey32v2, + ( + (item0, rs[j]) + for i, item0 in enumerate(rs) + for j in range(i + 1, len(rs)) + ), + ): # TODO: if some keys already recovered, test them on item before running mfkey32 on item # TODO: if some keys already recovered, remove corresponding items - cmd_base = f"{item0['uid']} {item0['nt']} {item0['nr']} {item0['ar']}" - cmd_base += f" {item1['nt']} {item1['nr']} {item1['ar']}" - if sys.platform == "win32": - cmd_recover = f"mfkey32v2.exe {cmd_base}" - else: - cmd_recover = f"./mfkey32v2 {cmd_base}" - # print(cmd_recover) - # Found Key: [e899c526c5cd] - # subprocess.run(cmd_final, cwd=os.path.abspath("../bin/"), shell=True) - process = self.sub_process(cmd_recover) - # wait end - process.wait_process() - # get output - output_str = process.get_output_sync() - # print(output_str) - sea_obj = re.search(r"([a-fA-F0-9]{12})", output_str, flags=re.MULTILINE) - if sea_obj is not None: - keys.add(sea_obj[1]) + if key is not None: + keys.add(key) print(f"{msg1}{n}{msg2}{len(keys)}{msg3}\r", end="") n += 1 print()