#!/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 . import sys import datetime import time import copy import argparse from util import print_msg, format_satoshis, print_stderr from bitcoin import is_valid, hash_160_to_bc_address, hash_160 from decimal import Decimal import bitcoin from transaction import Transaction class Command: def __init__(self, name, requires_network, requires_wallet, requires_password, params, options, description): self.name = name self.params = params self.options = options self.requires_network = requires_network self.requires_wallet = requires_wallet self.requires_password = requires_password self.description = description known_commands = {} def register_command(*args): global known_commands name = args[0] known_commands[name] = Command(*args) # command # requires_network # requires_wallet # requires_password # args # optional args register_command('contacts', False, True, False, [], [], 'Show your list of contacts') register_command('create', False, True, False, [], [], 'Create a new wallet') register_command('createmultisig', False, True, False, ['num','pubkeys'], [], 'Create multisig address') register_command('createrawtransaction', False, True, False, ['inputs', 'outputs'], [], 'Create an unsigned transaction. The syntax is similar to bitcoind.') register_command('deseed', False, True, False, [], [], 'Remove seed from wallet, creating a seedless, watching-only wallet.') register_command('decoderawtransaction', False, False, False, ['raw_tx'], [], 'Decode raw transaction') register_command('getprivatekeys', False, True, True, ['address'], [], 'Get the private keys of a given address') register_command('dumpprivkeys', False, True, True, [], [], 'Dump private keys from your wallet') register_command('freeze', False, True, True, ['address'], [], 'Freeze the funds at one of your wallet\'s addresses') register_command('getbalance', True, True, False, [], [], 'Return the balance of your wallet, or of one account in your wallet') register_command('getservers', True, False, False, [], [], 'Return the list of available servers') register_command('getaddressbalance', True, False, False, ['address'], [], 'Return the balance of an address') register_command('getaddresshistory', True, False, False, ['address'], [], 'Return the transaction history of a wallet address') register_command('getconfig', False, False, False, ['key'], [], 'Return a configuration variable') register_command('getpubkeys', False, True, False, ['address'], [], 'Return the public keys for a wallet address') register_command('getrawtransaction', True, False, False, ['txid'], [], 'Retrieve a transaction') register_command('getseed', False, True, True, [], [], 'Print the generation seed of your wallet.') register_command('getmpk', False, True, False, [], [], 'Return your wallet\'s master public key') register_command('help', False, False, False, [], [], 'Print help on a command.') register_command('history', True, True, False, [], [], 'Returns the transaction history of your wallet') register_command('importprivkey', False, True, True, ['privatekey'], [], 'Import a private key') register_command('ismine', False, True, False, ['address'], [], 'Return true if and only if address is in wallet') register_command('listaddresses', False, True, False, [], ['show_all', 'show_labels'], 'Returns your list of addresses.') register_command('listunspent', True, True, False, [], [], 'Returns the list of unspent inputs in your wallet.') register_command('getaddressunspent', True, False, False, ['address'], [], 'Returns the list of unspent inputs for an address.') register_command('mktx', False, True, True, ['recipient', 'amount'], ['tx_fee', 'from_addr', 'change_addr'], 'Create a signed transaction') register_command('payto', True, True, True, ['recipient', 'amount'], ['tx_fee', 'from_addr', 'change_addr'], 'Create and broadcast a transaction.') register_command('mktx_csv', False, True, True, ['csv_file'], ['tx_fee', 'from_addr', 'change_addr'], 'Create a signed transaction') register_command('payto_csv', True, True, True, ['csv_file'], ['tx_fee', 'from_addr', 'change_addr'], 'Create and broadcast a transaction.') register_command('password', False, True, True, [], [], 'Change your password') register_command('restore', True, True, False, [], ['gap_limit', 'mpk', 'concealed'], 'Restore a wallet') register_command('searchcontacts', False, True, False, ['query'], [], 'Search through contacts, return matching entries') register_command('setconfig', False, False, False, ['key', 'value'], [], 'Set a configuration variable') register_command('setlabel', False, True, False, ['txid', 'label'], [], 'Assign a label to an item') register_command('sendrawtransaction', True, False, False, ['raw_tx'], [], 'Broadcast a transaction to the network.') register_command('signtxwithkey', False, False, False, ['raw_tx', 'key'], [], 'Sign a serialized transaction with a key') register_command('signtxwithwallet', False, True, True, ['raw_tx'], [], 'Sign a serialized transaction with a wallet') register_command('signmessage', False, True, True, ['address', 'message'], [], 'Sign a message with a key') register_command('unfreeze', False, True, False, ['address'], [], 'Unfreeze the funds at one of your wallet\'s address') register_command('validateaddress', False, False, False, ['address'], [], 'Check that the address is valid') register_command('verifymessage', False, False, False, ['address', 'message'], [], 'Verify a signature') register_command('version', False, False, False, [], [], 'Return the version of your client') register_command('encrypt', False, False, False, ['pubkey', 'message'], [], 'Encrypt a message with a public key') register_command('decrypt', False, True, True, ['pubkey', 'message'], [], 'Decrypt a message encrypted with a public key') register_command('getmerkle', True, False, False, ['txid', 'height'], [], 'Get Merkle branch of a transaction included in a block') register_command('getproof', True, False, False, ['address'], [], 'Get Merkle branch of an address in the UTXO set') register_command('getutxoaddress', True, False, False, ['txid', 'pos'], [], 'Get the address of an unspent transaction output') register_command('sweep', True, False, False, ['privkey', 'destination_address'], ['tx_fee'], 'Sweep a private key.') register_command('make_seed', False, False, False, [], ['nbits', 'entropy', 'language'], 'Create a seed.') register_command('check_seed', False, False, False, ['seed'], ['entropy'], 'Check that a seed was generated with external entropy.') command_options = { 'password': ("-W", "--password", None, "Password"), 'concealed': ("-C", "--concealed", False, "Don't echo seed to console when restoring"), 'show_all': ("-a", "--all", False, "Show all addresses"), 'show_labels': ("-l", "--labels", False, "Show the labels of listed addresses"), 'tx_fee': ("-f", "--fee", None, "set tx fee"), 'from_addr': ("-F", "--fromaddr", None, "set source address for payto/mktx. if it isn't in the wallet, it will ask for the private key unless supplied in the format public_key:private_key. It's not saved in the wallet."), 'change_addr': ("-c", "--changeaddr", None, "set the change address for payto/mktx. default is a spare address, or the source address if it's not in the wallet"), 'nbits': (None, "--nbits", "128", "Number of bits of entropy"), 'entropy': (None, "--entropy", "1", "Custom entropy"), 'language': ("-L", "--lang", None, "Default language for wordlist"), 'gap_limit': ("-G", "--gap", None, "Gap limit"), 'mpk': (None, "--mpk", None, "Restore from master public key"), } def set_default_subparser(self, name, args=None): """see http://stackoverflow.com/questions/5176691/argparse-how-to-specify-a-default-subcommand""" subparser_found = False for arg in sys.argv[1:]: if arg in ['-h', '--help']: # global help if no subparser break else: for x in self._subparsers._actions: if not isinstance(x, argparse._SubParsersAction): continue for sp_name in x._name_parser_map.keys(): if sp_name in sys.argv[1:]: subparser_found = True if not subparser_found: # insert default in first position, this implies no # global options without a sub_parsers specified if args is None: sys.argv.insert(1, name) else: args.insert(0, name) argparse.ArgumentParser.set_default_subparser = set_default_subparser add_offline_option = lambda parser: parser.add_argument("-o", "--offline", action="store_true", dest="offline", default=False, help="Remain offline") def add_wallet_options(p): p.add_argument("-w", "--wallet", dest="wallet_path", help="wallet path") p.add_argument("-P", "--portable", action="store_true", dest="portable", default=False, help="Portable wallet") def get_parser(run_gui, run_daemon, run_cmdline): # parent parser, because set_default_subparser removes global options parent_parser = argparse.ArgumentParser('parent', add_help=False) parent_parser.add_argument("-v", "--verbose", action="store_true", dest="verbose", default=False, help="Show debugging information") # connection conn_parser = argparse.ArgumentParser(add_help=False) conn_parser.add_argument("-1", "--oneserver", action="store_true", dest="oneserver", default=False, help="connect to one server only") conn_parser.add_argument("-s", "--server", dest="server", default=None, help="set server host:port:protocol, where protocol is either t (tcp) or s (ssl)") conn_parser.add_argument("-p", "--proxy", dest="proxy", default=None, help="set proxy [type:]host[:port], where type is socks4,socks5 or http") # create main parser parser = argparse.ArgumentParser(parents=[parent_parser]) subparsers = parser.add_subparsers(dest='cmd') # gui parser_gui = subparsers.add_parser('gui', parents=[parent_parser, conn_parser], description="Run Electrum's Graphical User Interface.", help="Run GUI (default)") parser_gui.add_argument("url", nargs='?', default=None, help="bitcoin URI (or bip70 file)") parser_gui.set_defaults(func=run_gui) parser_gui.add_argument("-g", "--gui", dest="gui", help="select graphical user interface", choices=['qt', 'lite', 'gtk', 'text', 'stdio']) parser_gui.add_argument("-m", action="store_true", dest="hide_gui", default=False, help="hide GUI on startup") parser.add_argument("-L", "--lang", dest="language", default=None, help="default language used in GUI") add_offline_option(parser) add_wallet_options(parser) # daemon parser_daemon = subparsers.add_parser('daemon', help="Run Daemon") parser_daemon.add_argument("subcommand", choices=['start', 'status', 'stop']) parser_daemon.set_defaults(func=run_daemon) # create a parser for each command for cmdname in sorted(known_commands.keys()): cmd = known_commands[cmdname] parents = [parent_parser] p = subparsers.add_parser(cmdname, parents=parents, description=cmd.description, help=cmd.description) p.set_defaults(func=run_cmdline) if cmd.requires_network: add_offline_option(p) if cmd.requires_wallet: add_wallet_options(p) if cmd.requires_password: p.add_argument("-W", "--password", dest="password", default=None, help="password") for optname in cmd.options: a, b, default, help = command_options[optname] action = "store_true" if type(default) is bool else 'store' args = (a, b) if a else (b,) p.add_argument(*args, dest=optname, action=action, default=default, help=help) for param in cmd.params: p.add_argument(param) # default command is gui parser.set_default_subparser('gui') return parser class Commands: def __init__(self, wallet, network, callback = None): self.wallet = wallet self.network = network self._callback = callback self.password = None def _run(self, method, args, password_getter): cmd = known_commands[method] if cmd.requires_password and self.wallet.use_encryption: self.password = apply(password_getter,()) f = getattr(self, method) result = f(*args) self.password = None if self._callback: apply(self._callback, ()) return result def make_seed(self, nbits, custom_entropy, language): from mnemonic import Mnemonic s = Mnemonic(language).make_seed(nbits, custom_entropy=custom_entropy) return s.encode('utf8') def check_seed(self, seed, custom_entropy, language): from mnemonic import Mnemonic return Mnemonic(language).check_seed(seed, custom_entropy) def getaddresshistory(self, addr): return self.network.synchronous_get([ ('blockchain.address.get_history',[addr]) ])[0] def listunspent(self): l = copy.deepcopy(self.wallet.get_spendable_coins()) for i in l: i["value"] = str(Decimal(i["value"])/100000000) return l def getaddressunspent(self, addr): return self.network.synchronous_get([ ('blockchain.address.listunspent',[addr]) ])[0] def getutxoaddress(self, txid, num): r = self.network.synchronous_get([ ('blockchain.utxo.get_address',[txid, num]) ]) if r: return {'address':r[0] } def createrawtransaction(self, inputs, outputs): coins = self.wallet.get_spendable_coins(None) tx_inputs = [] for i in inputs: prevout_hash = i['txid'] prevout_n = i['vout'] for c in coins: if c['prevout_hash'] == prevout_hash and c['prevout_n'] == prevout_n: self.wallet.add_input_info(c) tx_inputs.append(c) break else: raise BaseException('Transaction output not in wallet', prevout_hash+":%d"%prevout_n) outputs = map(lambda x: ('address', x[0], int(1e8*x[1])), outputs.items()) tx = Transaction.from_io(tx_inputs, outputs) return tx def signtxwithkey(self, raw_tx, sec): tx = Transaction(raw_tx) pubkey = bitcoin.public_key_from_private_key(sec) tx.sign({ pubkey:sec }) return tx def signtxwithwallet(self, raw_tx): tx = Transaction(raw_tx) tx.deserialize() self.wallet.sign_transaction(tx, self.password) return tx def decoderawtransaction(self, raw): tx = Transaction(raw) tx.deserialize() return {'inputs':tx.inputs, 'outputs':tx.outputs} def sendrawtransaction(self, raw): tx = Transaction(raw) return self.network.synchronous_get([('blockchain.transaction.broadcast', [str(tx)])])[0] def createmultisig(self, num, pubkeys): assert isinstance(pubkeys, list) redeem_script = Transaction.multisig_script(pubkeys, num) address = hash_160_to_bc_address(hash_160(redeem_script.decode('hex')), 5) return {'address':address, 'redeemScript':redeem_script} def freeze(self,addr): return self.wallet.freeze(addr) def unfreeze(self,addr): return self.wallet.unfreeze(addr) def getprivatekeys(self, addr): return self.wallet.get_private_key(addr, self.password) def ismine(self, addr): return self.wallet.is_mine(addr) def dumpprivkeys(self, addresses = None): if addresses is None: addresses = self.wallet.addresses(True) return [self.wallet.get_private_key(address, self.password) for address in addresses] def validateaddress(self, addr): isvalid = is_valid(addr) out = { 'isvalid':isvalid } if isvalid: out['address'] = addr return out def getpubkeys(self, addr): out = { 'address':addr } out['pubkeys'] = self.wallet.get_public_keys(addr) return out def getbalance(self, account= None): if account is None: c, u, x = self.wallet.get_balance() else: c, u, x = self.wallet.get_account_balance(account) out = {"confirmed": str(Decimal(c)/100000000)} if u: out["unconfirmed"] = str(Decimal(u)/100000000) if x: out["unmatured"] = str(Decimal(x)/100000000) return out def getaddressbalance(self, addr): out = self.network.synchronous_get([ ('blockchain.address.get_balance',[addr]) ])[0] out["confirmed"] = str(Decimal(out["confirmed"])/100000000) out["unconfirmed"] = str(Decimal(out["unconfirmed"])/100000000) return out def getproof(self, addr): p = self.network.synchronous_get([ ('blockchain.address.get_proof',[addr]) ])[0] out = [] for i,s in p: out.append(i) return out def getmerkle(self, txid, height): return self.network.synchronous_get([ ('blockchain.transaction.get_merkle', [txid, int(height)]) ])[0] def getservers(self): while not self.network.is_up_to_date(): time.sleep(0.1) return self.network.get_servers() def version(self): import electrum # Needs to stay here to prevent ciruclar imports return electrum.ELECTRUM_VERSION def getmpk(self): return self.wallet.get_master_public_keys() def getseed(self): s = self.wallet.get_mnemonic(self.password) return s.encode('utf8') def importprivkey(self, sec): try: addr = self.wallet.import_key(sec,self.password) out = "Keypair imported: ", addr except Exception as e: out = "Error: Keypair import failed: " + str(e) return out def sweep(self, privkey, to_address, fee = 0.0001): fee = int(Decimal(fee)*100000000) return Transaction.sweep([privkey], self.network, to_address, fee) def signmessage(self, address, message): return self.wallet.sign_message(address, message, self.password) def verifymessage(self, address, signature, message): return bitcoin.verify_message(address, signature, message) def _mktx(self, outputs, fee = None, change_addr = None, domain = None): for to_address, amount in outputs: if not is_valid(to_address): raise Exception("Invalid Bitcoin address", to_address) if change_addr: if not is_valid(change_addr): raise Exception("Invalid Bitcoin address", change_addr) if domain is not None: for addr in domain: if not is_valid(addr): raise Exception("invalid Bitcoin address", addr) if not self.wallet.is_mine(addr): raise Exception("address not in wallet", addr) for k, v in self.wallet.labels.items(): if change_addr and v == change_addr: change_addr = k final_outputs = [] for to_address, amount in outputs: for k, v in self.wallet.labels.items(): if v == to_address: to_address = k print_msg("alias", to_address) break amount = int(100000000*amount) final_outputs.append(('address', to_address, amount)) if fee is not None: fee = int(100000000*fee) return self.wallet.mktx(final_outputs, self.password, fee , change_addr, domain) def mktx(self, to_address, amount, fee = None, change_addr = None, domain = None): tx = self._mktx([(to_address, amount)], fee, change_addr, domain) return tx def _read_csv(self, csvpath): import csv outputs = [] with open(csvpath, 'rb') as csvfile: csvReader = csv.reader(csvfile, delimiter=',') for row in csvReader: address, amount = row assert bitcoin.is_address(address) amount = Decimal(amount) outputs.append((address, amount)) return outputs def mktx_csv(self, path, fee = None, change_addr = None, domain = None): outputs = self._read_csv(path) print outputs tx = self._mktx(outputs, fee, change_addr, domain) return tx def payto(self, to_address, amount, fee = None, change_addr = None, domain = None): tx = self._mktx([(to_address, amount)], fee, change_addr, domain) r, h = self.wallet.sendtx( tx ) return h def payto_csv(self, path, fee = None, change_addr = None, domain = None): outputs = self._read_csv(path) tx = self._mktx(outputs, fee, change_addr, domain) r, h = self.wallet.sendtx( tx ) return h def history(self): balance = 0 out = [] for item in self.wallet.get_history(): tx_hash, conf, value, timestamp, balance = item try: time_str = datetime.datetime.fromtimestamp( timestamp).isoformat(' ')[:-3] except Exception: time_str = "----" label, is_default_label = self.wallet.get_label(tx_hash) out.append({'txid':tx_hash, 'date':"%16s"%time_str, 'label':label, 'value':format_satoshis(value), 'confirmations':conf}) return out def setlabel(self, key, label): self.wallet.set_label(key, label) def contacts(self): c = {} for addr in self.wallet.addressbook: c[addr] = self.wallet.labels.get(addr) return c def searchcontacts(self, query): results = {} for addr in self.wallet.addressbook: if query.lower() in self.wallet.labels.get(addr).lower(): results[addr] = self.wallet.labels.get(addr) return results def listaddresses(self, show_all = False, show_label = False): out = [] for addr in self.wallet.addresses(True): if show_all or not self.wallet.is_change(addr): if show_label: item = { 'address': addr } if show_label: label = self.wallet.labels.get(addr,'') if label: item['label'] = label else: item = addr out.append( item ) return out def getrawtransaction(self, tx_hash): if self.wallet: tx = self.wallet.transactions.get(tx_hash) if tx: return tx raw = self.network.synchronous_get([ ('blockchain.transaction.get',[tx_hash]) ])[0] if raw: return Transaction(raw) else: return "unknown transaction" def encrypt(self, pubkey, message): return bitcoin.encrypt_message(message, pubkey) def decrypt(self, pubkey, message): return self.wallet.decrypt_message(pubkey, message, self.password)