mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-18 03:10:58 +00:00
169 lines
5.9 KiB
Python
169 lines
5.9 KiB
Python
from common import *
|
|
from storage import cache
|
|
from trezor import wire
|
|
from trezor.crypto import bip39
|
|
from apps.common.keychain import get_keychain
|
|
from apps.common.paths import HARDENED
|
|
|
|
if not utils.BITCOIN_ONLY:
|
|
from apps.ethereum import CURVE
|
|
from apps.ethereum.keychain import (
|
|
PATTERNS_ADDRESS,
|
|
_schemas_from_address_n,
|
|
with_keychain_from_path,
|
|
with_keychain_from_chain_id,
|
|
)
|
|
from apps.ethereum.networks import by_chain_id, by_slip44
|
|
|
|
from trezor.messages import EthereumGetAddress
|
|
from trezor.messages import EthereumSignTx
|
|
|
|
|
|
@unittest.skipUnless(not utils.BITCOIN_ONLY, "altcoin")
|
|
class TestEthereumKeychain(unittest.TestCase):
|
|
def _check_keychain(self, keychain, slip44_id):
|
|
# valid address should succeed
|
|
valid_addresses = (
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED],
|
|
[44 | HARDENED, slip44_id | HARDENED, 19 | HARDENED],
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED, 0, 0],
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED, 0, 999],
|
|
)
|
|
for addr in valid_addresses:
|
|
keychain.derive(addr)
|
|
# invalid address should fail
|
|
invalid_addresses = (
|
|
[44 | HARDENED],
|
|
[44 | HARDENED, slip44_id | HARDENED],
|
|
[44 | HARDENED, 0 | HARDENED, 0 | HARDENED],
|
|
[42 | HARDENED, slip44_id | HARDENED, 0 | HARDENED],
|
|
[0 | HARDENED, slip44_id | HARDENED, 0 | HARDENED],
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED, 0],
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED, 0 | HARDENED, 0],
|
|
[44 | HARDENED, slip44_id | HARDENED, 0 | HARDENED, 0 | HARDENED, 0 | HARDENED],
|
|
)
|
|
for addr in invalid_addresses:
|
|
self.assertRaises(
|
|
wire.DataError, keychain.derive, addr,
|
|
)
|
|
|
|
def setUp(self):
|
|
cache.start_session()
|
|
seed = bip39.seed(" ".join(["all"] * 12), "")
|
|
cache.set(cache.APP_COMMON_SEED, seed)
|
|
|
|
def from_address_n(self, address_n):
|
|
schemas = _schemas_from_address_n(PATTERNS_ADDRESS, address_n)
|
|
return await_result(get_keychain(wire.DUMMY_CONTEXT, CURVE, schemas))
|
|
|
|
def test_from_address_n(self):
|
|
# valid keychain m/44'/60'/0'
|
|
keychain = self.from_address_n([44 | HARDENED, 60 | HARDENED, 0 | HARDENED])
|
|
self._check_keychain(keychain, 60)
|
|
|
|
def test_from_address_n_unknown(self):
|
|
# try Bitcoin slip44 id m/44'/0'/0'
|
|
schemas = tuple(_schemas_from_address_n(PATTERNS_ADDRESS, [44 | HARDENED, 0 | HARDENED, 0 | HARDENED]))
|
|
self.assertEqual(schemas, ())
|
|
|
|
def test_bad_address_n(self):
|
|
# keychain generated from valid slip44 id but invalid address m/0'/60'/0'
|
|
keychain = self.from_address_n([0 | HARDENED, 60 | HARDENED, 0 | HARDENED])
|
|
self._check_keychain(keychain, 60)
|
|
|
|
def test_with_keychain_from_path(self):
|
|
@with_keychain_from_path(*PATTERNS_ADDRESS)
|
|
async def handler(ctx, msg, keychain):
|
|
self._check_keychain(keychain, msg.address_n[1] & ~HARDENED)
|
|
|
|
await_result(
|
|
handler(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumGetAddress(
|
|
address_n=[44 | HARDENED, 60 | HARDENED, 0 | HARDENED]
|
|
),
|
|
)
|
|
)
|
|
await_result(
|
|
handler(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumGetAddress(
|
|
address_n=[44 | HARDENED, 108 | HARDENED, 0 | HARDENED]
|
|
),
|
|
)
|
|
)
|
|
|
|
with self.assertRaises(wire.DataError):
|
|
await_result(
|
|
handler(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumGetAddress(
|
|
address_n=[44 | HARDENED, 0 | HARDENED, 0 | HARDENED]
|
|
),
|
|
)
|
|
)
|
|
|
|
def test_with_keychain_from_chain_id(self):
|
|
@with_keychain_from_chain_id
|
|
async def handler_chain_id(ctx, msg, keychain):
|
|
slip44_id = msg.address_n[1] & ~HARDENED
|
|
# standard tests
|
|
self._check_keychain(keychain, slip44_id)
|
|
# provided address should succeed too
|
|
keychain.derive(msg.address_n)
|
|
|
|
await_result( # Ethereum
|
|
handler_chain_id(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumSignTx(
|
|
address_n=[44 | HARDENED, 60 | HARDENED, 0 | HARDENED],
|
|
chain_id=1,
|
|
gas_price=b"",
|
|
gas_limit=b"",
|
|
),
|
|
)
|
|
)
|
|
|
|
await_result( # Ethereum Classic
|
|
handler_chain_id(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumSignTx(
|
|
address_n=[44 | HARDENED, 61 | HARDENED, 0 | HARDENED],
|
|
chain_id=61,
|
|
gas_price=b"",
|
|
gas_limit=b"",
|
|
),
|
|
)
|
|
)
|
|
|
|
# Known chain-ids are allowed to use Ethereum derivation paths too, as there is
|
|
# no risk of replaying the transaction on the Ethereum chain
|
|
await_result( # ETH slip44 with ETC chain-id
|
|
handler_chain_id(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumSignTx(
|
|
address_n=[44 | HARDENED, 60 | HARDENED, 0 | HARDENED],
|
|
chain_id=61,
|
|
gas_price=b"",
|
|
gas_limit=b"",
|
|
),
|
|
)
|
|
)
|
|
|
|
with self.assertRaises(wire.DataError):
|
|
await_result( # chain_id and network mismatch
|
|
handler_chain_id(
|
|
wire.DUMMY_CONTEXT,
|
|
EthereumSignTx(
|
|
address_n=[44 | HARDENED, 61 | HARDENED, 0 | HARDENED],
|
|
chain_id=2,
|
|
gas_price=b"",
|
|
gas_limit=b"",
|
|
),
|
|
)
|
|
)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|