6 changed files with 380 additions and 293 deletions
@ -0,0 +1,254 @@ |
|||||
|
#!/usr/bin/env python |
||||
|
# |
||||
|
# Electrum - lightweight Bitcoin client |
||||
|
# Copyright (C) 2011 thomasv@gitorious |
||||
|
# |
||||
|
# This program is free software: you can redistribute it and/or modify |
||||
|
# it under the terms of the GNU General Public License as published by |
||||
|
# the Free Software Foundation, either version 3 of the License, or |
||||
|
# (at your option) any later version. |
||||
|
# |
||||
|
# This program 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 General Public License for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU General Public License |
||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
|
|
||||
|
|
||||
|
from util import * |
||||
|
from bitcoin import * |
||||
|
from decimal import Decimal |
||||
|
import bitcoin |
||||
|
|
||||
|
protected_commands = ['payto', 'password', 'mktx', 'get_seed', 'importprivkey','signmessage', 'signrawtransaction','dumpprivkey' ] |
||||
|
|
||||
|
class Commands: |
||||
|
|
||||
|
def __init__(self, wallet, interface): |
||||
|
self.wallet = wallet |
||||
|
self.interface = interface |
||||
|
|
||||
|
def _run(self, method, args, password_getter): |
||||
|
if method in protected_commands: |
||||
|
pw = apply(password_getter,()) |
||||
|
args += (pw,) |
||||
|
|
||||
|
f = eval('self.'+method) |
||||
|
apply(f,args) |
||||
|
|
||||
|
def get_history(self, addr): |
||||
|
h = self.wallet.get_history(addr) |
||||
|
if h is None: h = self.wallet.interface.synchronous_get([ ('blockchain.address.get_history',[addr]) ])[0] |
||||
|
print_json(h) |
||||
|
|
||||
|
def listunspent(self): |
||||
|
print_json(self.wallet.get_unspent_coins()) |
||||
|
|
||||
|
def createrawtransaction(self, inputs, outputs): |
||||
|
# convert to own format |
||||
|
for i in inputs: |
||||
|
i['tx_hash'] = i['txid'] |
||||
|
i['index'] = i['vout'] |
||||
|
outputs = map(lambda x: (x[0],int(1e8*x[1])), outputs.items()) |
||||
|
tx = Transaction.from_io(inputs, outputs) |
||||
|
print_msg( tx ) |
||||
|
|
||||
|
def signrawtransaction(self, raw_tx, input_info, private_keys, password): |
||||
|
tx = Transaction(raw_tx) |
||||
|
unspent_coins = self.wallet.get_unspent_coins() |
||||
|
|
||||
|
# convert private_keys to dict |
||||
|
pk = {} |
||||
|
for sec in private_keys: |
||||
|
address = bitcoin.address_from_private_key(sec) |
||||
|
pk[address] = sec |
||||
|
private_keys = pk |
||||
|
|
||||
|
for txin in tx.inputs: |
||||
|
# convert to own format |
||||
|
txin['tx_hash'] = txin['prevout_hash'] |
||||
|
txin['index'] = txin['prevout_n'] |
||||
|
|
||||
|
for item in input_info: |
||||
|
if item.get('txid') == txin['tx_hash'] and item.get('vout') == txin['index']: |
||||
|
txin['raw_output_script'] = item['scriptPubKey'] |
||||
|
txin['redeemScript'] = item.get('redeemScript') |
||||
|
txin['electrumKeyID'] = item.get('electrumKeyID') |
||||
|
break |
||||
|
else: |
||||
|
for item in unspent_coins: |
||||
|
if txin['tx_hash'] == item['tx_hash'] and txin['index'] == item['index']: |
||||
|
txin['raw_output_script'] = item['raw_output_script'] |
||||
|
break |
||||
|
else: |
||||
|
# if neither, we might want to get it from the server.. |
||||
|
raise |
||||
|
|
||||
|
# find the address: |
||||
|
import deserialize |
||||
|
if txin.get('electrumKeyID'): |
||||
|
n, for_change = txin.get('electrumKeyID') |
||||
|
sec = wallet.sequence.get_private_key(n, for_change, seed) |
||||
|
address = bitcoin.address_from_private_key(sec) |
||||
|
txin['address'] = address |
||||
|
private_keys[address] = sec |
||||
|
|
||||
|
elif txin.get("redeemScript"): |
||||
|
txin['address'] = bitcoin.hash_160_to_bc_address(bitcoin.hash_160(txin.get("redeemScript").decode('hex')), 5) |
||||
|
|
||||
|
elif txin.get("raw_output_script"): |
||||
|
addr = deserialize.get_address_from_output_script(txin.get("raw_output_script").decode('hex')) |
||||
|
sec = wallet.get_private_key(addr, password) |
||||
|
if sec: |
||||
|
private_keys[addr] = sec |
||||
|
txin['address'] = addr |
||||
|
|
||||
|
tx.sign( private_keys ) |
||||
|
print_json({ "hex":str(tx),"complete":tx.is_complete}) |
||||
|
|
||||
|
def decoderawtransaction(self, raw): |
||||
|
tx = Transaction(raw) |
||||
|
print_json( tx.deserialize() ) |
||||
|
|
||||
|
def sendrawtransaction(self, raw): |
||||
|
tx = Transaction(raw) |
||||
|
r, h = wallet.sendtx( tx ) |
||||
|
print_msg(h) |
||||
|
|
||||
|
def createmultisig(self, num, pubkeys): |
||||
|
assert isinstance(pubkeys, list) |
||||
|
print_json( Transaction.multisig_script(pubkeys, num) ) |
||||
|
|
||||
|
def freeze(self,addr): |
||||
|
print_msg(self.wallet.freeze(addr)) |
||||
|
|
||||
|
def unfreeze(self,addr): |
||||
|
print_msg(self.wallet.unfreeze(addr)) |
||||
|
|
||||
|
def prioritize(self, addr): |
||||
|
print_msg(self.wallet.prioritize(addr)) |
||||
|
|
||||
|
def unprioritize(self, addr): |
||||
|
print_msg(self.wallet.unprioritize(addr)) |
||||
|
|
||||
|
def dumpprivkey(self, addr, password): |
||||
|
sec = self.wallet.get_private_key(addr, password) |
||||
|
print_msg( sec ) |
||||
|
|
||||
|
def validateaddress(self,addr): |
||||
|
is_valid = self.wallet.is_valid(addr) |
||||
|
out = { 'isvalid':is_valid } |
||||
|
if is_valid: |
||||
|
is_mine = self.wallet.is_mine(addr) |
||||
|
out['address'] = addr |
||||
|
out['ismine'] = is_mine |
||||
|
if is_mine: |
||||
|
out['pubkey'] = self.wallet.get_public_key(addr) |
||||
|
|
||||
|
print_json(out) |
||||
|
|
||||
|
|
||||
|
def balance(self, addresses = []): |
||||
|
if addresses == []: |
||||
|
c, u = self.wallet.get_balance() |
||||
|
if u: |
||||
|
print_msg(Decimal( c ) / 100000000 , Decimal( u ) / 100000000) |
||||
|
else: |
||||
|
print_msg(Decimal( c ) / 100000000) |
||||
|
else: |
||||
|
for addr in addresses: |
||||
|
c, u = wallet.get_addr_balance(addr) |
||||
|
if u: |
||||
|
print_msg("%s %s, %s" % (addr, str(Decimal(c)/100000000), str(Decimal(u)/100000000))) |
||||
|
else: |
||||
|
print_msg("%s %s" % (addr, str(Decimal(c)/100000000))) |
||||
|
|
||||
|
|
||||
|
def get_seed(self, password): |
||||
|
import mnemonic |
||||
|
seed = self.wallet.decode_seed(password) |
||||
|
print_msg(seed + ' "' + ' '.join(mnemonic.mn_encode(seed)) + '"') |
||||
|
|
||||
|
def importprivkey(self, sec): |
||||
|
try: |
||||
|
addr = wallet.import_key(sec,password) |
||||
|
wallet.save() |
||||
|
print_msg("Keypair imported: ", addr) |
||||
|
except BaseException as e: |
||||
|
print_msg("Error: Keypair import failed: " + str(e)) |
||||
|
|
||||
|
|
||||
|
def sign_message(self, address, message, password): |
||||
|
print_msg(self.wallet.sign_message(address, message, password)) |
||||
|
|
||||
|
|
||||
|
def verify_message(self, address, signature, message): |
||||
|
try: |
||||
|
EC_KEY.verify_message(address, signature, message) |
||||
|
print_msg(True) |
||||
|
except BaseException as e: |
||||
|
print_error("Verification error: {0}".format(e)) |
||||
|
print_msg(False) |
||||
|
|
||||
|
|
||||
|
def mktx(self, to_address, amount, fee, change_addr, from_addr, password = None): |
||||
|
|
||||
|
for k, v in self.wallet.labels.items(): |
||||
|
if v == to_address: |
||||
|
to_address = k |
||||
|
print_msg("alias", to_address) |
||||
|
break |
||||
|
if change_addr and v == change_addr: |
||||
|
change_addr = k |
||||
|
|
||||
|
amount = int(10000000*amount) |
||||
|
if fee: fee = int(10000000*fee) |
||||
|
tx = self.wallet.mktx( [(to_address, amount)], password, fee , change_addr, from_addr) |
||||
|
|
||||
|
out = {"hex":str(tx), "complete":tx.is_complete} |
||||
|
if not tx.is_complete: |
||||
|
import json |
||||
|
out['input_info'] = repr(tx.input_info).replace(' ','') |
||||
|
print_json(out) |
||||
|
|
||||
|
|
||||
|
def payto(self, to_address, amount, fee, change_addr, from_addr, password = None): |
||||
|
|
||||
|
amount = int(10000000*amount) |
||||
|
if fee: fee = int(10000000*fee) |
||||
|
tx = self.wallet.mktx( [(to_address, amount)], password, fee, change_addr, from_addr ) |
||||
|
r, h = wallet.sendtx( tx ) |
||||
|
print_msg(h) |
||||
|
|
||||
|
|
||||
|
def history(self): |
||||
|
import datetime |
||||
|
balance = 0 |
||||
|
for item in self.wallet.get_tx_history(): |
||||
|
tx_hash, conf, is_mine, value, fee, balance, timestamp = item |
||||
|
try: |
||||
|
time_str = datetime.datetime.fromtimestamp( timestamp).isoformat(' ')[:-3] |
||||
|
except: |
||||
|
time_str = "----" |
||||
|
|
||||
|
label, is_default_label = self.wallet.get_label(tx_hash) |
||||
|
if not label: label = tx_hash |
||||
|
else: label = label + ' '*(64 - len(label) ) |
||||
|
|
||||
|
print_msg("%17s"%time_str, " " + label + " " + format_satoshis(value)+ " "+ format_satoshis(balance)) |
||||
|
print_msg("# balance: ", format_satoshis(balance)) |
||||
|
|
||||
|
|
||||
|
def setlabel(self, tx, label): |
||||
|
self.wallet.labels[tx] = label |
||||
|
self.wallet.save() |
||||
|
|
||||
|
|
||||
|
def contacts(self): |
||||
|
c = {} |
||||
|
for addr in self.wallet.addressbook: |
||||
|
c[addr] = self.wallet.labels.get(addr) |
||||
|
print_json(c) |
||||
|
|
Loading…
Reference in new issue