Files
worldmonitor/e2e/runtime-fetch.spec.ts

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);
});
});