|
@ -181,8 +181,9 @@ helpers.stubUtxos = function(server, wallet, amounts, cb) { |
|
|
}); |
|
|
}); |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
helpers.stubBroadcast = function(txid) { |
|
|
helpers.stubBroadcast = function(thirdPartyBroadcast) { |
|
|
blockchainExplorer.broadcast = sinon.stub().callsArgWith(1, null, txid); |
|
|
blockchainExplorer.broadcast = sinon.stub().callsArgWith(1, null, '112233'); |
|
|
|
|
|
blockchainExplorer.getTransaction = sinon.stub().callsArgWith(1, null, null); |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
helpers.stubHistory = function(txs) { |
|
|
helpers.stubHistory = function(txs) { |
|
@ -485,14 +486,14 @@ describe('Wallet service', function() { |
|
|
message: 'some message' |
|
|
message: 'some message' |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
var txpId; |
|
|
var txp; |
|
|
async.waterfall([ |
|
|
async.waterfall([ |
|
|
|
|
|
|
|
|
function(next) { |
|
|
function(next) { |
|
|
server.createTx(txOpts, next); |
|
|
server.createTx(txOpts, next); |
|
|
}, |
|
|
}, |
|
|
function(txp, next) { |
|
|
function(t, next) { |
|
|
txpId = txp.id; |
|
|
txp = t; |
|
|
async.eachSeries(_.range(2), function(i, next) { |
|
|
async.eachSeries(_.range(2), function(i, next) { |
|
|
var copayer = TestData.copayers[i]; |
|
|
var copayer = TestData.copayers[i]; |
|
|
helpers.getAuthServer(copayer.id45, function(server) { |
|
|
helpers.getAuthServer(copayer.id45, function(server) { |
|
@ -500,14 +501,17 @@ describe('Wallet service', function() { |
|
|
server.signTx({ |
|
|
server.signTx({ |
|
|
txProposalId: txp.id, |
|
|
txProposalId: txp.id, |
|
|
signatures: signatures, |
|
|
signatures: signatures, |
|
|
}, next); |
|
|
}, function(err, t) { |
|
|
|
|
|
txp = t; |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}, next); |
|
|
}, next); |
|
|
}, |
|
|
}, |
|
|
function(next) { |
|
|
function(next) { |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastTx({ |
|
|
server.broadcastTx({ |
|
|
txProposalId: txpId, |
|
|
txProposalId: txp.id, |
|
|
}, next); |
|
|
}, next); |
|
|
}, |
|
|
}, |
|
|
], function(err) { |
|
|
], function(err) { |
|
@ -525,7 +529,7 @@ describe('Wallet service', function() { |
|
|
one.text.should.contain(wallet.name); |
|
|
one.text.should.contain(wallet.name); |
|
|
one.text.should.contain('800,000'); |
|
|
one.text.should.contain('800,000'); |
|
|
should.exist(one.html); |
|
|
should.exist(one.html); |
|
|
one.html.should.contain('https://insight.bitpay.com/tx/999'); |
|
|
one.html.should.contain('https://insight.bitpay.com/tx/' + txp.txid); |
|
|
server.storage.fetchUnsentEmails(function(err, unsent) { |
|
|
server.storage.fetchUnsentEmails(function(err, unsent) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
unsent.should.be.empty; |
|
|
unsent.should.be.empty; |
|
@ -2817,6 +2821,7 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
it('should sign a TX with multiple inputs, different paths, and return raw', function(done) { |
|
|
it('should sign a TX with multiple inputs, different paths, and return raw', function(done) { |
|
|
|
|
|
blockchainExplorer.getTransaction = sinon.stub().callsArgWith(1, null, null); |
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
var tx = txs[0]; |
|
|
var tx = txs[0]; |
|
|
tx.id.should.equal(txid); |
|
|
tx.id.should.equal(txid); |
|
@ -3053,7 +3058,7 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
describe('#broadcastTx & #broadcastRawTx', function() { |
|
|
describe('#broadcastTx & #broadcastRawTx', function() { |
|
|
var server, wallet, txpid; |
|
|
var server, wallet, txpid, txid; |
|
|
beforeEach(function(done) { |
|
|
beforeEach(function(done) { |
|
|
helpers.createAndJoinWallet(1, 1, function(s, w) { |
|
|
helpers.createAndJoinWallet(1, 1, function(s, w) { |
|
|
server = s; |
|
|
server = s; |
|
@ -3074,6 +3079,7 @@ describe('Wallet service', function() { |
|
|
should.exist(txp); |
|
|
should.exist(txp); |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
|
|
|
txid = txp.txid; |
|
|
txpid = txp.id; |
|
|
txpid = txp.id; |
|
|
done(); |
|
|
done(); |
|
|
}); |
|
|
}); |
|
@ -3084,7 +3090,7 @@ describe('Wallet service', function() { |
|
|
|
|
|
|
|
|
it('should broadcast a tx', function(done) { |
|
|
it('should broadcast a tx', function(done) { |
|
|
var clock = sinon.useFakeTimers(1234000, 'Date'); |
|
|
var clock = sinon.useFakeTimers(1234000, 'Date'); |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastTx({ |
|
|
server.broadcastTx({ |
|
|
txProposalId: txpid |
|
|
txProposalId: txpid |
|
|
}, function(err) { |
|
|
}, function(err) { |
|
@ -3094,7 +3100,7 @@ describe('Wallet service', function() { |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
should.not.exist(txp.raw); |
|
|
should.not.exist(txp.raw); |
|
|
txp.txid.should.equal('999'); |
|
|
txp.txid.should.equal(txid); |
|
|
txp.isBroadcasted().should.be.true; |
|
|
txp.isBroadcasted().should.be.true; |
|
|
txp.broadcastedOn.should.equal(1234); |
|
|
txp.broadcastedOn.should.equal(1234); |
|
|
clock.restore(); |
|
|
clock.restore(); |
|
@ -3104,19 +3110,19 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
it('should broadcast a raw tx', function(done) { |
|
|
it('should broadcast a raw tx', function(done) { |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastRawTx({ |
|
|
server.broadcastRawTx({ |
|
|
network: 'testnet', |
|
|
network: 'testnet', |
|
|
rawTx: 'raw tx', |
|
|
rawTx: 'raw tx', |
|
|
}, function(err, txid) { |
|
|
}, function(err, txid) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
txid.should.equal('999'); |
|
|
should.exist(txid); |
|
|
done(); |
|
|
done(); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
it('should fail to brodcast a tx already marked as broadcasted', function(done) { |
|
|
it('should fail to brodcast a tx already marked as broadcasted', function(done) { |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastTx({ |
|
|
server.broadcastTx({ |
|
|
txProposalId: txpid |
|
|
txProposalId: txpid |
|
|
}, function(err) { |
|
|
}, function(err) { |
|
@ -3131,8 +3137,52 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
it('should auto process already broadcasted txs', function(done) { |
|
|
|
|
|
helpers.stubBroadcast(); |
|
|
|
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
txs.length.should.equal(1); |
|
|
|
|
|
blockchainExplorer.getTransaction = sinon.stub().callsArgWith(1, null, { |
|
|
|
|
|
txid: 999 |
|
|
|
|
|
}); |
|
|
|
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
txs.length.should.equal(0); |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
it('should process only broadcasted txs', function(done) { |
|
|
|
|
|
helpers.stubBroadcast(); |
|
|
|
|
|
var txOpts = helpers.createSimpleProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 9, TestData.copayers[0].privKey_1H_0, { |
|
|
|
|
|
message: 'some message 2' |
|
|
|
|
|
}); |
|
|
|
|
|
server.createTx(txOpts, function(err, txp) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
txs.length.should.equal(2); |
|
|
|
|
|
blockchainExplorer.getTransaction = sinon.stub().callsArgWith(1, null, { |
|
|
|
|
|
txid: 999 |
|
|
|
|
|
}); |
|
|
|
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
txs.length.should.equal(1); |
|
|
|
|
|
txs[0].status.should.equal('pending'); |
|
|
|
|
|
should.not.exist(txs[0].txid); |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
it('should fail to brodcast a not yet accepted tx', function(done) { |
|
|
it('should fail to brodcast a not yet accepted tx', function(done) { |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
var txOpts = helpers.createSimpleProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 9, TestData.copayers[0].privKey_1H_0, { |
|
|
var txOpts = helpers.createSimpleProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 9, TestData.copayers[0].privKey_1H_0, { |
|
|
message: 'some message' |
|
|
message: 'some message' |
|
|
}); |
|
|
}); |
|
@ -3161,7 +3211,7 @@ describe('Wallet service', function() { |
|
|
txProposalId: txpid |
|
|
txProposalId: txpid |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
should.not.exist(txp.txid); |
|
|
should.exist(txp.txid); |
|
|
txp.isBroadcasted().should.be.false; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
should.not.exist(txp.broadcastedOn); |
|
|
should.not.exist(txp.broadcastedOn); |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isAccepted().should.be.true; |
|
@ -3184,7 +3234,6 @@ describe('Wallet service', function() { |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
should.exist(txp.txid); |
|
|
should.exist(txp.txid); |
|
|
txp.txid.should.equal('999'); |
|
|
|
|
|
txp.isBroadcasted().should.be.true; |
|
|
txp.isBroadcasted().should.be.true; |
|
|
should.exist(txp.broadcastedOn); |
|
|
should.exist(txp.broadcastedOn); |
|
|
done(); |
|
|
done(); |
|
@ -3204,7 +3253,7 @@ describe('Wallet service', function() { |
|
|
txProposalId: txpid |
|
|
txProposalId: txpid |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
should.not.exist(txp.txid); |
|
|
should.exist(txp.txid); |
|
|
txp.isBroadcasted().should.be.false; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
should.not.exist(txp.broadcastedOn); |
|
|
should.not.exist(txp.broadcastedOn); |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isAccepted().should.be.true; |
|
@ -3221,7 +3270,7 @@ describe('Wallet service', function() { |
|
|
server = s; |
|
|
server = s; |
|
|
wallet = w; |
|
|
wallet = w; |
|
|
helpers.stubUtxos(server, wallet, _.range(1, 9), function() { |
|
|
helpers.stubUtxos(server, wallet, _.range(1, 9), function() { |
|
|
helpers.stubBroadcast('999'); |
|
|
helpers.stubBroadcast(); |
|
|
done(); |
|
|
done(); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
@ -3320,7 +3369,7 @@ describe('Wallet service', function() { |
|
|
txp.isPending().should.be.true; |
|
|
txp.isPending().should.be.true; |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isAccepted().should.be.true; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
txp.isBroadcasted().should.be.false; |
|
|
should.not.exist(txp.txid); |
|
|
should.exist(txp.txid); |
|
|
txp.actions.length.should.equal(2); |
|
|
txp.actions.length.should.equal(2); |
|
|
server.getNotifications({}, function(err, notifications) { |
|
|
server.getNotifications({}, function(err, notifications) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
@ -3692,7 +3741,7 @@ describe('Wallet service', function() { |
|
|
signatures: signatures, |
|
|
signatures: signatures, |
|
|
}, function(err) { |
|
|
}, function(err) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
helpers.stubBroadcast('1122334455'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastTx({ |
|
|
server.broadcastTx({ |
|
|
txProposalId: tx.id |
|
|
txProposalId: tx.id |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
@ -3710,6 +3759,38 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
it('should notify sign, acceptance, and broadcast, and emit (with 3rd party broadcast', function(done) { |
|
|
|
|
|
server.getPendingTxs({}, function(err, txs) { |
|
|
|
|
|
var tx = txs[2]; |
|
|
|
|
|
var signatures = helpers.clientSign(tx, TestData.copayers[0].xPrivKey); |
|
|
|
|
|
server.signTx({ |
|
|
|
|
|
txProposalId: tx.id, |
|
|
|
|
|
signatures: signatures, |
|
|
|
|
|
}, function(err) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
blockchainExplorer.broadcast = sinon.stub().callsArgWith(1, 'err'); |
|
|
|
|
|
blockchainExplorer.getTransaction = sinon.stub().callsArgWith(1, null, { |
|
|
|
|
|
txid: 11 |
|
|
|
|
|
}); |
|
|
|
|
|
server.broadcastTx({ |
|
|
|
|
|
txProposalId: tx.id |
|
|
|
|
|
}, function(err, txp) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
server.getNotifications({ |
|
|
|
|
|
limit: 3, |
|
|
|
|
|
reverse: true, |
|
|
|
|
|
}, function(err, notifications) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
var types = _.pluck(notifications, 'type'); |
|
|
|
|
|
types.should.deep.equal(['NewOutgoingTxByThirdParty', 'TxProposalFinallyAccepted', 'TxProposalAcceptedBy']); |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
describe('#removeWallet', function() { |
|
|
describe('#removeWallet', function() { |
|
@ -4192,13 +4273,13 @@ describe('Wallet service', function() { |
|
|
}, function(err, tx) { |
|
|
}, function(err, tx) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
|
|
|
|
|
|
helpers.stubBroadcast('1122334455'); |
|
|
helpers.stubBroadcast(); |
|
|
server.broadcastTx({ |
|
|
server.broadcastTx({ |
|
|
txProposalId: tx.id |
|
|
txProposalId: tx.id |
|
|
}, function(err, txp) { |
|
|
}, function(err, txp) { |
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
var txs = [{ |
|
|
var txs = [{ |
|
|
txid: '1122334455', |
|
|
txid: txp.txid, |
|
|
confirmations: 1, |
|
|
confirmations: 1, |
|
|
fees: 5460, |
|
|
fees: 5460, |
|
|
time: 1, |
|
|
time: 1, |
|
|