3ca4a11ca3
* feat: basic notification listener sturcuture * feat: clean up generated gql * chore: edited structure * feat: added basic repo * feat: ported comment email to job queue * feat: ported stream access request accepted * feat: added notification insertion * fix: minor typings * feat: delayed notifications * updated types * feat: fixed gql * notifications are listed * index on notifications * feat: while loop skiping for update locked * delayed notification for access request * take into account user prefrences * on comment view, notification is marked as read * feat: added gql notifications * feat: avoid raising errors * fix: error added scopes * fix: mr comments * fix: cursor and service method * feat: added stronger types to notifications and versioning logic * minor: rows updated
134 lines
4.6 KiB
TypeScript
134 lines
4.6 KiB
TypeScript
import type { MentionedInCommentData } from '@/modules/notifications/helpers/types'
|
|
import { NotificationType } from '@speckle/shared/notifications'
|
|
import { publishNotification } from '@/modules/notifications/services/publication/publishNotification'
|
|
import type { NotificationsStateManager } from '@/test/notificationsHelper'
|
|
import {
|
|
buildNotificationsStateTracker,
|
|
purgeNotifications
|
|
} from '@/test/notificationsHelper'
|
|
import { expect } from 'chai'
|
|
import {
|
|
InvalidNotificationError,
|
|
NotificationValidationError,
|
|
UnhandledNotificationError
|
|
} from '@/modules/notifications/errors'
|
|
import { NotificationJobResultsStatus } from '@/modules/notifications/services/publication/queue'
|
|
import { getEventBus } from '@/modules/shared/services/eventBus'
|
|
import { NotificationsEvents } from '@/modules/notifications/domain/events'
|
|
import { isNotificationListenerEnabled } from '@/modules/shared/helpers/envHelper'
|
|
|
|
!isNotificationListenerEnabled()
|
|
? describe('Notifications', () => {
|
|
let notificationsState: NotificationsStateManager
|
|
|
|
before(async () => {
|
|
await purgeNotifications()
|
|
notificationsState = await buildNotificationsStateTracker()
|
|
})
|
|
|
|
after(async () => {
|
|
notificationsState.destroy()
|
|
})
|
|
|
|
afterEach(() => {
|
|
notificationsState.reset()
|
|
})
|
|
|
|
it('can be emitted and routed to proper handler on consumption', async () => {
|
|
const targetUserId = '1234555'
|
|
const data: MentionedInCommentData = {
|
|
threadId: 'aaa',
|
|
commentId: 'bbb',
|
|
authorId: 'ccc',
|
|
streamId: 'ddd'
|
|
}
|
|
|
|
// Enqueue notification
|
|
const msgId = await publishNotification(NotificationType.MentionedInComment, {
|
|
targetUserId,
|
|
data
|
|
})
|
|
|
|
// Wait for ack
|
|
await notificationsState.waitForMsgAck(msgId)
|
|
|
|
const enqueuedMessage = notificationsState.collectedMessages().at(-1)!
|
|
expect(enqueuedMessage).to.be.ok
|
|
expect(enqueuedMessage?.targetUserId).to.eq(targetUserId)
|
|
expect(enqueuedMessage?.type).to.eq(NotificationType.MentionedInComment)
|
|
expect(enqueuedMessage?.data).to.deep.equalInAnyOrder(data)
|
|
})
|
|
|
|
it('fail safely when emitted with an unexpected structure', async () => {
|
|
// Enqueue notification with invalid structure
|
|
const msgId = await publishNotification(NotificationType.MentionedInComment, {
|
|
a: 1,
|
|
b: 2
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
} as any)
|
|
|
|
const { err } = await notificationsState.waitForMsgAck(msgId)
|
|
|
|
expect(err).to.be.ok
|
|
expect(err instanceof InvalidNotificationError).to.be.true
|
|
expect(err?.message).to.contain('invalid notification')
|
|
})
|
|
|
|
it('fail safely when emitted with an unexpected type', async () => {
|
|
// Enqueue notification with invalid structure
|
|
const msgId = await publishNotification('booooooooo' as NotificationType, {
|
|
targetUserId: '123',
|
|
data: {
|
|
a: 123
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
} as any
|
|
})
|
|
|
|
const { err } = await notificationsState.waitForMsgAck(msgId)
|
|
expect(err).to.be.ok
|
|
expect(err instanceof UnhandledNotificationError).to.be.true
|
|
})
|
|
|
|
const validationErrorDataSet = [
|
|
{
|
|
display: 'successful',
|
|
error: new NotificationValidationError('expected validation isue')
|
|
},
|
|
{ display: 'unsuccessful', error: new Error('ooohhhh') }
|
|
]
|
|
|
|
validationErrorDataSet.forEach(({ display, error }) => {
|
|
it(`fail with ${display} ack when handler throws ${error.name}`, async () => {
|
|
const data: MentionedInCommentData = {
|
|
threadId: 'aaa',
|
|
commentId: 'bbb',
|
|
authorId: 'ccc',
|
|
streamId: 'ddd'
|
|
}
|
|
|
|
getEventBus().listenOnce(NotificationsEvents.Received, () => {
|
|
throw error
|
|
})
|
|
|
|
const msgId = await publishNotification(NotificationType.MentionedInComment, {
|
|
targetUserId: '123',
|
|
data
|
|
})
|
|
|
|
const { err, result } = await notificationsState.waitForMsgAck(msgId)
|
|
|
|
const isValidationError = error instanceof NotificationValidationError
|
|
if (isValidationError) {
|
|
expect(err).to.be.not.ok
|
|
expect(result?.status).to.eq(NotificationJobResultsStatus.ValidationError)
|
|
} else {
|
|
expect(err).to.be.ok
|
|
expect(err?.name).to.eq(error.name)
|
|
expect(err?.message).to.eq(error.message)
|
|
expect(result).to.be.not.ok
|
|
}
|
|
})
|
|
})
|
|
})
|
|
: {}
|