Browse Source

Simplify interface to make_tx in coin chooser

Makes the coin chooser code simpler and easier to understand.
283
Neil Booth 9 years ago
parent
commit
a6ea9a0c71
  1. 47
      lib/coinchooser.py
  2. 15
      lib/wallet.py

47
lib/coinchooser.py

@ -27,21 +27,17 @@ class CoinChooser(PrintError):
def __init__(self, wallet): def __init__(self, wallet):
self.wallet = wallet self.wallet = wallet
def fee(self, tx, fixed_fee, fee_per_kb): def make_tx(self, coins, outputs, change_addrs, fee_estimator,
if fixed_fee is not None: dust_threshold):
return fixed_fee '''Select unspent coins to spend to pay outputs. If the change is
return tx.estimated_fee(fee_per_kb) greater than dust_threshold (after adding the change output to
the transaction) it is kept, otherwise none is sent and it is
def dust_threshold(self): added to the transaction fee.'''
return 182 * 3 * MIN_RELAY_TX_FEE/1000
def make_tx(self, coins, outputs, change_addrs, fixed_fee, fee_per_kb):
'''Select unspent coins to spend to pay outputs.'''
amount = sum(map(lambda x: x[2], outputs)) amount = sum(map(lambda x: x[2], outputs))
total = 0 total = 0
inputs = [] inputs = []
tx = Transaction.from_io(inputs, outputs) tx = Transaction.from_io(inputs, outputs)
fee = self.fee(tx, fixed_fee, fee_per_kb) fee = fee_estimator(tx)
# add inputs, sorted by age # add inputs, sorted by age
for item in coins: for item in coins:
v = item.get('value') v = item.get('value')
@ -51,7 +47,7 @@ class CoinChooser(PrintError):
# no need to estimate fee until we have reached desired amount # no need to estimate fee until we have reached desired amount
if total < amount + fee: if total < amount + fee:
continue continue
fee = self.fee(tx, fixed_fee, fee_per_kb) fee = fee_estimator(tx)
if total >= amount + fee: if total >= amount + fee:
break break
else: else:
@ -69,13 +65,13 @@ class CoinChooser(PrintError):
else: else:
break break
if removed: if removed:
fee = self.fee(tx, fixed_fee, fee_per_kb) fee = fee_estimator(tx)
for item in sorted(tx.inputs, key=itemgetter('value')): for item in sorted(tx.inputs, key=itemgetter('value')):
v = item.get('value') v = item.get('value')
if total - v >= amount + fee: if total - v >= amount + fee:
tx.inputs.remove(item) tx.inputs.remove(item)
total -= v total -= v
fee = self.fee(tx, fixed_fee, fee_per_kb) fee = fee_estimator(tx)
continue continue
break break
self.print_error("using %d inputs" % len(tx.inputs)) self.print_error("using %d inputs" % len(tx.inputs))
@ -83,22 +79,21 @@ class CoinChooser(PrintError):
# if change is above dust threshold, add a change output. # if change is above dust threshold, add a change output.
change_addr = change_addrs[0] change_addr = change_addrs[0]
change_amount = total - (amount + fee) change_amount = total - (amount + fee)
if fixed_fee is not None and change_amount > 0:
tx.outputs.append(('address', change_addr, change_amount)) # See if change would still be greater than dust after adding
elif change_amount > self.dust_threshold(): # the change to the transaction
if change_amount > dust_threshold:
tx.outputs.append(('address', change_addr, change_amount)) tx.outputs.append(('address', change_addr, change_amount))
# recompute fee including change output fee = fee_estimator(tx)
fee = tx.estimated_fee(fee_per_kb)
# remove change output # remove change output
tx.outputs.pop() tx.outputs.pop()
# if change is still above dust threshold, re-add change output.
change_amount = total - (amount + fee) change_amount = total - (amount + fee)
if change_amount > self.dust_threshold():
tx.outputs.append(('address', change_addr, change_amount)) # If change is still above dust threshold, keep the change.
self.print_error('change', change_amount) if change_amount > dust_threshold:
else: tx.outputs.append(('address', change_addr, change_amount))
self.print_error('not keeping dust', change_amount) self.print_error('change', change_amount)
else: elif change_amount:
self.print_error('not keeping dust', change_amount) self.print_error('not keeping dust', change_amount)
return tx return tx

15
lib/wallet.py

@ -905,8 +905,6 @@ class Abstract_Wallet(PrintError):
if type == 'address': if type == 'address':
assert is_address(data), "Address " + data + " is invalid!" assert is_address(data), "Address " + data + " is invalid!"
fee_per_kb = self.fee_per_kb(config)
# change address # change address
if change_addr: if change_addr:
change_addrs = [change_addr] change_addrs = [change_addr]
@ -926,9 +924,20 @@ class Abstract_Wallet(PrintError):
else: else:
change_addrs = [address] change_addrs = [address]
fee_per_kb = self.fee_per_kb(config)
def fee_estimator(tx):
if fixed_fee is not None:
return fixed_fee
return tx.estimated_fee(fee_per_kb)
# If a fixed fee is specified, keep even dust change
dust_threshold = 182 * 3 * MIN_RELAY_TX_FEE / 1000
if fixed_fee is None:
dust_threshold = 0
# Let the coin chooser select the coins to spend # Let the coin chooser select the coins to spend
tx = self.coin_chooser.make_tx(coins, outputs, change_addrs, tx = self.coin_chooser.make_tx(coins, outputs, change_addrs,
fixed_fee, fee_per_kb) fee_estimator, dust_threshold)
# Sort the inputs and outputs deterministically # Sort the inputs and outputs deterministically
tx.BIP_LI01_sort() tx.BIP_LI01_sort()

Loading…
Cancel
Save