Gateprep / frontend /src /context /AuthContext.test.jsx
banu4prasad's picture
add: new tests
672a9eb
Raw
History Blame Contribute Delete
7.43 kB
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'
import { render, screen, cleanup, act, waitFor } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { MemoryRouter } from 'react-router-dom'
import { AuthProvider, useAuth } from './AuthContext'
// ── Mocks ────────────────────────────────────────────────────────
const mockNavigate = vi.fn()
vi.mock('react-router-dom', async () => {
const actual = await vi.importActual('react-router-dom')
return { ...actual, useNavigate: () => mockNavigate }
})
vi.mock('swr', () => ({ mutate: vi.fn() }))
const mockGet = vi.fn()
const mockPost = vi.fn()
vi.mock('../api/client', () => ({
default: { get: (...args) => mockGet(...args), post: (...args) => mockPost(...args) },
AUTH_UNAUTHORIZED_EVENT: 'gateprep:auth-unauthorized',
startTokenRefresh: vi.fn(),
stopTokenRefresh: vi.fn(),
}))
// Helper to consume and display context values
function AuthConsumer() {
const { user, loading, sessionExpired, logout, saveUser } = useAuth()
return (
<div>
<span data-testid="loading">{String(loading)}</span>
<span data-testid="user">{user ? JSON.stringify(user) : 'null'}</span>
<span data-testid="session-expired">{String(sessionExpired)}</span>
<button data-testid="logout-btn" onClick={logout}>Logout</button>
<button data-testid="save-btn" onClick={() => saveUser({ id: 99, email: 'new@test.com', role: 'student', full_name: 'New' })}>Save</button>
</div>
)
}
function renderWithRouter(ui) {
return render(<MemoryRouter>{ui}</MemoryRouter>)
}
beforeEach(() => {
vi.clearAllMocks()
})
afterEach(() => {
cleanup()
})
// ── Tests ────────────────────────────────────────────────────────
describe('AuthProvider', () => {
it('initialises: fetches /auth/me and sets user on success', async () => {
const userData = { id: 1, email: 'a@b.com', role: 'admin', full_name: 'Admin' }
mockGet.mockResolvedValueOnce({ data: userData })
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
// Initially loading
expect(screen.getByTestId('loading').textContent).toBe('true')
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
expect(screen.getByTestId('user').textContent).toContain('"email":"a@b.com"')
const { startTokenRefresh } = await import('../api/client')
expect(startTokenRefresh).toHaveBeenCalledWith({ refreshNow: true })
})
it('initialises: sets user to null when /auth/me fails', async () => {
mockGet.mockRejectedValueOnce(new Error('Unauthorized'))
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
expect(screen.getByTestId('user').textContent).toBe('null')
})
it('logout: clears user, stops refresh, and navigates to /login', async () => {
const userData = { id: 1, email: 'a@b.com', role: 'admin', full_name: 'Admin' }
mockGet.mockResolvedValueOnce({ data: userData })
mockPost.mockResolvedValueOnce({}) // logout API call
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
await act(async () => {
await userEvent.click(screen.getByTestId('logout-btn'))
})
const { stopTokenRefresh } = await import('../api/client')
expect(stopTokenRefresh).toHaveBeenCalled()
expect(screen.getByTestId('user').textContent).toBe('null')
expect(mockNavigate).toHaveBeenCalledWith('/login', { replace: true })
})
it('logout: still clears state even if API call fails', async () => {
mockGet.mockResolvedValueOnce({ data: { id: 1, email: 'a@b.com', role: 'student', full_name: 'S' } })
mockPost.mockRejectedValueOnce(new Error('network error'))
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
await act(async () => {
await userEvent.click(screen.getByTestId('logout-btn'))
})
// User should still be cleared despite API failure
expect(screen.getByTestId('user').textContent).toBe('null')
expect(mockNavigate).toHaveBeenCalledWith('/login', { replace: true })
})
it('saveUser: sets user from provided data and starts token refresh', async () => {
mockGet.mockRejectedValueOnce(new Error('no session')) // initial load fails
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
await act(async () => {
await userEvent.click(screen.getByTestId('save-btn'))
})
expect(screen.getByTestId('user').textContent).toContain('"email":"new@test.com"')
expect(screen.getByTestId('session-expired').textContent).toBe('false')
const { startTokenRefresh } = await import('../api/client')
expect(startTokenRefresh).toHaveBeenCalled()
})
it('session expired: shows dialog on AUTH_UNAUTHORIZED_EVENT', async () => {
mockGet.mockResolvedValueOnce({ data: { id: 1, email: 'a@b.com', role: 'admin', full_name: 'A' } })
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
// Fire the unauthorized event
act(() => {
window.dispatchEvent(new CustomEvent('gateprep:auth-unauthorized'))
})
expect(screen.getByTestId('session-expired').textContent).toBe('true')
expect(screen.getByText('Session expired')).toBeTruthy()
expect(screen.getByText('Sign in')).toBeTruthy()
})
it('confirmSessionExpired: clicking "Sign in" clears state and navigates to /login', async () => {
mockGet.mockResolvedValueOnce({ data: { id: 1, email: 'a@b.com', role: 'admin', full_name: 'A' } })
renderWithRouter(
<AuthProvider><AuthConsumer /></AuthProvider>
)
await waitFor(() => {
expect(screen.getByTestId('loading').textContent).toBe('false')
})
act(() => {
window.dispatchEvent(new CustomEvent('gateprep:auth-unauthorized'))
})
await act(async () => {
await userEvent.click(screen.getByText('Sign in'))
})
const { stopTokenRefresh } = await import('../api/client')
expect(stopTokenRefresh).toHaveBeenCalled()
expect(screen.getByTestId('user').textContent).toBe('null')
expect(screen.getByTestId('session-expired').textContent).toBe('false')
expect(mockNavigate).toHaveBeenCalledWith('/login', expect.objectContaining({ replace: true }))
})
})
describe('useAuth', () => {
it('throws if used outside AuthProvider', () => {
// Suppress React error boundary console noise
const spy = vi.spyOn(console, 'error').mockImplementation(() => {})
function Naked() {
useAuth()
return null
}
expect(() => render(<Naked />)).toThrow('useAuth must be used within AuthProvider')
spy.mockRestore()
})
})