mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-02-23 04:52:01 +00:00
325 lines
11 KiB
Python
325 lines
11 KiB
Python
![]() |
# This file is part of the Trezor project.
|
||
|
#
|
||
|
# Copyright (C) 2012-2019 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>.
|
||
|
|
||
|
import json
|
||
|
from contextlib import contextmanager
|
||
|
from copy import deepcopy
|
||
|
from pathlib import Path
|
||
|
from typing import Any, Dict, Generator
|
||
|
|
||
|
import pytest
|
||
|
|
||
|
from trezorlib import debuglink, device, exceptions, messages, translations
|
||
|
from trezorlib.debuglink import TrezorClientDebugLink as Client
|
||
|
|
||
|
pytestmark = pytest.mark.skip_t1
|
||
|
|
||
|
HERE = Path(__file__).parent.resolve()
|
||
|
CORE = HERE.parent.parent / "core"
|
||
|
TRANSLATIONS = CORE / "embed" / "rust" / "src" / "ui" / "translations"
|
||
|
FONTS = TRANSLATIONS / "fonts"
|
||
|
ORDER_FILE = TRANSLATIONS / "order.json"
|
||
|
|
||
|
EN_JSON = TRANSLATIONS / "en.json"
|
||
|
CS_JSON = TRANSLATIONS / "cs.json"
|
||
|
FR_JSON = TRANSLATIONS / "fr.json"
|
||
|
|
||
|
MAX_DATA_LENGTH = {"T": 48 * 1024, "Safe 3": 32 * 1024}
|
||
|
|
||
|
|
||
|
def _read_confirm_word(file: Path) -> str:
|
||
|
content = json.loads(file.read_text())
|
||
|
return content["translations"]["words"]["confirm"]
|
||
|
|
||
|
|
||
|
ENGLISH_CONFIRM = _read_confirm_word(EN_JSON)
|
||
|
CZECH_CONFIRM = _read_confirm_word(CS_JSON)
|
||
|
FRENCH_CONFIRM = _read_confirm_word(FR_JSON)
|
||
|
|
||
|
|
||
|
@contextmanager
|
||
|
def _set_english_return_back(client: Client) -> Generator[Client, None, None]:
|
||
|
lang_before = client.features.language or ""
|
||
|
try:
|
||
|
_set_default_english(client)
|
||
|
yield client
|
||
|
finally:
|
||
|
if lang_before.startswith("en"):
|
||
|
_set_default_english(client)
|
||
|
elif lang_before == "cs":
|
||
|
_set_full_czech(client)
|
||
|
elif lang_before == "fr":
|
||
|
_set_full_french(client)
|
||
|
else:
|
||
|
raise RuntimeError(f"Unknown language: {lang_before}")
|
||
|
|
||
|
|
||
|
def _set_full_czech(client: Client):
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=translations.blob_from_file(CS_JSON, client.features.model or ""),
|
||
|
)
|
||
|
|
||
|
|
||
|
def _set_full_french(client: Client):
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=translations.blob_from_file(FR_JSON, client.features.model or ""),
|
||
|
)
|
||
|
|
||
|
|
||
|
def _set_default_english(client: Client):
|
||
|
with client:
|
||
|
device.change_language(client, language_data=b"")
|
||
|
|
||
|
|
||
|
def _get_data_from_dict(data: Dict[str, Any], client: Client) -> bytes:
|
||
|
return translations.blob_from_dict(
|
||
|
data,
|
||
|
font_dir=FONTS,
|
||
|
order_json_file=ORDER_FILE,
|
||
|
model=client.features.model or "",
|
||
|
)
|
||
|
|
||
|
|
||
|
def _check_ping_screen_texts(client: Client, title: str, right_button: str) -> None:
|
||
|
def ping_input_flow(client: Client, title: str, right_button: str):
|
||
|
yield
|
||
|
layout = client.debug.wait_layout()
|
||
|
assert layout.title() == title.upper()
|
||
|
assert layout.button_contents()[-1] == right_button.upper()
|
||
|
client.debug.press_yes()
|
||
|
|
||
|
# TT does not have a right button text (but a green OK tick)
|
||
|
if client.features.model == "T":
|
||
|
right_button = "-"
|
||
|
|
||
|
with client:
|
||
|
client.watch_layout(True)
|
||
|
client.set_input_flow(ping_input_flow(client, title, right_button))
|
||
|
ping = client.call(messages.Ping(message="ahoj!", button_protection=True))
|
||
|
assert ping == messages.Success(message="ahoj!")
|
||
|
|
||
|
|
||
|
def test_change_language_errors(client: Client):
|
||
|
with _set_english_return_back(client) as client:
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Translations too short
|
||
|
# Sending less data than the header length
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Translations too short"
|
||
|
), client:
|
||
|
bad_data = (translations.HEADER_LEN - 1) * b"a"
|
||
|
device.change_language(client, language_data=bad_data)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Translations too long
|
||
|
# Sending more than allowed by the flash capacity
|
||
|
max_length = MAX_DATA_LENGTH[client.features.model]
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Translations too long"
|
||
|
), client:
|
||
|
bad_data = (max_length + 1) * b"a"
|
||
|
device.change_language(client, language_data=bad_data)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid header data length
|
||
|
# Sending more data than advertised in the header
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid header data length"
|
||
|
), client:
|
||
|
good_data = translations.blob_from_file(
|
||
|
CS_JSON, client.features.model or ""
|
||
|
)
|
||
|
bad_data = good_data + b"abcd"
|
||
|
device.change_language(client, language_data=bad_data)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid header magic
|
||
|
# Does not match the expected magic
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid header magic"
|
||
|
), client:
|
||
|
good_data = translations.blob_from_file(
|
||
|
CS_JSON, client.features.model or ""
|
||
|
)
|
||
|
bad_data = 4 * b"a" + good_data[4:]
|
||
|
device.change_language(client, language_data=bad_data)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid header data
|
||
|
# Putting non-zero bytes where zero is expected
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid header data"
|
||
|
), client:
|
||
|
good_data = translations.blob_from_file(
|
||
|
CS_JSON, client.features.model or ""
|
||
|
)
|
||
|
pre_sig_pos = translations.HEADER_LEN - translations.SIG_LEN
|
||
|
bad_data = good_data[: pre_sig_pos - 4] + 4 * b"a" + good_data[pre_sig_pos:]
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=bad_data,
|
||
|
)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid data hash
|
||
|
# Changing the data after their hash has been calculated
|
||
|
with pytest.raises(exceptions.TrezorFailure, match="Invalid data hash"), client:
|
||
|
good_data = translations.blob_from_file(
|
||
|
CS_JSON, client.features.model or ""
|
||
|
)
|
||
|
bad_data = good_data[:-8] + 8 * b"a"
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=bad_data,
|
||
|
)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid translations version
|
||
|
# Change the version to one not matching the current device
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid translations version"
|
||
|
), client:
|
||
|
with open(CS_JSON, "r") as f:
|
||
|
data = json.load(f)
|
||
|
data["header"]["version"] = "3.5.4"
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=_get_data_from_dict(data, client),
|
||
|
)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid header version
|
||
|
# Version is not a valid semver with integers
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid header version"
|
||
|
), client:
|
||
|
with open(CS_JSON, "r") as f:
|
||
|
data = json.load(f)
|
||
|
data["header"]["version"] = "ABC.XYZ.DEF"
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=_get_data_from_dict(data, client),
|
||
|
)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Invalid translations signature
|
||
|
# Modifying the signature part of the header
|
||
|
with pytest.raises(
|
||
|
exceptions.TrezorFailure, match="Invalid translations signature"
|
||
|
), client:
|
||
|
good_data = translations.blob_from_file(
|
||
|
CS_JSON, client.features.model or ""
|
||
|
)
|
||
|
bad_data = (
|
||
|
good_data[: translations.HEADER_LEN - 8]
|
||
|
+ 8 * b"a"
|
||
|
+ good_data[translations.HEADER_LEN :]
|
||
|
)
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=bad_data,
|
||
|
)
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
_check_ping_screen_texts(client, ENGLISH_CONFIRM, ENGLISH_CONFIRM)
|
||
|
|
||
|
|
||
|
def test_full_language_change(client: Client):
|
||
|
with _set_english_return_back(client) as client:
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Setting cs language
|
||
|
_set_full_czech(client)
|
||
|
assert client.features.language == "cs"
|
||
|
_check_ping_screen_texts(client, CZECH_CONFIRM, CZECH_CONFIRM)
|
||
|
|
||
|
# Setting fr language
|
||
|
_set_full_french(client)
|
||
|
assert client.features.language == "fr"
|
||
|
_check_ping_screen_texts(client, FRENCH_CONFIRM, FRENCH_CONFIRM)
|
||
|
|
||
|
# Setting the default language via empty data
|
||
|
_set_default_english(client)
|
||
|
assert client.features.language == "en-US"
|
||
|
_check_ping_screen_texts(client, ENGLISH_CONFIRM, ENGLISH_CONFIRM)
|
||
|
|
||
|
|
||
|
def test_language_stays_after_wipe(client: Client):
|
||
|
with _set_english_return_back(client) as client:
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
_check_ping_screen_texts(client, ENGLISH_CONFIRM, ENGLISH_CONFIRM)
|
||
|
|
||
|
# Setting cs language
|
||
|
_set_full_czech(client)
|
||
|
assert client.features.language == "cs"
|
||
|
|
||
|
_check_ping_screen_texts(client, CZECH_CONFIRM, CZECH_CONFIRM)
|
||
|
|
||
|
# Wipe device
|
||
|
device.wipe(client)
|
||
|
assert client.features.language == "cs"
|
||
|
|
||
|
# Load it again
|
||
|
debuglink.load_device(
|
||
|
client,
|
||
|
mnemonic=" ".join(["all"] * 12),
|
||
|
pin=None,
|
||
|
passphrase_protection=False,
|
||
|
label="test",
|
||
|
)
|
||
|
assert client.features.language == "cs"
|
||
|
|
||
|
_check_ping_screen_texts(client, CZECH_CONFIRM, CZECH_CONFIRM)
|
||
|
|
||
|
|
||
|
def test_translations_renders_on_screen(client: Client):
|
||
|
with open(CS_JSON, "r") as f:
|
||
|
czech_data = json.load(f)
|
||
|
|
||
|
# Setting some values of words__confirm key and checking that in ping screen title
|
||
|
with _set_english_return_back(client) as client:
|
||
|
assert client.features.language == "en-US"
|
||
|
|
||
|
# Normal english
|
||
|
_check_ping_screen_texts(client, ENGLISH_CONFIRM, ENGLISH_CONFIRM)
|
||
|
|
||
|
# Normal czech
|
||
|
_set_full_czech(client)
|
||
|
assert client.features.language == "cs"
|
||
|
_check_ping_screen_texts(client, CZECH_CONFIRM, CZECH_CONFIRM)
|
||
|
|
||
|
# Modified czech - changed value
|
||
|
czech_data_copy = deepcopy(czech_data)
|
||
|
new_czech_confirm = "ABCD"
|
||
|
czech_data_copy["translations"]["words"]["confirm"] = new_czech_confirm
|
||
|
device.change_language(
|
||
|
client,
|
||
|
language_data=_get_data_from_dict(czech_data_copy, client),
|
||
|
)
|
||
|
_check_ping_screen_texts(client, new_czech_confirm, CZECH_CONFIRM)
|
||
|
|
||
|
# Modified czech - key deleted completely, english is shown
|
||
|
czech_data_copy = deepcopy(czech_data)
|
||
|
del czech_data_copy["translations"]["words"]["confirm"]
|
||
|
device.change_language(
|
||
|
client, language_data=_get_data_from_dict(czech_data_copy, client)
|
||
|
)
|
||
|
_check_ping_screen_texts(client, ENGLISH_CONFIRM, CZECH_CONFIRM)
|