2022-12-19 07:51:56 +01:00
|
|
|
import { CacheableResponsePlugin } from 'workbox-cacheable-response';
|
|
|
|
import { ExpirationPlugin } from 'workbox-expiration';
|
|
|
|
import { RegExpRoute, registerRoute, Route } from 'workbox-routing';
|
2023-01-10 10:08:10 +01:00
|
|
|
import {
|
|
|
|
CacheFirst,
|
|
|
|
NetworkFirst,
|
|
|
|
StaleWhileRevalidate,
|
|
|
|
} from 'workbox-strategies';
|
2022-12-19 07:51:56 +01:00
|
|
|
|
2023-01-02 07:22:31 +01:00
|
|
|
self.__WB_DISABLE_DEV_LOGS = true;
|
|
|
|
|
2023-09-14 17:21:43 +02:00
|
|
|
const assetsRoute = new Route(
|
|
|
|
({ request, sameOrigin }) => {
|
|
|
|
const isAsset =
|
|
|
|
request.destination === 'style' || request.destination === 'script';
|
|
|
|
const hasHash = /-[0-9a-f]{4,}\./i.test(request.url);
|
|
|
|
return sameOrigin && isAsset && hasHash;
|
|
|
|
},
|
|
|
|
new NetworkFirst({
|
|
|
|
cacheName: 'assets',
|
|
|
|
networkTimeoutSeconds: 5,
|
|
|
|
plugins: [
|
|
|
|
new CacheableResponsePlugin({
|
|
|
|
statuses: [0, 200],
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
registerRoute(assetsRoute);
|
|
|
|
|
2022-12-19 07:51:56 +01:00
|
|
|
const imageRoute = new Route(
|
|
|
|
({ request, sameOrigin }) => {
|
2022-12-20 06:21:08 +01:00
|
|
|
const isRemote = !sameOrigin;
|
|
|
|
const isImage = request.destination === 'image';
|
|
|
|
const isAvatar = request.url.includes('/avatars/');
|
2024-02-29 11:18:40 +01:00
|
|
|
const isCustomEmoji = request.url.includes('/custom/_emojis');
|
2022-12-20 06:21:08 +01:00
|
|
|
const isEmoji = request.url.includes('/emoji/');
|
2024-02-29 11:18:40 +01:00
|
|
|
return isRemote && isImage && (isAvatar || isCustomEmoji || isEmoji);
|
2022-12-19 07:51:56 +01:00
|
|
|
},
|
|
|
|
new CacheFirst({
|
|
|
|
cacheName: 'remote-images',
|
|
|
|
plugins: [
|
|
|
|
new ExpirationPlugin({
|
2023-02-25 03:31:50 +01:00
|
|
|
maxEntries: 50,
|
|
|
|
maxAgeSeconds: 3 * 24 * 60 * 60, // 3 days
|
2022-12-19 07:51:56 +01:00
|
|
|
purgeOnQuotaError: true,
|
|
|
|
}),
|
|
|
|
new CacheableResponsePlugin({
|
|
|
|
statuses: [0, 200],
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
registerRoute(imageRoute);
|
|
|
|
|
2023-04-14 11:23:41 +02:00
|
|
|
const iconsRoute = new Route(
|
|
|
|
({ request, sameOrigin }) => {
|
|
|
|
const isIcon = request.url.includes('/icons/');
|
|
|
|
return sameOrigin && isIcon;
|
|
|
|
},
|
|
|
|
new CacheFirst({
|
|
|
|
cacheName: 'icons',
|
|
|
|
plugins: [
|
|
|
|
new ExpirationPlugin({
|
|
|
|
maxEntries: 50,
|
|
|
|
maxAgeSeconds: 3 * 24 * 60 * 60, // 3 days
|
|
|
|
purgeOnQuotaError: true,
|
|
|
|
}),
|
|
|
|
new CacheableResponsePlugin({
|
|
|
|
statuses: [0, 200],
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
registerRoute(iconsRoute);
|
|
|
|
|
2023-02-23 13:54:23 +01:00
|
|
|
// 1-day cache for
|
|
|
|
// - /api/v1/instance
|
|
|
|
// - /api/v1/custom_emojis
|
|
|
|
// - /api/v1/preferences
|
|
|
|
// - /api/v1/lists/:id
|
2023-05-21 02:36:59 +02:00
|
|
|
// - /api/v1/announcements
|
2022-12-19 07:51:56 +01:00
|
|
|
const apiExtendedRoute = new RegExpRoute(
|
2023-05-21 02:36:59 +02:00
|
|
|
/^https?:\/\/[^\/]+\/api\/v\d+\/(instance|custom_emojis|preferences|lists\/\d+|announcements)$/,
|
2022-12-19 07:51:56 +01:00
|
|
|
new StaleWhileRevalidate({
|
|
|
|
cacheName: 'api-extended',
|
|
|
|
plugins: [
|
|
|
|
new ExpirationPlugin({
|
|
|
|
maxAgeSeconds: 24 * 60 * 60, // 1 day
|
|
|
|
}),
|
|
|
|
new CacheableResponsePlugin({
|
|
|
|
statuses: [0, 200],
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
registerRoute(apiExtendedRoute);
|
|
|
|
|
2023-01-02 07:23:00 +01:00
|
|
|
const apiRoute = new RegExpRoute(
|
|
|
|
// Matches:
|
|
|
|
// - statuses/:id/context - some contexts are really huge
|
|
|
|
/^https?:\/\/[^\/]+\/api\/v\d+\/(statuses\/\d+\/context)/,
|
2023-01-09 18:28:52 +01:00
|
|
|
new NetworkFirst({
|
2023-01-02 07:23:00 +01:00
|
|
|
cacheName: 'api',
|
2023-01-09 18:28:52 +01:00
|
|
|
networkTimeoutSeconds: 5,
|
2023-01-02 07:23:00 +01:00
|
|
|
plugins: [
|
|
|
|
new ExpirationPlugin({
|
|
|
|
maxAgeSeconds: 5 * 60, // 5 minutes
|
|
|
|
}),
|
|
|
|
new CacheableResponsePlugin({
|
|
|
|
statuses: [0, 200],
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
registerRoute(apiRoute);
|
2023-09-01 09:40:00 +02:00
|
|
|
|
|
|
|
// PUSH NOTIFICATIONS
|
|
|
|
// ==================
|
|
|
|
|
|
|
|
self.addEventListener('push', (event) => {
|
|
|
|
const { data } = event;
|
|
|
|
if (data) {
|
|
|
|
const payload = data.json();
|
|
|
|
console.log('PUSH payload', payload);
|
|
|
|
const {
|
|
|
|
access_token,
|
|
|
|
title,
|
|
|
|
body,
|
|
|
|
icon,
|
|
|
|
notification_id,
|
|
|
|
notification_type,
|
|
|
|
preferred_locale,
|
|
|
|
} = payload;
|
|
|
|
|
|
|
|
if (!!navigator.setAppBadge) {
|
|
|
|
if (notification_type === 'mention') {
|
|
|
|
navigator.setAppBadge(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
event.waitUntil(
|
|
|
|
self.registration.showNotification(title, {
|
|
|
|
body,
|
|
|
|
icon,
|
|
|
|
dir: 'auto',
|
2023-09-20 11:28:42 +02:00
|
|
|
badge: '/logo-badge-72.png',
|
2023-09-01 09:40:00 +02:00
|
|
|
lang: preferred_locale,
|
|
|
|
tag: notification_id,
|
|
|
|
timestamp: Date.now(),
|
|
|
|
data: {
|
|
|
|
access_token,
|
|
|
|
notification_type,
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
self.addEventListener('notificationclick', (event) => {
|
|
|
|
const payload = event.notification;
|
|
|
|
console.log('NOTIFICATION CLICK payload', payload);
|
|
|
|
const { badge, body, data, dir, icon, lang, tag, timestamp, title } = payload;
|
|
|
|
const { access_token, notification_type } = data;
|
2023-10-01 17:20:48 +02:00
|
|
|
const url = `/#/notifications?id=${tag}&access_token=${btoa(access_token)}`;
|
|
|
|
|
|
|
|
event.waitUntil(
|
|
|
|
(async () => {
|
|
|
|
const clients = await self.clients.matchAll({
|
2023-09-01 09:40:00 +02:00
|
|
|
type: 'window',
|
|
|
|
includeUncontrolled: true,
|
2023-10-01 17:20:48 +02:00
|
|
|
});
|
|
|
|
console.log('NOTIFICATION CLICK clients 1', clients);
|
|
|
|
if (clients.length && 'navigate' in clients[0]) {
|
|
|
|
console.log('NOTIFICATION CLICK clients 2', clients);
|
|
|
|
const bestClient =
|
|
|
|
clients.find(
|
|
|
|
(client) => client.focused || client.visibilityState === 'visible',
|
|
|
|
) || clients[0];
|
|
|
|
console.log('NOTIFICATION CLICK navigate', url);
|
|
|
|
if (bestClient) {
|
|
|
|
console.log('NOTIFICATION CLICK postMessage', bestClient);
|
2023-10-19 11:45:27 +02:00
|
|
|
bestClient.focus();
|
2023-10-01 17:20:48 +02:00
|
|
|
bestClient.postMessage?.({
|
|
|
|
type: 'notification',
|
|
|
|
id: tag,
|
|
|
|
accessToken: access_token,
|
|
|
|
});
|
2023-09-01 09:40:00 +02:00
|
|
|
} else {
|
|
|
|
console.log('NOTIFICATION CLICK openWindow', url);
|
2023-10-01 17:20:48 +02:00
|
|
|
await self.clients.openWindow(url);
|
2023-09-01 09:40:00 +02:00
|
|
|
}
|
2023-10-01 17:20:48 +02:00
|
|
|
// }
|
|
|
|
} else {
|
|
|
|
console.log('NOTIFICATION CLICK openWindow', url);
|
|
|
|
await self.clients.openWindow(url);
|
|
|
|
}
|
2023-10-16 15:38:14 +02:00
|
|
|
await event.notification.close();
|
2023-10-01 17:20:48 +02:00
|
|
|
})(),
|
|
|
|
);
|
2023-09-01 09:40:00 +02:00
|
|
|
});
|