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.
225 lines
5.0 KiB
225 lines
5.0 KiB
'use strict';
|
|
|
|
var _ = require('lodash');
|
|
var async = require('async');
|
|
var log = require('npmlog');
|
|
var request = require('request')
|
|
log.debug = log.verbose;
|
|
log.level = 'debug';
|
|
var fs = require('fs')
|
|
|
|
var Bitcore = require('bitcore')
|
|
var SignUtils = require('./signutils');
|
|
|
|
var BASE_URL = 'http://localhost:3001/copay/api/';
|
|
|
|
|
|
|
|
function _getUrl(path) {
|
|
return BASE_URL + path;
|
|
};
|
|
|
|
function _parseError(body) {
|
|
if (_.isString(body)) {
|
|
body = JSON.parse(body);
|
|
}
|
|
var code = body.code || 'ERROR';
|
|
var message = body.error || 'There was an unknown error processing the request';
|
|
log.error(code, message);
|
|
};
|
|
|
|
function _signRequest(url, args, privKey) {
|
|
var message = url + (args ? '|' + JSON.stringify(args) : '');
|
|
return SignUtils.sign(message, privKey);
|
|
};
|
|
|
|
function _createXPrivKey() {
|
|
return new Bitcore.HDPrivateKey().toString();
|
|
};
|
|
|
|
function CliLib(opts) {
|
|
if (!opts.filename) {
|
|
throw new Error('Please set the config filename');
|
|
}
|
|
this.filename = opts.filename;
|
|
};
|
|
|
|
|
|
CliLib.prototype_save = function (data) {
|
|
fs.writeFileSync(this.filename, JSON.stringify(data));
|
|
};
|
|
|
|
CliLib.prototype._load = function() {
|
|
try {
|
|
return JSON.parse(fs.readFileSync(this.filename));
|
|
} catch (ex) {}
|
|
};
|
|
|
|
CliLib.prototype._loadAndCheck = function() {
|
|
var data = this._load();
|
|
if (!data) {
|
|
log.error('Wallet file not found.');
|
|
process.exit(1);
|
|
}
|
|
if (data.n > 1) {
|
|
var pkrComplete = data.publicKeyRing && data.m && data.publicKeyRing.length === data.n;
|
|
if (!pkrComplete) {
|
|
log.warn('The file ' + this.filename + ' is incomplete. It will allow you to operate with the wallet but it should not be trusted as a backup. Please wait for all copayers to join the wallet and run the tool with -export flag.')
|
|
}
|
|
}
|
|
return data;
|
|
};
|
|
|
|
CliLib.prototype.createWallet = function(walletName, copayerName, m, n, network, cb) {
|
|
var self = this;
|
|
|
|
var data = this._load();
|
|
if (data) return cb('Only one wallet is supported in this version');
|
|
|
|
data = {
|
|
xPrivKey: _createXPrivKey(),
|
|
m: m,
|
|
};
|
|
|
|
var privKey = new Bitcore.PrivateKey();
|
|
var pubKey = privKey.toPublicKey();
|
|
|
|
var args = {
|
|
name: walletName,
|
|
m: m,
|
|
n: n,
|
|
pubKey: pubKey.toString(),
|
|
network: network || 'livenet',
|
|
};
|
|
|
|
request({
|
|
method: 'post',
|
|
url: _getUrl('v1/wallets'),
|
|
body: args,
|
|
json: true,
|
|
}, function(err, res, body) {
|
|
if (err) return cb(err);
|
|
if (res.statusCode != 200) {
|
|
_parseError(body);
|
|
return cb('Request error');
|
|
}
|
|
|
|
var walletId = body.walletId;
|
|
var secret = walletId + '|' + privKey.toString();
|
|
data.secret = secret;
|
|
|
|
self._save(data);
|
|
|
|
self._joinWallet(data, secret, copayerName, function(err) {
|
|
if (err) return cb(err);
|
|
|
|
return cb(null, data.secret);
|
|
});
|
|
});
|
|
};
|
|
|
|
CliLib.prototype._joinWallet = function(data, secret, copayerName, cb) {
|
|
var self = this;
|
|
|
|
var secretSplit = secret.split('|');
|
|
var walletId = secretSplit[0];
|
|
var privKey = Bitcore.PrivateKey.fromString(secretSplit[1]);
|
|
var pubKey = privKey.toPublicKey();
|
|
|
|
var xPubKey = new Bitcore.HDPublicKey(data.xPrivKey);
|
|
var xPubKeySignature = SignUtils.sign(xPubKey.toString(), privKey);
|
|
var signingPrivKey = xPubKey.derive('m/1/0');
|
|
|
|
var args = {
|
|
walletId: walletId,
|
|
name: copayerName,
|
|
xPubKey: xPubKey.toString(),
|
|
xPubKeySignature: xPubKeySignature,
|
|
};
|
|
|
|
request({
|
|
method: 'post',
|
|
url: _getUrl('v1/wallets/' + walletId + '/copayers'),
|
|
body: args,
|
|
json: true,
|
|
}, function(err, res, body) {
|
|
console.log('[clilib.js.123:err:]',err, body); //TODO
|
|
if (err) return cb(err);
|
|
if (res.statusCode != 200) {
|
|
_parseError(body);
|
|
return cb('Request error');
|
|
}
|
|
|
|
var wallet = body.wallet;
|
|
data.copayerId = body.copayerId;
|
|
data.signingPrivKey = signingPrivKey.toString();
|
|
data.m = wallet.m;
|
|
data.n = wallet.n;
|
|
data.publicKeyRing = wallet.publicKeyRing;
|
|
self._save(data);
|
|
|
|
return cb();
|
|
});
|
|
};
|
|
|
|
CliLib.prototype.joinWallet = function(secret, copayerName, cb) {
|
|
var self = this;
|
|
|
|
var data = this._load();
|
|
if (data) return cb('Only one wallet is supported in this version');
|
|
|
|
data = {
|
|
xPrivKey: _createXPrivKey(),
|
|
};
|
|
|
|
self._joinWallet(data, secret, copayerName, cb);
|
|
};
|
|
|
|
CliLib.prototype.status = function(cb) {
|
|
var self = this;
|
|
|
|
var data = this._loadAndCheck();
|
|
|
|
var url = 'v1/wallets/';
|
|
var signature = _signRequest(url, null, data.signingPrivKey);
|
|
|
|
request({
|
|
headers: {
|
|
'x-identity': data.copayerId,
|
|
'x-signature': signature,
|
|
},
|
|
method: 'get',
|
|
url: _getUrl(url),
|
|
json: true,
|
|
}, function(err, res, body) {
|
|
if (err) return cb(err);
|
|
if (res.statusCode != 200) {
|
|
_parseError(body);
|
|
return cb('Request error');
|
|
}
|
|
|
|
return cb(null, body);
|
|
});
|
|
};
|
|
|
|
CliLib.prototype.send = function(addressTo, amount, message, cb) {
|
|
|
|
};
|
|
|
|
CliLib.prototype.sign = function(proposalId, cb) {
|
|
|
|
};
|
|
|
|
CliLib.prototype.reject = function(proposalId, cb) {
|
|
|
|
};
|
|
|
|
CliLib.prototype.address = function(cb) {
|
|
|
|
};
|
|
|
|
CliLib.prototype.history = function(limit, cb) {
|
|
|
|
};
|
|
|
|
module.exports = CliLib;
|
|
|