2018-11-08 17:15:49 +00:00
|
|
|
# This file is part of the Trezor project.
|
|
|
|
#
|
2021-11-26 14:50:43 +00:00
|
|
|
# Copyright (C) 2012-2022 SatoshiLabs and contributors
|
2018-11-08 17:15:49 +00:00
|
|
|
#
|
|
|
|
# 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>.
|
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
import typing as t
|
2021-11-15 12:23:18 +00:00
|
|
|
from hashlib import blake2s
|
2018-10-12 13:58:55 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
from typing_extensions import Protocol, TypeGuard
|
|
|
|
|
|
|
|
from .. import messages
|
|
|
|
from ..tools import expect, session
|
|
|
|
from .core import VendorFirmware
|
|
|
|
from .legacy import LegacyFirmware, LegacyV2Firmware
|
|
|
|
|
|
|
|
# re-exports:
|
|
|
|
if True:
|
|
|
|
# indented block prevents isort from messing with these until we upgrade to 5.x
|
|
|
|
from .consts import * # noqa: F401, F403
|
|
|
|
from .core import * # noqa: F401, F403
|
|
|
|
from .legacy import * # noqa: F401, F403
|
|
|
|
from .util import ( # noqa: F401
|
|
|
|
FirmwareIntegrityError,
|
|
|
|
InvalidSignatureError,
|
|
|
|
Unsigned,
|
2019-12-18 12:12:07 +00:00
|
|
|
)
|
2022-09-06 09:18:05 +00:00
|
|
|
from .vendor import * # noqa: F401, F403
|
2018-06-13 17:04:18 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
if t.TYPE_CHECKING:
|
|
|
|
from ..client import TrezorClient
|
2018-06-13 17:04:18 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
T = t.TypeVar("T", bound="FirmwareType")
|
2020-02-05 11:23:34 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
class FirmwareType(Protocol):
|
|
|
|
@classmethod
|
|
|
|
def parse(cls: t.Type[T], data: bytes) -> T:
|
|
|
|
...
|
2018-06-13 17:04:18 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
def verify(self, public_keys: t.Sequence[bytes] = ()) -> None:
|
|
|
|
...
|
2018-06-13 17:04:18 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
def digest(self) -> bytes:
|
|
|
|
...
|
2018-06-13 17:04:18 +00:00
|
|
|
|
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
def parse(data: bytes) -> "FirmwareType":
|
2018-06-13 17:04:18 +00:00
|
|
|
try:
|
2022-09-06 09:18:05 +00:00
|
|
|
if data[:4] == b"TRZR":
|
|
|
|
return LegacyFirmware.parse(data)
|
|
|
|
elif data[:4] == b"TRZV":
|
|
|
|
return VendorFirmware.parse(data)
|
|
|
|
elif data[:4] == b"TRZF":
|
|
|
|
return LegacyV2Firmware.parse(data)
|
|
|
|
else:
|
|
|
|
raise ValueError("Unrecognized firmware image type")
|
2018-06-13 17:04:18 +00:00
|
|
|
except Exception as e:
|
2019-02-25 18:35:39 +00:00
|
|
|
raise FirmwareIntegrityError("Invalid firmware image") from e
|
2019-12-18 14:44:51 +00:00
|
|
|
|
2019-02-25 18:35:39 +00:00
|
|
|
|
2022-09-06 09:18:05 +00:00
|
|
|
def is_onev2(fw: "FirmwareType") -> TypeGuard[LegacyFirmware]:
|
|
|
|
return isinstance(fw, LegacyFirmware) and fw.embedded_v2 is not None
|
2018-06-13 17:04:18 +00:00
|
|
|
|
|
|
|
|
2018-08-10 14:05:14 +00:00
|
|
|
# ====== Client functions ====== #
|
2018-06-13 17:04:18 +00:00
|
|
|
|
|
|
|
|
feat(python): add full type information
WIP - typing the trezorctl apps
typing functions trezorlib/cli
addressing most of mypy issue for trezorlib apps and _internal folder
fixing broken device tests by changing asserts in debuglink.py
addressing most of mypy issues in trezorlib/cli folder
adding types to some untyped functions, mypy section in setup.cfg
typing what can be typed, some mypy fixes, resolving circular import issues
importing type objects in "if TYPE_CHECKING:" branch
fixing CI by removing assert in emulator, better ignore comments
CI assert fix, style fixes, new config options
fixup! CI assert fix, style fixes, new config options
type fixes after rebasing on master
fixing python3.6 and 3.7 unittests by importing Literal from typing_extensions
couple mypy and style fixes
fixes and improvements from code review
silencing all but one mypy issues
trial of typing the tools.expect function
fixup! trial of typing the tools.expect function
@expect and @session decorators correctly type-checked
Optional args in CLI where relevant, not using general list/tuple/dict where possible
python/Makefile commands, adding them into CI, ignoring last mypy issue
documenting overload for expect decorator, two mypy fixes coming from that
black style fix
improved typing of decorators, pyright config file
addressing or ignoring pyright errors, replacing mypy in CI by pyright
fixing incomplete assert causing device tests to fail
pyright issue that showed in CI but not locally, printing pyright version in CI
fixup! pyright issue that showed in CI but not locally, printing pyright version in CI
unifying type:ignore statements for pyright usage
resolving PIL.Image issues, pyrightconfig not excluding anything
replacing couple asserts with TypeGuard on safe_issubclass
better error handling of usb1 import for webusb
better error handling of hid import
small typing details found out by strict pyright mode
improvements from code review
chore(python): changing List to Sequence for protobuf messages
small code changes to reflect the protobuf change to Sequence
importing TypedDict from typing_extensions to support 3.6 and 3.7
simplify _format_access_list function
fixup! simplify _format_access_list function
typing tools folder
typing helper-scripts folder
some click typing
enforcing all functions to have typed arguments
reverting the changed argument name in tools
replacing TransportType with Transport
making PinMatrixRequest.type protobuf attribute required
reverting the protobuf change, making argument into get_pin Optional
small fixes in asserts
solving the session decorator type issues
fixup! solving the session decorator type issues
improvements from code review
fixing new pyright errors introduced after version increase
changing -> Iterable to -> Sequence in enumerate_devices, change in wait_for_devices
style change in debuglink.py
chore(python): adding type annotation to Sequences in messages.py
better "self and cls" types on Transport
fixup! better "self and cls" types on Transport
fixing some easy things from strict pyright run
2021-11-03 22:12:53 +00:00
|
|
|
@session
|
2021-02-09 12:38:42 +00:00
|
|
|
def update(
|
|
|
|
client: "TrezorClient",
|
|
|
|
data: bytes,
|
2022-09-06 09:18:05 +00:00
|
|
|
progress_update: t.Callable[[int], t.Any] = lambda _: None,
|
2021-02-09 12:38:42 +00:00
|
|
|
):
|
2018-06-13 17:04:18 +00:00
|
|
|
if client.features.bootloader_mode is False:
|
|
|
|
raise RuntimeError("Device must be in bootloader mode")
|
|
|
|
|
2018-10-12 13:58:55 +00:00
|
|
|
resp = client.call(messages.FirmwareErase(length=len(data)))
|
2018-06-13 17:04:18 +00:00
|
|
|
|
|
|
|
# TREZORv1 method
|
2018-10-12 13:58:55 +00:00
|
|
|
if isinstance(resp, messages.Success):
|
|
|
|
resp = client.call(messages.FirmwareUpload(payload=data))
|
2021-02-09 12:38:42 +00:00
|
|
|
progress_update(len(data))
|
2018-10-12 13:58:55 +00:00
|
|
|
if isinstance(resp, messages.Success):
|
|
|
|
return
|
|
|
|
else:
|
2021-09-27 10:13:51 +00:00
|
|
|
raise RuntimeError(f"Unexpected result {resp}")
|
2018-06-13 17:04:18 +00:00
|
|
|
|
2018-10-12 13:58:55 +00:00
|
|
|
# TREZORv2 method
|
|
|
|
while isinstance(resp, messages.FirmwareRequest):
|
feat(python): add full type information
WIP - typing the trezorctl apps
typing functions trezorlib/cli
addressing most of mypy issue for trezorlib apps and _internal folder
fixing broken device tests by changing asserts in debuglink.py
addressing most of mypy issues in trezorlib/cli folder
adding types to some untyped functions, mypy section in setup.cfg
typing what can be typed, some mypy fixes, resolving circular import issues
importing type objects in "if TYPE_CHECKING:" branch
fixing CI by removing assert in emulator, better ignore comments
CI assert fix, style fixes, new config options
fixup! CI assert fix, style fixes, new config options
type fixes after rebasing on master
fixing python3.6 and 3.7 unittests by importing Literal from typing_extensions
couple mypy and style fixes
fixes and improvements from code review
silencing all but one mypy issues
trial of typing the tools.expect function
fixup! trial of typing the tools.expect function
@expect and @session decorators correctly type-checked
Optional args in CLI where relevant, not using general list/tuple/dict where possible
python/Makefile commands, adding them into CI, ignoring last mypy issue
documenting overload for expect decorator, two mypy fixes coming from that
black style fix
improved typing of decorators, pyright config file
addressing or ignoring pyright errors, replacing mypy in CI by pyright
fixing incomplete assert causing device tests to fail
pyright issue that showed in CI but not locally, printing pyright version in CI
fixup! pyright issue that showed in CI but not locally, printing pyright version in CI
unifying type:ignore statements for pyright usage
resolving PIL.Image issues, pyrightconfig not excluding anything
replacing couple asserts with TypeGuard on safe_issubclass
better error handling of usb1 import for webusb
better error handling of hid import
small typing details found out by strict pyright mode
improvements from code review
chore(python): changing List to Sequence for protobuf messages
small code changes to reflect the protobuf change to Sequence
importing TypedDict from typing_extensions to support 3.6 and 3.7
simplify _format_access_list function
fixup! simplify _format_access_list function
typing tools folder
typing helper-scripts folder
some click typing
enforcing all functions to have typed arguments
reverting the changed argument name in tools
replacing TransportType with Transport
making PinMatrixRequest.type protobuf attribute required
reverting the protobuf change, making argument into get_pin Optional
small fixes in asserts
solving the session decorator type issues
fixup! solving the session decorator type issues
improvements from code review
fixing new pyright errors introduced after version increase
changing -> Iterable to -> Sequence in enumerate_devices, change in wait_for_devices
style change in debuglink.py
chore(python): adding type annotation to Sequences in messages.py
better "self and cls" types on Transport
fixup! better "self and cls" types on Transport
fixing some easy things from strict pyright run
2021-11-03 22:12:53 +00:00
|
|
|
assert resp.offset is not None
|
|
|
|
assert resp.length is not None
|
2021-02-09 12:38:42 +00:00
|
|
|
length = resp.length
|
|
|
|
payload = data[resp.offset : resp.offset + length]
|
2019-08-09 12:10:28 +00:00
|
|
|
digest = blake2s(payload).digest()
|
2018-10-12 13:58:55 +00:00
|
|
|
resp = client.call(messages.FirmwareUpload(payload=payload, hash=digest))
|
2021-02-09 12:38:42 +00:00
|
|
|
progress_update(length)
|
2018-10-12 13:58:55 +00:00
|
|
|
|
|
|
|
if isinstance(resp, messages.Success):
|
|
|
|
return
|
|
|
|
else:
|
2021-09-27 10:13:51 +00:00
|
|
|
raise RuntimeError(f"Unexpected message {resp}")
|
2022-04-22 22:19:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
@expect(messages.FirmwareHash, field="hash", ret_type=bytes)
|
2022-09-06 09:18:05 +00:00
|
|
|
def get_hash(client: "TrezorClient", challenge: t.Optional[bytes]):
|
2022-04-22 22:19:07 +00:00
|
|
|
return client.call(messages.GetFirmwareHash(challenge=challenge))
|