Merge pull request #3 from isocolsky/REST_auth
REST + command line client
This commit is contained in:
commit
6ef5283688
|
@ -30,3 +30,5 @@ node_modules
|
||||||
*.swp
|
*.swp
|
||||||
out/
|
out/
|
||||||
db/
|
db/
|
||||||
|
|
||||||
|
.bit
|
76
app.js
76
app.js
|
@ -3,14 +3,15 @@
|
||||||
var _ = require('lodash');
|
var _ = require('lodash');
|
||||||
var async = require('async');
|
var async = require('async');
|
||||||
var log = require('npmlog');
|
var log = require('npmlog');
|
||||||
var CopayServer = require('./lib/server');
|
|
||||||
var express = require('express');
|
var express = require('express');
|
||||||
var querystring = require('querystring');
|
var querystring = require('querystring');
|
||||||
|
var bodyParser = require('body-parser')
|
||||||
|
|
||||||
|
var CopayServer = require('./lib/server');
|
||||||
|
|
||||||
log.debug = log.verbose;
|
log.debug = log.verbose;
|
||||||
log.level = 'debug';
|
log.level = 'debug';
|
||||||
|
|
||||||
var POST_LIMIT = 1024 * 100 /* Max POST 100 kb */ ;
|
|
||||||
|
|
||||||
CopayServer.initialize();
|
CopayServer.initialize();
|
||||||
|
|
||||||
|
@ -32,6 +33,12 @@ var allowCORS = function(req, res, next) {
|
||||||
}
|
}
|
||||||
app.use(allowCORS);
|
app.use(allowCORS);
|
||||||
|
|
||||||
|
var POST_LIMIT = 1024 * 100 /* Max POST 100 kb */ ;
|
||||||
|
|
||||||
|
app.use(bodyParser.json({
|
||||||
|
limit: POST_LIMIT
|
||||||
|
}));
|
||||||
|
|
||||||
var port = process.env.COPAY_PORT || 3001;
|
var port = process.env.COPAY_PORT || 3001;
|
||||||
var router = express.Router();
|
var router = express.Router();
|
||||||
|
|
||||||
|
@ -60,82 +67,45 @@ function getCredentials(req) {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
copayerId: identity,
|
copayerId: identity,
|
||||||
|
signature: req.header('x-signature'),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
function getServerWithAuth(req, res, cb) {
|
function getServerWithAuth(req, res, cb) {
|
||||||
var credentials = getCredentials(req);
|
var credentials = getCredentials(req);
|
||||||
|
var auth = {
|
||||||
CopayServer.getInstanceWithAuth({
|
|
||||||
copayerId: credentials.copayerId,
|
copayerId: credentials.copayerId,
|
||||||
message: 'hello world!',
|
message: req.url + '|' + JSON.stringify(req.body),
|
||||||
signature: '3045022100addd20e5413865d65d561ad2979f2289a40d52594b1f804840babd9a63e4ebbf02204b86285e1fcab02df772e7a1325fc4b511ecad79a8f80a2bd1ad8bfa858ac3d4',
|
signature: credentials.signature,
|
||||||
}, function(err, server) {
|
};
|
||||||
|
|
||||||
|
CopayServer.getInstanceWithAuth(auth, function(err, server) {
|
||||||
if (err) return returnError(err, res);
|
if (err) return returnError(err, res);
|
||||||
return cb(server);
|
return cb(server);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function authenticate() {
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
function parsePost(req, res, cb) {
|
|
||||||
var queryData = '';
|
|
||||||
req.on('data', function(data) {
|
|
||||||
queryData += data;
|
|
||||||
if (queryData.length > POST_LIMIT) {
|
|
||||||
queryData = '';
|
|
||||||
res.writeHead(413, {
|
|
||||||
'Content-Type': 'text/plain'
|
|
||||||
});
|
|
||||||
res.end();
|
|
||||||
req.connection.destroy();
|
|
||||||
}
|
|
||||||
}).on('end', function() {
|
|
||||||
try {
|
|
||||||
var params = JSON.parse(queryData);
|
|
||||||
cb(params);
|
|
||||||
} catch (ex) {
|
|
||||||
returnError({
|
|
||||||
code: 400,
|
|
||||||
message: 'Unable to parse request'
|
|
||||||
}, res);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
router.post('/v1/wallets/', function(req, res) {
|
router.post('/v1/wallets/', function(req, res) {
|
||||||
parsePost(req, res, function(params) {
|
|
||||||
var server = CopayServer.getInstance();
|
var server = CopayServer.getInstance();
|
||||||
server.createWallet(params, function(err, wallet) {
|
server.createWallet(req.body, function(err, wallet) {
|
||||||
if (err) returnError(err, res);
|
if (err) returnError(err, res);
|
||||||
|
|
||||||
res.json(wallet);
|
res.json(wallet);
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post('/v1/wallets/:id/join/', function(req, res) {
|
router.post('/v1/wallets/:id/copayers/', function(req, res) {
|
||||||
parsePost(req, res, function(params) {
|
req.body.walletId = req.params['id'];
|
||||||
params.walletId = req.params['id'];
|
|
||||||
var server = CopayServer.getInstance();
|
var server = CopayServer.getInstance();
|
||||||
server.joinWallet(params, function(err) {
|
server.joinWallet(req.body, function(err) {
|
||||||
if (err) returnError(err, res);
|
if (err) returnError(err, res);
|
||||||
|
|
||||||
res.end();
|
res.end();
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
router.get('/v1/wallets/', function(req, res) {
|
router.get('/v1/wallets/', function(req, res) {
|
||||||
var credentials = getCredentials(req);
|
getServerWithAuth(req, res, function(server) {
|
||||||
|
|
||||||
CopayServer.getInstanceWithAuth(getCredentials(req) {
|
|
||||||
copayerId: credentials.copayerId,
|
|
||||||
message: 'hello world!',
|
|
||||||
signature: '3045022100addd20e5413865d65d561ad2979f2289a40d52594b1f804840babd9a63e4ebbf02204b86285e1fcab02df772e7a1325fc4b511ecad79a8f80a2bd1ad8bfa858ac3d4',
|
|
||||||
}, function(err, server) {
|
|
||||||
if (err) return returnError(err, res);
|
if (err) return returnError(err, res);
|
||||||
server.getWallet({}, function(err, wallet) {
|
server.getWallet({}, function(err, wallet) {
|
||||||
if (err) returnError(err, res);
|
if (err) returnError(err, res);
|
||||||
|
@ -145,14 +115,12 @@ router.get('/v1/wallets/', function(req, res) {
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post('/v1/addresses/', function(req, res) {
|
router.post('/v1/addresses/', function(req, res) {
|
||||||
parsePost(req, res, function(params) {
|
|
||||||
getServerWithAuth(req, res, function(server) {
|
getServerWithAuth(req, res, function(server) {
|
||||||
server.createAddress(params, function(err, address) {
|
server.createAddress(req.body, function(err, address) {
|
||||||
if (err) returnError(err, res);
|
if (err) returnError(err, res);
|
||||||
res.json(address);
|
res.json(address);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
router.get('/v1/addresses/', function(req, res) {
|
router.get('/v1/addresses/', function(req, res) {
|
||||||
|
|
|
@ -0,0 +1,145 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
var _ = require('lodash');
|
||||||
|
var async = require('async');
|
||||||
|
var log = require('npmlog');
|
||||||
|
var request = require('request')
|
||||||
|
var commander = require('commander')
|
||||||
|
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/';
|
||||||
|
|
||||||
|
var cli = {};
|
||||||
|
|
||||||
|
|
||||||
|
function _getUrl(path) {
|
||||||
|
return BASE_URL + path;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function signRequest(url, args) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
function save(data) {
|
||||||
|
fs.writeFileSync('./.bit', JSON.stringify(data));
|
||||||
|
};
|
||||||
|
|
||||||
|
function load() {
|
||||||
|
try {
|
||||||
|
return JSON.parse(fs.readFileSync('./.bit'));
|
||||||
|
} catch (ex) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
clilib.createWallet = function(walletName, copayerName, m, n, cb) {
|
||||||
|
var data = load();
|
||||||
|
if (!data) {
|
||||||
|
data = {};
|
||||||
|
data.xPrivKey = new Bitcore.HDPrivateKey().toString();
|
||||||
|
data.m = m;
|
||||||
|
}
|
||||||
|
var privKey = new Bitcore.PrivateKey();
|
||||||
|
var pubKey = privKey.toPublicKey();
|
||||||
|
|
||||||
|
var args = {
|
||||||
|
name: walletName,
|
||||||
|
m: m,
|
||||||
|
n: n,
|
||||||
|
pubKey: pubKey.toString(),
|
||||||
|
};
|
||||||
|
|
||||||
|
request({
|
||||||
|
method: 'post',
|
||||||
|
url: _getUrl('v1/wallets'),
|
||||||
|
body: args,
|
||||||
|
json: true,
|
||||||
|
}, function(err, res, body) {
|
||||||
|
if (err) return cb(err);
|
||||||
|
var walletId = body;
|
||||||
|
var secret = walletId + '|' + privKey.toString();
|
||||||
|
|
||||||
|
joinWallet(secret, copayerName, function(err) {
|
||||||
|
if (err) return cb(err);
|
||||||
|
|
||||||
|
save(data);
|
||||||
|
return cb(null, secret);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.joinWallet = function(secret, copayerName, cb) {
|
||||||
|
var data = load();
|
||||||
|
if (!data) {
|
||||||
|
data = {};
|
||||||
|
data.xPrivKey = new Bitcore.HDPrivateKey().toString();
|
||||||
|
}
|
||||||
|
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).toString();
|
||||||
|
var xPubKeySignature = SignUtils.sign(xPubKey, privKey);
|
||||||
|
|
||||||
|
var args = {
|
||||||
|
walletId: walletId,
|
||||||
|
name: copayerName,
|
||||||
|
xPubKey: xPubKey,
|
||||||
|
xPubKeySignature: xPubKeySignature,
|
||||||
|
};
|
||||||
|
|
||||||
|
request({
|
||||||
|
method: 'post',
|
||||||
|
url: _getUrl('v1/wallets/' + walletId + '/copayers'),
|
||||||
|
body: args,
|
||||||
|
json: true,
|
||||||
|
}, function(err, res, body) {
|
||||||
|
if (err) return cb(err);
|
||||||
|
|
||||||
|
var copayerId = body;
|
||||||
|
data.copayerId = copayerId;
|
||||||
|
save(data);
|
||||||
|
return status(cb);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.status = function(cb) {
|
||||||
|
request({
|
||||||
|
method: 'get',
|
||||||
|
url: _getUrl('v1/dump/'),
|
||||||
|
}, function(err, res, body) {
|
||||||
|
if (err) return cb(err);
|
||||||
|
|
||||||
|
console.log(body);
|
||||||
|
return cb();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.send = function(addressTo, amount, message, cb) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.sign = function(proposalId, cb) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.reject = function(proposalId, cb) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.address = function(cb) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
clilib.history = function(limit, cb) {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = clilib;
|
|
@ -124,6 +124,7 @@ CopayServer.prototype.createWallet = function(opts, cb) {
|
||||||
});
|
});
|
||||||
|
|
||||||
self.storage.storeWallet(wallet, function(err) {
|
self.storage.storeWallet(wallet, function(err) {
|
||||||
|
log.debug('Wallet created', wallet.id);
|
||||||
return cb(err, wallet.id);
|
return cb(err, wallet.id);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -164,6 +165,8 @@ CopayServer.prototype._verifySignature = function(text, signature, pubKey) {
|
||||||
CopayServer.prototype._notify = function(type, data) {
|
CopayServer.prototype._notify = function(type, data) {
|
||||||
var self = this;
|
var self = this;
|
||||||
|
|
||||||
|
log.debug('Notification', type, data);
|
||||||
|
|
||||||
var walletId = self.walletId || data.walletId;
|
var walletId = self.walletId || data.walletId;
|
||||||
$.checkState(walletId);
|
$.checkState(walletId);
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
"async": "^0.9.0",
|
"async": "^0.9.0",
|
||||||
"bitcore": "0.10.0",
|
"bitcore": "0.10.0",
|
||||||
"bitcore-explorers": "^0.9.1",
|
"bitcore-explorers": "^0.9.1",
|
||||||
|
"body-parser": "^1.11.0",
|
||||||
"commander": "^2.6.0",
|
"commander": "^2.6.0",
|
||||||
"express": "^4.10.0",
|
"express": "^4.10.0",
|
||||||
"inherits": "^2.0.1",
|
"inherits": "^2.0.1",
|
||||||
|
|
Loading…
Reference in New Issue