2017-08-04 14:25:17 -04:00
|
|
|
const dispatch = require('../dispatch/central-dispatch');
|
2017-07-21 16:46:08 -04:00
|
|
|
const log = require('../util/log');
|
|
|
|
|
|
|
|
const BlockType = require('./block-type');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} ArgumentInfo - Information about an extension block argument
|
|
|
|
* @property {ArgumentType} type - the type of value this argument can take
|
|
|
|
* @property {*|undefined} default - the default value of this argument (default: blank)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} BlockInfo - Information about an extension block
|
|
|
|
* @property {string} opcode - the block opcode
|
|
|
|
* @property {string|object} text - the human-readable text on this block
|
|
|
|
* @property {BlockType|undefined} blockType - the type of block (default: BlockType.COMMAND)
|
|
|
|
* @property {int|undefined} branchCount - the number of branches this block controls, if conditional (default: 0)
|
|
|
|
* @property {Boolean|undefined} isTerminal - true if this block ends a stack (default: false)
|
|
|
|
* @property {Boolean|undefined} blockAllThreads - true if all threads must wait for this block to run (default: false)
|
|
|
|
* @property {object.<string,ArgumentInfo>|undefined} arguments - information about this block's arguments, if any
|
2017-08-04 14:25:17 -04:00
|
|
|
* @property {string|Function|undefined} func - the method for this block on the extension service (default: opcode)
|
2017-07-21 16:46:08 -04:00
|
|
|
* @property {Array.<string>|undefined} filter - the list of targets for which this block should appear (default: all)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} CategoryInfo - Information about a block category
|
|
|
|
* @property {string} id - the unique ID of this category
|
|
|
|
* @property {string} color1 - the primary color for this category, in '#rrggbb' format
|
|
|
|
* @property {string} color2 - the secondary color for this category, in '#rrggbb' format
|
|
|
|
* @property {string} color3 - the tertiary color for this category, in '#rrggbb' format
|
2017-09-04 18:48:51 -04:00
|
|
|
* @property {Array.<BlockInfo>} block - the blocks in this category
|
2017-08-22 17:28:38 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {object} PendingExtensionWorker - Information about an extension worker still initializing
|
|
|
|
* @property {string} extensionURL - the URL of the extension to be loaded by this worker
|
|
|
|
* @property {Function} resolve - function to call on successful worker startup
|
|
|
|
* @property {Function} reject - function to call on failed worker startup
|
2017-07-21 16:46:08 -04:00
|
|
|
*/
|
|
|
|
|
2017-07-14 15:37:59 -04:00
|
|
|
class ExtensionManager {
|
|
|
|
constructor () {
|
|
|
|
/**
|
|
|
|
* The ID number to provide to the next extension worker.
|
|
|
|
* @type {int}
|
|
|
|
*/
|
|
|
|
this.nextExtensionWorker = 0;
|
|
|
|
|
|
|
|
/**
|
2017-08-22 17:28:38 -04:00
|
|
|
* FIFO queue of extensions which have been requested but not yet loaded in a worker,
|
|
|
|
* along with promise resolution functions to call once the worker is ready or failed.
|
|
|
|
*
|
|
|
|
* @type {Array.<PendingExtensionWorker>}
|
|
|
|
*/
|
|
|
|
this.pendingExtensions = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Map of worker ID to workers which have been allocated but have not yet finished initialization.
|
|
|
|
* @type {Array.<PendingExtensionWorker>}
|
2017-07-14 15:37:59 -04:00
|
|
|
*/
|
2017-08-22 17:28:38 -04:00
|
|
|
this.pendingWorkers = [];
|
2017-07-14 15:37:59 -04:00
|
|
|
|
2017-08-04 14:25:17 -04:00
|
|
|
dispatch.setService('extensions', this).catch(e => {
|
|
|
|
log.error(`ExtensionManager was unable to register extension service: ${JSON.stringify(e)}`);
|
|
|
|
});
|
2017-07-14 15:37:59 -04:00
|
|
|
}
|
|
|
|
|
2017-08-22 17:28:38 -04:00
|
|
|
/**
|
|
|
|
* Load an extension by URL
|
|
|
|
* @param {string} extensionURL - the URL for the extension to load
|
|
|
|
* @returns {Promise} resolved once the extension is loaded and initialized or rejected on failure
|
|
|
|
*/
|
2017-07-14 15:37:59 -04:00
|
|
|
loadExtensionURL (extensionURL) {
|
2017-08-22 17:28:38 -04:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// If we `require` this at the global level it breaks non-webpack targets, including tests
|
2017-08-29 14:43:09 -04:00
|
|
|
const ExtensionWorker = require('worker-loader?name=extension-worker.js!./extension-worker');
|
2017-07-14 15:37:59 -04:00
|
|
|
|
2017-08-22 17:28:38 -04:00
|
|
|
this.pendingExtensions.push({extensionURL, resolve, reject});
|
|
|
|
dispatch.addWorker(new ExtensionWorker());
|
|
|
|
});
|
2017-07-14 15:37:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
allocateWorker () {
|
|
|
|
const id = this.nextExtensionWorker++;
|
2017-08-22 17:28:38 -04:00
|
|
|
const workerInfo = this.pendingExtensions.shift();
|
|
|
|
this.pendingWorkers[id] = workerInfo;
|
|
|
|
return [id, workerInfo.extensionURL];
|
2017-07-14 15:37:59 -04:00
|
|
|
}
|
2017-07-21 16:46:08 -04:00
|
|
|
|
|
|
|
registerExtensionService (serviceName) {
|
2017-08-04 14:25:17 -04:00
|
|
|
dispatch.call(serviceName, 'getInfo').then(info => {
|
2017-07-21 16:46:08 -04:00
|
|
|
this._registerExtensionInfo(serviceName, info);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-22 17:28:38 -04:00
|
|
|
onWorkerInit (id, e) {
|
|
|
|
const workerInfo = this.pendingWorkers[id];
|
|
|
|
delete this.pendingWorkers[id];
|
|
|
|
if (e) {
|
|
|
|
workerInfo.reject(e);
|
|
|
|
} else {
|
|
|
|
workerInfo.resolve(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 00:00:59 -04:00
|
|
|
/**
|
|
|
|
* Register an internal (non-Worker) extension object
|
|
|
|
* @param {object} extensionObject - the extension object to register
|
2017-10-02 15:02:44 -04:00
|
|
|
* @returns {Promise} resolved once the extension is fully registered or rejected on failure
|
2017-09-27 00:00:59 -04:00
|
|
|
*/
|
|
|
|
_registerInternalExtension (extensionObject) {
|
|
|
|
const extensionInfo = extensionObject.getInfo();
|
|
|
|
const serviceName = `extension.internal.${extensionInfo.id}`;
|
2017-10-02 15:02:44 -04:00
|
|
|
return dispatch.setService(serviceName, extensionObject)
|
2017-09-27 00:00:59 -04:00
|
|
|
.then(() => dispatch.call('extensions', 'registerExtensionService', serviceName));
|
|
|
|
}
|
|
|
|
|
2017-07-21 16:46:08 -04:00
|
|
|
_registerExtensionInfo (serviceName, extensionInfo) {
|
2017-08-04 14:25:17 -04:00
|
|
|
extensionInfo = this._prepareExtensionInfo(serviceName, extensionInfo);
|
|
|
|
dispatch.call('runtime', '_registerExtensionPrimitives', extensionInfo).catch(e => {
|
|
|
|
log.error(`Failed to register primitives for extension on service ${serviceName}: ${JSON.stringify(e)}`);
|
|
|
|
});
|
2017-07-21 16:46:08 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Modify the provided text as necessary to ensure that it may be used as an attribute value in valid XML.
|
|
|
|
* @param {string} text - the text to be sanitized
|
|
|
|
* @returns {string} - the sanitized text
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_sanitizeID (text) {
|
|
|
|
return text.toString().replace(/[<"&]/, '_');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply minor cleanup and defaults for optional extension fields.
|
|
|
|
* TODO: make the ID unique in cases where two copies of the same extension are loaded.
|
2017-08-04 14:25:17 -04:00
|
|
|
* @param {string} serviceName - the name of the service hosting this extension block
|
2017-07-21 16:46:08 -04:00
|
|
|
* @param {ExtensionInfo} extensionInfo - the extension info to be sanitized
|
|
|
|
* @returns {ExtensionInfo} - a new extension info object with cleaned-up values
|
|
|
|
* @private
|
|
|
|
*/
|
2017-08-04 14:25:17 -04:00
|
|
|
_prepareExtensionInfo (serviceName, extensionInfo) {
|
2017-07-21 16:46:08 -04:00
|
|
|
extensionInfo = Object.assign({}, extensionInfo);
|
|
|
|
extensionInfo.id = this._sanitizeID(extensionInfo.id);
|
|
|
|
extensionInfo.name = extensionInfo.name || extensionInfo.id;
|
|
|
|
extensionInfo.blocks = extensionInfo.blocks || [];
|
|
|
|
extensionInfo.targetTypes = extensionInfo.targetTypes || [];
|
2017-08-04 14:25:17 -04:00
|
|
|
extensionInfo.blocks = extensionInfo.blocks.reduce((result, blockInfo) => {
|
|
|
|
try {
|
|
|
|
result.push(this._prepareBlockInfo(serviceName, blockInfo));
|
|
|
|
} catch (e) {
|
|
|
|
// TODO: more meaningful error reporting
|
2017-10-06 16:37:59 -04:00
|
|
|
log.error(`Error processing block: ${e.message}, Block:\n${JSON.stringify(blockInfo)}`);
|
2017-08-04 14:25:17 -04:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}, []);
|
2017-07-21 16:46:08 -04:00
|
|
|
return extensionInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply defaults for optional block fields.
|
2017-08-04 14:25:17 -04:00
|
|
|
* @param {string} serviceName - the name of the service hosting this extension block
|
2017-07-21 16:46:08 -04:00
|
|
|
* @param {BlockInfo} blockInfo - the block info from the extension
|
|
|
|
* @returns {BlockInfo} - a new block info object which has values for all relevant optional fields.
|
|
|
|
* @private
|
|
|
|
*/
|
2017-08-04 14:25:17 -04:00
|
|
|
_prepareBlockInfo (serviceName, blockInfo) {
|
2017-07-21 16:46:08 -04:00
|
|
|
blockInfo = Object.assign({}, {
|
|
|
|
blockType: BlockType.COMMAND,
|
|
|
|
terminal: false,
|
|
|
|
blockAllThreads: false,
|
|
|
|
arguments: {}
|
|
|
|
}, blockInfo);
|
|
|
|
blockInfo.opcode = this._sanitizeID(blockInfo.opcode);
|
|
|
|
blockInfo.func = blockInfo.func ? this._sanitizeID(blockInfo.func) : blockInfo.opcode;
|
2017-09-27 00:00:59 -04:00
|
|
|
blockInfo.text = blockInfo.text || blockInfo.opcode;
|
2017-08-04 14:25:17 -04:00
|
|
|
blockInfo.func = dispatch.call.bind(dispatch, serviceName, blockInfo.func);
|
2017-07-21 16:46:08 -04:00
|
|
|
return blockInfo;
|
|
|
|
}
|
2017-07-14 15:37:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = ExtensionManager;
|