first commit
This commit is contained in:
commit
bc1d07d669
12
README
Normal file
12
README
Normal file
@ -0,0 +1,12 @@
|
||||
pywallet.py 1.0
|
||||
|
||||
based on http://github.com/gavinandresen/bitcointools
|
||||
|
||||
Usage: pywallet.py [options]
|
||||
|
||||
Options:
|
||||
--version show program's version number and exit
|
||||
-h, --help show this help message and exit
|
||||
--dumpwallet dump wallet in json format
|
||||
--importprivkey=KEY import private key from vanitygen
|
||||
--datadir=DATADIR wallet directory (defaults to bitcoin default)
|
||||
693
pywallet.py
Normal file
693
pywallet.py
Normal file
@ -0,0 +1,693 @@
|
||||
# pywallet.py 1.0
|
||||
#
|
||||
# based on http://github.com/gavinandresen/bitcointools
|
||||
#
|
||||
# Usage: pywallet.py [options]
|
||||
#
|
||||
# Options:
|
||||
# --version show program's version number and exit
|
||||
# -h, --help show this help message and exit
|
||||
# --dumpwallet dump wallet in json format
|
||||
# --importprivkey=KEY import private key from vanitygen
|
||||
# --datadir=DATADIR wallet directory (defaults to bitcoin default)
|
||||
|
||||
from bsddb.db import *
|
||||
import os, sys, time
|
||||
import json
|
||||
import logging
|
||||
import struct
|
||||
import StringIO
|
||||
import traceback
|
||||
import socket
|
||||
import types
|
||||
import string
|
||||
import exceptions
|
||||
import Crypto.Hash.SHA256 as SHA256
|
||||
import Crypto.Hash.RIPEMD160 as RIPEMD160
|
||||
from Crypto.PublicKey.pubkey import *
|
||||
from ctypes import *
|
||||
|
||||
TESTNET = 0
|
||||
|
||||
json_db = {}
|
||||
private_keys = []
|
||||
|
||||
def determine_db_dir():
|
||||
import os
|
||||
import os.path
|
||||
import platform
|
||||
if platform.system() == "Darwin":
|
||||
return os.path.expanduser("~/Library/Application Support/Bitcoin/")
|
||||
elif platform.system() == "Windows":
|
||||
return os.path.join(os.environ['APPDATA'], "Bitcoin")
|
||||
return os.path.expanduser("~/.bitcoin")
|
||||
|
||||
dlls = list()
|
||||
if 'win' in sys.platform:
|
||||
for d in ('libeay32.dll', 'libssl32.dll', 'ssleay32.dll'):
|
||||
try:
|
||||
dlls.append( cdll.LoadLibrary(d) )
|
||||
except:
|
||||
pass
|
||||
else:
|
||||
dlls.append( cdll.LoadLibrary('libssl.so') )
|
||||
|
||||
class BIGNUM_Struct (Structure):
|
||||
_fields_ = [("d",c_void_p),("top",c_int),("dmax",c_int),("neg",c_int),("flags",c_int)]
|
||||
|
||||
class BN_CTX_Struct (Structure):
|
||||
_fields_ = [ ("_", c_byte) ]
|
||||
|
||||
BIGNUM = POINTER( BIGNUM_Struct )
|
||||
BN_CTX = POINTER( BN_CTX_Struct )
|
||||
|
||||
def load_func( name, args, returns = c_int):
|
||||
d = sys.modules[ __name__ ].__dict__
|
||||
f = None
|
||||
|
||||
for dll in dlls:
|
||||
try:
|
||||
f = getattr(dll, name)
|
||||
f.argtypes = args
|
||||
f.restype = returns
|
||||
d[ name ] = f
|
||||
return
|
||||
except:
|
||||
pass
|
||||
raise ImportError('Unable to load required functions from SSL dlls')
|
||||
|
||||
load_func( 'BN_new', [], BIGNUM )
|
||||
load_func( 'BN_CTX_new', [], BN_CTX )
|
||||
load_func( 'BN_CTX_free', [BN_CTX], None )
|
||||
load_func( 'BN_num_bits', [BIGNUM], c_int )
|
||||
load_func( 'BN_bn2bin', [BIGNUM, c_char_p] )
|
||||
load_func( 'BN_bin2bn', [c_char_p, c_int, BIGNUM], BIGNUM )
|
||||
load_func( 'EC_KEY_new_by_curve_name', [c_int], c_void_p )
|
||||
load_func( 'EC_KEY_get0_group', [c_void_p], c_void_p)
|
||||
load_func( 'EC_KEY_get0_private_key', [c_void_p], BIGNUM)
|
||||
load_func( 'EC_POINT_new', [c_void_p], c_void_p)
|
||||
load_func( 'EC_POINT_free', [c_void_p])
|
||||
load_func( 'EC_POINT_mul', [c_void_p, c_void_p, BIGNUM, c_void_p, BIGNUM, BN_CTX], c_int)
|
||||
load_func( 'EC_KEY_set_private_key', [c_void_p, BIGNUM], c_void_p)
|
||||
load_func( 'EC_KEY_set_public_key', [c_void_p, c_void_p], c_void_p)
|
||||
load_func( 'i2d_ECPrivateKey', [ c_void_p, POINTER(POINTER(c_char))], c_int )
|
||||
load_func( 'i2o_ECPublicKey', [ c_void_p, POINTER(POINTER(c_char))], c_int )
|
||||
|
||||
def BN_num_bytes(a):
|
||||
return ((BN_num_bits(a)+7)/8)
|
||||
|
||||
NID_secp256k1 = 714
|
||||
|
||||
pkey = 0
|
||||
|
||||
def EC_KEY_regenerate_key(eckey, priv_key):
|
||||
group = EC_KEY_get0_group(eckey)
|
||||
ctx = BN_CTX_new()
|
||||
pub_key = EC_POINT_new(group)
|
||||
EC_POINT_mul(group, pub_key, priv_key, None, None, ctx)
|
||||
EC_KEY_set_private_key(eckey, priv_key)
|
||||
EC_KEY_set_public_key(eckey, pub_key)
|
||||
EC_POINT_free(pub_key)
|
||||
BN_CTX_free(ctx)
|
||||
|
||||
def GetSecret(pkey):
|
||||
bn = EC_KEY_get0_private_key(pkey)
|
||||
nSize = BN_num_bytes(bn)
|
||||
b = create_string_buffer(nSize)
|
||||
BN_bn2bin(bn, b)
|
||||
return b.raw
|
||||
|
||||
def GetPrivKey(pkey):
|
||||
nSize = i2d_ECPrivateKey(pkey, None)
|
||||
p = create_string_buffer(nSize)
|
||||
i2d_ECPrivateKey(pkey, byref(cast(p, POINTER(c_char))))
|
||||
return p.raw
|
||||
|
||||
def GetPubKey(pkey):
|
||||
nSize = i2o_ECPublicKey(pkey, None)
|
||||
p = create_string_buffer(nSize)
|
||||
i2o_ECPublicKey(pkey, byref(cast(p, POINTER(c_char))))
|
||||
return p.raw
|
||||
|
||||
def Hash(data):
|
||||
h1 = SHA256.new(data).digest()
|
||||
h2 = SHA256.new(h1).digest()
|
||||
return h2
|
||||
|
||||
def EncodeBase58Check(vchIn):
|
||||
hash = Hash(vchIn)
|
||||
return b58encode(vchIn + hash[0:4])
|
||||
|
||||
def DecodeBase58Check(psz):
|
||||
vchRet = b58decode(psz, None)
|
||||
vch = vchRet[0:-4]
|
||||
chk = vchRet[-4:]
|
||||
hash = Hash(vch)[0:4]
|
||||
if hash != chk:
|
||||
return None
|
||||
else:
|
||||
return vch
|
||||
|
||||
def SecretToASecret(privkey):
|
||||
vchSecret = privkey[9:9+32]
|
||||
# add 1-byte version number
|
||||
vchIn = "\x80" + vchSecret
|
||||
return EncodeBase58Check(vchIn)
|
||||
|
||||
def ASecretToSecret(key):
|
||||
vch = DecodeBase58Check(key)
|
||||
if vch:
|
||||
return vch[1:]
|
||||
else:
|
||||
return False
|
||||
|
||||
def importprivkey(db, key):
|
||||
|
||||
vchSecret = ASecretToSecret(key)
|
||||
|
||||
if not vchSecret:
|
||||
return False
|
||||
|
||||
pkey = EC_KEY_new_by_curve_name(NID_secp256k1)
|
||||
bn = BN_bin2bn(vchSecret, 32, BN_new())
|
||||
EC_KEY_regenerate_key(pkey, bn)
|
||||
|
||||
secret = GetSecret(pkey)
|
||||
private_key = GetPrivKey(pkey)
|
||||
public_key = GetPubKey(pkey)
|
||||
addr = public_key_to_bc_address(public_key)
|
||||
|
||||
print "Address: %s" % addr
|
||||
print "Privkey: %s" % SecretToASecret(private_key)
|
||||
|
||||
type = 'key'
|
||||
data = { 'public_key' : public_key, 'private_key' : private_key }
|
||||
update_wallet(db, type, data)
|
||||
|
||||
type = 'name'
|
||||
data = { 'hash' : addr, 'name' : '' }
|
||||
update_wallet(db, type, data)
|
||||
|
||||
return True
|
||||
|
||||
__b58chars = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
|
||||
__b58base = len(__b58chars)
|
||||
|
||||
def b58encode(v):
|
||||
""" encode v, which is a string of bytes, to base58.
|
||||
"""
|
||||
|
||||
long_value = 0L
|
||||
for (i, c) in enumerate(v[::-1]):
|
||||
long_value += (256**i) * ord(c)
|
||||
|
||||
result = ''
|
||||
while long_value >= __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 = 0L
|
||||
for (i, c) in enumerate(v[::-1]):
|
||||
long_value += __b58chars.find(c) * (__b58base**i)
|
||||
|
||||
result = ''
|
||||
while long_value >= 256:
|
||||
div, mod = divmod(long_value, 256)
|
||||
result = chr(mod) + result
|
||||
long_value = div
|
||||
result = chr(long_value) + result
|
||||
|
||||
nPad = 0
|
||||
for c in v:
|
||||
if c == __b58chars[0]: nPad += 1
|
||||
else: break
|
||||
|
||||
result = chr(0)*nPad + result
|
||||
if length is not None and len(result) != length:
|
||||
return None
|
||||
|
||||
return result
|
||||
|
||||
def hash_160(public_key):
|
||||
h1 = SHA256.new(public_key).digest()
|
||||
h2 = RIPEMD160.new(h1).digest()
|
||||
return h2
|
||||
|
||||
def public_key_to_bc_address(public_key):
|
||||
h160 = hash_160(public_key)
|
||||
return hash_160_to_bc_address(h160)
|
||||
|
||||
def hash_160_to_bc_address(h160):
|
||||
|
||||
if TESTNET:
|
||||
vh160 = "\x6f" + h160 # \x6f is testnet
|
||||
else:
|
||||
vh160 = "\x00" + h160 # \x00 is version 0
|
||||
|
||||
h3=SHA256.new(SHA256.new(vh160).digest()).digest()
|
||||
addr=vh160+h3[0:4]
|
||||
return b58encode(addr)
|
||||
|
||||
def bc_address_to_hash_160(addr):
|
||||
bytes = b58decode(addr, 25)
|
||||
return bytes[1:21]
|
||||
|
||||
def long_hex(bytes):
|
||||
return bytes.encode('hex_codec')
|
||||
|
||||
def short_hex(bytes):
|
||||
t = bytes.encode('hex_codec')
|
||||
if len(t) < 32:
|
||||
return t
|
||||
return t[0:32]+"..."+t[-32:]
|
||||
|
||||
def create_env(db_dir=None):
|
||||
if db_dir is None:
|
||||
db_dir = determine_db_dir()
|
||||
db_env = DBEnv(0)
|
||||
r = db_env.open(db_dir, (DB_CREATE|DB_INIT_LOCK|DB_INIT_LOG|DB_INIT_MPOOL| DB_INIT_TXN|DB_THREAD|DB_RECOVER))
|
||||
return db_env
|
||||
|
||||
def parse_CAddress(vds):
|
||||
d = {}
|
||||
d['nVersion'] = vds.read_int32()
|
||||
d['nTime'] = vds.read_uint32()
|
||||
d['nServices'] = vds.read_uint64()
|
||||
d['pchReserved'] = vds.read_bytes(12)
|
||||
d['ip'] = socket.inet_ntoa(vds.read_bytes(4))
|
||||
d['port'] = vds.read_uint16()
|
||||
return d
|
||||
|
||||
def deserialize_CAddress(d):
|
||||
return d['ip']+":"+str(d['port'])+" (lastseen: %s)"%(time.ctime(d['nTime']),)
|
||||
|
||||
def parse_setting(setting, vds):
|
||||
if setting[0] == "f": # flag (boolean) settings
|
||||
return str(vds.read_boolean())
|
||||
elif setting[0:4] == "addr": # CAddress
|
||||
d = parse_CAddress(vds)
|
||||
return deserialize_CAddress(d)
|
||||
elif setting == "nTransactionFee":
|
||||
return vds.read_int64()
|
||||
elif setting == "nLimitProcessors":
|
||||
return vds.read_int32()
|
||||
return 'unknown setting'
|
||||
|
||||
class SerializationError(Exception):
|
||||
""" Thrown when there's a problem deserializing or serializing """
|
||||
|
||||
class BCDataStream(object):
|
||||
def __init__(self):
|
||||
self.input = None
|
||||
self.read_cursor = 0
|
||||
|
||||
def clear(self):
|
||||
self.input = None
|
||||
self.read_cursor = 0
|
||||
|
||||
def write(self, bytes): # Initialize with string of bytes
|
||||
if self.input is None:
|
||||
self.input = bytes
|
||||
else:
|
||||
self.input += bytes
|
||||
|
||||
def map_file(self, file, start): # Initialize with bytes from file
|
||||
self.input = mmap.mmap(file.fileno(), 0, access=mmap.ACCESS_READ)
|
||||
self.read_cursor = start
|
||||
def seek_file(self, position):
|
||||
self.read_cursor = position
|
||||
def close_file(self):
|
||||
self.input.close()
|
||||
|
||||
def read_string(self):
|
||||
# Strings are encoded depending on length:
|
||||
# 0 to 252 : 1-byte-length followed by bytes (if any)
|
||||
# 253 to 65,535 : byte'253' 2-byte-length followed by bytes
|
||||
# 65,536 to 4,294,967,295 : byte '254' 4-byte-length followed by bytes
|
||||
# ... and the Bitcoin client is coded to understand:
|
||||
# greater than 4,294,967,295 : byte '255' 8-byte-length followed by bytes of string
|
||||
# ... but I don't think it actually handles any strings that big.
|
||||
if self.input is None:
|
||||
raise SerializationError("call write(bytes) before trying to deserialize")
|
||||
|
||||
try:
|
||||
length = self.read_compact_size()
|
||||
except IndexError:
|
||||
raise SerializationError("attempt to read past end of buffer")
|
||||
|
||||
return self.read_bytes(length)
|
||||
|
||||
def write_string(self, string):
|
||||
# Length-encoded as with read-string
|
||||
self.write_compact_size(len(string))
|
||||
self.write(string)
|
||||
|
||||
def read_bytes(self, length):
|
||||
try:
|
||||
result = self.input[self.read_cursor:self.read_cursor+length]
|
||||
self.read_cursor += length
|
||||
return result
|
||||
except IndexError:
|
||||
raise SerializationError("attempt to read past end of buffer")
|
||||
|
||||
return ''
|
||||
|
||||
def read_boolean(self): return self.read_bytes(1)[0] != chr(0)
|
||||
def read_int16(self): return self._read_num('<h')
|
||||
def read_uint16(self): return self._read_num('<H')
|
||||
def read_int32(self): return self._read_num('<i')
|
||||
def read_uint32(self): return self._read_num('<I')
|
||||
def read_int64(self): return self._read_num('<q')
|
||||
def read_uint64(self): return self._read_num('<Q')
|
||||
|
||||
def write_boolean(self, val): return self.write(chr(1) if val else chr(0))
|
||||
def write_int16(self, val): return self._write_num('<h', val)
|
||||
def write_uint16(self, val): return self._write_num('<H', val)
|
||||
def write_int32(self, val): return self._write_num('<i', val)
|
||||
def write_uint32(self, val): return self._write_num('<I', val)
|
||||
def write_int64(self, val): return self._write_num('<q', val)
|
||||
def write_uint64(self, val): return self._write_num('<Q', val)
|
||||
|
||||
def read_compact_size(self):
|
||||
size = ord(self.input[self.read_cursor])
|
||||
self.read_cursor += 1
|
||||
if size == 253:
|
||||
size = self._read_num('<H')
|
||||
elif size == 254:
|
||||
size = self._read_num('<I')
|
||||
elif size == 255:
|
||||
size = self._read_num('<Q')
|
||||
return size
|
||||
|
||||
def write_compact_size(self, size):
|
||||
if size < 0:
|
||||
raise SerializationError("attempt to write size < 0")
|
||||
elif size < 253:
|
||||
self.write(chr(size))
|
||||
elif size < 2**16:
|
||||
self.write('\xfd')
|
||||
self._write_num('<H', size)
|
||||
elif size < 2**32:
|
||||
self.write('\xfe')
|
||||
self._write_num('<I', size)
|
||||
elif size < 2**64:
|
||||
self.write('\xff')
|
||||
self._write_num('<Q', size)
|
||||
|
||||
def _read_num(self, format):
|
||||
(i,) = struct.unpack_from(format, self.input, self.read_cursor)
|
||||
self.read_cursor += struct.calcsize(format)
|
||||
return i
|
||||
|
||||
def _write_num(self, format, num):
|
||||
s = struct.pack(format, num)
|
||||
self.write(s)
|
||||
|
||||
def open_wallet(db_env, writable=False):
|
||||
db = DB(db_env)
|
||||
flags = DB_THREAD | (DB_CREATE if writable else DB_RDONLY)
|
||||
try:
|
||||
r = db.open("wallet.dat", "main", DB_BTREE, flags)
|
||||
except DBError:
|
||||
r = True
|
||||
|
||||
if r is not None:
|
||||
logging.error("Couldn't open wallet.dat/main. Try quitting Bitcoin and running this again.")
|
||||
sys.exit(1)
|
||||
|
||||
return db
|
||||
|
||||
def parse_wallet(db, item_callback):
|
||||
kds = BCDataStream()
|
||||
vds = BCDataStream()
|
||||
|
||||
for (key, value) in db.items():
|
||||
d = { }
|
||||
|
||||
kds.clear(); kds.write(key)
|
||||
vds.clear(); vds.write(value)
|
||||
|
||||
type = kds.read_string()
|
||||
|
||||
d["__key__"] = key
|
||||
d["__value__"] = value
|
||||
d["__type__"] = type
|
||||
|
||||
try:
|
||||
if type == "tx":
|
||||
d["tx_id"] = kds.read_bytes(32)
|
||||
d.update(parse_WalletTx(vds))
|
||||
elif type == "name":
|
||||
d['hash'] = kds.read_string()
|
||||
d['name'] = vds.read_string()
|
||||
elif type == "version":
|
||||
d['version'] = vds.read_uint32()
|
||||
elif type == "setting":
|
||||
d['setting'] = kds.read_string()
|
||||
d['value'] = parse_setting(d['setting'], vds)
|
||||
elif type == "key":
|
||||
d['public_key'] = kds.read_bytes(kds.read_compact_size())
|
||||
d['private_key'] = vds.read_bytes(vds.read_compact_size())
|
||||
elif type == "wkey":
|
||||
d['public_key'] = kds.read_bytes(kds.read_compact_size())
|
||||
d['private_key'] = vds.read_bytes(vds.read_compact_size())
|
||||
d['created'] = vds.read_int64()
|
||||
d['expires'] = vds.read_int64()
|
||||
d['comment'] = vds.read_string()
|
||||
elif type == "defaultkey":
|
||||
d['key'] = vds.read_bytes(vds.read_compact_size())
|
||||
elif type == "pool":
|
||||
d['n'] = kds.read_int64()
|
||||
d['nVersion'] = vds.read_int32()
|
||||
d['nTime'] = vds.read_int64()
|
||||
d['public_key'] = vds.read_bytes(vds.read_compact_size())
|
||||
elif type == "acc":
|
||||
d['account'] = kds.read_string()
|
||||
d['nVersion'] = vds.read_int32()
|
||||
d['public_key'] = vds.read_bytes(vds.read_compact_size())
|
||||
elif type == "acentry":
|
||||
d['account'] = kds.read_string()
|
||||
d['n'] = kds.read_uint64()
|
||||
d['nVersion'] = vds.read_int32()
|
||||
d['nCreditDebit'] = vds.read_int64()
|
||||
d['nTime'] = vds.read_int64()
|
||||
d['otherAccount'] = vds.read_string()
|
||||
d['comment'] = vds.read_string()
|
||||
|
||||
item_callback(type, d)
|
||||
|
||||
except Exception, e:
|
||||
traceback.print_exc()
|
||||
print("ERROR parsing wallet.dat, type %s" % type)
|
||||
print("key data in hex: %s"%key.encode('hex_codec'))
|
||||
print("value data in hex: %s"%value.encode('hex_codec'))
|
||||
sys.exit(1)
|
||||
|
||||
def update_wallet(db, type, data):
|
||||
"""Write a single item to the wallet.
|
||||
db must be open with writable=True.
|
||||
type and data are the type code and data dictionary as parse_wallet would
|
||||
give to item_callback.
|
||||
data's __key__, __value__ and __type__ are ignored; only the primary data
|
||||
fields are used.
|
||||
"""
|
||||
d = data
|
||||
kds = BCDataStream()
|
||||
vds = BCDataStream()
|
||||
|
||||
# Write the type code to the key
|
||||
kds.write_string(type)
|
||||
vds.write("") # Ensure there is something
|
||||
|
||||
try:
|
||||
if type == "tx":
|
||||
raise NotImplementedError("Writing items of type 'tx'")
|
||||
kds.write(d['tx_id'])
|
||||
#d.update(parse_WalletTx(vds))
|
||||
elif type == "name":
|
||||
kds.write_string(d['hash'])
|
||||
vds.write_string(d['name'])
|
||||
elif type == "version":
|
||||
vds.write_uint32(d['version'])
|
||||
elif type == "setting":
|
||||
raise NotImplementedError("Writing items of type 'setting'")
|
||||
kds.write_string(d['setting'])
|
||||
#d['value'] = parse_setting(d['setting'], vds)
|
||||
elif type == "key":
|
||||
kds.write_string(d['public_key'])
|
||||
vds.write_string(d['private_key'])
|
||||
elif type == "wkey":
|
||||
kds.write_string(d['public_key'])
|
||||
vds.write_string(d['private_key'])
|
||||
vds.write_int64(d['created'])
|
||||
vds.write_int64(d['expires'])
|
||||
vds.write_string(d['comment'])
|
||||
elif type == "defaultkey":
|
||||
vds.write_string(d['key'])
|
||||
elif type == "pool":
|
||||
kds.write_int64(d['n'])
|
||||
vds.write_int32(d['nVersion'])
|
||||
vds.write_int64(d['nTime'])
|
||||
vds.write_string(d['public_key'])
|
||||
elif type == "acc":
|
||||
kds.write_string(d['account'])
|
||||
vds.write_int32(d['nVersion'])
|
||||
vds.write_string(d['public_key'])
|
||||
elif type == "acentry":
|
||||
kds.write_string(d['account'])
|
||||
kds.write_uint64(d['n'])
|
||||
vds.write_int32(d['nVersion'])
|
||||
vds.write_int64(d['nCreditDebit'])
|
||||
vds.write_int64(d['nTime'])
|
||||
vds.write_string(d['otherAccount'])
|
||||
vds.write_string(d['comment'])
|
||||
else:
|
||||
print "Unknown key type: "+type
|
||||
|
||||
# Write the key/value pair to the database
|
||||
db.put(kds.input, vds.input)
|
||||
|
||||
except Exception, e:
|
||||
print("ERROR writing to wallet.dat, type %s"%type)
|
||||
print("data dictionary: %r"%data)
|
||||
traceback.print_exc()
|
||||
|
||||
def rewrite_wallet(db_env, destFileName, pre_put_callback=None):
|
||||
db = open_wallet(db_env)
|
||||
|
||||
db_out = DB(db_env)
|
||||
try:
|
||||
r = db_out.open(destFileName, "main", DB_BTREE, DB_CREATE)
|
||||
except DBError:
|
||||
r = True
|
||||
|
||||
if r is not None:
|
||||
logging.error("Couldn't open %s."%destFileName)
|
||||
sys.exit(1)
|
||||
|
||||
def item_callback(type, d):
|
||||
if (pre_put_callback is None or pre_put_callback(type, d)):
|
||||
db_out.put(d["__key__"], d["__value__"])
|
||||
|
||||
parse_wallet(db, item_callback)
|
||||
db_out.close()
|
||||
db.close()
|
||||
|
||||
def read_wallet(json_db, db_env, print_wallet, print_wallet_transactions, transaction_filter):
|
||||
db = open_wallet(db_env)
|
||||
|
||||
json_db['keys'] = []
|
||||
json_db['pool'] = []
|
||||
json_db['names'] = {}
|
||||
|
||||
def item_callback(type, d):
|
||||
|
||||
if type == "name":
|
||||
json_db['names'][d['hash']] = d['name']
|
||||
|
||||
elif type == "version":
|
||||
json_db['version'] = d['version']
|
||||
|
||||
elif type == "setting":
|
||||
if not json_db.has_key('settings'): json_db['settings'] = {}
|
||||
json_db["settings"][d['setting']] = d['value']
|
||||
|
||||
elif type == "defaultkey":
|
||||
json_db['defaultkey'] = public_key_to_bc_address(d['key'])
|
||||
|
||||
elif type == "key":
|
||||
addr = public_key_to_bc_address(d['public_key'])
|
||||
sec = SecretToASecret(d['private_key'])
|
||||
private_keys.append(sec)
|
||||
json_db['keys'].append({'addr' : addr, 'sec' : sec})
|
||||
|
||||
elif type == "wkey":
|
||||
if not json_db.has_key('wkey'): json_db['wkey'] = []
|
||||
json_db['wkey']['created'] = d['created']
|
||||
|
||||
elif type == "pool":
|
||||
json_db['pool'].append( {'n': d['n'], 'addr': public_key_to_bc_address(d['public_key']), 'nTime' : d['nTime'] } )
|
||||
|
||||
elif type == "acc":
|
||||
json_db['acc'] = d['account']
|
||||
print("Account %s (current key: %s)"%(d['account'], public_key_to_bc_address(d['public_key'])))
|
||||
|
||||
elif type == "acentry":
|
||||
json_db['acentry'] = (d['account'], d['nCreditDebit'], d['otherAccount'], time.ctime(d['nTime']), d['n'], d['comment'])
|
||||
|
||||
parse_wallet(db, item_callback)
|
||||
|
||||
db.close()
|
||||
|
||||
for k in json_db['keys']:
|
||||
addr = k['addr']
|
||||
if (addr in json_db['names'].keys()):
|
||||
k["label"] = json_db['names'][addr]
|
||||
else:
|
||||
k["reserve"] = 1
|
||||
|
||||
del(json_db['pool'])
|
||||
del(json_db['names'])
|
||||
|
||||
from optparse import OptionParser
|
||||
|
||||
def main():
|
||||
parser = OptionParser(usage="%prog [options]", version="%prog 1.0")
|
||||
|
||||
parser.add_option("--dumpwallet", dest="dump", action="store_true",
|
||||
help="dump wallet in json format")
|
||||
|
||||
parser.add_option("--importprivkey", dest="key",
|
||||
help="import private key from vanitygen")
|
||||
|
||||
parser.add_option("--datadir", dest="datadir",
|
||||
help="wallet directory (defaults to bitcoin default)")
|
||||
|
||||
(options, args) = parser.parse_args()
|
||||
|
||||
if options.dump is None and options.key is None:
|
||||
print "A mandatory option is missing\n"
|
||||
parser.print_help()
|
||||
exit(0)
|
||||
|
||||
if options.datadir is None:
|
||||
db_dir = determine_db_dir()
|
||||
else:
|
||||
db_dir = options.datadir
|
||||
|
||||
db_env = create_env(db_dir)
|
||||
|
||||
read_wallet(json_db, db_env, True, True, "")
|
||||
|
||||
if options.dump:
|
||||
print json.dumps(json_db, sort_keys=True, indent=4)
|
||||
|
||||
elif options.key:
|
||||
if (options.key not in private_keys):
|
||||
db = open_wallet(db_env, writable=True)
|
||||
|
||||
if importprivkey(db, options.key):
|
||||
print "Imported successfully"
|
||||
else:
|
||||
print "Bad private key"
|
||||
|
||||
db.close()
|
||||
else:
|
||||
print "Already exists"
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
||||
Loading…
Reference in New Issue
Block a user