Simplify service worker

This commit is contained in:
Pieter Vander Vennet 2024-08-26 03:42:02 +02:00
parent 1008226ad1
commit 5c56651b5e

View file

@ -12,74 +12,84 @@ async function install() {
addEventListener("install", (e) => (<any>e).waitUntil(install())) addEventListener("install", (e) => (<any>e).waitUntil(install()))
addEventListener("activate", (e) => (<any>e).waitUntil(activate())) addEventListener("activate", (e) => (<any>e).waitUntil(activate()))
async function clearCaches(exceptVersion = undefined) {
const keys = await caches.keys()
await Promise.all(keys.map((k) => k !== version && caches.delete(k)))
console.log("Cleared caches")
}
async function activate() { async function activate() {
console.log("Activating service worker") console.log("Activating service worker")
caches await clearCaches(version)
.keys()
.then((keys) => {
// Remove all old caches
Promise.all(keys.map((key) => key !== version && caches.delete(key)))
})
.catch(console.error)
} }
function fetchAndCache(event) { async function fetchAndCache(event: ServiceWorkerFetchEvent): Promise<Response> {
return fetch(event.request).then((networkResponse) => { let networkResponse = await fetch(event.request)
return caches.open(version).then((cache) => { let cache = await caches.open(version)
cache.put(event.request, networkResponse.clone()) await cache.put(event.request, networkResponse.clone())
console.log("Cached", event.request) console.log("Cached", event.request)
return networkResponse return networkResponse
})
})
} }
const cacheFirst = async (event, attemptUpdate: boolean = false) => { async function cacheFirst(event: ServiceWorkerFetchEvent, attemptUpdate: boolean = false) {
await event.respondWith( const cacheResponse = await caches.match(event.request, { ignoreSearch: true })
caches.match(event.request, { ignoreSearch: true }).then((cacheResponse) => { if (cacheResponse === undefined) {
if (cacheResponse !== undefined) { return fetchAndCache(event)
}
console.debug("Loaded from cache: ", event.request) console.debug("Loaded from cache: ", event.request)
if (attemptUpdate) { if (attemptUpdate) {
fetchAndCache(event) fetchAndCache(event)
} }
return cacheResponse return cacheResponse
}
return fetchAndCache(event)
})
)
} }
self.addEventListener("fetch", async (e) => { const neverCache: RegExp[] = [
// Important: this lambda must run synchronously, as the browser will otherwise handle the request /\.html$/,
const event = <ServiceWorkerFetchEvent>e /service-worker/
try { ]
const neverCacheHost : RegExp[] = [
/127\.0\.0\.[0-9]+/,
/\.local/,
/\.gitpod\.io/,
/localhost/
]
async function handleRequest(event: ServiceWorkerFetchEvent) {
const origin = new URL(self.origin) const origin = new URL(self.origin)
const requestUrl = new URL(event.request.url) const requestUrl = new URL(event.request.url)
if (requestUrl.pathname.endsWith("service-worker-version")) { if (requestUrl.pathname.endsWith("service-worker-version")) {
console.log("Sending version number...") console.log("Sending version number...")
await event.respondWith( await event.respondWith(
new Response(JSON.stringify({ "service-worker-version": version })) new Response(JSON.stringify({ "service-worker-version": version })),
) )
return return
} }
if (requestUrl.pathname.endsWith("/service-worker-clear")) { if (requestUrl.pathname.endsWith("/service-worker-clear")) {
const keys = await caches.keys() await clearCaches()
await Promise.all(keys.map((k) => caches.delete(k))) await event.respondWith(
console.log("Cleared caches") new Response(JSON.stringify({ "cache-cleared": true })),
)
return return
} }
const shouldBeCached = const shouldBeCached =
origin.host === requestUrl.host && origin.host === requestUrl.host &&
origin.hostname !== "127.0.0.1" && !neverCacheHost.some(blacklisted => origin.host.match(blacklisted)) &&
origin.hostname !== "localhost" && !neverCache.some(blacklisted => event.request.url.match(blacklisted))
!origin.hostname.endsWith(".local") &&
!origin.host.endsWith(".gitpod.io") &&
origin.pathname.indexOf("service-worker") < 0
if (!shouldBeCached) { if (!shouldBeCached) {
console.debug("Not intercepting ", requestUrl.toString(), origin.host, requestUrl.host) console.debug("Not intercepting ", requestUrl.toString(), origin.host, requestUrl.host)
// We return _without_ calling event.respondWith, which signals the browser that it'll have to handle it himself // We return _without_ calling event.respondWith, which signals the browser that it'll have to handle it himself
return return
} }
await cacheFirst(event) await event.respondWith(await cacheFirst(event))
}
self.addEventListener("fetch", async (e) => {
// Important: this lambda must run synchronously, as the browser will otherwise handle the request
const event: ServiceWorkerFetchEvent = <ServiceWorkerFetchEvent>e
try {
await handleRequest(event)
} catch (e) { } catch (e) {
console.error("CRASH IN SW:", e) console.error("CRASH IN SW:", e)
await event.respondWith(fetch(event.request.url)) await event.respondWith(fetch(event.request.url))