This repository has been archived on 2024-07-22. You can view files and clone it, but cannot push or open issues or pull requests.
Zaimki/server/routes/admin.js

106 lines
3.6 KiB
JavaScript
Raw Normal View History

2020-10-31 13:33:59 -07:00
import { Router } from 'express';
import SQL from 'sql-template-strings';
2020-11-02 12:12:15 -08:00
import avatar from '../avatar';
2020-11-02 12:45:45 -08:00
import {config as socialLoginConfig} from "../social";
2020-11-24 15:54:02 -08:00
import {now, sortByValue} from "../../src/helpers";
2020-10-31 13:33:59 -07:00
const router = Router();
router.get('/admin/users', async (req, res) => {
if (!req.admin) {
return res.status(401).json({error: 'Unauthorised'});
}
const users = await req.db.all(SQL`
2020-11-02 12:12:15 -08:00
SELECT u.id, u.username, u.email, u.roles, u.avatarSource, p.locale
2020-10-31 13:33:59 -07:00
FROM users u
LEFT JOIN profiles p ON p.userId = u.id
ORDER BY u.roles ASC, u.id DESC
2020-10-31 13:33:59 -07:00
`);
2020-11-02 12:45:45 -08:00
const authenticators = await req.db.all(SQL`
SELECT userId, type FROM authenticators
WHERE type IN (`.append(Object.keys(socialLoginConfig).map(k => `'${k}'`).join(',')).append(SQL`)
AND (validUntil IS NULL OR validUntil > ${now()})
`));
2020-10-31 13:33:59 -07:00
const groupedUsers = {};
for (let user of users) {
if (groupedUsers[user.id] === undefined) {
groupedUsers[user.id] = {
...user,
locale: undefined,
profiles: user.locale ? [user.locale] : [],
2020-11-02 12:12:15 -08:00
avatar: await avatar(req.db, user),
2020-11-02 12:45:45 -08:00
socialConnections: [],
2020-10-31 13:33:59 -07:00
}
} else {
groupedUsers[user.id].profiles.push(user.locale);
}
}
2020-11-02 12:45:45 -08:00
for (let auth of authenticators) {
groupedUsers[auth.userId].socialConnections.push(auth.type);
}
2020-10-31 13:33:59 -07:00
return res.json(groupedUsers);
});
2020-11-24 15:54:02 -08:00
router.get('/admin/stats', async (req, res) => {
if (!req.admin) {
return res.status(401).json({error: 'Unauthorised'});
}
const users = {
overall: (await req.db.get(SQL`SELECT count(*) AS c FROM users`)).c,
admins: (await req.db.get(SQL`SELECT count(*) AS c FROM users WHERE roles=${'admin'}`)).c,
};
const locales = {};
for (let locale in req.locales) {
if (!req.locales.hasOwnProperty(locale)) { continue; }
2020-11-27 11:30:21 -08:00
const profiles = await req.db.all(SQL`SELECT pronouns, flags FROM profiles WHERE locale=${locale}`);
2020-11-24 15:54:02 -08:00
const pronouns = {}
2020-11-27 11:30:21 -08:00
const flags = {}
2020-11-24 15:54:02 -08:00
for (let profile of profiles) {
const pr = JSON.parse(profile.pronouns);
for (let pronoun in pr) {
if (!pr.hasOwnProperty(pronoun)) { continue; }
if (pronoun.includes(',') || pr[pronoun] < 0) {
continue;
}
const p = pronoun.replace(/^.*:\/\//, '').replace(/^\//, '').toLowerCase().replace(/^[a-z]+\.[^/]+\//, '');
if (pronouns[p] === undefined) {
pronouns[p] = 0;
}
pronouns[p] += pr[pronoun] === 1 ? 1 : 0.5;
2020-11-24 15:54:02 -08:00
}
2020-11-27 11:30:21 -08:00
const fl = JSON.parse(profile.flags);
for (let flag of fl) {
if (flags[flag] === undefined) {
flags[flag] = 0;
}
flags[flag] += 1;
}
2020-11-24 15:54:02 -08:00
}
locales[locale] = {
name: req.locales[locale].name,
url: req.locales[locale].url,
profiles: profiles.length,
pronouns: sortByValue(pronouns, true),
2020-11-27 11:30:21 -08:00
flags: sortByValue(flags, true),
2020-11-24 15:54:02 -08:00
nouns: {
approved: (await req.db.get(SQL`SELECT count(*) AS c FROM nouns WHERE locale=${locale} AND approved=1 AND deleted=0`)).c,
awaiting: (await req.db.get(SQL`SELECT count(*) AS c FROM nouns WHERE locale=${locale} AND approved=0 AND deleted=0`)).c,
2020-11-24 15:54:02 -08:00
}
};
}
return res.json({ users, locales });
});
2020-10-31 13:33:59 -07:00
export default router;