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:
		
							parent
							
								
									b61092bcb0
								
							
						
					
					
						commit
						8ab815ae93
					
				
					 5 changed files with 81 additions and 31 deletions
				
			
		|  | @ -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"} | ||||||
|  |  | ||||||
|  | @ -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(), |  | ||||||
|     ]); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -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(/\/[^/]*?\/[^/]*?$/, '/'); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  | @ -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]'; | ||||||
|  |  | ||||||
|  | @ -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(); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue