mirror of
https://github.com/thedevs-network/the-guard-bot
synced 2025-08-22 09:57:50 +00:00
Bump version to 1.5.0 Bump minimum node to latest supported LTS (12->18) Bump all possible deps, notably Telegraf to 4.13 string-replace-async and nedb-promise still need to be dealt with later
191 lines
3.8 KiB
JavaScript
191 lines
3.8 KiB
JavaScript
'use strict';
|
|
|
|
/**
|
|
* @typedef { { id: number } | { username: string } } UserQuery
|
|
* @exports UserQuery
|
|
*/
|
|
|
|
// Utils
|
|
const { strip } = require('../utils/cmd');
|
|
|
|
const Datastore = require('nedb-promise');
|
|
const ms = require('millisecond');
|
|
const R = require('ramda');
|
|
|
|
const User = new Datastore({
|
|
autoload: true,
|
|
filename: 'data/User.db',
|
|
timestampData: true,
|
|
});
|
|
|
|
User.ensureIndex({
|
|
fieldName: 'id',
|
|
unique: true,
|
|
});
|
|
|
|
User.ensureIndex({
|
|
fieldName: 'status',
|
|
});
|
|
|
|
// Migration
|
|
User.update(
|
|
{ username: '' },
|
|
{ $unset: { username: true } },
|
|
{ multi: true },
|
|
).then(() =>
|
|
User.ensureIndex({ fieldName: 'username', sparse: true }));
|
|
|
|
const normalizeTgUser = R.pipe(
|
|
R.pick([ 'first_name', 'id', 'last_name', 'username' ]),
|
|
R.evolve({ username: R.toLower }),
|
|
R.mergeRight({ first_name: '', last_name: '' }),
|
|
);
|
|
|
|
const getUpdatedDocument = R.prop(1);
|
|
|
|
const getUser = user =>
|
|
User.findOne(user);
|
|
|
|
const updateUser = async (rawTgUser) => {
|
|
const tgUser = normalizeTgUser(rawTgUser);
|
|
|
|
const { id, username } = tgUser;
|
|
|
|
const [ rawDbUser ] = await Promise.all([
|
|
getUser({ id }),
|
|
User.update({ $not: { id }, username }, { $unset: { username: true } }),
|
|
]);
|
|
|
|
if (rawDbUser === null) {
|
|
return User.update(
|
|
{ id },
|
|
{ status: 'member', warns: [], ...tgUser },
|
|
{ returnUpdatedDocs: true, upsert: true },
|
|
).then(getUpdatedDocument);
|
|
}
|
|
|
|
const dbUser = rawDbUser;
|
|
|
|
if (!R.whereEq(tgUser, dbUser)) {
|
|
return User.update(
|
|
{ id },
|
|
{ $set: tgUser },
|
|
{ returnUpdatedDocs: true },
|
|
).then(getUpdatedDocument);
|
|
}
|
|
|
|
return dbUser;
|
|
};
|
|
|
|
const admin = ({ id }) =>
|
|
User.update(
|
|
{ id },
|
|
{ $set: { status: 'admin', warns: [] } },
|
|
);
|
|
|
|
const getAdmins = () =>
|
|
User.find({ status: 'admin' });
|
|
|
|
const unadmin = ({ id }) =>
|
|
User.update({ id }, { $set: { status: 'member' } });
|
|
|
|
const isAdmin = (user) => {
|
|
if (!user) return false;
|
|
|
|
if (user.status) return user.status === 'admin';
|
|
|
|
return User.findOne({ id: user.id, status: 'admin' });
|
|
};
|
|
|
|
const ban = ({ id }, ban_details) =>
|
|
User.update(
|
|
{ id, $not: { status: 'admin' } },
|
|
{ $set: { ban_details, status: 'banned' } },
|
|
{ upsert: true },
|
|
);
|
|
|
|
const batchBan = (users, ban_details) =>
|
|
User.update(
|
|
{ $or: users.map(strip), $not: { status: 'admin' } },
|
|
{ $set: { ban_details, status: 'banned' } },
|
|
{ multi: true, returnUpdatedDocs: true },
|
|
).then(getUpdatedDocument);
|
|
|
|
const ensureExists = ({ id }) =>
|
|
id && User.insert({ id, status: 'member', warns: [] }).catch(R.F);
|
|
|
|
const unban = ({ id }) =>
|
|
User.update(
|
|
{ id },
|
|
{
|
|
$set: { status: 'member' },
|
|
$unset: { ban_details: true, ban_reason: true },
|
|
},
|
|
);
|
|
|
|
/**
|
|
* @param {UserQuery} user
|
|
*/
|
|
const permit = (user, { by_id, date }) =>
|
|
User.update(
|
|
user,
|
|
{ $set: { permit: { by_id, date } } },
|
|
{ returnUpdatedDocs: true },
|
|
).then(getUpdatedDocument);
|
|
|
|
/**
|
|
* @param {UserQuery} user
|
|
*/
|
|
permit.revoke = (user) =>
|
|
User.update(
|
|
{ permit: { $exists: true }, ...strip(user) },
|
|
{ $unset: { permit: true } },
|
|
{ returnUpdatedDocs: true },
|
|
).then(getUpdatedDocument);
|
|
|
|
permit.isValid = (p) => Date.now() - ms('24h') < p?.date;
|
|
|
|
const warn = ({ id }, reason, { amend }) =>
|
|
User.update(
|
|
{ id, $not: { status: 'admin' } },
|
|
{
|
|
$pop: { warns: +!!amend },
|
|
$push: { warns: reason },
|
|
$unset: { permit: true },
|
|
},
|
|
{ returnUpdatedDocs: true },
|
|
).then(getUpdatedDocument);
|
|
|
|
const unwarn = ({ id }, warnQuery) =>
|
|
User.update(
|
|
{ id },
|
|
{
|
|
$pull: { warns: warnQuery },
|
|
$set: { status: 'member' },
|
|
$unset: { ban_details: true, ban_reason: true },
|
|
},
|
|
);
|
|
|
|
const nowarns = query => unwarn(query, {});
|
|
|
|
const verifyCaptcha = ({ id }, captcha = true) =>
|
|
User.update({ id }, { $set: { captcha } });
|
|
|
|
module.exports = {
|
|
admin,
|
|
ban,
|
|
batchBan,
|
|
ensureExists,
|
|
getAdmins,
|
|
getUser,
|
|
isAdmin,
|
|
nowarns,
|
|
permit,
|
|
unadmin,
|
|
unban,
|
|
unwarn,
|
|
updateUser,
|
|
verifyCaptcha,
|
|
warn,
|
|
};
|