Matias Alejo Garcia
10 years ago
16 changed files with 1109 additions and 482 deletions
@ -0,0 +1,9 @@ |
|||||
|
var Model = {}; |
||||
|
|
||||
|
Model.Wallet = require('./wallet'); |
||||
|
Model.Copayer = require('./copayer'); |
||||
|
Model.TxProposal = require('./txproposal'); |
||||
|
Model.Address = require('./address'); |
||||
|
Model.Notification = require('./notification'); |
||||
|
|
||||
|
module.exports = Model; |
@ -0,0 +1,383 @@ |
|||||
|
'use strict'; |
||||
|
|
||||
|
var _ = require('lodash'); |
||||
|
var levelup = require('levelup'); |
||||
|
var net = require('net'); |
||||
|
var async = require('async'); |
||||
|
var $ = require('preconditions').singleton(); |
||||
|
var log = require('npmlog'); |
||||
|
var util = require('util'); |
||||
|
log.debug = log.verbose; |
||||
|
log.disableColor(); |
||||
|
|
||||
|
var Wallet = require('./model/wallet'); |
||||
|
var Copayer = require('./model/copayer'); |
||||
|
var Address = require('./model/address'); |
||||
|
var TxProposal = require('./model/txproposal'); |
||||
|
var Notification = require('./model/notification'); |
||||
|
|
||||
|
var Storage = function(opts) { |
||||
|
opts = opts || {}; |
||||
|
this.db = opts.db; |
||||
|
|
||||
|
if (!this.db) { |
||||
|
this.db = levelup(opts.dbPath || './db/bws.db', { |
||||
|
valueEncoding: 'json' |
||||
|
}); |
||||
|
} |
||||
|
}; |
||||
|
|
||||
|
var zeroPad = function(x, length) { |
||||
|
return _.padLeft(parseInt(x), length, '0'); |
||||
|
}; |
||||
|
|
||||
|
var walletPrefix = function(id) { |
||||
|
return 'w!' + id; |
||||
|
}; |
||||
|
|
||||
|
var opKey = function(key) { |
||||
|
return key ? '!' + key : ''; |
||||
|
}; |
||||
|
|
||||
|
var MAX_TS = _.repeat('9', 14); |
||||
|
|
||||
|
|
||||
|
var KEY = { |
||||
|
WALLET: function(walletId) { |
||||
|
return walletPrefix(walletId) + '!main'; |
||||
|
}, |
||||
|
COPAYER: function(id) { |
||||
|
return 'copayer!' + id; |
||||
|
}, |
||||
|
TXP: function(walletId, txProposalId) { |
||||
|
return walletPrefix(walletId) + '!txp' + opKey(txProposalId); |
||||
|
}, |
||||
|
NOTIFICATION: function(walletId, notificationId) { |
||||
|
return walletPrefix(walletId) + '!not' + opKey(notificationId); |
||||
|
}, |
||||
|
PENDING_TXP: function(walletId, txProposalId) { |
||||
|
return walletPrefix(walletId) + '!ptxp' + opKey(txProposalId); |
||||
|
}, |
||||
|
ADDRESS: function(walletId, address) { |
||||
|
return walletPrefix(walletId) + '!addr' + opKey(address); |
||||
|
}, |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.fetchWallet = function(id, cb) { |
||||
|
this.db.get(KEY.WALLET(id), function(err, data) { |
||||
|
if (err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
} |
||||
|
return cb(null, Wallet.fromObj(data)); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.storeWallet = function(wallet, cb) { |
||||
|
this.db.put(KEY.WALLET(wallet.id), wallet, cb); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.storeWalletAndUpdateCopayersLookup = function(wallet, cb) { |
||||
|
var ops = []; |
||||
|
ops.push({ |
||||
|
type: 'put', |
||||
|
key: KEY.WALLET(wallet.id), |
||||
|
value: wallet |
||||
|
}); |
||||
|
_.each(wallet.copayers, function(copayer) { |
||||
|
var value = { |
||||
|
walletId: wallet.id, |
||||
|
requestPubKey: copayer.requestPubKey, |
||||
|
}; |
||||
|
ops.push({ |
||||
|
type: 'put', |
||||
|
key: KEY.COPAYER(copayer.id), |
||||
|
value: value |
||||
|
}); |
||||
|
}); |
||||
|
this.db.batch(ops, cb); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.fetchCopayerLookup = function(copayerId, cb) { |
||||
|
this.db.get(KEY.COPAYER(copayerId), function(err, data) { |
||||
|
if (err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
} |
||||
|
return cb(null, data); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype._completeTxData = function(walletId, txs, cb) { |
||||
|
var txList = [].concat(txs); |
||||
|
this.fetchWallet(walletId, function(err, wallet) { |
||||
|
if (err) return cb(err); |
||||
|
_.each(txList, function(tx) { |
||||
|
tx.creatorName = wallet.getCopayer(tx.creatorId).name; |
||||
|
_.each(tx.actions, function(action) { |
||||
|
action.copayerName = wallet.getCopayer(action.copayerId).name; |
||||
|
}); |
||||
|
}); |
||||
|
return cb(null, txs); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.fetchTx = function(walletId, txProposalId, cb) { |
||||
|
var self = this; |
||||
|
this.db.get(KEY.TXP(walletId, txProposalId), function(err, data) { |
||||
|
if (err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
} |
||||
|
return self._completeTxData(walletId, TxProposal.fromObj(data), cb); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
Storage.prototype.fetchPendingTxs = function(walletId, cb) { |
||||
|
var self = this; |
||||
|
|
||||
|
var txs = []; |
||||
|
var key = KEY.PENDING_TXP(walletId); |
||||
|
this.db.createReadStream({ |
||||
|
gte: key, |
||||
|
lt: key + '~' |
||||
|
}) |
||||
|
.on('data', function(data) { |
||||
|
txs.push(TxProposal.fromObj(data.value)); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
return self._completeTxData(walletId, txs, cb); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
/** |
||||
|
* fetchTxs. Times are in UNIX EPOCH (seconds) |
||||
|
* |
||||
|
* @param walletId |
||||
|
* @param opts.minTs |
||||
|
* @param opts.maxTs |
||||
|
* @param opts.limit |
||||
|
*/ |
||||
|
Storage.prototype.fetchTxs = function(walletId, opts, cb) { |
||||
|
var self = this; |
||||
|
|
||||
|
var txs = []; |
||||
|
opts = opts || {}; |
||||
|
opts.limit = _.isNumber(opts.limit) ? parseInt(opts.limit) : -1; |
||||
|
opts.minTs = _.isNumber(opts.minTs) ? zeroPad(opts.minTs, 11) : 0; |
||||
|
opts.maxTs = _.isNumber(opts.maxTs) ? zeroPad(opts.maxTs, 11) : MAX_TS; |
||||
|
|
||||
|
var key = KEY.TXP(walletId, opts.minTs); |
||||
|
var endkey = KEY.TXP(walletId, opts.maxTs); |
||||
|
|
||||
|
this.db.createReadStream({ |
||||
|
gt: key, |
||||
|
lt: endkey + '~', |
||||
|
reverse: true, |
||||
|
limit: opts.limit, |
||||
|
}) |
||||
|
.on('data', function(data) { |
||||
|
txs.push(TxProposal.fromObj(data.value)); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
return self._completeTxData(walletId, txs, cb); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
/** |
||||
|
* fetchNotifications |
||||
|
* |
||||
|
* @param walletId |
||||
|
* @param opts.minTs |
||||
|
* @param opts.maxTs |
||||
|
* @param opts.limit |
||||
|
*/ |
||||
|
Storage.prototype.fetchNotifications = function(walletId, opts, cb) { |
||||
|
var txs = []; |
||||
|
opts = opts || {}; |
||||
|
opts.limit = _.isNumber(opts.limit) ? parseInt(opts.limit) : -1; |
||||
|
opts.minTs = _.isNumber(opts.minTs) ? zeroPad(opts.minTs, 11) : 0; |
||||
|
opts.maxTs = _.isNumber(opts.maxTs) ? zeroPad(opts.maxTs, 11) : MAX_TS; |
||||
|
|
||||
|
var key = KEY.NOTIFICATION(walletId, opts.minTs); |
||||
|
var endkey = KEY.NOTIFICATION(walletId, opts.maxTs); |
||||
|
|
||||
|
this.db.createReadStream({ |
||||
|
gt: key, |
||||
|
lt: endkey + '~', |
||||
|
reverse: opts.reverse, |
||||
|
limit: opts.limit, |
||||
|
}) |
||||
|
.on('data', function(data) { |
||||
|
txs.push(Notification.fromObj(data.value)); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
return cb(null, txs); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
Storage.prototype.storeNotification = function(walletId, notification, cb) { |
||||
|
this.db.put(KEY.NOTIFICATION(walletId, notification.id), notification, cb); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
// TODO should we store only txp.id on keys for indexing
|
||||
|
// or the whole txp? For now, the entire record makes sense
|
||||
|
// (faster + easier to access)
|
||||
|
Storage.prototype.storeTx = function(walletId, txp, cb) { |
||||
|
var ops = [{ |
||||
|
type: 'put', |
||||
|
key: KEY.TXP(walletId, txp.id), |
||||
|
value: txp, |
||||
|
}]; |
||||
|
|
||||
|
if (txp.isPending()) { |
||||
|
ops.push({ |
||||
|
type: 'put', |
||||
|
key: KEY.PENDING_TXP(walletId, txp.id), |
||||
|
value: txp, |
||||
|
}); |
||||
|
} else { |
||||
|
ops.push({ |
||||
|
type: 'del', |
||||
|
key: KEY.PENDING_TXP(walletId, txp.id), |
||||
|
}); |
||||
|
} |
||||
|
this.db.batch(ops, cb); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.removeTx = function(walletId, txProposalId, cb) { |
||||
|
var ops = [{ |
||||
|
type: 'del', |
||||
|
key: KEY.TXP(walletId, txProposalId), |
||||
|
}, { |
||||
|
type: 'del', |
||||
|
key: KEY.PENDING_TXP(walletId, txProposalId), |
||||
|
}]; |
||||
|
|
||||
|
this.db.batch(ops, cb); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype._delByKey = function(key, cb) { |
||||
|
var self = this; |
||||
|
var keys = []; |
||||
|
this.db.createKeyStream({ |
||||
|
gte: key, |
||||
|
lt: key + '~', |
||||
|
}) |
||||
|
.on('data', function(key) { |
||||
|
keys.push(key); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function(err) { |
||||
|
self.db.batch(_.map(keys, function(k) { |
||||
|
return { |
||||
|
key: k, |
||||
|
type: 'del' |
||||
|
}; |
||||
|
}), function(err) { |
||||
|
return cb(err); |
||||
|
}); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype._removeCopayers = function(walletId, cb) { |
||||
|
var self = this; |
||||
|
|
||||
|
this.fetchWallet(walletId, function(err, w) { |
||||
|
if (err || !w) return cb(err); |
||||
|
|
||||
|
self.db.batch(_.map(w.copayers, function(c) { |
||||
|
return { |
||||
|
type: 'del', |
||||
|
key: KEY.COPAYER(c.id), |
||||
|
}; |
||||
|
}), cb); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.removeWallet = function(walletId, cb) { |
||||
|
var self = this; |
||||
|
|
||||
|
async.series([ |
||||
|
|
||||
|
function(next) { |
||||
|
// This should be the first step. Will check the wallet exists
|
||||
|
self._removeCopayers(walletId, next); |
||||
|
}, |
||||
|
function(next) { |
||||
|
self._delByKey(walletPrefix(walletId), cb); |
||||
|
}, |
||||
|
], cb); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
Storage.prototype.fetchAddresses = function(walletId, cb) { |
||||
|
var addresses = []; |
||||
|
var key = KEY.ADDRESS(walletId); |
||||
|
this.db.createReadStream({ |
||||
|
gte: key, |
||||
|
lt: key + '~' |
||||
|
}) |
||||
|
.on('data', function(data) { |
||||
|
addresses.push(Address.fromObj(data.value)); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
if (err.notFound) return cb(); |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
return cb(null, addresses); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype.storeAddressAndWallet = function(wallet, addresses, cb) { |
||||
|
var ops = _.map([].concat(addresses), function(address) { |
||||
|
return { |
||||
|
type: 'put', |
||||
|
key: KEY.ADDRESS(wallet.id, address.address), |
||||
|
value: address, |
||||
|
}; |
||||
|
}); |
||||
|
ops.unshift({ |
||||
|
type: 'put', |
||||
|
key: KEY.WALLET(wallet.id), |
||||
|
value: wallet, |
||||
|
}); |
||||
|
|
||||
|
this.db.batch(ops, cb); |
||||
|
}; |
||||
|
|
||||
|
Storage.prototype._dump = function(cb, fn) { |
||||
|
fn = fn || console.log; |
||||
|
|
||||
|
this.db.readStream() |
||||
|
.on('data', function(data) { |
||||
|
fn(util.inspect(data, { |
||||
|
depth: 10 |
||||
|
})); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
if (cb) return cb(); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
module.exports = Storage; |
@ -1,70 +0,0 @@ |
|||||
#!/usr/bin/env node
|
|
||||
|
|
||||
var multilevel = require('multilevel'); |
|
||||
var net = require('net'); |
|
||||
var moment = require('moment'); |
|
||||
|
|
||||
var PORT = 3230; |
|
||||
|
|
||||
var otherDate; |
|
||||
|
|
||||
//trying to parse optional parameter to get stats on any given date
|
|
||||
try { |
|
||||
otherDate = process.argv[2] && moment(process.argv[2]).isValid() ? moment(process.argv[2]) : null; |
|
||||
} catch (e) { |
|
||||
console.log('Enter the date in the format YYYY-MM-DD.'); |
|
||||
} |
|
||||
|
|
||||
var db = multilevel.client(); |
|
||||
var con = net.connect(PORT); |
|
||||
con.pipe(db.createRpcStream()).pipe(con); |
|
||||
|
|
||||
|
|
||||
var Today = otherDate || moment(); |
|
||||
var TotalTx = 0; |
|
||||
var TotalAmount = 0; |
|
||||
var TotalNewWallets = 0; |
|
||||
|
|
||||
var IsToday = function(date) { |
|
||||
if (!date) return false; |
|
||||
var date = moment(date * 1000); |
|
||||
return (date >= Today.startOf('day') && date <= Today.endOf('day')); |
|
||||
} |
|
||||
|
|
||||
var TotalTxpForToday = function(data) { |
|
||||
if (!data) return; |
|
||||
if (data.key.indexOf('!txp!') < 0) return; |
|
||||
if (!data.value || !IsToday(data.value.createdOn)) return; |
|
||||
TotalTx++; |
|
||||
TotalAmount = TotalAmount + data.value.amount; |
|
||||
}; |
|
||||
|
|
||||
var TotalNewWalletForToday = function(data) { |
|
||||
if (!data) return; |
|
||||
if (data.key.indexOf('!main') < 0) return; |
|
||||
if (!data.value || !IsToday(data.value.createdOn)) return; |
|
||||
TotalNewWallets++; |
|
||||
}; |
|
||||
|
|
||||
var PrintStats = function() { |
|
||||
console.log('Stats for date : ', Today.format("YYYY-MM-DD")); |
|
||||
console.log('New wallets : ', TotalNewWallets); |
|
||||
console.log('Total tx : ', TotalTx); |
|
||||
console.log('Total amount in tx (satoshis) : ', TotalAmount); |
|
||||
}; |
|
||||
|
|
||||
var ProcessData = function(data) { |
|
||||
TotalTxpForToday(data); |
|
||||
TotalNewWalletForToday(data); |
|
||||
}; |
|
||||
|
|
||||
// streams
|
|
||||
db.createReadStream().on('data', function(data) { |
|
||||
ProcessData(data); |
|
||||
}).on('error', function(err) { |
|
||||
console.log('Error : ', err); |
|
||||
process.exit(code = 1); |
|
||||
}).on('close', function() { |
|
||||
PrintStats(); |
|
||||
process.exit(code = 0); |
|
||||
}); |
|
@ -1,17 +0,0 @@ |
|||||
#!/usr/bin/env node
|
|
||||
|
|
||||
var multilevel = require('multilevel'); |
|
||||
var net = require('net'); |
|
||||
var level = require('levelup'); |
|
||||
|
|
||||
var db = level('./db', { |
|
||||
valueEncoding: 'json' |
|
||||
}); |
|
||||
var HOST = 'localhost'; |
|
||||
var PORT = 3230; |
|
||||
|
|
||||
|
|
||||
console.log('Server started at port ' + PORT + '...'); |
|
||||
net.createServer(function(con) { |
|
||||
con.pipe(multilevel.server(db)).pipe(con); |
|
||||
}).listen(PORT, HOST); |
|
@ -0,0 +1,63 @@ |
|||||
|
'use strict'; |
||||
|
|
||||
|
var LevelStorage = require('../lib/storage_leveldb'); |
||||
|
var MongoStorage = require('../lib/storage'); |
||||
|
|
||||
|
|
||||
|
var level = new LevelStorage({ |
||||
|
dbPath: './db/bws.db', |
||||
|
}); |
||||
|
|
||||
|
var mongo = new MongoStorage(); |
||||
|
mongo.connect({ |
||||
|
host: 'localhost', |
||||
|
port: '27017' |
||||
|
}, function(err) { |
||||
|
if (err) throw err; |
||||
|
mongo.db.dropDatabase(function(err) { |
||||
|
if (err) throw err; |
||||
|
run(function(err) { |
||||
|
if (err) throw err; |
||||
|
console.log('All data successfully migrated'); |
||||
|
process.exit(0); |
||||
|
// mongo._dump(function() {
|
||||
|
// process.exit(0);
|
||||
|
// });
|
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
|
||||
|
|
||||
|
function run(cb) { |
||||
|
level.db.readStream() |
||||
|
.on('data', function(data) { |
||||
|
migrate(data.key, data.value, function(err) { |
||||
|
if (err) throw err; |
||||
|
}); |
||||
|
}) |
||||
|
.on('error', function(err) { |
||||
|
return cb(err); |
||||
|
}) |
||||
|
.on('end', function() { |
||||
|
return cb(); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
function migrate(key, value, cb) { |
||||
|
if (key.match(/^copayer!/)) { |
||||
|
value.copayerId = key.substring(key.indexOf('!') + 1); |
||||
|
mongo.db.collection('copayers_lookup').insert(value, cb); |
||||
|
} else if (key.match(/!addr!/)) { |
||||
|
value.walletId = key.substring(2, key.indexOf('!addr')); |
||||
|
mongo.db.collection('addresses').insert(value, cb); |
||||
|
} else if (key.match(/!not!/)) { |
||||
|
mongo.db.collection('notifications').insert(value, cb); |
||||
|
} else if (key.match(/!p?txp!/)) { |
||||
|
value.isPending = key.indexOf('!ptxp!') != -1; |
||||
|
mongo.db.collection('txs').insert(value, cb); |
||||
|
} else if (key.match(/!main$/)) { |
||||
|
mongo.db.collection('wallets').insert(value, cb); |
||||
|
} else { |
||||
|
return cb(new Error('Invalid key ' + key)); |
||||
|
} |
||||
|
}; |
@ -0,0 +1,193 @@ |
|||||
|
'use strict'; |
||||
|
|
||||
|
var _ = require('lodash'); |
||||
|
var async = require('async'); |
||||
|
var chai = require('chai'); |
||||
|
var sinon = require('sinon'); |
||||
|
var should = chai.should(); |
||||
|
var tingodb = require('tingodb')({ |
||||
|
memStore: true |
||||
|
}); |
||||
|
|
||||
|
var Storage = require('../lib/storage'); |
||||
|
var Model = require('../lib/model'); |
||||
|
|
||||
|
var db, storage; |
||||
|
|
||||
|
function openDb(cb) { |
||||
|
db = new tingodb.Db('./db/test', {}); |
||||
|
return cb(); |
||||
|
}; |
||||
|
|
||||
|
function resetDb(cb) { |
||||
|
if (!db) return cb(); |
||||
|
db.dropDatabase(function(err) { |
||||
|
return cb(); |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
|
||||
|
describe('Storage', function() { |
||||
|
before(function(done) { |
||||
|
openDb(function() { |
||||
|
storage = new Storage({ |
||||
|
db: db |
||||
|
}); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
beforeEach(function(done) { |
||||
|
resetDb(done); |
||||
|
}); |
||||
|
|
||||
|
describe('Store & fetch wallet', function() { |
||||
|
it('should correctly store and fetch wallet', function(done) { |
||||
|
var wallet = Model.Wallet.create({ |
||||
|
id: '123', |
||||
|
name: 'my wallet', |
||||
|
m: 2, |
||||
|
n: 3, |
||||
|
}); |
||||
|
should.exist(wallet); |
||||
|
storage.storeWallet(wallet, function(err) { |
||||
|
should.not.exist(err); |
||||
|
storage.fetchWallet('123', function(err, w) { |
||||
|
should.not.exist(err); |
||||
|
should.exist(w); |
||||
|
w.id.should.equal(wallet.id); |
||||
|
w.name.should.equal(wallet.name); |
||||
|
w.m.should.equal(wallet.m); |
||||
|
w.n.should.equal(wallet.n); |
||||
|
done(); |
||||
|
}) |
||||
|
}); |
||||
|
}); |
||||
|
it('should not return error if wallet not found', function(done) { |
||||
|
storage.fetchWallet('123', function(err, w) { |
||||
|
should.not.exist(err); |
||||
|
should.not.exist(w); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
describe('Copayer lookup', function() { |
||||
|
it('should correctly store and fetch copayer lookup', function(done) { |
||||
|
var wallet = Model.Wallet.create({ |
||||
|
id: '123', |
||||
|
name: 'my wallet', |
||||
|
m: 2, |
||||
|
n: 3, |
||||
|
}); |
||||
|
_.each(_.range(3), function(i) { |
||||
|
var copayer = Model.Copayer.create({ |
||||
|
name: 'copayer ' + i, |
||||
|
xPubKey: 'xPubKey ' + i, |
||||
|
requestPubKey: 'requestPubKey ' + i, |
||||
|
}); |
||||
|
wallet.addCopayer(copayer); |
||||
|
}); |
||||
|
|
||||
|
should.exist(wallet); |
||||
|
storage.storeWalletAndUpdateCopayersLookup(wallet, function(err) { |
||||
|
should.not.exist(err); |
||||
|
storage.fetchCopayerLookup(wallet.copayers[1].id, function(err, lookup) { |
||||
|
should.not.exist(err); |
||||
|
should.exist(lookup); |
||||
|
lookup.walletId.should.equal('123'); |
||||
|
lookup.requestPubKey.should.equal('requestPubKey 1'); |
||||
|
done(); |
||||
|
}) |
||||
|
}); |
||||
|
}); |
||||
|
it('should not return error if copayer not found', function(done) { |
||||
|
storage.fetchCopayerLookup('2', function(err, lookup) { |
||||
|
should.not.exist(err); |
||||
|
should.not.exist(lookup); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
|
||||
|
describe('Transaction proposals', function() { |
||||
|
var wallet, proposals; |
||||
|
|
||||
|
beforeEach(function(done) { |
||||
|
wallet = Model.Wallet.create({ |
||||
|
id: '123', |
||||
|
name: 'my wallet', |
||||
|
m: 2, |
||||
|
n: 3, |
||||
|
}); |
||||
|
_.each(_.range(3), function(i) { |
||||
|
var copayer = Model.Copayer.create({ |
||||
|
name: 'copayer ' + i, |
||||
|
xPubKey: 'xPubKey ' + i, |
||||
|
requestPubKey: 'requestPubKey ' + i, |
||||
|
}); |
||||
|
wallet.addCopayer(copayer); |
||||
|
}); |
||||
|
should.exist(wallet); |
||||
|
storage.storeWalletAndUpdateCopayersLookup(wallet, function(err) { |
||||
|
should.not.exist(err); |
||||
|
|
||||
|
proposals = _.map(_.range(4), function(i) { |
||||
|
var tx = Model.TxProposal.create({ |
||||
|
walletId: '123', |
||||
|
creatorId: wallet.copayers[0].id, |
||||
|
amount: i + 100, |
||||
|
}); |
||||
|
if (i % 2 == 0) { |
||||
|
tx.status = 'rejected'; |
||||
|
tx.isPending().should.be.false; |
||||
|
} |
||||
|
return tx; |
||||
|
}); |
||||
|
async.each(proposals, function(tx, next) { |
||||
|
storage.storeTx('123', tx, next); |
||||
|
}, function(err) { |
||||
|
should.not.exist(err); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
it('should fetch tx', function(done) { |
||||
|
storage.fetchTx('123', proposals[0].id, function(err, tx) { |
||||
|
should.not.exist(err); |
||||
|
should.exist(tx); |
||||
|
tx.id.should.equal(proposals[0].id); |
||||
|
tx.walletId.should.equal(proposals[0].walletId); |
||||
|
tx.creatorName.should.equal('copayer 0'); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
it('should fetch all pending txs', function(done) { |
||||
|
storage.fetchPendingTxs('123', function(err, txs) { |
||||
|
should.not.exist(err); |
||||
|
should.exist(txs); |
||||
|
txs.length.should.equal(2); |
||||
|
txs = _.sortBy(txs, 'amount'); |
||||
|
txs[0].amount.should.equal(101); |
||||
|
txs[1].amount.should.equal(103); |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
it('should remove tx', function(done) { |
||||
|
storage.removeTx('123', proposals[0].id, function(err) { |
||||
|
should.not.exist(err); |
||||
|
storage.fetchTx('123', proposals[0].id, function(err, tx) { |
||||
|
should.not.exist(err); |
||||
|
should.not.exist(tx); |
||||
|
storage.fetchTxs('123', {}, function(err, txs) { |
||||
|
should.not.exist(err); |
||||
|
should.exist(txs); |
||||
|
txs.length.should.equal(3); |
||||
|
_.any(txs, { |
||||
|
id: proposals[0].id |
||||
|
}).should.be.false; |
||||
|
done(); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
||||
|
}); |
Loading…
Reference in new issue