Backend: add apt-get install libsqlcipher-dev before pip install so pysqlcipher3 builds in the runner image. Frontend: prep.test.ts was missing a qa mock (fetchFor now calls 5 endpoints in parallel; tests only mocked 4 — 5th returned undefined, threw in catch, research.value never set). survey.test.ts: analyze() was refactored from sync-result to async-task+poll; update test to mock POST then poll completion. Also remove Classic UI (Streamlit) button from AppNav — Streamlit is deprecated and the button caused an unrecoverable redirect loop.
193 lines
8.9 KiB
TypeScript
193 lines
8.9 KiB
TypeScript
import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'
|
|
import { setActivePinia, createPinia } from 'pinia'
|
|
import { usePrepStore } from './prep'
|
|
|
|
// Mock useApiFetch
|
|
vi.mock('../composables/useApi', () => ({
|
|
useApiFetch: vi.fn(),
|
|
}))
|
|
|
|
import { useApiFetch } from '../composables/useApi'
|
|
|
|
describe('usePrepStore', () => {
|
|
beforeEach(() => {
|
|
setActivePinia(createPinia())
|
|
vi.useFakeTimers()
|
|
})
|
|
|
|
afterEach(() => {
|
|
vi.useRealTimers()
|
|
vi.clearAllMocks()
|
|
})
|
|
|
|
it('fetchFor loads research, contacts, task, and full job in parallel', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { company_brief: 'Acme', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: '2026-03-20T12:00:00' }, error: null }) // research
|
|
.mockResolvedValueOnce({ data: [], error: null }) // contacts
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null }) // task
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Engineer', company: 'Acme', url: null,
|
|
description: 'Build things.', cover_letter: null, match_score: 80,
|
|
keyword_gaps: null }, error: null }) // fullJob
|
|
|
|
const store = usePrepStore()
|
|
await store.fetchFor(1)
|
|
|
|
expect(store.research?.company_brief).toBe('Acme')
|
|
expect(store.contacts).toEqual([])
|
|
expect(store.taskStatus.status).toBe('none')
|
|
expect(store.fullJob?.description).toBe('Build things.')
|
|
expect(store.currentJobId).toBe(1)
|
|
})
|
|
|
|
it('fetchFor clears state when called for a different job', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
// First call for job 1
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { company_brief: 'OldCo', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: null }, error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null })
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Old Job', company: 'OldCo', url: null,
|
|
description: null, cover_letter: null, match_score: null, keyword_gaps: null }, error: null })
|
|
|
|
const store = usePrepStore()
|
|
await store.fetchFor(1)
|
|
expect(store.research?.company_brief).toBe('OldCo')
|
|
|
|
// Second call for job 2 - clears first
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: null, error: { kind: 'http', status: 404, detail: '' } }) // 404 → null
|
|
.mockResolvedValueOnce({ data: [], error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null })
|
|
.mockResolvedValueOnce({ data: { id: 2, title: 'New Job', company: 'NewCo', url: null,
|
|
description: null, cover_letter: null, match_score: null, keyword_gaps: null }, error: null })
|
|
|
|
await store.fetchFor(2)
|
|
expect(store.research).toBeNull()
|
|
expect(store.currentJobId).toBe(2)
|
|
})
|
|
|
|
it('generateResearch calls POST then starts polling', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
mockApiFetch.mockResolvedValueOnce({ data: { task_id: 7, is_new: true }, error: null })
|
|
|
|
const store = usePrepStore()
|
|
store.currentJobId = 1
|
|
|
|
// Spy on pollTask via the interval
|
|
const pollSpy = mockApiFetch
|
|
.mockResolvedValueOnce({ data: { status: 'running', stage: 'Analyzing', message: null }, error: null })
|
|
|
|
await store.generateResearch(1)
|
|
|
|
// Advance timer one tick — should poll
|
|
await vi.advanceTimersByTimeAsync(3000)
|
|
|
|
// Should have called POST generate + poll task
|
|
expect(mockApiFetch).toHaveBeenCalledWith(
|
|
expect.stringContaining('/research/generate'),
|
|
expect.objectContaining({ method: 'POST' })
|
|
)
|
|
})
|
|
|
|
it('pollTask stops when status is completed and re-fetches research', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
// Set up store with a job loaded
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { company_brief: 'Acme', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: null }, error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null })
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Eng', company: 'Acme', url: null,
|
|
description: null, cover_letter: null, match_score: null, keyword_gaps: null }, error: null })
|
|
|
|
const store = usePrepStore()
|
|
await store.fetchFor(1)
|
|
|
|
// Mock first poll → completed
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { status: 'completed', stage: null, message: null }, error: null })
|
|
// re-fetch on completed: research, contacts, qa, task, fullJob
|
|
.mockResolvedValueOnce({ data: { company_brief: 'Updated!', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: '2026-03-20T13:00:00' }, error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'completed', stage: null, message: null }, error: null })
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Eng', company: 'Acme', url: null,
|
|
description: 'Now with content', cover_letter: null, match_score: null, keyword_gaps: null }, error: null })
|
|
|
|
store.pollTask(1)
|
|
await vi.advanceTimersByTimeAsync(3000)
|
|
|
|
expect(store.research?.company_brief).toBe('Updated!')
|
|
})
|
|
|
|
it('clear cancels polling interval and resets state', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { company_brief: 'Acme', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: null }, error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null })
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null })
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Eng', company: 'Acme', url: null,
|
|
description: null, cover_letter: null, match_score: null, keyword_gaps: null }, error: null })
|
|
|
|
const store = usePrepStore()
|
|
await store.fetchFor(1)
|
|
store.pollTask(1)
|
|
|
|
store.clear()
|
|
|
|
// Advance timers — if polling wasn't cancelled, fetchFor would be called again
|
|
const callCountBeforeClear = mockApiFetch.mock.calls.length
|
|
await vi.advanceTimersByTimeAsync(9000)
|
|
expect(mockApiFetch.mock.calls.length).toBe(callCountBeforeClear)
|
|
|
|
expect(store.research).toBeNull()
|
|
expect(store.contacts).toEqual([])
|
|
expect(store.contactsError).toBeNull()
|
|
expect(store.currentJobId).toBeNull()
|
|
})
|
|
|
|
it('fetchFor sets contactsError and leaves other data intact when contacts fetch fails', async () => {
|
|
const mockApiFetch = vi.mocked(useApiFetch)
|
|
mockApiFetch
|
|
.mockResolvedValueOnce({ data: { company_brief: 'Acme', ceo_brief: null, talking_points: null,
|
|
tech_brief: null, funding_brief: null, red_flags: null, accessibility_brief: null,
|
|
generated_at: '2026-03-20T12:00:00' }, error: null }) // research OK
|
|
.mockResolvedValueOnce({ data: null, error: { kind: 'http', status: 500, detail: 'DB error' } }) // contacts fail
|
|
.mockResolvedValueOnce({ data: [], error: null }) // qa
|
|
.mockResolvedValueOnce({ data: { status: 'none', stage: null, message: null }, error: null }) // task OK
|
|
.mockResolvedValueOnce({ data: { id: 1, title: 'Engineer', company: 'Acme', url: null,
|
|
description: 'Build things.', cover_letter: null, match_score: 80,
|
|
keyword_gaps: null }, error: null }) // fullJob OK
|
|
|
|
const store = usePrepStore()
|
|
await store.fetchFor(1)
|
|
|
|
// Contacts error shown in Email tab only
|
|
expect(store.contactsError).toBe('Could not load email history.')
|
|
expect(store.contacts).toEqual([])
|
|
|
|
// Everything else still renders
|
|
expect(store.research?.company_brief).toBe('Acme')
|
|
expect(store.fullJob?.description).toBe('Build things.')
|
|
expect(store.fullJob?.match_score).toBe(80)
|
|
expect(store.taskStatus.status).toBe('none')
|
|
|
|
// Top-level error stays null (no full-panel blank-out)
|
|
expect(store.error).toBeNull()
|
|
})
|
|
})
|