1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2024-11-22 07:28:10 +00:00

tests/ui: move reports and add index.html

This commit is contained in:
Tomas Susanka 2020-01-09 14:25:45 +00:00
parent c604b8f7fa
commit 1103a14c48
5 changed files with 103 additions and 26 deletions

View File

@ -54,8 +54,7 @@ core unix device ui test:
paths: paths:
- trezor.log - trezor.log
- ci/ui_test_records/ - ci/ui_test_records/
- tests/ui_tests/fixtures/*/failure_diff.html - tests/ui_tests/reports/
- tests/ui_tests/fixtures/*/success.html
- tests/junit.xml - tests/junit.xml
when: always when: always
expire_in: 1 week expire_in: 1 week

View File

@ -26,6 +26,7 @@ from trezorlib.transport import enumerate_devices, get_transport
from . import ui_tests from . import ui_tests
from .device_handler import BackgroundDeviceHandler from .device_handler import BackgroundDeviceHandler
from .ui_tests import get_test_name, report
def get_device(): def get_device():
@ -145,6 +146,35 @@ def client(request):
client.close() client.close()
def pytest_sessionstart(session):
if session.config.getoption("ui") == "test":
report.clear_dir()
def pytest_sessionfinish(session, exitstatus):
if session.config.getoption("ui") != "test":
return
reporter = session.config.pluginmanager.get_plugin("terminalreporter")
# intentionally set(), because there are multiple stages for one test in the TestReport items
test_names = {"passed": set(), "failed": set()}
for status, test in reporter.stats.items():
if status in ("deselected", "warnings"):
continue
if status in ("passed", "failed"):
# iterate through the stages to get the test name
for t in test:
test_names[status].add(get_test_name(t.nodeid))
report.index(test_names, exitstatus)
def pytest_terminal_summary(terminalreporter, exitstatus, config):
terminalreporter.writer.line(
"\nUI tests summary: %s" % (report.REPORTS_PATH / "index.html")
)
def pytest_addoption(parser): def pytest_addoption(parser):
parser.addoption( parser.addoption(
"--ui", "--ui",

View File

@ -1,3 +1,4 @@
*.png *.png
*.html *.html
*.zip *.zip
reports/

View File

@ -8,6 +8,8 @@ import pytest
from . import report from . import report
UI_TESTS_DIR = Path(__file__).parent.resolve()
def get_test_name(node_id): def get_test_name(node_id):
# Test item name is usually function name, but when parametrization is used, # Test item name is usually function name, but when parametrization is used,
@ -67,28 +69,24 @@ def _process_tested(fixture_test_path, test_name):
_rename_records(actual_path) _rename_records(actual_path)
if actual_hash != expected_hash: if actual_hash != expected_hash:
file_path = report.failure( file_path = report.failed(
fixture_test_path, test_name, actual_hash, expected_hash fixture_test_path, test_name, actual_hash, expected_hash
) )
if (fixture_test_path / "success.html").exists():
(fixture_test_path / "success.html").unlink()
pytest.fail( pytest.fail(
"Hash of {} differs.\nExpected: {}\nActual: {}\nDiff file: {}".format( "Hash of {} differs.\nExpected: {}\nActual: {}\nDiff file: {}".format(
test_name, expected_hash, actual_hash, file_path test_name, expected_hash, actual_hash, file_path
) )
) )
else: else:
report.success(fixture_test_path, test_name, actual_hash) report.passed(fixture_test_path, test_name, actual_hash)
if (fixture_test_path / "failure_diff.html").exists():
(fixture_test_path / "failure_diff.html").unlink()
@contextmanager @contextmanager
def screen_recording(client, request): def screen_recording(client, request):
test_ui = request.config.getoption("ui") test_ui = request.config.getoption("ui")
test_name = get_test_name(request.node.nodeid) test_name = get_test_name(request.node.nodeid)
fixture_test_path = Path(__file__).parent.resolve() / "fixtures" / test_name fixture_test_path = UI_TESTS_DIR / "fixtures" / test_name
if test_ui == "record": if test_ui == "record":
screen_path = fixture_test_path / "recorded" screen_path = fixture_test_path / "recorded"
@ -97,7 +95,9 @@ def screen_recording(client, request):
else: else:
raise ValueError("Invalid 'ui' option.") raise ValueError("Invalid 'ui' option.")
_check_fixture_directory(fixture_test_path, screen_path) # remove previous files
shutil.rmtree(screen_path, ignore_errors=True)
screen_path.mkdir()
try: try:
client.debug.start_recording(str(screen_path)) client.debug.start_recording(str(screen_path))

View File

@ -1,13 +1,18 @@
import base64 import base64
import filecmp import filecmp
import shutil
from datetime import datetime
from distutils.dir_util import copy_tree from distutils.dir_util import copy_tree
from itertools import zip_longest from itertools import zip_longest
from pathlib import Path
import dominate import dominate
from dominate.tags import div, h1, hr, i, img, p, table, td, th, tr from dominate.tags import a, div, h1, h2, hr, i, img, p, table, td, th, tr
from . import download from . import download
REPORTS_PATH = Path(__file__).parent.resolve() / "reports"
def _image(src): def _image(src):
with td(): with td():
@ -47,20 +52,62 @@ def _write(fixture_test_path, doc, filename):
return fixture_test_path / filename return fixture_test_path / filename
def failure(fixture_test_path, test_name, actual_hash, expected_hash): def _report_links(tests, status):
if status not in ("failed", "passed"):
raise ValueError("Different status than failed/passed is not yet supported.")
if not tests:
i("None!")
return
with table(border=1):
with tr():
th("Link to report")
for test in tests:
with tr():
td(a(test, href=REPORTS_PATH / status / (test + ".html")))
def clear_dir():
# delete and create the reports dir to clear previous entries
shutil.rmtree(REPORTS_PATH, ignore_errors=True)
REPORTS_PATH.mkdir()
(REPORTS_PATH / "failed").mkdir()
(REPORTS_PATH / "passed").mkdir()
def index(tests, status):
title = "UI Test report " + datetime.now().strftime("%Y-%m-%d %H:%M:%S")
doc = dominate.document(title=title)
with doc:
h1("UI Test report")
if status == 0:
p("All tests succeeded!", style="color: green; font-weight: bold;")
else:
p("Some tests failed!", style="color: red; font-weight: bold;")
hr()
h2("Failed", style="color: red;")
_report_links(tests["failed"], "failed")
h2("Passed", style="color: green;")
_report_links(tests["passed"], "passed")
return _write(REPORTS_PATH, doc, "index.html")
def failed(fixture_test_path, test_name, actual_hash, expected_hash):
doc = dominate.document(title=test_name) doc = dominate.document(title=test_name)
recorded_path = fixture_test_path / "recorded" recorded_path = fixture_test_path / "recorded"
actual_path = fixture_test_path / "actual" actual_path = fixture_test_path / "actual"
if not recorded_path.exists(): if not recorded_path.exists():
recorded_path.mkdir() recorded_path.mkdir()
download.fetch_recorded(expected_hash, recorded_path) download.fetch_recorded(expected_hash, recorded_path)
recorded = sorted(recorded_path.iterdir()) recorded_screens = sorted(recorded_path.iterdir())
actual = sorted(actual_path.iterdir()) actual_screens = sorted(actual_path.iterdir())
if not recorded: if not recorded_screens:
return return
with doc: with doc:
@ -71,24 +118,24 @@ def failure(fixture_test_path, test_name, actual_hash, expected_hash):
th("Expected") th("Expected")
th("Actual") th("Actual")
for r, a in zip_longest(recorded, actual): for recorded, actual in zip_longest(recorded_screens, actual_screens):
if r and a and filecmp.cmp(a, r): if recorded and actual and filecmp.cmp(actual, recorded):
background = "white" background = "white"
else: else:
background = "red" background = "red"
with tr(bgcolor=background): with tr(bgcolor=background):
_image(r) _image(recorded)
_image(a) _image(actual)
return _write(fixture_test_path, doc, "failure_diff.html") return _write(REPORTS_PATH / "failed", doc, test_name + ".html")
def success(fixture_test_path, test_name, actual_hash): def passed(fixture_test_path, test_name, actual_hash):
copy_tree(str(fixture_test_path / "actual"), str(fixture_test_path / "recorded")) copy_tree(str(fixture_test_path / "actual"), str(fixture_test_path / "recorded"))
doc = dominate.document(title=test_name) doc = dominate.document(title=test_name)
actual_path = fixture_test_path / "actual" actual_path = fixture_test_path / "actual"
actual = sorted(actual_path.iterdir()) actual_screens = sorted(actual_path.iterdir())
with doc: with doc:
_header(test_name, actual_hash, actual_hash) _header(test_name, actual_hash, actual_hash)
@ -97,8 +144,8 @@ def success(fixture_test_path, test_name, actual_hash):
with tr(): with tr():
th("Recorded") th("Recorded")
for a in actual: for screen in actual_screens:
with tr(): with tr():
_image(a) _image(screen)
return _write(fixture_test_path, doc, "success.html") return _write(REPORTS_PATH / "passed", doc, test_name + ".html")