mirror of
https://github.com/ggml-org/llama.cpp.git
synced 2026-06-27 23:50:20 -05:00
* feat: Add basic PWA support and service worker for offline caching * feat: Vite PWA implementation WIP * feat: Improve PWA icons generation * feat: Add PWA workbox to server routes * feat: Include `version.json` in static assets * feat: Add HTTP cache headers for PWA static assets * feat: Update app name for `apple-mobile-web-app-title` * feat: Implement PWA versioning and automatic update detection * chore: Update `.gitignore` files * feat: Splash Screens * feat: Add dark mode favicon support * refactor: Cleanup * fix: Use dark logo for dark splash screens * refactor: Simplify favicons SVG code * fix: Adjust caching and polling for reliable service worker updates * fix: Add missing favicon entry * fix: Align PWA service worker configuration with SvelteKit build structure * fix: Replace hashed bundle paths with versioned static paths * test: Add PWA tests * ci: Add build output for unit tests * refactor: Cleanup * fix: Server build & release versioning * chore: Update package-lock.json * chore: Increase PWA cache size * chore: Update packages * feat: Update favicons * refactor: Post-merge fix * feat: support explicit build version for PWA cache busting * fix: CI * feat: Improve PWA Refresh Alert UI * feat: Add toggleable build version display * refactor: Cleanup * feat: Add version mismatch detection and manual app reload * refactor: replace dynamic imports with static * refactor: Cleanup * feat: Add safe space for `pwa-<size>.png` rendered icons * fix: use relative paths for PWA assets to support base path deployment * feat: add PWA mode detection via URL query parameter * feat: Use ?cache=true for SW-cached PWA assets * refactor: Build process cleanup * refactor: Decouple PWA versioning and remove ?cache=true workaround * chore: Update README logo * feat: Include PWA Assets generation in build script * refactor: `usePwa` hook for core layout * fix: Relativize base vite plugin * fix: remove unnecessary backslash escapes in test regexes * test: update static asset paths for API Key test * refactor: Move SvelteKit PWA Options config to constants * ui: fix update notification never appearing Keep the PWA hook object intact instead of destructuring needRefreshByStorage, which freezes the reactive getter. Also exclude loading.html from PWA precache to prevent 404 errors and broken SW installation.
107 lines
3.7 KiB
TypeScript
107 lines
3.7 KiB
TypeScript
import { expect, test } from '@playwright/test';
|
|
|
|
test.describe('PWA Service Worker', () => {
|
|
test('service worker is registered', async ({ page }) => {
|
|
await page.goto('/');
|
|
|
|
const swURL = await page.evaluate(async () => {
|
|
const registration = await Promise.race([
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore - type inference differs from browser runtime
|
|
navigator.serviceWorker.ready,
|
|
new Promise((_, reject) =>
|
|
setTimeout(() => reject(new Error('Service worker registration failed: timeout')), 15000)
|
|
)
|
|
]);
|
|
// @ts-expect-error registration is of type unknown
|
|
return registration.active?.scriptURL;
|
|
});
|
|
|
|
expect(swURL).toBeTruthy();
|
|
expect(swURL).toContain('/sw.js');
|
|
});
|
|
|
|
test('service worker has precache configured', async ({ page }) => {
|
|
await page.goto('/');
|
|
|
|
await page.evaluate(async () => {
|
|
await navigator.serviceWorker.ready;
|
|
});
|
|
|
|
const swActive = await page.evaluate(async () => {
|
|
const reg = await navigator.serviceWorker.ready;
|
|
return reg.active?.scriptURL ?? null;
|
|
});
|
|
|
|
expect(swActive).toBeTruthy();
|
|
|
|
const swResponse = await page.request.get(swActive!);
|
|
const swContent = await swResponse.text();
|
|
|
|
// Precache contains SvelteKit content-hashed bundle paths
|
|
expect(swContent).toMatch(/"_app\/immutable\/bundle\.[a-zA-Z0-9-]+\.js"/);
|
|
expect(swContent).toMatch(/"_app\/immutable\/assets\/bundle\.[a-zA-Z0-9-]+\.css"/);
|
|
expect(swContent).toMatch(/"manifest\.webmanifest"/);
|
|
expect(swContent).toMatch(/"_app\/version\.json"/);
|
|
expect(swContent).toMatch(/NavigationRoute/);
|
|
expect(swContent).toMatch(/api-cache/);
|
|
});
|
|
|
|
test('offline mode - page loads when offline after caching', async ({ browser }) => {
|
|
const context = await browser.newContext();
|
|
const offlinePage = await context.newPage();
|
|
|
|
await offlinePage.goto('/');
|
|
await offlinePage.waitForLoadState('networkidle');
|
|
|
|
await offlinePage.evaluate(async () => {
|
|
await navigator.serviceWorker.ready;
|
|
});
|
|
|
|
await offlinePage.waitForTimeout(2000);
|
|
|
|
await context.setOffline(true);
|
|
await offlinePage.goto('/');
|
|
|
|
const bodyText = await offlinePage.locator('body').textContent();
|
|
expect(bodyText).toBeTruthy();
|
|
|
|
await context.close();
|
|
});
|
|
|
|
test('version.json is accessible and contains version', async ({ page }) => {
|
|
const versionResponse = await page.request.get('/_app/version.json');
|
|
expect(versionResponse.ok()).toBeTruthy();
|
|
|
|
const versionData = await versionResponse.json();
|
|
expect(versionData).toHaveProperty('version');
|
|
expect(typeof versionData.version).toBe('string');
|
|
expect(versionData.version.length).toBeGreaterThan(0);
|
|
});
|
|
|
|
test('manifest.webmanifest is accessible and valid', async ({ page }) => {
|
|
const response = await page.request.get('/manifest.webmanifest');
|
|
expect(response.ok()).toBeTruthy();
|
|
|
|
const manifest = await response.json();
|
|
expect(manifest).toHaveProperty('name', 'llama-ui');
|
|
expect(manifest).toHaveProperty('short_name', 'llama-ui');
|
|
expect(manifest).toHaveProperty('start_url', './');
|
|
expect(manifest).toHaveProperty('display', 'standalone');
|
|
expect(manifest.icons).toBeTruthy();
|
|
expect(manifest.icons.length).toBeGreaterThan(0);
|
|
});
|
|
|
|
test('index.html contains content-hashed bundle references', async ({ page }) => {
|
|
const response = await page.request.get('/');
|
|
expect(response.ok()).toBeTruthy();
|
|
|
|
const html = await response.text();
|
|
|
|
// SvelteKit outputs content-hashed bundle names in _app/immutable/
|
|
expect(html).toMatch(/href="(\.\/|\/)_app\/immutable\/bundle\.[a-zA-Z0-9-]+\.js"/);
|
|
expect(html).toMatch(/href="(\.\/|\/)_app\/immutable\/assets\/bundle\.[a-zA-Z0-9-]+\.css"/);
|
|
expect(html).toMatch(/import\("(\.\/|\/)_app\/immutable\/bundle\.[a-zA-Z0-9-]+\.js"\)/);
|
|
});
|
|
});
|