# 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 . from collections import defaultdict import os import subprocess import tempfile import time from trezorlib.debuglink import TrezorClientDebugLink from trezorlib.transport import TransportException, get_transport BINDIR = os.path.dirname(os.path.abspath(__file__)) + "/emulators" ENV = {"SDL_VIDEODRIVER": "dummy"} ROOT = os.path.dirname(os.path.abspath(__file__)) + "/../" LOCAL_BUILDS = { "core": ROOT + "core/build/unix/micropython", "legacy": ROOT + "legacy/firmware/trezor.elf", } BIN_DIR = os.path.dirname(os.path.abspath(__file__)) + "/emulators" def check_version(tag, ver_emu): if tag.startswith("v") and len(tag.split(".")) == 3: assert tag == "v" + ".".join(["%d" % i for i in ver_emu]) def check_file(gen, tag): if tag.startswith("/"): filename = tag else: filename = "%s/trezor-emu-%s-%s" % (BIN_DIR, gen, tag) if not os.path.exists(filename): raise ValueError(filename + " not found. Do not forget to build firmware.") def get_tags(): files = os.listdir(BIN_DIR) if not files: raise ValueError( "No files found. Use download_emulators.sh to download emulators." ) result = defaultdict(list) for f in sorted(files): try: _, _, gen, tag = f.split("-", maxsplit=3) result[gen].append(tag) except ValueError: pass return result ALL_TAGS = get_tags() class EmulatorWrapper: def __init__(self, gen, tag, storage=None): self.gen = gen self.tag = tag self.workdir = tempfile.TemporaryDirectory() if storage: open(self._storage_file(), "wb").write(storage) def __enter__(self): if self.tag.startswith("/"): # full path+filename provided args = [self.tag] else: # only gen+tag provided args = ["%s/trezor-emu-%s-%s" % (BINDIR, self.gen, self.tag)] env = ENV if self.gen == "core": args += ["-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 self.client = None self.process = subprocess.Popen( args, cwd=self.workdir.name, env=ENV, stdout=open(os.devnull, "w") ) # wait until emulator is listening i = 0 while True: try: i += 1 if i > 100: self.__exit__(None, None, None) raise RuntimeError("Can't connect to emulator") self.transport = get_transport("udp:127.0.0.1:21324") except TransportException: time.sleep(0.1) continue break self.client = TrezorClientDebugLink(self.transport) self.client.open() # check whether the reported version matches the expected one if self.tag[0] == "v": version = "v%d.%d.%d" % ( self.client.features["major_version"], self.client.features["minor_version"], self.client.features["patch_version"], ) assert self.tag == version, "expected: %s reported: %s" % ( self.tag, version, ) return self def __exit__(self, exc_type, exc_value, traceback): if self.client: self.client.close() self.process.terminate() try: self.process.wait(1) except subprocess.TimeoutExpired: self.process.kill() 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()