Files
NodeBB/src/controllers/accounts/settings.js
2023-10-05 12:53:00 -04:00

249 lines
7.7 KiB
JavaScript

'use strict';
const nconf = require('nconf');
const winston = require('winston');
const _ = require('lodash');
const jwt = require('jsonwebtoken');
const util = require('util');
const user = require('../../user');
const languages = require('../../languages');
const meta = require('../../meta');
const plugins = require('../../plugins');
const notifications = require('../../notifications');
const db = require('../../database');
const helpers = require('../helpers');
const accountHelpers = require('./helpers');
const slugify = require('../../slugify');
const settingsController = module.exports;
settingsController.get = async function (req, res, next) {
const userData = await accountHelpers.getUserDataByUserSlug(req.params.userslug, req.uid, req.query);
if (!userData) {
return next();
}
const [settings, languagesData] = await Promise.all([
user.getSettings(userData.uid),
languages.list(),
]);
userData.settings = settings;
userData.languages = languagesData;
if (userData.isAdmin && userData.isSelf) {
userData.acpLanguages = _.cloneDeep(languagesData);
}
const data = await plugins.hooks.fire('filter:user.customSettings', {
settings: settings,
customSettings: [],
uid: req.uid,
});
const [notificationSettings, routes, bsSkinOptions] = await Promise.all([
getNotificationSettings(userData),
getHomePageRoutes(userData),
getSkinOptions(userData),
]);
userData.customSettings = data.customSettings;
userData.homePageRoutes = routes;
userData.bootswatchSkinOptions = bsSkinOptions;
userData.notificationSettings = notificationSettings;
userData.disableEmailSubscriptions = meta.config.disableEmailSubscriptions;
userData.dailyDigestFreqOptions = [
{ value: 'off', name: '[[user:digest-off]]', selected: userData.settings.dailyDigestFreq === 'off' },
{ value: 'day', name: '[[user:digest-daily]]', selected: userData.settings.dailyDigestFreq === 'day' },
{ value: 'week', name: '[[user:digest-weekly]]', selected: userData.settings.dailyDigestFreq === 'week' },
{ value: 'biweek', name: '[[user:digest-biweekly]]', selected: userData.settings.dailyDigestFreq === 'biweek' },
{ value: 'month', name: '[[user:digest-monthly]]', selected: userData.settings.dailyDigestFreq === 'month' },
];
userData.languages.forEach((language) => {
language.selected = language.code === userData.settings.userLang;
});
if (userData.isAdmin && userData.isSelf) {
userData.acpLanguages.forEach((language) => {
language.selected = language.code === userData.settings.acpLang;
});
}
const notifFreqOptions = [
'all',
'first',
'everyTen',
'threshold',
'logarithmic',
'disabled',
];
userData.upvoteNotifFreq = notifFreqOptions.map(
name => ({ name: name, selected: name === userData.settings.upvoteNotifFreq })
);
userData.categoryWatchState = { [userData.settings.categoryWatchState]: true };
userData.disableCustomUserSkins = meta.config.disableCustomUserSkins || 0;
userData.allowUserHomePage = meta.config.allowUserHomePage === 1 ? 1 : 0;
userData.hideFullname = meta.config.hideFullname || 0;
userData.hideEmail = meta.config.hideEmail || 0;
userData.inTopicSearchAvailable = plugins.hooks.hasListeners('filter:topic.search');
userData.maxTopicsPerPage = meta.config.maxTopicsPerPage;
userData.maxPostsPerPage = meta.config.maxPostsPerPage;
userData.title = '[[pages:account/settings]]';
userData.breadcrumbs = helpers.buildBreadcrumbs([{ text: userData.username, url: `/user/${userData.userslug}` }, { text: '[[user:settings]]' }]);
res.render('account/settings', userData);
};
const unsubscribable = ['digest', 'notification'];
const jwtVerifyAsync = util.promisify((token, callback) => {
jwt.verify(token, nconf.get('secret'), (err, payload) => callback(err, payload));
});
const doUnsubscribe = async (payload) => {
if (payload.template === 'digest') {
await Promise.all([
user.setSetting(payload.uid, 'dailyDigestFreq', 'off'),
user.updateDigestSetting(payload.uid, 'off'),
]);
} else if (payload.template === 'notification') {
const current = await db.getObjectField(`user:${payload.uid}:settings`, `notificationType_${payload.type}`);
await user.setSetting(payload.uid, `notificationType_${payload.type}`, (current === 'notificationemail' ? 'notification' : 'none'));
}
return true;
};
settingsController.unsubscribe = async (req, res) => {
try {
const payload = await jwtVerifyAsync(req.params.token);
if (!payload || !unsubscribable.includes(payload.template)) {
return;
}
await doUnsubscribe(payload);
res.render('unsubscribe', {
payload,
});
} catch (err) {
res.render('unsubscribe', {
error: err.message,
});
}
};
settingsController.unsubscribePost = async function (req, res) {
let payload;
try {
payload = await jwtVerifyAsync(req.params.token);
if (!payload || !unsubscribable.includes(payload.template)) {
return res.sendStatus(404);
}
} catch (err) {
return res.sendStatus(403);
}
try {
await doUnsubscribe(payload);
res.sendStatus(200);
} catch (err) {
winston.error(`[settings/unsubscribe] One-click unsubscribe failed with error: ${err.message}`);
res.sendStatus(500);
}
};
async function getNotificationSettings(userData) {
const privilegedTypes = [];
const privileges = await user.getPrivileges(userData.uid);
if (privileges.isAdmin) {
privilegedTypes.push('notificationType_new-register');
}
if (privileges.isAdmin || privileges.isGlobalMod || privileges.isModeratorOfAnyCategory) {
privilegedTypes.push('notificationType_post-queue', 'notificationType_new-post-flag');
}
if (privileges.isAdmin || privileges.isGlobalMod) {
privilegedTypes.push('notificationType_new-user-flag');
}
const results = await plugins.hooks.fire('filter:user.notificationTypes', {
types: notifications.baseTypes.slice(),
privilegedTypes: privilegedTypes,
});
function modifyType(type) {
const setting = userData.settings[type];
return {
name: type,
label: `[[notifications:${type.replace(/_/g, '-')}]]`,
none: setting === 'none',
notification: setting === 'notification',
email: setting === 'email',
notificationemail: setting === 'notificationemail',
};
}
if (meta.config.disableChat) {
results.types = results.types.filter(type => type !== 'notificationType_new-chat');
}
return results.types.map(modifyType).concat(results.privilegedTypes.map(modifyType));
}
async function getHomePageRoutes(userData) {
let routes = await helpers.getHomePageRoutes(userData.uid);
// Set selected for each route
let customIdx;
let hasSelected = false;
routes = routes.map((route, idx) => {
if (route.route === userData.settings.homePageRoute) {
route.selected = true;
hasSelected = true;
} else {
route.selected = false;
}
if (route.route === 'custom') {
customIdx = idx;
}
return route;
});
if (!hasSelected && customIdx && userData.settings.homePageRoute !== 'none') {
routes[customIdx].selected = true;
}
return routes;
}
async function getSkinOptions(userData) {
const defaultSkin = _.capitalize(meta.config.bootswatchSkin) || '[[user:no-skin]]';
const bootswatchSkinOptions = [
{ name: '[[user:no-skin]]', value: 'noskin' },
{ name: `[[user:default, ${defaultSkin}]]`, value: '' },
];
const customSkins = await meta.settings.get('custom-skins');
if (customSkins && Array.isArray(customSkins['custom-skin-list'])) {
customSkins['custom-skin-list'].forEach((customSkin) => {
bootswatchSkinOptions.push({
name: customSkin['custom-skin-name'],
value: slugify(customSkin['custom-skin-name']),
});
});
}
bootswatchSkinOptions.push(
...meta.css.supportedSkins.map(skin => ({ name: _.capitalize(skin), value: skin }))
);
bootswatchSkinOptions.forEach((skin) => {
skin.selected = skin.value === userData.settings.bootswatchSkin;
});
return bootswatchSkinOptions;
}