-
-
Notifications
You must be signed in to change notification settings - Fork 63
/
sw-template.js
357 lines (303 loc) · 12 KB
/
sw-template.js
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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
/* eslint-disable no-console */
import { precacheAndRoute } from "workbox-precaching";
import { registerRoute } from "workbox-routing";
import { CacheFirst, NetworkFirst, Strategy, StrategyHandler } from "workbox-strategies";
import { ExpirationPlugin, CacheExpiration } from "workbox-expiration";
import { createCacheKey } from "workbox-precaching/utils/createCacheKey";
/*
this comment will attempt to explain the caching strategy employed by this service worker
at a high level, it strives to enable a good compromise of speed and freshness without surprising behavior
the runtime manifest provides hashes for the files it contains, preventing stale files from being served.
when loaded, the sw will:
purge any files in the precache that are no longer in the manifest
fetch any files that are missing from the precache
purge any files in the runtime cache that have a different revision from the manifest
purge any images in the external-images cache that have not been recently accessed (7 days)
routes are resolved in descending order.
any file request:
Is it in the Precache Manifest (essential files: script, html, css, some fonts, most data)?
serve it from the cache
Is it in the Runtime Manifest (optional files: images and adventure data)?
Has it been cached?
yes: serve it from the cache
no: fetch it from the network and cache it
Is it a font?
Is it in the font-cache?
yes: serve it from the font cache
no: fetch it from the network and cache it
Is it an image (external images)?
Fetch it from the network, and cache it
If the fetch fails, serve from the cache
Default is load from network.
A file request not caught by any above policy won't be available offline.
*/
// nabbed from https://github.com/GoogleChrome/workbox/blob/0cc6975f17b60d67a71d8b717e73ef7ddb79a891/packages/workbox-core/src/_private/waitUntil.ts#L19-L26
/**
* wait until an event happens
* @param {ExtendableEvent} event event to wait until on
* @param {() => Promise<any>} asyncFn the function to pass to waitUntil
* @returns {Promise<any>}
*/
function waitUntil (
event,
asyncFn,
) {
const returnPromise = asyncFn();
event.waitUntil(returnPromise);
return returnPromise;
}
const offlineAlert = async (url) => {
console.log(`fetch failure - we are offline, cannot access ${url}`);
const clients = await self.clients.matchAll({type: "window"});
let payload = "generic";
if (/\.(?:png|gif|webm|jpg|webp|jpeg|svg)$/m.test(url)) payload = "image";
else if (/\.json$/m.test(url)) payload = "json";
for (const client of clients) {
client.postMessage({type: "FETCH_ERROR", payload});
}
};
/**
* Reset, wiping all caches and un-registering the service worker.
*
* @return {Promise<void>}
*/
const resetAll = async () => {
// Clear all caches, as these persist between service workers
const cacheNames = await caches.keys();
for (const cacheName of cacheNames) {
await caches.delete(cacheName);
// See: https://github.com/GoogleChrome/workbox/issues/2234
const cacheExpiration = new CacheExpiration(cacheName, {maxEntries: 1});
await cacheExpiration.delete();
console.log(`deleted cache "${cacheName}"`);
}
await self.registration.unregister();
const clients = await self.clients.matchAll();
clients.forEach(client => client.navigate(client.url));
};
addEventListener("message", (event) => {
switch (event.data.type) {
case "RESET": {
console.log("Resetting...");
event.waitUntil(resetAll());
break;
}
}
});
/*
routes take precedence in order listed. if a higher route and a lower route both match a file, the higher route will resolve it
https://stackoverflow.com/questions/52423473/workbox-routing-registerroute-idempotence
*/
// the self value is replaced with key: value pair of file: hash, to allow workbox to carry files over between caches if they match
precacheAndRoute(self.__WB_PRECACHE_MANIFEST);
class RevisionCacheFirst extends Strategy {
cacheRoutesAbortController = null;
constructor () {
super({ cacheName: "runtime-revision" });
// bind this for activate method
this.activate = this.activate.bind(this);
this.cacheRoutes = this.cacheRoutes.bind(this);
addEventListener("message", (event) => {
switch (event.data.type) {
case "CACHE_ROUTES": {
this.cacheRoutesAbortController = new AbortController();
event.waitUntil(this.cacheRoutes(event.data, this.cacheRoutesAbortController.signal));
break;
}
case "CANCEL_CACHE_ROUTES": {
console.log("aborting cache!");
this.cacheRoutesAbortController?.abort();
this.cacheRoutesAbortController = null;
break;
}
}
});
}
/**
* @param {Request} request
* @param {StrategyHandler} handler
* @returns {Promise<Response | undefined>}
*/
async _handle (request, handler) {
/** the full url of the request, https://example.com/slug/ */
const url = request.url;
/**
* the route of the url, with a query string for the revision
*
* this way, we can invalidate the cache entry if the revision is wrong
*/
const cacheKey = createCacheKey({url, revision: runtimeManifest.get(url)}).cacheKey;
console.log(`trying to resolve ${url} with key ${cacheKey}`);
const cacheResponse = await handler.cacheMatch(cacheKey);
// undefined is returned if we don't have a cache response for the key
if (cacheResponse !== undefined) return cacheResponse;
// we need to fetch the request from the network and store it with revision for next time
console.log(`fetching ${url} over the network for RevisionFirstCache`);
try {
const fetchResponse = await handler.fetch(request);
// no await because it can happen later
handler.cachePut(cacheKey, fetchResponse.clone());
return fetchResponse;
} catch (e) {
// no await because it can happen later
offlineAlert(url);
// empty response, we cant get the file
return new Response();
}
}
/**
* the cache busting portion of the Strategy.
* Iterate the cache, and remove anything that is not in the manifest, or from a different revision.
*
* call this from the activate event
*
* @param {ExtendableEvent} event
* @returns {Promise}
*/
activate (event) {
return waitUntil(event, async () => {
const cache = await caches.open(this.cacheName);
const currentCacheKeys = (await cache.keys()).map(request => request.url);
const validCacheKeys = new Set(Array.from(runtimeManifest).map(([url, revision]) => createCacheKey({url, revision}).cacheKey));
// queue up all the deletions
await Promise.allSettled(
currentCacheKeys.map(async key => {
// this will happen if a revision is updated or a file is no longer included in the glob
if (!validCacheKeys.has(key)) {
// we can save space by deleting this element - it wouldn't be served bc the revision is wrong
console.log(`deleting ${key} from the cache because its revision does not match`);
await cache.delete(key);
}
}),
);
});
}
/**
* Preload runtime cache routes. This method is called based on a message from the "frontend".
* The method is passed a regex which all possible files are matched against.
* Files which match and have not already been loaded will be attempted to be fetched.
*
* File fetching is done with a pool strategy, where async function closures pop urls off an array.
* If a fetch throws an error, it will not be retried, and it will kill the worker.
* This intentionally reduces the number of concurrent requests.
* If a system or server is struggling under the load, it could cause fetches to fail,
* in a manner that is opaque to this code.
*
* When all the original workers have died or finished, any failures will be reported.
*
* @param {{payload: {routeRegex: RegExp}}} data the data sent with the request
* @param {AbortSignal} signal signal to abort the operation
*/
async cacheRoutes (data, signal) {
const cache = await caches.open(this.cacheName);
const currentCacheKeys = new Set((await cache.keys()).map(request => request.url));
const validCacheKeys = Array.from(runtimeManifest).map(([url, revision]) => createCacheKey({url, revision}).cacheKey);
const routeRegex = data.payload.routeRegex;
/**
* These are the keys which have not been cached yet AND match the regex for routes
*/
const routesToCache = validCacheKeys.filter((key) => !currentCacheKeys.has(key) && routeRegex.test(key));
const fetchTotal = routesToCache.length;
let fetched = 0;
/**
* This is an async function to let clients know the status of route caching.
* It can take up to 1 ms, so it can be called without an await to let it resolve in downtime.
*/
const postProgress = async () => {
const clients = await self.clients.matchAll({type: "window"});
for (const client of clients) {
client.postMessage({type: "CACHE_ROUTES_PROGRESS", payload: {fetched, fetchTotal}});
}
};
// First call, and awaited, so that pages show a loading bar to indicate fetching has started
await postProgress();
// early escape if there is no work to do.
if (fetchTotal === 0) return;
/**
* The number of fetches to run at the same time
*/
const concurrentFetches = 5;
const fetchPromise = async () => {
while (true) {
// each instance of this function will keep popping urls off the array until there are none left
const url = routesToCache.pop();
if (url === undefined || signal.aborted) return;
// this regex is a very bad idea, but it trims the cache version off the url
const cleanUrl = url.replace(/\?__WB_REVISION__=\w+$/m, "");
const response = await fetch(cleanUrl);
// this await could be omitted to further speed up fetching at risk of failure during error
await cache.put(url, response);
fetched++;
postProgress();
}
};
// an array / pool of functions who will fetch files and cache them
const fetchPromises = [];
for (let i = 0; i < concurrentFetches; i++) {
fetchPromises.push(fetchPromise());
}
// wait for each function to die, or empty the routesToCache
const fetchResults = await Promise.allSettled(fetchPromises);
// determine if any functions died and report them
const errorResults = fetchResults.filter(fetchResult => fetchResult.status === "rejected");
if (errorResults.length > 0) {
const clients = await self.clients.matchAll({type: "window"});
for (const client of clients) client.postMessage({type: "CACHE_ROUTES_ERROR", payload: { errors: errorResults }});
}
}
}
/**
* Map([url, revision])
*
* __WB_RUNTIME_MANIFEST is injected as [route, revision] array, mapped into [url, revision], and constructed as map
*/
const runtimeManifest = new Map(self.__WB_RUNTIME_MANIFEST.map(
([
route,
revision,
]) =>
[
`${self.location.origin}/${route}`,
revision,
],
));
const revisionCacheFirst = new RevisionCacheFirst();
registerRoute(
({request}) => runtimeManifest.has(request.url),
revisionCacheFirst,
);
// purge the old entries from cache
addEventListener("activate", revisionCacheFirst.activate);
/*
this tells workbox to cache fonts, and serve them cache first after first load
this works on the assumption that fonts are static assets and won't change
*/
registerRoute(({request}) => request.destination === "font", new CacheFirst({
cacheName: "font-cache",
}));
/*
the base case route - for images that have fallen through every other route
this is external images, for homebrew as an example
*/
registerRoute(({request}) => request.destination === "image", new NetworkFirst({
cacheName: "external-image-cache",
plugins: [
// this is a safeguard against an utterly massive cache - these numbers may need tweaking
new ExpirationPlugin({maxAgeSeconds: 7 /* days */ * 24 * 60 * 60, maxEntries: 100, purgeOnQuotaError: true}),
],
}));
addEventListener("install", () => {
self.skipWaiting();
});
// this only serves to delete cache from old versions of page - pre sw rework
addEventListener("activate", event => {
event.waitUntil((async () => {
const cacheNames = await caches.keys();
for (const cacheName of cacheNames) {
if (/\d+\.\d+\.\d+/.test(cacheName)) {
await caches.delete(cacheName);
console.log(`deleted cache: ${cacheName} because it is from old service worker`);
}
}
})());
});