From 50e9a09a8c572b45f68f7e3361d4a41aeb740761 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Tue, 20 May 2014 15:11:17 +1000 Subject: [PATCH] Wallet: cleanup createTx control flow Unknowingly this also revealed a subtle bug in the previous implementation which allowed the creation of transactions even when no UTXOs existed. --- src/wallet.js | 49 +++++++++++++++++++++---------------------------- test/wallet.js | 18 ++++++++++++++---- 2 files changed, 35 insertions(+), 32 deletions(-) diff --git a/src/wallet.js b/src/wallet.js index cb5963e..e29409d 100644 --- a/src/wallet.js +++ b/src/wallet.js @@ -179,34 +179,40 @@ function Wallet(seed, options) { } this.createTx = function(to, value, fixedFee, changeAddress) { - checkDust(value) + if (isDust(value)) throw new Error("Value must be above dust threshold") + + var utxos = getCandidateOutputs(value) + var accum = 0 + var subTotal = value var tx = new Transaction() tx.addOutput(to, value) - var utxo = getCandidateOutputs(value) - var totalInValue = 0 - for(var i=0; i 0 && !isDust(change)) { - tx.addOutput(changeAddress || getChangeAddress(), change) + subTotal = value + fee + if (accum >= subTotal) { + var change = accum - subTotal + + if (!isDust(change)) { + tx.addOutput(changeAddress || getChangeAddress(), change) + } + + break } - break } - checkInsufficientFund(totalInValue, value, fee) + if (accum < subTotal) { + throw new Error('Not enough funds: ' + accum + ' < ' + subTotal) + } this.sign(tx) - return tx } @@ -232,12 +238,6 @@ function Wallet(seed, options) { return amount <= me.dustThreshold } - function checkDust(value){ - if (isNullOrUndefined(value) || isDust(value)) { - throw new Error("Value must be above dust threshold") - } - } - function getCandidateOutputs(value){ var unspent = [] for (var key in me.outputs){ @@ -263,13 +263,6 @@ function Wallet(seed, options) { return me.changeAddresses[me.changeAddresses.length - 1] } - function checkInsufficientFund(totalInValue, value, fee) { - if(totalInValue < value + fee) { - throw new Error('Not enough money to send funds including transaction fee. Have: ' + - totalInValue + ', needed: ' + (value + fee)) - } - } - this.sign = function(tx) { tx.ins.forEach(function(inp,i) { var output = me.outputs[inp.outpoint.hash + ':' + inp.outpoint.index] diff --git a/test/wallet.js b/test/wallet.js index 567bfd4..c91e422 100644 --- a/test/wallet.js +++ b/test/wallet.js @@ -492,13 +492,23 @@ describe('Wallet', function() { describe('testnet', function(){ it('should create transaction', function(){ - var to = 'mt7MyTVVEWnbwpF5hBn6fgnJcv95Syk2ue' var wallet = new Wallet(seed, {network: 'testnet'}) - var tx = wallet.createTx(to, value) + var address = wallet.generateAddress() + wallet.setUnspentOutputs([{ + hash: fakeTxHash(0), + outputIndex: 0, + address: address, + value: value + }]) + + var to = 'mt7MyTVVEWnbwpF5hBn6fgnJcv95Syk2ue' + var toValue = value - 20000 + + var tx = wallet.createTx(to, toValue) assert.equal(tx.outs.length, 1) assert.equal(tx.outs[0].address.toString(), to) - assert.equal(tx.outs[0].value, value) + assert.equal(tx.outs[0].value, toValue) }) }) @@ -605,7 +615,7 @@ describe('Wallet', function() { assert.throws(function() { wallet.createTx(to, value) - }, /Not enough money to send funds including transaction fee. Have: 1420000, needed: 1420001/) + }, /Not enough funds: 1420000 < 1420001/) }) }) })