mirror of
https://github.com/Sendouc/sendou.ink.git
synced 2026-04-04 16:55:03 -05:00
155 lines
4.1 KiB
TypeScript
155 lines
4.1 KiB
TypeScript
import type { TFunction } from "i18next";
|
|
import pLimit from "p-limit";
|
|
import { WebPushError } from "web-push";
|
|
import { IS_E2E_TEST_RUN } from "~/utils/e2e";
|
|
import type { NotificationSubscription } from "../../../db/tables";
|
|
import { i18next } from "../../../modules/i18n/i18next.server";
|
|
import { logger } from "../../../utils/logger";
|
|
import * as NotificationRepository from "../NotificationRepository.server";
|
|
import type { Notification } from "../notifications-types";
|
|
import {
|
|
mapMetaForTranslation,
|
|
notificationLink,
|
|
} from "../notifications-utils";
|
|
import webPush, { webPushEnabled } from "./webPush.server";
|
|
|
|
/**
|
|
* Create notifications both in the database and send push notifications to users (if enabled).
|
|
*/
|
|
export async function notify({
|
|
userIds,
|
|
notification,
|
|
defaultSeenUserIds,
|
|
}: {
|
|
/** Array of user ids to notify */
|
|
userIds: Array<number>;
|
|
/** Array of user ids that should have the notification marked as seen by default */
|
|
defaultSeenUserIds?: Array<number>;
|
|
/** Notification to send (same for all users) */
|
|
notification: Notification;
|
|
}) {
|
|
if (userIds.length === 0) {
|
|
return;
|
|
}
|
|
|
|
const dededuplicatedUserIds = Array.from(new Set(userIds));
|
|
|
|
if (isNotificationAlreadySent(notification, dededuplicatedUserIds)) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
await NotificationRepository.insert(
|
|
notification,
|
|
dededuplicatedUserIds.map((userId) => ({
|
|
userId,
|
|
seen: defaultSeenUserIds?.includes(userId) ? 1 : 0,
|
|
})),
|
|
);
|
|
} catch (e) {
|
|
logger.error("Failed to notify users", e);
|
|
}
|
|
|
|
const subscriptions = await NotificationRepository.subscriptionsByUserIds(
|
|
dededuplicatedUserIds,
|
|
);
|
|
if (subscriptions.length > 0) {
|
|
const t = await i18next.getFixedT("en-US", ["common"]);
|
|
|
|
const limit = pLimit(50);
|
|
|
|
await Promise.all(
|
|
subscriptions.map(({ id, subscription }) =>
|
|
limit(() =>
|
|
sendPushNotification({
|
|
subscription,
|
|
subscriptionId: id,
|
|
notification,
|
|
t,
|
|
}),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
const sentNotifications = new Set<string>();
|
|
|
|
export function clearSentNotificationsForTesting() {
|
|
sentNotifications.clear();
|
|
}
|
|
|
|
// deduplicates notifications as a failsafe & anti-abuse mechanism
|
|
function isNotificationAlreadySent(
|
|
notification: Notification,
|
|
userIds: Array<number>,
|
|
) {
|
|
// e2e tests should not be affected by this
|
|
if (IS_E2E_TEST_RUN) {
|
|
return false;
|
|
}
|
|
|
|
// bulk notifications are typically not something you can repeat
|
|
if (userIds.length > 10) {
|
|
return false;
|
|
}
|
|
|
|
const sortedUserIds = [...userIds].sort((a, b) => a - b).join(",");
|
|
const key = `${notification.type}-${JSON.stringify(notification.meta)}-${sortedUserIds}`;
|
|
if (sentNotifications.has(key)) {
|
|
return true;
|
|
}
|
|
sentNotifications.add(key);
|
|
|
|
if (sentNotifications.size > 10_000) {
|
|
sentNotifications.clear();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
async function sendPushNotification({
|
|
subscription,
|
|
subscriptionId,
|
|
notification,
|
|
t,
|
|
}: {
|
|
subscription: NotificationSubscription;
|
|
subscriptionId: number;
|
|
notification: Notification;
|
|
t: TFunction<["common"], undefined>;
|
|
}) {
|
|
if (!webPushEnabled) return;
|
|
|
|
try {
|
|
await webPush.sendNotification(
|
|
subscription,
|
|
JSON.stringify(pushNotificationOptions(notification, t)),
|
|
);
|
|
} catch (err) {
|
|
if (!(err instanceof WebPushError)) {
|
|
logger.error("Failed to send push notification (unknown error)", err);
|
|
// if we get "Not Found" or "Gone" we should delete the subscription as it is expired or no longer valid
|
|
} else if (err.statusCode === 404 || err.statusCode === 410) {
|
|
await NotificationRepository.deleteSubscriptionById(subscriptionId);
|
|
} else {
|
|
logger.error("Failed to send push notification", err);
|
|
}
|
|
}
|
|
}
|
|
|
|
function pushNotificationOptions(
|
|
notification: Notification,
|
|
t: TFunction<["common"], undefined>,
|
|
): Parameters<ServiceWorkerRegistration["showNotification"]>[1] & {
|
|
title: string;
|
|
} {
|
|
const meta = mapMetaForTranslation(notification, "en-US");
|
|
return {
|
|
title: t(`common:notifications.title.${notification.type}`),
|
|
body: t(`common:notifications.text.${notification.type}`, meta),
|
|
icon: notification.pictureUrl ?? "/static-assets/img/app-icon.png",
|
|
data: { url: notificationLink(notification) },
|
|
};
|
|
}
|