Initial commit - Church Music Database

This commit is contained in:
2026-01-27 18:04:50 -06:00
commit d367261867
336 changed files with 103545 additions and 0 deletions

View File

@@ -0,0 +1,199 @@
/* Service Worker for Church Music Management System */
const CACHE_NAME = "church-music-v1";
const API_CACHE_NAME = "church-music-api-v1";
// Static assets to cache immediately
const STATIC_ASSETS = [
"/",
"/index.html",
"/static/css/main.css",
"/static/js/main.js",
"/favicon.ico",
"/manifest.json",
];
// API endpoints to cache with time-based expiration
const API_CACHE_DURATION = 3 * 60 * 1000; // 3 minutes
// Install event - cache static assets
self.addEventListener("install", (event) => {
console.log("[Service Worker] Installing...");
event.waitUntil(
caches
.open(CACHE_NAME)
.then((cache) => {
console.log("[Service Worker] Caching static assets");
// Don't fail if some assets are missing
return Promise.allSettled(
STATIC_ASSETS.map((url) =>
cache
.add(url)
.catch((err) =>
console.log(`[Service Worker] Failed to cache ${url}:`, err)
)
)
);
})
.then(() => self.skipWaiting()) // Activate immediately
);
});
// Activate event - clean up old caches
self.addEventListener("activate", (event) => {
console.log("[Service Worker] Activating...");
event.waitUntil(
caches
.keys()
.then((cacheNames) => {
return Promise.all(
cacheNames
.filter((name) => name !== CACHE_NAME && name !== API_CACHE_NAME)
.map((name) => {
console.log("[Service Worker] Deleting old cache:", name);
return caches.delete(name);
})
);
})
.then(() => self.clients.claim()) // Take control immediately
);
});
// Fetch event - serve from cache with network fallback
self.addEventListener("fetch", (event) => {
const { request } = event;
const url = new URL(request.url);
// Skip cross-origin requests
if (url.origin !== location.origin) {
return;
}
// Handle API requests separately
if (url.pathname.startsWith("/api/")) {
event.respondWith(handleApiRequest(request));
return;
}
// Handle static assets with cache-first strategy
event.respondWith(
caches.match(request).then((cachedResponse) => {
if (cachedResponse) {
console.log("[Service Worker] Serving from cache:", request.url);
return cachedResponse;
}
// Not in cache, fetch from network
return fetch(request)
.then((response) => {
// Cache successful responses
if (response && response.status === 200) {
const responseClone = response.clone();
caches.open(CACHE_NAME).then((cache) => {
cache.put(request, responseClone);
});
}
return response;
})
.catch((error) => {
console.error("[Service Worker] Fetch failed:", error);
// Return offline page if available
return caches.match("/offline.html").catch(() => {
return new Response("Offline - Please check your connection", {
status: 503,
statusText: "Service Unavailable",
headers: new Headers({
"Content-Type": "text/plain",
}),
});
});
});
})
);
});
// Handle API requests with network-first, cache-fallback strategy
async function handleApiRequest(request) {
const url = new URL(request.url);
// Only cache GET requests
if (request.method !== "GET") {
return fetch(request);
}
try {
// Try network first
const response = await fetch(request);
if (response && response.status === 200) {
// Clone and cache the response with timestamp
const responseClone = response.clone();
const cache = await caches.open(API_CACHE_NAME);
// Add timestamp header for expiration
const cachedResponse = new Response(await responseClone.blob(), {
status: responseClone.status,
statusText: responseClone.statusText,
headers: {
...Object.fromEntries(responseClone.headers.entries()),
"sw-cached-at": Date.now().toString(),
},
});
await cache.put(request, cachedResponse);
console.log("[Service Worker] Cached API response:", url.pathname);
}
return response;
} catch (error) {
console.log("[Service Worker] Network failed, trying cache:", url.pathname);
// Network failed, try cache
const cachedResponse = await caches.match(request);
if (cachedResponse) {
// Check if cache is still fresh
const cachedAt = cachedResponse.headers.get("sw-cached-at");
const now = Date.now();
if (cachedAt && now - parseInt(cachedAt) < API_CACHE_DURATION) {
console.log("[Service Worker] Serving fresh cached API response");
return cachedResponse;
} else {
console.log("[Service Worker] Cached API response expired");
// Return stale cache with warning header
return new Response(await cachedResponse.blob(), {
status: cachedResponse.status,
statusText: cachedResponse.statusText,
headers: {
...Object.fromEntries(cachedResponse.headers.entries()),
"sw-cache-status": "stale",
},
});
}
}
// No cache available
throw error;
}
}
// Listen for messages from the client
self.addEventListener("message", (event) => {
if (event.data && event.data.type === "SKIP_WAITING") {
self.skipWaiting();
}
if (event.data && event.data.type === "CLEAR_CACHE") {
caches
.keys()
.then((cacheNames) => {
return Promise.all(cacheNames.map((name) => caches.delete(name)));
})
.then(() => {
event.ports[0].postMessage({ success: true });
});
}
});