Add basic auth

This commit is contained in:
wltsmrz
2015-10-19 13:52:38 -07:00
parent 98422e4153
commit f65b673451
5 changed files with 97 additions and 40 deletions

View File

@@ -149,6 +149,9 @@ function Remote(options = {}) {
if (!Array.isArray(this.servers)) { if (!Array.isArray(this.servers)) {
throw new TypeError('servers must be an array'); throw new TypeError('servers must be an array');
} }
if (!(_.isUndefined(this.basic_auth) || _.isString(this.basic_auth))) {
throw new TypeError('basic_auth must be a string');
}
this.setMaxListeners(this.max_listeners); this.setMaxListeners(this.max_listeners);

View File

@@ -5,6 +5,7 @@ const assert = require('assert');
const util = require('util'); const util = require('util');
const url = require('url'); const url = require('url');
const LRU = require('lru-cache'); const LRU = require('lru-cache');
const HttpsProxyAgent = require('https-proxy-agent');
const EventEmitter = require('events').EventEmitter; const EventEmitter = require('events').EventEmitter;
const RippleError = require('./rippleerror').RippleError; const RippleError = require('./rippleerror').RippleError;
const Amount = require('./amount').Amount; const Amount = require('./amount').Amount;
@@ -418,7 +419,6 @@ Server.prototype.reconnect = function() {
Server.prototype.connect = function() { Server.prototype.connect = function() {
const self = this; const self = this;
const WebSocket = Server.websocketConstructor(); const WebSocket = Server.websocketConstructor();
if (!WebSocket) { if (!WebSocket) {
@@ -442,23 +442,24 @@ Server.prototype.connect = function() {
log.info(this.getServerID(), 'connect'); log.info(this.getServerID(), 'connect');
} }
if (this._remote.proxy !== undefined) { const wsOptions = {};
const parsed = url.parse(this._opts.url);
const opts = url.parse(this._remote.proxy);
opts.secureEndpoint = parsed.protocol === 'wss:';
let HttpsProxyAgent;
try {
HttpsProxyAgent = require('https-proxy-agent');
} catch (error) {
throw new Error('"proxy" option is not supported in the browser');
}
const agent = new HttpsProxyAgent(opts);
this._ws = new WebSocket(this._opts.url, {agent: agent}); if (!_.isUndefined(this._remote.proxy)) {
} else { const proxyOptions = _.merge(url.parse(this._remote.proxy), {
this._ws = new WebSocket(this._opts.url); secureEndpoint: /^wss/.test(this._opts.url)
});
wsOptions.agent = new HttpsProxyAgent(proxyOptions);
} }
if (!_.isUndefined(this._remote.basic_auth)) {
const authOptions = this._remote.basic_auth;
const auth = new Buffer(authOptions).toString('base64');
wsOptions.headers = {Authorization: `Basic ${auth}`};
}
this._ws = new WebSocket(this._opts.url, wsOptions);
const ws = this._ws; const ws = this._ws;
this._shouldConnect = true; this._shouldConnect = true;
@@ -505,11 +506,13 @@ Server.prototype.connect = function() {
self.emit('socket_error'); self.emit('socket_error');
if (self._remote.trace) { if (self._remote.trace) {
log.info(self.getServerID(), 'onerror:', e.data || e); log.info(`${self.getServerID()} onerror: ${e.toString()}`);
} }
if (Server.TLS_ERRORS.indexOf(e.message) !== -1) { if (Server.TLS_ERRORS.includes(e.message)) {
// Unrecoverable throw e;
}
if (e.message.includes('unexpected')) {
throw e; throw e;
} }

View File

@@ -1,4 +1,11 @@
'use strict'; 'use strict';
/* eslint-disable max-len */
// Enable core-js polyfills. This allows use of ES6/7 extensions listed here:
// https://github.com/zloirock/core-js/blob/fb0890f32dabe8d4d88a4350d1b268446127132e/shim.js#L1-L103
/* eslint-enable max-len */
require('babel-core/polyfill');
const core = require('./core'); const core = require('./core');
const RippleAPI = require('./api'); const RippleAPI = require('./api');

View File

@@ -792,11 +792,7 @@ describe('Remote', function() {
it('Get server', function() { it('Get server', function() {
remote.addServer('wss://sasdf.ripple.com:443'); remote.addServer('wss://sasdf.ripple.com:443');
remote.connect(); remote._servers.concat(remote).forEach(s => s._connected = true);
remote._connected = true;
remote._servers.forEach(function(s) {
s._connected = true;
});
const message = { const message = {
type: 'ledgerClosed', type: 'ledgerClosed',
@@ -828,19 +824,6 @@ describe('Remote', function() {
assert.strictEqual(remote._servers.length, 2); assert.strictEqual(remote._servers.length, 2);
assert.strictEqual(remote.getServer(), null); assert.strictEqual(remote.getServer(), null);
}); });
it('Get server -- primary server', function() {
const server = remote.addServer({
host: 'sasdf.ripple.com',
port: 443,
secure: true,
primary: true
});
remote.connect();
server._connected = true;
assert.strictEqual(remote.getServer().getServerID(), server.getServerID());
});
it('Parse binary transaction', function() { it('Parse binary transaction', function() {
const binaryTransaction = require('./fixtures/binary-transaction.json'); const binaryTransaction = require('./fixtures/binary-transaction.json');

View File

@@ -1,6 +1,6 @@
'use strict'; 'use strict';
/* eslint-disable no-new */ /* eslint-disable no-new, max-len */
const _ = require('lodash'); const _ = require('lodash');
const assert = require('assert'); const assert = require('assert');
@@ -417,6 +417,67 @@ describe('Server', function() {
server.connect(); server.connect();
}); });
it('Connect -- with basic auth', function(done) {
const port = 5748;
function verifyClient(info, callback) {
assert.strictEqual(info.req.headers.authorization, 'Basic dXNlcm5hbWU6cGFzc3dvcmQ=');
callback(true);
}
const wss = new ws.Server({port, verifyClient});
function handleSubscribeRequest(_ws, message) {
const m = JSON.parse(message);
assert.deepEqual(m, {
command: 'subscribe',
id: 0,
streams: ['ledger', 'server']
});
_ws.send(JSON.stringify({
id: 0,
status: 'success',
type: 'response',
result: {
fee_base: 10,
fee_ref: 10,
ledger_hash:
'1838539EE12463C36F2C53B079D807C697E3D93A1936B717E565A4A912E11776',
ledger_index: 7053695,
ledger_time: 455414390,
load_base: 256,
load_factor: 256,
random:
'E56C9154D9BE94D49C581179356C2E084E16D18D74E8B09093F2D61207625E6A',
reserve_base: 20000000,
reserve_inc: 5000000,
server_status: 'full',
validated_ledgers: '32570-7053695'
}
}));
}
wss.once('connection', function(_ws) {
_ws.once('message', _.partial(handleSubscribeRequest, _ws));
});
const remote = new Remote({
basic_auth: 'username:password'
});
const server = new Server(remote, `ws://localhost:${port}`);
server.once('connect', function() {
server.once('disconnect', function() {
wss.close();
done();
});
server.disconnect();
});
server.connect();
});
it('Connect - already connected', function(done) { it('Connect - already connected', function(done) {
const server = new Server(new Remote(), 'ws://localhost:5748'); const server = new Server(new Remote(), 'ws://localhost:5748');
server._connected = true; server._connected = true;