mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08: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