Files
xen-orchestra/packages/xo-lib/index.js

321 lines
7.3 KiB
JavaScript
Raw Normal View History

2014-07-26 10:15:29 +02:00
'use strict';
//====================================================================
var assign = require('lodash.assign');
2015-02-05 11:53:27 +01:00
var Bluebird = require('bluebird');
2015-02-05 17:05:49 +01:00
var EventEmitter = require('events').EventEmitter;
var inherits = require('util').inherits;
var jsonRpc = require('json-rpc');
2015-02-10 11:40:11 +01:00
var makeError = require('make-error');
2015-02-05 17:05:49 +01:00
var MethodNotFound = require('json-rpc/errors').MethodNotFound;
2015-02-05 11:53:27 +01:00
var WebSocket = require('ws');
2014-07-26 10:15:29 +02:00
2015-02-10 14:42:56 +01:00
var createCollection = require('./collection');
2014-07-26 10:15:29 +02:00
//====================================================================
2015-02-10 17:53:33 +01:00
// Expose Bluebird for now to ease integration (e.g. with Angular.js).
exports.setScheduler = Bluebird.setScheduler;
//====================================================================
2015-02-05 13:23:26 +01:00
function makeDeferred() {
var resolve, reject;
var promise = new Bluebird(function (resolve_, reject_) {
resolve = resolve_;
reject = reject_;
});
return {
promise: promise,
reject: reject,
resolve: resolve,
};
}
2015-02-05 13:19:47 +01:00
function startsWith(string, target) {
return (string.lastIndexOf(target, 0) === 0);
}
//====================================================================
2015-02-10 11:40:11 +01:00
function returnThis() {
/* jshint validthis: true */
return this;
}
// Returns an iterator to the Fibonacci sequence.
function fibonacci(start) {
var prev = 0;
var curr = start || 1;
var iterator = {
next: function () {
var tmp = curr;
curr += prev;
prev = tmp;
return {
done: false,
value: prev,
};
},
};
// Make the iterator a true iterable (ES6).
if (typeof Symbol !== 'undefined') {
iterator[Symbol.iterator] = returnThis;
}
return iterator;
}
//====================================================================
2014-07-28 13:21:19 +02:00
// Fix URL if necessary.
2015-02-05 12:41:25 +01:00
var URL_RE = /^(?:(?:http|ws)(s)?:\/\/)?(.*?)\/*(?:\/api\/)?$/;
2015-02-05 11:57:05 +01:00
function fixUrl(url) {
2015-02-05 12:41:25 +01:00
var matches = URL_RE.exec(url);
var isSecure = !!matches[1];
var rest = matches[2];
2014-07-28 13:21:19 +02:00
return [
2015-02-05 12:41:25 +01:00
isSecure ? 'wss' : 'ws',
'://',
rest,
'/api/',
2014-07-28 13:21:19 +02:00
].join('');
2015-02-05 11:57:05 +01:00
}
2015-02-10 11:40:11 +01:00
exports.fixUrl = fixUrl;
2014-07-28 13:21:19 +02:00
2014-07-26 10:15:29 +02:00
//====================================================================
function getCurrentUrl() {
/* global window: false */
if (typeof window === undefined) {
throw new Error('cannot get current URL');
}
return window.location.host + window.location.pathname;
}
//====================================================================
2015-02-10 11:40:11 +01:00
var ConnectionLost = makeError('ConnectionLost');
// Low level interface to XO.
function Api(url) {
2015-02-05 17:05:49 +01:00
// Super constructor.
EventEmitter.call(this);
2014-07-26 10:15:29 +02:00
2015-02-05 17:05:49 +01:00
// Fix the URL (ensure correct protocol and /api/ path).
this._url = fixUrl(url || getCurrentUrl());
2014-07-26 10:15:29 +02:00
2015-02-05 17:05:49 +01:00
// Will contains the WebSocket.
this._socket = null;
// The JSON-RPC server.
var this_ = this;
this._jsonRpc = jsonRpc.createServer(function (message) {
if (message.type === 'notification') {
this_.emit('notification', message);
} else {
// This object does not support requests.
throw new MethodNotFound(message.method);
}
}).on('data', function (message) {
this_._socket.send(JSON.stringify(message));
});
2015-02-05 11:57:05 +01:00
}
2015-02-10 11:40:11 +01:00
inherits(Api, EventEmitter);
2014-07-26 10:15:29 +02:00
2015-02-10 11:40:11 +01:00
assign(Api.prototype, {
2014-07-26 10:15:29 +02:00
close: function () {
2015-02-05 11:57:05 +01:00
if (this._socket) {
2014-07-26 10:15:29 +02:00
this._socket.close();
}
},
2015-02-05 13:23:26 +01:00
connect: Bluebird.method(function () {
2015-02-05 17:05:49 +01:00
if (this._socket) {
2015-02-05 13:23:26 +01:00
return;
2014-07-26 10:15:29 +02:00
}
2015-02-05 13:23:26 +01:00
var deferred = makeDeferred();
var opts = {};
2015-02-05 13:19:47 +01:00
if (startsWith(this._url, 'wss')) {
2014-07-26 23:42:04 +02:00
// Due to imperfect TLS implementation in XO-Server.
opts.rejectUnauthorized = false;
}
2015-02-05 13:22:49 +01:00
var socket = this._socket = new WebSocket(this._url, '', opts);
2014-07-26 10:15:29 +02:00
2015-02-05 17:05:49 +01:00
// Used to avoid binding listeners to this object.
var this_ = this;
2014-07-26 10:15:29 +02:00
// When the socket opens, send any queued requests.
2015-02-05 13:22:49 +01:00
socket.addEventListener('open', function () {
2014-07-26 10:15:29 +02:00
// Resolves the promise.
deferred.resolve();
2015-02-10 11:40:11 +01:00
this_.emit('connected');
2015-02-05 17:05:49 +01:00
});
2014-07-26 10:15:29 +02:00
2015-02-05 17:05:49 +01:00
socket.addEventListener('message', function (message) {
this_._jsonRpc.write(message.data);
});
2014-07-26 10:15:29 +02:00
2015-02-05 13:22:49 +01:00
socket.addEventListener('close', function () {
2015-02-05 17:05:49 +01:00
this_._socket = null;
2014-07-26 10:15:29 +02:00
2015-02-10 11:40:11 +01:00
this_._jsonRpc.failPendingRequests(new ConnectionLost());
// Only emit this event if connected before.
if (deferred.promise.isFulfilled()) {
this_.emit('disconnected');
}
2015-02-05 17:05:49 +01:00
});
2014-07-26 10:15:29 +02:00
2015-02-05 13:23:26 +01:00
socket.addEventListener('error', function (error) {
2014-07-26 10:15:29 +02:00
// Fails the connect promise if possible.
deferred.reject(error);
});
return deferred.promise;
2015-02-05 13:23:26 +01:00
}),
2014-07-26 10:15:29 +02:00
call: function (method, params) {
2015-02-05 17:05:49 +01:00
var jsonRpc = this._jsonRpc;
2014-07-26 10:15:29 +02:00
2015-02-05 17:05:49 +01:00
return this.connect().then(function () {
return jsonRpc.request(method, params);
});
2014-07-26 10:15:29 +02:00
},
});
2015-02-10 11:40:11 +01:00
exports.Api = Api;
//====================================================================
2015-02-10 14:42:56 +01:00
var objectsOptions = {
indexes: [
'ref',
'type',
'UUID',
],
key: function (item) {
return item.UUID || item.ref;
},
};
2015-02-10 11:40:11 +01:00
// High level interface to Xo.
//
// Handle auto-reconnect, sign in & objects cache.
function Xo(opts) {
var self = this;
this._api = new Api(opts.url);
this._auth = opts.auth;
this._backOff = fibonacci(1e3);
2015-02-10 14:42:56 +01:00
this.objects = createCollection(objectsOptions);
2015-02-10 17:12:10 +01:00
this.status = 'disconnected';
2015-02-10 11:40:11 +01:00
this.user = null;
// Promise representing the connection status.
this._connection = null;
self._api.on('disconnected', function () {
self._connection = null;
2015-02-10 14:42:56 +01:00
self.objects.clear();
2015-02-10 17:12:10 +01:00
self.status = 'disconnected';
2015-02-10 17:53:43 +01:00
// Automatically reconnect.
self.connect();
2015-02-10 11:40:11 +01:00
});
self._api.on('notification', function (notification) {
if (notification.method !== 'all') {
return;
}
2015-02-10 14:42:56 +01:00
var method = (
notification.params.type === 'exit' ?
'unset' :
'set'
) + 'Multiple';
2015-02-10 11:40:11 +01:00
2015-02-10 14:42:56 +01:00
self.objects[method](notification.params.items);
2015-02-10 11:40:11 +01:00
});
}
2015-02-10 17:12:10 +01:00
function tryConnect() {
/* jshint validthis: true */
2015-02-10 11:40:11 +01:00
2015-02-10 17:12:10 +01:00
this.status = 'connecting';
return this._api.connect().bind(this).catch(function () {
return Bluebird.delay(this._backOff.next().value).then(tryConnect);
});
}
function onSuccessfulConnection() {
/* jshint validthis: true */
// FIXME: session.signIn() should work with both token and password.
return this._api.call(
this._auth.token ?
'session.signInWithToken' :
'session.signInWithPassword',
this._auth
).bind(this).then(function (user) {
this.user = user;
this.status = 'connected';
this._api.call('xo.getAllObjects').bind(this).then(function (objects) {
this.objects.setMultiple(objects);
2015-02-10 11:40:11 +01:00
});
2015-02-10 17:12:10 +01:00
});
}
function onFailedConnection() {
/* jshint validthis: true */
this.status = 'disconnected';
}
Xo.prototype.connect = function () {
if (this._connection) {
return this._connection;
}
this._connection = tryConnect.call(this).then(
onSuccessfulConnection, onFailedConnection
);
return this._connection;
};
2015-02-10 11:40:11 +01:00
2015-02-10 17:12:10 +01:00
Xo.prototype.call = function (method, params) {
// TODO: prevent session.*() from being because it may interfere
// with this class session management.
return this.connect().then(function () {
var self = this;
return this._api.call(method, params).catch(ConnectionLost, function () {
// Retry automatically.
return self.call(method, params);
2015-02-10 11:40:11 +01:00
});
2015-02-10 17:12:10 +01:00
});
};
2015-02-10 11:40:11 +01:00
exports.Xo = Xo;
2014-07-26 10:15:29 +02:00
//====================================================================
2015-02-10 14:47:16 +01:00
function createXo(opts) {
return new Xo(opts);
}
exports = module.exports = assign(createXo, module.exports);