1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-10 15:30:55 +00:00

Introduce coininfo.b58_hash

This commit is contained in:
Yura Pakhuchiy 2018-07-01 23:47:43 +07:00 committed by Pavol Rusnak
parent b48cc1d6f8
commit d09547fec3
6 changed files with 34 additions and 27 deletions

View File

@ -1,3 +1,5 @@
from trezor.crypto.base58 import groestl512d_32, sha256d_32
class CoinInfo: class CoinInfo:
def __init__( def __init__(
@ -36,8 +38,10 @@ class CoinInfo:
self.bip115 = bip115 self.bip115 = bip115
self.curve_name = curve_name self.curve_name = curve_name
if curve_name == 'secp256k1-groestl': if curve_name == 'secp256k1-groestl':
self.b58_hash = groestl512d_32
self.sign_hash_double = False self.sign_hash_double = False
else: else:
self.b58_hash = sha256d_32
self.sign_hash_double = True self.sign_hash_double = True

View File

@ -34,7 +34,7 @@ def get_address(script_type: InputScriptType, coin: CoinInfo, node, multisig=Non
'Multisig not enabled on this coin') 'Multisig not enabled on this coin')
pubkeys = multisig_get_pubkeys(multisig) pubkeys = multisig_get_pubkeys(multisig)
address = address_multisig_p2sh(pubkeys, multisig.m, coin.address_type_p2sh) address = address_multisig_p2sh(pubkeys, multisig.m, coin)
if coin.cashaddr_prefix is not None: if coin.cashaddr_prefix is not None:
address = address_to_cashaddr(address, coin) address = address_to_cashaddr(address, coin)
return address return address
@ -67,32 +67,32 @@ def get_address(script_type: InputScriptType, coin: CoinInfo, node, multisig=Non
# p2wsh multisig nested in p2sh # p2wsh multisig nested in p2sh
if multisig is not None: if multisig is not None:
pubkeys = multisig_get_pubkeys(multisig) pubkeys = multisig_get_pubkeys(multisig)
return address_multisig_p2wsh_in_p2sh(pubkeys, multisig.m, coin.address_type_p2sh) return address_multisig_p2wsh_in_p2sh(pubkeys, multisig.m, coin)
# p2wpkh nested in p2sh # p2wpkh nested in p2sh
return address_p2wpkh_in_p2sh(node.public_key(), coin.address_type_p2sh) return address_p2wpkh_in_p2sh(node.public_key(), coin)
else: else:
raise AddressError(FailureType.ProcessError, raise AddressError(FailureType.ProcessError,
'Invalid script type') 'Invalid script type')
def address_multisig_p2sh(pubkeys: bytes, m: int, addrtype: int): def address_multisig_p2sh(pubkeys: bytes, m: int, coin: CoinInfo):
if addrtype is None: if coin.address_type_p2sh is None:
raise AddressError(FailureType.ProcessError, raise AddressError(FailureType.ProcessError,
'Multisig not enabled on this coin') 'Multisig not enabled on this coin')
redeem_script = output_script_multisig(pubkeys, m) redeem_script = output_script_multisig(pubkeys, m)
redeem_script_hash = sha256_ripemd160_digest(redeem_script) redeem_script_hash = sha256_ripemd160_digest(redeem_script)
return address_p2sh(redeem_script_hash, addrtype) return address_p2sh(redeem_script_hash, coin)
def address_multisig_p2wsh_in_p2sh(pubkeys: bytes, m: int, addrtype: int): def address_multisig_p2wsh_in_p2sh(pubkeys: bytes, m: int, coin: CoinInfo):
if addrtype is None: if coin.address_type_p2sh is None:
raise AddressError(FailureType.ProcessError, raise AddressError(FailureType.ProcessError,
'Multisig not enabled on this coin') 'Multisig not enabled on this coin')
witness_script = output_script_multisig(pubkeys, m) witness_script = output_script_multisig(pubkeys, m)
witness_script_hash = sha256(witness_script).digest() witness_script_hash = sha256(witness_script).digest()
return address_p2wsh_in_p2sh(witness_script_hash, addrtype) return address_p2wsh_in_p2sh(witness_script_hash, coin)
def address_multisig_p2wsh(pubkeys: bytes, m: int, hrp: str): def address_multisig_p2wsh(pubkeys: bytes, m: int, hrp: str):
@ -104,27 +104,27 @@ def address_multisig_p2wsh(pubkeys: bytes, m: int, hrp: str):
return address_p2wsh(witness_script_hash, hrp) return address_p2wsh(witness_script_hash, hrp)
def address_pkh(pubkey: bytes, addrtype: int) -> str: def address_pkh(pubkey: bytes, coin: CoinInfo) -> str:
s = addrtype_bytes(addrtype) + sha256_ripemd160_digest(pubkey) s = addrtype_bytes(coin.address_type) + sha256_ripemd160_digest(pubkey)
return base58.encode_check(bytes(s)) return base58.encode_check(bytes(s), coin.b58_hash)
def address_p2sh(redeem_script_hash: bytes, addrtype: int) -> str: def address_p2sh(redeem_script_hash: bytes, coin: CoinInfo) -> str:
s = addrtype_bytes(addrtype) + redeem_script_hash s = addrtype_bytes(coin.address_type_p2sh) + redeem_script_hash
return base58.encode_check(bytes(s)) return base58.encode_check(bytes(s), coin.b58_hash)
def address_p2wpkh_in_p2sh(pubkey: bytes, addrtype: int) -> str: def address_p2wpkh_in_p2sh(pubkey: bytes, coin: CoinInfo) -> str:
pubkey_hash = ecdsa_hash_pubkey(pubkey) pubkey_hash = ecdsa_hash_pubkey(pubkey)
redeem_script = output_script_native_p2wpkh_or_p2wsh(pubkey_hash) redeem_script = output_script_native_p2wpkh_or_p2wsh(pubkey_hash)
redeem_script_hash = sha256_ripemd160_digest(redeem_script) redeem_script_hash = sha256_ripemd160_digest(redeem_script)
return address_p2sh(redeem_script_hash, addrtype) return address_p2sh(redeem_script_hash, coin)
def address_p2wsh_in_p2sh(witness_script_hash: bytes, addrtype: int) -> str: def address_p2wsh_in_p2sh(witness_script_hash: bytes, coin: CoinInfo) -> str:
redeem_script = output_script_native_p2wpkh_or_p2wsh(witness_script_hash) redeem_script = output_script_native_p2wpkh_or_p2wsh(witness_script_hash)
redeem_script_hash = sha256_ripemd160_digest(redeem_script) redeem_script_hash = sha256_ripemd160_digest(redeem_script)
return address_p2sh(redeem_script_hash, addrtype) return address_p2sh(redeem_script_hash, coin)
def address_p2wpkh(pubkey: bytes, hrp: str) -> str: def address_p2wpkh(pubkey: bytes, hrp: str) -> str:
@ -153,7 +153,7 @@ def decode_bech32_address(prefix: str, address: str) -> bytes:
def address_to_cashaddr(address: str, coin: CoinInfo) -> str: def address_to_cashaddr(address: str, coin: CoinInfo) -> str:
raw = base58.decode_check(address) raw = base58.decode_check(address, coin.b58_hash)
version, data = raw[0], raw[1:] version, data = raw[0], raw[1:]
if version == coin.address_type: if version == coin.address_type:
version = cashaddr.ADDRESS_TYPE_P2KH version = cashaddr.ADDRESS_TYPE_P2KH

View File

@ -509,7 +509,7 @@ def output_derive_script(o: TxOutputType, coin: CoinInfo, root: bip32.HDNode) ->
raise ValueError('Unknown cashaddr address type') raise ValueError('Unknown cashaddr address type')
raw_address = bytes([version]) + data raw_address = bytes([version]) + data
else: else:
raw_address = base58.decode_check(o.address) raw_address = base58.decode_check(o.address, coin.b58_hash)
if address_type.check(coin.address_type, raw_address): if address_type.check(coin.address_type, raw_address):
# p2pkh # p2pkh

View File

@ -38,11 +38,11 @@ async def verify_message(ctx, msg):
raise wire.ProcessError('Invalid signature') raise wire.ProcessError('Invalid signature')
if script_type == SPENDADDRESS: if script_type == SPENDADDRESS:
addr = address_pkh(pubkey, coin.address_type) addr = address_pkh(pubkey, coin)
if coin.cashaddr_prefix is not None: if coin.cashaddr_prefix is not None:
addr = address_to_cashaddr(addr, coin) addr = address_to_cashaddr(addr, coin)
elif script_type == SPENDP2SHWITNESS: elif script_type == SPENDP2SHWITNESS:
addr = address_p2wpkh_in_p2sh(pubkey, coin.address_type_p2sh) addr = address_p2wpkh_in_p2sh(pubkey, coin)
elif script_type == SPENDWITNESS: elif script_type == SPENDWITNESS:
addr = address_p2wpkh(pubkey, coin.bech32_prefix) addr = address_p2wpkh(pubkey, coin.bech32_prefix)
else: else:

View File

@ -59,19 +59,22 @@ def decode(string: str) -> bytes:
return bytes((b for b in reversed(result + [0] * (origlen - newlen)))) return bytes((b for b in reversed(result + [0] * (origlen - newlen))))
def _dsha256_32(data: bytes) -> bytes: def sha256d_32(data: bytes) -> bytes:
from .hashlib import sha256 from .hashlib import sha256
return sha256(sha256(data).digest()).digest()[:4] return sha256(sha256(data).digest()).digest()[:4]
def groestl512d_32(data: bytes) -> bytes:
from .hashlib import groestl512
return groestl512(groestl512(data).digest()).digest()[:4]
def encode_check(data: bytes, digestfunc=_dsha256_32) -> str: def encode_check(data: bytes, digestfunc=sha256d_32) -> str:
''' '''
Convert bytes to base58 encoded string, append checksum. Convert bytes to base58 encoded string, append checksum.
''' '''
return encode(data + digestfunc(data)) return encode(data + digestfunc(data))
def decode_check(string: str, digestfunc=_dsha256_32) -> bytes: def decode_check(string: str, digestfunc=sha256d_32) -> bytes:
''' '''
Convert base58 encoded string to bytes and verify checksum. Convert base58 encoded string to bytes and verify checksum.
''' '''

View File

@ -1 +1 @@
from trezorcrypto import blake256, blake2b, blake2s, ripemd160, sha1, sha256, sha512, sha3_256, sha3_512 # noqa: F401 from trezorcrypto import blake256, blake2b, blake2s, groestl512, ripemd160, sha1, sha256, sha512, sha3_256, sha3_512 # noqa: F401