mirror of
https://github.com/ether/etherpad-lite.git
synced 2025-04-21 16:06:16 -04:00

There are two main benefits: * HTML is no longer printed in the startup debug logs. * `require()` is no longer called on client-side files. This eliminates "Failed to load <file> for <plugin>: ReferenceError: window is not defined" errors when users visit `/admin/plugins/info`.
135 lines
4.3 KiB
JavaScript
135 lines
4.3 KiB
JavaScript
'use strict';
|
|
|
|
const eejs = require('../../eejs');
|
|
const settings = require('../../utils/Settings');
|
|
const installer = require('../../../static/js/pluginfw/installer');
|
|
const pluginDefs = require('../../../static/js/pluginfw/plugin_defs');
|
|
const plugins = require('../../../static/js/pluginfw/plugins');
|
|
const semver = require('semver');
|
|
const UpdateCheck = require('../../utils/UpdateCheck');
|
|
|
|
exports.expressCreateServer = (hookName, args, cb) => {
|
|
args.app.get('/admin/plugins', (req, res) => {
|
|
res.send(eejs.require('ep_etherpad-lite/templates/admin/plugins.html', {
|
|
plugins: pluginDefs.plugins,
|
|
req,
|
|
errors: [],
|
|
}));
|
|
});
|
|
|
|
args.app.get('/admin/plugins/info', (req, res) => {
|
|
const gitCommit = settings.getGitCommit();
|
|
const epVersion = settings.getEpVersion();
|
|
|
|
res.send(eejs.require('ep_etherpad-lite/templates/admin/plugins-info.html', {
|
|
gitCommit,
|
|
epVersion,
|
|
installedPlugins: `<pre>${plugins.formatPlugins().replace(/, /g, '\n')}</pre>`,
|
|
installedParts: `<pre>${plugins.formatParts()}</pre>`,
|
|
installedServerHooks: `<div>${plugins.formatHooks('hooks', true)}</div>`,
|
|
installedClientHooks: `<div>${plugins.formatHooks('client_hooks', true)}</div>`,
|
|
latestVersion: UpdateCheck.getLatestVersion(),
|
|
req,
|
|
}));
|
|
});
|
|
|
|
return cb();
|
|
};
|
|
|
|
exports.socketio = (hookName, args, cb) => {
|
|
const io = args.io.of('/pluginfw/installer');
|
|
io.on('connection', (socket) => {
|
|
const {session: {user: {is_admin: isAdmin} = {}} = {}} = socket.conn.request;
|
|
if (!isAdmin) return;
|
|
|
|
socket.on('getInstalled', (query) => {
|
|
// send currently installed plugins
|
|
const installed =
|
|
Object.keys(pluginDefs.plugins).map((plugin) => pluginDefs.plugins[plugin].package);
|
|
|
|
socket.emit('results:installed', {installed});
|
|
});
|
|
|
|
socket.on('checkUpdates', async () => {
|
|
// Check plugins for updates
|
|
try {
|
|
const results = await installer.getAvailablePlugins(/* maxCacheAge:*/ 60 * 10);
|
|
|
|
const updatable = Object.keys(pluginDefs.plugins).filter((plugin) => {
|
|
if (!results[plugin]) return false;
|
|
|
|
const latestVersion = results[plugin].version;
|
|
const currentVersion = pluginDefs.plugins[plugin].package.version;
|
|
|
|
return semver.gt(latestVersion, currentVersion);
|
|
});
|
|
|
|
socket.emit('results:updatable', {updatable});
|
|
} catch (err) {
|
|
console.warn(err.stack || err.toString());
|
|
|
|
socket.emit('results:updatable', {updatable: {}});
|
|
}
|
|
});
|
|
|
|
socket.on('getAvailable', async (query) => {
|
|
try {
|
|
const results = await installer.getAvailablePlugins(/* maxCacheAge:*/ false);
|
|
socket.emit('results:available', results);
|
|
} catch (er) {
|
|
console.error(er);
|
|
socket.emit('results:available', {});
|
|
}
|
|
});
|
|
|
|
socket.on('search', async (query) => {
|
|
try {
|
|
const results = await installer.search(query.searchTerm, /* maxCacheAge:*/ 60 * 10);
|
|
let res = Object.keys(results)
|
|
.map((pluginName) => results[pluginName])
|
|
.filter((plugin) => !pluginDefs.plugins[plugin.name]);
|
|
res = sortPluginList(res, query.sortBy, query.sortDir)
|
|
.slice(query.offset, query.offset + query.limit);
|
|
socket.emit('results:search', {results: res, query});
|
|
} catch (er) {
|
|
console.error(er);
|
|
|
|
socket.emit('results:search', {results: {}, query});
|
|
}
|
|
});
|
|
|
|
socket.on('install', (pluginName) => {
|
|
installer.install(pluginName, (err) => {
|
|
if (err) console.warn(err.stack || err.toString());
|
|
|
|
socket.emit('finished:install', {
|
|
plugin: pluginName,
|
|
code: err ? err.code : null,
|
|
error: err ? err.message : null,
|
|
});
|
|
});
|
|
});
|
|
|
|
socket.on('uninstall', (pluginName) => {
|
|
installer.uninstall(pluginName, (err) => {
|
|
if (err) console.warn(err.stack || err.toString());
|
|
|
|
socket.emit('finished:uninstall', {plugin: pluginName, error: err ? err.message : null});
|
|
});
|
|
});
|
|
});
|
|
return cb();
|
|
};
|
|
|
|
const sortPluginList = (plugins, property, /* ASC?*/dir) => plugins.sort((a, b) => {
|
|
if (a[property] < b[property]) {
|
|
return dir ? -1 : 1;
|
|
}
|
|
|
|
if (a[property] > b[property]) {
|
|
return dir ? 1 : -1;
|
|
}
|
|
|
|
// a must be equal to b
|
|
return 0;
|
|
});
|