You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
435 lines
12 KiB
435 lines
12 KiB
'use strict';
|
|
|
|
var _ = require('lodash');
|
|
var $ = require('preconditions').singleton();
|
|
var async = require('async');
|
|
var log = require('npmlog');
|
|
log.debug = log.verbose;
|
|
var Bitcore = require('bitcore');
|
|
var PublicKey = Bitcore.PublicKey;
|
|
var Explorers = require('bitcore-explorers');
|
|
|
|
var Lock = require('./lock');
|
|
var Storage = require('./storage');
|
|
|
|
var Wallet = require('./model/wallet');
|
|
var Copayer = require('./model/copayer');
|
|
var Address = require('./model/address');
|
|
var TxProposal = require('./model/txproposal');
|
|
|
|
/**
|
|
* Creates an instance of the Copay server.
|
|
* @constructor
|
|
* @param {Object} opts
|
|
* @param {Storage} [opts.storage] - The storage provider.
|
|
*/
|
|
function CopayServer(opts) {
|
|
opts = opts || {};
|
|
this.storage = opts.storage || new Storage();
|
|
};
|
|
|
|
|
|
/**
|
|
* Creates a new wallet.
|
|
* @param {Object} opts
|
|
* @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.
|
|
*/
|
|
CopayServer.prototype.createWallet = function (opts, cb) {
|
|
var self = this, pubKey;
|
|
|
|
try {
|
|
pubKey = new PublicKey.fromString(opts.pubKey);
|
|
} catch (e) {
|
|
return cb(e.toString());
|
|
};
|
|
|
|
self.storage.fetchWallet(opts.id, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
if (wallet) return cb('Wallet already exists');
|
|
|
|
var wallet = new Wallet({
|
|
id: opts.id,
|
|
name: opts.name,
|
|
m: opts.m,
|
|
n: opts.n,
|
|
network: opts.network || 'livenet',
|
|
pubKey: pubKey,
|
|
});
|
|
|
|
self.storage.storeWallet(wallet, cb);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Retrieves a wallet from storage.
|
|
* @param {Object} opts
|
|
* @param {string} opts.id - The wallet id.
|
|
* @returns {Object} wallet
|
|
*/
|
|
CopayServer.prototype.getWallet = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.storage.fetchWallet(opts.id, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
if (!wallet) return cb('Wallet not found');
|
|
|
|
return cb(null, wallet);
|
|
});
|
|
};
|
|
|
|
|
|
CopayServer.prototype._runLocked = function (walletId, cb, task) {
|
|
var self = this;
|
|
|
|
Lock.get(walletId, function (lock) {
|
|
var _cb = function () {
|
|
cb.apply(null, arguments);
|
|
lock.free();
|
|
};
|
|
task(_cb);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Joins a wallet in creation.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.id - The copayer id.
|
|
* @param {string} opts.name - The copayer name.
|
|
* @param {number} opts.xPubKey - Extended Public Key for this copayer.
|
|
* @param {number} opts.xPubKeySignature - Signature of xPubKey using the wallet pubKey.
|
|
*/
|
|
CopayServer.prototype.joinWallet = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self._runLocked(opts.walletId, cb, function (cb) {
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
if (_.find(wallet.copayers, { xPubKey: opts.xPubKey })) return cb('Copayer already in wallet');
|
|
if (wallet.copayers.length == wallet.n) return cb('Wallet full');
|
|
// TODO: validate copayer's extended public key using the public key from this wallet
|
|
// Note: use Bitcore.crypto.ecdsa .verify()
|
|
|
|
var copayer = new Copayer({
|
|
id: opts.id,
|
|
name: opts.name,
|
|
xPubKey: opts.xPubKey,
|
|
xPubKeySignature: opts.xPubKeySignature,
|
|
});
|
|
|
|
wallet.addCopayer(copayer);
|
|
|
|
self.storage.storeWallet(wallet, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb();
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._doCreateAddress = function (pkr, index, isChange) {
|
|
throw 'not implemented';
|
|
};
|
|
|
|
/**
|
|
* Creates a new address.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {truthy} opts.isChange - Indicates whether this is a regular address or a change address.
|
|
* @returns {Address} address
|
|
*/
|
|
CopayServer.prototype.createAddress = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self._runLocked(opts.walletId, cb, function (cb) {
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
|
|
var index = wallet.addressIndex++;
|
|
self.storage.storeWallet(wallet, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
var address = self._doCreateAddress(wallet.publicKeyRing, index, opts.isChange);
|
|
self.storage.storeAddress(opts.walletId, address, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb(null, address);
|
|
});
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Verifies that a given message was actually sent by an authorized copayer.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.copayerId - The wallet id.
|
|
* @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.
|
|
*/
|
|
CopayServer.prototype.verifyMessageSignature = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
|
|
var copayer = wallet.getCopayer(opts.copayerId);
|
|
if (!copayer) return cb('Copayer not found');
|
|
|
|
var isValid = self._doVerifyMessageSignature(copayer.xPubKey, opts.message, opts.signature);
|
|
return cb(null, isValid);
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._doVerifyMessageSignature = function (pubKey, message, signature) {
|
|
throw 'not implemented';
|
|
};
|
|
|
|
CopayServer.prototype._getBlockExplorer = function (provider, network) {
|
|
var url;
|
|
|
|
switch (provider) {
|
|
default:
|
|
case 'insight':
|
|
switch (network) {
|
|
default:
|
|
case 'livenet':
|
|
url = 'https://insight.bitpay.com:443';
|
|
break;
|
|
case 'testnet':
|
|
url = 'https://test-insight.bitpay.com:443'
|
|
break;
|
|
}
|
|
return new Explorers.Insight(url, network);
|
|
break;
|
|
}
|
|
};
|
|
|
|
CopayServer.prototype._getUtxos = function (opts, cb) {
|
|
var self = this;
|
|
|
|
// Get addresses for this wallet
|
|
self.storage.fetchAddresses(opts.walletId, function (err, addresses) {
|
|
if (err) return cb(err);
|
|
if (addresses.length == 0) return cb('The wallet has no addresses');
|
|
|
|
var addresses = _.pluck(addresses, 'address');
|
|
|
|
var bc = self._getBlockExplorer('insight', opts.network);
|
|
bc.getUnspentUtxos(addresses, function (err, utxos) {
|
|
if (err) return cb(err);
|
|
|
|
self.getPendingTxs({ walletId: opts.walletId }, function (err, txps) {
|
|
if (err) return cb(err);
|
|
|
|
var inputs = _.chain(txps)
|
|
.pluck('inputs')
|
|
.flatten()
|
|
.map(function (utxo) { return utxo.txid + '|' + utxo.vout });
|
|
|
|
var dictionary = _.groupBy(utxos, function (utxo) {
|
|
return utxo.txid + '|' + utxo.vout;
|
|
});
|
|
|
|
_.each(inputs, function (input) {
|
|
if (dictionary[input]) {
|
|
dictionary[input].locked = true;
|
|
}
|
|
});
|
|
|
|
return cb(null, utxos);
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
|
|
/**
|
|
* Creates a new transaction proposal.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @returns {Object} balance - Total amount & locked amount.
|
|
*/
|
|
CopayServer.prototype.getBalance = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self._getUtxos({ walletId: opts.walletId }, function (err, utxos) {
|
|
if (err) return cb(err);
|
|
|
|
var balance = {};
|
|
balance.totalAmount = _.reduce(utxos, function(sum, utxo) { return sum + utxo.amount; }, 0);
|
|
balance.lockedAmount = _.reduce(_.without(utxos, { locked: true }), function(sum, utxo) { return sum + utxo.amount; }, 0);
|
|
|
|
return cb(null, balance);
|
|
});
|
|
};
|
|
|
|
|
|
CopayServer.prototype._createRawTx = function (txp) {
|
|
var rawTx = new Bitcore.Transaction()
|
|
.from(tx.inputs)
|
|
.to(txp.toAddress, txp.amount)
|
|
.change(txp.changeAddress);
|
|
|
|
return rawTx;
|
|
};
|
|
|
|
CopayServer.prototype._selectUtxos = function (txp, utxos) {
|
|
var i = 0;
|
|
var total = 0;
|
|
var selected = [];
|
|
var inputs = _.sortBy(utxos, 'amount');
|
|
while (i < inputs.length) {
|
|
selected.push(inputs[i]);
|
|
total += inputs[i].amount;
|
|
if (total >= txp.amount) {
|
|
break;
|
|
}
|
|
i++;
|
|
};
|
|
return selected;
|
|
};
|
|
|
|
|
|
/**
|
|
* Creates a new transaction proposal.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.copayerId - The wallet id.
|
|
* @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.
|
|
* @returns {TxProposal} Transaction proposal.
|
|
*/
|
|
CopayServer.prototype.createTx = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
|
|
self._getUtxos({ walletId: wallet.id }, function (err, utxos) {
|
|
if (err) return cb(err);
|
|
|
|
utxos = _.without(utxos, { locked: true });
|
|
|
|
var txp = new TxProposal({
|
|
creatorId: opts.copayerId,
|
|
toAddress: opts.toAddress,
|
|
amount: opts.amount,
|
|
inputs: self._selectUtxos(opts.amount, utxos),
|
|
changeAddress: opts.changeAddress,
|
|
requiredSignatures: wallet.m,
|
|
maxRejections: wallet.n - wallet.m,
|
|
});
|
|
|
|
txp.rawTx = self._createRawTx(txp);
|
|
|
|
self.storage.storeTx(wallet.id, txp, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb(null, txp);
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._broadcastTx = function (rawTx, cb) {
|
|
// TODO: this should attempt to broadcast _all_ accepted and not-yet broadcasted (status=='accepted') txps?
|
|
cb = cb || function () {};
|
|
|
|
throw 'not implemented';
|
|
};
|
|
|
|
/**
|
|
* Sign a transaction proposal.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.copayerId - The wallet id.
|
|
* @param {string} opts.txProposalId - The identifier of the transaction.
|
|
* @param {string} opts.signature - The signature of the tx for this copayer.
|
|
*/
|
|
CopayServer.prototype.signTx = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.fetchTx(opts.walletId, opts.txProposalId, function (err, txp) {
|
|
if (err) return cb(err);
|
|
if (!txp) return cb('Transaction proposal not found');
|
|
var action = _.find(txp.actions, { copayerId: opts.copayerId });
|
|
if (action) return cb('Copayer already voted on this transaction proposal');
|
|
if (txp.status != 'pending') return cb('The transaction proposal is not pending');
|
|
|
|
txp.sign(opts.copayerId, opts.signature);
|
|
|
|
self.storage.storeTx(opts.walletId, txp, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
if (txp.status == 'accepted');
|
|
self._broadcastTx(txp.rawTx, function (err, txid) {
|
|
if (err) return cb(err);
|
|
|
|
tx.setBroadcasted(txid);
|
|
self.storage.storeTx(opts.walletId, txp, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Reject a transaction proposal.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.copayerId - The wallet id.
|
|
* @param {string} opts.txProposalId - The identifier of the transaction.
|
|
*/
|
|
CopayServer.prototype.rejectTx = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.fetchTx(opts.walletId, opts.txProposalId, function (err, txp) {
|
|
if (err) return cb(err);
|
|
if (!txp) return cb('Transaction proposal not found');
|
|
var action = _.find(txp.actions, { copayerId: opts.copayerId });
|
|
if (action) return cb('Copayer already voted on this transaction proposal');
|
|
if (txp.status != 'pending') return cb('The transaction proposal is not pending');
|
|
|
|
txp.reject(opts.copayerId);
|
|
|
|
self.storage.storeTx(opts.walletId, txp, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb();
|
|
});
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Retrieves all pending transaction proposals.
|
|
* @param {Object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @returns {TxProposal[]} Transaction proposal.
|
|
*/
|
|
CopayServer.prototype.getPendingTxs = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.storage.fetchTxs(opts.walletId, function (err, txps) {
|
|
if (err) return cb(err);
|
|
|
|
var pending = _.filter(txps, { status: 'pending' });
|
|
|
|
return cb(null, pending);
|
|
});
|
|
};
|
|
|
|
|
|
module.exports = CopayServer;
|
|
|