import gql from 'graphql-tag' import Factory, { cleanDatabase } from '@db/factories' import { createTestClient } from 'apollo-server-testing' import { getNeode, getDriver } from '@db/neo4j' import createServer, { pubsub } from '@src/server' import { createGroupMutation, joinGroupMutation, leaveGroupMutation, changeGroupMemberRoleMutation, removeUserFromGroupMutation, } from '@graphql/groups' import { createMessageMutation } from '@graphql/messages' import { createRoomMutation } from '@graphql/rooms' const sendMailMock = jest.fn() jest.mock('../helpers/email/sendMail', () => ({ sendMail: () => sendMailMock(), })) const chatMessageTemplateMock = jest.fn() jest.mock('../helpers/email/templateBuilder', () => ({ chatMessageTemplate: () => chatMessageTemplateMock(), })) let isUserOnlineMock = jest.fn() jest.mock('../helpers/isUserOnline', () => ({ isUserOnline: () => isUserOnlineMock(), })) let server, query, mutate, notifiedUser, authenticatedUser let publishSpy const driver = getDriver() const neode = getNeode() const categoryIds = ['cat9'] const createPostMutation = gql` mutation ($id: ID, $title: String!, $postContent: String!, $categoryIds: [ID]!) { CreatePost(id: $id, title: $title, content: $postContent, categoryIds: $categoryIds) { id title content } } ` const updatePostMutation = gql` mutation ($id: ID!, $title: String!, $postContent: String!, $categoryIds: [ID]!) { UpdatePost(id: $id, content: $postContent, title: $title, categoryIds: $categoryIds) { title content } } ` const createCommentMutation = gql` mutation ($id: ID, $postId: ID!, $commentContent: String!) { CreateComment(id: $id, postId: $postId, content: $commentContent) { id content } } ` beforeAll(async () => { await cleanDatabase() publishSpy = jest.spyOn(pubsub, 'publish') const createServerResult = createServer({ context: () => { return { user: authenticatedUser, neode, driver, } }, }) server = createServerResult.server const createTestClientResult = createTestClient(server) query = createTestClientResult.query mutate = createTestClientResult.mutate }) afterAll(async () => { await cleanDatabase() driver.close() }) beforeEach(async () => { publishSpy.mockClear() notifiedUser = await neode.create( 'User', { id: 'you', name: 'Al Capone', slug: 'al-capone', }, { email: 'test@example.org', password: '1234', }, ) await neode.create('Category', { id: 'cat9', name: 'Democracy & Politics', icon: 'university', }) }) // TODO: avoid database clean after each test in the future if possible for performance and flakyness reasons by filling the database step by step, see issue https://github.com/Ocelot-Social-Community/Ocelot-Social/issues/4543 afterEach(async () => { await cleanDatabase() }) describe('notifications', () => { const notificationQuery = gql` query ($read: Boolean) { notifications(read: $read, orderBy: updatedAt_desc) { read reason createdAt relatedUser { id } from { __typename ... on Post { id content } ... on Comment { id content } ... on Group { id } } } } ` describe('authenticated', () => { beforeEach(async () => { authenticatedUser = await notifiedUser.toJson() }) describe('given another user', () => { let title let postContent let postAuthor const createPostAction = async () => { authenticatedUser = await postAuthor.toJson() await mutate({ mutation: createPostMutation, variables: { id: 'p47', title, postContent, categoryIds, }, }) authenticatedUser = await notifiedUser.toJson() } let commentContent let commentAuthor const createCommentOnPostAction = async () => { await createPostAction() authenticatedUser = await commentAuthor.toJson() await mutate({ mutation: createCommentMutation, variables: { id: 'c47', postId: 'p47', commentContent, }, }) authenticatedUser = await notifiedUser.toJson() } describe('comments on my post', () => { beforeEach(async () => { title = 'My post' postContent = 'My post content.' postAuthor = notifiedUser }) describe('commenter is not me', () => { beforeEach(async () => { commentContent = 'Commenters comment.' commentAuthor = await neode.create( 'User', { id: 'commentAuthor', name: 'Mrs Comment', slug: 'mrs-comment', }, { email: 'commentauthor@example.org', password: '1234', }, ) }) it('sends me a notification', async () => { await createCommentOnPostAction() const expected = expect.objectContaining({ data: { notifications: [ { read: false, createdAt: expect.any(String), reason: 'commented_on_post', from: { __typename: 'Comment', id: 'c47', content: commentContent, }, relatedUser: null, }, ], }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) it('sends me no notification if I have blocked the comment author', async () => { await notifiedUser.relateTo(commentAuthor, 'blocked') await createCommentOnPostAction() const expected = expect.objectContaining({ data: { notifications: [] }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) }) describe('commenter is me', () => { beforeEach(async () => { commentContent = 'My comment.' commentAuthor = notifiedUser }) it('sends me no notification', async () => { await createCommentOnPostAction() const expected = expect.objectContaining({ data: { notifications: [] }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) }) }) beforeEach(async () => { postAuthor = await neode.create( 'User', { id: 'postAuthor', name: 'Mrs Post', slug: 'mrs-post', }, { email: 'post-author@example.org', password: '1234', }, ) }) describe('mentions me in a post', () => { beforeEach(async () => { title = 'Mentioning Al Capone' postContent = 'Hey @al-capone how do you do?' }) it('sends me a notification', async () => { await createPostAction() const expectedContent = 'Hey @al-capone how do you do?' await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toMatchObject({ errors: undefined, data: { notifications: [ { read: false, createdAt: expect.any(String), reason: 'mentioned_in_post', from: { __typename: 'Post', id: 'p47', content: expectedContent, }, }, ], }, }) }) it('publishes `NOTIFICATION_ADDED` to me', async () => { await createPostAction() expect(publishSpy).toHaveBeenCalledWith( 'NOTIFICATION_ADDED', expect.objectContaining({ notificationAdded: expect.objectContaining({ reason: 'mentioned_in_post', to: expect.objectContaining({ id: 'you', }), }), }), ) expect(publishSpy).toHaveBeenCalledTimes(1) }) describe('updates the post and mentions me again', () => { const updatePostAction = async () => { const updatedContent = ` One more mention to @al-capone and again: @al-capone and again @al-capone ` authenticatedUser = await postAuthor.toJson() await mutate({ mutation: updatePostMutation, variables: { id: 'p47', title, postContent: updatedContent, categoryIds, }, }) authenticatedUser = await notifiedUser.toJson() } it('creates no duplicate notification for the same resource', async () => { const expectedUpdatedContent = '
One more mention to

@al-capone

and again:

@al-capone

and again

@al-capone

' await createPostAction() await updatePostAction() const expected = expect.objectContaining({ data: { notifications: [ { read: false, createdAt: expect.any(String), reason: 'mentioned_in_post', from: { __typename: 'Post', id: 'p47', content: expectedUpdatedContent, }, relatedUser: null, }, ], }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) describe('if the notification was marked as read earlier', () => { const markAsReadAction = async () => { const mutation = gql` mutation ($id: ID!) { markAsRead(id: $id) { read } } ` await mutate({ mutation, variables: { id: 'p47' } }) } describe('but the next mention happens after the notification was marked as read', () => { it('sets the `read` attribute to false again', async () => { await createPostAction() await markAsReadAction() const { data: { notifications: [{ read: readBefore }], }, } = await query({ query: notificationQuery, }) await updatePostAction() const { data: { notifications: [{ read: readAfter }], }, } = await query({ query: notificationQuery, }) expect(readBefore).toEqual(true) expect(readAfter).toEqual(false) }) it('does not update the `createdAt` attribute', async () => { await createPostAction() await markAsReadAction() const { data: { notifications: [{ createdAt: createdAtBefore }], }, } = await query({ query: notificationQuery, }) await updatePostAction() const { data: { notifications: [{ createdAt: createdAtAfter }], }, } = await query({ query: notificationQuery, }) expect(createdAtBefore).toBeTruthy() expect(Date.parse(createdAtBefore)).toEqual(expect.any(Number)) expect(createdAtAfter).toBeTruthy() expect(Date.parse(createdAtAfter)).toEqual(expect.any(Number)) expect(createdAtBefore).toEqual(createdAtAfter) }) }) }) }) describe('but the author of the post blocked me', () => { beforeEach(async () => { await postAuthor.relateTo(notifiedUser, 'blocked') }) it('sends no notification', async () => { await createPostAction() const expected = expect.objectContaining({ data: { notifications: [] }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) it('does not publish `NOTIFICATION_ADDED`', async () => { await createPostAction() expect(publishSpy).not.toHaveBeenCalled() }) }) }) describe('mentions me in a comment', () => { beforeEach(async () => { title = 'Post where I get mentioned in a comment' postContent = 'Content of post where I get mentioned in a comment.' }) describe('I am not blocked at all', () => { beforeEach(async () => { commentContent = 'One mention about me with @al-capone.' commentAuthor = await neode.create( 'User', { id: 'commentAuthor', name: 'Mrs Comment', slug: 'mrs-comment', }, { email: 'comment-author@example.org', password: '1234', }, ) }) it('sends only one notification with reason mentioned_in_comment', async () => { postAuthor = await neode.create( 'User', { id: 'MrPostAuthor', name: 'Mr Author', slug: 'mr-author', }, { email: 'post-author@example.org', password: '1234', }, ) await createCommentOnPostAction() const expected = expect.objectContaining({ data: { notifications: [ { read: false, createdAt: expect.any(String), reason: 'mentioned_in_comment', from: { __typename: 'Comment', id: 'c47', content: commentContent, }, relatedUser: null, }, ], }, }) await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toEqual(expected) }) beforeEach(async () => { title = "Post where I'm the author and I get mentioned in a comment" postContent = 'Content of post where I get mentioned in a comment.' postAuthor = notifiedUser }) it('sends only one notification with reason commented_on_post, no notification with reason mentioned_in_comment', async () => { await createCommentOnPostAction() const expected = { data: { notifications: [ { read: false, createdAt: expect.any(String), reason: 'commented_on_post', from: { __typename: 'Comment', id: 'c47', content: commentContent, }, relatedUser: null, }, ], }, } await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toMatchObject({ ...expected, errors: undefined }) }) }) describe('but the author of the post blocked me', () => { beforeEach(async () => { await postAuthor.relateTo(notifiedUser, 'blocked') commentContent = 'One mention about me with @al-capone.' commentAuthor = await neode.create( 'User', { id: 'commentAuthor', name: 'Mrs Comment', slug: 'mrs-comment', }, { email: 'comment-author@example.org', password: '1234', }, ) }) it('sends no notification', async () => { await createCommentOnPostAction() await expect( query({ query: notificationQuery, variables: { read: false, }, }), ).resolves.toMatchObject({ data: { notifications: [] }, errors: undefined, }) }) it('does not publish `NOTIFICATION_ADDED` to authenticated user', async () => { await createCommentOnPostAction() expect(publishSpy).toHaveBeenCalledWith( 'NOTIFICATION_ADDED', expect.objectContaining({ notificationAdded: expect.objectContaining({ reason: 'commented_on_post', to: expect.objectContaining({ id: 'postAuthor', // that's expected, it's not me but the post author }), }), }), ) expect(publishSpy).toHaveBeenCalledTimes(1) }) }) }) }) }) describe('chat email notifications', () => { let chatSender let chatReceiver let roomId beforeEach(async () => { jest.clearAllMocks() chatSender = await neode.create( 'User', { id: 'chatSender', name: 'chatSender', slug: 'chatSender', }, { email: 'chatSender@example.org', password: '1234', }, ) chatReceiver = await Factory.build( 'user', { id: 'chatReceiver', name: 'chatReceiver', slug: 'chatReceiver' }, { email: 'user@example.org' }, ) authenticatedUser = await chatSender.toJson() const room = await mutate({ mutation: createRoomMutation(), variables: { userId: 'chatReceiver', }, }) roomId = room.data.CreateRoom.id }) describe('chatReceiver is online', () => { it('sends no email', async () => { isUserOnlineMock = jest.fn().mockReturnValue(true) await mutate({ mutation: createMessageMutation(), variables: { roomId, content: 'Some nice message to chatReceiver', }, }) expect(sendMailMock).not.toHaveBeenCalled() expect(chatMessageTemplateMock).not.toHaveBeenCalled() }) }) describe('chatReceiver is offline', () => { it('sends an email', async () => { isUserOnlineMock = jest.fn().mockReturnValue(false) await mutate({ mutation: createMessageMutation(), variables: { roomId, content: 'Some nice message to chatReceiver', }, }) expect(sendMailMock).toHaveBeenCalledTimes(1) expect(chatMessageTemplateMock).toHaveBeenCalledTimes(1) }) }) describe('chatReceiver has blocked chatSender', () => { it('sends no email', async () => { isUserOnlineMock = jest.fn().mockReturnValue(false) await chatReceiver.relateTo(chatSender, 'blocked') await mutate({ mutation: createMessageMutation(), variables: { roomId, content: 'Some nice message to chatReceiver', }, }) expect(sendMailMock).not.toHaveBeenCalled() expect(chatMessageTemplateMock).not.toHaveBeenCalled() }) }) describe('chatReceiver has disabled email notifications', () => { it('sends no email', async () => { isUserOnlineMock = jest.fn().mockReturnValue(false) await chatReceiver.update({ sendNotificationEmails: false }) await mutate({ mutation: createMessageMutation(), variables: { roomId, content: 'Some nice message to chatReceiver', }, }) expect(sendMailMock).not.toHaveBeenCalled() expect(chatMessageTemplateMock).not.toHaveBeenCalled() }) }) }) describe('group notifications', () => { let groupOwner beforeEach(async () => { groupOwner = await neode.create( 'User', { id: 'group-owner', name: 'Group Owner', slug: 'group-owner', }, { email: 'owner@example.org', password: '1234', }, ) authenticatedUser = await groupOwner.toJson() await mutate({ mutation: createGroupMutation(), variables: { id: 'closed-group', name: 'The Closed Group', about: 'Will test the closed group!', description: 'Some description' + Array(50).join('_'), groupType: 'public', actionRadius: 'regional', categoryIds, }, }) }) describe('user joins group', () => { beforeEach(async () => { authenticatedUser = await notifiedUser.toJson() await mutate({ mutation: joinGroupMutation(), variables: { groupId: 'closed-group', userId: authenticatedUser.id, }, }) authenticatedUser = await groupOwner.toJson() }) it('has the notification in database', async () => { await expect( query({ query: notificationQuery, }), ).resolves.toMatchObject({ data: { notifications: [ { read: false, reason: 'user_joined_group', createdAt: expect.any(String), from: { __typename: 'Group', id: 'closed-group', }, relatedUser: { id: 'you', }, }, ], }, errors: undefined, }) }) }) describe('user leaves group', () => { beforeEach(async () => { authenticatedUser = await notifiedUser.toJson() await mutate({ mutation: joinGroupMutation(), variables: { groupId: 'closed-group', userId: authenticatedUser.id, }, }) await mutate({ mutation: leaveGroupMutation(), variables: { groupId: 'closed-group', userId: authenticatedUser.id, }, }) authenticatedUser = await groupOwner.toJson() }) it('has two the notification in database', async () => { await expect( query({ query: notificationQuery, }), ).resolves.toMatchObject({ data: { notifications: [ { read: false, reason: 'user_left_group', createdAt: expect.any(String), from: { __typename: 'Group', id: 'closed-group', }, relatedUser: { id: 'you', }, }, { read: false, reason: 'user_joined_group', createdAt: expect.any(String), from: { __typename: 'Group', id: 'closed-group', }, relatedUser: { id: 'you', }, }, ], }, errors: undefined, }) }) }) describe('user role in group changes', () => { beforeEach(async () => { authenticatedUser = await notifiedUser.toJson() await mutate({ mutation: joinGroupMutation(), variables: { groupId: 'closed-group', userId: authenticatedUser.id, }, }) authenticatedUser = await groupOwner.toJson() await mutate({ mutation: changeGroupMemberRoleMutation(), variables: { groupId: 'closed-group', userId: 'you', roleInGroup: 'admin', }, }) authenticatedUser = await notifiedUser.toJson() }) it('has notification in database', async () => { await expect( query({ query: notificationQuery, }), ).resolves.toMatchObject({ data: { notifications: [ { read: false, reason: 'changed_group_member_role', createdAt: expect.any(String), from: { __typename: 'Group', id: 'closed-group', }, relatedUser: { id: 'group-owner', }, }, ], }, errors: undefined, }) }) }) describe('user is removed from group', () => { beforeEach(async () => { authenticatedUser = await notifiedUser.toJson() await mutate({ mutation: joinGroupMutation(), variables: { groupId: 'closed-group', userId: authenticatedUser.id, }, }) authenticatedUser = await groupOwner.toJson() await mutate({ mutation: removeUserFromGroupMutation(), variables: { groupId: 'closed-group', userId: 'you', }, }) authenticatedUser = await notifiedUser.toJson() }) it('has notification in database', async () => { await expect( query({ query: notificationQuery, }), ).resolves.toMatchObject({ data: { notifications: [ { read: false, reason: 'removed_user_from_group', createdAt: expect.any(String), from: { __typename: 'Group', id: 'closed-group', }, relatedUser: { id: 'group-owner', }, }, ], }, errors: undefined, }) }) }) }) })