Files
NodeBB/src/socket.io/admin.js

370 lines
9.9 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2014-01-13 12:05:13 -05:00
2019-09-13 18:24:21 -04:00
const async = require('async');
const winston = require('winston');
const fs = require('fs');
const path = require('path');
const nconf = require('nconf');
const meta = require('../meta');
const plugins = require('../plugins');
const widgets = require('../widgets');
const user = require('../user');
const userDigest = require('../user/digest');
const userEmail = require('../user/email');
const logger = require('../logger');
const events = require('../events');
const notifications = require('../notifications');
const emailer = require('../emailer');
const db = require('../database');
const analytics = require('../analytics');
const websockets = require('../socket.io/index');
const index = require('./index');
const getAdminSearchDict = require('../admin/search').getDictionary;
const utils = require('../../public/src/utils');
const SocketAdmin = module.exports;
SocketAdmin.user = require('./admin/user');
SocketAdmin.categories = require('./admin/categories');
SocketAdmin.groups = require('./admin/groups');
SocketAdmin.tags = require('./admin/tags');
SocketAdmin.rewards = require('./admin/rewards');
SocketAdmin.navigation = require('./admin/navigation');
SocketAdmin.rooms = require('./admin/rooms');
SocketAdmin.social = require('./admin/social');
SocketAdmin.themes = {};
SocketAdmin.plugins = {};
SocketAdmin.widgets = {};
SocketAdmin.config = {};
SocketAdmin.settings = {};
SocketAdmin.email = {};
SocketAdmin.analytics = {};
SocketAdmin.logs = {};
SocketAdmin.errors = {};
SocketAdmin.uploads = {};
2014-01-10 16:00:03 -05:00
SocketAdmin.before = function (socket, method, data, next) {
2017-05-26 00:02:20 -04:00
async.waterfall([
function (next) {
user.isAdministrator(socket.uid, next);
},
function (isAdmin) {
if (isAdmin) {
return next();
}
winston.warn('[socket.io] Call to admin method ( ' + method + ' ) blocked (accessed by uid ' + socket.uid + ')');
next(new Error('[[error:no-privileges]]'));
},
], next);
2014-01-13 12:01:42 -05:00
};
SocketAdmin.restart = function (socket, data, callback) {
logRestart(socket);
meta.restart();
callback();
};
function logRestart(socket) {
2015-02-01 19:11:58 -05:00
events.log({
type: 'restart',
uid: socket.uid,
2017-02-17 19:31:21 -07:00
ip: socket.ip,
2015-02-01 19:11:58 -05:00
});
db.setObject('lastrestart', {
uid: socket.uid,
ip: socket.ip,
timestamp: Date.now(),
});
}
2019-09-13 18:24:21 -04:00
SocketAdmin.reload = async function (socket) {
await require('../meta/build').buildAll();
await events.log({
type: 'build',
uid: socket.uid,
ip: socket.ip,
});
logRestart(socket);
meta.restart();
};
SocketAdmin.fireEvent = function (socket, data, callback) {
2015-09-25 16:16:07 -04:00
index.server.emit(data.name, data.payload || {});
2016-03-09 19:13:36 +02:00
callback();
};
SocketAdmin.themes.getInstalled = function (socket, data, callback) {
2014-01-17 12:42:19 -05:00
meta.themes.get(callback);
2014-01-10 16:00:03 -05:00
};
SocketAdmin.themes.set = async function (socket, data) {
2016-03-09 19:13:36 +02:00
if (!data) {
throw new Error('[[error:invalid-data]]');
}
if (data.type === 'local') {
await widgets.reset();
2014-01-17 12:42:19 -05:00
}
2014-03-31 16:19:57 -04:00
data.ip = socket.ip;
data.uid = socket.uid;
2017-06-02 15:01:24 -04:00
await meta.themes.set(data);
2014-05-14 18:05:17 -04:00
};
SocketAdmin.plugins.toggleActive = async function (socket, plugin_id) {
require('../posts/cache').reset();
const data = await plugins.toggleActive(plugin_id);
await events.log({
type: 'plugin-' + (data.active ? 'activate' : 'deactivate'),
text: plugin_id,
uid: socket.uid,
});
return data;
};
SocketAdmin.plugins.toggleInstall = async function (socket, data) {
require('../posts/cache').reset();
const pluginData = await plugins.toggleInstall(data.id, data.version);
await events.log({
type: 'plugin-' + (pluginData.installed ? 'install' : 'uninstall'),
text: data.id,
version: data.version,
uid: socket.uid,
});
return pluginData;
2014-01-10 16:00:03 -05:00
};
SocketAdmin.plugins.getActive = function (socket, data, callback) {
2015-02-23 15:55:35 -05:00
plugins.getActive(callback);
};
2019-09-13 18:24:21 -04:00
SocketAdmin.plugins.orderActivePlugins = async function (socket, data) {
data = data.filter(plugin => plugin && plugin.name);
await Promise.all(data.map(plugin => db.sortedSetAdd('plugins:active', plugin.order || 0, plugin.name)));
2015-02-23 15:55:35 -05:00
};
SocketAdmin.plugins.upgrade = function (socket, data, callback) {
plugins.upgrade(data.id, data.version, callback);
2014-10-09 18:42:05 -04:00
};
SocketAdmin.widgets.set = function (socket, data, callback) {
2017-09-21 14:30:12 -04:00
if (!Array.isArray(data)) {
2014-04-09 21:26:37 -04:00
return callback(new Error('[[error:invalid-data]]'));
}
2017-09-21 14:30:12 -04:00
async.eachSeries(data, widgets.setArea, callback);
};
2019-09-13 18:24:21 -04:00
SocketAdmin.config.set = async function (socket, data) {
2016-03-09 19:13:36 +02:00
if (!data) {
2019-09-13 18:24:21 -04:00
throw new Error('[[error:invalid-data]]');
2014-01-17 12:42:19 -05:00
}
2019-09-13 18:24:21 -04:00
const _data = {};
2016-12-16 15:55:34 +03:00
_data[data.key] = data.value;
2019-09-13 18:24:21 -04:00
await SocketAdmin.config.setMultiple(socket, _data);
2014-01-13 12:05:13 -05:00
};
2019-09-13 18:24:21 -04:00
SocketAdmin.config.setMultiple = async function (socket, data) {
2016-03-09 19:13:36 +02:00
if (!data) {
2019-09-13 18:24:21 -04:00
throw new Error('[[error:invalid-data]]');
}
2018-10-04 13:42:53 -04:00
var changes = {};
2018-12-01 16:59:20 -05:00
data = meta.configs.deserialize(data);
2018-10-04 13:42:53 -04:00
Object.keys(data).forEach(function (key) {
if (data[key] !== meta.config[key]) {
changes[key] = data[key];
changes[key + '_old'] = meta.config[key];
}
});
2019-09-13 18:24:21 -04:00
await meta.configs.setMultiple(data);
var setting;
for (var field in data) {
if (data.hasOwnProperty(field)) {
setting = {
key: field,
value: data[field],
};
plugins.fireHook('action:config.set', setting);
logger.monitorConfig({ io: index.server }, setting);
}
}
if (Object.keys(changes).length) {
changes.type = 'config-change';
changes.uid = socket.uid;
changes.ip = socket.ip;
await events.log(changes);
}
};
SocketAdmin.config.remove = function (socket, key, callback) {
2016-12-16 15:55:34 +03:00
meta.configs.remove(key, callback);
2014-01-13 12:05:13 -05:00
};
SocketAdmin.settings.get = function (socket, data, callback) {
meta.settings.get(data.hash, callback);
};
2019-09-13 18:24:21 -04:00
SocketAdmin.settings.set = async function (socket, data) {
await meta.settings.set(data.hash, data.values);
const eventData = data.values;
eventData.type = 'settings-change';
eventData.uid = socket.uid;
eventData.ip = socket.ip;
eventData.hash = data.hash;
await events.log(eventData);
};
SocketAdmin.settings.clearSitemapCache = function (socket, data, callback) {
2015-01-31 13:10:43 -05:00
require('../sitemap').clearCache();
callback();
};
SocketAdmin.email.test = function (socket, data, callback) {
var payload = {
subject: '[[email:test-email.subject]]',
};
switch (data.template) {
case 'digest':
userDigest.execute({
2017-11-28 15:42:27 -05:00
interval: 'alltime',
subscribers: [socket.uid],
}, callback);
break;
case 'banned':
Object.assign(payload, {
username: 'test-user',
until: utils.toISOString(Date.now()),
reason: 'Test Reason',
});
emailer.send(data.template, socket.uid, payload, callback);
break;
case 'welcome':
userEmail.sendValidationEmail(socket.uid, {
force: 1,
}, callback);
break;
case 'notification':
async.waterfall([
function (next) {
notifications.create({
type: 'test',
bodyShort: '[[email:notif.test.short]]',
bodyLong: '[[email:notif.test.long]]',
nid: 'uid:' + socket.uid + ':test',
path: '/',
from: socket.uid,
}, next);
},
function (notifObj, next) {
emailer.send('notification', socket.uid, {
path: notifObj.path,
subject: utils.stripHTMLTags(notifObj.subject || '[[notifications:new_notification]]'),
intro: utils.stripHTMLTags(notifObj.bodyShort),
body: notifObj.bodyLong || '',
notification: notifObj,
showUnsubscribe: true,
}, next);
},
], callback);
break;
default:
emailer.send(data.template, socket.uid, payload, callback);
break;
}
2014-07-30 17:12:07 -04:00
};
SocketAdmin.analytics.get = function (socket, data, callback) {
2017-02-28 15:16:49 +03:00
if (!data || !data.graph || !data.units) {
return callback(new Error('[[error:invalid-data]]'));
}
// Default returns views from past 24 hours, by hour
if (!data.amount) {
if (data.units === 'days') {
data.amount = 30;
} else {
data.amount = 24;
}
}
2018-10-24 11:24:37 -04:00
const getStats = data.units === 'days' ? analytics.getDailyStatsForSet : analytics.getHourlyStatsForSet;
2017-02-28 15:16:49 +03:00
if (data.graph === 'traffic') {
async.parallel({
uniqueVisitors: function (next) {
2018-10-24 11:24:37 -04:00
getStats('analytics:uniquevisitors', data.until || Date.now(), data.amount, next);
2017-02-28 15:16:49 +03:00
},
pageviews: function (next) {
2018-10-24 11:24:37 -04:00
getStats('analytics:pageviews', data.until || Date.now(), data.amount, next);
},
2018-10-24 11:34:57 -04:00
pageviewsRegistered: function (next) {
2018-10-24 11:24:37 -04:00
getStats('analytics:pageviews:registered', data.until || Date.now(), data.amount, next);
},
2018-10-24 11:34:57 -04:00
pageviewsGuest: function (next) {
2018-10-24 11:24:37 -04:00
getStats('analytics:pageviews:guest', data.until || Date.now(), data.amount, next);
},
2018-10-24 11:34:57 -04:00
pageviewsBot: function (next) {
2018-10-24 11:24:37 -04:00
getStats('analytics:pageviews:bot', data.until || Date.now(), data.amount, next);
2017-02-28 15:16:49 +03:00
},
2017-05-11 16:53:30 -04:00
summary: function (next) {
analytics.getSummary(next);
2017-02-28 15:16:49 +03:00
},
}, function (err, data) {
data.pastDay = data.pageviews.reduce(function (a, b) { return parseInt(a, 10) + parseInt(b, 10); });
data.pageviews[data.pageviews.length - 1] = parseInt(data.pageviews[data.pageviews.length - 1], 10) + analytics.getUnwrittenPageviews();
callback(err, data);
});
}
};
SocketAdmin.logs.get = function (socket, data, callback) {
2014-11-24 12:20:28 -05:00
meta.logs.get(callback);
};
SocketAdmin.logs.clear = function (socket, data, callback) {
2014-11-24 12:20:28 -05:00
meta.logs.clear(callback);
};
SocketAdmin.errors.clear = function (socket, data, callback) {
2016-05-24 23:04:57 -04:00
meta.errors.clear(callback);
};
2018-12-18 13:43:54 -05:00
SocketAdmin.deleteEvents = function (socket, eids, callback) {
events.deleteEvents(eids, callback);
};
SocketAdmin.deleteAllEvents = function (socket, data, callback) {
2015-02-01 19:11:58 -05:00
events.deleteAll(callback);
2014-10-06 18:19:33 -04:00
};
2019-09-13 18:24:21 -04:00
SocketAdmin.getSearchDict = async function (socket) {
const settings = await user.getSettings(socket.uid);
var lang = settings.userLang || meta.config.defaultLang || 'en-GB';
return await getAdminSearchDict(lang);
};
2016-12-15 14:47:42 +03:00
SocketAdmin.deleteAllSessions = function (socket, data, callback) {
user.auth.deleteAllSessions(callback);
};
2017-05-25 15:16:55 -04:00
SocketAdmin.reloadAllSessions = function (socket, data, callback) {
websockets.in('uid_' + socket.uid).emit('event:livereload');
callback();
};
2014-10-07 16:21:12 -04:00
2018-01-26 18:56:17 -05:00
SocketAdmin.uploads.delete = function (socket, pathToFile, callback) {
pathToFile = path.join(nconf.get('upload_path'), pathToFile);
if (!pathToFile.startsWith(nconf.get('upload_path'))) {
return callback(new Error('[[error:invalid-path]]'));
}
fs.unlink(pathToFile, callback);
};
require('../promisify')(SocketAdmin);