const CACHE_VERSION = 'orly-v1'; const STATIC_ASSETS = [ '/', '/index.html', '/bundle.js', '/bundle.css', '/global.css', '/favicon.png', '/icon-192.png', '/icon-512.png', '/orly.png' ]; self.addEventListener('install', (event) => { event.waitUntil( caches.open(CACHE_VERSION).then((cache) => { return cache.addAll(STATIC_ASSETS); }) ); self.skipWaiting(); }); self.addEventListener('activate', (event) => { event.waitUntil( caches.keys().then((cacheNames) => { return Promise.all( cacheNames .filter((name) => name !== CACHE_VERSION) .map((name) => caches.delete(name)) ); }) ); self.clients.claim(); }); self.addEventListener('fetch', (event) => { const url = new URL(event.request.url); // Skip WebSocket requests if (url.protocol === 'ws:' || url.protocol === 'wss:') { return; } // Skip non-GET requests if (event.request.method !== 'GET') { return; } // API calls: network-first with cache fallback if (url.pathname.startsWith('/api/')) { event.respondWith( fetch(event.request) .then((response) => { if (response.ok) { const clone = response.clone(); caches.open(CACHE_VERSION).then((cache) => { cache.put(event.request, clone); }); } return response; }) .catch(() => { return caches.match(event.request); }) ); return; } // Static assets: cache-first with network fallback event.respondWith( caches.match(event.request).then((cached) => { if (cached) { // Update cache in background fetch(event.request).then((response) => { if (response.ok) { caches.open(CACHE_VERSION).then((cache) => { cache.put(event.request, response); }); } }).catch(() => {}); return cached; } return fetch(event.request).then((response) => { if (response.ok) { const clone = response.clone(); caches.open(CACHE_VERSION).then((cache) => { cache.put(event.request, clone); }); } return response; }); }) ); });