2020-12-08 03:20:59 -05:00
|
|
|
'use strict';
|
|
|
|
|
2024-03-09 23:07:09 +01:00
|
|
|
|
|
|
|
import {PadQueryResult, PadSearchQuery} from "../../types/PadSearchQuery";
|
|
|
|
import {PadType} from "../../types/PadType";
|
2024-05-27 15:35:20 -04:00
|
|
|
import log4js from 'log4js';
|
2024-03-09 23:07:09 +01:00
|
|
|
|
2020-12-08 03:20:59 -05:00
|
|
|
const eejs = require('../../eejs');
|
2021-11-10 18:24:54 -05:00
|
|
|
const fsp = require('fs').promises;
|
2020-12-08 03:20:59 -05:00
|
|
|
const hooks = require('../../../static/js/pluginfw/hooks');
|
2021-02-15 01:23:02 -05:00
|
|
|
const plugins = require('../../../static/js/pluginfw/plugins');
|
2020-12-08 03:20:59 -05:00
|
|
|
const settings = require('../../utils/Settings');
|
2024-03-09 23:07:09 +01:00
|
|
|
const UpdateCheck = require('../../utils/UpdateCheck');
|
|
|
|
const padManager = require('../../db/PadManager');
|
|
|
|
const api = require('../../db/API');
|
2024-09-14 15:54:30 +02:00
|
|
|
const cleanup = require('../../utils/Cleanup');
|
2024-03-09 23:07:09 +01:00
|
|
|
|
|
|
|
|
|
|
|
const queryPadLimit = 12;
|
2024-05-27 15:35:20 -04:00
|
|
|
const logger = log4js.getLogger('adminSettings');
|
2012-11-02 13:16:15 +00:00
|
|
|
|
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
exports.socketio = (hookName: string, {io}: any) => {
|
|
|
|
io.of('/settings').on('connection', (socket: any) => {
|
|
|
|
// @ts-ignore
|
|
|
|
const {session: {user: {is_admin: isAdmin} = {}} = {}} = socket.conn.request;
|
|
|
|
if (!isAdmin) return;
|
|
|
|
|
|
|
|
socket.on('load', async (query: string): Promise<any> => {
|
|
|
|
let data;
|
|
|
|
try {
|
|
|
|
data = await fsp.readFile(settings.settingsFilename, 'utf8');
|
|
|
|
} catch (err) {
|
2024-05-27 15:35:20 -04:00
|
|
|
return logger.error(`Error loading settings: ${err}`);
|
2024-04-21 17:58:51 +02:00
|
|
|
}
|
|
|
|
// if showSettingsInAdminPage is set to false, then return NOT_ALLOWED in the result
|
|
|
|
if (settings.showSettingsInAdminPage === false) {
|
|
|
|
socket.emit('settings', {results: 'NOT_ALLOWED'});
|
|
|
|
} else {
|
|
|
|
socket.emit('settings', {results: data});
|
|
|
|
}
|
|
|
|
});
|
2012-11-02 13:16:15 +00:00
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
socket.on('saveSettings', async (newSettings: string) => {
|
2024-05-27 15:35:20 -04:00
|
|
|
logger.info('Admin request to save settings through a socket on /admin/settings');
|
|
|
|
try {
|
|
|
|
await fsp.writeFile(settings.settingsFilename, newSettings);
|
|
|
|
} catch (err) {
|
|
|
|
logger.error(`Error saving settings: ${err}`);
|
|
|
|
}
|
2024-04-21 17:58:51 +02:00
|
|
|
socket.emit('saveprogress', 'saved');
|
|
|
|
});
|
2012-11-02 13:16:15 +00:00
|
|
|
|
2024-03-09 23:07:09 +01:00
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
type ShoutMessage = {
|
|
|
|
message: string,
|
|
|
|
sticky: boolean,
|
|
|
|
}
|
2024-03-09 23:07:09 +01:00
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
socket.on('shout', (message: ShoutMessage) => {
|
|
|
|
const messageToSend = {
|
|
|
|
type: "COLLABROOM",
|
|
|
|
data: {
|
|
|
|
type: "shoutMessage",
|
|
|
|
payload: {
|
|
|
|
message: message,
|
|
|
|
timestamp: Date.now()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-09 23:07:09 +01:00
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
io.of('/settings').emit('shout', messageToSend);
|
|
|
|
io.sockets.emit('shout', messageToSend);
|
|
|
|
})
|
2024-03-09 23:07:09 +01:00
|
|
|
|
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
socket.on('help', () => {
|
|
|
|
const gitCommit = settings.getGitCommit();
|
|
|
|
const epVersion = settings.getEpVersion();
|
|
|
|
|
|
|
|
const hooks: Map<string, Map<string, string>> = plugins.getHooks('hooks', false);
|
|
|
|
const clientHooks: Map<string, Map<string, string>> = plugins.getHooks('client_hooks', false);
|
|
|
|
|
|
|
|
function mapToObject(map: Map<string, any>) {
|
|
|
|
let obj = Object.create(null);
|
|
|
|
for (let [k, v] of map) {
|
|
|
|
if (v instanceof Map) {
|
|
|
|
obj[k] = mapToObject(v);
|
|
|
|
} else {
|
|
|
|
obj[k] = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return obj;
|
2024-03-09 23:07:09 +01:00
|
|
|
}
|
2024-04-21 17:58:51 +02:00
|
|
|
|
|
|
|
socket.emit('reply:help', {
|
|
|
|
gitCommit,
|
|
|
|
epVersion,
|
|
|
|
installedPlugins: plugins.getPlugins(),
|
|
|
|
installedParts: plugins.getParts(),
|
|
|
|
installedServerHooks: mapToObject(hooks),
|
|
|
|
installedClientHooks: mapToObject(clientHooks),
|
|
|
|
latestVersion: UpdateCheck.getLatestVersion(),
|
2024-03-09 23:07:09 +01:00
|
|
|
})
|
2024-04-21 17:58:51 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
socket.on('padLoad', async (query: PadSearchQuery) => {
|
|
|
|
const {padIDs} = await padManager.listAllPads();
|
|
|
|
|
|
|
|
const data: {
|
|
|
|
total: number,
|
|
|
|
results?: PadQueryResult[]
|
|
|
|
} = {
|
|
|
|
total: padIDs.length,
|
|
|
|
};
|
|
|
|
let result: string[] = padIDs;
|
|
|
|
let maxResult;
|
|
|
|
|
|
|
|
// Filter out matches
|
|
|
|
if (query.pattern) {
|
|
|
|
result = result.filter((padName: string) => padName.includes(query.pattern));
|
2024-03-09 23:07:09 +01:00
|
|
|
}
|
2024-04-21 17:58:51 +02:00
|
|
|
|
|
|
|
data.total = result.length;
|
|
|
|
|
|
|
|
maxResult = result.length - 1;
|
|
|
|
if (maxResult < 0) {
|
|
|
|
maxResult = 0;
|
|
|
|
}
|
|
|
|
|
2024-08-06 20:43:10 +02:00
|
|
|
// Reset to default values if out of bounds
|
2024-04-21 17:58:51 +02:00
|
|
|
if (query.offset && query.offset < 0) {
|
|
|
|
query.offset = 0;
|
|
|
|
} else if (query.offset > maxResult) {
|
|
|
|
query.offset = maxResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (query.limit && query.limit < 0) {
|
2024-08-06 20:43:10 +02:00
|
|
|
// Too small
|
2024-04-21 17:58:51 +02:00
|
|
|
query.limit = 0;
|
|
|
|
} else if (query.limit > queryPadLimit) {
|
2024-08-06 20:43:10 +02:00
|
|
|
// Too big
|
2024-04-21 17:58:51 +02:00
|
|
|
query.limit = queryPadLimit;
|
|
|
|
}
|
|
|
|
|
2024-08-06 20:43:10 +02:00
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
if (query.sortBy === 'padName') {
|
|
|
|
result = result.sort((a, b) => {
|
|
|
|
if (a < b) return query.ascending ? -1 : 1;
|
|
|
|
if (a > b) return query.ascending ? 1 : -1;
|
|
|
|
return 0;
|
|
|
|
}).slice(query.offset, query.offset + query.limit);
|
|
|
|
|
|
|
|
data.results = await Promise.all(result.map(async (padName: string) => {
|
|
|
|
const pad = await padManager.getPad(padName);
|
|
|
|
const revisionNumber = pad.getHeadRevisionNumber()
|
|
|
|
const userCount = api.padUsersCount(padName).padUsersCount;
|
|
|
|
const lastEdited = await pad.getLastEdit();
|
|
|
|
|
|
|
|
return {
|
|
|
|
padName,
|
|
|
|
lastEdited,
|
|
|
|
userCount,
|
|
|
|
revisionNumber
|
|
|
|
}
|
|
|
|
}));
|
2024-08-06 20:43:10 +02:00
|
|
|
} else if (query.sortBy === "revisionNumber") {
|
2024-04-21 17:58:51 +02:00
|
|
|
const currentWinners: PadQueryResult[] = []
|
2024-08-06 20:43:10 +02:00
|
|
|
const padMapping = [] as {padId: string, revisionNumber: number}[]
|
2024-04-21 17:58:51 +02:00
|
|
|
for (let res of result) {
|
|
|
|
const pad = await padManager.getPad(res);
|
2024-08-06 20:43:10 +02:00
|
|
|
const revisionNumber = pad.getHeadRevisionNumber()
|
|
|
|
padMapping.push({padId: res, revisionNumber})
|
2024-04-21 17:58:51 +02:00
|
|
|
}
|
2024-08-06 20:43:10 +02:00
|
|
|
padMapping.sort((a, b) => {
|
|
|
|
if (a.revisionNumber < b.revisionNumber) return query.ascending ? -1 : 1;
|
|
|
|
if (a.revisionNumber > b.revisionNumber) return query.ascending ? 1 : -1;
|
|
|
|
return 0;
|
|
|
|
})
|
|
|
|
|
|
|
|
for (const padRetrieval of padMapping.slice(query.offset, query.offset + query.limit)) {
|
|
|
|
let pad = await padManager.getPad(padRetrieval.padId);
|
|
|
|
currentWinners.push({
|
|
|
|
padName: padRetrieval.padId,
|
|
|
|
lastEdited: await pad.getLastEdit(),
|
|
|
|
userCount: api.padUsersCount(pad.padName).padUsersCount,
|
|
|
|
revisionNumber: padRetrieval.revisionNumber
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
data.results = currentWinners;
|
|
|
|
} else if (query.sortBy === "userCount") {
|
|
|
|
const currentWinners: PadQueryResult[] = []
|
|
|
|
const padMapping = [] as {padId: string, userCount: number}[]
|
|
|
|
for (let res of result) {
|
|
|
|
const userCount = api.padUsersCount(res).padUsersCount
|
|
|
|
padMapping.push({padId: res, userCount})
|
|
|
|
}
|
|
|
|
padMapping.sort((a, b) => {
|
|
|
|
if (a.userCount < b.userCount) return query.ascending ? -1 : 1;
|
|
|
|
if (a.userCount > b.userCount) return query.ascending ? 1 : -1;
|
|
|
|
return 0;
|
|
|
|
})
|
|
|
|
|
|
|
|
for (const padRetrieval of padMapping.slice(query.offset, query.offset + query.limit)) {
|
|
|
|
let pad = await padManager.getPad(padRetrieval.padId);
|
|
|
|
currentWinners.push({
|
|
|
|
padName: padRetrieval.padId,
|
|
|
|
lastEdited: await pad.getLastEdit(),
|
|
|
|
userCount: padRetrieval.userCount,
|
|
|
|
revisionNumber: pad.getHeadRevisionNumber()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
data.results = currentWinners;
|
|
|
|
} else if (query.sortBy === "lastEdited") {
|
|
|
|
const currentWinners: PadQueryResult[] = []
|
|
|
|
const padMapping = [] as {padId: string, lastEdited: string}[]
|
|
|
|
for (let res of result) {
|
|
|
|
const pad = await padManager.getPad(res);
|
|
|
|
const lastEdited = await pad.getLastEdit();
|
|
|
|
padMapping.push({padId: res, lastEdited})
|
|
|
|
}
|
|
|
|
padMapping.sort((a, b) => {
|
|
|
|
if (a.lastEdited < b.lastEdited) return query.ascending ? -1 : 1;
|
|
|
|
if (a.lastEdited > b.lastEdited) return query.ascending ? 1 : -1;
|
|
|
|
return 0;
|
|
|
|
})
|
|
|
|
|
|
|
|
for (const padRetrieval of padMapping.slice(query.offset, query.offset + query.limit)) {
|
|
|
|
let pad = await padManager.getPad(padRetrieval.padId);
|
|
|
|
currentWinners.push({
|
|
|
|
padName: padRetrieval.padId,
|
|
|
|
lastEdited: padRetrieval.lastEdited,
|
|
|
|
userCount: api.padUsersCount(pad.padName).padUsersCount,
|
|
|
|
revisionNumber: pad.getHeadRevisionNumber()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
data.results = currentWinners;
|
2024-04-21 17:58:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
socket.emit('results:padLoad', data);
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
socket.on('deletePad', async (padId: string) => {
|
|
|
|
const padExists = await padManager.doesPadExists(padId);
|
|
|
|
if (padExists) {
|
2024-05-27 15:35:20 -04:00
|
|
|
logger.info(`Deleting pad: ${padId}`);
|
2024-04-21 17:58:51 +02:00
|
|
|
const pad = await padManager.getPad(padId);
|
|
|
|
await pad.remove();
|
|
|
|
socket.emit('results:deletePad', padId);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2024-09-14 15:54:30 +02:00
|
|
|
socket.on('cleanupPadRevisions', async (padId: string) => {
|
|
|
|
if (!settings.cleanup.enabled) {
|
|
|
|
socket.emit('results:cleanupPadRevisions', {
|
|
|
|
error: 'Cleanup disabled. Enable cleanup in settings.json: cleanup.enabled => true',
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const padExists = await padManager.doesPadExists(padId);
|
|
|
|
if (padExists) {
|
|
|
|
logger.info(`Cleanup pad revisions: ${padId}`);
|
|
|
|
try {
|
|
|
|
const result = await cleanup.deleteRevisions(padId, settings.cleanup.keepRevisions)
|
|
|
|
if (result) {
|
|
|
|
socket.emit('results:cleanupPadRevisions', {
|
|
|
|
padId: padId,
|
|
|
|
keepRevisions: settings.cleanup.keepRevisions,
|
|
|
|
});
|
|
|
|
logger.info('successful cleaned up pad: ', padId)
|
|
|
|
} else {
|
|
|
|
socket.emit('results:cleanupPadRevisions', {
|
|
|
|
error: 'Error cleaning up pad',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (err: any) {
|
|
|
|
logger.error(`Error in pad ${padId}: ${err.stack || err}`);
|
|
|
|
socket.emit('results:cleanupPadRevisions', {
|
|
|
|
error: err.toString(),
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
socket.on('restartServer', async () => {
|
2024-05-27 15:35:20 -04:00
|
|
|
logger.info('Admin request to restart server through a socket on /admin/settings');
|
2024-04-21 17:58:51 +02:00
|
|
|
settings.reloadSettings();
|
|
|
|
await plugins.update();
|
|
|
|
await hooks.aCallAll('loadSettings', {settings});
|
|
|
|
await hooks.aCallAll('restartServer');
|
|
|
|
});
|
2012-11-02 13:16:15 +00:00
|
|
|
});
|
2020-11-23 13:24:19 -05:00
|
|
|
};
|
2024-03-09 23:07:09 +01:00
|
|
|
|
|
|
|
|
2024-04-21 17:58:51 +02:00
|
|
|
const searchPad = async (query: PadSearchQuery) => {
|
2024-03-09 23:07:09 +01:00
|
|
|
|
|
|
|
}
|