mirror of
https://github.com/immich-app/immich.git
synced 2025-11-20 21:02:34 +09:00
* Migrate from npm to pnpm across entire project • Update all GitHub workflow files to use pnpm instead of npm • Replace npm commands with pnpm equivalents in devcontainer scripts • Remove package-lock.json files and update to use pnpm-lock.yaml • Consolidate node version references to use server/.nvmrc * Refine pnpm migration based on review feedback • Replace SKIP_SHARP_FILTERING with SHARP_IGNORE_GLOBAL_LIBVIPS environment variable • Improve Sharp package filtering to include specific Linux architectures for Docker builds • Optimize Dockerfile dependency caching with improved layer structure • Clean up workspace configuration and remove redundant settings * Address additional review feedback for pnpm migration • Fix node-version-file paths in GitHub workflow configurations • Refactor .pnpmfile.cjs to use switch statement for better code organization • Correct cache type typo in fix-format workflow • Simplify Vite configuration by merging configs inline • Update package description for consistency * Use 'server/.nvmrc' for fix-format.yml GHA * Delete npm locks * Remove Docker volume isolation for node_modules directories • Remove volume mounts for node_modules and related directories • Allow shared access between host and container filesystem • Update init container to handle file ownership with conditional existence check * Remove unused Docker volumes and volume mounts • Remove node_modules volume mounts from devcontainer configuration • Remove unused named volumes for pnpm-store, node_modules, and cache directories • Clean up Docker Compose configuration after removing volume isolation * Fix typescript-sdk package issues • Remove unknown "build" dependency that was incorrectly added to package.json • Update pnpm-lock.yaml to reflect dependency removal * Add pnpm setup to mobile workflow for translation formatting • Add pnpm action setup step to mobile unit tests workflow • Required for translation file formatting and sorting operations --------- Co-authored-by: Jason Rasmussen <jason@rasm.me>
114 lines
3.0 KiB
TypeScript
114 lines
3.0 KiB
TypeScript
import { version } from '$service-worker';
|
|
import { APP_RESOURCES, checkCache, getCacheKey, setCached } from './cache';
|
|
|
|
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) {
|
|
const response = await checkCache(request);
|
|
if (response) {
|
|
return response;
|
|
}
|
|
|
|
try {
|
|
return await fetchWithCancellation(request);
|
|
} catch {
|
|
return new Response(undefined, {
|
|
status: 499,
|
|
statusText: 'Request canceled: Instructions unclear, accidentally interrupted myself',
|
|
});
|
|
}
|
|
}
|
|
|
|
async function fetchWithCancellation(request: URL | Request | string) {
|
|
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, cacheKey);
|
|
return response;
|
|
} finally {
|
|
pendingLoads.delete(cacheKey);
|
|
}
|
|
}
|
|
|
|
function checkResponse(response: Response) {
|
|
if (!(response instanceof Response)) {
|
|
throw new TypeError('Fetch did not return a valid Response object');
|
|
}
|
|
}
|
|
|
|
function isIgnoredFileType(pathname: string): boolean {
|
|
return /\.(png|ico|txt|json|ts|ttf|css|js|svelte)$/.test(pathname);
|
|
}
|
|
|
|
function isIgnoredPath(pathname: string): boolean {
|
|
return (
|
|
/^\/(src|api)(\/.*)?$/.test(pathname) || /node_modules/.test(pathname) || /^\/@(vite|id)(\/.*)?$/.test(pathname)
|
|
);
|
|
}
|
|
|
|
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);
|
|
|
|
// Only handle requests to the same origin
|
|
if (url.origin !== self.location.origin) {
|
|
return;
|
|
}
|
|
|
|
// Do not cache app resources
|
|
if (APP_RESOURCES.includes(url.pathname)) {
|
|
return;
|
|
}
|
|
|
|
// Cache requests for thumbnails
|
|
if (isAssetRequest(url.pathname)) {
|
|
event.respondWith(getCachedOrFetch(event.request));
|
|
return;
|
|
}
|
|
|
|
// Do not cache ignored file types or paths
|
|
if (isIgnoredFileType(url.pathname) || isIgnoredPath(url.pathname)) {
|
|
return;
|
|
}
|
|
|
|
// At this point, the only remaining requests for top level routes
|
|
// so serve the Svelte SPA fallback page
|
|
const slash = new URL('/', url.origin);
|
|
event.respondWith(getCachedOrFetch(slash));
|
|
}
|