mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-15 09:50:57 +00:00
20f3658e7e
[no changelog]
259 lines
8.2 KiB
Python
259 lines
8.2 KiB
Python
import hashlib
|
|
import json
|
|
import re
|
|
import shutil
|
|
from contextlib import contextmanager
|
|
from pathlib import Path
|
|
from typing import Dict, Generator, Optional, Set
|
|
|
|
import pytest
|
|
from _pytest.outcomes import Failed
|
|
from PIL import Image
|
|
|
|
from trezorlib.debuglink import TrezorClientDebugLink as Client
|
|
|
|
from .reporting import testreport
|
|
|
|
UI_TESTS_DIR = Path(__file__).resolve().parent
|
|
SCREENS_DIR = UI_TESTS_DIR / "screens"
|
|
HASH_FILE = UI_TESTS_DIR / "fixtures.json"
|
|
SUGGESTION_FILE = UI_TESTS_DIR / "fixtures.suggestion.json"
|
|
FIXTURES_DIFF = UI_TESTS_DIR / "fixtures.json.diff"
|
|
FILE_HASHES: Dict[str, str] = {}
|
|
ACTUAL_HASHES: Dict[str, str] = {}
|
|
PROCESSED: Set[str] = set()
|
|
FAILED_TESTS: Set[str] = set()
|
|
|
|
# T1/TT, to be set in screen_recording(), as we do not know it beforehand
|
|
# TODO: it is not the cleanest, we could create a class out of this file
|
|
MODEL = ""
|
|
|
|
|
|
def get_test_name(node_id: str) -> str:
|
|
# Test item name is usually function name, but when parametrization is used,
|
|
# parameters are also part of the name. Some functions have very long parameter
|
|
# names (tx hashes etc) that run out of maximum allowable filename length, so
|
|
# we limit the name to first 100 chars. This is not a problem with txhashes.
|
|
new_name = node_id.replace("tests/device_tests/", "")
|
|
# remove ::TestClass:: if present because it is usually the same as the test file name
|
|
new_name = re.sub(r"::.*?::", "-", new_name)
|
|
new_name = new_name.replace("/", "-") # in case there is "/"
|
|
if len(new_name) <= 100:
|
|
return new_name
|
|
return new_name[:91] + "-" + hashlib.sha256(new_name.encode()).hexdigest()[:8]
|
|
|
|
|
|
def _process_recorded(screen_path: Path, test_name: str) -> None:
|
|
# calculate hash
|
|
actual_hash = _hash_files(screen_path)
|
|
FILE_HASHES[test_name] = actual_hash
|
|
ACTUAL_HASHES[test_name] = actual_hash
|
|
_rename_records(screen_path)
|
|
testreport.recorded(screen_path, test_name, actual_hash)
|
|
|
|
|
|
def _rename_records(screen_path: Path) -> None:
|
|
# rename screenshots
|
|
for index, record in enumerate(sorted(screen_path.iterdir())):
|
|
record.replace(screen_path / f"{index:08}.png")
|
|
|
|
|
|
def _hash_files(path: Path) -> str:
|
|
files = path.iterdir()
|
|
hasher = hashlib.sha256()
|
|
for file in sorted(files):
|
|
hasher.update(_get_bytes_from_png(str(file)))
|
|
|
|
return hasher.digest().hex()
|
|
|
|
|
|
def _get_bytes_from_png(png_file: str) -> bytes:
|
|
"""Decode a PNG file into bytes representing all the pixels.
|
|
|
|
Is necessary because Linux and Mac are using different PNG encoding libraries,
|
|
and we need the file hashes to be the same on both platforms.
|
|
"""
|
|
return Image.open(png_file).tobytes()
|
|
|
|
|
|
def _process_tested(fixture_test_path: Path, test_name: str) -> None:
|
|
actual_path = fixture_test_path / "actual"
|
|
actual_hash = _hash_files(actual_path)
|
|
ACTUAL_HASHES[test_name] = actual_hash
|
|
|
|
_rename_records(actual_path)
|
|
|
|
expected_hash = FILE_HASHES.get(test_name)
|
|
if expected_hash is None:
|
|
pytest.fail(f"Hash of {test_name} not found in fixtures.json")
|
|
|
|
if actual_hash != expected_hash:
|
|
assert expected_hash is not None
|
|
file_path = testreport.failed(
|
|
fixture_test_path, test_name, actual_hash, expected_hash
|
|
)
|
|
|
|
# Writing the diff to a file, so that we can process it later
|
|
# Appending a new JSON object, not having to regenerate the
|
|
# whole file (which could cause issues with multiple processes/threads)
|
|
with open(FIXTURES_DIFF, "a") as f:
|
|
diff = {
|
|
"test_name": test_name,
|
|
"actual_hash": actual_hash,
|
|
}
|
|
f.write(json.dumps(diff) + "\n")
|
|
|
|
pytest.fail(
|
|
f"Hash of {test_name} differs.\n"
|
|
f"Expected: {expected_hash}\n"
|
|
f"Actual: {actual_hash}\n"
|
|
f"Diff file: {file_path}"
|
|
)
|
|
else:
|
|
testreport.passed(fixture_test_path, test_name, actual_hash)
|
|
|
|
|
|
def get_last_call_test_result(request: pytest.FixtureRequest) -> Optional[bool]:
|
|
# if test did not finish, e.g. interrupted by Ctrl+C, the pytest_runtest_makereport
|
|
# did not create the attribute we need
|
|
if not hasattr(request.node, "rep_call"):
|
|
return None
|
|
|
|
return request.node.rep_call.passed
|
|
|
|
|
|
@contextmanager
|
|
def screen_recording(
|
|
client: Client, request: pytest.FixtureRequest
|
|
) -> Generator[None, None, None]:
|
|
test_ui = request.config.getoption("ui")
|
|
test_name = get_test_name(request.node.nodeid)
|
|
|
|
# Differentiating test names between T1 and TT
|
|
# Making the model global for other functions
|
|
global MODEL
|
|
MODEL = f"T{client.features.model}"
|
|
|
|
test_name = f"{MODEL}_{test_name}"
|
|
|
|
screens_test_path = SCREENS_DIR / test_name
|
|
|
|
if test_ui == "record":
|
|
screen_path = screens_test_path / "recorded"
|
|
else:
|
|
screen_path = screens_test_path / "actual"
|
|
|
|
if not screens_test_path.exists():
|
|
screens_test_path.mkdir()
|
|
# remove previous files
|
|
shutil.rmtree(screen_path, ignore_errors=True)
|
|
screen_path.mkdir()
|
|
|
|
try:
|
|
client.debug.start_recording(str(screen_path))
|
|
yield
|
|
finally:
|
|
# Wait for response to Initialize, which gives the emulator time to catch up
|
|
# and redraw the homescreen. Otherwise there's a race condition between that
|
|
# and stopping recording.
|
|
client.init_device()
|
|
client.debug.stop_recording()
|
|
|
|
if test_ui:
|
|
PROCESSED.add(test_name)
|
|
if get_last_call_test_result(request) is False:
|
|
FAILED_TESTS.add(test_name)
|
|
|
|
if test_ui == "record":
|
|
_process_recorded(screen_path, test_name)
|
|
else:
|
|
_process_tested(screens_test_path, test_name)
|
|
|
|
|
|
def list_missing() -> Set[str]:
|
|
# Only listing the ones for the current model
|
|
relevant_cases = {
|
|
case for case in FILE_HASHES.keys() if case.startswith(f"{MODEL}_")
|
|
}
|
|
return relevant_cases - PROCESSED
|
|
|
|
|
|
def read_fixtures() -> None:
|
|
if not HASH_FILE.exists():
|
|
raise ValueError("File fixtures.json not found.")
|
|
global FILE_HASHES
|
|
FILE_HASHES = json.loads(HASH_FILE.read_text())
|
|
|
|
|
|
def write_fixtures(remove_missing: bool) -> None:
|
|
HASH_FILE.write_text(_get_fixtures_content(FILE_HASHES, remove_missing))
|
|
|
|
|
|
def write_fixtures_suggestion(
|
|
remove_missing: bool, only_passed_tests: bool = False
|
|
) -> None:
|
|
SUGGESTION_FILE.write_text(
|
|
_get_fixtures_content(ACTUAL_HASHES, remove_missing, only_passed_tests)
|
|
)
|
|
|
|
|
|
def update_fixtures_with_diff() -> int:
|
|
"""Update the fixtures.json file with the actual hashes from the diff file.
|
|
|
|
Use-case is that the UI test run will generate the differing hashes,
|
|
and with this function we can simply update the fixtures.json file
|
|
without having to call the UI tests again in recording mode.
|
|
"""
|
|
if not FIXTURES_DIFF.exists():
|
|
raise ValueError(f"File {FIXTURES_DIFF} not found.")
|
|
|
|
read_fixtures()
|
|
|
|
changes_amount = 0
|
|
with open(FIXTURES_DIFF) as f:
|
|
for line in f:
|
|
changes_amount += 1
|
|
diff = json.loads(line)
|
|
FILE_HASHES[diff["test_name"]] = diff["actual_hash"]
|
|
|
|
write_fixtures(remove_missing=False)
|
|
|
|
# Returning the amount of updated hashes
|
|
return changes_amount
|
|
|
|
|
|
def _get_fixtures_content(
|
|
fixtures: Dict[str, str], remove_missing: bool, only_passed_tests: bool = False
|
|
) -> str:
|
|
if remove_missing:
|
|
# Not removing the ones for different model
|
|
nonrelevant_cases = {
|
|
f: h for f, h in FILE_HASHES.items() if not f.startswith(f"{MODEL}_")
|
|
}
|
|
|
|
filtered_processed_tests = PROCESSED
|
|
if only_passed_tests:
|
|
filtered_processed_tests = PROCESSED - FAILED_TESTS
|
|
|
|
processed_fixtures = {i: fixtures[i] for i in filtered_processed_tests}
|
|
fixtures = {**nonrelevant_cases, **processed_fixtures}
|
|
else:
|
|
fixtures = fixtures
|
|
|
|
return json.dumps(fixtures, indent="", sort_keys=True) + "\n"
|
|
|
|
|
|
def main() -> None:
|
|
read_fixtures()
|
|
for record in SCREENS_DIR.iterdir():
|
|
if not (record / "actual").exists():
|
|
continue
|
|
|
|
try:
|
|
_process_tested(record, record.name)
|
|
print("PASSED:", record.name)
|
|
except Failed:
|
|
print("FAILED:", record.name)
|
|
|
|
testreport.generate_reports()
|