Files
outline/server/models/Notification.ts
Hemachandar 7f17a51e11 feat: Add email headers to enhance threading (#7477)
* feat: Add email headers to enhance threading

* tsc

* review

* change comment mentioned email subject

* paginated load

* rename util method

* add tests

* group events

* test for unsupported notification

* typo

* review
2024-09-03 17:20:56 -07:00

291 lines
6.0 KiB
TypeScript

import crypto from "crypto";
import chunk from "lodash/chunk";
import type {
InferAttributes,
InferCreationAttributes,
SaveOptions,
} from "sequelize";
import { Op } from "sequelize";
import {
Table,
ForeignKey,
Column,
PrimaryKey,
IsUUID,
CreatedAt,
BelongsTo,
DataType,
Default,
AllowNull,
Scopes,
AfterCreate,
DefaultScope,
} from "sequelize-typescript";
import { NotificationEventType } from "@shared/types";
import env from "@server/env";
import Model from "@server/models/base/Model";
import {
getEmailMessageId,
getEmailThreadEventGroup,
isEmailThreadSupportedNotification,
MaxMessagesInEmailThread,
} from "@server/utils/emails";
import Collection from "./Collection";
import Comment from "./Comment";
import Document from "./Document";
import Event from "./Event";
import Revision from "./Revision";
import Team from "./Team";
import User from "./User";
import Fix from "./decorators/Fix";
@Scopes(() => ({
withTeam: {
include: [
{
association: "team",
},
],
},
withDocument: {
include: [
{
association: "document",
},
],
},
withComment: {
include: [
{
association: "comment",
},
],
},
withActor: {
include: [
{
association: "actor",
},
],
},
withUser: {
include: [
{
association: "user",
},
],
},
}))
@DefaultScope(() => ({
include: [
{
association: "document",
required: false,
},
{
association: "comment",
required: false,
},
{
association: "actor",
required: false,
},
],
}))
@Table({
tableName: "notifications",
modelName: "notification",
updatedAt: false,
})
@Fix
class Notification extends Model<
InferAttributes<Notification>,
Partial<InferCreationAttributes<Notification>>
> {
@IsUUID(4)
@PrimaryKey
@Default(DataType.UUIDV4)
@Column(DataType.UUID)
id: string;
@AllowNull
@Column
emailedAt?: Date | null;
@AllowNull
@Column
viewedAt: Date | null;
@AllowNull
@Column
archivedAt: Date | null;
@CreatedAt
createdAt: Date;
@Column(DataType.STRING)
event: NotificationEventType;
// associations
@BelongsTo(() => User, "userId")
user: User;
@ForeignKey(() => User)
@Column(DataType.UUID)
userId: string;
@BelongsTo(() => User, "actorId")
actor: User;
@AllowNull
@ForeignKey(() => User)
@Column(DataType.UUID)
actorId: string;
@BelongsTo(() => Comment, "commentId")
comment: Comment;
@AllowNull
@ForeignKey(() => Comment)
@Column(DataType.UUID)
commentId: string;
@BelongsTo(() => Document, "documentId")
document: Document;
@AllowNull
@ForeignKey(() => Document)
@Column(DataType.UUID)
documentId: string;
@BelongsTo(() => Revision, "revisionId")
revision: Revision;
@AllowNull
@ForeignKey(() => Revision)
@Column(DataType.UUID)
revisionId: string;
@BelongsTo(() => Collection, "collectionId")
collection: Collection;
@AllowNull
@ForeignKey(() => Collection)
@Column(DataType.UUID)
collectionId: string;
@BelongsTo(() => Team, "teamId")
team: Team;
@ForeignKey(() => Team)
@Column(DataType.UUID)
teamId: string;
@AfterCreate
static async createEvent(
model: Notification,
options: SaveOptions<InferAttributes<Notification>>
) {
const params = {
name: "notifications.create",
userId: model.userId,
modelId: model.id,
teamId: model.teamId,
commentId: model.commentId,
documentId: model.documentId,
collectionId: model.collectionId,
actorId: model.actorId,
};
if (options.transaction) {
options.transaction.afterCommit(() => void Event.schedule(params));
return;
}
await Event.schedule(params);
}
/**
* Returns a token that can be used to mark this notification as read
* without being logged in.
*
* @returns A string token
*/
public get pixelToken() {
const hash = crypto.createHash("sha256");
hash.update(`${this.id}-${env.SECRET_KEY}`);
return hash.digest("hex");
}
/**
* Returns a URL that can be used to mark this notification as read
* without being logged in.
*
* @returns A URL
*/
public get pixelUrl() {
return `${env.URL}/api/notifications.pixel?token=${this.pixelToken}&id=${this.id}`;
}
/**
* Returns the past message ids which are used to setup the thread chain in email clients.
*
* @param notification Notification for which the past notifications are fetched - used for determining the properties that form a thread.
* @returns An array of email message ids that form a thread.
*/
public static async emailReferences(
notification: Notification
): Promise<string[] | undefined> {
if (!isEmailThreadSupportedNotification(notification.event)) {
return;
}
const events = getEmailThreadEventGroup(notification.event);
if (!events) {
return;
}
const prevNotifications: Notification[] = [];
await this.findAllInBatches<Notification>(
{
attributes: ["id"],
where: {
id: {
[Op.ne]: notification.id,
},
event: {
[Op.in]: events,
},
documentId: notification.documentId,
userId: notification.userId,
},
order: [["createdAt", "ASC"]],
offset: 0,
batchLimit: 100,
},
async (notifications) => void prevNotifications.push(...notifications)
);
const emailThreads = chunk(prevNotifications, MaxMessagesInEmailThread);
const lastThread = emailThreads.at(-1);
// Don't return anything if there are no past notifications (or) the limit is reached.
// This will start a new thread in the email clients.
// Also ensures we don't face header limit errors.
if (
!lastThread ||
lastThread.length === 0 ||
lastThread.length === MaxMessagesInEmailThread
) {
return;
}
// Return references from the last thread.
return lastThread.map((notif) => getEmailMessageId(notif.id));
}
}
export default Notification;