-
Notifications
You must be signed in to change notification settings - Fork 1.7k
/
serviceWorker.js
96 lines (92 loc) · 4.04 KB
/
serviceWorker.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
async function handleFetch(request) {
// Perform the original fetch request and store the result in order to modify the response.
try {
var r = await fetch(request);
}
catch (e) {
console.error(e)
}
if (r.status === 0) {
return r;
}
// We add headers to the original response its headers, in order to enable cross-origin-isolation. And make it independent of the server config.
const newHeaders = new Headers(r.headers);
// COEP & COOP for cross-origin-isolation.
newHeaders.set("Cross-Origin-Embedder-Policy", "require-corp");
newHeaders.set("Cross-Origin-Opener-Policy", "same-origin");
newHeaders.set("Cross-Origin-Resource-Policy", "cross-origin");
/**
* This workaround is necessary due to a limitation of CheerpOS, which relies on the response URL being set to the resolved URL.
* When constructing a new response object, the URL is not set by the Response() constructor and the serviceworker respondwith() method will set the url to event.request.url in case of an empty string.
* To address this, we set the location URL to the resolved response URL and set the status code to 301 in the new Response object.
* This causes the request to bounce back to the serviceworker from Cheerpos, with the event.request.url now set to the resolved URL, which allows the respondWith method to properly set the response URL in our new response.
* https://developer.mozilla.org/en-US/docs/Web/API/FetchEvent/respondWith.
*/
if (r.redirected === true)
newHeaders.set("location", r.url);
// In case of a redirection, we set the status to 301, and body to null, in order to not transfer too much data needlessly
const moddedResponse = new Response(r.redirected === true ? null : r.body, {
headers: newHeaders,
status: r.redirected === true ? 301 : r.status,
statusText: r.statusText,
});
return moddedResponse;
}
function serviceWorkerInit() {
// Init the service worker.
self.addEventListener("install", () => self.skipWaiting());
self.addEventListener("activate", e => e.waitUntil(self.clients.claim()));
// Listen for fetch requests and call handleFetch function.
self.addEventListener("fetch", function (e) {
try {
e.respondWith(handleFetch(e.request));
} catch (err) {
console.log("Serviceworker NetworkError:" + err);
}
});
}
async function doRegister() {
try {
const registration = await navigator.serviceWorker.register(window.document.currentScript.src);
console.log("Service Worker registered", registration.scope);
// EventListener to make sure that the page gets reloaded when a new serviceworker gets installed.
// f.e on first access.
registration.addEventListener("updatefound", () => {
console.log("Reloading the page to transfer control to the Service Worker.");
try {
window.location.reload();
} catch (err) {
console.log("Service Worker failed reloading the page. ERROR:" + err);
};
});
// When the registration is active, but it's not controlling the page, we reload the page to have it take control.
// This f.e occurs when you hard-reload (shift + refresh). https://www.w3.org/TR/service-workers/#navigator-service-worker-controller
if (registration.active && !navigator.serviceWorker.controller) {
console.log("Reloading the page to transfer control to the Service Worker.");
try {
window.location.reload();
} catch (err) {
console.log("Service Worker failed reloading the page. ERROR:" + err);
};
}
}
catch {
console.error("Service Worker failed to register:", e)
}
}
async function serviceWorkerRegister() {
if (window.crossOriginIsolated) return;
if (!window.isSecureContext) {
console.log("Service Worker not registered, a secure context is required.");
return;
}
// Register the service worker and reload the page to transfer control to the serviceworker.
if ("serviceWorker" in navigator)
await doRegister();
else
console.log("Service worker is not supported in this browser");
}
if (typeof window === 'undefined') // If the script is running in a Service Worker context
serviceWorkerInit()
else // If the script is running in the browser context
serviceWorkerRegister();