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
191 lines
5.4 KiB
Python
191 lines
5.4 KiB
Python
"""
|
|
Helper functions for communication with Gitlab.
|
|
|
|
Allowing for interaction with the test results, e.g. with UI tests.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import re
|
|
from dataclasses import dataclass
|
|
from pathlib import Path
|
|
from typing import Any, Iterable, Iterator
|
|
|
|
import requests
|
|
|
|
AnyDict = dict[Any, Any]
|
|
|
|
HERE = Path(__file__).parent
|
|
|
|
BRANCHES_API_TEMPLATE = "https://gitlab.com/satoshilabs/trezor/trezor-firmware/-/pipelines.json?scope=branches&page={}"
|
|
GRAPHQL_API = "https://gitlab.com/api/graphql"
|
|
RAW_REPORT_URL_TEMPLATE = (
|
|
"https://gitlab.com/satoshilabs/trezor/trezor-firmware/-/jobs/{}/raw"
|
|
)
|
|
|
|
UI_JOBS_ENGLISH = [
|
|
"core click R test",
|
|
"core device R test",
|
|
"core click test",
|
|
"core device test",
|
|
"core persistence test",
|
|
"legacy device test",
|
|
]
|
|
|
|
UI_JOBS_FRENCH = [
|
|
"core click R test french",
|
|
"core device R test french",
|
|
"core click test french",
|
|
"core device test french",
|
|
]
|
|
|
|
UI_JOBS_CZECH = [
|
|
"core click R test czech",
|
|
"core device R test czech",
|
|
"core click test czech",
|
|
"core device test czech",
|
|
]
|
|
|
|
UI_JOB_NAMES = UI_JOBS_ENGLISH + UI_JOBS_FRENCH + UI_JOBS_CZECH
|
|
|
|
SAVE_GRAPHQL_RESULTS = False
|
|
|
|
|
|
@dataclass
|
|
class TestResult:
|
|
failed: int = 0
|
|
passed: int = 0
|
|
error: int = 0
|
|
|
|
@classmethod
|
|
def from_line(cls, line: str) -> TestResult:
|
|
self = TestResult()
|
|
for key in self.__annotations__:
|
|
match = re.search(rf"(\d+) {key}", line)
|
|
if match:
|
|
setattr(self, key, int(match.group(1)))
|
|
return self
|
|
|
|
@classmethod
|
|
def from_job_id(cls, job_id: str) -> TestResult:
|
|
report_link = RAW_REPORT_URL_TEMPLATE.format(job_id)
|
|
raw_content = requests.get(report_link).text
|
|
result_pattern = r"= .* passed.*s \(\d.*\) ="
|
|
result_line_match = re.search(result_pattern, raw_content)
|
|
if not result_line_match:
|
|
print("No results yet.")
|
|
return TestResult()
|
|
return cls.from_line(result_line_match.group(0))
|
|
|
|
|
|
def _get_gitlab_branches(page: int) -> list[AnyDict]:
|
|
return requests.get(BRANCHES_API_TEMPLATE.format(page)).json()["pipelines"]
|
|
|
|
|
|
def _get_branch_obj(branch_name: str) -> AnyDict:
|
|
# Trying first 10 pages of branches
|
|
for page in range(1, 11):
|
|
branches = _get_gitlab_branches(page)
|
|
for branch_obj in branches:
|
|
if branch_obj["ref"]["name"] == branch_name:
|
|
return branch_obj
|
|
raise ValueError(f"Branch {branch_name} not found")
|
|
|
|
|
|
def _get_pipeline_jobs_info(pipeline_iid: int) -> AnyDict:
|
|
# Getting just the stuff we need - the job names and IDs
|
|
graphql_query = """
|
|
query getJobsFromPipeline($projectPath: ID!, $iid: ID!) {
|
|
project(fullPath: $projectPath) {
|
|
pipeline(iid: $iid) {
|
|
stages {
|
|
nodes {
|
|
groups {
|
|
nodes {
|
|
jobs {
|
|
nodes {
|
|
id
|
|
name
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"""
|
|
query = {
|
|
"query": graphql_query,
|
|
"variables": {
|
|
"projectPath": "satoshilabs/trezor/trezor-firmware",
|
|
"iid": pipeline_iid,
|
|
},
|
|
}
|
|
return requests.post(GRAPHQL_API, json=query).json()
|
|
|
|
|
|
def _yield_pipeline_jobs(pipeline_iid: int) -> Iterator[AnyDict]:
|
|
jobs_info = _get_pipeline_jobs_info(pipeline_iid)
|
|
if SAVE_GRAPHQL_RESULTS: # for development purposes
|
|
with open("jobs_info.json", "w") as f:
|
|
json.dump(jobs_info, f, indent=2)
|
|
stages = jobs_info["data"]["project"]["pipeline"]["stages"]["nodes"]
|
|
for stage in stages:
|
|
nodes = stage["groups"]["nodes"]
|
|
for node in nodes:
|
|
jobs = node["jobs"]["nodes"]
|
|
for job in jobs:
|
|
yield job
|
|
|
|
|
|
def _get_job_ui_fixtures_results(job: AnyDict) -> AnyDict:
|
|
print(f"Checking job {job['name']}")
|
|
job_id = job["id"].split("/")[-1]
|
|
|
|
job_results = TestResult.from_job_id(job_id)
|
|
if job_results.failed:
|
|
print(f"ERROR: Job {job['name']} failed - {job_results}")
|
|
return {}
|
|
|
|
url = f"https://satoshilabs.gitlab.io/-/trezor/trezor-firmware/-/jobs/{job_id}/artifacts/tests/ui_tests/fixtures.results.json"
|
|
response = requests.get(url)
|
|
if response.status_code != 200:
|
|
print("No UI results found")
|
|
return {}
|
|
return response.json()
|
|
|
|
|
|
def get_jobs_of_interest(
|
|
only_jobs: Iterable[str] | None, exclude_jobs: Iterable[str] | None
|
|
) -> Iterable[str]:
|
|
if only_jobs and exclude_jobs:
|
|
raise ValueError("Cannot specify both only_jobs and exclude_jobs")
|
|
if only_jobs:
|
|
return [job for job in UI_JOB_NAMES if job in only_jobs]
|
|
if exclude_jobs:
|
|
return [job for job in UI_JOB_NAMES if job not in exclude_jobs]
|
|
return UI_JOB_NAMES
|
|
|
|
|
|
def get_branch_ui_fixtures_results(
|
|
branch_name: str, jobs_of_interest: Iterable[str] | None = None
|
|
) -> dict[str, AnyDict]:
|
|
print(f"Checking branch {branch_name}")
|
|
|
|
if jobs_of_interest is None:
|
|
jobs_of_interest = UI_JOB_NAMES
|
|
|
|
branch_obj = _get_branch_obj(branch_name)
|
|
pipeline_iid = branch_obj["iid"]
|
|
|
|
def yield_key_value() -> Iterator[tuple[str, AnyDict]]:
|
|
for job in _yield_pipeline_jobs(pipeline_iid):
|
|
for ui_job_name in jobs_of_interest:
|
|
if job["name"] == ui_job_name:
|
|
yield job["name"], _get_job_ui_fixtures_results(job)
|
|
|
|
return dict(yield_key_value())
|