2017-08-05 02:52:31 -04:00
|
|
|
const SharedDispatch = require('./shared-dispatch');
|
|
|
|
|
2017-07-12 18:48:36 -04:00
|
|
|
const log = require('../util/log');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class serves as the central broker for message dispatch. It expects to operate on the main thread / Window and
|
|
|
|
* it must be informed of any Worker threads which will participate in the messaging system. From any context in the
|
|
|
|
* messaging system, the dispatcher's "call" method can call any method on any "service" provided in any participating
|
|
|
|
* context. The dispatch system will forward function arguments and return values across worker boundaries as needed.
|
|
|
|
* @see {WorkerDispatch}
|
|
|
|
*/
|
2017-08-05 02:52:31 -04:00
|
|
|
class CentralDispatch extends SharedDispatch {
|
2017-07-12 18:48:36 -04:00
|
|
|
constructor () {
|
2017-08-05 02:52:31 -04:00
|
|
|
super();
|
2017-07-12 18:48:36 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Map of channel name to worker or local service provider.
|
|
|
|
* If the entry is a Worker, the service is provided by an object on that worker.
|
|
|
|
* Otherwise, the service is provided locally and methods on the service will be called directly.
|
|
|
|
* @see {setService}
|
|
|
|
* @type {object.<Worker|object>}
|
|
|
|
*/
|
|
|
|
this.services = {};
|
|
|
|
|
2017-07-16 02:51:59 -04:00
|
|
|
/**
|
|
|
|
* The constructor we will use to recognize workers.
|
|
|
|
* @type {Function}
|
|
|
|
*/
|
|
|
|
this.workerClass = (typeof Worker === 'undefined' ? null : Worker);
|
|
|
|
|
2017-07-12 18:48:36 -04:00
|
|
|
/**
|
|
|
|
* List of workers attached to this dispatcher.
|
|
|
|
* @type {Array}
|
|
|
|
*/
|
|
|
|
this.workers = [];
|
|
|
|
}
|
|
|
|
|
2019-03-20 14:45:21 -04:00
|
|
|
/**
|
|
|
|
* Synchronously call a particular method on a particular service provided locally.
|
|
|
|
* Calling this function on a remote service will fail.
|
|
|
|
* @param {string} service - the name of the service.
|
|
|
|
* @param {string} method - the name of the method.
|
|
|
|
* @param {*} [args] - the arguments to be copied to the method, if any.
|
|
|
|
* @returns {*} - the return value of the service method.
|
|
|
|
*/
|
|
|
|
callSync (service, method, ...args) {
|
|
|
|
const {provider, isRemote} = this._getServiceProvider(service);
|
|
|
|
if (provider) {
|
|
|
|
if (isRemote) {
|
2019-03-25 16:32:51 -04:00
|
|
|
throw new Error(`Cannot use 'callSync' on remote provider for service ${service}.`);
|
2019-03-20 14:45:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return provider[method].apply(provider, args);
|
|
|
|
}
|
2019-03-25 16:32:51 -04:00
|
|
|
throw new Error(`Provider not found for service: ${service}`);
|
2019-03-20 14:45:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Synchronously set a local object as the global provider of the specified service.
|
|
|
|
* WARNING: Any method on the provider can be called from any worker within the dispatch system.
|
|
|
|
* @param {string} service - a globally unique string identifying this service. Examples: 'vm', 'gui', 'extension9'.
|
|
|
|
* @param {object} provider - a local object which provides this service.
|
|
|
|
*/
|
|
|
|
setServiceSync (service, provider) {
|
|
|
|
if (this.services.hasOwnProperty(service)) {
|
|
|
|
log.warn(`Central dispatch replacing existing service provider for ${service}`);
|
|
|
|
}
|
|
|
|
this.services[service] = provider;
|
|
|
|
}
|
|
|
|
|
2017-07-12 18:48:36 -04:00
|
|
|
/**
|
|
|
|
* Set a local object as the global provider of the specified service.
|
2017-07-21 16:13:45 -04:00
|
|
|
* WARNING: Any method on the provider can be called from any worker within the dispatch system.
|
2017-07-12 18:48:36 -04:00
|
|
|
* @param {string} service - a globally unique string identifying this service. Examples: 'vm', 'gui', 'extension9'.
|
|
|
|
* @param {object} provider - a local object which provides this service.
|
2017-07-21 16:13:45 -04:00
|
|
|
* @returns {Promise} - a promise which will resolve once the service is registered.
|
2017-07-12 18:48:36 -04:00
|
|
|
*/
|
|
|
|
setService (service, provider) {
|
2017-08-05 02:52:31 -04:00
|
|
|
/** Return a promise for consistency with {@link WorkerDispatch#setService} */
|
|
|
|
try {
|
2019-03-20 14:45:21 -04:00
|
|
|
this.setServiceSync(service, provider);
|
2017-08-05 02:52:31 -04:00
|
|
|
return Promise.resolve();
|
|
|
|
} catch (e) {
|
|
|
|
return Promise.reject(e);
|
|
|
|
}
|
2017-07-12 18:48:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a worker to the message dispatch system. The worker must implement a compatible message dispatch framework.
|
|
|
|
* The dispatcher will immediately attempt to "handshake" with the worker.
|
|
|
|
* @param {Worker} worker - the worker to add into the dispatch system.
|
|
|
|
*/
|
|
|
|
addWorker (worker) {
|
|
|
|
if (this.workers.indexOf(worker) === -1) {
|
|
|
|
this.workers.push(worker);
|
2017-07-16 02:51:59 -04:00
|
|
|
worker.onmessage = this._onMessage.bind(this, worker);
|
2017-08-05 02:52:31 -04:00
|
|
|
this._remoteCall(worker, 'dispatch', 'handshake').catch(e => {
|
|
|
|
log.error(`Could not handshake with worker: ${JSON.stringify(e)}`);
|
|
|
|
});
|
2017-07-12 18:48:36 -04:00
|
|
|
} else {
|
2017-08-05 02:52:31 -04:00
|
|
|
log.warn('Central dispatch ignoring attempt to add duplicate worker');
|
2017-07-12 18:48:36 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-05 02:52:31 -04:00
|
|
|
* Fetch the service provider object for a particular service name.
|
|
|
|
* @override
|
|
|
|
* @param {string} service - the name of the service to look up
|
|
|
|
* @returns {{provider:(object|Worker), isRemote:boolean}} - the means to contact the service, if found
|
|
|
|
* @protected
|
2017-07-12 18:48:36 -04:00
|
|
|
*/
|
2017-08-05 02:52:31 -04:00
|
|
|
_getServiceProvider (service) {
|
|
|
|
const provider = this.services[service];
|
|
|
|
return provider && {
|
|
|
|
provider,
|
2018-03-07 18:13:15 -05:00
|
|
|
isRemote: Boolean(this.workerClass && provider instanceof this.workerClass)
|
2017-08-05 02:52:31 -04:00
|
|
|
};
|
2017-07-12 18:48:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-05 02:52:31 -04:00
|
|
|
* Handle a call message sent to the dispatch service itself
|
|
|
|
* @override
|
|
|
|
* @param {Worker} worker - the worker which sent the message.
|
|
|
|
* @param {DispatchCallMessage} message - the message to be handled.
|
|
|
|
* @returns {Promise|undefined} - a promise for the results of this operation, if appropriate
|
|
|
|
* @protected
|
2017-07-12 18:48:36 -04:00
|
|
|
*/
|
2017-08-05 02:52:31 -04:00
|
|
|
_onDispatchMessage (worker, message) {
|
|
|
|
let promise;
|
|
|
|
switch (message.method) {
|
|
|
|
case 'setService':
|
|
|
|
promise = this.setService(message.args[0], worker);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
log.error(`Central dispatch received message for unknown method: ${message.method}`);
|
2017-07-12 18:48:36 -04:00
|
|
|
}
|
2017-08-05 02:52:31 -04:00
|
|
|
return promise;
|
2017-07-12 18:48:36 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-16 02:51:59 -04:00
|
|
|
module.exports = new CentralDispatch();
|