aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Database
diff options
context:
space:
mode:
authorFuwn <[email protected]>2026-03-28 06:20:36 +0000
committerFuwn <[email protected]>2026-03-28 06:20:36 +0000
commit78bf2502cbf1e26980abf6c0ffb7f0c74b917a3b (patch)
tree9f4a84c90fa7fcfa945f1c9e0e637ba80b9cca79 /src/lib/Database
parentfix(cache): preserve hydrated client state (diff)
downloaddue.moe-78bf2502cbf1e26980abf6c0ffb7f0c74b917a3b.tar.xz
due.moe-78bf2502cbf1e26980abf6c0ffb7f0c74b917a3b.zip
fix(notifications): support per-device push subscriptions
Diffstat (limited to 'src/lib/Database')
-rw-r--r--src/lib/Database/SB/User/notifications.ts41
1 files changed, 38 insertions, 3 deletions
diff --git a/src/lib/Database/SB/User/notifications.ts b/src/lib/Database/SB/User/notifications.ts
index 058171a9..21ad827b 100644
--- a/src/lib/Database/SB/User/notifications.ts
+++ b/src/lib/Database/SB/User/notifications.ts
@@ -8,6 +8,16 @@ export interface UserNotifications {
fingerprint: string;
}
+const subscriptionEndpoint = (subscription: JSON) => {
+ if (typeof subscription !== "object" || subscription === null) return null;
+
+ return "endpoint" in subscription &&
+ typeof subscription.endpoint === "string" &&
+ subscription.endpoint.length
+ ? subscription.endpoint
+ : null;
+};
+
export const getUserSubscription = async (userId: number) =>
await sb.from("user_notifications").select("*").eq("user_id", userId);
@@ -33,13 +43,38 @@ export const setUserSubscription = async (
userId: number,
subscription: JSON,
fingerprint: string,
-) =>
- await sb.from("user_notifications").upsert(
+) => {
+ const endpoint = subscriptionEndpoint(subscription);
+
+ if (endpoint) {
+ const { data } = await sb
+ .from("user_notifications")
+ .select("fingerprint, subscription")
+ .eq("user_id", userId);
+
+ const staleFingerprints =
+ data
+ ?.filter(
+ (existing) =>
+ existing.fingerprint !== fingerprint &&
+ subscriptionEndpoint(existing.subscription as JSON) === endpoint,
+ )
+ .map((existing) => existing.fingerprint) ?? [];
+
+ await Promise.all(
+ staleFingerprints.map(async (staleFingerprint) => {
+ await deleteUserSubscription(userId, staleFingerprint);
+ }),
+ );
+ }
+
+ return await sb.from("user_notifications").upsert(
{
user_id: userId,
updated_at: new Date().toISOString(),
subscription: subscription,
fingerprint,
},
- { onConflict: "user_id" },
+ { onConflict: "user_id,fingerprint" },
);
+};