|
@ -9,13 +9,14 @@ var sinon = require('sinon'); |
|
|
var should = chai.should(); |
|
|
var should = chai.should(); |
|
|
var levelup = require('levelup'); |
|
|
var levelup = require('levelup'); |
|
|
var memdown = require('memdown'); |
|
|
var memdown = require('memdown'); |
|
|
|
|
|
var mongodb = require('mongodb'); |
|
|
var log = require('npmlog'); |
|
|
var log = require('npmlog'); |
|
|
log.debug = log.verbose; |
|
|
log.debug = log.verbose; |
|
|
|
|
|
|
|
|
var Utils = require('../../lib/utils'); |
|
|
var Utils = require('../../lib/utils'); |
|
|
var WalletUtils = require('bitcore-wallet-utils'); |
|
|
var WalletUtils = require('bitcore-wallet-utils'); |
|
|
var Bitcore = WalletUtils.Bitcore; |
|
|
var Bitcore = WalletUtils.Bitcore; |
|
|
var Storage = require('../../lib/storage'); |
|
|
var Storage = require('../../lib/storage_mongo'); |
|
|
var BlockchainMonitor = require('../../lib/blockchainmonitor'); |
|
|
var BlockchainMonitor = require('../../lib/blockchainmonitor'); |
|
|
|
|
|
|
|
|
var Wallet = require('../../lib/model/wallet'); |
|
|
var Wallet = require('../../lib/model/wallet'); |
|
@ -208,24 +209,56 @@ helpers.createAddresses = function(server, wallet, main, change, cb) { |
|
|
|
|
|
|
|
|
var db, storage, blockchainExplorer; |
|
|
var db, storage, blockchainExplorer; |
|
|
|
|
|
|
|
|
|
|
|
function openDb(cb) { |
|
|
describe('Wallet service', function() { |
|
|
function dropDb(cb) { |
|
|
beforeEach(function() { |
|
|
db.dropDatabase(function(err) { |
|
|
db = levelup(memdown, { |
|
|
should.not.exist(err); |
|
|
valueEncoding: 'json' |
|
|
return cb(); |
|
|
}); |
|
|
}); |
|
|
storage = new Storage({ |
|
|
}; |
|
|
db: db |
|
|
if (db) { |
|
|
|
|
|
return dropDb(cb); |
|
|
|
|
|
} else { |
|
|
|
|
|
var url = 'mongodb://localhost:27017/bws'; |
|
|
|
|
|
mongodb.MongoClient.connect(url, function(err, _db) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
db = _db; |
|
|
|
|
|
return dropDb(cb); |
|
|
}); |
|
|
}); |
|
|
blockchainExplorer = sinon.stub(); |
|
|
} |
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
WalletService.initialize({ |
|
|
function closeDb(cb) { |
|
|
storage: storage, |
|
|
if (db) { |
|
|
blockchainExplorer: blockchainExplorer, |
|
|
db.close(true, function(err) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
db = null; |
|
|
|
|
|
return cb(); |
|
|
}); |
|
|
}); |
|
|
helpers.offset = 0; |
|
|
} else { |
|
|
}); |
|
|
return cb(); |
|
|
|
|
|
} |
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
describe('Wallet service', function() { |
|
|
|
|
|
beforeEach(function(done) { |
|
|
|
|
|
openDb(function() { |
|
|
|
|
|
storage = new Storage({ |
|
|
|
|
|
db: db |
|
|
|
|
|
}); |
|
|
|
|
|
blockchainExplorer = sinon.stub(); |
|
|
|
|
|
|
|
|
|
|
|
WalletService.initialize({ |
|
|
|
|
|
storage: storage, |
|
|
|
|
|
blockchainExplorer: blockchainExplorer, |
|
|
|
|
|
}); |
|
|
|
|
|
helpers.offset = 0; |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
after(function(done) { |
|
|
|
|
|
closeDb(done); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
describe('#getInstanceWithAuth', function() { |
|
|
describe('#getInstanceWithAuth', function() { |
|
|
beforeEach(function() {}); |
|
|
beforeEach(function() {}); |
|
@ -1172,6 +1205,7 @@ describe('Wallet service', function() { |
|
|
helpers.stubUtxos(server, wallet, _.range(1, 9), function() { |
|
|
helpers.stubUtxos(server, wallet, _.range(1, 9), function() { |
|
|
var txOpts = helpers.createProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 10, null, TestData.copayers[0].privKey_1H_0); |
|
|
var txOpts = helpers.createProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 10, null, TestData.copayers[0].privKey_1H_0); |
|
|
server.createTx(txOpts, function(err, tx) { |
|
|
server.createTx(txOpts, function(err, tx) { |
|
|
|
|
|
|
|
|
should.not.exist(err); |
|
|
should.not.exist(err); |
|
|
should.exist(tx); |
|
|
should.exist(tx); |
|
|
txid = tx.id; |
|
|
txid = tx.id; |
|
@ -1814,9 +1848,7 @@ describe('Wallet service', function() { |
|
|
var server, wallet, clock; |
|
|
var server, wallet, clock; |
|
|
|
|
|
|
|
|
beforeEach(function(done) { |
|
|
beforeEach(function(done) { |
|
|
if (server) return done(); |
|
|
|
|
|
this.timeout(5000); |
|
|
this.timeout(5000); |
|
|
console.log('\tCreating TXS...'); |
|
|
|
|
|
clock = sinon.useFakeTimers(); |
|
|
clock = sinon.useFakeTimers(); |
|
|
helpers.createAndJoinWallet(1, 1, function(s, w) { |
|
|
helpers.createAndJoinWallet(1, 1, function(s, w) { |
|
|
server = s; |
|
|
server = s; |
|
@ -1824,7 +1856,7 @@ describe('Wallet service', function() { |
|
|
helpers.stubUtxos(server, wallet, _.range(10), function() { |
|
|
helpers.stubUtxos(server, wallet, _.range(10), function() { |
|
|
var txOpts = helpers.createProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 0.1, null, TestData.copayers[0].privKey_1H_0); |
|
|
var txOpts = helpers.createProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 0.1, null, TestData.copayers[0].privKey_1H_0); |
|
|
async.eachSeries(_.range(10), function(i, next) { |
|
|
async.eachSeries(_.range(10), function(i, next) { |
|
|
clock.tick(10000); |
|
|
clock.tick(10 * 1000); |
|
|
server.createTx(txOpts, function(err, tx) { |
|
|
server.createTx(txOpts, function(err, tx) { |
|
|
next(); |
|
|
next(); |
|
|
}); |
|
|
}); |
|
@ -1883,17 +1915,18 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
it('should txs from times 50 to 70', function(done) { |
|
|
it('should txs from times 50 to 70', |
|
|
server.getTxs({ |
|
|
function(done) { |
|
|
minTs: 50, |
|
|
server.getTxs({ |
|
|
maxTs: 70, |
|
|
minTs: 50, |
|
|
}, function(err, txps) { |
|
|
maxTs: 70, |
|
|
should.not.exist(err); |
|
|
}, function(err, txps) { |
|
|
var times = _.pluck(txps, 'createdOn'); |
|
|
should.not.exist(err); |
|
|
times.should.deep.equal([70, 60, 50]); |
|
|
var times = _.pluck(txps, 'createdOn'); |
|
|
done(); |
|
|
times.should.deep.equal([70, 60, 50]); |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
describe('Notifications', function() { |
|
|
describe('Notifications', function() { |
|
@ -2073,62 +2106,110 @@ describe('Wallet service', function() { |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
it('should delete a wallet', function(done) { |
|
|
it('should delete a wallet', function(done) { |
|
|
var i = 0; |
|
|
server.removeWallet({}, function(err) { |
|
|
var count = function() { |
|
|
should.not.exist(err); |
|
|
return ++i; |
|
|
server.getWallet({}, function(err, w) { |
|
|
}; |
|
|
should.exist(err); |
|
|
server.storage._dump(function() { |
|
|
err.message.should.equal('Wallet not found'); |
|
|
i.should.above(1); |
|
|
should.not.exist(w); |
|
|
server.removeWallet({}, function(err) { |
|
|
async.parallel([ |
|
|
i = 0; |
|
|
|
|
|
server.storage._dump(function() { |
|
|
function(next) { |
|
|
server.storage._dump(); |
|
|
server.storage.fetchAddresses(wallet.id, function(err, items) { |
|
|
i.should.equal(0); |
|
|
items.length.should.equal(0); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server.storage.fetchTxs(wallet.id, {}, function(err, items) { |
|
|
|
|
|
items.length.should.equal(0); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server.storage.fetchNotifications(wallet.id, {}, function(err, items) { |
|
|
|
|
|
items.length.should.equal(0); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
], function(err) { |
|
|
|
|
|
should.not.exist(err); |
|
|
done(); |
|
|
done(); |
|
|
}, count); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}, count); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
// creates 2 wallet, and deletes only 1.
|
|
|
// creates 2 wallet, and deletes only 1.
|
|
|
it('should delete a wallet, and only that wallet', function(done) { |
|
|
it('should delete a wallet, and only that wallet', function(done) { |
|
|
var i = 0; |
|
|
var server2, wallet2; |
|
|
var db = []; |
|
|
async.series([ |
|
|
var cat = function(data) { |
|
|
|
|
|
db.push(data); |
|
|
function(next) { |
|
|
}; |
|
|
helpers.offset = 1; |
|
|
server.storage._dump(function() { |
|
|
helpers.createAndJoinWallet(1, 1, function(s, w) { |
|
|
var before = _.clone(db); |
|
|
server2 = s; |
|
|
db.length.should.above(1); |
|
|
wallet2 = w; |
|
|
|
|
|
|
|
|
helpers.offset = 1; |
|
|
helpers.stubUtxos(server2, wallet2, _.range(1, 3), function() { |
|
|
helpers.createAndJoinWallet(2, 3, function(s, w) { |
|
|
var txOpts = helpers.createProposalOpts('18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', 0.1, 'some message', TestData.copayers[1].privKey_1H_0); |
|
|
server = s; |
|
|
async.eachSeries(_.range(2), function(i, next) { |
|
|
wallet = w; |
|
|
server2.createTx(txOpts, function(err, tx) { |
|
|
|
|
|
should.not.exist(err); |
|
|
helpers.stubUtxos(server, wallet, _.range(2), function() { |
|
|
next(err); |
|
|
var txOpts = { |
|
|
}); |
|
|
toAddress: '18PzpUFkFZE8zKWUPvfykkTxmB9oMR8qP7', |
|
|
}, next); |
|
|
amount: helpers.toSatoshi(0.1), |
|
|
}); |
|
|
}; |
|
|
|
|
|
async.eachSeries(_.range(2), function(i, next) { |
|
|
|
|
|
server.createTx(txOpts, function(err, tx) { |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, function() { |
|
|
|
|
|
server.removeWallet({}, function(err) { |
|
|
|
|
|
db = []; |
|
|
|
|
|
server.storage._dump(function() { |
|
|
|
|
|
var after = _.clone(db); |
|
|
|
|
|
after.should.deep.equal(before); |
|
|
|
|
|
done(); |
|
|
|
|
|
}, cat); |
|
|
|
|
|
}); |
|
|
|
|
|
}, cat); |
|
|
|
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}, |
|
|
}, cat); |
|
|
function(next) { |
|
|
|
|
|
server.removeWallet({}, next); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server.getWallet({}, function(err, wallet) { |
|
|
|
|
|
should.exist(err); |
|
|
|
|
|
err.message.should.contain('not found'); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server2.getWallet({}, function(err, wallet) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
should.exist(wallet); |
|
|
|
|
|
wallet.id.should.equal(wallet2.id); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server2.getMainAddresses({}, function(err, addresses) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
should.exist(addresses); |
|
|
|
|
|
addresses.length.should.above(0); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server2.getTxs({}, function(err, txs) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
should.exist(txs); |
|
|
|
|
|
txs.length.should.equal(2); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
function(next) { |
|
|
|
|
|
server2.getNotifications({}, function(err, notifications) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
should.exist(notifications); |
|
|
|
|
|
notifications.length.should.above(0); |
|
|
|
|
|
next(); |
|
|
|
|
|
}); |
|
|
|
|
|
}, |
|
|
|
|
|
], function(err) { |
|
|
|
|
|
should.not.exist(err); |
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
@ -2967,29 +3048,32 @@ describe('Wallet service', function() { |
|
|
describe('Blockchain monitor', function() { |
|
|
describe('Blockchain monitor', function() { |
|
|
var addressSubscriber; |
|
|
var addressSubscriber; |
|
|
|
|
|
|
|
|
beforeEach(function() { |
|
|
beforeEach(function(done) { |
|
|
db = levelup(memdown, { |
|
|
|
|
|
valueEncoding: 'json' |
|
|
|
|
|
}); |
|
|
|
|
|
storage = new Storage({ |
|
|
|
|
|
db: db |
|
|
|
|
|
}); |
|
|
|
|
|
blockchainExplorer = sinon.stub(); |
|
|
|
|
|
|
|
|
|
|
|
WalletService.initialize({ |
|
|
|
|
|
storage: storage, |
|
|
|
|
|
blockchainExplorer: blockchainExplorer, |
|
|
|
|
|
}); |
|
|
|
|
|
helpers.offset = 0; |
|
|
|
|
|
|
|
|
|
|
|
addressSubscriber = sinon.stub(); |
|
|
addressSubscriber = sinon.stub(); |
|
|
addressSubscriber.subscribe = sinon.stub(); |
|
|
addressSubscriber.subscribe = sinon.stub(); |
|
|
sinon.stub(BlockchainMonitor.prototype, '_getAddressSubscriber').onFirstCall().returns(addressSubscriber); |
|
|
sinon.stub(BlockchainMonitor.prototype, '_getAddressSubscriber').onFirstCall().returns(addressSubscriber); |
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
openDb(function() { |
|
|
|
|
|
storage = new Storage({ |
|
|
|
|
|
db: db |
|
|
|
|
|
}); |
|
|
|
|
|
blockchainExplorer = sinon.stub(); |
|
|
|
|
|
|
|
|
|
|
|
WalletService.initialize({ |
|
|
|
|
|
storage: storage, |
|
|
|
|
|
blockchainExplorer: blockchainExplorer, |
|
|
|
|
|
}); |
|
|
|
|
|
helpers.offset = 0; |
|
|
|
|
|
|
|
|
|
|
|
done(); |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
afterEach(function() { |
|
|
afterEach(function() { |
|
|
BlockchainMonitor.prototype._getAddressSubscriber.restore(); |
|
|
BlockchainMonitor.prototype._getAddressSubscriber.restore(); |
|
|
}); |
|
|
}); |
|
|
|
|
|
after(function(done) { |
|
|
|
|
|
closeDb(done); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
it('should subscribe wallet', function(done) { |
|
|
it('should subscribe wallet', function(done) { |
|
|
var monitor = new BlockchainMonitor(); |
|
|
var monitor = new BlockchainMonitor(); |
|
|