2019-01-28 14:44:18 +00:00
|
|
|
from ubinascii import unhexlify
|
|
|
|
|
|
|
|
from trezor import wire
|
|
|
|
|
2019-01-15 13:10:46 +00:00
|
|
|
from apps.common import HARDENED, paths
|
|
|
|
from apps.ethereum import networks
|
2018-08-30 11:45:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
"""
|
2019-01-15 13:10:46 +00:00
|
|
|
We believe Ethereum should use 44'/60'/a' for everything, because it is
|
2018-08-30 11:45:09 +00:00
|
|
|
account-based, rather than UTXO-based. Unfortunately, lot of Ethereum
|
|
|
|
tools (MEW, Metamask) do not use such scheme and set a = 0 and then
|
|
|
|
iterate the address index i. Therefore for compatibility reasons we use
|
|
|
|
the same scheme: 44'/60'/0'/0/i and only the i is being iterated.
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2019-01-15 13:10:46 +00:00
|
|
|
def validate_path_for_get_public_key(path: list) -> bool:
|
|
|
|
"""
|
|
|
|
This should be 44'/60'/0', but other non-hardened items are allowed.
|
|
|
|
"""
|
|
|
|
length = len(path)
|
|
|
|
if length < 3 or length > 5:
|
|
|
|
return False
|
|
|
|
if path[0] != 44 | HARDENED:
|
|
|
|
return False
|
|
|
|
if path[1] not in networks.all_slip44_ids_hardened():
|
|
|
|
return False
|
|
|
|
if path[2] != 0 | HARDENED:
|
|
|
|
return False
|
|
|
|
if length > 3 and paths.is_hardened(path[3]):
|
|
|
|
return False
|
|
|
|
if length > 4 and paths.is_hardened(path[4]):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2018-08-30 11:45:09 +00:00
|
|
|
def validate_full_path(path: list) -> bool:
|
|
|
|
"""
|
|
|
|
Validates derivation path to equal 44'/60'/0'/0/i,
|
|
|
|
where `i` is an address index from 0 to 1 000 000.
|
|
|
|
"""
|
|
|
|
if len(path) != 5:
|
|
|
|
return False
|
|
|
|
if path[0] != 44 | HARDENED:
|
|
|
|
return False
|
2019-01-15 13:10:46 +00:00
|
|
|
if path[1] not in networks.all_slip44_ids_hardened():
|
2018-08-30 11:45:09 +00:00
|
|
|
return False
|
|
|
|
if path[2] != 0 | HARDENED:
|
|
|
|
return False
|
|
|
|
if path[3] != 0:
|
|
|
|
return False
|
|
|
|
if path[4] > 1000000:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2019-01-28 14:44:18 +00:00
|
|
|
def address_from_bytes(address_bytes: bytes, network=None) -> str:
|
|
|
|
"""
|
|
|
|
Converts address in bytes to a checksummed string as defined
|
|
|
|
in https://github.com/ethereum/EIPs/blob/master/EIPS/eip-55.md
|
|
|
|
"""
|
2018-08-30 11:45:09 +00:00
|
|
|
from ubinascii import hexlify
|
|
|
|
from trezor.crypto.hashlib import sha3_256
|
|
|
|
|
|
|
|
rskip60 = network is not None and network.rskip60
|
|
|
|
|
2019-01-28 14:44:18 +00:00
|
|
|
hx = hexlify(address_bytes).decode()
|
2018-08-30 11:45:09 +00:00
|
|
|
|
|
|
|
prefix = str(network.chain_id) + "0x" if rskip60 else ""
|
|
|
|
hs = sha3_256(prefix + hx, keccak=True).digest()
|
|
|
|
h = ""
|
|
|
|
|
|
|
|
for i in range(20):
|
|
|
|
l = hx[i * 2]
|
|
|
|
if hs[i] & 0x80 and l >= "a" and l <= "f":
|
|
|
|
l = l.upper()
|
|
|
|
h += l
|
|
|
|
l = hx[i * 2 + 1]
|
|
|
|
if hs[i] & 0x08 and l >= "a" and l <= "f":
|
|
|
|
l = l.upper()
|
|
|
|
h += l
|
|
|
|
|
|
|
|
return "0x" + h
|
2019-01-28 14:44:18 +00:00
|
|
|
|
|
|
|
|
2019-01-29 14:35:09 +00:00
|
|
|
def bytes_from_address(address: str) -> bytes:
|
2019-01-28 14:44:18 +00:00
|
|
|
if len(address) == 40:
|
|
|
|
return unhexlify(address)
|
|
|
|
|
|
|
|
elif len(address) == 42:
|
|
|
|
if address[0:2] not in ("0x", "0X"):
|
|
|
|
raise wire.ProcessError("Ethereum: invalid beginning of an address")
|
|
|
|
return unhexlify(address[2:])
|
|
|
|
|
2019-01-29 14:35:09 +00:00
|
|
|
elif len(address) == 0:
|
|
|
|
return bytes()
|
|
|
|
|
2019-01-28 14:44:18 +00:00
|
|
|
raise wire.ProcessError("Ethereum: Invalid address length")
|