aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Database/SB/User/notifications.ts
blob: 21ad827b621239e0402a7fa8b7f4291d9b539826 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
import sb from "../../sb.server";

export interface UserNotifications {
	created_at: string;
	updated_at: string;
	user_id: number;
	subscription: JSON;
	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);

export const getUserSubscriptions = async () => {
	const { data, error } = await sb.from("user_notifications").select("*");

	if (error) return [];

	return data as UserNotifications[];
};

export const deleteUserSubscription = async (
	userId: number,
	fingerprint: string,
) =>
	await sb
		.from("user_notifications")
		.delete()
		.eq("user_id", userId)
		.eq("fingerprint", fingerprint);

export const setUserSubscription = async (
	userId: number,
	subscription: JSON,
	fingerprint: string,
) => {
	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,fingerprint" },
	);
};