mirror of
https://github.com/outline/outline.git
synced 2026-03-03 01:30:06 -06:00
* 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
291 lines
6.0 KiB
TypeScript
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;
|