bitcore-wallet-service/lib/client/api.js

560 lines
14 KiB
JavaScript
Raw Normal View History

2015-02-12 11:42:32 -08:00
'use strict';
var _ = require('lodash');
2015-02-16 14:54:38 -08:00
var $ = require('preconditions').singleton();
2015-02-15 08:14:36 -08:00
var util = require('util');
2015-02-12 11:42:32 -08:00
var async = require('async');
var log = require('npmlog');
var request = require('request')
log.debug = log.verbose;
var Bitcore = require('bitcore')
2015-02-17 11:42:47 -08:00
var WalletUtils = require('../walletutils');
2015-02-16 08:10:48 -08:00
var Verifier = require('./verifier');
2015-02-16 11:23:42 -08:00
var ServerCompromisedError = require('./servercompromisederror')
2015-02-12 11:42:32 -08:00
2015-02-13 11:07:47 -08:00
var BASE_URL = 'http://localhost:3001/copay/api';
2015-02-12 11:42:32 -08:00
2015-02-18 13:14:24 -08:00
var WALLET_CRITICAL_DATA = ['xPrivKey', 'm', 'publicKeyRing', 'sharedEncryptingKey'];
2015-02-17 10:13:42 -08:00
2015-02-20 10:11:30 -08:00
function _encryptMessage(message, encryptingKey) {
if (!message) return null;
return WalletUtils.encryptMessage(message, encryptingKey);
};
2015-02-20 10:11:30 -08:00
function _decryptMessage(message, encryptingKey) {
if (!message) return '';
2015-02-18 13:14:24 -08:00
try {
return WalletUtils.decryptMessage(message, encryptingKey);
} catch (ex) {
return '<ECANNOTDECRYPT>';
}
2015-02-13 07:45:05 -08:00
};
2015-02-12 11:50:10 -08:00
2015-02-20 10:11:30 -08:00
function _processTxps(txps, encryptingKey) {
_.each([].concat(txps), function(txp) {
txp.decryptedMessage = _decryptMessage(txp.message, encryptingKey);
_.each(txp.actions, function(action) {
action.comment = _decryptMessage(action.comment, encryptingKey);
});
});
};
2015-02-12 19:00:54 -08:00
function _parseError(body) {
if (_.isString(body)) {
2015-02-13 07:45:05 -08:00
try {
2015-02-13 08:35:20 -08:00
body = JSON.parse(body);
2015-02-13 07:45:05 -08:00
} catch (e) {
2015-02-13 08:35:20 -08:00
body = {
error: body
};
2015-02-13 07:45:05 -08:00
}
2015-02-12 19:00:54 -08:00
}
var code = body.code || 'ERROR';
var message = body.error || 'There was an unknown error processing the request';
log.error(code, message);
2015-02-19 16:37:13 -08:00
return {
message: message,
code: code
};
2015-02-12 19:00:54 -08:00
};
2015-02-15 06:12:04 -08:00
function _signRequest(method, url, args, privKey) {
var message = method.toLowerCase() + '|' + url + '|' + JSON.stringify(args);
2015-02-17 11:42:47 -08:00
return WalletUtils.signMessage(message, privKey);
2015-02-12 19:00:54 -08:00
};
2015-02-12 11:42:32 -08:00
2015-02-15 06:33:04 -08:00
function API(opts) {
2015-02-15 08:14:36 -08:00
if (!opts.storage) {
throw new Error('Must provide storage option');
}
this.storage = opts.storage;
this.verbose = !!opts.verbose;
2015-02-16 10:02:20 -08:00
this.request = request || opts.request;
2015-02-17 15:31:03 -08:00
this.baseUrl = opts.baseUrl || BASE_URL;
2015-02-19 16:37:13 -08:00
this.basePath = this.baseUrl.replace(/http.?:\/\/[a-zA-Z0-9:-]*\//, '/');
2015-02-15 08:14:36 -08:00
if (this.verbose) {
log.level = 'debug';
2015-02-19 07:32:10 -08:00
} else {
log.level = 'info';
2015-02-12 18:57:16 -08:00
}
2015-02-12 11:42:32 -08:00
};
2015-02-16 16:10:14 -08:00
API.prototype._tryToComplete = function(data, cb) {
var self = this;
var url = '/v1/wallets/';
2015-02-17 06:48:19 -08:00
self._doGetRequest(url, data, function(err, ret) {
2015-02-16 16:10:14 -08:00
if (err) return cb(err);
2015-02-17 06:48:19 -08:00
var wallet = ret.wallet;
2015-02-16 21:20:04 -08:00
2015-02-17 06:48:19 -08:00
if (wallet.status != 'complete')
2015-02-16 21:20:04 -08:00
return cb('Wallet Incomplete');
2015-02-16 16:10:14 -08:00
2015-02-17 06:48:19 -08:00
if (!Verifier.checkCopayers(wallet.copayers, data.walletPrivKey, data.xPrivKey, data.n))
2015-02-19 15:04:05 -08:00
return cb(new ServerCompromisedError(
'Copayers in the wallet could not be verified to have known the wallet secret'));
2015-02-16 16:10:14 -08:00
2015-02-17 06:48:19 -08:00
data.publicKeyRing = _.pluck(wallet.copayers, 'xPubKey')
self.storage.save(data, function(err) {
return cb(err, data);
});
});
};
2015-02-16 16:10:14 -08:00
2015-02-18 12:14:56 -08:00
API.prototype._load = function(cb) {
2015-02-16 21:20:04 -08:00
var self = this;
2015-02-16 15:23:25 -08:00
this.storage.load(function(err, data) {
if (err || !data) {
return cb(err || 'Wallet file not found.');
}
2015-02-18 12:14:56 -08:00
return cb(null, data);
});
};
API.prototype._loadAndCheck = function(cb) {
var self = this;
this._load(function(err, data) {
if (err) return cb(err);
2015-02-16 15:23:25 -08:00
if (data.n > 1) {
var pkrComplete = data.publicKeyRing && data.m && data.publicKeyRing.length === data.n;
2015-02-16 14:54:38 -08:00
2015-02-16 16:10:14 -08:00
if (!pkrComplete) {
return self._tryToComplete(data, cb);
2015-02-16 15:23:25 -08:00
}
2015-02-12 19:00:54 -08:00
}
2015-02-16 15:23:25 -08:00
return cb(null, data);
});
2015-02-12 13:54:17 -08:00
};
2015-02-15 06:12:04 -08:00
2015-02-15 08:03:48 -08:00
API.prototype._doRequest = function(method, url, args, data, cb) {
2015-02-17 10:13:42 -08:00
var reqSignature;
data = data || {};
if (data.signingPrivKey)
reqSignature = _signRequest(method, url, args, data.signingPrivKey);
2015-02-17 15:31:03 -08:00
var absUrl = this.baseUrl + url;
2015-02-15 08:14:36 -08:00
var args = {
2015-02-19 07:32:10 -08:00
// relUrl: only for testing with `supertest`
relUrl: this.basePath + url,
2015-02-13 20:28:43 -08:00
headers: {
'x-identity': data.copayerId,
'x-signature': reqSignature,
},
2015-02-15 06:12:04 -08:00
method: method,
2015-02-13 20:28:43 -08:00
url: absUrl,
body: args,
json: true,
2015-02-15 08:14:36 -08:00
};
2015-02-16 21:20:04 -08:00
log.verbose('Request Args', util.inspect(args, {
depth: 10
}));
2015-02-16 10:02:20 -08:00
this.request(args, function(err, res, body) {
2015-02-16 21:20:04 -08:00
log.verbose(util.inspect(body, {
depth: 10
}));
2015-02-13 20:28:43 -08:00
if (err) return cb(err);
2015-02-19 15:04:05 -08:00
2015-02-13 20:28:43 -08:00
if (res.statusCode != 200) {
2015-02-19 15:04:05 -08:00
return cb(_parseError(body));
2015-02-13 20:28:43 -08:00
}
2015-02-16 10:02:20 -08:00
2015-02-13 20:28:43 -08:00
return cb(null, body);
});
};
2015-02-15 06:33:04 -08:00
API.prototype._doPostRequest = function(url, args, data, cb) {
2015-02-13 20:28:43 -08:00
return this._doRequest('post', url, args, data, cb);
};
2015-02-15 06:33:04 -08:00
API.prototype._doGetRequest = function(url, data, cb) {
2015-02-13 20:28:43 -08:00
return this._doRequest('get', url, {}, data, cb);
};
2015-02-17 22:12:22 -08:00
API.prototype._initData = function(network, walletPrivKey, m, n) {
var xPrivKey = new Bitcore.HDPrivateKey(network);
var xPubKey = (new Bitcore.HDPublicKey(xPrivKey)).toString();
2015-02-18 13:14:24 -08:00
var signingPrivKey = (new Bitcore.HDPrivateKey(xPrivKey)).derive('m/1/0').privateKey;
2015-02-18 13:29:00 -08:00
var sharedEncryptingKey = Bitcore.crypto.Hash.sha256(walletPrivKey.toBuffer()).slice(0, 16).toString('base64');
2015-02-18 11:47:15 -08:00
var copayerId = WalletUtils.xPubToCopayerId(xPubKey);
2015-02-17 22:12:22 -08:00
var data = {
copayerId: copayerId,
xPrivKey: xPrivKey.toString(),
publicKeyRing: [xPubKey],
network: network,
m: m,
n: n,
2015-02-18 13:14:24 -08:00
signingPrivKey: signingPrivKey.toWIF(),
2015-02-17 22:12:22 -08:00
walletPrivKey: walletPrivKey.toWIF(),
2015-02-18 13:14:24 -08:00
sharedEncryptingKey: sharedEncryptingKey,
2015-02-17 22:12:22 -08:00
};
return data;
};
API.prototype._doJoinWallet = function(walletId, walletPrivKey, xPubKey, copayerName, cb) {
var args = {
walletId: walletId,
name: copayerName,
xPubKey: xPubKey,
xPubKeySignature: WalletUtils.signMessage(xPubKey, walletPrivKey),
};
var url = '/v1/wallets/' + walletId + '/copayers';
this._doPostRequest(url, args, {}, function(err, body) {
if (err) return cb(err);
return cb(null, body.wallet);
});
};
2015-02-15 06:33:04 -08:00
API.prototype.createWallet = function(walletName, copayerName, m, n, network, cb) {
2015-02-12 19:00:54 -08:00
var self = this;
2015-02-13 13:24:35 -08:00
network = network || 'livenet';
2015-02-13 17:59:05 -08:00
if (!_.contains(['testnet', 'livenet'], network))
return cb('Invalid network');
2015-02-12 19:00:54 -08:00
2015-02-16 15:23:25 -08:00
this.storage.load(function(err, data) {
if (data)
return cb('Storage already contains a wallet');
2015-02-17 10:13:42 -08:00
var walletPrivKey = new Bitcore.PrivateKey();
2015-02-16 15:23:25 -08:00
var args = {
name: walletName,
m: m,
n: n,
2015-02-17 10:13:42 -08:00
pubKey: walletPrivKey.toPublicKey().toString(),
2015-02-16 15:23:25 -08:00
network: network,
};
var url = '/v1/wallets/';
2015-02-17 10:13:42 -08:00
self._doPostRequest(url, args, {}, function(err, body) {
2015-02-16 15:23:25 -08:00
if (err) return cb(err);
2015-02-15 06:33:04 -08:00
2015-02-16 15:23:25 -08:00
var walletId = body.walletId;
2015-02-15 13:26:05 -08:00
2015-02-17 22:12:22 -08:00
var secret = WalletUtils.toSecret(walletId, walletPrivKey, network);
var data = self._initData(network, walletPrivKey, m, n);
self._doJoinWallet(walletId, walletPrivKey, data.publicKeyRing[0], copayerName,
function(err, wallet) {
if (err) return cb(err);
self.storage.save(data, function(err) {
return cb(err, n > 1 ? secret : null);
});
});
2015-02-12 11:42:32 -08:00
});
});
};
2015-02-17 22:34:11 -08:00
API.prototype.reCreateWallet = function(walletName, cb) {
var self = this;
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
var walletPrivKey = new Bitcore.PrivateKey();
var args = {
name: walletName,
m: data.m,
n: data.n,
pubKey: walletPrivKey.toPublicKey().toString(),
network: data.network,
};
var url = '/v1/wallets/';
self._doPostRequest(url, args, {}, function(err, body) {
if (err) return cb(err);
var walletId = body.walletId;
var secret = WalletUtils.toSecret(walletId, walletPrivKey, data.network);
var i = 0;
async.each(data.publicKeyRing, function(xpub, next) {
var copayerName = 'recovered Copayer #' + i;
self._doJoinWallet(walletId, walletPrivKey, data.publicKeyRing[i++], copayerName, next);
}, function(err) {
return cb(err);
});
});
});
};
2015-02-15 06:33:04 -08:00
API.prototype.joinWallet = function(secret, copayerName, cb) {
2015-02-12 19:00:54 -08:00
var self = this;
2015-02-12 13:54:17 -08:00
2015-02-16 15:23:25 -08:00
this.storage.load(function(err, data) {
if (data)
return cb('Storage already contains a wallet');
2015-02-12 13:54:17 -08:00
2015-02-17 22:12:22 -08:00
var secretData = WalletUtils.fromSecret(secret);
var data = self._initData(secretData.network, secretData.walletPrivKey);
self._doJoinWallet(secretData.walletId, secretData.walletPrivKey, data.publicKeyRing[0], copayerName,
function(err, wallet) {
if (err) return cb(err);
data.m = wallet.m;
data.n = wallet.n;
self.storage.save(data, cb);
});
2015-02-16 15:23:25 -08:00
});
2015-02-12 11:42:32 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.getStatus = function(cb) {
2015-02-12 19:00:54 -08:00
var self = this;
2015-02-18 12:14:56 -08:00
this._load(function(err, data) {
2015-02-12 11:42:32 -08:00
if (err) return cb(err);
2015-02-16 15:23:25 -08:00
var url = '/v1/wallets/';
2015-02-20 10:11:30 -08:00
self._doGetRequest(url, data, function(err, result) {
_processTxps(result.pendingTxps, data.sharedEncryptingKey);
return cb(err, result, data.copayerId);
2015-02-16 15:23:25 -08:00
});
2015-02-12 11:42:32 -08:00
});
};
2015-02-13 07:45:05 -08:00
/**
* send
*
* @param opts
* @param opts.toAddress
* @param opts.amount
* @param opts.message
2015-02-13 07:45:05 -08:00
*/
API.prototype.sendTxProposal = function(opts, cb) {
2015-02-19 16:37:13 -08:00
$.checkArgument(opts);
$.shouldBeNumber(opts.amount);
2015-02-13 06:38:25 -08:00
var self = this;
2015-02-16 21:20:04 -08:00
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
2015-02-13 07:55:07 -08:00
var args = {
toAddress: opts.toAddress,
2015-02-19 16:37:13 -08:00
amount: opts.amount,
2015-02-20 10:11:30 -08:00
message: _encryptMessage(opts.message, data.sharedEncryptingKey),
};
var hash = WalletUtils.getProposalHash(args.toAddress, args.amount, args.message);
args.proposalSignature = WalletUtils.signMessage(hash, data.signingPrivKey);
2015-02-19 11:21:50 -08:00
log.debug('Generating & signing tx proposal hash -> Hash: ', hash, ' Signature: ', args.proposalSignature);
2015-02-13 07:55:07 -08:00
2015-02-16 21:20:04 -08:00
var url = '/v1/txproposals/';
self._doPostRequest(url, args, data, cb);
});
2015-02-13 07:55:07 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.createAddress = function(cb) {
2015-02-13 07:55:07 -08:00
var self = this;
2015-02-16 21:20:04 -08:00
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
2015-02-16 15:23:25 -08:00
2015-02-16 21:20:04 -08:00
var url = '/v1/addresses/';
self._doPostRequest(url, {}, data, function(err, address) {
if (err) return cb(err);
if (!Verifier.checkAddress(data, address)) {
return cb(new ServerCompromisedError('Server sent fake address'));
}
2015-02-16 16:10:14 -08:00
2015-02-16 21:20:04 -08:00
return cb(null, address);
2015-02-16 15:23:25 -08:00
});
2015-02-16 21:20:04 -08:00
});
2015-02-12 11:42:32 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.history = function(limit, cb) {
2015-02-12 11:42:32 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.getBalance = function(cb) {
2015-02-13 08:35:20 -08:00
var self = this;
2015-02-16 16:10:14 -08:00
this._loadAndCheck(function(err, data) {
2015-02-16 15:23:25 -08:00
if (err) return cb(err);
var url = '/v1/balance/';
self._doGetRequest(url, data, cb);
});
2015-02-13 08:35:20 -08:00
};
2015-02-17 08:59:24 -08:00
API.prototype.export = function(cb) {
2015-02-13 08:35:20 -08:00
var self = this;
2015-02-17 09:55:57 -08:00
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
2015-02-17 10:48:28 -08:00
var v = [];
2015-02-17 12:51:35 -08:00
var myXPubKey = (new Bitcore.HDPublicKey(data.xPrivKey)).toString();
2015-02-17 10:48:28 -08:00
_.each(WALLET_CRITICAL_DATA, function(k) {
2015-02-17 12:51:35 -08:00
var d;
if (k === 'publicKeyRing') {
d = _.without(data[k], myXPubKey);
} else {
d = data[k];
}
v.push(d);
2015-02-17 10:48:28 -08:00
});
return cb(null, JSON.stringify(v));
2015-02-17 10:13:42 -08:00
});
}
API.prototype.import = function(str, cb) {
var self = this;
2015-02-17 10:48:28 -08:00
this.storage.load(function(err, data) {
if (data)
return cb('Storage already contains a wallet');
2015-02-17 10:13:42 -08:00
2015-02-17 10:48:28 -08:00
data = {};
2015-02-17 10:13:42 -08:00
2015-02-17 10:48:28 -08:00
var inData = JSON.parse(str);
var i = 0;
_.each(WALLET_CRITICAL_DATA, function(k) {
data[k] = inData[i++];
if (!data[k])
return cb('Invalid wallet data');
});
2015-02-17 12:51:35 -08:00
var xPubKey = (new Bitcore.HDPublicKey(data.xPrivKey)).toString();
2015-02-19 12:38:48 -08:00
data.publicKeyRing.unshift(xPubKey);
2015-02-17 15:26:58 -08:00
data.copayerId = WalletUtils.xPubToCopayerId(xPubKey);
2015-02-17 22:34:11 -08:00
data.n = data.publicKeyRing.length;
2015-02-17 10:48:28 -08:00
data.signingPrivKey = (new Bitcore.HDPrivateKey(data.xPrivKey)).derive('m/1/0').privateKey.toWIF();
2015-02-17 15:26:58 -08:00
data.network = data.xPrivKey.substr(0, 4) === 'tprv' ? 'testnet' : 'livenet';
2015-02-17 10:48:28 -08:00
self.storage.save(data, cb);
});
};
2015-02-17 08:59:24 -08:00
2015-02-20 07:25:21 -08:00
/**
2015-02-20 10:11:30 -08:00
*
2015-02-20 07:25:21 -08:00
* opts.doNotVerify
* @return {undefined}
*/
2015-02-17 10:13:42 -08:00
2015-02-17 08:59:24 -08:00
API.prototype.getTxProposals = function(opts, cb) {
var self = this;
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
var url = '/v1/txproposals/';
self._doGetRequest(url, data, function(err, txps) {
if (err) return cb(err);
2015-02-20 10:11:30 -08:00
_processTxps(txps, data.sharedEncryptingKey);
2015-02-20 06:52:01 -08:00
2015-02-20 10:11:30 -08:00
var fake = _.any(txps, function(txp) {
return (!opts.doNotVerify && !Verifier.checkTxProposal(data, txp));
});
2015-02-20 06:52:01 -08:00
if (fake)
return cb(new ServerCompromisedError('Server sent fake transaction proposal'));
return cb(null, txps);
});
2015-02-17 08:59:24 -08:00
});
2015-02-13 08:35:20 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.signTxProposal = function(txp, cb) {
2015-02-19 13:11:57 -08:00
$.checkArgument(txp.creatorId);
2015-02-13 12:02:56 -08:00
2015-02-19 16:37:13 -08:00
var self = this;
2015-02-17 08:59:24 -08:00
this._loadAndCheck(function(err, data) {
if (err) return cb(err);
2015-02-13 12:02:56 -08:00
2015-02-17 15:26:58 -08:00
if (!Verifier.checkTxProposal(data, txp)) {
return cb(new ServerCompromisedError('Server sent fake transaction proposal'));
}
2015-02-17 06:52:29 -08:00
2015-02-17 08:59:24 -08:00
//Derive proper key to sign, for each input
var privs = [],
derived = {};
2015-02-13 12:02:56 -08:00
2015-02-17 08:59:24 -08:00
var network = new Bitcore.Address(txp.toAddress).network.name;
var xpriv = new Bitcore.HDPrivateKey(data.xPrivKey, network);
2015-02-13 12:02:56 -08:00
2015-02-17 08:59:24 -08:00
_.each(txp.inputs, function(i) {
if (!derived[i.path]) {
derived[i.path] = xpriv.derive(i.path).privateKey;
}
privs.push(derived[i.path]);
});
2015-02-13 12:02:56 -08:00
2015-02-17 08:59:24 -08:00
var t = new Bitcore.Transaction();
_.each(txp.inputs, function(i) {
t.from(i, i.publicKeys, txp.requiredSignatures);
});
2015-02-13 12:02:56 -08:00
2015-02-17 08:59:24 -08:00
t.to(txp.toAddress, txp.amount)
2015-02-19 11:21:50 -08:00
.change(txp.changeAddress.address)
2015-02-17 08:59:24 -08:00
.sign(privs);
2015-02-13 13:24:35 -08:00
2015-02-17 08:59:24 -08:00
var signatures = [];
_.each(privs, function(p) {
var s = t.getSignatures(p)[0].signature.toDER().toString('hex');
signatures.push(s);
});
2015-02-13 13:53:49 -08:00
2015-02-17 08:59:24 -08:00
var url = '/v1/txproposals/' + txp.id + '/signatures/';
var args = {
signatures: signatures
};
2015-02-16 15:23:25 -08:00
2015-02-17 08:59:24 -08:00
self._doPostRequest(url, args, data, cb);
});
2015-02-13 12:02:56 -08:00
};
2015-02-15 06:33:04 -08:00
API.prototype.rejectTxProposal = function(txp, reason, cb) {
2015-02-19 13:11:57 -08:00
$.checkArgument(cb);
2015-02-13 17:51:40 -08:00
2015-02-19 16:37:13 -08:00
var self = this;
2015-02-16 16:10:14 -08:00
this._loadAndCheck(
function(err, data) {
if (err) return cb(err);
2015-02-16 15:23:25 -08:00
2015-02-16 16:10:14 -08:00
var url = '/v1/txproposals/' + txp.id + '/rejections/';
var args = {
2015-02-20 10:11:30 -08:00
reason: _encryptMessage(reason, data.sharedEncryptingKey) || '',
2015-02-16 16:10:14 -08:00
};
self._doPostRequest(url, args, data, cb);
});
2015-02-13 17:51:40 -08:00
};
2015-02-13 08:35:20 -08:00
2015-02-15 13:52:48 -08:00
API.prototype.broadcastTxProposal = function(txp, cb) {
var self = this;
2015-02-16 16:10:14 -08:00
this._loadAndCheck(
function(err, data) {
if (err) return cb(err);
2015-02-16 15:23:25 -08:00
2015-02-16 16:10:14 -08:00
var url = '/v1/txproposals/' + txp.id + '/broadcast/';
self._doPostRequest(url, {}, data, cb);
});
2015-02-15 13:52:48 -08:00
};
2015-02-15 08:03:48 -08:00
API.prototype.removeTxProposal = function(txp, cb) {
2015-02-14 07:54:00 -08:00
var self = this;
2015-02-16 16:10:14 -08:00
this._loadAndCheck(
function(err, data) {
if (err) return cb(err);
var url = '/v1/txproposals/' + txp.id;
self._doRequest('delete', url, {}, data, cb);
});
2015-02-14 07:54:00 -08:00
};
2015-02-15 06:33:04 -08:00
module.exports = API;