mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-31 03:18:24 +00:00 
			
		
		
		
	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:
		| @@ -1,18 +1,26 @@ | ||||
| const {UseServiceWorker, AppSubUrl, AppVer} = window.config; | ||||
| const cachePrefix = 'static-cache-v'; // actual version is set in the service worker script | ||||
| import {joinPaths} from '../utils.js'; | ||||
|  | ||||
| async function unregister() { | ||||
|   const registrations = await navigator.serviceWorker.getRegistrations(); | ||||
|   await Promise.all(registrations.map((registration) => { | ||||
|     return registration.active && registration.unregister(); | ||||
|   })); | ||||
| const {UseServiceWorker, AppSubUrl, AssetUrlPrefix, AppVer} = window.config; | ||||
| const cachePrefix = 'static-cache-v'; // actual version is set in the service worker script | ||||
| const workerAssetPath = joinPaths(AssetUrlPrefix, 'serviceworker.js'); | ||||
|  | ||||
| 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() { | ||||
|   const cacheKeys = await caches.keys(); | ||||
|   await Promise.all(cacheKeys.map((key) => { | ||||
|     return key.startsWith(cachePrefix) && caches.delete(key); | ||||
|   })); | ||||
|   for (const key of await caches.keys()) { | ||||
|     if (key.startsWith(cachePrefix)) caches.delete(key); | ||||
|   } | ||||
| } | ||||
|  | ||||
| async function checkCacheValidity() { | ||||
| @@ -30,24 +38,20 @@ export default async function initServiceWorker() { | ||||
|   if (!('serviceWorker' in navigator)) return; | ||||
|  | ||||
|   if (UseServiceWorker) { | ||||
|     // unregister all service workers where scriptURL does not match the current one | ||||
|     await unregisterOtherWorkers(); | ||||
|     try { | ||||
|       // 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 | ||||
|       await Promise.all([ | ||||
|         checkCacheValidity(), | ||||
|         navigator.serviceWorker.register(`${AppSubUrl}/assets/serviceworker.js`), | ||||
|       ]); | ||||
|       await checkCacheValidity(); | ||||
|       await navigator.serviceWorker.register(joinPaths(AppSubUrl, workerAssetPath)); | ||||
|     } catch (err) { | ||||
|       console.error(err); | ||||
|       await Promise.all([ | ||||
|         invalidateCache(), | ||||
|         unregister(), | ||||
|       ]); | ||||
|       await invalidateCache(); | ||||
|       await unregisterAll(); | ||||
|     } | ||||
|   } else { | ||||
|     await Promise.all([ | ||||
|       invalidateCache(), | ||||
|       unregister(), | ||||
|     ]); | ||||
|     await invalidateCache(); | ||||
|     await unregisterAll(); | ||||
|   } | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user