Files
habitica/website/server/libs/chatReporting/groupChatReporter.js
2019-10-10 20:11:50 +02:00

107 lines
3.6 KiB
JavaScript

import nconf from 'nconf';
import moment from 'moment';
import ChatReporter from './chatReporter';
import {
BadRequest,
NotFound,
} from '../errors';
import { sendTxn } from '../email';
import * as slack from '../slack';
import { model as Group } from '../../models/group';
import { chatModel as Chat } from '../../models/message';
import apiError from '../apiError';
const COMMUNITY_MANAGER_EMAIL = nconf.get('EMAILS_COMMUNITY_MANAGER_EMAIL');
const FLAG_REPORT_EMAILS = nconf
.get('FLAG_REPORT_EMAIL')
.split(',')
.map(email => ({ email, canSend: true }));
const USER_AGE_FOR_FLAGGING = 3; // accounts less than this many days old don't increment flagCount
export default class GroupChatReporter extends ChatReporter {
constructor (req, res) {
super(req, res);
this.user = res.locals.user;
this.groupId = req.params.groupId;
}
async validate () {
this.req.checkParams('groupId', apiError('groupIdRequired')).notEmpty();
this.req.checkParams('chatId', apiError('chatIdRequired')).notEmpty();
const validationErrors = this.req.validationErrors();
if (validationErrors) throw validationErrors;
const group = await Group.getGroup({
user: this.user,
groupId: this.groupId,
optionalMembership: this.user.contributor.admin,
});
if (!group) throw new NotFound(this.res.t('groupNotFound'));
const message = await Chat.findOne({ _id: this.req.params.chatId }).exec();
if (!message) throw new NotFound(this.res.t('messageGroupChatNotFound'));
if (message.uuid === 'system') throw new BadRequest(this.res.t('messageCannotFlagSystemMessages', { communityManagerEmail: COMMUNITY_MANAGER_EMAIL }));
const userComment = this.req.body.comment;
return { message, group, userComment };
}
async notify (group, message, userComment, automatedComment = '') {
let emailVariables = await this.getMessageVariables(group, message);
emailVariables = emailVariables.concat([
{ name: 'REPORTER_COMMENT', content: userComment || '' },
]);
sendTxn(FLAG_REPORT_EMAILS, 'flag-report-to-mods-with-comments', emailVariables);
slack.sendFlagNotification({
authorEmail: this.authorEmail,
flagger: this.user,
group,
message,
userComment,
automatedComment,
});
}
async flagGroupMessage (group, message, increaseFlagCount) {
// Log user ids that have flagged the message
if (!message.flags) message.flags = {};
// TODO fix error type
if (message.flags[this.user._id] && !this.user.contributor.admin) throw new NotFound(this.res.t('messageGroupChatFlagAlreadyReported'));
message.flags[this.user._id] = true;
message.markModified('flags');
// Log total number of flags (publicly viewable)
if (!message.flagCount) message.flagCount = 0;
if (this.user.contributor.admin) {
// Arbitrary amount, higher than 2
message.flagCount = 5;
} else if (increaseFlagCount) {
message.flagCount += 1;
}
await message.save();
}
async flag () {
const { message, group, userComment } = await this.validate();
let increaseFlagCount = true;
let automatedComment = '';
if (moment().diff(this.user.auth.timestamps.created, 'days') < USER_AGE_FOR_FLAGGING) {
increaseFlagCount = false;
automatedComment = `The post's flag count has not been increased because the flagger's account is less than ${USER_AGE_FOR_FLAGGING} days old.`;
// This is to prevent trolls from making new accounts to maliciously flag-and-hide.
}
await this.notify(group, message, userComment, automatedComment);
await this.flagGroupMessage(group, message, increaseFlagCount);
return message;
}
}