Unregister non-matching serviceworkers (#15834)

* Unregister non-matching serviceworkers

With the addition of the /assets url, users who visited a previous
version of the site now may have two active service workers, one with
the old scope `/` and one with scope `/assets`. This check for
serviceworkers that do not match the current script path and unregisters
them.

Also included is a small refactor to publicpath.js which was simplified
because AssetUrlPrefix is always present now. Also it makes use of the
new joinPaths helper too.

Fixes: https://github.com/go-gitea/gitea/pull/15823
This commit is contained in:
silverwind 2021-05-12 20:36:53 +02:00 committed by GitHub
parent b61092bcb0
commit 8ab815ae93
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 81 additions and 31 deletions

View file

@ -789,6 +789,7 @@ var (
"debug", "debug",
"error", "error",
"explore", "explore",
"favicon.ico",
"ghost", "ghost",
"help", "help",
"install", "install",
@ -807,10 +808,10 @@ var (
"repo", "repo",
"robots.txt", "robots.txt",
"search", "search",
"serviceworker.js",
"stars", "stars",
"template", "template",
"user", "user",
"favicon.ico",
} }
reservedUserPatterns = []string{"*.keys", "*.gpg"} reservedUserPatterns = []string{"*.keys", "*.gpg"}

View file

@ -1,18 +1,26 @@
const {UseServiceWorker, AppSubUrl, AppVer} = window.config; import {joinPaths} from '../utils.js';
const cachePrefix = 'static-cache-v'; // actual version is set in the service worker script
async function unregister() { const {UseServiceWorker, AppSubUrl, AssetUrlPrefix, AppVer} = window.config;
const registrations = await navigator.serviceWorker.getRegistrations(); const cachePrefix = 'static-cache-v'; // actual version is set in the service worker script
await Promise.all(registrations.map((registration) => { const workerAssetPath = joinPaths(AssetUrlPrefix, 'serviceworker.js');
return registration.active && registration.unregister();
})); async function unregisterAll() {
for (const registration of await navigator.serviceWorker.getRegistrations()) {
if (registration.active) await registration.unregister();
}
}
async function unregisterOtherWorkers() {
for (const registration of await navigator.serviceWorker.getRegistrations()) {
const scriptURL = registration.active?.scriptURL || '';
if (!scriptURL.endsWith(workerAssetPath)) await registration.unregister();
}
} }
async function invalidateCache() { async function invalidateCache() {
const cacheKeys = await caches.keys(); for (const key of await caches.keys()) {
await Promise.all(cacheKeys.map((key) => { if (key.startsWith(cachePrefix)) caches.delete(key);
return key.startsWith(cachePrefix) && caches.delete(key); }
}));
} }
async function checkCacheValidity() { async function checkCacheValidity() {
@ -30,24 +38,20 @@ export default async function initServiceWorker() {
if (!('serviceWorker' in navigator)) return; if (!('serviceWorker' in navigator)) return;
if (UseServiceWorker) { if (UseServiceWorker) {
// unregister all service workers where scriptURL does not match the current one
await unregisterOtherWorkers();
try { try {
// normally we'd serve the service worker as a static asset from AssetUrlPrefix but // normally we'd serve the service worker as a static asset from AssetUrlPrefix but
// the spec strictly requires it to be same-origin so it has to be AppSubUrl to work // the spec strictly requires it to be same-origin so it has to be AppSubUrl to work
await Promise.all([ await checkCacheValidity();
checkCacheValidity(), await navigator.serviceWorker.register(joinPaths(AppSubUrl, workerAssetPath));
navigator.serviceWorker.register(`${AppSubUrl}/assets/serviceworker.js`),
]);
} catch (err) { } catch (err) {
console.error(err); console.error(err);
await Promise.all([ await invalidateCache();
invalidateCache(), await unregisterAll();
unregister(),
]);
} }
} else { } else {
await Promise.all([ await invalidateCache();
invalidateCache(), await unregisterAll();
unregister(),
]);
} }
} }

View file

@ -1,10 +1,6 @@
// This sets up the URL prefix used in webpack's chunk loading. // This sets up the URL prefix used in webpack's chunk loading.
// This file must be imported before any lazy-loading is being attempted. // This file must be imported before any lazy-loading is being attempted.
import {joinPaths} from './utils.js';
const {AssetUrlPrefix} = window.config; const {AssetUrlPrefix} = window.config;
if (AssetUrlPrefix) { __webpack_public_path__ = joinPaths(AssetUrlPrefix, '/');
__webpack_public_path__ = AssetUrlPrefix.endsWith('/') ? AssetUrlPrefix : `${AssetUrlPrefix}/`;
} else {
const url = new URL(document.currentScript.src);
__webpack_public_path__ = url.pathname.replace(/\/[^/]*?\/[^/]*?$/, '/');
}

View file

@ -9,6 +9,16 @@ export function extname(path = '') {
return ext || ''; return ext || '';
} }
// join a list of path segments with slashes, ensuring no double slashes
export function joinPaths(...parts) {
let str = '';
for (const part of parts) {
if (!part) continue;
str = !str ? part : `${str.replace(/\/$/, '')}/${part.replace(/^\//, '')}`;
}
return str;
}
// test whether a variable is an object // test whether a variable is an object
export function isObject(obj) { export function isObject(obj) {
return Object.prototype.toString.call(obj) === '[object Object]'; return Object.prototype.toString.call(obj) === '[object Object]';

View file

@ -1,5 +1,5 @@
import { import {
basename, extname, isObject, uniq, stripTags, basename, extname, isObject, uniq, stripTags, joinPaths,
} from './utils.js'; } from './utils.js';
test('basename', () => { test('basename', () => {
@ -15,6 +15,45 @@ test('extname', () => {
expect(extname('file.js')).toEqual('.js'); expect(extname('file.js')).toEqual('.js');
}); });
test('joinPaths', () => {
expect(joinPaths('', '')).toEqual('');
expect(joinPaths('', 'b')).toEqual('b');
expect(joinPaths('', '/b')).toEqual('/b');
expect(joinPaths('', '/b/')).toEqual('/b/');
expect(joinPaths('a', '')).toEqual('a');
expect(joinPaths('/a', '')).toEqual('/a');
expect(joinPaths('/a/', '')).toEqual('/a/');
expect(joinPaths('a', 'b')).toEqual('a/b');
expect(joinPaths('a', '/b')).toEqual('a/b');
expect(joinPaths('/a', '/b')).toEqual('/a/b');
expect(joinPaths('/a', '/b')).toEqual('/a/b');
expect(joinPaths('/a/', '/b')).toEqual('/a/b');
expect(joinPaths('/a', '/b/')).toEqual('/a/b/');
expect(joinPaths('/a/', '/b/')).toEqual('/a/b/');
expect(joinPaths('', '', '')).toEqual('');
expect(joinPaths('', 'b', '')).toEqual('b');
expect(joinPaths('', 'b', 'c')).toEqual('b/c');
expect(joinPaths('', '', 'c')).toEqual('c');
expect(joinPaths('', '/b', '/c')).toEqual('/b/c');
expect(joinPaths('/a', '', '/c')).toEqual('/a/c');
expect(joinPaths('/a', '/b', '')).toEqual('/a/b');
expect(joinPaths('', '/')).toEqual('/');
expect(joinPaths('a', '/')).toEqual('a/');
expect(joinPaths('', '/', '/')).toEqual('/');
expect(joinPaths('/', '/')).toEqual('/');
expect(joinPaths('/', '')).toEqual('/');
expect(joinPaths('/', 'b')).toEqual('/b');
expect(joinPaths('/', 'b/')).toEqual('/b/');
expect(joinPaths('/', '', '/')).toEqual('/');
expect(joinPaths('/', 'b', '/')).toEqual('/b/');
expect(joinPaths('/', 'b/', '/')).toEqual('/b/');
expect(joinPaths('a', '/', '/')).toEqual('a/');
expect(joinPaths('/', '/', 'c')).toEqual('/c');
expect(joinPaths('/', '/', 'c/')).toEqual('/c/');
});
test('isObject', () => { test('isObject', () => {
expect(isObject({})).toBeTrue(); expect(isObject({})).toBeTrue();
expect(isObject([])).toBeFalse(); expect(isObject([])).toBeFalse();