flocore-node/lib/services/wallet-api/index.js
2017-01-16 19:41:27 -05:00

382 lines
9.5 KiB
JavaScript

'use strict';
var async = require('async');
var BaseService = require('../../service');
var inherits = require('util').inherits;
var index = require('../../');
var log = index.log;
var errors = index.errors;
var bitcore = require('bitcore-lib');
var Networks = bitcore.Networks;
var levelup = require('levelup');
var leveldown = require('leveldown');
var multer = require('multer');
var storage = multer.memoryStorage();
var upload = multer({ storage: storage });
var validators = require('./validators');
var utils = require('./utils');
var _ = require('lodash');
var bodyParser = require('body-parser');
var LRU = require('lru-cache');
var WalletService = function(options) {
BaseService.call(this, options);
this._dbOptions = {
keyEncoding: 'string',
valueEncoding: 'json'
};
this._db = levelup(options.dbPath, this._dbOptions);
this._cache = LRU({
max: 500 * 1024 * 1024,
length: function(n, key) {
return Buffer.byteLength(n, 'utf8');
},
maxAge: 30 * 60 * 1000
});
};
inherits(WalletService, BaseService);
WalletService.dependencies = [
'bitcoind',
'web'
];
WalletService.prototype.getAPIMethods = function() {
return [];
};
WalletService.prototype.start = function(callback) {
setImmediate(callback);
};
WalletService.prototype.stop = function(callback) {
setImmediate(callback);
};
WalletService.prototype.getPublishEvents = function() {
return [];
};
WalletService.prototype._endpointUTXOs = function() {
var self = this;
return function(req, res) {
req.setTimeout(600000);
var walletId = req.params.walletId;
var queryMempool = req.query.queryMempool === false ? false : true;
//var tip = self.node.bitcoind.tip;
// TODO: get the height of the tip
//var height = tip;
var height = null;
var options = {
queryMempool: queryMempool
};
self._getUtxos(walletId, height, options, function(err, utxos) {
if(err) {
return utils.sendError(err, res);
}
res.status(200).jsonp({
utxos: utxos,
height: height
});
});
};
};
WalletService.prototype._endpointGetBalance= function() {
var self = this;
return function(req, res) {
req.setTimeout(600000);
var walletId = req.params.walletId;
var queryMempool = req.query.queryMempool === false ? false : true;
var byAddress = req.query.byAddress;
//var tip = self.node.bitcoind.tip;
// TODO: get the height of the tip
//var height = tip;
var height = null;
var options = {
queryMempool: queryMempool,
byAddress: byAddress
};
self._getBalance(walletId, height, options, function(err, result) {
if(err) {
return utils.sendError(err, res);
}
res.status(200).jsonp(result);
});
};
};
WalletService.prototype._endpointGetAddresses = function() {
var self = this;
return function(req, res) {
var walletId = req.params.walletId;
self._getAddresses(walletId, function(err, addresses) {
if(err) {
return utils.sendError(err, res);
}
if(!addresses) {
return res.status(404).send('Not found');
}
res.status(200).jsonp({
addresses: addresses
});
});
};
};
WalletService.prototype._endpointPostAddresses = function() {
var self = this;
return function(req, res) {
var addresses = req.addresses;
var walletId = utils.getWalletId();
self._storeAddresses(walletId, addresses, function(err, hash) {
if(err) {
return utils.sendError(err, res);
}
res.status(201).jsonp({
walletId: walletId,
});
});
};
};
WalletService.prototype._endpointGetTransactions = function() {
var self = this;
return function(req, res) {
req.setTimeout(600000);
var walletId = req.params.walletId;
var options = {
start: req.query.start,
end : req.query.end,
from: req.query.from,
to: req.query.to
};
self._getTransactions(walletId, options, function(err, transactions, totalCount) {
if(err) {
return utils.sendError(err, res);
}
res.status(200).jsonp({
transactions: transactions,
totalCount: totalCount
});
});
};
};
WalletService.prototype._endpointPutAddresses = function() {
var self = this;
return function(req, res) {
var newAddresses = req.body;
if(!Array.isArray(req.body)) {
return utils.sendError(new Error('Must PUT an array'), res);
}
var walletId = req.params.walletId;
self._getAddresses(walletId, function(err, oldAddresses) {
if(err) {
return utils.sendError(err, res);
}
if(!oldAddresses) {
return res.status(404).send('Not found');
}
var allAddresses = _.union(oldAddresses, newAddresses);
var amountAdded = allAddresses.length - oldAddresses.length;
self._storeAddresses(walletId, allAddresses, function(err) {
if(err) {
return utils.sendError(err, res);
}
res.status(200).jsonp({
walletId: walletId,
amountAdded: amountAdded
});
});
});
};
};
WalletService.prototype._getUtxos = function(walletId, height, options, callback) {
// TODO get the balance only to this height
var self = this;
self._getAddresses(walletId, function(err, addresses) {
if(err) {
return callback(err);
}
self.node.services.bitcoind.getAddressUnspentOutputs(addresses, options, callback);
});
};
WalletService.prototype._getBalance = function(walletId, height, options, callback) {
// TODO get the balance only to this height
var self = this;
self._getAddresses(walletId, function(err, addresses) {
if(err) {
return callback(err);
}
self.node.services.bitcoind.getAddressUnspentOutputs(addresses, options, function(err, utxos) {
if(err) {
return callback(err);
}
if (options.byAddress) {
var result = self._getBalanceByAddress(addresses, utxos);
return callback(null, {
addresses: result,
height: null
});
}
var balance = 0;
utxos.forEach(function(utxo) {
balance += utxo.satoshis;
});
callback(null, {
balance: balance,
height: null
});
});
});
};
WalletService.prototype._getBalanceByAddress = function(addresses, utxos) {
var res = {};
utxos.forEach(function(utxo) {
if (res[utxo.address]) {
res[utxo.address] += utxo.satoshis;
} else {
res[utxo.address] = utxo.satoshis;
}
});
addresses.forEach(function(address) {
res[address] = res[address] || 0;
});
return res;
};
WalletService.prototype._chunkAdresses = function(addresses) {
var maxLength = this.node.services.bitcoind.maxAddressesQuery;
var groups = [];
var groupsCount = Math.ceil(addresses.length / maxLength);
for(var i = 0; i < groupsCount; i++) {
groups.push(addresses.slice(i * maxLength, Math.min(maxLength * (i + 1), addresses.length)));
}
return groups;
};
WalletService.prototype._getTransactions = function(walletId, options, callback) {
var self = this;
var transactions = [];
var opts = {
start: options.start,
end: options.end
};
var key = walletId + opts.start + opts.end;
if (!self._cache.peek(key)) {
self._getAddresses(walletId, function(err, addresses) {
if(err) {
return callback(err);
}
if (!addresses) {
return callback(new Error('wallet not found'));
}
var addressGroups = self._chunkAdresses(addresses);
async.eachSeries(addressGroups, function(addresses, next) {
self.node.services.bitcoind.getAddressHistory(addresses, opts, function(err, history) {
if(err) {
return next(err);
}
var groupTransactions = history.items.map(function(item) {
return item.tx;
});
transactions = _.union(transactions, groupTransactions);
next();
});
}, function(err) {
if(err) {
return callback(err);
}
self._cache.set(key, JSON.stringify(transactions));
finish();
});
});
} else {
try {
transactions = JSON.parse(self._cache.get(key));
finish();
} catch(e) {
self._cache.del(key);
return callback(e);
}
}
function finish() {
var from = options.from || 0;
var to = options.to || transactions.length;
callback(null, transactions.slice(from, to), transactions.length);
}
};
WalletService.prototype._getAddresses = function(walletId, callback) {
this._db.get(walletId, callback);
};
WalletService.prototype._storeAddresses = function(walletId, addresses, callback) {
this._db.put(walletId, addresses, callback);
};
WalletService.prototype._endpointGetInfo = function() {
return function(req, res) {
res.jsonp({result: 'ok'});
};
};
WalletService.prototype.setupRoutes = function(app, express) {
var s = this;
var v = validators;
app.use(bodyParser.json());
app.get('/info',
s._endpointGetInfo()
);
app.get('/wallets/:walletId/utxos',
s._endpointUTXOs()
);
app.get('/wallets/:walletId/balance',
s._endpointGetBalance()
);
app.get('/wallets/:walletId',
s._endpointGetAddresses()
);
app.put('/wallets/:walletId/addresses',
s._endpointPutAddresses()
);
app.get('/wallets/:walletId/transactions',
s._endpointGetTransactions()
);
app.post('/wallets',
upload.single('addresses'),
v.checkAddresses,
s._endpointPostAddresses()
);
};
WalletService.prototype.getRoutePrefix = function() {
return 'wallet-api';
};
module.exports = WalletService;