From 7415c962a6d1ae9a76974c428a6c3977d9e5beb5 Mon Sep 17 00:00:00 2001 From: M1nd3r Date: Fri, 22 Nov 2024 09:25:49 +0100 Subject: [PATCH] wip --- core/src/apps/debug/__init__.py | 27 ++++++++++++++++--- tests/click_tests/test_autolock.py | 2 +- tests/click_tests/test_lock.py | 8 ++++-- tests/device_handler.py | 23 +++++++++++++--- tests/upgrade_tests/test_firmware_upgrades.py | 21 +++++++++++---- .../test_passphrase_consistency.py | 27 ++++++++++--------- 6 files changed, 81 insertions(+), 27 deletions(-) diff --git a/core/src/apps/debug/__init__.py b/core/src/apps/debug/__init__.py index 027b62a0fb..9357bd100e 100644 --- a/core/src/apps/debug/__init__.py +++ b/core/src/apps/debug/__init__.py @@ -208,17 +208,20 @@ if __debug__: msg: DebugLinkDecision, ) -> DebugLinkState | None: from trezor import ui, workflow - + log.debug(__name__, "decision 1") workflow.idle_timer.touch() x = msg.x # local_cache_attribute y = msg.y # local_cache_attribute - + log.debug(__name__, "decision 2") await wait_until_layout_is_running() assert isinstance(ui.CURRENT_LAYOUT, ui.Layout) layout_change_box.clear() + log.debug(__name__, "decision 3") try: + log.debug(__name__, "decision try") + # click on specific coordinates, with possible hold if x is not None and y is not None: await _layout_click(x, y, msg.hold_ms or 0) @@ -230,7 +233,13 @@ if __debug__: elif msg.button is not None: await _layout_event(msg.button) elif msg.input is not None: - ui.CURRENT_LAYOUT._emit_message(msg.input) + log.debug(__name__, "decision input") + try: + ui.CURRENT_LAYOUT._emit_message(msg.input) + except Exception as e: + print(type(e)) + log.debug(__name__, "decision input end") + else: raise RuntimeError("Invalid DebugLinkDecision message") @@ -244,6 +253,7 @@ if __debug__: # If no exception was raised, the layout did not shut down. That means that it # just updated itself. The update is already live for the caller to retrieve. + log.debug(__name__, "decision end") def _state( thp_pairing_code_entry_code: int | None = None, @@ -431,22 +441,31 @@ if __debug__: ctx.iface.iface_num(), msg_type, ) - + log.debug(__name__, "here 1") if msg.type not in WORKFLOW_HANDLERS: + log.debug(__name__, "here a") + await ctx.write(message_handler.unexpected_message()) continue elif req_type is None: + log.debug(__name__, "here b") + # Message type is in workflow handlers but not in protobuf # definitions. This indicates a deprecated message. # We put a no-op handler for those messages. # XXX return a Failure here? await ctx.write(Success()) continue + log.debug(__name__, "here 3") req_msg = message_handler.wrap_protobuf_load(msg.data, req_type) try: + log.debug(__name__, "here 4") + res_msg = await WORKFLOW_HANDLERS[msg.type](req_msg) + log.debug(__name__, "here 5") + except Exception as exc: # Log and ignore, never die. log.exception(__name__, exc) diff --git a/tests/click_tests/test_autolock.py b/tests/click_tests/test_autolock.py index e841d60f0c..12669bb860 100644 --- a/tests/click_tests/test_autolock.py +++ b/tests/click_tests/test_autolock.py @@ -59,7 +59,7 @@ CENTER_BUTTON = buttons.grid35(1, 2) def set_autolock_delay(device_handler: "BackgroundDeviceHandler", delay_ms: int): debug = device_handler.debuglink() - device_handler.run(device.apply_settings, auto_lock_delay_ms=delay_ms) # type: ignore + device_handler.run_with_session(device.apply_settings, auto_lock_delay_ms=delay_ms) # type: ignore assert "PinKeyboard" in debug.read_layout().all_components() diff --git a/tests/click_tests/test_lock.py b/tests/click_tests/test_lock.py index 4b719885e6..b3656dfd29 100644 --- a/tests/click_tests/test_lock.py +++ b/tests/click_tests/test_lock.py @@ -19,7 +19,7 @@ from typing import TYPE_CHECKING import pytest -from trezorlib import models +from trezorlib import messages, models from trezorlib.debuglink import LayoutType from .. import buttons, common @@ -34,6 +34,9 @@ PIN4 = "1234" @pytest.mark.setup_client(pin=PIN4) def test_hold_to_lock(device_handler: "BackgroundDeviceHandler"): debug = device_handler.debuglink() + session = device_handler.client.get_management_session() + session.call(messages.LockDevice()) + session.refresh_features() short_duration = { models.T1B1: 500, @@ -59,9 +62,10 @@ def test_hold_to_lock(device_handler: "BackgroundDeviceHandler"): assert device_handler.features().unlocked is False # unlock with message - device_handler.run(common.get_test_address) + device_handler.run_with_session(common.get_test_address) assert "PinKeyboard" in debug.read_layout().all_components() + time.sleep(10) debug.input("1234") assert device_handler.result() diff --git a/tests/device_handler.py b/tests/device_handler.py index 45ec1df9f7..a380ea80a3 100644 --- a/tests/device_handler.py +++ b/tests/device_handler.py @@ -33,6 +33,9 @@ class NullUI: else: raise NotImplementedError("NullUI should not be used with T1") + def debug_callback_button(self, session: Any, msg: Any) -> Any: + raise RuntimeError("unexpected call to a fake debuglink") + class BackgroundDeviceHandler: _pool = ThreadPoolExecutor() @@ -60,13 +63,28 @@ class BackgroundDeviceHandler: with self.debuglink().wait_for_layout_change(): self.task = self._pool.submit(function, self.client, *args, **kwargs) + def run_with_session( + self, function: Callable[..., Any], *args: Any, **kwargs: Any + ) -> None: + """Runs some function that interacts with a device. + + Makes sure the UI is updated before returning. + """ + if self.task is not None: + raise RuntimeError("Wait for previous task first") + + # wait for the first UI change triggered by the task running in the background + with self.debuglink().wait_for_layout_change(): + session = self.client.get_session() + self.task = self._pool.submit(function, session, *args, **kwargs) + def kill_task(self) -> None: if self.task is not None: # Force close the client, which should raise an exception in a client # waiting on IO. Does not work over Bridge, because bridge doesn't have # a close() method. - while self.client.session_counter > 0: - self.client.close() + # while self.client.session_counter > 0: + # self.client.close() try: self.task.result(timeout=1) except Exception: @@ -90,7 +108,6 @@ class BackgroundDeviceHandler: def features(self) -> "Features": if self.task is not None: raise RuntimeError("Cannot query features while task is running") - self.client.init_device() return self.client.features def debuglink(self) -> "DebugLink": diff --git a/tests/upgrade_tests/test_firmware_upgrades.py b/tests/upgrade_tests/test_firmware_upgrades.py index bafe67f511..e742e25ce5 100644 --- a/tests/upgrade_tests/test_firmware_upgrades.py +++ b/tests/upgrade_tests/test_firmware_upgrades.py @@ -21,6 +21,7 @@ import pytest from shamir_mnemonic import shamir from trezorlib import btc, debuglink, device, exceptions, fido, models +from trezorlib.debuglink import SessionDebugWrapper as Session from trezorlib.messages import ( ApplySettings, BackupAvailability, @@ -57,15 +58,19 @@ STRENGTH = 128 @for_all() def test_upgrade_load(gen: str, tag: str) -> None: def asserts(client: "Client"): + client.refresh_features() assert not client.features.pin_protection assert not client.features.passphrase_protection assert client.features.initialized assert client.features.label == LABEL - assert btc.get_address(client, "Bitcoin", PATH) == ADDRESS + assert ( + btc.get_address(client.get_session(passphrase=""), "Bitcoin", PATH) + == ADDRESS + ) with EmulatorWrapper(gen, tag) as emu: debuglink.load_device_by_mnemonic( - emu.client, + emu.client.get_management_session(), mnemonic=MNEMONIC, pin="", passphrase_protection=False, @@ -164,11 +169,14 @@ def test_upgrade_wipe_code(gen: str, tag: str): assert client.features.initialized assert client.features.label == LABEL client.use_pin_sequence([PIN]) - assert btc.get_address(client, "Bitcoin", PATH) == ADDRESS + assert ( + btc.get_address(client.get_session(passphrase=""), "Bitcoin", PATH) + == ADDRESS + ) with EmulatorWrapper(gen, tag) as emu: debuglink.load_device_by_mnemonic( - emu.client, + emu.client.get_management_session(), mnemonic=MNEMONIC, pin=PIN, passphrase_protection=False, @@ -177,7 +185,10 @@ def test_upgrade_wipe_code(gen: str, tag: str): # Set wipe code. emu.client.use_pin_sequence([PIN, WIPE_CODE, WIPE_CODE]) - device.change_wipe_code(emu.client) + session = Session(emu.client.get_management_session()) + session.refresh_features() + raise Exception("client", str(emu.client.pin_callback)) + device.change_wipe_code(session) device_id = emu.client.features.device_id asserts(emu.client) diff --git a/tests/upgrade_tests/test_passphrase_consistency.py b/tests/upgrade_tests/test_passphrase_consistency.py index 67a4c406fb..9a778f6055 100644 --- a/tests/upgrade_tests/test_passphrase_consistency.py +++ b/tests/upgrade_tests/test_passphrase_consistency.py @@ -21,6 +21,7 @@ import pytest from trezorlib import btc, device, mapping, messages, models, protobuf from trezorlib._internal.emulator import Emulator from trezorlib.tools import parse_path +from trezorlib.debuglink import SessionDebugWrapper as Session from ..emulators import EmulatorWrapper from . import for_all @@ -47,11 +48,12 @@ def emulator(gen: str, tag: str) -> Iterator[Emulator]: with EmulatorWrapper(gen, tag) as emu: # set up a passphrase-protected device device.reset( - emu.client, + emu.client.get_management_session(), pin_protection=False, skip_backup=True, ) - resp = emu.client.call( + emu.client = emu.client.get_new_client() + resp = emu.client.get_management_session().call( ApplySettingsCompat(use_passphrase=True, passphrase_source=SOURCE_HOST) ) assert isinstance(resp, messages.Success) @@ -87,11 +89,11 @@ def test_passphrase_works(emulator: Emulator): messages.ButtonRequest, messages.Address, ] - - with emulator.client: + emu_session = emulator.client.get_session(passphrase="") + with emulator.client, Session(emu_session) as session: emulator.client.use_passphrase("TREZOR") - emulator.client.set_expected_responses(expected_responses) - btc.get_address(emulator.client, "Testnet", parse_path("44h/1h/0h/0/0")) + session.set_expected_responses(expected_responses) + btc.get_address(session, "Testnet", parse_path("44h/1h/0h/0/0")) @for_all( @@ -131,13 +133,14 @@ def test_init_device(emulator: Emulator): messages.Address, ] - with emulator.client: + emu_session = emulator.client.get_session(passphrase="") + with emulator.client, Session(emu_session) as session: emulator.client.use_passphrase("TREZOR") - emulator.client.set_expected_responses(expected_responses) + session.set_expected_responses(expected_responses) - btc.get_address(emulator.client, "Testnet", parse_path("44h/1h/0h/0/0")) + btc.get_address(session, "Testnet", parse_path("44h/1h/0h/0/0")) # in TT < 2.3.0 session_id will only be available after PassphraseStateRequest - session_id = emulator.client.session_id - emulator.client.init_device() + session_id = emulator.client.features.session_id + # emulator.client.init_device() btc.get_address(emulator.client, "Testnet", parse_path("44h/1h/0h/0/0")) - assert session_id == emulator.client.session_id + assert session_id == emulator.client.features.session_id