etherpad-lite/src/static/js/pluginfw/hooks.js

209 lines
5.6 KiB
JavaScript
Raw Normal View History

2012-05-28 17:23:43 -07:00
var _ = require("underscore");
// Maps the name of a server-side hook to a string explaining the deprecation
// (e.g., 'use the foo hook instead').
//
// If you want to deprecate the fooBar hook, do the following:
//
// const hooks = require('ep_etherpad-lite/static/js/pluginfw/hooks');
// hooks.deprecationNotices.fooBar = 'use the newSpiffy hook instead';
//
exports.deprecationNotices = {};
const deprecationWarned = {};
function checkDeprecation(hook) {
const notice = exports.deprecationNotices[hook.hook_name];
if (notice == null) return;
if (deprecationWarned[hook.hook_fn_name]) return;
console.warn('%s hook used by the %s plugin (%s) is deprecated: %s',
hook.hook_name, hook.part.name, hook.hook_fn_name, notice);
deprecationWarned[hook.hook_fn_name] = true;
}
2012-03-27 22:23:28 +02:00
exports.bubbleExceptions = true
var hookCallWrapper = function (hook, hook_name, args, cb) {
if (cb === undefined) cb = function (x) { return x; };
checkDeprecation(hook);
// Normalize output to list for both sync and async cases
var normalize = function(x) {
if (x === undefined) return [];
return x;
}
var normalizedhook = function () {
return normalize(hook.hook_fn(hook_name, args, function (x) {
return cb(normalize(x));
}));
}
2012-03-27 22:23:28 +02:00
if (exports.bubbleExceptions) {
return normalizedhook();
2012-03-27 22:23:28 +02:00
} else {
try {
return normalizedhook();
2012-03-27 22:23:28 +02:00
} catch (ex) {
console.error([hook_name, hook.part.full_name, ex.stack || ex]);
}
}
}
exports.syncMapFirst = function (lst, fn) {
var i;
var result;
for (i = 0; i < lst.length; i++) {
result = fn(lst[i])
if (result.length) return result;
}
return [];
}
exports.mapFirst = function (lst, fn, cb) {
var i = 0;
2014-12-14 22:01:28 +01:00
var next = function () {
if (i >= lst.length) return cb(null, []);
fn(lst[i++], function (err, result) {
if (err) return cb(err);
if (result.length) return cb(null, result);
next();
});
}
next();
}
/* Don't use Array.concat as it flatterns arrays within the array */
exports.flatten = function (lst) {
var res = [];
if (lst != undefined && lst != null) {
for (var i = 0; i < lst.length; i++) {
if (lst[i] != undefined && lst[i] != null) {
for (var j = 0; j < lst[i].length; j++) {
res.push(lst[i][j]);
}
}
}
}
return res;
}
exports.callAll = function (hook_name, args) {
2012-03-27 22:23:28 +02:00
if (!args) args = {};
2013-02-18 08:29:25 -08:00
if (exports.plugins){
if (exports.plugins.hooks[hook_name] === undefined) return [];
return _.flatten(_.map(exports.plugins.hooks[hook_name], function (hook) {
return hookCallWrapper(hook, hook_name, args);
}), true);
}
}
async function aCallAll(hook_name, args, cb) {
2012-03-27 22:23:28 +02:00
if (!args) args = {};
if (!cb) cb = function () {};
2012-05-28 18:58:55 -07:00
if (exports.plugins.hooks[hook_name] === undefined) return cb(null, []);
var hooksPromises = exports.plugins.hooks[hook_name].map(async function(hook, index){
return await hookCallWrapper(hook, hook_name, args, function (res) {
return Promise.resolve(res);
});
});
var result = await Promise.all(hooksPromises);
// after forEach
cb(null, _.flatten(result, true));
}
/* return a Promise if cb is not supplied */
exports.aCallAll = function (hook_name, args, cb) {
if (cb === undefined) {
try{
return new Promise(function(resolve, reject) {
aCallAll(hook_name, args, function(err, res) {
return err ? reject(err) : resolve(res);
});
});
}catch(e){
$.gritter.removeAll();
$.gritter.add("Please update your web browser")
}
} else {
return aCallAll(hook_name, args, cb);
}
}
exports.callFirst = function (hook_name, args) {
2012-03-27 22:23:28 +02:00
if (!args) args = {};
2012-05-28 18:58:55 -07:00
if (exports.plugins.hooks[hook_name] === undefined) return [];
return exports.syncMapFirst(exports.plugins.hooks[hook_name], function (hook) {
return hookCallWrapper(hook, hook_name, args);
});
}
function aCallFirst(hook_name, args, cb) {
2012-03-27 22:23:28 +02:00
if (!args) args = {};
if (!cb) cb = function () {};
2012-05-28 18:58:55 -07:00
if (exports.plugins.hooks[hook_name] === undefined) return cb(null, []);
exports.mapFirst(
2012-05-28 18:58:55 -07:00
exports.plugins.hooks[hook_name],
function (hook, cb) {
hookCallWrapper(hook, hook_name, args, function (res) { cb(null, res); });
},
cb
);
}
/* return a Promise if cb is not supplied */
exports.aCallFirst = function (hook_name, args, cb) {
if (cb === undefined) {
return new Promise(function(resolve, reject) {
aCallFirst(hook_name, args, function(err, res) {
return err ? reject(err) : resolve(res);
});
});
} else {
return aCallFirst(hook_name, args, cb);
}
}
exports.callAllStr = function(hook_name, args, sep, pre, post) {
if (sep == undefined) sep = '';
if (pre == undefined) pre = '';
if (post == undefined) post = '';
var newCallhooks = [];
var callhooks = exports.callAll(hook_name, args);
for (var i = 0, ii = callhooks.length; i < ii; i++) {
newCallhooks[i] = pre + callhooks[i] + post;
}
return newCallhooks.join(sep || "");
}
/*
* Returns an array containing the names of the installed client-side plugins
*
* If no client-side plugins are installed, returns an empty array.
* Duplicate names are always discarded.
*
* A client-side plugin is a plugin that implements at least one client_hook
*
* EXAMPLE:
* No plugins: []
* Some plugins: [ 'ep_adminpads', 'ep_add_buttons', 'ep_activepads' ]
*/
exports.clientPluginNames = function() {
if (!(exports.plugins)) {
return [];
}
var client_plugin_names = _.uniq(
exports.plugins.parts
.filter(function(part) { return part.hasOwnProperty('client_hooks'); })
.map(function(part) { return 'plugin-' + part['plugin']; })
);
return client_plugin_names;
}