mirror of
https://github.com/koala73/worldmonitor.git
synced 2026-05-14 02:56:21 +02:00
359 lines
13 KiB
TypeScript
359 lines
13 KiB
TypeScript
import { expect, test } from '@playwright/test';
|
|
|
|
test.describe('desktop runtime routing guardrails', () => {
|
|
test('detectDesktopRuntime covers packaged tauri hosts', async ({ page }) => {
|
|
await page.goto('/tests/runtime-harness.html');
|
|
|
|
const result = await page.evaluate(async () => {
|
|
const runtime = await import('/src/services/runtime.ts');
|
|
return {
|
|
tauriHost: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'https:',
|
|
locationHost: 'tauri.localhost',
|
|
locationOrigin: 'https://tauri.localhost',
|
|
}),
|
|
tauriScheme: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'tauri:',
|
|
locationHost: '',
|
|
locationOrigin: 'tauri://localhost',
|
|
}),
|
|
tauriUa: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0 Tauri/2.0',
|
|
locationProtocol: 'https:',
|
|
locationHost: 'example.com',
|
|
locationOrigin: 'https://example.com',
|
|
}),
|
|
tauriGlobal: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: true,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'https:',
|
|
locationHost: 'example.com',
|
|
locationOrigin: 'https://example.com',
|
|
}),
|
|
secureLocalhost: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'https:',
|
|
locationHost: 'localhost',
|
|
locationOrigin: 'https://localhost',
|
|
}),
|
|
insecureLocalhost: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'http:',
|
|
locationHost: 'localhost:5173',
|
|
locationOrigin: 'http://localhost:5173',
|
|
}),
|
|
webHost: runtime.detectDesktopRuntime({
|
|
hasTauriGlobals: false,
|
|
userAgent: 'Mozilla/5.0',
|
|
locationProtocol: 'https:',
|
|
locationHost: 'worldmonitor.app',
|
|
locationOrigin: 'https://worldmonitor.app',
|
|
}),
|
|
};
|
|
});
|
|
|
|
expect(result.tauriHost).toBe(true);
|
|
expect(result.tauriScheme).toBe(true);
|
|
expect(result.tauriUa).toBe(true);
|
|
expect(result.tauriGlobal).toBe(true);
|
|
expect(result.secureLocalhost).toBe(true);
|
|
expect(result.insecureLocalhost).toBe(false);
|
|
expect(result.webHost).toBe(false);
|
|
});
|
|
|
|
test('runtime fetch patch falls back to cloud for local failures', async ({ page }) => {
|
|
await page.goto('/tests/runtime-harness.html');
|
|
|
|
const result = await page.evaluate(async () => {
|
|
const runtime = await import('/src/services/runtime.ts');
|
|
const globalWindow = window as unknown as Record<string, unknown>;
|
|
const originalFetch = window.fetch.bind(window);
|
|
|
|
const calls: string[] = [];
|
|
const responseJson = (body: unknown, status = 200) =>
|
|
new Response(JSON.stringify(body), {
|
|
status,
|
|
headers: { 'content-type': 'application/json' },
|
|
});
|
|
|
|
window.fetch = (async (input: RequestInfo | URL) => {
|
|
const url =
|
|
typeof input === 'string'
|
|
? input
|
|
: input instanceof URL
|
|
? input.toString()
|
|
: input.url;
|
|
|
|
calls.push(url);
|
|
|
|
if (url.includes('127.0.0.1:46123/api/fred-data')) {
|
|
return responseJson({ error: 'missing local api key' }, 500);
|
|
}
|
|
if (url.includes('worldmonitor.app/api/fred-data')) {
|
|
return responseJson({ observations: [{ value: '321.5' }] }, 200);
|
|
}
|
|
|
|
if (url.includes('127.0.0.1:46123/api/stablecoin-markets')) {
|
|
throw new Error('ECONNREFUSED');
|
|
}
|
|
if (url.includes('worldmonitor.app/api/stablecoin-markets')) {
|
|
return responseJson({ stablecoins: [{ symbol: 'USDT' }] }, 200);
|
|
}
|
|
|
|
return responseJson({ ok: true }, 200);
|
|
}) as typeof window.fetch;
|
|
|
|
const previousTauri = globalWindow.__TAURI__;
|
|
globalWindow.__TAURI__ = { core: { invoke: () => Promise.resolve(null) } };
|
|
delete globalWindow.__wmFetchPatched;
|
|
|
|
try {
|
|
runtime.installRuntimeFetchPatch();
|
|
|
|
const fredResponse = await window.fetch('/api/fred-data?series_id=CPIAUCSL');
|
|
const fredBody = await fredResponse.json() as { observations?: Array<{ value: string }> };
|
|
|
|
const stableResponse = await window.fetch('/api/stablecoin-markets');
|
|
const stableBody = await stableResponse.json() as { stablecoins?: Array<{ symbol: string }> };
|
|
|
|
return {
|
|
fredStatus: fredResponse.status,
|
|
fredValue: fredBody.observations?.[0]?.value ?? null,
|
|
stableStatus: stableResponse.status,
|
|
stableSymbol: stableBody.stablecoins?.[0]?.symbol ?? null,
|
|
calls,
|
|
};
|
|
} finally {
|
|
window.fetch = originalFetch;
|
|
delete globalWindow.__wmFetchPatched;
|
|
if (previousTauri === undefined) {
|
|
delete globalWindow.__TAURI__;
|
|
} else {
|
|
globalWindow.__TAURI__ = previousTauri;
|
|
}
|
|
}
|
|
});
|
|
|
|
expect(result.fredStatus).toBe(200);
|
|
expect(result.fredValue).toBe('321.5');
|
|
expect(result.stableStatus).toBe(200);
|
|
expect(result.stableSymbol).toBe('USDT');
|
|
|
|
expect(result.calls.some((url) => url.includes('127.0.0.1:46123/api/fred-data'))).toBe(true);
|
|
expect(result.calls.some((url) => url.includes('worldmonitor.app/api/fred-data'))).toBe(true);
|
|
expect(result.calls.some((url) => url.includes('127.0.0.1:46123/api/stablecoin-markets'))).toBe(true);
|
|
expect(result.calls.some((url) => url.includes('worldmonitor.app/api/stablecoin-markets'))).toBe(true);
|
|
});
|
|
|
|
test('update badge picks architecture-correct desktop download url', async ({ page }) => {
|
|
await page.goto('/tests/runtime-harness.html');
|
|
|
|
const result = await page.evaluate(async () => {
|
|
const { App } = await import('/src/App.ts');
|
|
const globalWindow = window as unknown as {
|
|
__TAURI__?: { core?: { invoke?: (command: string) => Promise<unknown> } };
|
|
};
|
|
const previousTauri = globalWindow.__TAURI__;
|
|
const releaseUrl = 'https://github.com/koala73/worldmonitor/releases/latest';
|
|
|
|
const appProto = App.prototype as unknown as {
|
|
resolveUpdateDownloadUrl: (releaseUrl: string) => Promise<string>;
|
|
mapDesktopDownloadPlatform: (os: string, arch: string) => string | null;
|
|
};
|
|
const fakeApp = {
|
|
mapDesktopDownloadPlatform: appProto.mapDesktopDownloadPlatform,
|
|
};
|
|
|
|
try {
|
|
globalWindow.__TAURI__ = {
|
|
core: {
|
|
invoke: async (command: string) => {
|
|
if (command !== 'get_desktop_runtime_info') throw new Error(`Unexpected command: ${command}`);
|
|
return { os: 'macos', arch: 'aarch64' };
|
|
},
|
|
},
|
|
};
|
|
const macArm = await appProto.resolveUpdateDownloadUrl.call(fakeApp, releaseUrl);
|
|
|
|
globalWindow.__TAURI__ = {
|
|
core: {
|
|
invoke: async () => ({ os: 'windows', arch: 'amd64' }),
|
|
},
|
|
};
|
|
const windowsX64 = await appProto.resolveUpdateDownloadUrl.call(fakeApp, releaseUrl);
|
|
|
|
globalWindow.__TAURI__ = {
|
|
core: {
|
|
invoke: async () => ({ os: 'linux', arch: 'x86_64' }),
|
|
},
|
|
};
|
|
const linuxFallback = await appProto.resolveUpdateDownloadUrl.call(fakeApp, releaseUrl);
|
|
|
|
return { macArm, windowsX64, linuxFallback };
|
|
} finally {
|
|
if (previousTauri === undefined) {
|
|
delete globalWindow.__TAURI__;
|
|
} else {
|
|
globalWindow.__TAURI__ = previousTauri;
|
|
}
|
|
}
|
|
});
|
|
|
|
expect(result.macArm).toBe('https://worldmonitor.app/api/download?platform=macos-arm64');
|
|
expect(result.windowsX64).toBe('https://worldmonitor.app/api/download?platform=windows-exe');
|
|
expect(result.linuxFallback).toBe('https://github.com/koala73/worldmonitor/releases/latest');
|
|
});
|
|
|
|
test('loadMarkets keeps Yahoo-backed data when Finnhub is skipped', async ({ page }) => {
|
|
await page.goto('/tests/runtime-harness.html');
|
|
|
|
const result = await page.evaluate(async () => {
|
|
const { App } = await import('/src/App.ts');
|
|
const originalFetch = window.fetch.bind(window);
|
|
|
|
const calls: string[] = [];
|
|
const toUrl = (input: RequestInfo | URL): string => {
|
|
if (typeof input === 'string') return new URL(input, window.location.origin).toString();
|
|
if (input instanceof URL) return input.toString();
|
|
return new URL(input.url, window.location.origin).toString();
|
|
};
|
|
const responseJson = (body: unknown, status = 200) =>
|
|
new Response(JSON.stringify(body), {
|
|
status,
|
|
headers: { 'content-type': 'application/json' },
|
|
});
|
|
|
|
const yahooChart = (symbol: string) => {
|
|
const base = symbol.length * 100;
|
|
return {
|
|
chart: {
|
|
result: [{
|
|
meta: {
|
|
regularMarketPrice: base + 1,
|
|
previousClose: base,
|
|
},
|
|
indicators: {
|
|
quote: [{ close: [base - 2, base - 1, base, base + 1] }],
|
|
},
|
|
}],
|
|
},
|
|
};
|
|
};
|
|
|
|
const marketRenders: number[] = [];
|
|
const marketConfigErrors: string[] = [];
|
|
const heatmapRenders: number[] = [];
|
|
const heatmapConfigErrors: string[] = [];
|
|
const commoditiesRenders: number[] = [];
|
|
const commoditiesConfigErrors: string[] = [];
|
|
const cryptoRenders: number[] = [];
|
|
const apiStatuses: Array<{ name: string; status: string }> = [];
|
|
|
|
window.fetch = (async (input: RequestInfo | URL) => {
|
|
const url = toUrl(input);
|
|
calls.push(url);
|
|
const parsed = new URL(url);
|
|
|
|
if (parsed.pathname === '/api/finnhub') {
|
|
return responseJson({
|
|
quotes: [],
|
|
skipped: true,
|
|
reason: 'FINNHUB_API_KEY not configured',
|
|
});
|
|
}
|
|
|
|
if (parsed.pathname === '/api/yahoo-finance') {
|
|
const symbol = parsed.searchParams.get('symbol') ?? 'UNKNOWN';
|
|
return responseJson(yahooChart(symbol));
|
|
}
|
|
|
|
if (parsed.pathname === '/api/coingecko') {
|
|
return responseJson([
|
|
{ id: 'bitcoin', current_price: 50000, price_change_percentage_24h: 1.2, sparkline_in_7d: { price: [1, 2, 3] } },
|
|
{ id: 'ethereum', current_price: 3000, price_change_percentage_24h: -0.5, sparkline_in_7d: { price: [1, 2, 3] } },
|
|
{ id: 'solana', current_price: 120, price_change_percentage_24h: 2.1, sparkline_in_7d: { price: [1, 2, 3] } },
|
|
]);
|
|
}
|
|
|
|
return responseJson({});
|
|
}) as typeof window.fetch;
|
|
|
|
const fakeApp = {
|
|
latestMarkets: [] as Array<unknown>,
|
|
panels: {
|
|
markets: {
|
|
renderMarkets: (data: Array<unknown>) => marketRenders.push(data.length),
|
|
showConfigError: (message: string) => marketConfigErrors.push(message),
|
|
},
|
|
heatmap: {
|
|
renderHeatmap: (data: Array<unknown>) => heatmapRenders.push(data.length),
|
|
showConfigError: (message: string) => heatmapConfigErrors.push(message),
|
|
},
|
|
commodities: {
|
|
renderCommodities: (data: Array<unknown>) => commoditiesRenders.push(data.length),
|
|
showConfigError: (message: string) => commoditiesConfigErrors.push(message),
|
|
},
|
|
crypto: {
|
|
renderCrypto: (data: Array<unknown>) => cryptoRenders.push(data.length),
|
|
},
|
|
},
|
|
statusPanel: {
|
|
updateApi: (name: string, payload: { status?: string }) => {
|
|
apiStatuses.push({ name, status: payload.status ?? '' });
|
|
},
|
|
},
|
|
};
|
|
|
|
try {
|
|
await (App.prototype as unknown as { loadMarkets: (thisArg: unknown) => Promise<void> })
|
|
.loadMarkets.call(fakeApp);
|
|
|
|
const commoditySymbols = ['^VIX', 'GC=F', 'CL=F', 'NG=F', 'SI=F', 'HG=F'];
|
|
const commodityYahooCalls = commoditySymbols.map((symbol) =>
|
|
calls.some((url) => {
|
|
const parsed = new URL(url);
|
|
return parsed.pathname === '/api/yahoo-finance' && parsed.searchParams.get('symbol') === symbol;
|
|
})
|
|
);
|
|
|
|
return {
|
|
marketRenders,
|
|
marketConfigErrors,
|
|
heatmapRenders,
|
|
heatmapConfigErrors,
|
|
commoditiesRenders,
|
|
commoditiesConfigErrors,
|
|
cryptoRenders,
|
|
apiStatuses,
|
|
latestMarketsCount: fakeApp.latestMarkets.length,
|
|
commodityYahooCalls,
|
|
};
|
|
} finally {
|
|
window.fetch = originalFetch;
|
|
}
|
|
});
|
|
|
|
expect(result.marketRenders.some((count) => count > 0)).toBe(true);
|
|
expect(result.latestMarketsCount).toBeGreaterThan(0);
|
|
expect(result.marketConfigErrors.length).toBe(0);
|
|
|
|
expect(result.heatmapRenders.length).toBe(0);
|
|
expect(result.heatmapConfigErrors).toEqual(['FINNHUB_API_KEY not configured — add in Settings']);
|
|
|
|
expect(result.commoditiesRenders.some((count) => count > 0)).toBe(true);
|
|
expect(result.commoditiesConfigErrors.length).toBe(0);
|
|
expect(result.commodityYahooCalls.every(Boolean)).toBe(true);
|
|
|
|
expect(result.cryptoRenders.some((count) => count > 0)).toBe(true);
|
|
expect(result.apiStatuses.some((entry) => entry.name === 'Finnhub' && entry.status === 'error')).toBe(true);
|
|
expect(result.apiStatuses.some((entry) => entry.name === 'CoinGecko' && entry.status === 'ok')).toBe(true);
|
|
});
|
|
});
|