|
@ -15,7 +15,7 @@ var ServerCompromisedError = require('./servercompromisederror') |
|
|
|
|
|
|
|
|
var BASE_URL = 'http://localhost:3001/copay/api'; |
|
|
var BASE_URL = 'http://localhost:3001/copay/api'; |
|
|
|
|
|
|
|
|
var WALLET_CRITICAL_DATA = ['xPrivKey', 'm', 'publicKeyRing']; |
|
|
var WALLET_CRITICAL_DATA = ['xPrivKey', 'm', 'publicKeyRing', 'sharedEncryptingKey']; |
|
|
|
|
|
|
|
|
function _encryptProposalMessage(message, encryptingKey) { |
|
|
function _encryptProposalMessage(message, encryptingKey) { |
|
|
if (!message) return null; |
|
|
if (!message) return null; |
|
@ -24,7 +24,11 @@ function _encryptProposalMessage(message, encryptingKey) { |
|
|
|
|
|
|
|
|
function _decryptProposalMessage(message, encryptingKey) { |
|
|
function _decryptProposalMessage(message, encryptingKey) { |
|
|
if (!message) return ''; |
|
|
if (!message) return ''; |
|
|
return WalletUtils.decryptMessage(message, encryptingKey); |
|
|
try { |
|
|
|
|
|
return WalletUtils.decryptMessage(message, encryptingKey); |
|
|
|
|
|
} catch (ex) { |
|
|
|
|
|
return '<ECANNOTDECRYPT>'; |
|
|
|
|
|
} |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
function _parseError(body) { |
|
|
function _parseError(body) { |
|
@ -161,8 +165,9 @@ API.prototype._doGetRequest = function(url, data, cb) { |
|
|
|
|
|
|
|
|
API.prototype._initData = function(network, walletPrivKey, m, n) { |
|
|
API.prototype._initData = function(network, walletPrivKey, m, n) { |
|
|
var xPrivKey = new Bitcore.HDPrivateKey(network); |
|
|
var xPrivKey = new Bitcore.HDPrivateKey(network); |
|
|
var signingPrivKey = (new Bitcore.HDPrivateKey(xPrivKey)).derive('m/1/0').privateKey.toWIF(); |
|
|
|
|
|
var xPubKey = (new Bitcore.HDPublicKey(xPrivKey)).toString(); |
|
|
var xPubKey = (new Bitcore.HDPublicKey(xPrivKey)).toString(); |
|
|
|
|
|
var signingPrivKey = (new Bitcore.HDPrivateKey(xPrivKey)).derive('m/1/0').privateKey; |
|
|
|
|
|
var sharedEncryptingKey = Bitcore.crypto.Hash.sha256(signingPrivKey.toBuffer()).slice(0, 16).toString('base64'); |
|
|
var copayerId = WalletUtils.xPubToCopayerId(xPubKey); |
|
|
var copayerId = WalletUtils.xPubToCopayerId(xPubKey); |
|
|
|
|
|
|
|
|
var data = { |
|
|
var data = { |
|
@ -172,8 +177,9 @@ API.prototype._initData = function(network, walletPrivKey, m, n) { |
|
|
network: network, |
|
|
network: network, |
|
|
m: m, |
|
|
m: m, |
|
|
n: n, |
|
|
n: n, |
|
|
signingPrivKey: signingPrivKey, |
|
|
signingPrivKey: signingPrivKey.toWIF(), |
|
|
walletPrivKey: walletPrivKey.toWIF(), |
|
|
walletPrivKey: walletPrivKey.toWIF(), |
|
|
|
|
|
sharedEncryptingKey: sharedEncryptingKey, |
|
|
}; |
|
|
}; |
|
|
return data; |
|
|
return data; |
|
|
}; |
|
|
}; |
|
@ -291,8 +297,8 @@ API.prototype.getStatus = function(cb) { |
|
|
var url = '/v1/wallets/'; |
|
|
var url = '/v1/wallets/'; |
|
|
self._doGetRequest(url, data, function(err, body) { |
|
|
self._doGetRequest(url, data, function(err, body) { |
|
|
_.each(body.pendingTxps, function(txp) { |
|
|
_.each(body.pendingTxps, function(txp) { |
|
|
txp.message = _decryptProposalMessage(txp.message, data.signingPrivKey); |
|
|
txp.message = _decryptProposalMessage(txp.message, data.sharedEncryptingKey); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
return cb(err, body, data.copayerId); |
|
|
return cb(err, body, data.copayerId); |
|
|
}); |
|
|
}); |
|
@ -323,7 +329,7 @@ API.prototype.sendTxProposal = function(opts, cb) { |
|
|
var args = { |
|
|
var args = { |
|
|
toAddress: opts.toAddress, |
|
|
toAddress: opts.toAddress, |
|
|
amount: amount, |
|
|
amount: amount, |
|
|
message: _encryptProposalMessage(opts.message, data.signingPrivKey), |
|
|
message: _encryptProposalMessage(opts.message, data.sharedEncryptingKey), |
|
|
}; |
|
|
}; |
|
|
var hash = WalletUtils.getProposalHash(args.toAddress, args.amount, args.message); |
|
|
var hash = WalletUtils.getProposalHash(args.toAddress, args.amount, args.message); |
|
|
args.proposalSignature = WalletUtils.signMessage(hash, data.signingPrivKey); |
|
|
args.proposalSignature = WalletUtils.signMessage(hash, data.signingPrivKey); |
|
@ -428,7 +434,7 @@ API.prototype.getTxProposals = function(opts, cb) { |
|
|
if (err) return cb(err); |
|
|
if (err) return cb(err); |
|
|
|
|
|
|
|
|
_.each(txps, function(txp) { |
|
|
_.each(txps, function(txp) { |
|
|
txp.message = _decryptProposalMessage(txp.message, data.signingPrivKey); |
|
|
txp.message = _decryptProposalMessage(txp.message, data.sharedEncryptingKey); |
|
|
}); |
|
|
}); |
|
|
return cb(null, txps); |
|
|
return cb(null, txps); |
|
|
}); |
|
|
}); |
|
|