2017-02-18 01:56:23 -07:00
|
|
|
'use strict';
|
2014-03-12 17:59:29 -04:00
|
|
|
|
2020-07-09 23:57:54 -04:00
|
|
|
const validator = require('validator');
|
2016-10-07 17:35:24 +03:00
|
|
|
|
2019-08-14 22:48:57 -04:00
|
|
|
const user = require('../../user');
|
|
|
|
|
const meta = require('../../meta');
|
|
|
|
|
const db = require('../../database');
|
|
|
|
|
const pagination = require('../../pagination');
|
|
|
|
|
const events = require('../../events');
|
|
|
|
|
const plugins = require('../../plugins');
|
2020-11-16 22:47:23 +03:00
|
|
|
const privileges = require('../../privileges');
|
2019-08-14 22:48:57 -04:00
|
|
|
const utils = require('../../utils');
|
2014-03-12 17:59:29 -04:00
|
|
|
|
2019-08-14 22:48:57 -04:00
|
|
|
const usersController = module.exports;
|
2014-03-12 17:59:29 -04:00
|
|
|
|
2020-10-13 00:29:38 -04:00
|
|
|
const userFields = [
|
|
|
|
|
'uid', 'username', 'userslug', 'email', 'postcount', 'joindate', 'banned',
|
2020-10-13 23:02:49 -04:00
|
|
|
'reputation', 'picture', 'flags', 'lastonline', 'email:confirmed',
|
2020-10-13 00:29:38 -04:00
|
|
|
];
|
|
|
|
|
|
|
|
|
|
usersController.index = async function (req, res) {
|
|
|
|
|
if (req.query.query) {
|
|
|
|
|
await usersController.search(req, res);
|
|
|
|
|
} else {
|
2020-10-13 01:49:12 -04:00
|
|
|
await getUsers(req, res);
|
2020-10-13 00:29:38 -04:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2020-10-13 01:49:12 -04:00
|
|
|
async function getUsers(req, res) {
|
2020-10-13 00:29:38 -04:00
|
|
|
const sortDirection = req.query.sortDirection || 'desc';
|
|
|
|
|
const reverse = sortDirection === 'desc';
|
|
|
|
|
|
|
|
|
|
const page = parseInt(req.query.page, 10) || 1;
|
|
|
|
|
let resultsPerPage = parseInt(req.query.resultsPerPage, 10) || 50;
|
|
|
|
|
if (![50, 100, 250, 500].includes(resultsPerPage)) {
|
|
|
|
|
resultsPerPage = 50;
|
|
|
|
|
}
|
2020-10-13 01:49:12 -04:00
|
|
|
let sortBy = validator.escape(req.query.sortBy || '');
|
2020-10-13 21:04:07 -04:00
|
|
|
const filterBy = Array.isArray(req.query.filters || []) ? (req.query.filters || []) : [req.query.filters];
|
2020-10-13 00:29:38 -04:00
|
|
|
const start = Math.max(0, page - 1) * resultsPerPage;
|
|
|
|
|
const stop = start + resultsPerPage - 1;
|
|
|
|
|
|
|
|
|
|
function buildSet() {
|
|
|
|
|
const sortToSet = {
|
|
|
|
|
postcount: 'users:postcount',
|
|
|
|
|
reputation: 'users:reputation',
|
|
|
|
|
joindate: 'users:joindate',
|
2020-10-16 23:30:22 -04:00
|
|
|
lastonline: 'users:online',
|
2020-10-13 00:29:38 -04:00
|
|
|
flags: 'users:flags',
|
|
|
|
|
};
|
|
|
|
|
|
2020-10-13 01:49:12 -04:00
|
|
|
const set = [];
|
|
|
|
|
if (sortBy) {
|
|
|
|
|
set.push(sortToSet[sortBy]);
|
|
|
|
|
}
|
2020-10-13 14:55:56 -04:00
|
|
|
if (filterBy.includes('unverified')) {
|
|
|
|
|
set.push('group:unverified-users:members');
|
|
|
|
|
}
|
|
|
|
|
if (filterBy.includes('verified')) {
|
|
|
|
|
set.push('group:verified-users:members');
|
2020-10-13 00:29:38 -04:00
|
|
|
}
|
|
|
|
|
if (filterBy.includes('banned')) {
|
|
|
|
|
set.push('users:banned');
|
|
|
|
|
}
|
2020-10-13 01:49:12 -04:00
|
|
|
if (!set.length) {
|
|
|
|
|
set.push('users:online');
|
2020-10-19 18:56:39 -04:00
|
|
|
sortBy = 'lastonline';
|
2020-10-13 01:49:12 -04:00
|
|
|
}
|
2020-10-13 00:29:38 -04:00
|
|
|
return set.length > 1 ? set : set[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getCount(set) {
|
|
|
|
|
if (Array.isArray(set)) {
|
|
|
|
|
return await db.sortedSetIntersectCard(set);
|
|
|
|
|
}
|
|
|
|
|
return await db.sortedSetCard(set);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getUids(set) {
|
|
|
|
|
let uids = [];
|
|
|
|
|
if (Array.isArray(set)) {
|
|
|
|
|
const weights = set.map((s, index) => (index ? 0 : 1));
|
|
|
|
|
uids = await db[reverse ? 'getSortedSetRevIntersect' : 'getSortedSetIntersect']({
|
|
|
|
|
sets: set,
|
|
|
|
|
start: start,
|
|
|
|
|
stop: stop,
|
|
|
|
|
weights: weights,
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
uids = await db[reverse ? 'getSortedSetRevRange' : 'getSortedSetRange'](set, start, stop);
|
|
|
|
|
}
|
|
|
|
|
return uids;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getUsersWithFields(set) {
|
|
|
|
|
const uids = await getUids(set);
|
2020-10-13 22:57:18 -04:00
|
|
|
const [isAdmin, userData, lastonline] = await Promise.all([
|
2020-10-13 00:29:38 -04:00
|
|
|
user.isAdministrator(uids),
|
|
|
|
|
user.getUsersWithFields(uids, userFields, req.uid),
|
2020-10-13 22:57:18 -04:00
|
|
|
db.sortedSetScores('users:online', uids),
|
2020-10-13 00:29:38 -04:00
|
|
|
]);
|
|
|
|
|
userData.forEach((user, index) => {
|
|
|
|
|
if (user) {
|
|
|
|
|
user.administrator = isAdmin[index];
|
2020-10-13 23:40:04 -04:00
|
|
|
const timestamp = lastonline[index] || user.joindate;
|
2020-10-13 23:02:49 -04:00
|
|
|
user.lastonline = timestamp;
|
2020-10-13 23:40:04 -04:00
|
|
|
user.lastonlineISO = utils.toISOString(timestamp);
|
2020-10-13 00:29:38 -04:00
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return userData;
|
|
|
|
|
}
|
|
|
|
|
const set = buildSet();
|
|
|
|
|
const [count, users] = await Promise.all([
|
|
|
|
|
getCount(set),
|
|
|
|
|
getUsersWithFields(set),
|
|
|
|
|
]);
|
|
|
|
|
|
2020-11-16 22:47:23 +03:00
|
|
|
await render(req, res, {
|
2020-10-13 00:29:38 -04:00
|
|
|
users: users.filter(user => user && parseInt(user.uid, 10)),
|
|
|
|
|
page: page,
|
|
|
|
|
pageCount: Math.max(1, Math.ceil(count / resultsPerPage)),
|
|
|
|
|
resultsPerPage: resultsPerPage,
|
|
|
|
|
reverse: reverse,
|
|
|
|
|
sortBy: sortBy,
|
2020-10-13 01:29:21 -04:00
|
|
|
});
|
2020-10-13 00:29:38 -04:00
|
|
|
}
|
2016-10-07 17:35:24 +03:00
|
|
|
|
2020-07-09 23:57:54 -04:00
|
|
|
usersController.search = async function (req, res) {
|
2020-10-13 01:29:21 -04:00
|
|
|
const sortDirection = req.query.sortDirection || 'desc';
|
|
|
|
|
const reverse = sortDirection === 'desc';
|
2020-07-09 23:57:54 -04:00
|
|
|
const page = parseInt(req.query.page, 10) || 1;
|
|
|
|
|
let resultsPerPage = parseInt(req.query.resultsPerPage, 10) || 50;
|
|
|
|
|
if (![50, 100, 250, 500].includes(resultsPerPage)) {
|
|
|
|
|
resultsPerPage = 50;
|
|
|
|
|
}
|
2020-10-13 20:37:38 -04:00
|
|
|
|
2020-07-09 23:57:54 -04:00
|
|
|
const searchData = await user.search({
|
|
|
|
|
uid: req.uid,
|
|
|
|
|
query: req.query.query,
|
|
|
|
|
searchBy: req.query.searchBy,
|
2020-10-13 01:29:21 -04:00
|
|
|
sortBy: req.query.sortBy,
|
|
|
|
|
sortDirection: sortDirection,
|
2020-10-13 20:37:38 -04:00
|
|
|
filters: req.query.filters,
|
2020-07-09 23:57:54 -04:00
|
|
|
page: page,
|
|
|
|
|
resultsPerPage: resultsPerPage,
|
|
|
|
|
findUids: async function (query, searchBy, hardCap) {
|
|
|
|
|
if (!query || query.length < 2) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
2020-12-16 10:41:45 -05:00
|
|
|
query = String(query).toLowerCase();
|
2020-07-09 23:57:54 -04:00
|
|
|
if (!query.endsWith('*')) {
|
|
|
|
|
query += '*';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const data = await db.getSortedSetScan({
|
2021-02-03 23:59:08 -07:00
|
|
|
key: `${searchBy}:sorted`,
|
2020-07-09 23:57:54 -04:00
|
|
|
match: query,
|
2020-12-16 10:41:45 -05:00
|
|
|
limit: hardCap || (resultsPerPage * 10),
|
2020-07-09 23:57:54 -04:00
|
|
|
});
|
2020-09-11 23:20:49 -04:00
|
|
|
return data.map(data => data.split(':').pop());
|
2020-07-09 23:57:54 -04:00
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const uids = searchData.users.map(user => user && user.uid);
|
|
|
|
|
const userInfo = await user.getUsersFields(uids, ['email', 'flags', 'lastonline', 'joindate']);
|
|
|
|
|
|
2021-02-04 00:01:39 -07:00
|
|
|
searchData.users.forEach((user, index) => {
|
2020-07-09 23:57:54 -04:00
|
|
|
if (user && userInfo[index]) {
|
|
|
|
|
user.email = userInfo[index].email;
|
|
|
|
|
user.flags = userInfo[index].flags || 0;
|
|
|
|
|
user.lastonlineISO = userInfo[index].lastonlineISO;
|
|
|
|
|
user.joindateISO = userInfo[index].joindateISO;
|
|
|
|
|
}
|
2014-03-12 17:59:29 -04:00
|
|
|
});
|
2020-07-09 23:57:54 -04:00
|
|
|
searchData.query = validator.escape(String(req.query.query || ''));
|
|
|
|
|
searchData.resultsPerPage = resultsPerPage;
|
2020-10-13 01:29:21 -04:00
|
|
|
searchData.sortBy = req.query.sortBy;
|
|
|
|
|
searchData.reverse = reverse;
|
2020-11-16 22:47:23 +03:00
|
|
|
await render(req, res, searchData);
|
2014-10-03 14:14:41 -04:00
|
|
|
};
|
|
|
|
|
|
2019-08-14 22:48:57 -04:00
|
|
|
usersController.registrationQueue = async function (req, res) {
|
|
|
|
|
const page = parseInt(req.query.page, 10) || 1;
|
|
|
|
|
const itemsPerPage = 20;
|
|
|
|
|
const start = (page - 1) * 20;
|
|
|
|
|
const stop = start + itemsPerPage - 1;
|
|
|
|
|
|
|
|
|
|
const data = await utils.promiseParallel({
|
|
|
|
|
registrationQueueCount: db.sortedSetCard('registration:queue'),
|
|
|
|
|
users: user.getRegistrationQueue(start, stop),
|
2020-11-20 16:06:26 -05:00
|
|
|
customHeaders: plugins.hooks.fire('filter:admin.registrationQueue.customHeaders', { headers: [] }),
|
2019-08-14 22:48:57 -04:00
|
|
|
invites: getInvites(),
|
|
|
|
|
});
|
2021-02-04 00:06:15 -07:00
|
|
|
const pageCount = Math.max(1, Math.ceil(data.registrationQueueCount / itemsPerPage));
|
2019-08-14 22:48:57 -04:00
|
|
|
data.pagination = pagination.create(page, pageCount);
|
|
|
|
|
data.customHeaders = data.customHeaders.headers;
|
|
|
|
|
res.render('admin/manage/registration', data);
|
2015-06-27 21:26:19 -04:00
|
|
|
};
|
|
|
|
|
|
2019-08-14 22:48:57 -04:00
|
|
|
async function getInvites() {
|
|
|
|
|
const invitations = await user.getAllInvites();
|
|
|
|
|
const uids = invitations.map(invite => invite.uid);
|
|
|
|
|
let usernames = await user.getUsersFields(uids, ['username']);
|
|
|
|
|
usernames = usernames.map(user => user.username);
|
|
|
|
|
|
2021-02-04 00:01:39 -07:00
|
|
|
invitations.forEach((invites, index) => {
|
2019-08-14 22:48:57 -04:00
|
|
|
invites.username = usernames[index];
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
async function getUsernamesByEmails(emails) {
|
2020-10-18 16:54:42 -04:00
|
|
|
const uids = await db.sortedSetScores('email:uid', emails.map(email => String(email).toLowerCase()));
|
2019-08-14 22:48:57 -04:00
|
|
|
const usernames = await user.getUsersFields(uids, ['username']);
|
|
|
|
|
return usernames.map(user => user.username);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
usernames = await Promise.all(invitations.map(invites => getUsernamesByEmails(invites.invitations)));
|
|
|
|
|
|
2021-02-04 00:01:39 -07:00
|
|
|
invitations.forEach((invites, index) => {
|
|
|
|
|
invites.invitations = invites.invitations.map((email, i) => ({
|
|
|
|
|
email: email,
|
|
|
|
|
username: usernames[index][i] === '[[global:guest]]' ? '' : usernames[index][i],
|
|
|
|
|
}));
|
2019-08-14 22:48:57 -04:00
|
|
|
});
|
|
|
|
|
return invitations;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-16 22:47:23 +03:00
|
|
|
async function render(req, res, data) {
|
2016-01-20 16:12:57 +02:00
|
|
|
data.pagination = pagination.create(data.page, data.pageCount, req.query);
|
2016-04-27 11:01:27 +03:00
|
|
|
|
2021-02-06 14:10:15 -07:00
|
|
|
const { registrationType } = meta.config;
|
2016-04-27 11:01:27 +03:00
|
|
|
|
|
|
|
|
data.inviteOnly = registrationType === 'invite-only' || registrationType === 'admin-invite-only';
|
|
|
|
|
data.adminInviteOnly = registrationType === 'admin-invite-only';
|
2021-02-03 23:59:08 -07:00
|
|
|
data[`sort_${data.sortBy}`] = true;
|
2020-10-13 21:04:07 -04:00
|
|
|
if (req.query.searchBy) {
|
2021-02-03 23:59:08 -07:00
|
|
|
data[`searchBy_${validator.escape(String(req.query.searchBy))}`] = true;
|
2020-10-13 21:04:07 -04:00
|
|
|
}
|
|
|
|
|
const filterBy = Array.isArray(req.query.filters || []) ? (req.query.filters || []) : [req.query.filters];
|
2021-02-04 00:01:39 -07:00
|
|
|
filterBy.forEach((filter) => {
|
2021-02-03 23:59:08 -07:00
|
|
|
data[`filterBy_${validator.escape(String(filter))}`] = true;
|
2020-10-13 17:07:26 -04:00
|
|
|
});
|
2020-11-27 15:51:56 -05:00
|
|
|
data.userCount = parseInt(await db.getObjectField('global', 'userCount'), 10);
|
2020-11-16 22:47:23 +03:00
|
|
|
if (data.adminInviteOnly) {
|
|
|
|
|
data.showInviteButton = await privileges.users.isAdministrator(req.uid);
|
2020-11-16 15:42:39 -05:00
|
|
|
} else {
|
|
|
|
|
data.showInviteButton = await privileges.users.hasInvitePrivilege(req.uid);
|
2020-11-16 22:47:23 +03:00
|
|
|
}
|
|
|
|
|
|
2016-01-20 16:12:57 +02:00
|
|
|
res.render('admin/manage/users', data);
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-27 15:38:22 -05:00
|
|
|
usersController.getCSV = async function (req, res, next) {
|
|
|
|
|
await events.log({
|
2016-08-19 12:40:52 -04:00
|
|
|
type: 'getUsersCSV',
|
2018-01-31 15:20:17 -05:00
|
|
|
uid: req.uid,
|
2017-02-17 19:31:21 -07:00
|
|
|
ip: req.ip,
|
2016-08-19 12:40:52 -04:00
|
|
|
});
|
2020-11-27 15:38:22 -05:00
|
|
|
const path = require('path');
|
|
|
|
|
const { baseDir } = require('../../constants').paths;
|
|
|
|
|
res.sendFile('users.csv', {
|
|
|
|
|
root: path.join(baseDir, 'build/export'),
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'text/csv',
|
|
|
|
|
'Content-Disposition': 'attachment; filename=users.csv',
|
|
|
|
|
},
|
2021-02-04 00:01:39 -07:00
|
|
|
}, (err) => {
|
2020-11-27 15:38:22 -05:00
|
|
|
if (err) {
|
|
|
|
|
if (err.code === 'ENOENT') {
|
|
|
|
|
res.locals.isAPI = false;
|
|
|
|
|
return next();
|
|
|
|
|
}
|
|
|
|
|
return next(err);
|
|
|
|
|
}
|
|
|
|
|
});
|
2014-03-12 17:59:29 -04:00
|
|
|
};
|