2023-08-11 15:57:32 +00:00
|
|
|
import re
|
2022-10-25 10:46:37 +00:00
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
|
2023-05-04 12:16:40 +00:00
|
|
|
from shamir_mnemonic import shamir # type: ignore
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2024-09-02 11:01:16 +00:00
|
|
|
from trezorlib.debuglink import LayoutType
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
from .. import buttons
|
2023-08-11 15:57:32 +00:00
|
|
|
from .. import translations as TR
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from trezorlib.debuglink import DebugLink
|
|
|
|
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2023-05-04 12:16:40 +00:00
|
|
|
def confirm_new_wallet(debug: "DebugLink") -> None:
|
2024-11-14 14:11:03 +00:00
|
|
|
assert debug.read_layout().title() == TR.reset__title_create_wallet
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.OK)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
|
|
|
debug.click(buttons.TAP_TO_CONFIRM)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right()
|
|
|
|
debug.press_right()
|
2024-11-14 14:11:03 +00:00
|
|
|
assert (
|
|
|
|
TR.backup__new_wallet_successfully_created in debug.read_layout().text_content()
|
|
|
|
or TR.backup__new_wallet_created in debug.read_layout().text_content()
|
2023-11-21 15:55:46 +00:00
|
|
|
)
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type is LayoutType.Delizia:
|
2023-11-21 15:55:46 +00:00
|
|
|
debug.swipe_up()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
|
2023-08-11 15:57:32 +00:00
|
|
|
def confirm_read(debug: "DebugLink", middle_r: bool = False) -> None:
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.OK)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2023-08-11 15:57:32 +00:00
|
|
|
page_count = debug.read_layout().page_count()
|
|
|
|
if page_count > 1:
|
|
|
|
for _ in range(page_count - 1):
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right()
|
2023-06-26 08:50:28 +00:00
|
|
|
if middle_r:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_middle()
|
2023-05-12 09:19:35 +00:00
|
|
|
else:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
|
2024-06-04 10:47:09 +00:00
|
|
|
def cancel_backup(
|
|
|
|
debug: "DebugLink", middle_r: bool = False, confirm: bool = False
|
|
|
|
) -> None:
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.CANCEL)
|
|
|
|
debug.click(buttons.CANCEL)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.CORNER_BUTTON)
|
|
|
|
debug.click(buttons.VERTICAL_MENU[0])
|
2024-06-04 10:47:09 +00:00
|
|
|
if confirm:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2024-06-04 10:47:09 +00:00
|
|
|
debug.click(buttons.TAP_TO_CONFIRM)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_left()
|
|
|
|
debug.press_left()
|
2024-04-11 14:21:17 +00:00
|
|
|
|
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
def set_selection(debug: "DebugLink", button: tuple[int, int], diff: int) -> None:
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type in (LayoutType.Bolt, LayoutType.Delizia):
|
2023-05-12 09:19:35 +00:00
|
|
|
assert "NumberInputDialog" in debug.read_layout().all_components()
|
|
|
|
for _ in range(diff):
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(button)
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.OK)
|
2024-05-27 22:19:01 +00:00
|
|
|
else:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2023-05-12 09:19:35 +00:00
|
|
|
layout = debug.read_layout()
|
2024-11-14 14:11:03 +00:00
|
|
|
if (
|
|
|
|
layout.title()
|
|
|
|
in TR.reset__title_number_of_shares + TR.words__title_threshold
|
|
|
|
):
|
2023-05-12 09:19:35 +00:00
|
|
|
# Special info screens
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.press_right()
|
2023-05-12 09:19:35 +00:00
|
|
|
assert "NumberInput" in layout.all_components()
|
2024-05-27 22:19:01 +00:00
|
|
|
if button == buttons.reset_minus(debug.model.internal_name):
|
2023-05-12 09:19:35 +00:00
|
|
|
for _ in range(diff):
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_left()
|
2023-05-12 09:19:35 +00:00
|
|
|
else:
|
|
|
|
for _ in range(diff):
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right()
|
|
|
|
debug.press_middle()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
|
2024-04-23 10:26:46 +00:00
|
|
|
def read_words(debug: "DebugLink", do_htc: bool = True) -> list[str]:
|
2022-10-25 10:46:37 +00:00
|
|
|
words: list[str] = []
|
2023-06-26 08:50:28 +00:00
|
|
|
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type is LayoutType.Caesar:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right()
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2023-05-12 09:19:35 +00:00
|
|
|
|
|
|
|
# Swiping through all the pages and loading the words
|
2023-08-11 15:57:32 +00:00
|
|
|
layout = debug.read_layout()
|
2023-06-26 08:50:28 +00:00
|
|
|
for _ in range(layout.page_count() - 1):
|
|
|
|
words.extend(layout.seed_words())
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.swipe_up()
|
2023-05-04 12:16:40 +00:00
|
|
|
assert layout is not None
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type in (LayoutType.Bolt, LayoutType.Delizia):
|
2023-05-12 09:19:35 +00:00
|
|
|
words.extend(layout.seed_words())
|
2022-10-25 10:46:37 +00:00
|
|
|
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2024-05-27 22:19:01 +00:00
|
|
|
|
2023-05-04 12:16:40 +00:00
|
|
|
# There is hold-to-confirm button
|
|
|
|
if do_htc:
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.click(buttons.OK, hold_ms=1500)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2023-11-21 15:55:46 +00:00
|
|
|
debug.click(buttons.TAP_TO_CONFIRM, hold_ms=1500)
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.press_right(hold_ms=1200)
|
2023-05-04 12:16:40 +00:00
|
|
|
else:
|
|
|
|
# It would take a very long time to test 16-of-16 with doing 1500 ms HTC after
|
|
|
|
# each word set
|
|
|
|
debug.press_yes()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
return words
|
|
|
|
|
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
def confirm_words(debug: "DebugLink", words: list[str]) -> None:
|
2025-01-17 12:04:11 +00:00
|
|
|
if debug.layout_type is LayoutType.Delizia:
|
2023-11-21 12:51:59 +00:00
|
|
|
debug.swipe_up()
|
2024-05-27 22:19:01 +00:00
|
|
|
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.read_layout()
|
2025-01-08 13:45:24 +00:00
|
|
|
if debug.layout_type is LayoutType.Bolt:
|
2024-11-14 14:11:03 +00:00
|
|
|
assert TR.regexp("reset__select_word_x_of_y_template").match(
|
|
|
|
layout.text_content()
|
|
|
|
)
|
2023-05-12 09:19:35 +00:00
|
|
|
for _ in range(3):
|
|
|
|
# "Select word 3 of 20"
|
|
|
|
# ^
|
2023-11-21 12:51:59 +00:00
|
|
|
word_pos_match = re.search(r"\d+", debug.read_layout().text_content())
|
2023-08-11 15:57:32 +00:00
|
|
|
assert word_pos_match is not None
|
|
|
|
word_pos = int(word_pos_match.group(0))
|
2023-05-12 09:19:35 +00:00
|
|
|
# Unifying both the buttons and words to lowercase
|
|
|
|
btn_texts = [
|
|
|
|
text.lower() for text in layout.tt_check_seed_button_contents()
|
|
|
|
]
|
|
|
|
wanted_word = words[word_pos - 1].lower()
|
|
|
|
button_pos = btn_texts.index(wanted_word)
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.click(buttons.RESET_WORD_CHECK[button_pos])
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Delizia:
|
2024-11-14 14:11:03 +00:00
|
|
|
assert TR.regexp("reset__select_word_x_of_y_template").match(layout.subtitle())
|
2024-06-04 13:40:42 +00:00
|
|
|
for _ in range(3):
|
|
|
|
# "Select word 3 of 20"
|
|
|
|
# ^
|
2023-11-21 12:51:59 +00:00
|
|
|
word_pos_match = re.search(r"\d+", debug.read_layout().subtitle())
|
2024-06-04 13:40:42 +00:00
|
|
|
assert word_pos_match is not None
|
|
|
|
word_pos = int(word_pos_match.group(0))
|
|
|
|
# Unifying both the buttons and words to lowercase
|
|
|
|
btn_texts = [
|
|
|
|
text.lower() for text in layout.tt_check_seed_button_contents()
|
|
|
|
]
|
|
|
|
wanted_word = words[word_pos - 1].lower()
|
|
|
|
button_pos = btn_texts.index(wanted_word)
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.click(buttons.VERTICAL_MENU[button_pos])
|
2025-01-17 12:04:11 +00:00
|
|
|
elif debug.layout_type is LayoutType.Caesar:
|
2024-11-14 14:11:03 +00:00
|
|
|
assert TR.reset__select_correct_word in layout.text_content()
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.press_right()
|
2023-05-12 09:19:35 +00:00
|
|
|
for _ in range(3):
|
|
|
|
# "SELECT 2ND WORD"
|
|
|
|
# ^
|
2023-08-11 15:57:32 +00:00
|
|
|
word_pos_match = re.search(r"\d+", layout.title())
|
|
|
|
assert word_pos_match is not None
|
|
|
|
word_pos = int(word_pos_match.group(0))
|
|
|
|
|
2023-05-12 09:19:35 +00:00
|
|
|
wanted_word = words[word_pos - 1].lower()
|
|
|
|
|
|
|
|
while not layout.get_middle_choice() == wanted_word:
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.press_right()
|
2023-05-12 09:19:35 +00:00
|
|
|
|
2023-11-21 12:51:59 +00:00
|
|
|
layout = debug.press_middle()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
def validate_mnemonics(mnemonics: list[str], expected_ems: bytes) -> None:
|
2019-11-04 14:45:54 +00:00
|
|
|
# We expect these combinations to recreate the secret properly
|
|
|
|
# In case of click tests the mnemonics are always XofX so no need for combinations
|
2021-02-03 12:39:26 +00:00
|
|
|
groups = shamir.decode_mnemonics(mnemonics)
|
|
|
|
ems = shamir.recover_ems(groups)
|
|
|
|
assert expected_ems == ems.ciphertext
|