Notifications refactor (#5151
* Ongoing * refactor * test * Add cleanup task * refactor
This commit is contained in:
145
server/queues/tasks/RevisionCreatedNotificationsTask.ts
Normal file
145
server/queues/tasks/RevisionCreatedNotificationsTask.ts
Normal file
@@ -0,0 +1,145 @@
|
||||
import { subHours } from "date-fns";
|
||||
import { differenceBy } from "lodash";
|
||||
import { Op } from "sequelize";
|
||||
import { NotificationEventType } from "@shared/types";
|
||||
import { createSubscriptionsForDocument } from "@server/commands/subscriptionCreator";
|
||||
import env from "@server/env";
|
||||
import Logger from "@server/logging/Logger";
|
||||
import { Document, Revision, Notification, User, View } from "@server/models";
|
||||
import DocumentHelper from "@server/models/helpers/DocumentHelper";
|
||||
import NotificationHelper from "@server/models/helpers/NotificationHelper";
|
||||
import { RevisionEvent } from "@server/types";
|
||||
import BaseTask, { TaskPriority } from "./BaseTask";
|
||||
|
||||
export default class RevisionCreatedNotificationsTask extends BaseTask<
|
||||
RevisionEvent
|
||||
> {
|
||||
public async perform(event: RevisionEvent) {
|
||||
const [document, revision] = await Promise.all([
|
||||
Document.findByPk(event.documentId, { includeState: true }),
|
||||
Revision.findByPk(event.modelId),
|
||||
]);
|
||||
|
||||
if (!document || !revision) {
|
||||
return;
|
||||
}
|
||||
|
||||
await createSubscriptionsForDocument(document, event);
|
||||
|
||||
// Send notifications to mentioned users first
|
||||
const before = await revision.previous();
|
||||
const oldMentions = before ? DocumentHelper.parseMentions(before) : [];
|
||||
const newMentions = DocumentHelper.parseMentions(document);
|
||||
const mentions = differenceBy(newMentions, oldMentions, "id");
|
||||
const userIdsMentioned: string[] = [];
|
||||
|
||||
for (const mention of mentions) {
|
||||
const recipient = await User.findByPk(mention.modelId);
|
||||
if (
|
||||
recipient &&
|
||||
recipient.id !== mention.actorId &&
|
||||
recipient.subscribedToEventType(
|
||||
NotificationEventType.MentionedInDocument
|
||||
)
|
||||
) {
|
||||
await Notification.create({
|
||||
event: NotificationEventType.MentionedInDocument,
|
||||
userId: recipient.id,
|
||||
revisionId: event.modelId,
|
||||
actorId: document.updatedBy.id,
|
||||
teamId: document.teamId,
|
||||
documentId: document.id,
|
||||
});
|
||||
userIdsMentioned.push(recipient.id);
|
||||
}
|
||||
}
|
||||
|
||||
const recipients = (
|
||||
await NotificationHelper.getDocumentNotificationRecipients(
|
||||
document,
|
||||
NotificationEventType.UpdateDocument,
|
||||
document.lastModifiedById,
|
||||
true
|
||||
)
|
||||
).filter((recipient) => !userIdsMentioned.includes(recipient.id));
|
||||
if (!recipients.length) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (const recipient of recipients) {
|
||||
const notify = await this.shouldNotify(document, recipient);
|
||||
|
||||
if (notify) {
|
||||
await Notification.create({
|
||||
event: NotificationEventType.UpdateDocument,
|
||||
userId: recipient.id,
|
||||
revisionId: event.modelId,
|
||||
actorId: document.updatedBy.id,
|
||||
teamId: document.teamId,
|
||||
documentId: document.id,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private shouldNotify = async (
|
||||
document: Document,
|
||||
user: User
|
||||
): Promise<boolean> => {
|
||||
// Create only a single notification in a 6 hour window
|
||||
const notification = await Notification.findOne({
|
||||
order: [["createdAt", "DESC"]],
|
||||
where: {
|
||||
userId: user.id,
|
||||
documentId: document.id,
|
||||
emailedAt: {
|
||||
[Op.not]: null,
|
||||
[Op.gte]: subHours(new Date(), 6),
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
if (notification) {
|
||||
if (env.ENVIRONMENT === "development") {
|
||||
Logger.info(
|
||||
"processor",
|
||||
`would have suppressed notification to ${user.id}, but not in development`
|
||||
);
|
||||
} else {
|
||||
Logger.info(
|
||||
"processor",
|
||||
`suppressing notification to ${user.id} as recently notified`
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// If this recipient has viewed the document since the last update was made
|
||||
// then we can avoid sending them a useless notification, yay.
|
||||
const view = await View.findOne({
|
||||
where: {
|
||||
userId: user.id,
|
||||
documentId: document.id,
|
||||
updatedAt: {
|
||||
[Op.gt]: document.updatedAt,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
if (view) {
|
||||
Logger.info(
|
||||
"processor",
|
||||
`suppressing notification to ${user.id} because update viewed`
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
public get options() {
|
||||
return {
|
||||
priority: TaskPriority.Background,
|
||||
};
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user