Advertisement
joric

bitcoin address generator (address.py)

Jun 4th, 2012
4,787
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Python 5.53 KB | None | 0 0
  1. # Joric/bitcoin-dev, june 2012, public domain
  2. import hashlib
  3. import ctypes
  4. import ctypes.util
  5.  
  6. ssl = ctypes.cdll.LoadLibrary (ctypes.util.find_library ('ssl') or 'libeay32')
  7.  
  8. def check_result (val, func, args):
  9.     if val == 0: raise ValueError
  10.     else: return ctypes.c_void_p (val)
  11.  
  12. ssl.EC_KEY_new_by_curve_name.restype = ctypes.c_void_p
  13. ssl.EC_KEY_new_by_curve_name.errcheck = check_result
  14.  
  15. class KEY:
  16.     def __init__(self):
  17.         NID_secp256k1 = 714
  18.         self.k = ssl.EC_KEY_new_by_curve_name(NID_secp256k1)
  19.         self.compressed = False
  20.         self.POINT_CONVERSION_COMPRESSED = 2
  21.         self.POINT_CONVERSION_UNCOMPRESSED = 4
  22.  
  23.     def __del__(self):
  24.         if ssl:
  25.             ssl.EC_KEY_free(self.k)
  26.         self.k = None
  27.  
  28.     def generate(self, secret=None):
  29.         if secret:
  30.             priv_key = ssl.BN_bin2bn(secret, 32, ssl.BN_new())
  31.             group = ssl.EC_KEY_get0_group(self.k)
  32.             pub_key = ssl.EC_POINT_new(group)
  33.             ctx = ssl.BN_CTX_new()
  34.             ssl.EC_POINT_mul(group, pub_key, priv_key, None, None, ctx)
  35.             ssl.EC_KEY_set_private_key(self.k, priv_key)
  36.             ssl.EC_KEY_set_public_key(self.k, pub_key)
  37.             ssl.EC_POINT_free(pub_key)
  38.             ssl.BN_CTX_free(ctx)
  39.             return self.k
  40.         else:
  41.             return ssl.EC_KEY_generate_key(self.k)
  42.  
  43.     def set_privkey(self, key):
  44.         self.mb = ctypes.create_string_buffer(key)
  45.         ssl.d2i_ECPrivateKey(ctypes.byref(self.k), ctypes.byref(ctypes.pointer(self.mb)), len(key))
  46.  
  47.     def set_pubkey(self, key):
  48.         self.mb = ctypes.create_string_buffer(key)
  49.         ssl.o2i_ECPublicKey(ctypes.byref(self.k), ctypes.byref(ctypes.pointer(self.mb)), len(key))
  50.  
  51.     def get_privkey(self):
  52.         size = ssl.i2d_ECPrivateKey(self.k, 0)
  53.         mb_pri = ctypes.create_string_buffer(size)
  54.         ssl.i2d_ECPrivateKey(self.k, ctypes.byref(ctypes.pointer(mb_pri)))
  55.         return mb_pri.raw
  56.  
  57.     def get_pubkey(self):
  58.         size = ssl.i2o_ECPublicKey(self.k, 0)
  59.         mb = ctypes.create_string_buffer(size)
  60.         ssl.i2o_ECPublicKey(self.k, ctypes.byref(ctypes.pointer(mb)))
  61.         return mb.raw
  62.  
  63.     def get_secret(self):
  64.         bn = ssl.EC_KEY_get0_private_key(self.k);
  65.         bytes = (ssl.BN_num_bits(bn) + 7) / 8
  66.         mb = ctypes.create_string_buffer(bytes)
  67.         n = ssl.BN_bn2bin(bn, mb);
  68.         return mb.raw.rjust(32, chr(0))
  69.  
  70.     def set_compressed(self, compressed):
  71.         self.compressed = compressed
  72.         if compressed:
  73.             form = self.POINT_CONVERSION_COMPRESSED
  74.         else:
  75.             form = self.POINT_CONVERSION_UNCOMPRESSED
  76.         ssl.EC_KEY_set_conv_form(self.k, form)
  77.  
  78. def dhash(s):
  79.     return hashlib.sha256(hashlib.sha256(s).digest()).digest()
  80.  
  81. def rhash(s):
  82.     h1 = hashlib.new('ripemd160')
  83.     h1.update(hashlib.sha256(s).digest())
  84.     return h1.digest()
  85.  
  86. b58_digits = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
  87.  
  88. def base58_encode(n):
  89.     l = []
  90.     while n > 0:
  91.         n, r = divmod(n, 58)
  92.         l.insert(0,(b58_digits[r]))
  93.     return ''.join(l)
  94.  
  95. def base58_decode(s):
  96.     n = 0
  97.     for ch in s:
  98.         n *= 58
  99.         digit = b58_digits.index(ch)
  100.         n += digit
  101.     return n
  102.  
  103. def base58_encode_padded(s):
  104.     res = base58_encode(int('0x' + s.encode('hex'), 16))
  105.     pad = 0
  106.     for c in s:
  107.         if c == chr(0):
  108.             pad += 1
  109.         else:
  110.             break
  111.     return b58_digits[0] * pad + res
  112.  
  113. def base58_decode_padded(s):
  114.     pad = 0
  115.     for c in s:
  116.         if c == b58_digits[0]:
  117.             pad += 1
  118.         else:
  119.             break
  120.     h = '%x' % base58_decode(s)
  121.     if len(h) % 2:
  122.         h = '0' + h
  123.     res = h.decode('hex')
  124.     return chr(0) * pad + res
  125.  
  126. def base58_check_encode(s, version=0):
  127.     vs = chr(version) + s
  128.     check = dhash(vs)[:4]
  129.     return base58_encode_padded(vs + check)
  130.  
  131. def base58_check_decode(s, version=0):
  132.     k = base58_decode_padded(s)
  133.     v0, data, check0 = k[0], k[1:-4], k[-4:]
  134.     check1 = dhash(v0 + data)[:4]
  135.     if check0 != check1:
  136.         raise BaseException('checksum error')
  137.     if version != ord(v0):
  138.         raise BaseException('version mismatch')
  139.     return data
  140.  
  141. def gen_eckey(passphrase=None, secret=None, pkey=None, compressed=False, rounds=1):
  142.     k = KEY()
  143.     if passphrase:
  144.         secret = passphrase.encode('utf8')
  145.         for i in xrange(rounds):
  146.             secret = hashlib.sha256(secret).digest()
  147.     if pkey:
  148.         secret = base58_check_decode(pkey, 128)
  149.         compressed = len(secret) == 33
  150.         secret = secret[0:32]
  151.     k.generate(secret)
  152.     k.set_compressed(compressed)
  153.     return k
  154.  
  155. def get_addr(k):
  156.     pubkey = k.get_pubkey()
  157.     secret = k.get_secret()
  158.     hash160 = rhash(pubkey)
  159.     addr = base58_check_encode(hash160)
  160.     payload = secret
  161.     if k.compressed:
  162.         payload = secret + chr(1)
  163.     pkey = base58_check_encode(payload, 128)
  164.     return addr, pkey
  165.  
  166. def test():
  167.     # random uncompressed
  168.     print get_addr(gen_eckey())
  169.     # random compressed
  170.     print get_addr(gen_eckey(compressed=True))
  171.     # by secret
  172.     print get_addr(gen_eckey(secret=('%064x' % 0xdeadbabe).decode('hex')))
  173.     # by passphrase
  174.     print get_addr(gen_eckey(passphrase='Satoshi Nakamoto'))
  175.     # by private key, uncompressed
  176.     print get_addr(gen_eckey(pkey='5K1HkbYffstTZDuV4riUWMbAMkQh57b8798uoy9pXYUDYeUHe7F'))
  177.     # by private key, compressed
  178.     print get_addr(gen_eckey(pkey='L3ATL5R9Exe1ubuAnHVgNgTKZEUKkDvWYAWkLUCyyvzzxRjtgyFe'))
  179.  
  180. if __name__ == '__main__':
  181.     test()
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement