'use strict'; var winston = require('winston'); var async = require('async'); module.exports = function (Plugins) { Plugins.deprecatedHooks = { 'filter:user.custom_fields': null, // remove in v1.1.0 'filter:post.save': 'filter:post.create', 'filter:user.profileLinks': 'filter:user.profileMenu', 'action:post.flag': 'action:flag.create', 'action:flag.create': 'action:flags.create', 'action:flag.update': 'action:flags.update', }; /* `data` is an object consisting of (* is required): `data.hook`*, the name of the NodeBB hook `data.method`*, the method called in that plugin `data.priority`, the relative priority of the method when it is eventually called (default: 10) */ Plugins.registerHook = function (id, data, callback) { callback = callback || function () {}; function register() { Plugins.loadedHooks[data.hook] = Plugins.loadedHooks[data.hook] || []; Plugins.loadedHooks[data.hook].push(data); callback(); } if (!data.hook) { winston.warn('[plugins/' + id + '] registerHook called with invalid data.hook', data); return callback(); } var method; if (Object.keys(Plugins.deprecatedHooks).indexOf(data.hook) !== -1) { winston.warn('[plugins/' + id + '] Hook `' + data.hook + '` is deprecated, ' + (Plugins.deprecatedHooks[data.hook] ? 'please use `' + Plugins.deprecatedHooks[data.hook] + '` instead.' : 'there is no alternative.' )); } else { // handle hook's startsWith, i.e. action:homepage.get var parts = data.hook.split(':'); if (parts.length > 2) { parts.pop(); } parts.join(':'); } if (data.hook && data.method) { data.id = id; if (!data.priority) { data.priority = 10; } if (typeof data.method === 'string' && data.method.length > 0) { method = data.method.split('.').reduce(function (memo, prop) { if (memo && memo[prop]) { return memo[prop]; } // Couldn't find method by path, aborting return null; }, Plugins.libraries[data.id]); // Write the actual method reference to the hookObj data.method = method; register(); } else if (typeof data.method === 'function') { register(); } else { winston.warn('[plugins/' + id + '] Hook method mismatch: ' + data.hook + ' => ' + data.method); return callback(); } } }; Plugins.unregisterHook = function (id, hook, method) { var hooks = Plugins.loadedHooks[hook] || []; Plugins.loadedHooks[hook] = hooks.filter(function (hookData) { return hookData && hookData.id !== id && hookData.method !== method; }); }; Plugins.fireHook = function (hook, params, callback) { callback = typeof callback === 'function' ? callback : function () {}; var hookList = Plugins.loadedHooks[hook]; var hookType = hook.split(':')[0]; switch (hookType) { case 'filter': fireFilterHook(hook, hookList, params, callback); break; case 'action': fireActionHook(hook, hookList, params, callback); break; case 'static': fireStaticHook(hook, hookList, params, callback); break; default: winston.warn('[plugins] Unknown hookType: ' + hookType + ', hook : ' + hook); break; } }; function fireFilterHook(hook, hookList, params, callback) { if (!Array.isArray(hookList) || !hookList.length) { return callback(null, params); } async.reduce(hookList, params, function (params, hookObj, next) { if (typeof hookObj.method !== 'function') { if (global.env === 'development') { winston.warn('[plugins] Expected method for hook \'' + hook + '\' in plugin \'' + hookObj.id + '\' not found, skipping.'); } return next(null, params); } hookObj.method(params, next); }, callback); } function fireActionHook(hook, hookList, params, callback) { if (!Array.isArray(hookList) || !hookList.length) { return callback(); } async.each(hookList, function (hookObj, next) { if (typeof hookObj.method !== 'function') { if (global.env === 'development') { winston.warn('[plugins] Expected method for hook \'' + hook + '\' in plugin \'' + hookObj.id + '\' not found, skipping.'); } return next(); } hookObj.method(params); next(); }, callback); } function fireStaticHook(hook, hookList, params, callback) { if (!Array.isArray(hookList) || !hookList.length) { return callback(); } async.each(hookList, function (hookObj, next) { if (typeof hookObj.method === 'function') { var timedOut = false; var timeoutId = setTimeout(function () { winston.warn('[plugins] Callback timed out, hook \'' + hook + '\' in plugin \'' + hookObj.id + '\''); timedOut = true; next(); }, 5000); try { hookObj.method(params, function () { clearTimeout(timeoutId); if (!timedOut) { next.apply(null, arguments); } }); } catch (err) { winston.error('[plugins] Error executing \'' + hook + '\' in plugin \'' + hookObj.id + '\''); winston.error(err); clearTimeout(timeoutId); next(); } } else { next(); } }, callback); } Plugins.hasListeners = function (hook) { return !!(Plugins.loadedHooks[hook] && Plugins.loadedHooks[hook].length > 0); }; };