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
356 lines
11 KiB
Python
356 lines
11 KiB
Python
# This file is part of the Trezor project.
|
|
#
|
|
# Copyright (C) 2012-2023 SatoshiLabs and contributors
|
|
#
|
|
# This library is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Lesser General Public License version 3
|
|
# as published by the Free Software Foundation.
|
|
#
|
|
# This library is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the License along with this library.
|
|
# If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>.
|
|
|
|
from contextlib import contextmanager
|
|
from enum import Enum
|
|
from typing import TYPE_CHECKING, Generator
|
|
|
|
import pytest
|
|
|
|
from trezorlib import device, exceptions
|
|
|
|
from .. import buttons
|
|
from .. import translations as TR
|
|
from .common import go_back, go_next, navigate_to_action_and_press
|
|
|
|
if TYPE_CHECKING:
|
|
from trezorlib.debuglink import DebugLink
|
|
|
|
from ..device_handler import BackgroundDeviceHandler
|
|
|
|
|
|
pytestmark = pytest.mark.skip_t1
|
|
|
|
PIN_CANCELLED = pytest.raises(exceptions.TrezorFailure, match="PIN entry cancelled")
|
|
PIN_INVALID = pytest.raises(exceptions.TrezorFailure, match="PIN invalid")
|
|
|
|
PIN4 = "1234"
|
|
PIN24 = "875163065288639289952973"
|
|
PIN50 = "31415926535897932384626433832795028841971693993751"
|
|
PIN60 = PIN50 + "9" * 10
|
|
|
|
|
|
def _get_possible_btns(path: str) -> str:
|
|
return "|".join(TR.translate(path))
|
|
|
|
|
|
DELETE = _get_possible_btns("inputs.delete")
|
|
SHOW = _get_possible_btns("inputs.show")
|
|
ENTER = _get_possible_btns("inputs.enter")
|
|
|
|
|
|
TR_PIN_ACTIONS = [
|
|
DELETE,
|
|
SHOW,
|
|
ENTER,
|
|
"0",
|
|
"1",
|
|
"2",
|
|
"3",
|
|
"4",
|
|
"5",
|
|
"6",
|
|
"7",
|
|
"8",
|
|
"9",
|
|
]
|
|
|
|
|
|
class Situation(Enum):
|
|
PIN_INPUT = 1
|
|
PIN_SETUP = 2
|
|
PIN_CHANGE = 3
|
|
WIPE_CODE_SETUP = 4
|
|
|
|
|
|
@contextmanager
|
|
def prepare(
|
|
device_handler: "BackgroundDeviceHandler",
|
|
situation: Situation = Situation.PIN_INPUT,
|
|
old_pin: str = "",
|
|
) -> Generator["DebugLink", None, None]:
|
|
debug = device_handler.debuglink()
|
|
# So that the digit order is the same. Needed for UI tests.
|
|
# Even though it should be done in conftest::client fixture (used by device_handler),
|
|
# without reseeding "again", the results are still random.
|
|
debug.reseed(0)
|
|
|
|
# Setup according to the wanted situation
|
|
if situation == Situation.PIN_INPUT:
|
|
# Any action triggering the PIN dialogue
|
|
device_handler.run(device.apply_settings, auto_lock_delay_ms=300_000) # type: ignore
|
|
elif situation == Situation.PIN_SETUP:
|
|
# Set new PIN
|
|
device_handler.run(device.change_pin) # type: ignore
|
|
TR.assert_in(debug.wait_layout().text_content(), "pin.turn_on")
|
|
if debug.model == "T":
|
|
go_next(debug)
|
|
elif debug.model == "Safe 3":
|
|
go_next(debug, wait=True)
|
|
go_next(debug, wait=True)
|
|
go_next(debug, wait=True)
|
|
go_next(debug, wait=True)
|
|
elif situation == Situation.PIN_CHANGE:
|
|
# Change PIN
|
|
device_handler.run(device.change_pin) # type: ignore
|
|
_input_see_confirm(debug, old_pin)
|
|
TR.assert_in(debug.wait_layout().text_content(), "pin.change")
|
|
go_next(debug, wait=True)
|
|
_input_see_confirm(debug, old_pin)
|
|
elif situation == Situation.WIPE_CODE_SETUP:
|
|
# Set wipe code
|
|
device_handler.run(device.change_wipe_code) # type: ignore
|
|
if old_pin:
|
|
_input_see_confirm(debug, old_pin)
|
|
TR.assert_in(debug.wait_layout().text_content(), "wipe_code.turn_on")
|
|
go_next(debug, wait=True)
|
|
if debug.model == "Safe 3":
|
|
go_next(debug, wait=True)
|
|
go_next(debug, wait=True)
|
|
go_next(debug, wait=True)
|
|
if old_pin:
|
|
debug.wait_layout()
|
|
_input_see_confirm(debug, old_pin)
|
|
|
|
debug.wait_layout()
|
|
_assert_pin_entry(debug)
|
|
yield debug
|
|
go_next(debug)
|
|
device_handler.result()
|
|
|
|
|
|
def _assert_pin_entry(debug: "DebugLink") -> None:
|
|
assert "PinKeyboard" in debug.read_layout().all_components()
|
|
|
|
|
|
def _input_pin(debug: "DebugLink", pin: str, check: bool = False) -> None:
|
|
"""Input the PIN"""
|
|
if check:
|
|
before = debug.read_layout().pin()
|
|
|
|
if debug.model == "T":
|
|
digits_order = debug.read_layout().tt_pin_digits_order()
|
|
for digit in pin:
|
|
digit_index = digits_order.index(digit)
|
|
coords = buttons.pin_passphrase_index(digit_index)
|
|
debug.click(coords, wait=True)
|
|
elif debug.model == "Safe 3":
|
|
for digit in pin:
|
|
navigate_to_action_and_press(debug, digit, TR_PIN_ACTIONS)
|
|
|
|
if check:
|
|
after = debug.read_layout().pin()
|
|
assert before + pin == after
|
|
|
|
|
|
def _see_pin(debug: "DebugLink") -> None:
|
|
"""Navigate to "SHOW" and press it"""
|
|
if debug.model == "T":
|
|
debug.click(buttons.TOP_ROW, wait=True)
|
|
elif debug.model == "Safe 3":
|
|
navigate_to_action_and_press(debug, SHOW, TR_PIN_ACTIONS)
|
|
|
|
|
|
def _delete_pin(debug: "DebugLink", digits_to_delete: int, check: bool = True) -> None:
|
|
"""Navigate to "DELETE" and press it how many times requested"""
|
|
if check:
|
|
before = debug.read_layout().pin()
|
|
|
|
for _ in range(digits_to_delete):
|
|
if debug.model == "T":
|
|
debug.click(buttons.pin_passphrase_grid(9), wait=True)
|
|
elif debug.model == "Safe 3":
|
|
navigate_to_action_and_press(debug, DELETE, TR_PIN_ACTIONS)
|
|
|
|
if check:
|
|
after = debug.read_layout().pin()
|
|
assert before[:-digits_to_delete] == after
|
|
|
|
|
|
def _delete_all(debug: "DebugLink", check: bool = True) -> None:
|
|
"""Navigate to "DELETE" and hold it until all digits are deleted"""
|
|
if debug.model == "T":
|
|
debug.click_hold(buttons.pin_passphrase_grid(9), hold_ms=1500)
|
|
elif debug.model == "Safe 3":
|
|
navigate_to_action_and_press(debug, DELETE, TR_PIN_ACTIONS, hold_ms=1000)
|
|
|
|
if check:
|
|
after = debug.read_layout().pin()
|
|
assert after == ""
|
|
|
|
|
|
def _cancel_pin(debug: "DebugLink") -> None:
|
|
"""Navigate to "CANCEL" and press it"""
|
|
# It is the same button as DELETE
|
|
# TODO: implement cancel PIN for TR?
|
|
_delete_pin(debug, 1, check=False)
|
|
|
|
|
|
def _confirm_pin(debug: "DebugLink") -> None:
|
|
"""Navigate to "ENTER" and press it"""
|
|
if debug.model == "T":
|
|
debug.click(buttons.pin_passphrase_grid(11), wait=True)
|
|
elif debug.model == "Safe 3":
|
|
navigate_to_action_and_press(debug, ENTER, TR_PIN_ACTIONS)
|
|
|
|
|
|
def _input_see_confirm(debug: "DebugLink", pin: str) -> None:
|
|
_input_pin(debug, pin)
|
|
_see_pin(debug)
|
|
_confirm_pin(debug)
|
|
|
|
|
|
def _enter_two_times(debug: "DebugLink", pin1: str, pin2: str) -> None:
|
|
_input_see_confirm(debug, pin1)
|
|
|
|
if debug.model == "Safe 3":
|
|
# Please re-enter
|
|
go_next(debug, wait=True)
|
|
|
|
_input_see_confirm(debug, pin2)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN4)
|
|
def test_pin_short(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_see_confirm(debug, PIN4)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN24)
|
|
def test_pin_long(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_see_confirm(debug, PIN24)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN4)
|
|
def test_pin_empty_cannot_send(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_see_confirm(debug, "")
|
|
_input_see_confirm(debug, PIN4)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN24)
|
|
def test_pin_long_delete(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_pin(debug, PIN24)
|
|
_see_pin(debug)
|
|
|
|
_delete_pin(debug, 10)
|
|
_see_pin(debug)
|
|
|
|
_input_see_confirm(debug, PIN24[-10:])
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN4)
|
|
def test_pin_delete_hold(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_pin(debug, PIN4)
|
|
_see_pin(debug)
|
|
|
|
_delete_all(debug)
|
|
|
|
_input_see_confirm(debug, PIN4)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN60[:50])
|
|
def test_pin_longer_than_max(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_pin(debug, PIN60, check=False)
|
|
|
|
# What is over 50 digits was not entered
|
|
# TODO: do some UI change when limit is reached?
|
|
assert debug.read_layout().pin() == PIN60[:50]
|
|
|
|
_see_pin(debug)
|
|
_confirm_pin(debug)
|
|
|
|
|
|
@pytest.mark.setup_client(pin=PIN4)
|
|
def test_pin_incorrect(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler) as debug:
|
|
_input_see_confirm(debug, "1235")
|
|
_input_see_confirm(debug, PIN4)
|
|
|
|
|
|
@pytest.mark.skip_tr("TODO: will we support cancelling on TR?")
|
|
@pytest.mark.setup_client(pin=PIN4)
|
|
def test_pin_cancel(device_handler: "BackgroundDeviceHandler"):
|
|
with PIN_CANCELLED, prepare(device_handler) as debug:
|
|
_input_pin(debug, PIN4)
|
|
_see_pin(debug)
|
|
_delete_pin(debug, len(PIN4))
|
|
_see_pin(debug)
|
|
_cancel_pin(debug)
|
|
|
|
|
|
@pytest.mark.setup_client()
|
|
def test_pin_setup(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler, Situation.PIN_SETUP) as debug:
|
|
_enter_two_times(debug, PIN4, PIN4)
|
|
|
|
|
|
@pytest.mark.setup_client()
|
|
def test_pin_setup_mismatch(device_handler: "BackgroundDeviceHandler"):
|
|
with PIN_CANCELLED, prepare(device_handler, Situation.PIN_SETUP) as debug:
|
|
_enter_two_times(debug, "1", "2")
|
|
if debug.model == "T":
|
|
go_next(debug)
|
|
_cancel_pin(debug)
|
|
elif debug.model == "Safe 3":
|
|
debug.press_middle()
|
|
debug.press_no()
|
|
|
|
|
|
@pytest.mark.setup_client(pin="1")
|
|
def test_pin_change(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler, Situation.PIN_CHANGE, old_pin="1") as debug:
|
|
_enter_two_times(debug, "2", "2")
|
|
|
|
|
|
@pytest.mark.setup_client(pin="1")
|
|
def test_wipe_code_setup(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler, Situation.WIPE_CODE_SETUP, old_pin="1") as debug:
|
|
_enter_two_times(debug, "2", "2")
|
|
|
|
|
|
# @pytest.mark.setup_client(pin="1")
|
|
# @pytest.mark.timeout(15)
|
|
# @pytest.mark.xfail(reason="It will disconnect from the emulator")
|
|
# def test_wipe_code_setup_and_trigger(device_handler: "BackgroundDeviceHandler"):
|
|
# with prepare(device_handler, Situation.WIPE_CODE_SETUP, old_pin="1") as debug:
|
|
# _enter_two_times(debug, "2", "2")
|
|
# device_handler.client.lock()
|
|
# with prepare(device_handler) as debug:
|
|
# _input_see_confirm(debug, "2")
|
|
|
|
|
|
@pytest.mark.setup_client(pin="1")
|
|
def test_wipe_code_same_as_pin(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler, Situation.WIPE_CODE_SETUP, old_pin="1") as debug:
|
|
_input_see_confirm(debug, "1")
|
|
# Try again
|
|
go_next(debug, wait=True)
|
|
_enter_two_times(debug, "2", "2")
|
|
|
|
|
|
@pytest.mark.setup_client()
|
|
def test_pin_same_as_wipe_code(device_handler: "BackgroundDeviceHandler"):
|
|
with prepare(device_handler, Situation.WIPE_CODE_SETUP) as debug:
|
|
_enter_two_times(debug, "1", "1")
|
|
with PIN_INVALID, prepare(device_handler, Situation.PIN_SETUP) as debug:
|
|
_enter_two_times(debug, "1", "1")
|
|
go_back(debug, r_middle=True)
|