1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-15 18:00:59 +00:00

Emulator launcher scripts (#796)

Emulator launcher scripts
This commit is contained in:
matejcik 2020-01-24 16:18:18 +01:00 committed by GitHub
commit 75264a07a8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
26 changed files with 753 additions and 471 deletions

View File

@ -93,6 +93,12 @@ core unix regular build:
script: script:
- cd core - cd core
- pipenv run make build_unix - pipenv run make build_unix
artifacts:
name: "$CI_JOB_NAME-$CI_COMMIT_SHORT_SHA"
paths:
- core/build/unix/micropython
- core/src/trezor/res/resources.py
expire_in: 1 week
core unix frozen regular build: core unix frozen regular build:
stage: build stage: build
@ -104,13 +110,13 @@ core unix frozen regular build:
name: "$CI_JOB_NAME-$CI_COMMIT_SHORT_SHA" name: "$CI_JOB_NAME-$CI_COMMIT_SHORT_SHA"
paths: paths:
- core/build/unix/micropython - core/build/unix/micropython
- core/src/trezor/res/resources.py
expire_in: 1 week expire_in: 1 week
core unix frozen btconly build: core unix frozen btconly debug build:
stage: build stage: build
<<: *only_changes_core <<: *only_changes_core
variables: variables:
PYOPT: "0"
BITCOIN_ONLY: "1" BITCOIN_ONLY: "1"
script: script:
- cd core - cd core
@ -120,7 +126,6 @@ core unix frozen btconly build:
name: "$CI_JOB_NAME-$CI_COMMIT_SHORT_SHA" name: "$CI_JOB_NAME-$CI_COMMIT_SHORT_SHA"
paths: paths:
- core/build/unix/micropython-bitcoinonly - core/build/unix/micropython-bitcoinonly
- core/src/trezor/res/resources.py
expire_in: 1 week expire_in: 1 week
core unix frozen debug build: core unix frozen debug build:

View File

@ -33,7 +33,7 @@ core unix unit test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix regular build
script: script:
- cd core - cd core
- pipenv run make test - pipenv run make test
@ -42,20 +42,19 @@ core unix device ui test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix frozen debug build
script: script:
- cd core - cd core
- pipenv run make test_emu_ui - pipenv run make test_emu_ui
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
- cd ../ci - cd ../ci
- pipenv run python prepare_ui_artifacts.py - pipenv run python prepare_ui_artifacts.py
artifacts: artifacts:
name: core-unix-device-ui-test name: core-unix-device-ui-test
paths: paths:
- trezor.log
- ci/ui_test_records/ - ci/ui_test_records/
- tests/ui_tests/reports/ - tests/ui_tests/reports/
- tests/junit.xml - tests/junit.xml
- tests/trezor.log
when: always when: always
expire_in: 1 week expire_in: 1 week
reports: reports:
@ -65,20 +64,19 @@ core unix device test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix frozen debug build
variables: variables:
TREZOR_PROFILING: 1 TREZOR_PROFILING: 1
script: script:
- cd core - cd core
- pipenv run make test_emu - pipenv run make test_emu
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
- sync - sync
- sleep 1 - sleep 1
- mv ./src/.coverage .coverage.test_emu - mv ./src/.coverage .coverage.test_emu
artifacts: artifacts:
name: core-unix-device-test name: core-unix-device-test
paths: paths:
- trezor.log - tests/trezor.log
- tests/junit.xml - tests/junit.xml
- core/.coverage.* - core/.coverage.*
expire_in: 1 week expire_in: 1 week
@ -90,18 +88,17 @@ core unix btconly device test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen btconly build - core unix frozen btconly debug build
variables: variables:
MICROPYTHON: "../build/unix/micropython-bitcoinonly" MICROPYTHON: "build/unix/micropython-bitcoinonly"
TREZOR_PYTEST_SKIP_ALTCOINS: 1 TREZOR_PYTEST_SKIP_ALTCOINS: 1
script: script:
- cd core - cd core
- pipenv run make test_emu - pipenv run make test_emu
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
artifacts: artifacts:
name: core-unix-btconly-device-test name: core-unix-btconly-device-test
paths: paths:
- trezor.log - tests/trezor.log
- tests/junit.xml - tests/junit.xml
expire_in: 1 week expire_in: 1 week
when: always when: always
@ -112,20 +109,19 @@ core unix monero test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix frozen debug build
variables: variables:
TREZOR_PROFILING: 1 TREZOR_PROFILING: 1
script: script:
- cd core - cd core
- pipenv run make test_emu_monero - pipenv run make test_emu_monero
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
- sync - sync
- sleep 1 - sleep 1
- mv ./src/.coverage .coverage.test_emu_monero - mv ./src/.coverage .coverage.test_emu_monero
artifacts: artifacts:
name: core-unix-monero-test name: core-unix-monero-test
paths: paths:
- trezor.log - tests/trezor.log
- core/.coverage.* - core/.coverage.*
expire_in: 1 week expire_in: 1 week
when: always when: always
@ -134,21 +130,20 @@ core unix u2f test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix frozen debug build
variables: variables:
TREZOR_PROFILING: 1 TREZOR_PROFILING: 1
script: script:
- make -C tests/fido_tests/u2f-tests-hid - make -C tests/fido_tests/u2f-tests-hid
- cd core - cd core
- pipenv run make test_emu_u2f - pipenv run make test_emu_u2f
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
- sync - sync
- sleep 1 - sleep 1
- mv ./src/.coverage .coverage.test_emu_u2f - mv ./src/.coverage .coverage.test_emu_u2f
artifacts: artifacts:
name: core-unix-u2f-test name: core-unix-u2f-test
paths: paths:
- trezor.log - tests/trezor.log
- core/.coverage.* - core/.coverage.*
expire_in: 1 week expire_in: 1 week
when: always when: always
@ -157,20 +152,19 @@ core unix fido2 test:
stage: test stage: test
<<: *only_changes_core <<: *only_changes_core
dependencies: dependencies:
- core unix frozen regular build - core unix frozen debug build
variables: variables:
TREZOR_PROFILING: 1 TREZOR_PROFILING: 1
script: script:
- cd core - cd core
- pipenv run make test_emu_fido2 - pipenv run make test_emu_fido2
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
- sync - sync
- sleep 1 - sleep 1
- mv ./src/.coverage .coverage.test_emu_fido2 - mv ./src/.coverage .coverage.test_emu_fido2
artifacts: artifacts:
name: core-unix-fido2-test name: core-unix-fido2-test
paths: paths:
- trezor.log - tests/trezor.log
- tests/junit.xml - tests/junit.xml
- core/.coverage.* - core/.coverage.*
expire_in: 1 week expire_in: 1 week
@ -186,11 +180,10 @@ core unix click test:
script: script:
- cd core - cd core
- pipenv run make test_emu_click - pipenv run make test_emu_click
- cp /var/tmp/trezor.log ${CI_PROJECT_DIR}
artifacts: artifacts:
name: core-unix-click-test name: core-unix-click-test
paths: paths:
- trezor.log - tests/trezor.log
- tests/junit.xml - tests/junit.xml
reports: reports:
junit: tests/junit.xml junit: tests/junit.xml

View File

@ -38,6 +38,17 @@ FIRMWARE_MAXSIZE = 1703936
GITREV=$(shell git describe --always --dirty | tr '-' '_') GITREV=$(shell git describe --always --dirty | tr '-' '_')
CFLAGS += -DGITREV=$(GITREV) CFLAGS += -DGITREV=$(GITREV)
TESTPATH = $(CURDIR)/../tests
EMU = $(CURDIR)/emu.py
EMU_LOG_FILE ?= $(TESTPATH)/trezor.log
EMU_TEST_ARGS = --disable-animation --headless --output=$(EMU_LOG_FILE) --temporary-profile
EMU_TEST = $(EMU) $(EMU_TEST_ARGS) -c
JUNIT_XML ?= $(TESTPATH)/junit.xml
PYTEST = pytest --junitxml=$(JUNIT_XML)
TREZOR_FIDO2_UDP_PORT = 21326
## help commands: ## help commands:
help: ## show this help help: ## show this help
@ -57,7 +68,7 @@ run: ## run unix port
cd src ; ../$(UNIX_BUILD_DIR)/micropython cd src ; ../$(UNIX_BUILD_DIR)/micropython
emu: ## run emulator emu: ## run emulator
./emu.sh $(EMU)
## test commands: ## test commands:
@ -65,25 +76,27 @@ test: ## run unit tests
cd tests ; ./run_tests.sh $(TESTOPTS) cd tests ; ./run_tests.sh $(TESTOPTS)
test_emu: ## run selected device tests from python-trezor test_emu: ## run selected device tests from python-trezor
cd tests ; ./run_tests_device_emu.sh $(TESTOPTS) $(EMU_TEST) $(PYTEST) $(TESTPATH)/device_tests $(TESTOPTS)
test_emu_monero: ## run selected monero device tests from monero-agent test_emu_monero: ## run selected monero device tests from monero-agent
cd tests ; ./run_tests_device_emu_monero.sh $(TESTOPTS) cd tests ; ./run_tests_device_emu_monero.sh $(TESTOPTS)
test_emu_u2f: ## run selected u2f device tests from u2f-tests-hid test_emu_u2f: ## run selected u2f device tests from u2f-tests-hid
cd tests ; ./run_tests_device_emu_u2f.sh $(TESTOPTS) $(EMU_TEST) --slip0014 $(TESTPATH)/fido_tests/u2f-tests-hid/HIDTest $(TREZOR_FIDO2_UDP_PORT) $(TESTOPTS)
$(EMU_TEST) --slip0014 $(TESTPATH)/fido_tests/u2f-tests-hid/U2FTest $(TREZOR_FIDO2_UDP_PORT) $(TESTOPTS)
test_emu_fido2: ## run fido2 device tests test_emu_fido2: ## run fido2 device tests
cd tests ; ./run_tests_device_emu_fido2.sh $(TESTOPTS) cd $(TESTPATH)/fido_tests/fido2 ; \
$(EMU_TEST) $(PYTEST) --sim tests/standard/ --vendor trezor $(TESTOPTS)
test_emu_click: ## run click tests test_emu_click: ## run click tests
cd tests ; ./run_tests_click_emu.sh $(TESTOPTS) $(EMU_TEST) $(PYTEST) $(TESTPATH)/click_tests $(TESTOPTS)
test_emu_ui: ## run ui integration tests test_emu_ui: ## run ui integration tests
cd tests ; ./run_tests_device_emu.sh --ui=test -m "not skip_ui" $(TESTOPTS) $(EMU_TEST) $(PYTEST) $(TESTPATH)/device_tests --ui=test -m "not skip_ui" $(TESTOPTS)
test_emu_ui_record: ## record and hash screens for ui integration tests test_emu_ui_record: ## record and hash screens for ui integration tests
cd tests ; ./run_tests_device_emu.sh --ui=record -m "not skip_ui" $(TESTOPTS) $(EMU_TEST) $(PYTEST) $(TESTPATH)/device_tests --ui=record -m "not skip_ui" $(TESTOPTS)
pylint: ## run pylint on application sources and tests pylint: ## run pylint on application sources and tests
pylint -E $(shell find src tests -name *.py) pylint -E $(shell find src tests -name *.py)

View File

@ -692,6 +692,9 @@ MP_NOINLINE int main_(int argc, char **argv) {
return ret & 0xff; return ret & 0xff;
} }
#ifdef TREZOR_EMULATOR_FROZEN
uint mp_import_stat(const char *path) { return MP_IMPORT_STAT_NO_EXIST; }
#else
uint mp_import_stat(const char *path) { uint mp_import_stat(const char *path) {
struct stat st; struct stat st;
if (stat(path, &st) == 0) { if (stat(path, &st) == 0) {
@ -703,6 +706,7 @@ uint mp_import_stat(const char *path) {
} }
return MP_IMPORT_STAT_NO_EXIST; return MP_IMPORT_STAT_NO_EXIST;
} }
#endif
void nlr_jump_fail(void *val) { void nlr_jump_fail(void *val) {
printf("FATAL: uncaught NLR %p\n", val); printf("FATAL: uncaught NLR %p\n", val);

269
core/emu.py Executable file
View File

@ -0,0 +1,269 @@
#!/usr/bin/env python3
import gzip
import os
import platform
import signal
import subprocess
import sys
import tempfile
import time
from pathlib import Path
import click
import trezorlib.debuglink
import trezorlib.device
from trezorlib._internal.emulator import CoreEmulator
try:
import inotify.adapters
except ImportError:
inotify = None
HERE = Path(__file__).parent.resolve()
MICROPYTHON = HERE / "build" / "unix" / "micropython"
SRC_DIR = HERE / "src"
SD_CARD_GZ = HERE / "trezor.sdcard.gz"
PROFILING_WRAPPER = HERE / "prof" / "prof.py"
PROFILE_BASE = Path.home() / ".trezoremu"
def run_command_with_emulator(emulator, command):
with emulator:
# first start the subprocess
process = subprocess.Popen(command)
# After the subprocess is started, ignore SIGINT in parent
# (so that we don't need to handle KeyboardInterrupts)
signal.signal(signal.SIGINT, signal.SIG_IGN)
# SIGINTs will be sent to all children by the OS, so we should be able to safely
# wait for their exit.
return process.wait()
def run_emulator(emulator):
with emulator:
signal.signal(signal.SIGINT, signal.SIG_IGN)
return emulator.wait()
def watch_emulator(emulator):
watch = inotify.adapters.InotifyTree(str(SRC_DIR))
try:
for _, type_names, _, _ in watch.event_gen(yield_nones=False):
if "IN_CLOSE_WRITE" in type_names:
click.echo("Restarting...")
emulator.restart()
except KeyboardInterrupt:
emulator.stop()
return 0
def run_debugger(emulator):
os.chdir(emulator.workdir)
env = emulator.make_env()
if platform.system() == "Darwin":
env["PATH"] = "/usr/bin"
os.execvpe(
"lldb",
["lldb", "-f", emulator.executable, "--"] + emulator.make_args(),
env,
)
else:
os.execvpe(
"gdb", ["gdb", "--args", emulator.executable] + emulator.make_args(), env
)
@click.command(context_settings=dict(ignore_unknown_options=True))
# fmt: off
@click.option("-a", "--disable-animation", is_flag=True, default=os.environ.get("TREZOR_DISABLE_ANIMATION") == "1", help="Disable animation")
@click.option("-c", "--command", "run_command", is_flag=True, help="Run command while emulator is running")
@click.option("-d", "--production", is_flag=True, default=os.environ.get("PYOPT") == "1", help="Production mode (debuglink disabled)")
@click.option("-D", "--debugger", is_flag=True, help="Run emulator in debugger (gdb/lldb)")
@click.option("--executable", type=click.Path(exists=True, dir_okay=False), default=os.environ.get("MICROPYTHON"), help="Alternate emulator executable")
@click.option("-g", "--profiling", is_flag=True, default=os.environ.get("TREZOR_PROFILING"), help="Run with profiler wrapper")
@click.option("-h", "--headless", is_flag=True, help="Headless mode (no display)")
@click.option("--heap-size", metavar="SIZE", default="20M", help="Configure heap size")
@click.option("--main", help="Path to python main file")
@click.option("--mnemonic", "mnemonics", multiple=True, help="Initialize device with given mnemonic. Specify multiple times for Shamir shares.")
@click.option("--log-memory", is_flag=True, default=os.environ.get("TREZOR_LOG_MEMORY") == "1", help="Print memory usage after workflows")
@click.option("-o", "--output", type=click.File("w"), default="-", help="Redirect emulator output to file")
@click.option("-p", "--profile", metavar="NAME", help="Profile name or path")
@click.option("-P", "--port", metavar="PORT", type=int, default=int(os.environ.get("TREZOR_UDP_PORT", 0)) or None, help="UDP port number")
@click.option("-q", "--quiet", is_flag=True, help="Silence emulator output")
@click.option("-s", "--slip0014", is_flag=True, help="Initialize device with SLIP-14 seed (all all all...)")
@click.option("-t", "--temporary-profile", is_flag=True, help="Create an empty temporary profile")
@click.option("-w", "--watch", is_flag=True, help="Restart emulator if sources change")
@click.option("-X", "--extra-arg", "extra_args", multiple=True, help="Extra argument to pass to micropython")
# fmt: on
@click.argument("command", nargs=-1, type=click.UNPROCESSED)
def cli(
disable_animation,
run_command,
production,
debugger,
executable,
profiling,
headless,
heap_size,
main,
mnemonics,
log_memory,
profile,
port,
output,
quiet,
slip0014,
temporary_profile,
watch,
extra_args,
command,
):
"""Run the trezor-core emulator.
If -c is specified, extra arguments are treated as a command that is executed with
the running emulator. This command can access the following environment variables:
\b
TREZOR_PROFILE_DIR - path to storage directory
TREZOR_PATH - trezorlib connection string
TREZOR_UDP_PORT - UDP port on which the emulator listens
TREZOR_FIDO2_UDP_PORT - UDP port for FIDO2
By default, emulator output goes to stdout. If silenced with -q, it is redirected
to $TREZOR_PROFILE_DIR/trezor.log. You can also specify a custom path with -o.
"""
if executable:
executable = Path(executable)
else:
executable = MICROPYTHON
if command and not run_command:
raise click.ClickException("Extra arguments found. Did you mean to use -c?")
if watch and (command or debugger or frozen):
raise click.ClickException("Cannot use -w together with -c or -D or -F")
if watch and inotify is None:
raise click.ClickException("inotify module is missing, install with pip")
if main and profiling:
raise click.ClickException("Cannot use --main and -g together")
if slip0014 and mnemonics:
raise click.ClickException("Cannot use -s and --mnemonic together")
if slip0014:
mnemonics = [" ".join(["all"] * 12)]
if mnemonics and debugger:
raise click.ClickException("Cannot load mnemonics when running in debugger")
if mnemonics and production:
raise click.ClickException("Cannot load mnemonics in production mode")
if profiling:
main_args = [str(PROFILING_WRAPPER)]
elif main:
main_args = [main]
else:
main_args = ["-m", "main"]
if profile and temporary_profile:
raise click.ClickException("Cannot use -p and -t together")
tempdir = None
if profile:
if "/" in profile:
profile_dir = Path(profile)
else:
profile_dir = PROFILE_BASE / profile
elif temporary_profile:
tempdir = tempfile.TemporaryDirectory(prefix="trezor-emulator-")
profile_dir = Path(tempdir.name)
# unpack empty SD card for faster start-up
with gzip.open(SD_CARD_GZ, "rb") as gz:
(profile_dir / "trezor.sdcard").write_bytes(gz.read())
elif "TREZOR_PROFILE_DIR" in os.environ:
profile_dir = Path(os.environ["TREZOR_PROFILE_DIR"])
else:
profile_dir = Path("/var/tmp")
if quiet:
output = None
emulator = CoreEmulator(
executable,
profile_dir,
logfile=output,
port=port,
headless=headless,
debug=not production,
extra_args=extra_args,
main_args=main_args,
heap_size=heap_size,
disable_animation=disable_animation,
workdir=SRC_DIR,
)
emulator_env = dict(
TREZOR_PATH=f"udp:127.0.0.1:{emulator.port}",
TREZOR_PROFILE_DIR=str(profile_dir.resolve()),
TREZOR_UDP_PORT=str(emulator.port),
TREZOR_FIDO2_UDP_PORT=str(emulator.port + 2),
TREZOR_SRC=str(SRC_DIR),
)
os.environ.update(emulator_env)
for k, v in emulator_env.items():
click.echo(f"{k}={v}")
if log_memory:
os.environ["TREZOR_LOG_MEMORY"] = "1"
if debugger:
run_debugger(emulator)
raise RuntimeError("run_debugger should not return")
click.echo("Waiting for emulator to come up... ", err=True)
start = time.monotonic()
emulator.start()
end = time.monotonic()
click.echo(f"Emulator ready after {end - start:.3f} seconds", err=True)
if mnemonics:
if slip0014:
label = "SLIP-0014"
elif profile:
label = profile_dir.name
else:
label = "Emulator"
trezorlib.device.wipe(emulator.client)
trezorlib.debuglink.load_device(
emulator.client,
mnemonics,
pin=None,
passphrase_protection=False,
label=label,
)
if run_command:
ret = run_command_with_emulator(emulator, command)
elif watch:
ret = watch_emulator(emulator)
else:
ret = run_emulator(emulator)
if tempdir is not None:
tempdir.cleanup()
sys.exit(ret)
if __name__ == "__main__":
cli()

View File

@ -1,35 +1,10 @@
#!/usr/bin/env bash #!/bin/sh
PYOPT="${PYOPT:-1}"
MICROPYTHON="${MICROPYTHON:-${PWD}/build/unix/micropython}" if [ -n "$1" ]; then
TREZOR_SRC=$(cd "${PWD}/src/"; pwd) echo "This is just a compatibility wrapper. Use emu.py if you want features."
BROWSER="${BROWSER:-chromium}" exit 1
source ./trezor_cmd.sh
cd "${TREZOR_SRC}"
case "$1" in
"-d")
shift
OPERATING_SYSTEM=$(uname)
if [ "$OPERATING_SYSTEM" = "Darwin" ]; then
PATH=/usr/bin /usr/bin/lldb -f $MICROPYTHON -- $ARGS $* $MAIN
else
gdb --args $MICROPYTHON $ARGS $* $MAIN
fi fi
;;
"-r") cd src
shift ../build/unix/micropython -O$PYOPT -X heapsize=20M -m main
while true; do
$MICROPYTHON $ARGS $* $MAIN &
UPY_PID=$!
find -name '*.py' | inotifywait -q -e close_write --fromfile -
echo Restarting ...
kill $UPY_PID
done
;;
*)
echo "Starting emulator: $MICROPYTHON $ARGS $* $MAIN"
$MICROPYTHON $ARGS $* $MAIN 2>&1 | tee "${TREZOR_LOGFILE}"
exit ${PIPESTATUS[0]}
esac

View File

@ -1,10 +1,13 @@
import sys import sys
import uos
from uio import open from uio import open
from uos import getenv
sys.path.insert(0, uos.getenv("TREZOR_SRC")) # We need to insert "" to sys.path so that the frozen build can import main from the
del uos # frozen modules, and regular build can import it from current directory.
sys.path.insert(0, "")
PATH_PREFIX = (getenv("TREZOR_SRC") or ".") + "/"
class Coverage: class Coverage:
@ -22,7 +25,7 @@ class Coverage:
this_file = globals()["__file__"] this_file = globals()["__file__"]
for filename in self.__files: for filename in self.__files:
if not filename == this_file: if not filename == this_file:
lines[filename] = list(self.__files[filename]) lines[PATH_PREFIX + filename] = list(self.__files[filename])
return lines_execution return lines_execution

View File

@ -38,7 +38,7 @@ if __debug__:
current_content = None # type: Optional[List[str]] current_content = None # type: Optional[List[str]]
def screenshot() -> bool: def screenshot() -> bool:
if utils.SAVE_SCREEN or save_screen: if save_screen:
ui.display.save(save_screen_directory + "/refresh-") ui.display.save(save_screen_directory + "/refresh-")
return True return True
return False return False

View File

@ -19,13 +19,9 @@ if __debug__:
if EMULATOR: if EMULATOR:
import uos import uos
TEST = int(uos.getenv("TREZOR_TEST") or "0")
DISABLE_ANIMATION = int(uos.getenv("TREZOR_DISABLE_ANIMATION") or "0") DISABLE_ANIMATION = int(uos.getenv("TREZOR_DISABLE_ANIMATION") or "0")
SAVE_SCREEN = int(uos.getenv("TREZOR_SAVE_SCREEN") or "0")
LOG_MEMORY = int(uos.getenv("TREZOR_LOG_MEMORY") or "0") LOG_MEMORY = int(uos.getenv("TREZOR_LOG_MEMORY") or "0")
else: else:
TEST = 0
SAVE_SCREEN = 0
LOG_MEMORY = 0 LOG_MEMORY = 0
if False: if False:

View File

@ -1,37 +0,0 @@
#!/usr/bin/env bash
: "${RUN_TEST_EMU:=1}"
CORE_DIR="$(SHELL_SESSION_FILE='' && cd "$( dirname "${BASH_SOURCE[0]}" )/.." >/dev/null 2>&1 && pwd )"
MICROPYTHON="${MICROPYTHON:-$CORE_DIR/build/unix/micropython}"
TREZOR_SRC="${CORE_DIR}/src"
PYOPT="${PYOPT:-0}"
upy_pid=""
# run emulator if RUN_TEST_EMU
if [[ $RUN_TEST_EMU > 0 ]]; then
source ../trezor_cmd.sh
# remove flash and sdcard files before run to prevent inconsistent states
mv "${TREZOR_PROFILE_DIR}/trezor.flash" "${TREZOR_PROFILE_DIR}/trezor.flash.bkp" 2>/dev/null
mv "${TREZOR_PROFILE_DIR}/trezor.sdcard" "${TREZOR_PROFILE_DIR}/trezor.sdcard.bkp" 2>/dev/null
cd "${TREZOR_SRC}"
echo "Starting emulator: $MICROPYTHON $ARGS ${MAIN}"
TREZOR_TEST=1 \
TREZOR_DISABLE_ANIMATION=1 \
$MICROPYTHON $ARGS "${MAIN}" &> "${TREZOR_LOGFILE}" &
upy_pid=$!
cd -
sleep 30
fi
# run tests
error=0
if ! pytest --junitxml=../../tests/junit.xml ../../tests/device_tests "$@"; then
error=1
fi
kill $upy_pid
exit $error

View File

@ -1,40 +0,0 @@
#!/usr/bin/env bash
: "${RUN_TEST_EMU:=1}"
CORE_DIR="$(SHELL_SESSION_FILE='' && cd "$( dirname "${BASH_SOURCE[0]}" )/.." >/dev/null 2>&1 && pwd )"
MICROPYTHON="${MICROPYTHON:-$CORE_DIR/build/unix/micropython}"
TREZOR_SRC="${CORE_DIR}/src"
DISABLE_ANIMATION=1
PYOPT="${PYOPT:-0}"
upy_pid=""
# run emulator if RUN_TEST_EMU
if [[ $RUN_TEST_EMU > 0 ]]; then
source ../trezor_cmd.sh
# remove flash and sdcard files before run to prevent inconsistent states
mv "${TREZOR_PROFILE_DIR}/trezor.flash" "${TREZOR_PROFILE_DIR}/trezor.flash.bkp" 2>/dev/null
mv "${TREZOR_PROFILE_DIR}/trezor.sdcard" "${TREZOR_PROFILE_DIR}/trezor.sdcard.bkp" 2>/dev/null
cd "${TREZOR_SRC}"
echo "Starting emulator: $MICROPYTHON $ARGS ${MAIN}"
TREZOR_TEST=1 \
TREZOR_DISABLE_ANIMATION=$DISABLE_ANIMATION \
$MICROPYTHON $ARGS "${MAIN}" &> "${TREZOR_LOGFILE}" &
upy_pid=$!
cd -
sleep 30
fi
cd ../../tests/fido_tests/fido2
# run tests
error=0
export TREZOR_FIDO2_UDP_PORT=21326
if ! pytest --junitxml=../../tests/junit.xml --sim tests/standard/ --vendor trezor "$@"; then
error=1
fi
kill $upy_pid
exit $error

View File

@ -5,30 +5,22 @@
: "${RUN_TEST_EMU:=1}" : "${RUN_TEST_EMU:=1}"
CORE_DIR="$(SHELL_SESSION_FILE='' && cd "$( dirname "${BASH_SOURCE[0]}" )/.." >/dev/null 2>&1 && pwd )" CORE_DIR="$(SHELL_SESSION_FILE='' && cd "$( dirname "${BASH_SOURCE[0]}" )/.." >/dev/null 2>&1 && pwd )"
MICROPYTHON="${MICROPYTHON:-$CORE_DIR/build/unix/micropython}"
TREZOR_SRC="${CORE_DIR}/src"
DISABLE_ANIMATION=1
PYOPT="${PYOPT:-0}"
upy_pid="" upy_pid=""
# run emulator if RUN_TEST_EMU # run emulator if RUN_TEST_EMU
if [[ $RUN_TEST_EMU > 0 ]]; then if [[ $RUN_TEST_EMU > 0 ]]; then
source ../trezor_cmd.sh t=$(mktemp)
../emu.py \
# remove flash and sdcard files before run to prevent inconsistent states --disable-animation \
mv "${TREZOR_PROFILE_DIR}/trezor.flash" "${TREZOR_PROFILE_DIR}/trezor.flash.bkp" 2>/dev/null --temporary-profile \
mv "${TREZOR_PROFILE_DIR}/trezor.sdcard" "${TREZOR_PROFILE_DIR}/trezor.sdcard.bkp" 2>/dev/null --headless \
--output=../../tests/trezor.log \
cd "${TREZOR_SRC}" > $t &
echo "Starting emulator: $MICROPYTHON $ARGS ${MAIN}" trezorctl wait-for-emulator
source $t
TREZOR_TEST=1 \ upy_pid=$(cat $TREZOR_PROFILE_DIR/trezor.pid)
TREZOR_DISABLE_ANIMATION=$DISABLE_ANIMATION \ rm $t
$MICROPYTHON $ARGS "${MAIN}" &> "${TREZOR_LOGFILE}" &
upy_pid=$!
cd -
sleep 30
fi fi
DOCKER_ID="" DOCKER_ID=""

View File

@ -1,46 +0,0 @@
#!/usr/bin/env bash
: "${RUN_TEST_EMU:=1}"
CORE_DIR="$(SHELL_SESSION_FILE='' && cd "$( dirname "${BASH_SOURCE[0]}" )/.." >/dev/null 2>&1 && pwd )"
MICROPYTHON="${MICROPYTHON:-$CORE_DIR/build/unix/micropython}"
TREZOR_SRC="${CORE_DIR}/src"
DISABLE_ANIMATION=1
PYOPT="${PYOPT:-0}"
upy_pid=""
# run emulator if RUN_TEST_EMU
if [[ $RUN_TEST_EMU > 0 ]]; then
source ../trezor_cmd.sh
# remove flash and sdcard files before run to prevent inconsistent states
mv "${TREZOR_PROFILE_DIR}/trezor.flash" "${TREZOR_PROFILE_DIR}/trezor.flash.bkp" 2>/dev/null
mv "${TREZOR_PROFILE_DIR}/trezor.sdcard" "${TREZOR_PROFILE_DIR}/trezor.sdcard.bkp" 2>/dev/null
cd "${TREZOR_SRC}"
echo "Starting emulator: $MICROPYTHON $ARGS ${MAIN}"
TREZOR_TEST=1 \
TREZOR_DISABLE_ANIMATION=$DISABLE_ANIMATION \
$MICROPYTHON $ARGS "${MAIN}" &> "${TREZOR_LOGFILE}" &
upy_pid=$!
cd -
sleep 30
fi
# run tests
error=0
TREZOR_FIDO2_UDP_PORT=21326
# missuse loaddevice test to initialize the device
if ! pytest ../../tests/device_tests -k "test_msg_loaddevice" "$@"; then
error=1
fi
if ! ../../tests/fido_tests/u2f-tests-hid/HIDTest "${TREZOR_FIDO2_UDP_PORT}" "$@"; then
error=1
fi
if ! ../../tests/fido_tests/u2f-tests-hid/U2FTest "${TREZOR_FIDO2_UDP_PORT}" "$@"; then
error=1
fi
kill $upy_pid
exit $error

BIN
core/trezor.sdcard.gz Normal file

Binary file not shown.

View File

@ -1,78 +0,0 @@
#!/usr/bin/env bash
# expected inputs:
# TREZOR_SRC -- directory containing python code for uMP
if [[ ! "${TREZOR_SRC}" ]]; then echo "expecting TREZOR_SRC"; exit 0; fi
# optional inputs:
# TREZOR_PROFILE -- profile name (directory) in ~/.trezoremu or full path
# TREZOR_PROFILING -- wrap the uMP/python in the profiler script
# outputs:
## uMP
# PYOPT
# HEAPSIZE
# ARGS -- uMP arguments
# MAIN -- uMP file to execute
## Trezor core
# TREZOR_PROFILE_DIR
# TREZOR_PROFILE_NAME
# TREZOR_UDP_PORT
## this script
# TREZOR_SRC
# TREZOR_LOGFILE
## python-trezor
# TREZOR_PATH -- connect string
# defaults
PYOPT="${PYOPT:-1}"
HEAPSIZE="${HEAPSIZE:-20M}"
TREZOR_PROFILE="${TREZOR_PROFILE:-/var/tmp}"
TREZOR_PROFILE_DIR="${TREZOR_PROFILE}"
TREZOR_PROFILE_NAME="${TREZOR_PROFILE}"
# for profile names create profile directory if not existent
if ! [[ "$TREZOR_PROFILE" == "/"* ]]; then
TREZOR_PROFILE_DIR="${HOME}/.trezoremu/${TREZOR_PROFILE}"
if ! [[ -d "${TREZOR_PROFILE_DIR}" ]]; then
mkdir -p "${TREZOR_PROFILE_DIR}"
PORT=$(( ( RANDOM % 1000 ) + 1 + 21324 ))
echo "# autogenerated config" > "${TREZOR_PROFILE_DIR}/emu.config"
echo "TREZOR_UDP_PORT=\"\${TREZOR_UDP_PORT:-${PORT}}\"" >> "${TREZOR_PROFILE_DIR}/emu.config"
fi
fi
# load profile config
if [[ -f "${TREZOR_PROFILE_DIR}/emu.config" ]]; then
source "${TREZOR_PROFILE_DIR}/emu.config"
fi
# for profiling wrap
if [[ "$TREZOR_PROFILING" -gt 0 ]]; then
MAIN="${TREZOR_SRC}/../prof/prof.py"
else
MAIN="${TREZOR_SRC}/main.py"
fi
TREZOR_LOGFILE="${TREZOR_PROFILE_DIR}/trezor.log"
TREZOR_UDP_PORT="${TREZOR_UDP_PORT:-21324}"
TREZOR_PATH="${TREZOR_PATH:-udp:127.0.0.1:${TREZOR_UDP_PORT}}"
echo "Trezor^emu profile name: ${TREZOR_PROFILE_NAME}"
echo "Trezor^emu profile directory: ${TREZOR_PROFILE_DIR}"
echo "Trezor^emu log file: ${TREZOR_LOGFILE}"
echo "Trezor^emu UDP port: ${TREZOR_UDP_PORT}"
echo "Trezor^emu path: ${TREZOR_PATH}"
echo "Trezor^emu src: ${TREZOR_SRC}"
export TREZOR_PROFILE_NAME="${TREZOR_PROFILE_NAME}"
export TREZOR_PROFILE_DIR="${TREZOR_PROFILE_DIR}"
export TREZOR_LOGFILE="${TREZOR_LOGFILE}"
export TREZOR_UDP_PORT="${TREZOR_UDP_PORT}"
export TREZOR_PATH="${TREZOR_PATH}"
export TREZOR_SRC="${TREZOR_SRC}"
ARGS="-O${PYOPT} -X heapsize=${HEAPSIZE}"

View File

@ -59,7 +59,7 @@ pipenv run make build_unix
Now you can start the emulator: Now you can start the emulator:
```sh ```sh
./emu.sh ./emu.py
``` ```
The emulator has a number of interesting features all documented in the [Emulator](../emulator/index.md) section. The emulator has a number of interesting features all documented in the [Emulator](../emulator/index.md) section.

View File

@ -11,81 +11,117 @@ Emulator significantly speeds up development and has several features to help yo
## How to run ## How to run
1. [build](../build/emulator.md) the emulator 1. [build](../build/emulator.md) the emulator
2. run `emu.sh` 2. run `emu.py` inside the pipenv environment:
- either enter `pipenv shell` first, and then use `./emu.py`
- or always use `pipenv run ./emu.py`
3. to use [bridge](https://github.com/trezor/trezord-go) with the emulator support, start it with `trezord -e 21324` 3. to use [bridge](https://github.com/trezor/trezord-go) with the emulator support, start it with `trezord -e 21324`
Now you can use the emulator the same way as you use the device, for example you can visit our Wallet (https://wallet.trezor.io), use our Python CLI tool (`trezorctl`) etc. Simply click to emulate screen touches. Now you can use the emulator the same way as you use the device, for example you can visit our Wallet (https://wallet.trezor.io), use our Python CLI tool (`trezorctl`) etc. Simply click to emulate screen touches.
## Features ## Features
### Debug mode Run `./emu.py --help` to see all supported command line options and shortcuts. The
sections below only list long option names and most notable features.
To allow debug link (to run tests), see exceptions and log output, run emulator with `PYOPT=0 ./emu.sh`. To properly distinguish the debug mode from production there is a tiny red square in the top right corner. The debug mode is obviously disabled on production firmwares. ### Debug and production mode
By default the emulator runs in debug mode. Debuglink is available (on port 21325 by
default), exceptions and log output goes to console. To indicate debug mode, there is a
red square in the upper right corner of Trezor screen.
![emulator](emulator-debug.png) ![emulator](emulator-debug.png)
To enable production mode, run `./emu.py --production`, or set environment variable `PYOPT=1`.
### Initialize with mnemonic words ### Initialize with mnemonic words
If the debug mode is enabled, you can load the device with any recovery seed directly from the console. This feature is otherwise disabled. To enter seed use `trezorctl`: In debug mode, the emulator can be pre-configured with a mnemonic phrase.
To use a specific mnemonic phrase:
```sh ```sh
trezorctl -m "your mnemonic words" ./emu.py --mnemonic "such deposit very security much theme..."
``` ```
or to use the "all all all" seed defined in [SLIP-14](https://github.com/satoshilabs/slips/blob/master/slip-0014.md): When using Shamir shares, repeat the `--mnemonic` option:
```sh ```sh
trezorctl -s ./emu.py --mnemonic "your first share" --mnemonic "your second share" ...
``` ```
Shamir Backup is also supported: To use the "all all all" seed defined in [SLIP-14](https://github.com/satoshilabs/slips/blob/master/slip-0014.md):
```sh ```sh
trezorctl -m "share 1 words" -m "share 2 words" ./emu.py --slip0014
``` ```
### Storage and Profiles ### Storage and Profiles
Internal Trezor's storage is emulated and stored in the `/var/tmp/trezor.flash` file on default. Deleting this file is similar to calling _wipe device_. You can also find `/var/tmp/trezor.sdcard` for SD card and `/var/tmp/trezor.log`, which contains the communication log, the same as is in the emulator's stdout. Internal Trezor's storage is emulated and stored in the `/var/tmp/trezor.flash` file on
default. Deleting this file is similar to calling _wipe device_. You can also find
`/var/tmp/trezor.sdcard` for SD card.
To run emulator with different files set the environment variable **TREZOR_PROFILE** like so: You can specify a different location for the storage and log files via the `-p` /
`--profile` option:
```sh ```sh
TREZOR_PROFILE=foobar ./emu.sh ./emu.py -p foobar
``` ```
This will create a profile directory in your home ``` ~/.trezoremu/foobar``` containing emulator run files. Alternatively you can set a full path like so: This will create a profile directory in your home `~/.trezoremu/foobar` containing
emulator run files. Alternatively you can set a full path like so:
```sh ```sh
TREZOR_PROFILE=/var/tmp/foobar ./emu.sh ./emu.py -p /var/tmp/foobar
``` ```
You can also set a full profile path to `TREZOR_PROFILE_DIR` environment variable.
Specifying `-t` / `--temporary-profile` will start the emulator in a clean temporary
profile that will be erased when the emulator stops. This is useful, e.g., for tests.
### Logging
By default, emulator output goes to stdout. When silenced with `--quiet`, it is
redirected to `${TREZOR_PROFILE_DIR}/trezor.log`. You can specify an alternate output
file with `--output`.
### Running subcommands with the emulator
In scripts, it is often necessary to start the emulator, run a commmand while it is
available, and then stop it. The following command runs the device test suite using the
emulator:
```sh
./emu.py --command pytest ../tests/device_tests
```
### Profiling support
Run `./emu.py --profiling`, or set environment variable `TREZOR_PROFILING=1`, to run the
emulator with a profiling wrapper that generates statistics of executed lines.
### Memory statistics
Run `./emu.py --log-memory`, or set environment variable `TREZOR_LOG_MEMORY=1`, to dump
memory usage information after each workflow task is finished.
### Run in gdb ### Run in gdb
Running `emu.sh` with `-d` runs emulator inside gdb/lldb. Running `./emu.py --debugger` runs emulator inside gdb/lldb.
### Watch for file changes ### Watch for file changes
Running `emu.sh` with `-r` watches for file changes and reloads the emulator if any occur. Note that this does not do rebuild, i.e. this works for MicroPython code (which is interpreted) but if you make C changes, you need to rebuild your self. Running `./emu.py --watch` watches for file changes and reloads the emulator if any
occur. Note that this does not do rebuild, i.e. this works for MicroPython code (which
is interpreted) but if you make C changes, you need to rebuild yourself.
### Print screen ### Print screen
Press `p` on your keyboard to capture emulator's screen. You will find a png screenshot in the `src` directory. Press `p` on your keyboard to capture emulator's screen. You will find a png screenshot
in the `src` directory.
### Environment Variables ### Disable animation
#### Auto print screen Run `./emu.py --disable-animation`, or set environment variable
`TREZOR_DISABLE_ANIMATION=1` to disable all animations.
If ``` TREZOR_SAVE_SCREEN=1 ``` is set, the emulator makes print screen on every screen change.
#### Memory statistics
If ```TREZOR_LOG_MEMORY=1``` is set, the emulator prints memory usage information after each workflow task is finished.
#### Disable animations
```TREZOR_DISABLE_ANIMATION=1``` disables fading and other animations, which speeds up the UI workflows significantly (useful for tests). This is also requirement for UI integration tests.
#### Tests
```TREZOR_TEST``` informs whether device tests are to be run. Currently unused.

View File

@ -25,10 +25,9 @@ environment:
pipenv shell pipenv shell
``` ```
If you want to test against the emulator, run it in a separate terminal from the `core` If you want to test against the emulator, run it in a separate terminal:
subdirectory:
```sh ```sh
PYOPT=0 ./emu.sh ./core/emu.py
``` ```
Now you can run the test suite with `pytest` from the root directory: Now you can run the test suite with `pytest` from the root directory:

View File

@ -6,6 +6,13 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
_At the moment, the project does **not** adhere to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). That is expected to change with version 1.0._ _At the moment, the project does **not** adhere to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). That is expected to change with version 1.0._
## [0.11.7] - Unreleased
### Added
- built-in functionality of UdpTransport to wait until an emulator comes up, and the
related command `trezorctl wait-for-emulator`
## [0.11.6] - 2019-12-30 ## [0.11.6] - 2019-12-30
[0.11.6]: https://github.com/trezor/trezor-firmware/compare/python/v0.11.5...python/v0.11.6 [0.11.6]: https://github.com/trezor/trezor-firmware/compare/python/v0.11.5...python/v0.11.6

View File

@ -0,0 +1,237 @@
# 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 os
import subprocess
import time
from pathlib import Path
from trezorlib.debuglink import TrezorClientDebugLink
from trezorlib.transport.udp import UdpTransport
def _rm_f(path):
try:
path.unlink()
except FileNotFoundError:
pass
class Emulator:
STORAGE_FILENAME = None
def __init__(
self,
executable,
profile_dir,
*,
logfile=None,
storage=None,
headless=False,
debug=True,
extra_args=()
):
self.executable = Path(executable).resolve()
if not executable.exists():
raise ValueError(
"emulator executable not found: {}".format(self.executable)
)
self.profile_dir = Path(profile_dir).resolve()
if not self.profile_dir.exists():
self.profile_dir.mkdir(parents=True)
elif not self.profile_dir.is_dir():
raise ValueError("profile_dir is not a directory")
self.workdir = self.profile_dir
self.storage = self.profile_dir / self.STORAGE_FILENAME
if storage:
self.storage.write_bytes(storage)
if logfile:
self.logfile = logfile
else:
self.logfile = self.profile_dir / "trezor.log"
self.client = None
self.process = None
self.port = 21324
self.headless = headless
self.debug = debug
self.extra_args = list(extra_args)
def make_args(self):
return []
def make_env(self):
return os.environ.copy()
def _get_transport(self):
return UdpTransport("127.0.0.1:{}".format(self.port))
def wait_until_ready(self, timeout=30):
transport = self._get_transport()
transport.open()
start = time.monotonic()
try:
while True:
if transport._ping():
break
if self.process.poll() is not None:
raise RuntimeError("Emulator proces died")
elapsed = time.monotonic() - start
if elapsed >= timeout:
raise RuntimeError("Can't connect to emulator")
time.sleep(0.1)
finally:
transport.close()
def wait(self, timeout=None):
ret = self.process.wait(timeout=None)
self.stop()
return ret
def launch_process(self):
args = self.make_args()
env = self.make_env()
if hasattr(self.logfile, "write"):
output = self.logfile
else:
output = open(self.logfile, "w")
return subprocess.Popen(
[self.executable] + args + self.extra_args,
cwd=self.workdir,
stdout=output,
stderr=subprocess.STDOUT,
env=env,
)
def start(self):
if self.process:
if self.process.poll() is not None:
# process has died, stop and start again
self.stop()
else:
# process is running, no need to start again
return
self.process = self.launch_process()
self.wait_until_ready()
(self.profile_dir / "trezor.pid").write_text(str(self.process.pid) + "\n")
(self.profile_dir / "trezor.port").write_text(str(self.port) + "\n")
transport = self._get_transport()
self.client = TrezorClientDebugLink(transport, auto_interact=self.debug)
self.client.open()
def stop(self):
if self.client:
self.client.close()
self.client = None
if self.process:
self.process.terminate()
try:
self.process.wait(1)
except subprocess.TimeoutExpired:
self.process.kill()
_rm_f(self.profile_dir / "trezor.pid")
_rm_f(self.profile_dir / "trezor.port")
self.process = None
def restart(self):
self.stop()
self.start()
def __enter__(self):
return self
def __exit__(self, exc_type, exc_value, traceback):
self.stop()
def get_storage(self):
return self.storage.read_bytes()
class CoreEmulator(Emulator):
STORAGE_FILENAME = "trezor.flash"
def __init__(
self,
*args,
port=None,
main_args=("-m", "main"),
workdir=None,
sdcard=None,
disable_animation=True,
heap_size="20M",
**kwargs
):
super().__init__(*args, **kwargs)
if workdir is not None:
self.workdir = Path(workdir).resolve()
self.sdcard = self.profile_dir / "trezor.sdcard"
if sdcard is not None:
self.sdcard.write_bytes(sdcard)
if port:
self.port = port
self.disable_animation = disable_animation
self.main_args = list(main_args)
self.heap_size = heap_size
def make_env(self):
env = super().make_env()
env.update(
TREZOR_PROFILE_DIR=str(self.profile_dir),
TREZOR_PROFILE=str(self.profile_dir),
TREZOR_UDP_PORT=str(self.port),
)
if self.headless:
env["SDL_VIDEODRIVER"] = "dummy"
if self.disable_animation:
env["TREZOR_DISABLE_FADE"] = "1"
env["TREZOR_DISABLE_ANIMATION"] = "1"
return env
def make_args(self):
pyopt = "-O0" if self.debug else "-O1"
return (
[pyopt, "-X", "heapsize={}".format(self.heap_size)]
+ self.main_args
+ self.extra_args
)
class LegacyEmulator(Emulator):
STORAGE_FILENAME = "emulator.img"
def make_env(self):
env = super().make_env()
if self.headless:
env["SDL_VIDEODRIVER"] = "dummy"
return env

View File

@ -19,12 +19,14 @@
import json import json
import os import os
import sys import sys
import time
import click import click
from .. import coins, log, messages, protobuf, ui from .. import coins, log, messages, protobuf, ui
from ..client import TrezorClient from ..client import TrezorClient
from ..transport import enumerate_devices, get_transport from ..transport import enumerate_devices, get_transport
from ..transport.udp import UdpTransport
from . import ( from . import (
binance, binance,
btc, btc,
@ -180,7 +182,7 @@ def print_result(res, path, verbose, is_json):
click.echo("%s: %s" % (k, v)) click.echo("%s: %s" % (k, v))
elif isinstance(res, protobuf.MessageType): elif isinstance(res, protobuf.MessageType):
click.echo(protobuf.format_message(res)) click.echo(protobuf.format_message(res))
else: elif res is not None:
click.echo(res) click.echo(res)
@ -250,6 +252,28 @@ def usb_reset():
WebUsbTransport.enumerate(usb_reset=True) WebUsbTransport.enumerate(usb_reset=True)
@cli.command()
@click.option("-t", "--timeout", type=float, default=10, help="Timeout in seconds")
@click.pass_context
def wait_for_emulator(ctx, timeout):
"""Wait until Trezor Emulator comes up.
Tries to connect to emulator and returns when it succeeds.
"""
path = ctx.parent.params.get("path")
if path:
if not path.startswith("udp:"):
raise click.ClickException("You must use UDP path, not {}".format(path))
path = path.replace("udp:", "")
start = time.monotonic()
UdpTransport(path).wait_until_ready(timeout)
end = time.monotonic()
if ctx.parent.params.get("verbose"):
click.echo("Waited for {:.3f} seconds".format(end - start))
# #
# Basic coin functions # Basic coin functions
# #

View File

@ -15,6 +15,7 @@
# If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>. # If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>.
import socket import socket
import time
from typing import Iterable, Optional, cast from typing import Iterable, Optional, cast
from . import TransportException from . import TransportException
@ -60,7 +61,7 @@ class UdpTransport(ProtocolBasedTransport):
return d return d
else: else:
raise TransportException( raise TransportException(
"No Trezor device found at address {}".format(path) "No Trezor device found at address {}".format(d.get_path())
) )
finally: finally:
d.close() d.close()
@ -84,6 +85,22 @@ class UdpTransport(ProtocolBasedTransport):
path = path.replace("{}:".format(cls.PATH_PREFIX), "") path = path.replace("{}:".format(cls.PATH_PREFIX), "")
return cls._try_path(path) return cls._try_path(path)
def wait_until_ready(self, timeout: float = 10) -> None:
try:
self.open()
self.socket.settimeout(0)
start = time.monotonic()
while True:
if self._ping():
break
elapsed = time.monotonic() - start
if elapsed >= timeout:
raise TransportException("Timed out waiting for connection.")
time.sleep(0.05)
finally:
self.close()
def open(self) -> None: def open(self) -> None:
self.socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) self.socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
self.socket.connect(self.device) self.socket.connect(self.device)

View File

@ -15,23 +15,22 @@
# If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>. # If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>.
import gzip import gzip
import os
import subprocess
import tempfile import tempfile
import time
from collections import defaultdict from collections import defaultdict
from pathlib import Path
from trezorlib.debuglink import TrezorClientDebugLink from trezorlib._internal.emulator import CoreEmulator, LegacyEmulator
from trezorlib.transport.udp import UdpTransport
ROOT = Path(__file__).parent.parent.resolve()
BINDIR = ROOT / "tests" / "emulators"
ROOT = os.path.abspath(os.path.dirname(__file__) + "/..")
BINDIR = ROOT + "/tests/emulators"
LOCAL_BUILD_PATHS = { LOCAL_BUILD_PATHS = {
"core": ROOT + "/core/build/unix/micropython", "core": ROOT / "core" / "build" / "unix" / "micropython",
"legacy": ROOT + "/legacy/firmware/trezor.elf", "legacy": ROOT / "legacy" / "firmware" / "trezor.elf",
} }
SD_CARD_GZ = ROOT + "/tests/trezor.sdcard.gz" CORE_SRC_DIR = ROOT / "core" / "src"
SD_CARD_GZ = ROOT / "core" / "trezor.sdcard.gz"
ENV = {"SDL_VIDEODRIVER": "dummy"} ENV = {"SDL_VIDEODRIVER": "dummy"}
@ -44,11 +43,11 @@ def check_version(tag, version_tuple):
def filename_from_tag(gen, tag): def filename_from_tag(gen, tag):
return f"{BINDIR}/trezor-emu-{gen}-{tag}" return BINDIR / f"trezor-emu-{gen}-{tag}"
def get_tags(): def get_tags():
files = os.listdir(BINDIR) files = list(BINDIR.iterdir())
if not files: if not files:
raise ValueError( raise ValueError(
"No files found. Use download_emulators.sh to download emulators." "No files found. Use download_emulators.sh to download emulators."
@ -58,7 +57,7 @@ def get_tags():
for f in sorted(files): for f in sorted(files):
try: try:
# example: "trezor-emu-core-v2.1.1" # example: "trezor-emu-core-v2.1.1"
_, _, gen, tag = f.split("-", maxsplit=3) _, _, gen, tag = f.name.split("-", maxsplit=3)
result[gen].append(tag) result[gen].append(tag)
except ValueError: except ValueError:
pass pass
@ -69,116 +68,40 @@ ALL_TAGS = get_tags()
class EmulatorWrapper: class EmulatorWrapper:
def __init__(self, gen, tag=None, executable=None, storage=None): def __init__(self, gen, tag=None, storage=None):
self.gen = gen if tag is not None:
self.tag = tag executable = filename_from_tag(gen, tag)
if executable is not None:
self.executable = executable
elif tag is not None:
self.executable = filename_from_tag(gen, tag)
else: else:
self.executable = LOCAL_BUILD_PATHS[gen] executable = LOCAL_BUILD_PATHS[gen]
if not os.path.exists(self.executable): if not executable.exists():
raise ValueError(f"emulator executable not found: {self.executable}") raise ValueError(f"emulator executable not found: {executable}")
self.workdir = tempfile.TemporaryDirectory() self.profile_dir = tempfile.TemporaryDirectory()
if storage: if executable == LOCAL_BUILD_PATHS["core"]:
open(self._storage_file(), "wb").write(storage) workdir = CORE_SRC_DIR
else:
workdir = None
if gen == "legacy":
self.emulator = LegacyEmulator(
executable, self.profile_dir.name, storage=storage, headless=True,
)
elif gen == "core":
with gzip.open(SD_CARD_GZ, "rb") as gz: with gzip.open(SD_CARD_GZ, "rb") as gz:
with open(self.workdir.name + "/trezor.sdcard", "wb") as sd: self.emulator = CoreEmulator(
sd.write(gz.read()) executable,
self.profile_dir.name,
self.client = None storage=storage,
workdir=workdir,
def _get_params_core(self): sdcard=gz.read(),
env = ENV.copy() headless=True,
args = [self.executable, "-m", "main"]
# for firmware 2.1.2 and newer
env["TREZOR_PROFILE_DIR"] = self.workdir.name
# for firmware 2.1.1 and older
env["TREZOR_PROFILE"] = self.workdir.name
if self.executable == LOCAL_BUILD_PATHS["core"]:
cwd = ROOT + "/core/src"
else:
cwd = self.workdir.name
return env, args, cwd
def _get_params_legacy(self):
env = ENV.copy()
args = [self.executable]
cwd = self.workdir.name
return env, args, cwd
def _get_params(self):
if self.gen == "core":
return self._get_params_core()
elif self.gen == "legacy":
return self._get_params_legacy()
else:
raise ValueError("Unknown gen")
def start(self):
env, args, cwd = self._get_params()
self.process = subprocess.Popen(
args, cwd=cwd, env=env, stdout=open(os.devnull, "w")
) )
# wait until emulator is listening
transport = UdpTransport("127.0.0.1:21324")
transport.open()
for _ in range(300):
if transport._ping():
break
if self.process.poll() is not None:
self._cleanup()
raise RuntimeError("Emulator proces died")
time.sleep(0.1)
else:
# could not connect after 300 attempts * 0.1s = 30s of waiting
self._cleanup()
raise RuntimeError("Can't connect to emulator")
transport.close()
self.client = TrezorClientDebugLink(transport)
self.client.open()
check_version(self.tag, self.client.version)
def stop(self):
if self.client:
self.client.close()
self.process.terminate()
try:
self.process.wait(1)
except subprocess.TimeoutExpired:
self.process.kill()
def restart(self):
self.stop()
self.start()
def __enter__(self): def __enter__(self):
self.start() self.emulator.start()
return self return self.emulator
def __exit__(self, exc_type, exc_value, traceback): def __exit__(self, exc_type, exc_value, traceback):
self._cleanup() self.emulator.stop()
self.profile_dir.cleanup()
def _cleanup(self):
self.stop()
self.workdir.cleanup()
def _storage_file(self):
if self.gen == "legacy":
return self.workdir.name + "/emulator.img"
elif self.gen == "core":
return self.workdir.name + "/trezor.flash"
else:
raise ValueError("Unknown gen")
def storage(self):
return open(self._storage_file(), "rb").read()

View File

@ -28,12 +28,11 @@ from ..upgrade_tests import core_only
@pytest.fixture @pytest.fixture
def emulator(): def emulator():
emu = EmulatorWrapper("core") with EmulatorWrapper("core") as emu:
with emu:
yield emu yield emu
def _restart(device_handler: BackgroundDeviceHandler, emulator: EmulatorWrapper): def _restart(device_handler, emulator):
device_handler.restart(emulator) device_handler.restart(emulator)
return device_handler.debuglink() return device_handler.debuglink()

View File

@ -2,7 +2,7 @@ import os
import pytest import pytest
from ..emulators import EmulatorWrapper from ..emulators import LOCAL_BUILD_PATHS
SELECTED_GENS = [ SELECTED_GENS = [
gen.strip() for gen in os.environ.get("TREZOR_UPGRADE_TEST", "").split(",") if gen gen.strip() for gen in os.environ.get("TREZOR_UPGRADE_TEST", "").split(",") if gen
@ -15,17 +15,8 @@ if SELECTED_GENS:
else: else:
# if no selection was provided, select those for which we have emulators # if no selection was provided, select those for which we have emulators
try: LEGACY_ENABLED = LOCAL_BUILD_PATHS["legacy"].exists()
EmulatorWrapper("legacy") CORE_ENABLED = LOCAL_BUILD_PATHS["core"].exists()
LEGACY_ENABLED = True
except Exception:
LEGACY_ENABLED = False
try:
EmulatorWrapper("core")
CORE_ENABLED = True
except Exception:
CORE_ENABLED = False
legacy_only = pytest.mark.skipif( legacy_only = pytest.mark.skipif(

View File

@ -96,7 +96,7 @@ def test_upgrade_load(gen, from_tag, to_tag):
) )
device_id = emu.client.features.device_id device_id = emu.client.features.device_id
asserts(from_tag, emu.client) asserts(from_tag, emu.client)
storage = emu.storage() storage = emu.get_storage()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu: with EmulatorWrapper(gen, to_tag, storage=storage) as emu:
assert device_id == emu.client.features.device_id assert device_id == emu.client.features.device_id
@ -128,7 +128,7 @@ def test_upgrade_reset(gen, from_tag, to_tag):
device_id = emu.client.features.device_id device_id = emu.client.features.device_id
asserts(from_tag, emu.client) asserts(from_tag, emu.client)
address = btc.get_address(emu.client, "Bitcoin", PATH) address = btc.get_address(emu.client, "Bitcoin", PATH)
storage = emu.storage() storage = emu.get_storage()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu: with EmulatorWrapper(gen, to_tag, storage=storage) as emu:
assert device_id == emu.client.features.device_id assert device_id == emu.client.features.device_id
@ -162,7 +162,7 @@ def test_upgrade_reset_skip_backup(gen, from_tag, to_tag):
device_id = emu.client.features.device_id device_id = emu.client.features.device_id
asserts(from_tag, emu.client) asserts(from_tag, emu.client)
address = btc.get_address(emu.client, "Bitcoin", PATH) address = btc.get_address(emu.client, "Bitcoin", PATH)
storage = emu.storage() storage = emu.get_storage()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu: with EmulatorWrapper(gen, to_tag, storage=storage) as emu:
assert device_id == emu.client.features.device_id assert device_id == emu.client.features.device_id
@ -196,7 +196,7 @@ def test_upgrade_reset_no_backup(gen, from_tag, to_tag):
device_id = emu.client.features.device_id device_id = emu.client.features.device_id
asserts(from_tag, emu.client) asserts(from_tag, emu.client)
address = btc.get_address(emu.client, "Bitcoin", PATH) address = btc.get_address(emu.client, "Bitcoin", PATH)
storage = emu.storage() storage = emu.get_storage()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu: with EmulatorWrapper(gen, to_tag, storage=storage) as emu:
assert device_id == emu.client.features.device_id assert device_id == emu.client.features.device_id
@ -222,7 +222,7 @@ def test_upgrade_shamir_recovery(gen, from_tag, to_tag):
assert "2 more shares" in layout.text assert "2 more shares" in layout.text
device_id = emu.client.features.device_id device_id = emu.client.features.device_id
storage = emu.storage() storage = emu.get_storage()
device_handler.check_finalize() device_handler.check_finalize()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu, BackgroundDeviceHandler( with EmulatorWrapper(gen, to_tag, storage=storage) as emu, BackgroundDeviceHandler(
@ -258,7 +258,7 @@ def test_upgrade_u2f(gen, from_tag, to_tag):
counter = fido.get_next_counter(emu.client) counter = fido.get_next_counter(emu.client)
assert counter == 11 assert counter == 11
storage = emu.storage() storage = emu.get_storage()
with EmulatorWrapper(gen, to_tag, storage=storage) as emu: with EmulatorWrapper(gen, to_tag, storage=storage) as emu:
counter = fido.get_next_counter(emu.client) counter = fido.get_next_counter(emu.client)