mirror of
https://github.com/azaion/ui.git
synced 2026-06-21 20:21:13 +00:00
38eb87fb08
Scaffolds the Blackbox test project per AZ-456 / environment.md across
the three profiles:
- fast : Vitest 3.x + jsdom + MSW 2.x + RTL/jest-dom; tests/setup.ts
boots the MSW Node server with onUnhandledRequest:'error',
afterEach resets handlers, clears bearer + navigate-to-login
spy. Default handlers ship for every suite service plus OWM
and tile stand-ins. Fixtures mirror seed_* in test-data.md.
- e2e : Playwright ^1.49 with chromium + firefox projects against the
suite docker-compose stack; owm-stub + tile-stub Bun servers,
playwright-runner image, seeds.sql for the test-db.
- static: scripts/run-tests.sh extended — tsc --noEmit (test config),
vite build, ripgrep checks (with grep -r fallback), CSV
report at test-output/static-report.csv per AC-7 columns.
Smoke tests cover AC-3, AC-4 (fast, 5 tests, PASS) and AC-1, AC-2,
AC-5, AC-8 (e2e, gated by Risk 4 docker availability). Static profile
(13 checks) PASS — STC-SEC1 (no literal OWM key) lifted from
QUARANTINE per AZ-447 with a narrowed pattern.
Files:
+24 tests/**, +10 e2e/**, +vitest.config.ts, +tsconfig.test.json
~package.json (test scripts + devDeps for vitest, @testing-library/*,
msw, @playwright/test, jsdom, @types/node, @vitest/coverage-v8)
~scripts/run-tests.sh, scripts/run-performance-tests.sh — switched
RESULTS_DIR to test-output/, compose path to project-local
~.gitignore — added /test-output/
Verification:
bun run test:fast → 11 / 11 PASS
./scripts/run-tests.sh → static 13/13 + fast 11/11 PASS, exit 0
Tracker: AZ-456 → In Testing.
Co-authored-by: Cursor <cursoragent@cursor.com>
60 lines
2.0 KiB
TypeScript
60 lines
2.0 KiB
TypeScript
// SSE stand-in for the fast profile. MSW 2.x does not have first-class
|
|
// EventSource support (see AZ-456 Risk 3); jsdom does not ship one either.
|
|
// `simulateSseStream` returns a fake EventSource that tests inject in place
|
|
// of the global where production code allows it (e2e Playwright covers
|
|
// SSE end-to-end where the real EventSource is exercised against the suite).
|
|
|
|
export interface SseEvent<T = unknown> {
|
|
event?: string
|
|
data: T
|
|
id?: string
|
|
}
|
|
|
|
export interface FakeEventSource extends EventTarget {
|
|
readyState: 0 | 1 | 2
|
|
url: string
|
|
close(): void
|
|
emit<T>(e: SseEvent<T>): void
|
|
emitError(err?: Event): void
|
|
}
|
|
|
|
const READY_CONNECTING = 0 as const
|
|
const READY_OPEN = 1 as const
|
|
const READY_CLOSED = 2 as const
|
|
|
|
export function createFakeEventSource(url = 'about:blank'): FakeEventSource {
|
|
const target = new EventTarget() as FakeEventSource
|
|
;(target as { readyState: 0 | 1 | 2 }).readyState = READY_CONNECTING
|
|
;(target as { url: string }).url = url
|
|
|
|
// Move to "open" on the next microtask so listeners attached synchronously
|
|
// after construction still see the open event (matches the production
|
|
// EventSource handshake behavior closely enough for assertions).
|
|
queueMicrotask(() => {
|
|
;(target as { readyState: 0 | 1 | 2 }).readyState = READY_OPEN
|
|
target.dispatchEvent(new Event('open'))
|
|
})
|
|
|
|
target.close = () => {
|
|
;(target as { readyState: 0 | 1 | 2 }).readyState = READY_CLOSED
|
|
}
|
|
target.emit = <T>(e: SseEvent<T>) => {
|
|
if (target.readyState !== READY_OPEN) return
|
|
const payload = typeof e.data === 'string' ? e.data : JSON.stringify(e.data)
|
|
const message = new MessageEvent(e.event ?? 'message', { data: payload, lastEventId: e.id })
|
|
target.dispatchEvent(message)
|
|
}
|
|
target.emitError = (err?: Event) => {
|
|
target.dispatchEvent(err ?? new Event('error'))
|
|
}
|
|
return target
|
|
}
|
|
|
|
export function simulateSseStream<T = unknown>(events: Array<SseEvent<T>>): FakeEventSource {
|
|
const source = createFakeEventSource()
|
|
queueMicrotask(() => {
|
|
for (const e of events) source.emit(e)
|
|
})
|
|
return source
|
|
}
|