bitcore-wallet-service/lib/server.js

1342 lines
38 KiB
JavaScript
Raw Normal View History

2015-01-27 05:18:45 -08:00
'use strict';
var _ = require('lodash');
var $ = require('preconditions').singleton();
var async = require('async');
var log = require('npmlog');
log.debug = log.verbose;
2015-04-18 02:55:24 -07:00
log.disableColor();
2015-02-02 10:29:14 -08:00
2015-03-12 07:34:41 -07:00
var WalletUtils = require('bitcore-wallet-utils');
var Bitcore = WalletUtils.Bitcore;
2015-01-31 14:56:50 -08:00
var PublicKey = Bitcore.PublicKey;
2015-02-01 11:50:58 -08:00
var HDPublicKey = Bitcore.HDPublicKey;
2015-02-06 10:15:54 -08:00
var Address = Bitcore.Address;
2015-01-27 05:18:45 -08:00
2015-02-04 08:31:02 -08:00
var ClientError = require('./clienterror');
2015-02-02 11:00:32 -08:00
var Utils = require('./utils');
2015-04-07 13:02:08 -07:00
var Lock = require('./lock');
2015-01-27 05:18:45 -08:00
var Storage = require('./storage');
2015-05-06 06:00:09 -07:00
var MessageBroker = require('./messagebroker');
2015-03-30 16:16:51 -07:00
var BlockchainExplorer = require('./blockchainexplorer');
2015-01-27 11:40:21 -08:00
2015-01-27 05:18:45 -08:00
var Wallet = require('./model/wallet');
var Copayer = require('./model/copayer');
2015-01-27 11:40:21 -08:00
var Address = require('./model/address');
var TxProposal = require('./model/txproposal');
2015-02-18 12:05:02 -08:00
var Notification = require('./model/notification');
2015-01-27 05:18:45 -08:00
2015-02-06 12:56:51 -08:00
var initialized = false;
2015-04-15 06:59:25 -07:00
var lock, storage, blockchainExplorer, blockchainExplorerOpts;
2015-05-06 06:00:09 -07:00
var messageBroker;
2015-02-06 12:56:51 -08:00
2015-03-23 08:50:00 -07:00
2015-01-27 07:54:17 -08:00
/**
2015-03-04 09:04:34 -08:00
* Creates an instance of the Bitcore Wallet Service.
2015-01-27 07:54:17 -08:00
* @constructor
2015-02-06 12:56:51 -08:00
*/
2015-02-20 12:32:19 -08:00
function WalletService() {
2015-02-20 12:23:42 -08:00
if (!initialized)
2015-02-19 12:38:48 -08:00
throw new Error('Server not initialized');
2015-04-07 13:02:08 -07:00
this.lock = lock;
2015-02-06 12:56:51 -08:00
this.storage = storage;
2015-03-30 16:16:51 -07:00
this.blockchainExplorer = blockchainExplorer;
2015-04-15 06:59:25 -07:00
this.blockchainExplorerOpts = blockchainExplorerOpts;
2015-05-06 06:00:09 -07:00
this.messageBroker = messageBroker;
2015-02-12 05:26:13 -08:00
this.notifyTicker = 0;
2015-02-06 12:56:51 -08:00
};
/**
* Initializes global settings for all instances.
2015-01-28 05:52:45 -08:00
* @param {Object} opts
* @param {Storage} [opts.storage] - The storage provider.
2015-03-30 16:16:51 -07:00
* @param {Storage} [opts.blockchainExplorer] - The blockchainExporer provider.
2015-04-21 10:43:35 -07:00
* @param {Callback} cb
2015-01-27 07:54:17 -08:00
*/
2015-04-21 10:43:35 -07:00
WalletService.initialize = function(opts, cb) {
$.shouldBeFunction(cb);
2015-02-02 12:07:18 -08:00
opts = opts || {};
2015-04-07 13:02:08 -07:00
lock = opts.lock || new Lock(opts.lockOpts);
2015-03-30 16:16:51 -07:00
blockchainExplorer = opts.blockchainExplorer;
2015-04-15 06:59:25 -07:00
blockchainExplorerOpts = opts.blockchainExplorerOpts;
2015-04-21 10:43:35 -07:00
if (initialized)
return cb();
2015-05-04 14:23:56 -07:00
function initStorage(cb) {
if (opts.storage) {
storage = opts.storage;
return cb();
}
2015-04-21 10:43:35 -07:00
var newStorage = new Storage();
newStorage.connect(opts.storageOpts, function(err) {
if (err) return cb(err);
storage = newStorage;
return cb();
});
2015-05-04 14:23:56 -07:00
};
2015-05-06 06:00:09 -07:00
function initMessageBroker(cb) {
if (opts.messageBroker) {
messageBroker = opts.messageBroker;
} else {
messageBroker = new MessageBroker(opts.messageBrokerOpts);
2015-05-04 14:23:56 -07:00
}
2015-05-05 09:04:29 -07:00
return cb();
2015-05-04 14:23:56 -07:00
};
async.series([
function(next) {
initStorage(next);
},
function(next) {
2015-05-06 06:00:09 -07:00
initMessageBroker(next);
2015-05-04 14:23:56 -07:00
},
], function(err) {
if (err) {
log.error('Could not initialize', err);
throw err;
}
initialized = true;
return cb();
});
2015-04-21 10:43:35 -07:00
};
WalletService.shutDown = function(cb) {
2015-04-23 08:25:36 -07:00
if (!initialized) return cb();
storage.disconnect(function(err) {
if (err) return cb(err);
initialized = false;
return cb();
});
2015-01-27 05:18:45 -08:00
};
2015-02-20 12:32:19 -08:00
WalletService.getInstance = function() {
return new WalletService();
2015-02-09 10:30:16 -08:00
};
2015-02-06 12:56:51 -08:00
/**
* Gets an instance of the server after authenticating the copayer.
* @param {Object} opts
* @param {string} opts.copayerId - The copayer id making the request.
* @param {string} opts.message - The contents of the request to be signed.
* @param {string} opts.signature - Signature of message to be verified using the copayer's requestPubKey
2015-02-06 12:56:51 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.getInstanceWithAuth = function(opts, cb) {
2015-02-06 12:56:51 -08:00
2015-02-12 11:42:32 -08:00
if (!Utils.checkRequired(opts, ['copayerId', 'message', 'signature']))
2015-02-12 05:26:13 -08:00
return cb(new ClientError('Required argument missing'));
2015-02-06 12:56:51 -08:00
2015-02-20 12:32:19 -08:00
var server = new WalletService();
2015-02-07 08:13:29 -08:00
server.storage.fetchCopayerLookup(opts.copayerId, function(err, copayer) {
2015-02-06 12:56:51 -08:00
if (err) return cb(err);
2015-02-09 10:30:16 -08:00
if (!copayer) return cb(new ClientError('NOTAUTHORIZED', 'Copayer not found'));
2015-02-06 12:56:51 -08:00
var isValid = server._verifySignature(opts.message, opts.signature, copayer.requestPubKey);
2015-02-21 14:29:42 -08:00
if (!isValid)
return cb(new ClientError('NOTAUTHORIZED', 'Invalid signature'));
2015-02-02 10:56:53 -08:00
2015-02-06 12:56:51 -08:00
server.copayerId = opts.copayerId;
server.walletId = copayer.walletId;
return cb(null, server);
});
2015-02-02 10:56:53 -08:00
};
2015-01-27 05:18:45 -08:00
2015-04-08 11:18:28 -07:00
WalletService.prototype._runLocked = function(cb, task) {
$.checkState(this.walletId);
this.lock.runLocked(this.walletId, cb, task);
};
2015-02-06 12:56:51 -08:00
2015-01-27 07:54:17 -08:00
/**
* Creates a new wallet.
2015-01-27 11:40:21 -08:00
* @param {Object} opts
2015-01-27 07:54:17 -08:00
* @param {string} opts.id - The wallet id.
* @param {string} opts.name - The wallet name.
* @param {number} opts.m - Required copayers.
* @param {number} opts.n - Total copayers.
* @param {string} opts.pubKey - Public key to verify copayers joining have access to the wallet secret.
* @param {string} [opts.network = 'livenet'] - The Bitcoin network for this wallet.
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.createWallet = function(opts, cb) {
2015-02-02 15:13:13 -08:00
var self = this,
pubKey;
2015-01-27 05:18:45 -08:00
2015-02-11 07:10:47 -08:00
if (!Utils.checkRequired(opts, ['name', 'm', 'n', 'pubKey']))
return cb(new ClientError('Required argument missing'));
2015-02-07 08:13:29 -08:00
2015-02-08 08:16:41 -08:00
if (_.isEmpty(opts.name)) return cb(new ClientError('Invalid wallet name'));
2015-02-07 08:13:29 -08:00
if (!Wallet.verifyCopayerLimits(opts.m, opts.n))
return cb(new ClientError('Invalid combination of required copayers / total copayers'));
2015-02-02 12:07:18 -08:00
var network = opts.network || 'livenet';
2015-02-07 08:13:29 -08:00
if (network != 'livenet' && network != 'testnet')
return cb(new ClientError('Invalid network'));
2015-02-02 10:29:14 -08:00
2015-01-31 14:56:50 -08:00
try {
pubKey = new PublicKey.fromString(opts.pubKey);
2015-02-24 05:36:14 -08:00
} catch (ex) {
return cb(new ClientError('Invalid public key'));
2015-01-31 14:56:50 -08:00
};
2015-01-30 06:58:28 -08:00
2015-03-31 13:28:01 -07:00
var newWallet;
async.series([
function(acb) {
if (!opts.id)
return acb();
2015-02-07 08:13:29 -08:00
2015-03-31 13:28:01 -07:00
self.storage.fetchWallet(opts.id, function(err, wallet) {
2015-04-02 08:28:25 -07:00
if (wallet) return acb(new ClientError('WEXISTS', 'Wallet already exists'));
2015-03-31 13:28:01 -07:00
return acb(err);
});
},
function(acb) {
var wallet = Wallet.create({
name: opts.name,
m: opts.m,
n: opts.n,
network: network,
pubKey: pubKey.toString(),
id: opts.id,
});
self.storage.storeWallet(wallet, function(err) {
log.debug('Wallet created', wallet.id, network);
newWallet = wallet;
return acb(err);
});
}
], function(err) {
return cb(err, newWallet ? newWallet.id : null);
2015-02-02 12:07:18 -08:00
});
2015-01-27 05:18:45 -08:00
};
2015-01-27 07:54:17 -08:00
/**
* Retrieves a wallet from storage.
2015-01-27 11:40:21 -08:00
* @param {Object} opts
2015-02-02 06:55:03 -08:00
* @returns {Object} wallet
2015-01-27 07:54:17 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getWallet = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-27 05:18:45 -08:00
2015-02-06 12:56:51 -08:00
self.storage.fetchWallet(self.walletId, function(err, wallet) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-04 08:31:02 -08:00
if (!wallet) return cb(new ClientError('Wallet not found'));
2015-02-02 12:07:18 -08:00
return cb(null, wallet);
});
2015-01-27 05:18:45 -08:00
};
2015-01-28 05:36:49 -08:00
2015-03-31 13:28:01 -07:00
/**
* Replace temporary request key
* @param {Object} opts
* @param {string} opts.name - The copayer name.
* @param {string} opts.xPubKey - Extended Public Key for this copayer.
* @param {string} opts.requestPubKey - Public Key used to check requests from this copayer.
2015-04-29 08:11:29 -07:00
* @param {string} opts.copayerSignature - S(name|xPubKey|requestPubKey). Used by other copayers to verify that the copayer joining knows the wallet secret.
2015-03-31 13:28:01 -07:00
*/
WalletService.prototype.replaceTemporaryRequestKey = function(opts, cb) {
var self = this;
if (!Utils.checkRequired(opts, ['name', 'xPubKey', 'requestPubKey', 'copayerSignature']))
return cb(new ClientError('Required argument missing'));
if (_.isEmpty(opts.name))
return cb(new ClientError('Invalid copayer name'));
if (opts.isTemporaryRequestKey)
return cb(new ClientError('Bad arguments'));
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-03-31 13:28:01 -07:00
self.storage.fetchWallet(self.walletId, function(err, wallet) {
if (err) return cb(err);
if (!wallet) return cb(new ClientError('Wallet not found'));
var hash = WalletUtils.getCopayerHash(opts.name, opts.xPubKey, opts.requestPubKey);
if (!self._verifySignature(hash, opts.copayerSignature, wallet.pubKey)) {
return cb(new ClientError());
}
var oldCopayerData = _.find(wallet.copayers, {
id: self.copayerId
});
$.checkState(oldCopayerData);
2015-04-03 12:09:48 -07:00
if (oldCopayerData.xPubKey !== opts.xPubKey || !oldCopayerData.isTemporaryRequestKey)
2015-03-31 13:28:01 -07:00
return cb(new ClientError('CDATAMISMATCH', 'Copayer data mismatch'));
if (wallet.copayers.length != wallet.n)
return cb(new ClientError('WNOTFULL', 'Replace only works on full wallets'));
2015-04-01 13:55:40 -07:00
wallet.updateCopayerRequestKey(self.copayerId, opts.requestPubKey, opts.copayerSignature);
2015-03-31 13:28:01 -07:00
self.storage.storeWalletAndUpdateCopayersLookup(wallet, function(err) {
if (err) return cb(err);
2015-04-01 09:31:22 -07:00
self._notify('CopayerUpdated', {
walletId: opts.walletId,
copayerId: self.copayerId,
copayerName: opts.name,
2015-04-30 16:31:45 -07:00
}, false, function() {
return cb(null, {
copayerId: self.copayerId,
wallet: wallet
});
2015-03-31 13:28:01 -07:00
});
});
});
});
};
2015-02-01 06:41:16 -08:00
/**
* Verifies a signature
* @param text
* @param signature
* @param pubKey
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype._verifySignature = function(text, signature, pubKey) {
2015-02-17 11:42:47 -08:00
return WalletUtils.verifyMessage(text, signature, pubKey);
2015-02-01 06:41:16 -08:00
};
2015-02-11 11:00:16 -08:00
/**
2015-02-11 18:11:30 -08:00
* _notify
2015-02-11 11:00:16 -08:00
*
2015-04-02 07:57:47 -07:00
* @param {String} type
* @param {Object} data
* @param {Boolean} isGlobal - If true, the notification is not issued on behalf of any particular copayer (defaults to false)
2015-02-11 11:00:16 -08:00
*/
2015-04-30 16:31:45 -07:00
WalletService.prototype._notify = function(type, data, isGlobal, cb) {
2015-02-11 10:42:49 -08:00
var self = this;
2015-02-12 11:42:32 -08:00
log.debug('Notification', type, data);
2015-02-11 18:11:30 -08:00
var walletId = self.walletId || data.walletId;
var copayerId = self.copayerId || data.copayerId;
2015-02-11 18:11:30 -08:00
$.checkState(walletId);
2015-02-17 16:20:08 -08:00
var n = Notification.create({
2015-02-11 10:42:49 -08:00
type: type,
data: data,
2015-02-12 05:26:13 -08:00
ticker: this.notifyTicker++,
2015-04-02 07:57:47 -07:00
creatorId: isGlobal ? null : copayerId,
2015-03-30 07:24:33 -07:00
walletId: walletId,
2015-02-11 10:42:49 -08:00
});
2015-02-11 18:11:30 -08:00
this.storage.storeNotification(walletId, n, function() {
2015-05-06 06:00:09 -07:00
self.messageBroker.send(n);
2015-04-30 16:31:45 -07:00
if (cb) return cb();
2015-02-11 10:42:49 -08:00
});
};
2015-01-27 07:54:17 -08:00
/**
* Joins a wallet in creation.
2015-01-27 11:40:21 -08:00
* @param {Object} opts
2015-01-27 07:54:17 -08:00
* @param {string} opts.walletId - The wallet id.
* @param {string} opts.name - The copayer name.
2015-03-09 14:11:25 -07:00
* @param {string} opts.xPubKey - Extended Public Key for this copayer.
* @param {string} opts.requestPubKey - Public Key used to check requests from this copayer.
2015-03-10 09:55:07 -07:00
* @param {string} opts.copayerSignature - S(name|xPubKey|requestPubKey). Used by other copayers to verify the that the copayer joining knows the wallet secret.
2015-03-31 13:28:01 -07:00
* @param {string} opts.isTemporaryRequestKey - requestPubKey will be marked as 'temporary' (only used for Copay migration)
2015-01-27 07:54:17 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.joinWallet = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-27 05:18:45 -08:00
2015-03-10 07:23:23 -07:00
if (!Utils.checkRequired(opts, ['walletId', 'name', 'xPubKey', 'requestPubKey', 'copayerSignature']))
2015-02-11 07:10:47 -08:00
return cb(new ClientError('Required argument missing'));
2015-02-02 10:29:14 -08:00
2015-02-12 11:42:32 -08:00
if (_.isEmpty(opts.name))
2015-02-12 05:26:13 -08:00
return cb(new ClientError('Invalid copayer name'));
2015-02-08 08:36:19 -08:00
2015-04-08 11:18:28 -07:00
self.walletId = opts.walletId;
self._runLocked(cb, function(cb) {
2015-02-06 12:56:51 -08:00
self.storage.fetchWallet(opts.walletId, function(err, wallet) {
2015-03-31 13:28:01 -07:00
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-06 12:56:51 -08:00
if (!wallet) return cb(new ClientError('Wallet not found'));
2015-02-01 06:41:16 -08:00
2015-03-10 07:23:23 -07:00
var hash = WalletUtils.getCopayerHash(opts.name, opts.xPubKey, opts.requestPubKey);
if (!self._verifySignature(hash, opts.copayerSignature, wallet.pubKey)) {
2015-02-04 08:31:02 -08:00
return cb(new ClientError());
2015-02-01 06:41:16 -08:00
}
2015-02-02 06:55:03 -08:00
if (_.find(wallet.copayers, {
xPubKey: opts.xPubKey
2015-02-04 08:31:02 -08:00
})) return cb(new ClientError('CINWALLET', 'Copayer already in wallet'));
2015-02-17 12:36:45 -08:00
2015-02-12 11:42:32 -08:00
if (wallet.copayers.length == wallet.n)
2015-02-12 05:26:13 -08:00
return cb(new ClientError('WFULL', 'Wallet full'));
2015-02-02 06:55:03 -08:00
2015-02-17 15:26:58 -08:00
var copayer = Copayer.create({
2015-02-02 12:07:18 -08:00
name: opts.name,
2015-02-02 04:36:55 -08:00
copayerIndex: wallet.copayers.length,
2015-03-10 09:48:46 -07:00
xPubKey: opts.xPubKey,
2015-03-09 14:11:25 -07:00
requestPubKey: opts.requestPubKey,
2015-03-10 09:48:46 -07:00
signature: opts.copayerSignature,
2015-03-31 13:28:01 -07:00
isTemporaryRequestKey: !!opts.isTemporaryRequestKey,
2015-02-02 12:07:18 -08:00
});
2015-02-02 15:13:13 -08:00
2015-02-17 12:36:45 -08:00
self.storage.fetchCopayerLookup(copayer.id, function(err, res) {
2015-02-12 19:00:54 -08:00
if (err) return cb(err);
2015-02-17 12:36:45 -08:00
if (res)
return cb(new ClientError('CREGISTERED', 'Copayer ID already registered on server'));
2015-02-17 12:36:45 -08:00
wallet.addCopayer(copayer);
self.storage.storeWalletAndUpdateCopayersLookup(wallet, function(err) {
if (err) return cb(err);
self._notify('NewCopayer', {
walletId: opts.walletId,
copayerId: copayer.id,
copayerName: copayer.name,
2015-04-30 16:31:45 -07:00
}, false, function() {
return cb(null, {
copayerId: copayer.id,
wallet: wallet
});
2015-02-17 12:36:45 -08:00
});
2015-02-12 19:00:54 -08:00
});
2015-02-02 12:07:18 -08:00
});
});
});
2015-01-27 05:18:45 -08:00
};
2015-01-27 11:40:21 -08:00
/**
* Creates a new address.
* @param {Object} opts
2015-02-02 06:55:03 -08:00
* @returns {Address} address
2015-01-27 11:40:21 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.createAddress = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-02-06 12:56:51 -08:00
self.getWallet({}, function(err, wallet) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-12 11:42:32 -08:00
if (!wallet.isComplete())
2015-02-12 05:26:13 -08:00
return cb(new ClientError('Wallet is not complete'));
2015-02-02 15:13:13 -08:00
var address = wallet.createAddress(false);
2015-02-02 11:32:13 -08:00
2015-02-08 15:46:02 -08:00
self.storage.storeAddressAndWallet(wallet, address, function(err) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-04 11:18:36 -08:00
2015-03-30 08:45:43 -07:00
self._notify('NewAddress', {
address: address.address,
2015-04-30 16:31:45 -07:00
}, false, function() {
return cb(null, address);
2015-03-30 08:45:43 -07:00
});
2015-02-02 12:07:18 -08:00
});
});
});
2015-01-27 05:18:45 -08:00
};
2015-02-03 12:32:40 -08:00
/**
* Get all addresses.
* @param {Object} opts
* @returns {Address[]}
*/
2015-02-22 08:04:23 -08:00
WalletService.prototype.getMainAddresses = function(opts, cb) {
2015-02-03 12:32:40 -08:00
var self = this;
2015-02-06 12:56:51 -08:00
self.storage.fetchAddresses(self.walletId, function(err, addresses) {
2015-02-03 12:32:40 -08:00
if (err) return cb(err);
var onlyMain = _.reject(addresses, {
isChange: true
});
return cb(null, onlyMain);
2015-02-03 12:32:40 -08:00
});
};
2015-01-28 07:06:34 -08:00
/**
* Verifies that a given message was actually sent by an authorized copayer.
* @param {Object} opts
* @param {string} opts.message - The message to verify.
* @param {string} opts.signature - The signature of message to verify.
* @returns {truthy} The result of the verification.
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.verifyMessageSignature = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-28 07:06:34 -08:00
2015-02-11 07:10:47 -08:00
if (!Utils.checkRequired(opts, ['message', 'signature']))
return cb(new ClientError('Required argument missing'));
2015-02-02 10:29:14 -08:00
2015-02-06 12:56:51 -08:00
self.getWallet({}, function(err, wallet) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-01-28 09:21:09 -08:00
2015-02-06 12:56:51 -08:00
var copayer = wallet.getCopayer(self.copayerId);
2015-01-28 07:06:34 -08:00
var isValid = self._verifySignature(opts.message, opts.signature, copayer.requestPubKey);
2015-02-02 12:07:18 -08:00
return cb(null, isValid);
});
2015-01-28 07:06:34 -08:00
};
2015-01-27 05:18:45 -08:00
2015-03-30 16:16:51 -07:00
WalletService.prototype._getBlockchainExplorer = function(provider, network) {
if (!this.blockchainExplorer) {
2015-04-17 14:25:41 -07:00
var opts = {};
2015-04-15 06:59:25 -07:00
if (this.blockchainExplorerOpts && this.blockchainExplorerOpts[network]) {
2015-04-17 14:25:41 -07:00
opts = this.blockchainExplorerOpts[network];
2015-04-15 06:59:25 -07:00
}
opts.provider = provider;
opts.network = network;
this.blockchainExplorer = new BlockchainExplorer(opts);
2015-02-02 12:07:18 -08:00
}
2015-03-30 11:34:05 -07:00
2015-03-30 16:16:51 -07:00
return this.blockchainExplorer;
2015-01-27 05:18:45 -08:00
};
2015-02-05 12:22:38 -08:00
/**
* _getUtxos
*
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype._getUtxos = function(cb) {
2015-02-02 12:07:18 -08:00
var self = this;
// Get addresses for this wallet
2015-02-06 12:56:51 -08:00
self.storage.fetchAddresses(self.walletId, function(err, addresses) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-28 16:12:03 -08:00
if (addresses.length == 0) return cb(null, []);
2015-02-02 12:07:18 -08:00
2015-02-04 06:43:12 -08:00
var addressStrs = _.pluck(addresses, 'address');
var addressToPath = _.indexBy(addresses, 'address'); // TODO : check performance
2015-02-13 08:35:20 -08:00
var networkName = Bitcore.Address(addressStrs[0]).toObject().network;
2015-02-02 12:07:18 -08:00
2015-03-30 16:16:51 -07:00
var bc = self._getBlockchainExplorer('insight', networkName);
2015-02-13 13:24:35 -08:00
bc.getUnspentUtxos(addressStrs, function(err, inutxos) {
2015-04-14 11:18:33 -07:00
if (err) return cb(new Error('Could not fetch unspend outputs:' + err));
2015-02-13 13:24:35 -08:00
var utxos = _.map(inutxos, function(i) {
2015-02-21 20:01:15 -08:00
return _.pick(i.toObject(), ['txid', 'vout', 'address', 'scriptPubKey', 'amount', 'satoshis']);
2015-02-13 13:24:35 -08:00
});
2015-02-06 12:56:51 -08:00
self.getPendingTxs({}, function(err, txps) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-21 06:31:15 -08:00
var utxoKey = function(utxo) {
return utxo.txid + '|' + utxo.vout
};
2015-02-02 12:07:18 -08:00
var inputs = _.chain(txps)
.pluck('inputs')
.flatten()
2015-02-21 06:31:15 -08:00
.map(utxoKey)
.value();
2015-02-02 12:07:18 -08:00
2015-02-04 11:18:36 -08:00
var dictionary = _.reduce(utxos, function(memo, utxo) {
2015-02-21 06:31:15 -08:00
memo[utxoKey(utxo)] = utxo;
return memo;
}, {});
2015-02-02 12:07:18 -08:00
2015-02-02 15:13:13 -08:00
_.each(inputs, function(input) {
2015-02-02 12:07:18 -08:00
if (dictionary[input]) {
dictionary[input].locked = true;
}
});
2015-02-03 18:17:06 -08:00
2015-02-04 06:43:12 -08:00
// Needed for the clients to sign UTXOs
_.each(utxos, function(utxo) {
2015-02-18 11:47:15 -08:00
utxo.satoshis = utxo.satoshis ? +utxo.satoshis : Utils.strip(utxo.amount * 1e8);
delete utxo.amount;
2015-02-04 06:43:12 -08:00
utxo.path = addressToPath[utxo.address].path;
utxo.publicKeys = addressToPath[utxo.address].publicKeys;
});
2015-02-03 18:17:06 -08:00
2015-02-02 12:07:18 -08:00
return cb(null, utxos);
});
});
});
2015-01-27 05:18:45 -08:00
};
WalletService.prototype._totalizeUtxos = function(utxos) {
var balance = {};
balance.totalAmount = Utils.strip(_.reduce(utxos, function(sum, utxo) {
return sum + utxo.satoshis;
}, 0));
balance.lockedAmount = Utils.strip(_.reduce(_.filter(utxos, {
locked: true
}), function(sum, utxo) {
return sum + utxo.satoshis;
}, 0));
return balance;
};
2015-01-30 12:37:30 -08:00
/**
* Creates a new transaction proposal.
* @param {Object} opts
2015-02-02 06:55:03 -08:00
* @returns {Object} balance - Total amount & locked amount.
2015-01-30 12:37:30 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getBalance = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-30 12:37:30 -08:00
2015-02-06 12:56:51 -08:00
self._getUtxos(function(err, utxos) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-01-30 12:37:30 -08:00
var balance = self._totalizeUtxos(utxos);
2015-01-30 12:37:30 -08:00
2015-03-06 09:58:22 -08:00
// Compute balance by address
var byAddress = {};
_.each(_.indexBy(utxos, 'address'), function(value, key) {
byAddress[key] = {
address: key,
path: value.path,
amount: 0,
};
});
_.each(utxos, function(utxo) {
byAddress[utxo.address].amount += utxo.satoshis;
});
balance.byAddress = _.values(byAddress);
2015-02-02 12:07:18 -08:00
return cb(null, balance);
});
2015-01-30 12:37:30 -08:00
};
2015-03-11 11:04:42 -07:00
WalletService.prototype._selectTxInputs = function(txp, cb) {
var self = this;
2015-01-30 12:37:30 -08:00
2015-03-11 11:04:42 -07:00
self._getUtxos(function(err, utxos) {
if (err) return cb(err);
var balance = self._totalizeUtxos(utxos);
2015-03-25 08:17:41 -07:00
if (balance.totalAmount < txp.amount)
return cb(new ClientError('INSUFFICIENTFUNDS', 'Insufficient funds'));
if ((balance.totalAmount - balance.lockedAmount) < txp.amount)
return cb(new ClientError('LOCKEDFUNDS', 'Funds are locked by pending transaction proposals'));
2015-03-11 11:04:42 -07:00
utxos = _.reject(utxos, {
locked: true
});
2015-02-04 11:18:36 -08:00
2015-03-11 11:04:42 -07:00
var i = 0;
var total = 0;
var selected = [];
var inputs = _.sortBy(utxos, 'amount');
2015-03-25 08:17:41 -07:00
var bitcoreTx, bitcoreError;
2015-03-11 11:04:42 -07:00
while (i < inputs.length) {
selected.push(inputs[i]);
total += inputs[i].satoshis;
2015-03-25 08:17:41 -07:00
i++;
2015-03-11 11:04:42 -07:00
2015-03-25 08:17:41 -07:00
if (total >= txp.amount) {
2015-03-11 11:04:42 -07:00
try {
txp.inputs = selected;
bitcoreTx = txp.getBitcoreTx();
2015-03-25 08:17:41 -07:00
bitcoreError = bitcoreTx.getSerializationError({
disableIsFullySigned: true,
});
if (!bitcoreError) {
txp.inputPaths = _.pluck(txp.inputs, 'path');
txp.fee = bitcoreTx.getFee();
return cb();
2015-03-11 11:04:42 -07:00
}
2015-03-25 08:17:41 -07:00
} catch (ex) {
return cb(ex);
2015-02-16 10:00:41 -08:00
}
2015-02-16 09:27:01 -08:00
}
2015-03-11 11:04:42 -07:00
};
2015-03-25 08:17:41 -07:00
if (bitcoreError instanceof Bitcore.errors.Transaction.FeeError) {
return cb(new ClientError('INSUFFICIENTFUNDS', 'Insufficient funds for fee'));
}
if (bitcoreError instanceof Bitcore.errors.Transaction.DustOutputs) {
return cb(new ClientError('DUSTAMOUNT', 'Amount below dust threshold'));
}
2015-03-25 12:02:31 -07:00
return cb(bitcoreError || new Error('Could not select tx inputs'));
2015-03-11 11:04:42 -07:00
});
2015-01-30 13:29:46 -08:00
};
2015-01-27 07:54:17 -08:00
/**
* Creates a new transaction proposal.
2015-01-27 11:40:21 -08:00
* @param {Object} opts
2015-01-27 07:54:17 -08:00
* @param {string} opts.toAddress - Destination address.
* @param {number} opts.amount - Amount to transfer in satoshi.
* @param {string} opts.message - A message to attach to this transaction.
2015-03-25 12:44:47 -07:00
* @param {string} opts.proposalSignature - S(toAddress|amount|message|payProUrl). Used by other copayers to verify the proposal.
* @param {string} opts.payProUrl - Options: Paypro URL for peers to verify TX
2015-02-02 06:55:03 -08:00
* @returns {TxProposal} Transaction proposal.
2015-01-27 07:54:17 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.createTx = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-27 05:18:45 -08:00
2015-02-13 11:57:28 -08:00
if (!Utils.checkRequired(opts, ['toAddress', 'amount', 'proposalSignature']))
2015-02-11 07:10:47 -08:00
return cb(new ClientError('Required argument missing'));
2015-02-02 10:29:14 -08:00
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-02-08 13:29:58 -08:00
self.getWallet({}, function(err, wallet) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-08 13:29:58 -08:00
if (!wallet.isComplete()) return cb(new ClientError('Wallet is not complete'));
2015-02-10 05:22:23 -08:00
var copayer = wallet.getCopayer(self.copayerId);
2015-03-26 13:52:59 -07:00
var hash = WalletUtils.getProposalHash(opts.toAddress, opts.amount, opts.message, opts.payProUrl);
if (!self._verifySignature(hash, opts.proposalSignature, copayer.requestPubKey))
2015-02-11 18:11:30 -08:00
return cb(new ClientError('Invalid proposal signature'));
2015-01-28 12:06:29 -08:00
2015-02-08 14:10:06 -08:00
var toAddress;
try {
toAddress = new Bitcore.Address(opts.toAddress);
} catch (ex) {
return cb(new ClientError('INVALIDADDRESS', 'Invalid address'));
}
2015-02-11 18:11:30 -08:00
if (toAddress.network != wallet.getNetworkName())
return cb(new ClientError('INVALIDADDRESS', 'Incorrect address network'));
2015-02-08 14:10:06 -08:00
2015-02-24 05:36:14 -08:00
if (opts.amount <= 0)
return cb(new ClientError('Invalid amount'));
2015-02-16 10:00:41 -08:00
if (opts.amount < Bitcore.Transaction.DUST_AMOUNT)
2015-02-16 09:41:12 -08:00
return cb(new ClientError('DUSTAMOUNT', 'Amount below dust threshold'));
2015-01-30 12:37:30 -08:00
2015-03-11 11:04:42 -07:00
var changeAddress = wallet.createAddress(true);
2015-01-30 12:37:30 -08:00
2015-03-11 11:04:42 -07:00
var txp = TxProposal.create({
2015-03-30 07:29:19 -07:00
walletId: self.walletId,
2015-03-11 11:04:42 -07:00
creatorId: self.copayerId,
toAddress: opts.toAddress,
amount: opts.amount,
message: opts.message,
proposalSignature: opts.proposalSignature,
2015-03-25 12:44:47 -07:00
payProUrl: opts.payProUrl,
2015-03-11 11:04:42 -07:00
changeAddress: changeAddress,
requiredSignatures: wallet.m,
requiredRejections: Math.min(wallet.m, wallet.n - wallet.m + 1),
});
2015-01-28 07:06:34 -08:00
2015-03-11 11:04:42 -07:00
self._selectTxInputs(txp, function(err) {
if (err) return cb(err);
2015-02-16 10:00:41 -08:00
2015-03-11 11:04:42 -07:00
$.checkState(txp.inputs);
2015-02-08 13:29:58 -08:00
2015-02-08 15:46:02 -08:00
self.storage.storeAddressAndWallet(wallet, changeAddress, function(err) {
2015-02-08 13:29:58 -08:00
if (err) return cb(err);
2015-01-28 07:06:34 -08:00
2015-02-08 15:46:02 -08:00
self.storage.storeTx(wallet.id, txp, function(err) {
2015-02-08 13:29:58 -08:00
if (err) return cb(err);
2015-02-11 18:11:30 -08:00
self._notify('NewTxProposal', {
amount: opts.amount
2015-04-30 16:31:45 -07:00
}, false, function() {
return cb(null, txp);
2015-02-11 18:11:30 -08:00
});
2015-02-08 15:46:02 -08:00
});
2015-02-08 13:29:58 -08:00
});
2015-02-02 12:07:18 -08:00
});
});
});
2015-02-02 15:13:13 -08:00
};
2015-01-28 07:06:34 -08:00
2015-02-04 10:45:08 -08:00
/**
* Retrieves a tx from storage.
* @param {Object} opts
2015-02-26 05:41:55 -08:00
* @param {string} opts.txProposalId - The tx id.
2015-02-04 10:45:08 -08:00
* @returns {Object} txProposal
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getTx = function(opts, cb) {
2015-02-04 10:45:08 -08:00
var self = this;
2015-02-26 05:41:55 -08:00
self.storage.fetchTx(self.walletId, opts.txProposalId, function(err, txp) {
2015-02-04 10:45:08 -08:00
if (err) return cb(err);
if (!txp) return cb(new ClientError('Transaction proposal not found'));
return cb(null, txp);
});
};
2015-02-09 13:07:15 -08:00
/**
2015-02-10 11:11:44 -08:00
* removeWallet
*
* @param opts
* @param cb
* @return {undefined}
2015-02-09 13:07:15 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.removeWallet = function(opts, cb) {
2015-02-10 11:11:44 -08:00
var self = this;
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-02-10 11:11:44 -08:00
self.storage.removeWallet(self.walletId, cb);
});
2015-02-09 13:07:15 -08:00
};
2015-02-10 11:30:58 -08:00
/**
* removePendingTx
*
* @param opts
2015-02-15 08:03:48 -08:00
* @param {string} opts.txProposalId - The tx id.
2015-02-10 11:30:58 -08:00
* @return {undefined}
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.removePendingTx = function(opts, cb) {
2015-02-10 11:30:58 -08:00
var self = this;
2015-02-15 08:03:48 -08:00
if (!Utils.checkRequired(opts, ['txProposalId']))
2015-02-10 11:30:58 -08:00
return cb(new ClientError('Required argument missing'));
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-02-10 11:30:58 -08:00
2015-02-11 07:05:21 -08:00
self.getTx({
2015-02-26 05:41:55 -08:00
txProposalId: opts.txProposalId,
2015-02-11 07:05:21 -08:00
}, function(err, txp) {
2015-02-10 11:30:58 -08:00
if (err) return cb(err);
if (!txp.isPending())
2015-02-23 13:31:27 -08:00
return cb(new ClientError('TXNOTPENDING', 'Transaction proposal not pending'));
2015-02-10 11:30:58 -08:00
2015-02-10 13:04:50 -08:00
if (txp.creatorId !== self.copayerId)
2015-02-11 07:05:21 -08:00
return cb(new ClientError('Only creators can remove pending proposals'));
2015-02-10 13:04:50 -08:00
2015-02-10 11:30:58 -08:00
var actors = txp.getActors();
2015-02-11 07:05:21 -08:00
if (actors.length > 1 || (actors.length == 1 && actors[0] !== self.copayerId))
2015-02-23 13:31:27 -08:00
return cb(new ClientError('TXACTIONED', 'Cannot remove a proposal signed/rejected by other copayers'));
2015-02-10 11:30:58 -08:00
2015-04-30 16:31:45 -07:00
self._notify('TxProposalRemoved', {}, false, function() {
self.storage.removeTx(self.walletId, txp.id, cb);
});
2015-02-10 11:30:58 -08:00
});
});
};
2015-02-09 13:07:15 -08:00
2015-02-20 12:32:19 -08:00
WalletService.prototype._broadcastTx = function(txp, cb) {
2015-02-16 09:27:01 -08:00
var raw;
try {
raw = txp.getRawTx();
} catch (ex) {
return cb(ex);
}
2015-03-30 16:16:51 -07:00
var bc = this._getBlockchainExplorer('insight', txp.getNetworkName());
2015-02-05 12:22:38 -08:00
bc.broadcast(raw, function(err, txid) {
return cb(err, txid);
})
2015-01-28 08:28:18 -08:00
};
2015-01-28 07:06:34 -08:00
/**
* Sign a transaction proposal.
* @param {Object} opts
2015-01-28 08:28:18 -08:00
* @param {string} opts.txProposalId - The identifier of the transaction.
2015-02-04 11:18:36 -08:00
* @param {string} opts.signatures - The signatures of the inputs of this tx for this copayer (in apperance order)
2015-01-28 07:06:34 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.signTx = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-28 07:06:34 -08:00
2015-02-11 07:10:47 -08:00
if (!Utils.checkRequired(opts, ['txProposalId', 'signatures']))
return cb(new ClientError('Required argument missing'));
2015-02-02 10:29:14 -08:00
2015-02-06 12:56:51 -08:00
self.getWallet({}, function(err, wallet) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-01-27 05:18:45 -08:00
2015-02-05 10:50:18 -08:00
self.getTx({
2015-02-26 05:41:55 -08:00
txProposalId: opts.txProposalId
2015-02-05 10:50:18 -08:00
}, function(err, txp) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-10 11:30:58 -08:00
2015-02-05 10:50:18 -08:00
var action = _.find(txp.actions, {
2015-02-10 11:30:58 -08:00
copayerId: self.copayerId
2015-02-05 10:50:18 -08:00
});
2015-02-05 12:22:38 -08:00
if (action)
2015-02-05 10:50:18 -08:00
return cb(new ClientError('CVOTED', 'Copayer already voted on this transaction proposal'));
2015-02-23 13:31:27 -08:00
if (!txp.isPending())
2015-02-05 10:50:18 -08:00
return cb(new ClientError('TXNOTPENDING', 'The transaction proposal is not pending'));
2015-01-28 11:40:07 -08:00
2015-02-06 12:56:51 -08:00
var copayer = wallet.getCopayer(self.copayerId);
2015-01-27 05:18:45 -08:00
2015-02-13 16:00:12 -08:00
if (!txp.sign(self.copayerId, opts.signatures, copayer.xPubKey))
2015-02-05 10:50:18 -08:00
return cb(new ClientError('BADSIGNATURES', 'Bad signatures'));
2015-02-06 12:56:51 -08:00
self.storage.storeTx(self.walletId, txp, function(err) {
2015-02-05 10:50:18 -08:00
if (err) return cb(err);
2015-04-30 16:31:45 -07:00
async.parallel([
function(done) {
self._notify('TxProposalAcceptedBy', {
txProposalId: opts.txProposalId,
copayerId: self.copayerId,
}, false, done);
},
function(done) {
if (txp.isAccepted()) {
self._notify('TxProposalFinallyAccepted', {
txProposalId: opts.txProposalId,
}, false, done);
} else {
done();
}
},
], function() {
return cb(null, txp);
2015-02-11 18:11:30 -08:00
});
2015-02-05 10:50:18 -08:00
});
2015-02-02 12:07:18 -08:00
});
});
2015-02-02 06:55:03 -08:00
};
2015-01-27 05:18:45 -08:00
2015-02-15 13:52:48 -08:00
/**
* Broadcast a transaction proposal.
* @param {Object} opts
* @param {string} opts.txProposalId - The identifier of the transaction.
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.broadcastTx = function(opts, cb) {
2015-02-15 13:52:48 -08:00
var self = this;
if (!Utils.checkRequired(opts, ['txProposalId']))
return cb(new ClientError('Required argument missing'));
self.getWallet({}, function(err, wallet) {
if (err) return cb(err);
self.getTx({
2015-02-26 05:41:55 -08:00
txProposalId: opts.txProposalId
2015-02-15 13:52:48 -08:00
}, function(err, txp) {
if (err) return cb(err);
if (txp.status == 'broadcasted')
return cb(new ClientError('TXALREADYBROADCASTED', 'The transaction proposal is already broadcasted'));
if (txp.status != 'accepted')
return cb(new ClientError('TXNOTACCEPTED', 'The transaction proposal is not accepted'));
self._broadcastTx(txp, function(err, txid) {
if (err) return cb(err);
txp.setBroadcasted(txid);
self.storage.storeTx(self.walletId, txp, function(err) {
if (err) return cb(err);
self._notify('NewOutgoingTx', {
txProposalId: opts.txProposalId,
txid: txid
2015-04-30 16:31:45 -07:00
}, false, function() {
return cb(null, txp);
2015-02-15 13:52:48 -08:00
});
});
});
});
});
};
2015-01-29 09:57:26 -08:00
/**
* Reject a transaction proposal.
* @param {Object} opts
* @param {string} opts.txProposalId - The identifier of the transaction.
2015-02-02 10:29:14 -08:00
* @param {string} [opts.reason] - A message to other copayers explaining the rejection.
2015-01-29 09:57:26 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.rejectTx = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-29 09:57:26 -08:00
2015-02-11 07:10:47 -08:00
if (!Utils.checkRequired(opts, ['txProposalId']))
return cb(new ClientError('Required argument missing'));
2015-02-02 10:29:14 -08:00
2015-02-04 11:27:36 -08:00
self.getTx({
2015-02-26 05:41:55 -08:00
txProposalId: opts.txProposalId
2015-02-04 11:27:36 -08:00
}, function(err, txp) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-02-11 06:27:52 -08:00
2015-02-02 15:13:13 -08:00
var action = _.find(txp.actions, {
2015-02-06 12:56:51 -08:00
copayerId: self.copayerId
2015-02-02 15:13:13 -08:00
});
2015-02-11 18:11:30 -08:00
if (action)
return cb(new ClientError('CVOTED', 'Copayer already voted on this transaction proposal'));
if (txp.status != 'pending')
return cb(new ClientError('TXNOTPENDING', 'The transaction proposal is not pending'));
2015-01-29 09:57:26 -08:00
2015-02-15 10:46:29 -08:00
txp.reject(self.copayerId, opts.reason);
2015-01-29 09:57:26 -08:00
2015-02-06 12:56:51 -08:00
self.storage.storeTx(self.walletId, txp, function(err) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-01-29 09:57:26 -08:00
2015-04-30 16:31:45 -07:00
async.parallel([
2015-02-11 18:11:30 -08:00
2015-04-30 16:31:45 -07:00
function(done) {
self._notify('TxProposalRejectedBy', {
txProposalId: opts.txProposalId,
copayerId: self.copayerId,
}, false, done);
},
function(done) {
if (txp.status == 'rejected') {
self._notify('TxProposalFinallyRejected', {
txProposalId: opts.txProposalId,
}, false, done);
} else {
done();
}
},
], function() {
return cb(null, txp);
});
2015-02-02 12:07:18 -08:00
});
});
2015-01-29 09:57:26 -08:00
};
2015-01-28 07:06:34 -08:00
/**
2015-02-21 17:35:12 -08:00
* Retrieves pending transaction proposals.
2015-01-28 07:06:34 -08:00
* @param {Object} opts
2015-02-02 06:55:03 -08:00
* @returns {TxProposal[]} Transaction proposal.
2015-01-28 07:06:34 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getPendingTxs = function(opts, cb) {
2015-02-02 12:07:18 -08:00
var self = this;
2015-01-27 05:18:45 -08:00
2015-02-06 12:51:21 -08:00
self.storage.fetchPendingTxs(self.walletId, function(err, txps) {
2015-02-02 12:07:18 -08:00
if (err) return cb(err);
2015-01-28 12:40:37 -08:00
2015-02-06 23:09:45 -08:00
return cb(null, txps);
2015-02-02 12:07:18 -08:00
});
2015-01-27 05:18:45 -08:00
};
2015-02-06 23:09:45 -08:00
/**
2015-02-21 17:35:12 -08:00
* Retrieves all transaction proposals in the range (maxTs-minTs)
2015-02-12 05:26:13 -08:00
* Times are in UNIX EPOCH
*
2015-02-06 23:09:45 -08:00
* @param {Object} opts.minTs (defaults to 0)
* @param {Object} opts.maxTs (defaults to now)
* @param {Object} opts.limit
2015-04-20 15:45:45 -07:00
* @returns {TxProposal[]} Transaction proposals, newer first
2015-02-06 23:09:45 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getTxs = function(opts, cb) {
2015-02-06 23:09:45 -08:00
var self = this;
self.storage.fetchTxs(self.walletId, opts, function(err, txps) {
if (err) return cb(err);
return cb(null, txps);
});
};
2015-02-11 18:13:19 -08:00
/**
2015-02-21 17:35:12 -08:00
* Retrieves notifications in the range (maxTs-minTs).
2015-02-12 05:26:13 -08:00
* Times are in UNIX EPOCH. Order is assured even for events with the same time
*
2015-02-11 18:13:19 -08:00
* @param {Object} opts.minTs (defaults to 0)
* @param {Object} opts.maxTs (defaults to now)
* @param {Object} opts.limit
2015-02-12 05:26:13 -08:00
* @param {Object} opts.reverse (default false)
2015-02-12 11:42:32 -08:00
* @returns {Notification[]} Notifications
2015-02-11 18:13:19 -08:00
*/
2015-02-20 12:32:19 -08:00
WalletService.prototype.getNotifications = function(opts, cb) {
2015-02-11 18:13:19 -08:00
var self = this;
self.storage.fetchNotifications(self.walletId, opts, function(err, notifications) {
if (err) return cb(err);
return cb(null, notifications);
});
};
2015-02-21 17:35:12 -08:00
WalletService.prototype._normalizeTxHistory = function(txs) {
return _.map(txs, function(tx) {
var inputs = _.map(tx.vin, function(item) {
return {
address: item.addr,
amount: item.valueSat,
}
});
var outputs = _.map(tx.vout, function(item) {
var itemAddr;
// If classic multisig, ignore
if (item.scriptPubKey && item.scriptPubKey.addresses.length == 1) {
itemAddr = item.scriptPubKey.addresses[0];
}
return {
address: itemAddr,
amount: parseInt((item.value * 1e8).toFixed(0)),
}
});
return {
txid: tx.txid,
confirmations: tx.confirmations,
fees: parseInt((tx.fees * 1e8).toFixed(0)),
2015-02-22 18:26:21 -08:00
time: !_.isNaN(tx.time) ? tx.time : Math.floor(Date.now() / 1000),
2015-02-21 17:35:12 -08:00
inputs: inputs,
outputs: outputs,
};
});
};
/**
* Retrieves all transactions (incoming & outgoing)
2015-02-21 17:35:12 -08:00
* Times are in UNIX EPOCH
*
2015-03-17 15:46:01 -07:00
* @param {Object} opts
* @param {Number} opts.skip (defaults to 0)
2015-03-17 15:46:01 -07:00
* @param {Number} opts.limit
2015-02-21 17:35:12 -08:00
* @returns {TxProposal[]} Transaction proposals, first newer
*/
WalletService.prototype.getTxHistory = function(opts, cb) {
var self = this;
function decorate(txs, addresses, proposals) {
var indexedAddresses = _.indexBy(addresses, 'address');
var indexedProposals = _.indexBy(proposals, 'txid');
2015-02-21 17:35:12 -08:00
function sum(items, isMine, isChange) {
var filter = {};
if (_.isBoolean(isMine)) filter.isMine = isMine;
if (_.isBoolean(isChange)) filter.isChange = isChange;
return _.reduce(_.where(items, filter),
function(memo, item) {
return memo + item.amount;
}, 0);
};
function classify(items) {
return _.map(items, function(item) {
2015-02-21 17:35:12 -08:00
var address = indexedAddresses[item.address];
return {
address: item.address,
amount: item.amount,
isMine: !!address,
isChange: address ? address.isChange : false,
}
2015-02-21 17:35:12 -08:00
});
};
2015-03-19 09:17:51 -07:00
var now = Math.floor(Date.now() / 1000);
2015-02-21 17:35:12 -08:00
return _.map(txs, function(tx) {
var inputs = classify(tx.inputs);
var outputs = classify(tx.outputs);
var amountIn = sum(inputs, true);
var amountOut = sum(outputs, true, false);
var amountOutChange = sum(outputs, true, true);
var amount, action, addressTo;
2015-02-21 17:35:12 -08:00
if (amountIn == (amountOut + amountOutChange + (amountIn > 0 ? tx.fees : 0))) {
amount = amountOut;
action = 'moved';
2015-02-21 17:35:12 -08:00
} else {
amount = amountIn - amountOut - amountOutChange - (amountIn > 0 ? tx.fees : 0);
action = amount > 0 ? 'sent' : 'received';
2015-02-21 17:35:12 -08:00
}
amount = Math.abs(amount);
if (action == 'sent' || action == 'moved') {
var firstExternalOutput = _.find(outputs, {
isMine: false
});
addressTo = firstExternalOutput ? firstExternalOutput.address : 'N/A';
2015-02-21 17:35:12 -08:00
};
2015-02-11 18:13:19 -08:00
var newTx = {
txid: tx.txid,
action: action,
amount: amount,
fees: tx.fees,
2015-03-19 09:17:51 -07:00
time: tx.time || now,
addressTo: addressTo,
confirmations: tx.confirmations,
};
2015-02-21 17:35:12 -08:00
var proposal = indexedProposals[tx.txid];
if (proposal) {
newTx.proposalId = proposal.id;
newTx.creatorName = proposal.creatorName;
newTx.message = proposal.message;
newTx.actions = _.map(proposal.actions, function(action) {
2015-02-22 18:26:21 -08:00
return _.pick(action, ['createdOn', 'type', 'copayerId', 'copayerName', 'comment']);
});
// newTx.sentTs = proposal.sentTs;
// newTx.merchant = proposal.merchant;
//newTx.paymentAckMemo = proposal.paymentAckMemo;
2015-02-21 17:35:12 -08:00
}
return newTx;
2015-02-21 17:35:12 -08:00
});
};
2015-02-21 19:12:05 -08:00
function paginate(txs) {
var skip = opts.skip || 0;
var limited = _.isNumber(opts.limit) && opts.limit != -1;
2015-03-17 06:59:00 -07:00
var sliced = _.slice(_.sortBy(txs, function(tx) {
2015-03-18 07:56:49 -07:00
return -tx.time;
}), skip);
2015-03-17 06:59:00 -07:00
return limited ? _.take(sliced, opts.limit) : sliced;
2015-02-21 19:12:05 -08:00
};
2015-02-21 17:35:12 -08:00
// Get addresses for this wallet
self.storage.fetchAddresses(self.walletId, function(err, addresses) {
if (err) return cb(err);
if (addresses.length == 0) return cb(null, []);
var addressStrs = _.pluck(addresses, 'address');
var networkName = Bitcore.Address(addressStrs[0]).toObject().network;
2015-03-30 16:16:51 -07:00
var bc = self._getBlockchainExplorer('insight', networkName);
2015-02-21 17:35:12 -08:00
async.parallel([
function(next) {
2015-04-20 12:05:02 -07:00
self.storage.fetchTxs(self.walletId, {}, function(err, txps) {
2015-02-21 17:35:12 -08:00
if (err) return next(err);
next(null, txps);
});
},
function(next) {
bc.getTransactions(addressStrs, null, null, function(err, txs) {
2015-04-14 11:28:29 -07:00
if (err) return next(new Error('Could not fetch transactions: ' + err));
2015-02-21 17:35:12 -08:00
next(null, self._normalizeTxHistory(txs));
});
},
], function(err, res) {
if (err) return cb(err);
var proposals = res[0];
var txs = res[1];
txs = paginate(decorate(txs, addresses, proposals));
2015-02-21 17:35:12 -08:00
return cb(null, txs);
});
});
};
2015-02-11 18:13:19 -08:00
2015-02-06 23:09:45 -08:00
2015-04-01 12:42:12 -07:00
WalletService.scanConfig = {
2015-04-01 13:21:06 -07:00
SCAN_WINDOW: 20,
DERIVATION_DELAY: 10, // in milliseconds
2015-04-01 12:42:12 -07:00
};
/**
* Scan the blockchain looking for addresses having some activity
*
* @param {Object} opts
* @param {Boolean} opts.includeCopayerBranches (defaults to false)
*/
WalletService.prototype.scan = function(opts, cb) {
2015-04-01 12:42:12 -07:00
var self = this;
opts = opts || {};
2015-04-01 13:48:54 -07:00
function deriveAddresses(size, derivator, cb) {
2015-04-01 14:25:18 -07:00
async.mapSeries(_.range(size), function(i, next) {
2015-04-01 13:21:06 -07:00
setTimeout(function() {
2015-04-17 14:25:41 -07:00
next(null, derivator.derive());
2015-04-01 13:21:06 -07:00
}, WalletService.scanConfig.DERIVATION_DELAY)
2015-04-01 12:42:12 -07:00
}, cb);
};
2015-04-15 09:23:30 -07:00
function checkActivity(addresses, networkName, cb) {
var bc = self._getBlockchainExplorer('insight', networkName);
2015-04-01 12:42:12 -07:00
bc.getAddressActivity(addresses, cb);
};
2015-04-01 13:48:54 -07:00
function scanBranch(derivator, cb) {
2015-04-01 12:42:12 -07:00
var activity = true;
var allAddresses = [];
2015-04-15 09:23:30 -07:00
var networkName;
2015-04-01 12:42:12 -07:00
async.whilst(function() {
return activity;
}, function(next) {
2015-04-01 13:48:54 -07:00
deriveAddresses(WalletService.scanConfig.SCAN_WINDOW, derivator, function(err, addresses) {
2015-04-01 12:42:12 -07:00
if (err) return next(err);
2015-04-15 09:23:30 -07:00
networkName = networkName || Bitcore.Address(addresses[0].address).toObject().network;
checkActivity(_.pluck(addresses, 'address'), networkName, function(err, thereIsActivity) {
2015-04-14 11:28:29 -07:00
if (err) return next(new Error('Could not fetch TX activity:' + err));
2015-04-01 12:42:12 -07:00
activity = thereIsActivity;
2015-04-17 14:25:41 -07:00
if (thereIsActivity) {
allAddresses.push(addresses);
} else {
derivator.rewind(WalletService.scanConfig.SCAN_WINDOW);
}
2015-04-01 12:42:12 -07:00
next();
});
});
}, function(err) {
return cb(err, _.flatten(allAddresses));
});
2015-04-01 12:42:12 -07:00
};
2015-04-08 11:18:28 -07:00
self._runLocked(cb, function(cb) {
2015-04-01 12:42:12 -07:00
self.getWallet({}, function(err, wallet) {
if (err) return cb(err);
2015-04-02 07:18:39 -07:00
if (!wallet.isComplete()) return cb(new ClientError('Wallet is not complete'));
2015-04-01 12:42:12 -07:00
2015-04-15 06:57:10 -07:00
wallet.scanStatus = 'running';
self.storage.storeWallet(wallet, function(err) {
if (err) return cb(err);
2015-04-01 12:42:12 -07:00
2015-04-15 06:57:10 -07:00
var derivators = [];
_.each([false, true], function(isChange) {
2015-04-17 14:25:41 -07:00
derivators.push({
derive: _.bind(wallet.createAddress, wallet, isChange),
rewind: _.bind(wallet.addressManager.rewindIndex, wallet.addressManager, isChange),
});
2015-04-15 06:57:10 -07:00
if (opts.includeCopayerBranches) {
_.each(wallet.copayers, function(copayer) {
2015-04-17 14:25:41 -07:00
derivators.push({
derive: _.bind(copayer.createAddress, copayer, wallet, isChange),
rewind: _.bind(copayer.addressManager.rewindIndex, copayer.addressManager, isChange),
});
2015-04-15 06:57:10 -07:00
});
}
});
async.eachSeries(derivators, function(derivator, next) {
scanBranch(derivator, function(err, addresses) {
if (err) return next(err);
self.storage.storeAddressAndWallet(wallet, addresses, next);
});
}, function(err) {
wallet.scanStatus = err ? 'error' : 'success';
self.storage.storeWallet(wallet, function() {
return cb(err);
});
2015-04-01 12:42:12 -07:00
});
2015-04-15 06:57:10 -07:00
});
2015-04-01 12:42:12 -07:00
});
});
2015-04-02 07:18:39 -07:00
};
/**
* Start a scan process.
*
* @param {Object} opts
* @param {Boolean} opts.includeCopayerBranches (defaults to false)
*/
WalletService.prototype.startScan = function(opts, cb) {
var self = this;
function scanFinished(err) {
2015-04-14 11:41:27 -07:00
var data = {
2015-04-15 06:57:10 -07:00
result: err ? 'error' : 'success',
2015-04-14 11:41:27 -07:00
};
if (err) data.error = err;
2015-04-15 06:57:10 -07:00
self._notify('ScanFinished', data, true);
2015-04-02 07:18:39 -07:00
};
2015-04-03 14:49:08 -07:00
self.getWallet({}, function(err, wallet) {
if (err) return cb(err);
if (!wallet.isComplete()) return cb(new ClientError('Wallet is not complete'));
2015-04-02 07:18:39 -07:00
2015-04-15 06:57:10 -07:00
setTimeout(function() {
self.scan(opts, scanFinished);
}, 100);
2015-04-15 06:57:10 -07:00
return cb(null, {
started: true
});
2015-04-02 07:18:39 -07:00
});
};
2015-04-02 07:18:39 -07:00
2015-02-20 12:32:19 -08:00
module.exports = WalletService;
2015-02-09 10:30:16 -08:00
module.exports.ClientError = ClientError;