mirror of
https://github.com/immich-app/immich
synced 2025-11-14 17:36:12 +00:00
116 lines
2.9 KiB
TypeScript
116 lines
2.9 KiB
TypeScript
import { version } from '$service-worker';
|
|
|
|
const useCache = true;
|
|
const CACHE = `cache-${version}`;
|
|
|
|
export const isURL = (request: URL | RequestInfo): request is URL => (request as URL).href !== undefined;
|
|
export const isRequest = (request: RequestInfo): request is Request => (request as Request).url !== undefined;
|
|
|
|
export async function deleteOldCaches() {
|
|
for (const key of await caches.keys()) {
|
|
if (key !== CACHE) {
|
|
await caches.delete(key);
|
|
}
|
|
}
|
|
}
|
|
|
|
const pendingLoads = new Map<string, AbortController>();
|
|
|
|
export async function cancelLoad(urlString: string) {
|
|
const pending = pendingLoads.get(urlString);
|
|
if (pending) {
|
|
pending.abort();
|
|
pendingLoads.delete(urlString);
|
|
}
|
|
}
|
|
|
|
export async function getCachedOrFetch(request: URL | Request | string, cancelable: boolean = false) {
|
|
const cached = await checkCache(request);
|
|
if (cached.response) {
|
|
return cached.response;
|
|
}
|
|
|
|
try {
|
|
if (!cancelable) {
|
|
const response = await fetch(request);
|
|
checkResponse(response);
|
|
return response;
|
|
}
|
|
|
|
return await fetchWithCancellation(request, cached.cache);
|
|
} catch {
|
|
return new Response(undefined, {
|
|
status: 499,
|
|
statusText: 'Request canceled: Instructions unclear, accidentally interrupted myself',
|
|
});
|
|
}
|
|
}
|
|
|
|
async function fetchWithCancellation(request: URL | Request | string, cache: Cache) {
|
|
const cacheKey = getCacheKey(request);
|
|
const cancelToken = new AbortController();
|
|
|
|
try {
|
|
pendingLoads.set(cacheKey, cancelToken);
|
|
const response = await fetch(request, {
|
|
signal: cancelToken.signal,
|
|
});
|
|
|
|
checkResponse(response);
|
|
setCached(response, cache, cacheKey);
|
|
return response;
|
|
} finally {
|
|
pendingLoads.delete(cacheKey);
|
|
}
|
|
}
|
|
|
|
async function checkCache(url: URL | Request | string) {
|
|
if (!useCache) {
|
|
return;
|
|
}
|
|
const cache = await caches.open(CACHE);
|
|
const response = await cache.match(url);
|
|
return { cache, response };
|
|
}
|
|
|
|
async function setCached(response: Response, cache: Cache, cacheKey: URL | Request | string) {
|
|
if (response.status === 200) {
|
|
cache.put(cacheKey, response.clone());
|
|
}
|
|
}
|
|
|
|
function checkResponse(response: Response) {
|
|
if (!(response instanceof Response)) {
|
|
throw new TypeError('invalid response from fetch');
|
|
}
|
|
}
|
|
|
|
function getCacheKey(request: URL | Request | string) {
|
|
if (isURL(request)) {
|
|
return request.toString();
|
|
} else if (isRequest(request)) {
|
|
return request.url;
|
|
} else {
|
|
return request;
|
|
}
|
|
}
|
|
|
|
function isAssetRequest(pathname: string): boolean {
|
|
return /^\/api\/assets\/[a-f0-9-]+\/(original|thumbnail)/.test(pathname);
|
|
}
|
|
|
|
export function handleFetchEvent(event: FetchEvent): void {
|
|
if (event.request.method !== 'GET') {
|
|
return;
|
|
}
|
|
|
|
const url = new URL(event.request.url);
|
|
if (url.origin !== self.location.origin) {
|
|
return;
|
|
}
|
|
|
|
if (isAssetRequest(url.pathname)) {
|
|
event.respondWith(getCachedOrFetch(event.request, true));
|
|
return;
|
|
}
|
|
}
|