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.
269 lines
7.2 KiB
269 lines
7.2 KiB
'use strict';
|
|
|
|
var _ = require('lodash');
|
|
var $ = require('preconditions').singleton();
|
|
var async = require('async');
|
|
var log = require('npmlog');
|
|
log.debug = log.verbose;
|
|
|
|
var Lock = require('./lock');
|
|
var Storage = require('./storage');
|
|
var Wallet = require('./model/wallet');
|
|
var Copayer = require('./model/copayer');
|
|
|
|
/**
|
|
* Creates an instance of the Copay server.
|
|
* @constructor
|
|
*/
|
|
function CopayServer(opts) {
|
|
opts = opts || {};
|
|
this.storage = new Storage(opts);
|
|
};
|
|
|
|
|
|
/**
|
|
* 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;
|
|
|
|
self.getWallet({ id: 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: opts.pubKey,
|
|
});
|
|
|
|
self.storage.storeWallet(wallet, cb);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Retrieves a wallet from storage.
|
|
* @param {object} opts
|
|
* @param {string} opts.id - The wallet id.
|
|
* @param {truthy} opts.includeCopayers - Fetch wallet along with list of copayers.
|
|
* @returns {Object} wallet
|
|
*/
|
|
CopayServer.prototype.getWallet = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.storage.fetchWallet(opts.id, function (err, wallet) {
|
|
if (err || !wallet) return cb(err);
|
|
if (opts.includeCopayers) {
|
|
self.storage.fetchCopayers(wallet.id, function (err, copayers) {
|
|
if (err) return cb(err);
|
|
wallet.copayers = copayers || [];
|
|
return cb(null, wallet);
|
|
});
|
|
} else {
|
|
return cb(null, wallet);
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* 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;
|
|
|
|
Lock.get(opts.walletId, function (lock) {
|
|
var _cb = function (err, res) {
|
|
cb(err, res);
|
|
lock.free();
|
|
};
|
|
|
|
self.getWallet({ id: opts.walletId, includeCopayers: true }, function (err, wallet) {
|
|
if (err) return _cb(err);
|
|
if (!wallet) return _cb('Wallet not found');
|
|
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({
|
|
walletId: wallet.id,
|
|
id: opts.id,
|
|
name: opts.name,
|
|
xPubKey: opts.xPubKey,
|
|
xPubKeySignature: opts.xPubKeySignature,
|
|
});
|
|
|
|
self.storage.storeCopayer(copayer, function (err) {
|
|
if (err) return _cb(err);
|
|
if ((wallet.copayers.length + 1) < wallet.n) return _cb();
|
|
|
|
wallet.status = 'complete';
|
|
wallet.publicKeyRing = _.pluck(wallet.copayers, 'xPubKey');
|
|
wallet.publicKeyRing.push(copayer.xPubKey);
|
|
self.storage.storeWallet(wallet, _cb);
|
|
});
|
|
});
|
|
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._doCreateAddress = function (pkr, isChange) {
|
|
throw 'not implemented';
|
|
};
|
|
|
|
// opts = {
|
|
// walletId,
|
|
// isChange,
|
|
// };
|
|
CopayServer.prototype.createAddress = function (opts, cb) {
|
|
var self = this;
|
|
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
if (!wallet) return cb('Wallet not found');
|
|
|
|
var address = self._doCreateAddress(wallet.publicKeyRing, opts.isChange);
|
|
return cb(null, address);
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._verifyMessageSignature = function (copayerId, 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 Bitcore.Insight(url, network);
|
|
break;
|
|
}
|
|
};
|
|
|
|
CopayServer.prototype._getUtxos = function (opts, cb) {
|
|
var self = this;
|
|
|
|
// Get addresses for this wallet
|
|
self.storage.getAddresses(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 = _getBlockExplorer('insight', opts.network);
|
|
bc.getUnspentUtxos(addresses, function (err, utxos) {
|
|
if (err) return cb(err);
|
|
|
|
// TODO: filter 'locked' utxos
|
|
|
|
return cb(null, utxos);
|
|
});
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype._doCreateTx = function (opts, cb) {
|
|
var tx = new Bitcore.Transaction()
|
|
.from(opts.utxos)
|
|
.to(opts.toAddress, opts.amount)
|
|
.change(opts.changeAddress);
|
|
|
|
return tx;
|
|
};
|
|
|
|
// requestSignature, // S(toAddress + amount + otToken) using this copayers privKey
|
|
// // using this signature, the server can
|
|
// };
|
|
|
|
// result = {
|
|
// ntxid,
|
|
// rawTx,
|
|
// };
|
|
/**
|
|
* Creates a new transaction proposal.
|
|
* @param {object} opts
|
|
* @param {string} opts.walletId - The wallet id.
|
|
* @param {string} opts.copayerId - The wallet id.
|
|
* @param {truthy} opts.otToken - A one-time token used to avoid reply attacks.
|
|
* @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.
|
|
* @param {string} opts.requestSignature - Signature of the request (toAddress + amount + otToken).
|
|
* @returns {Object} result
|
|
* @returns {Object} result.ntxid - Id of the transaction proposal.
|
|
* @returns {Object} result.rawTx - Raw transaction.
|
|
*/
|
|
CopayServer.prototype.createTx = function (opts, cb) {
|
|
// Client generates a unique token and signs toAddress + amount + token.
|
|
// This way we authenticate + avoid replay attacks.
|
|
var self = this;
|
|
|
|
self.getWallet({ id: opts.walletId }, function (err, wallet) {
|
|
if (err) return cb(err);
|
|
if (!wallet) return cb('Wallet not found');
|
|
|
|
var msg = '' + opts.toAddress + opts.amount + opts.otToken;
|
|
if (!self._verifyMessageSignature(opts.copayerId, msg, opts.requestSignature)) return cb('Invalid request');
|
|
|
|
|
|
var txArgs = {
|
|
toAddress: opts.toAddress,
|
|
amount: opts.amount,
|
|
changeAddress: opts.changeAddress,
|
|
};
|
|
|
|
self._getUtxos({ walletId: wallet.id }, function (err, utxos) {
|
|
if (err) return cb('Could not retrieve UTXOs');
|
|
txArgs.utxos = utxos;
|
|
self._doCreateTx(txArgs, function (err, tx) {
|
|
if (err) return cb('Could not create transaction');
|
|
|
|
self.storage.storeTx(tx, function (err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb(null, {
|
|
ntxid: tx.ntxid,
|
|
rawTx: tx.raw,
|
|
});
|
|
});
|
|
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
CopayServer.prototype.getPendingTxs = function (opts, cb) {
|
|
var self = this;
|
|
|
|
//self.storage.get
|
|
};
|
|
|
|
|
|
module.exports = CopayServer;
|
|
|