mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-11-16 04:29:08 +00:00
7f1a5ac4c1
WIP - refactor and extend font generation for non-ascii characters WIP - add czech characters mapping between UTF8 value and index WIP - regenerate font files with czech characters WIP - shorten czech button text, it was causing SHUTDOWN for some reason WIP - support UTF8 characters in fonts.c WIP - account for translation in tests WIP - small fixes WIP - fix last test WIP - support UTF8 also in Rust font operations WIP - add a script to find non-translated english strings in micropython code WIP - add a validator script for checking missing micropython translations WIP - translate remaining altcoins and other apps in core (fido, sdcard, TT layouts, ...) WIP - generate czech glyphs for TT fonts WIP - modify gen_font.py to account for negative bearing czech characters WIP - extend translation validation scripts, move them into core/tools WIP - translate TT layouts in Rust WIP - fix tests WIP - fix inverse coloring of nonprintable glyph WIP - add build and test pipelines for Czech language WIP - merge both JSON files together WIP - run new isort WIP - unify all the translation in Rust, expose to micropython TEMP - leave en_merged.json file, so it is accessible by translators with old link WIP - fixes WIP - add french characters and translation via Google Translator WIP - skip rustfmt in mako-created files WIP - revert all the font height changes causing false-positive UI diff WIP - fixes after rebase WIP - fix broken translations WIP - revert some wording changes causing UI diff WIP - improve validation and translate scripts, translate missing strings WIP - sort all keys alphabetically WIP - remove any usage of translation in bootloader WIP - add newline at the end of JSON file WIP - fix bitcoin-only strings check WIP - fix python support check WIP - add some missing translations WIP - fix SD card device test WIP - fix pystyle WIP - fix rust unittests WIP - fix click tests WIP - flag errors in french translations WIP - add script transferring translations data into a byte blob WIP - regenerate fr.rs WIP - store and read language translations from flash WIP - storing language name in storage WIP - sending language_data in apply_settings protobuf message WIP - separate protobuf message for translations, fixes WIP - set up translations area for TT as well WIP - get rid of TREZOR_LANG env variable during build WIP - make the firmware buildable for TT WIP - add basic device tests WIP - set language for tests WIP - counting with language when writing fixtures WIP - add todos WIP - fix CI WIP - unify translations, make titles CAPITAL WIP - translate missing english WIP - skip translations messages for T1 WIP - not changing tests names for english WIP - fix flake8 WIP - no test language setting for T1 WIP - clippy lint about complex data type WIP - fix some english UI diff for TR WIP - fix cstyle WIP - minimize the usage of #[cfg(feature = "micropython")] outside translations module WIP - minimize TT's UI diff WIP - fix ruststyle WIP - fix TR build WIP - advanced Shamir text change WIP - storing the language name as the first item in the translation data WIP - modify and extend tests after storing language name WIP - modify checklist sentence WIP - add TEST_LANG into Makefile for all the emu tests WIP - default arguments WIP - reimplement default arguments remove unneeded pub from get_info function WIP - Rust handling of object attributes lookups from upy - thanks Matejcik! WIP - generate mock interface for attribute-based translations lookups WIP - change function calls to object attributes WIP - symbolic link for unix/translations.c WIP - fix and improve the reading of translations - thanks Matejcik! WIP - add support for multiple languages in removing missing tests WIP - fix multiple-accounts warning in tests WIP - fix encoding of newlines in translations WIP - fix czech tutorial text WIP - fix czech click tests WIP - do not translate wire error messages WIP - add language options to click tests as well WIP - setup czech device tests in CI WIP - setup czech click tests in CI WIP - record czech device tests for TR WIP - record czech click tests for TR WIP - record czech device tests for TT WIP - record czech click tests for TT WIP - pystyle WIP - cstyle WIP - fix Rust micropython import dependency WIP - fix czech recordings WIP - support french translations in tests WIP - shorten some french words to fix the tests WIP - fix micropython cfg compilation WIP - record french click tests for TR WIP - record french device tests for TR WIP - record french device tests for TT WIP - record french click tests for TT WIP - fix french translations - shorten them WIP - translate missing french words WIP - fix click tests WIP - add french tests into CI WIP - pystyle WIP - allow for czech/french tests in update script WIP - update czech fixtures WIP - update french fixtures WIP - ruststyle WIP - disallow MPU to run it on hardware WIP - cstyle WIP - change translations delimiter from * to \x00 WIP - change translations protobufs WIP - remove language handling from storage WIP - add header into JSON files WIP - count with header in translations blob WIP - yml style fixes WIP - fix proto gen WIP - verify version and data hash WIP - fix loading test translations feat(core): allow access to translations area in firmware [no changelog] WIP - fixes after rebase WIP - increase the TT's translations area to 3 sectors WIP - dynamically read the maximum translations size WIP - record non-english tests from CI WIP - loading font data from translations blob WIP - bump translations version WIP - include czech and french glyph data WIP - whitelist another negative-bearing glyph WIP - remove czech/french glyphs from common font files WIP - fix language tests WIP - specific fonts for specific models WIP - revert the non-ascii font hardcoding WIP - include missing BIG font into nonprintable logic WIP - minor Rust code improvements WIP - include newlines at the end of json files WIP - move glyph Rust function to librust_fonts.h WIP - add all fonts into translations file WIP - move fonts into its own dir WIP - reflect separate dir for fonts WIP - not putting translations trezorhal into bootloader WIP - write and read multiple fonts into translations data WIP - silence pyright issue/notissue WIP - delete no more used translations/*.py imports WIP - fix bootloader builds by introducing translations feature and TRANSLATIONS flag WIP - fix TT's bootloader Rust build WIP - fix tests in non-english languages WIP - not search for UTF-8 when there are no translations data WIP - add colons to strings where missing WIP - fix language loading in tests WIP - fix signmessage input flow to work in all languages WIP - create offset table for translation strings WIP - code improvements WIP - record foreign language fixtures + sync with main in english WIP - do alignment check before reading u16 data WIP - allocate blob in RAM for translations data WIP - add TODO for blob generation WIP - record non-english device tests WIP - use bytes.align_to instead of messing with pointers WIP - fixtures WIP - remove unused import WIP - add order.py WIP - add order.json WIP - take order.json into account in creating general.rs WIP - take order.json into account in generating the blob WIP - style WIP - sort the language files WIP - remove unused file WIP - code improvements WIP - add TODO for homescreen notification WIP - translate plural forms WIP - translate time intervals WIP - sign translations with dev keys, validate signatures, improve robustness WIP - improve tests for translations WIP - add `trezorctl utils sign-translations` for production signing of the blob WIP - pyright fix WIP - changing TR progress loader offset - it was colliding with title WIP - show indeterminate loader when loading translations data WIP - record new and updated language tests WIP - show the change language title/prompt in the target language WIP - sort keys WIP - add crowdin-cli into shell.nix WIP - add crowdin sync script
365 lines
13 KiB
Python
365 lines
13 KiB
Python
from trezorlib import messages
|
|
from trezorlib.debuglink import TrezorClientDebugLink as Client
|
|
|
|
from . import translations as TR
|
|
from .common import BRGeneratorType
|
|
|
|
B = messages.ButtonRequestType
|
|
|
|
|
|
class PinFlow:
|
|
def __init__(self, client: Client):
|
|
self.client = client
|
|
self.debug = self.client.debug
|
|
|
|
def setup_new_pin(
|
|
self, pin: str, second_different_pin: str | None = None
|
|
) -> BRGeneratorType:
|
|
yield # Enter PIN
|
|
assert "PinKeyboard" in self.debug.wait_layout().all_components()
|
|
self.debug.input(pin)
|
|
if self.debug.model == "Safe 3":
|
|
yield # Reenter PIN
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "pin.reenter_to_confirm"
|
|
)
|
|
self.debug.press_yes()
|
|
yield # Enter PIN again
|
|
assert "PinKeyboard" in self.debug.wait_layout().all_components()
|
|
if second_different_pin is not None:
|
|
self.debug.input(second_different_pin)
|
|
else:
|
|
self.debug.input(pin)
|
|
|
|
|
|
class BackupFlow:
|
|
def __init__(self, client: Client):
|
|
self.client = client
|
|
self.debug = self.client.debug
|
|
|
|
def confirm_new_wallet(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self.debug.wait_layout().text_content(), "reset.by_continuing")
|
|
if self.debug.model == "Safe 3":
|
|
self.debug.press_right()
|
|
self.debug.press_yes()
|
|
|
|
|
|
class RecoveryFlow:
|
|
def __init__(self, client: Client):
|
|
self.client = client
|
|
self.debug = self.client.debug
|
|
|
|
def _text_content(self) -> str:
|
|
return self.debug.wait_layout().text_content()
|
|
|
|
def confirm_recovery(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "reset.by_continuing")
|
|
if self.debug.model == "Safe 3":
|
|
self.debug.press_right()
|
|
self.debug.press_yes()
|
|
|
|
def confirm_dry_run(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.check_dry_run")
|
|
self.debug.press_yes()
|
|
|
|
def setup_slip39_recovery(self, num_words: int) -> BRGeneratorType:
|
|
if self.debug.model == "Safe 3":
|
|
yield from self.tr_recovery_homescreen()
|
|
yield from self.input_number_of_words(num_words)
|
|
yield from self.enter_any_share()
|
|
|
|
def setup_bip39_recovery(self, num_words: int) -> BRGeneratorType:
|
|
if self.debug.model == "Safe 3":
|
|
yield from self.tr_recovery_homescreen()
|
|
yield from self.input_number_of_words(num_words)
|
|
yield from self.enter_your_backup()
|
|
|
|
def tr_recovery_homescreen(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.num_of_words")
|
|
self.debug.press_yes()
|
|
|
|
def enter_your_backup(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.enter_backup")
|
|
is_dry_run = any(
|
|
title in self.debug.wait_layout().title()
|
|
for title in TR.translate("recovery.title_dry_run")
|
|
)
|
|
if self.debug.model == "Safe 3" and not is_dry_run:
|
|
# Normal recovery has extra info (not dry run)
|
|
self.debug.press_right(wait=True)
|
|
self.debug.press_right(wait=True)
|
|
self.debug.press_yes()
|
|
|
|
def enter_any_share(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.enter_any_share")
|
|
is_dry_run = any(
|
|
title in self.debug.wait_layout().title()
|
|
for title in TR.translate("recovery.title_dry_run")
|
|
)
|
|
if self.debug.model == "Safe 3" and not is_dry_run:
|
|
# Normal recovery has extra info (not dry run)
|
|
self.debug.press_right(wait=True)
|
|
self.debug.press_right(wait=True)
|
|
self.debug.press_yes()
|
|
|
|
def abort_recovery(self, confirm: bool) -> BRGeneratorType:
|
|
yield
|
|
if self.debug.model == "Safe 3":
|
|
TR.assert_in(self._text_content(), "recovery.num_of_words")
|
|
else:
|
|
TR.assert_in(self._text_content(), "recovery.enter_any_share")
|
|
self.debug.press_no()
|
|
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.wanna_cancel_recovery")
|
|
if self.debug.model == "Safe 3":
|
|
self.debug.press_right()
|
|
if confirm:
|
|
self.debug.press_yes()
|
|
else:
|
|
self.debug.press_no()
|
|
|
|
def input_number_of_words(self, num_words: int) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.MnemonicWordCount
|
|
if self.debug.model == "Safe 3":
|
|
TR.assert_in(self.debug.wait_layout().title(), "word_count.title")
|
|
else:
|
|
TR.assert_in(self._text_content(), "recovery.num_of_words")
|
|
self.debug.input(str(num_words))
|
|
|
|
def warning_invalid_recovery_seed(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.invalid_seed_entered")
|
|
self.debug.press_yes()
|
|
|
|
def warning_invalid_recovery_share(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.invalid_share_entered")
|
|
self.debug.press_yes()
|
|
|
|
def warning_group_threshold_reached(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.group_threshold_reached")
|
|
self.debug.press_yes()
|
|
|
|
def warning_share_already_entered(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.share_already_entered")
|
|
self.debug.press_yes()
|
|
|
|
def warning_share_from_another_shamir(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.share_from_another_shamir")
|
|
self.debug.press_yes()
|
|
|
|
def success_share_group_entered(self) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_in(self._text_content(), "recovery.you_have_entered")
|
|
self.debug.press_yes()
|
|
|
|
def success_wallet_recovered(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Success
|
|
TR.assert_in(self._text_content(), "recovery.wallet_recovered")
|
|
self.debug.press_yes()
|
|
|
|
def success_bip39_dry_run_valid(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Success
|
|
TR.assert_in(self._text_content(), "recovery.dry_run_bip39_valid_match")
|
|
self.debug.press_yes()
|
|
|
|
def success_slip39_dryrun_valid(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Success
|
|
TR.assert_in(self._text_content(), "recovery.dry_run_slip39_valid_match")
|
|
self.debug.press_yes()
|
|
|
|
def warning_slip39_dryrun_mismatch(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.dry_run_slip39_valid_mismatch")
|
|
self.debug.press_yes()
|
|
|
|
def warning_bip39_dryrun_mismatch(self) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.Warning
|
|
TR.assert_in(self._text_content(), "recovery.dry_run_bip39_valid_mismatch")
|
|
self.debug.press_yes()
|
|
|
|
def success_more_shares_needed(
|
|
self, count_needed: int | None = None
|
|
) -> BRGeneratorType:
|
|
yield
|
|
# TODO: do this plural assert
|
|
# assert (
|
|
# "1 more share needed" in self._text_content()
|
|
# or "More shares needed" in self._text_content()
|
|
# )
|
|
if count_needed is not None:
|
|
assert str(count_needed) in self._text_content()
|
|
self.debug.press_yes()
|
|
|
|
def input_mnemonic(self, mnemonic: list[str]) -> BRGeneratorType:
|
|
br = yield
|
|
assert br.code == B.MnemonicInput
|
|
assert "MnemonicKeyboard" in self.debug.wait_layout().all_components()
|
|
for _, word in enumerate(mnemonic):
|
|
# TODO: do this format assert
|
|
# if self.debug.model == "Safe 3":
|
|
# assert f"WORD {index + 1}" in self.debug.wait_layout().title()
|
|
# else:
|
|
# assert f"Type word {index + 1}" in self._text_content()
|
|
self.debug.input(word)
|
|
|
|
def input_all_slip39_shares(
|
|
self,
|
|
shares: list[str],
|
|
has_groups: bool = False,
|
|
click_info: bool = False,
|
|
) -> BRGeneratorType:
|
|
for index, share in enumerate(shares):
|
|
mnemonic = share.split(" ")
|
|
yield from self.input_mnemonic(mnemonic)
|
|
|
|
if index < len(shares) - 1:
|
|
if has_groups:
|
|
yield from self.success_share_group_entered()
|
|
if self.debug.model == "T" and click_info:
|
|
yield from self.tt_click_info()
|
|
yield from self.success_more_shares_needed()
|
|
|
|
def tt_click_info(
|
|
self,
|
|
) -> BRGeneratorType:
|
|
# Moving through the INFO button
|
|
self.debug.press_info()
|
|
yield
|
|
self.debug.swipe_up()
|
|
self.debug.press_yes()
|
|
|
|
|
|
class EthereumFlow:
|
|
GO_BACK = (16, 220)
|
|
|
|
def __init__(self, client: Client):
|
|
self.client = client
|
|
self.debug = self.client.debug
|
|
|
|
def confirm_data(self, info: bool = False, cancel: bool = False) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_equals(
|
|
self.debug.wait_layout().title(), "ethereum.title_confirm_data"
|
|
)
|
|
if info:
|
|
self.debug.press_info()
|
|
elif cancel:
|
|
self.debug.press_no()
|
|
else:
|
|
self.debug.press_yes()
|
|
|
|
def paginate_data(self) -> BRGeneratorType:
|
|
br = yield
|
|
TR.assert_equals(
|
|
self.debug.wait_layout().title(), "ethereum.title_confirm_data"
|
|
)
|
|
assert br.pages is not None
|
|
for i in range(br.pages):
|
|
self.debug.wait_layout()
|
|
if i < br.pages - 1:
|
|
self.debug.swipe_up()
|
|
self.debug.press_yes()
|
|
|
|
def paginate_data_go_back(self) -> BRGeneratorType:
|
|
br = yield
|
|
TR.assert_equals(
|
|
self.debug.wait_layout().title(), "ethereum.title_confirm_data"
|
|
)
|
|
assert br.pages is not None
|
|
assert br.pages > 2
|
|
if self.debug.model == "T":
|
|
self.debug.swipe_up(wait=True)
|
|
self.debug.swipe_up(wait=True)
|
|
self.debug.click(self.GO_BACK)
|
|
else:
|
|
self.debug.press_right()
|
|
self.debug.press_right()
|
|
self.debug.press_left()
|
|
self.debug.press_left()
|
|
self.debug.press_left()
|
|
|
|
def confirm_tx(
|
|
self,
|
|
cancel: bool = False,
|
|
info: bool = False,
|
|
go_back_from_summary: bool = False,
|
|
) -> BRGeneratorType:
|
|
yield
|
|
TR.assert_equals(self.debug.wait_layout().title(), "send.title_recipient")
|
|
|
|
if self.debug.model == "T":
|
|
if cancel:
|
|
self.debug.press_no()
|
|
else:
|
|
self.debug.press_yes()
|
|
yield
|
|
TR.assert_equals(
|
|
self.debug.wait_layout().title(), "words.title_summary"
|
|
)
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "send.maximum_fee"
|
|
)
|
|
if go_back_from_summary:
|
|
self.debug.press_no()
|
|
yield
|
|
self.debug.press_yes()
|
|
yield
|
|
if info:
|
|
self.debug.press_info(wait=True)
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "ethereum.gas_limit"
|
|
)
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "ethereum.gas_price"
|
|
)
|
|
self.debug.press_no(wait=True)
|
|
self.debug.press_yes()
|
|
else:
|
|
if cancel:
|
|
self.debug.press_left()
|
|
else:
|
|
self.debug.press_right()
|
|
yield
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "send.maximum_fee"
|
|
)
|
|
if go_back_from_summary:
|
|
self.debug.press_left()
|
|
yield
|
|
self.debug.press_right()
|
|
yield
|
|
if info:
|
|
self.debug.press_right(wait=True)
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "ethereum.gas_limit"
|
|
)
|
|
self.debug.press_right(wait=True)
|
|
TR.assert_in(
|
|
self.debug.wait_layout().text_content(), "ethereum.gas_price"
|
|
)
|
|
self.debug.press_left(wait=True)
|
|
self.debug.press_left(wait=True)
|
|
self.debug.press_middle()
|