You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
trezor-firmware/core/src/apps/cardano/byron_addresses.py

116 lines
3.7 KiB

from micropython import const
from typing import TYPE_CHECKING
from trezor.crypto import crc
from trezor.wire import ProcessError
from apps.common import cbor
from .helpers import protocol_magics
if TYPE_CHECKING:
from . import seed
_PROTOCOL_MAGIC_KEY = const(2)
"""
This is the legacy implementation of Byron addresses. Byron
addresses should however remain supported in Shelley with
exactly the same implementation, thus it is kept here
with base58 encoding and all the nuances of Byron addresses.
"""
def derive(keychain: seed.Keychain, path: list, protocol_magic: int) -> bytes:
from .helpers.utils import derive_public_key
# get_address_attributes
# protocol magic is included in Byron addresses only on testnets
if protocol_magics.is_mainnet(protocol_magic):
address_attributes = {}
else:
address_attributes = {_PROTOCOL_MAGIC_KEY: cbor.encode(protocol_magic)}
# _get_address_root
extpubkey = derive_public_key(keychain, path, extended=True)
address_root = _address_hash([0, [0, extpubkey], address_attributes])
address_type = 0
address_data = [address_root, address_attributes, address_type]
address_data_encoded = cbor.encode(address_data)
# _encode_raw
return cbor.encode(
[cbor.Tagged(24, address_data_encoded), crc.crc32(address_data_encoded)]
)
def validate(address: bytes, protocol_magic: int) -> None:
address_data_encoded = _decode_raw(address)
_validate_protocol_magic(address_data_encoded, protocol_magic)
def _address_hash(data: list) -> bytes:
from trezor.crypto import hashlib
cbor_data = cbor.encode(data)
sha_data_hash = hashlib.sha3_256(cbor_data).digest()
res = hashlib.blake2b(data=sha_data_hash, outlen=28).digest()
return res
def _decode_raw(address: bytes) -> bytes:
from trezor import log
try:
address_unpacked = cbor.decode(address)
except ValueError as e:
if __debug__:
log.exception(__name__, e)
raise ProcessError("Invalid address")
if not isinstance(address_unpacked, list) or len(address_unpacked) != 2:
raise ProcessError("Invalid address")
address_data_encoded = address_unpacked[0]
if not isinstance(address_data_encoded, bytes):
raise ProcessError("Invalid address")
address_crc = address_unpacked[1]
if not isinstance(address_crc, int):
raise ProcessError("Invalid address")
if address_crc != crc.crc32(address_data_encoded):
raise ProcessError("Invalid address")
return address_data_encoded
def _validate_protocol_magic(address_data_encoded: bytes, protocol_magic: int) -> None:
"""
Determines whether the correct protocol magic (or none)
is included in the address. Addresses on mainnet don't
contain protocol magic, but addresses on the testnet do.
"""
address_data = cbor.decode(address_data_encoded)
if not isinstance(address_data, list) or len(address_data) < 2:
raise ProcessError("Invalid address")
attributes = address_data[1]
if protocol_magics.is_mainnet(protocol_magic):
if _PROTOCOL_MAGIC_KEY in attributes:
raise ProcessError("Output address network mismatch")
else: # testnet
if len(attributes) == 0 or _PROTOCOL_MAGIC_KEY not in attributes:
raise ProcessError("Output address network mismatch")
protocol_magic_cbor = attributes[_PROTOCOL_MAGIC_KEY]
address_protocol_magic = cbor.decode(protocol_magic_cbor)
if not isinstance(address_protocol_magic, int):
raise ProcessError("Invalid address")
if address_protocol_magic != protocol_magic:
raise ProcessError("Output address network mismatch")