Scrum4Me/tests/setup.ts
Madhura68 48d1e11a2a feat(PBI-74): product-workspace store skelet + test-infra (Story 1)
Skelet voor de nieuwe `product-workspace-store` die op termijn de gefragmenteerde
`backlog-store`/`planner-store`/`selection-store`/`product-store` vervangt. Deze
PR levert alleen het skelet + tests; UI-consumers worden in latere stories
omgezet.

- vitest naar jsdom + tests/setup.ts (MemoryStorage, default fetch-stub) — G6/G8
- stores/product-workspace/{types,store,selectors,restore}.ts — immer-middleware,
  alle slices en acties (hydrate, setActive*, ensure*Loaded met activeRequestId-
  guard, applyRealtimeEvent, resyncActiveScopes/loadedScopes, optimistic
  mutations). Restore-wiring in setters volgt in Story 4 (T-857/T-858).
- selectors gebruiken module-level EMPTY refs (G1) en documenteren useShallow-
  vereiste (G2)
- 34 nieuwe unit-tests dekken §Testing setup-checklist uit het ontwerp:
  hydrateSnapshot, selection-cascade, applyRealtimeEvent (I/U/D + parent-move +
  ander-product + unknown-entity → resync), delete-cleanup, race-safe loaders,
  ensureTaskLoaded _detail-flag, resyncActiveScopes ensure-keten, restore-hints
  read/write/clear, optimistic mutation rollback/settle/SSE-echo idempotent
- docs/api/rest-contract.md: audit-sectie met de vier ontbrekende
  ensure*Loaded-endpoints (worden toegevoegd in Story 7 / T-870)

Refs: PBI-74, ST-1318, T-837..T-843
Bron-ontwerp: docs/plans/zustand-store-rearchitecture.md
Implementatieplan: docs/plans/zustand-workspace-store-implementation.md

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-10 00:55:22 +02:00

84 lines
2.4 KiB
TypeScript

import { beforeEach, vi } from 'vitest'
// G6: vitest 4 + jsdom 29 mist localStorage/sessionStorage op globalThis.
// MemoryStorage-binding zodat tests zonder echte browser draaien.
class MemoryStorage implements Storage {
private store = new Map<string, string>()
get length(): number {
return this.store.size
}
clear(): void {
this.store.clear()
}
getItem(key: string): string | null {
return this.store.has(key) ? this.store.get(key)! : null
}
key(index: number): string | null {
return Array.from(this.store.keys())[index] ?? null
}
removeItem(key: string): void {
this.store.delete(key)
}
setItem(key: string, value: string): void {
this.store.set(key, String(value))
}
}
const localStorageMemory = new MemoryStorage()
const sessionStorageMemory = new MemoryStorage()
Object.defineProperty(globalThis, 'localStorage', {
value: localStorageMemory,
configurable: true,
writable: true,
})
Object.defineProperty(globalThis, 'sessionStorage', {
value: sessionStorageMemory,
configurable: true,
writable: true,
})
if (typeof window !== 'undefined') {
Object.defineProperty(window, 'localStorage', {
value: localStorageMemory,
configurable: true,
writable: true,
})
Object.defineProperty(window, 'sessionStorage', {
value: sessionStorageMemory,
configurable: true,
writable: true,
})
}
// G7: maak globalThis.fetch herconfigureerbaar zodat vi.spyOn / vi.fn-stubs
// de eigenschap kunnen redefineren. Default Node 24 / jsdom 29 binding is
// vaak non-configurable, wat vi.spyOn(globalThis, 'fetch') laat falen.
const originalFetch = (globalThis as { fetch?: typeof fetch }).fetch
Object.defineProperty(globalThis, 'fetch', {
value: originalFetch,
configurable: true,
writable: true,
})
beforeEach(() => {
localStorageMemory.clear()
sessionStorageMemory.clear()
vi.restoreAllMocks()
// Default fetch-stub voorkomt dat fire-and-forget ensure*Loaded calls
// (b.v. via setActivePbi) lekken naar het echte network. Tests die
// specifieke responses willen overrulen dit met vi.spyOn/vi.fn.
// G8: mockImplementation (niet mockResolvedValue) zodat elke call een
// verse Response krijgt — body wordt anders maar één keer leesbaar.
;(globalThis as { fetch: typeof fetch }).fetch = vi
.fn()
.mockImplementation(() =>
Promise.resolve(new Response('null', { status: 200 })),
) as unknown as typeof fetch
})