import find from 'lodash/find'; import moment from 'moment'; import { IncomingWebhook } from '@slack/webhook'; import { createAndPopulateGroup, translate as t, } from '../../../../helpers/api-integration/v3'; describe('POST /chat/:chatId/flag', () => { let user; let admin; let anotherUser; let newUser; let group; let members; let userToDelete; const TEST_MESSAGE = 'Test Message'; const USER_AGE_FOR_FLAGGING = 3; beforeEach(async () => { ({ group, groupLeader: user, members } = await createAndPopulateGroup({ groupDetails: { name: 'Test Guild', type: 'guild', privacy: 'private', }, leaderDetails: { 'auth.timestamps.created': moment().subtract(USER_AGE_FOR_FLAGGING + 1, 'days').toDate(), }, members: 4, upgradeToGroupPlan: true, })); [admin, anotherUser, newUser, userToDelete] = members; await user.updateOne({ permissions: {} }); await admin.updateOne({ permissions: { moderator: true } }); await anotherUser.updateOne({ 'auth.timestamps.created': moment().subtract(USER_AGE_FOR_FLAGGING + 1, 'days').toDate() }); await newUser.updateOne({ 'auth.timestamps.created': moment().subtract(1, 'days').toDate() }); await userToDelete.updateOne({ 'auth.timestamps.created': moment().subtract(1, 'days').toDate(), 'purchased.plan.dateTerminated': moment().subtract(1, 'minutes').toDate(), }); sandbox.stub(IncomingWebhook.prototype, 'send').returns(Promise.resolve()); }); afterEach(() => { sandbox.restore(); }); it('Returns an error when chat message is not found', async () => { await expect(user.post(`/groups/${group._id}/chat/incorrectMessage/flag`)) .to.eventually.be.rejected.and.eql({ code: 404, error: 'NotFound', message: t('messageGroupChatNotFound'), }); }); it('Allows players to flag their own message', async () => { const message = await user.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); await expect(user.post(`/groups/${group._id}/chat/${message.message.id}/flag`)).to.eventually.be.ok; }); it('Flags a chat and sends normal message to moderator Slack when user is not new', async () => { const { message } = await anotherUser.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); const flagResult = await user.post(`/groups/${group._id}/chat/${message.id}/flag`); expect(flagResult.flags[user._id]).to.equal(true); expect(flagResult.flagCount).to.equal(1); const groupWithFlags = await admin.get(`/groups/${group._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck.flags[user._id]).to.equal(true); // Slack message to mods const timestamp = `${moment(message.timestamp).utc().format('YYYY-MM-DD HH:mm')} UTC`; /* eslint-disable camelcase */ expect(IncomingWebhook.prototype.send).to.be.calledWith({ text: `${user.profile.name} (${user.id}; language: en) flagged a group message`, attachments: [{ fallback: 'Flag Message', color: 'danger', author_name: `@${anotherUser.auth.local.username} ${anotherUser.profile.name} (${anotherUser.auth.local.email}; ${anotherUser._id})\n${timestamp}`, title: 'Flag in Test Guild - (private guild)', title_link: undefined, text: TEST_MESSAGE, footer: ``, mrkdwn_in: [ 'text', ], }], }); /* eslint-enable camelcase */ }); it('Does not increment message flag count and sends different message to moderator Slack when user is new', async () => { const automatedComment = `The post's flag count has not been increased because the flagger's account is less than ${USER_AGE_FOR_FLAGGING} days old.`; const { message } = await newUser.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); const flagResult = await newUser.post(`/groups/${group._id}/chat/${message.id}/flag`); expect(flagResult.flags[newUser._id]).to.equal(true); expect(flagResult.flagCount).to.equal(0); const groupWithFlags = await admin.get(`/groups/${group._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck.flags[newUser._id]).to.equal(true); // Slack message to mods const timestamp = `${moment(message.timestamp).utc().format('YYYY-MM-DD HH:mm')} UTC`; /* eslint-disable camelcase */ expect(IncomingWebhook.prototype.send).to.be.calledWith({ text: `${newUser.profile.name} (${newUser.id}; language: en) flagged a group message`, attachments: [{ fallback: 'Flag Message', color: 'danger', author_name: `@${newUser.auth.local.username} ${newUser.profile.name} (${newUser.auth.local.email}; ${newUser._id})\n${timestamp}`, title: 'Flag in Test Guild - (private guild)', title_link: undefined, text: TEST_MESSAGE, footer: ` ${automatedComment}`, mrkdwn_in: [ 'text', ], }], }); /* eslint-enable camelcase */ }); it('Flags a chat when the author\'s account was deleted', async () => { const { message } = await userToDelete.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); await userToDelete.del('/user', { password: 'password', }); const flagResult = await user.post(`/groups/${group._id}/chat/${message.id}/flag`); expect(flagResult.flags[user._id]).to.equal(true); expect(flagResult.flagCount).to.equal(1); const groupWithFlags = await admin.get(`/groups/${group._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck.flags[user._id]).to.equal(true); }); it('Flags a chat with a higher flag acount when an admin flags the message', async () => { const { message } = await user.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); const flagResult = await admin.post(`/groups/${group._id}/chat/${message.id}/flag`); expect(flagResult.flags[admin._id]).to.equal(true); expect(flagResult.flagCount).to.equal(5); const groupWithFlags = await admin.get(`/groups/${group._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck.flags[admin._id]).to.equal(true); expect(messageToCheck.flagCount).to.equal(5); }); it('allows admin to flag a message in a private group', async () => { const privateGroup = await user.post('/groups', { name: 'Test party', type: 'party', privacy: 'private', }); await user.post(`/groups/${privateGroup._id}/invite`, { uuids: [anotherUser._id], }); await anotherUser.post(`/groups/${privateGroup._id}/join`); const { message } = await user.post(`/groups/${privateGroup._id}/chat`, { message: TEST_MESSAGE }); const flagResult = await admin.post(`/groups/${privateGroup._id}/chat/${message.id}/flag`); expect(flagResult.flags[admin._id]).to.equal(true); expect(flagResult.flagCount).to.equal(5); const groupWithFlags = await anotherUser.get(`/groups/${privateGroup._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck).to.not.exist; }); it('does not allow non member to flag message in private group', async () => { const privateGroup = await user.post('/groups', { name: 'Test party', type: 'party', privacy: 'private', }); const { message } = await user.post(`/groups/${privateGroup._id}/chat`, { message: TEST_MESSAGE }); await expect(anotherUser.post(`/groups/${privateGroup._id}/chat/${message.id}/flag`)) .to.eventually.be.rejected.and.eql({ code: 404, error: 'NotFound', message: t('groupNotFound'), }); }); it('Returns an error when user tries to flag a message that they already flagged', async () => { const { message } = await anotherUser.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); await user.post(`/groups/${group._id}/chat/${message.id}/flag`); await expect(user.post(`/groups/${group._id}/chat/${message.id}/flag`)) .to.eventually.be.rejected.and.eql({ code: 404, error: 'NotFound', message: t('messageGroupChatFlagAlreadyReported'), }); }); it('shows a hidden message to the original poster', async () => { const { message } = await user.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); await admin.post(`/groups/${group._id}/chat/${message.id}/flag`); const groupWithFlags = await user.get(`/groups/${group._id}`); const messageToCheck = find(groupWithFlags.chat, { id: message.id }); expect(messageToCheck).to.exist; const auGroupWithFlags = await anotherUser.get(`/groups/${group._id}`); const auMessageToCheck = find(auGroupWithFlags.chat, { id: message.id }); expect(auMessageToCheck).to.not.exist; }); it('validates that the message belongs to the passed group', async () => { const { group: anotherGroup, groupLeader: anotherLeader } = await createAndPopulateGroup({ groupDetails: { name: 'Another Guild', type: 'guild', privacy: 'private', }, upgradeToGroupPlan: true, }); const message = await anotherUser.post(`/groups/${group._id}/chat`, { message: TEST_MESSAGE }); await expect(anotherLeader.post(`/groups/${anotherGroup._id}/chat/${message.message.id}/flag`)) .to.eventually.be.rejected.and.eql({ code: 404, error: 'NotFound', message: t('messageGroupChatNotFound'), }); }); });