# This file is part of the Trezor project. # # Copyright (C) 2012-2018 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 . import functools import hashlib import struct import unicodedata from typing import NewType, List from .coins import slip44 HARDENED_FLAG = 1 << 31 Address = NewType('Address', List[int]) def H_(x: int) -> int: """ Shortcut function that "hardens" a number in a BIP44 path. """ return x | HARDENED_FLAG def btc_hash(data): """ Double-SHA256 hash as used in BTC """ return hashlib.sha256(hashlib.sha256(data).digest()).digest() def hash_160(public_key): md = hashlib.new('ripemd160') md.update(hashlib.sha256(public_key).digest()) return md.digest() def hash_160_to_bc_address(h160, address_type): vh160 = struct.pack('= __b58base: div, mod = divmod(long_value, __b58base) result = __b58chars[mod] + result long_value = div result = __b58chars[long_value] + result # Bitcoin does a little leading-zero-compression: # leading 0-bytes in the input become leading-1s nPad = 0 for c in v: if c == 0: nPad += 1 else: break return (__b58chars[0] * nPad) + result def b58decode(v, length): """ decode v into a string of len bytes.""" long_value = 0 for (i, c) in enumerate(v[::-1]): long_value += __b58chars.find(c) * (__b58base ** i) result = b'' while long_value >= 256: div, mod = divmod(long_value, 256) result = struct.pack('B', mod) + result long_value = div result = struct.pack('B', long_value) + result nPad = 0 for c in v: if c == __b58chars[0]: nPad += 1 else: break result = b'\x00' * nPad + result if length is not None and len(result) != length: return None return result def parse_path(nstr: str) -> Address: """ Convert BIP32 path string to list of uint32 integers with hardened flags. Several conventions are supported to set the hardened flag: -1, 1', 1h e.g.: "0/1h/1" -> [0, 0x80000001, 1] :param nstr: path string :return: list of integers """ if not nstr: return [] n = nstr.split('/') # m/a/b/c => a/b/c if n[0] == 'm': n = n[1:] # coin_name/a/b/c => 44'/SLIP44_constant'/a/b/c if n[0] in slip44: coin_id = slip44[n[0]] n[0:1] = ['44h', '{}h'.format(coin_id)] def str_to_harden(x: str) -> int: if x.startswith('-'): return H_(abs(int(x))) elif x.endswith(('h', "'")): return H_(int(x[:-1])) else: return int(x) try: return [str_to_harden(x) for x in n] except Exception: raise ValueError('Invalid BIP32 path', nstr) def normalize_nfc(txt): ''' Normalize message to NFC and return bytes suitable for protobuf. This seems to be bitcoin-qt standard of doing things. ''' if isinstance(txt, bytes): txt = txt.decode('utf-8') return unicodedata.normalize('NFC', txt).encode('utf-8') class CallException(Exception): pass class expect: # Decorator checks if the method # returned one of expected protobuf messages # or raises an exception def __init__(self, expected, field=None): self.expected = expected self.field = field def __call__(self, f): @functools.wraps(f) def wrapped_f(*args, **kwargs): ret = f(*args, **kwargs) if not isinstance(ret, self.expected): raise RuntimeError("Got %s, expected %s" % (ret.__class__, self.expected)) if self.field is not None: return getattr(ret, self.field) else: return ret return wrapped_f def session(f): # Decorator wraps a BaseClient method # with session activation / deactivation @functools.wraps(f) def wrapped_f(*args, **kwargs): __tracebackhide__ = True # pytest traceback hiding - this function won't appear in tracebacks client = args[0] client.transport.session_begin() try: return f(*args, **kwargs) finally: client.transport.session_end() return wrapped_f