veza/apps/web/src/router/index.test.tsx
2025-12-12 21:34:34 -05:00

251 lines
6.9 KiB
TypeScript

import { describe, it, expect, beforeEach, vi } from 'vitest';
import { render, screen } from '@testing-library/react';
import { MemoryRouter } from 'react-router-dom';
import { AppRouter } from './index';
import { useAuthStore } from '@/stores/auth';
// Mock the stores
vi.mock('@/stores/auth', () => ({
useAuthStore: vi.fn(),
}));
// Mock lazy components
vi.mock('@/components/ui/LazyComponent', () => ({
LazyLogin: () => <div>Login Page</div>,
LazyRegister: () => <div>Register Page</div>,
LazyForgotPassword: () => <div>Forgot Password Page</div>,
LazyDashboard: () => <div>Dashboard Page</div>,
LazyChat: () => <div>Chat Page</div>,
LazyLibrary: () => <div>Library Page</div>,
LazyProfile: () => <div>Profile Page</div>,
LazySettings: () => <div>Settings Page</div>,
LazyNotFound: () => <div>404 Not Found</div>,
LazyServerError: () => <div>500 Server Error</div>,
}));
// Mock Layout component
vi.mock('@/components/layout/Layout', () => ({
Layout: ({ children }: { children: React.ReactNode }) => (
<div data-testid="layout">{children}</div>
),
}));
// Mock LoadingSpinner
vi.mock('@/components/ui/loading-spinner', () => ({
LoadingSpinner: () => <div>Loading...</div>,
}));
describe('AppRouter', () => {
beforeEach(() => {
vi.clearAllMocks();
});
describe('Public Routes', () => {
it('should render login page for /login route when not authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/login']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('Login Page')).toBeInTheDocument();
});
it('should redirect to dashboard when authenticated user tries to access login', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: true,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/login']}>
<AppRouter />
</MemoryRouter>,
);
// Should redirect to dashboard (mocked as Dashboard Page)
expect(screen.getByText('Dashboard Page')).toBeInTheDocument();
});
it('should render register page for /register route when not authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/register']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('Register Page')).toBeInTheDocument();
});
});
describe('Protected Routes', () => {
it('should render dashboard page when authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: true,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/dashboard']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByTestId('layout')).toBeInTheDocument();
expect(screen.getByText('Dashboard Page')).toBeInTheDocument();
});
it('should redirect to login when not authenticated user tries to access protected route', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/dashboard']}>
<AppRouter />
</MemoryRouter>,
);
// Should redirect to login
expect(screen.getByText('Login Page')).toBeInTheDocument();
});
it('should show loading spinner when checking authentication', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: true,
} as any);
render(
<MemoryRouter initialEntries={['/dashboard']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('Loading...')).toBeInTheDocument();
});
it('should render chat page when authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: true,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/chat']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByTestId('layout')).toBeInTheDocument();
expect(screen.getByText('Chat Page')).toBeInTheDocument();
});
it('should render library page when authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: true,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/library']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByTestId('layout')).toBeInTheDocument();
expect(screen.getByText('Library Page')).toBeInTheDocument();
});
});
describe('Error Routes', () => {
it('should render 404 page for /404 route', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/404']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('404 Not Found')).toBeInTheDocument();
});
it('should render 500 page for /500 route', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/500']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('500 Server Error')).toBeInTheDocument();
});
it('should redirect to 404 for unknown routes', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/unknown-route']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('404 Not Found')).toBeInTheDocument();
});
});
describe('Default Routes', () => {
it('should redirect root path to dashboard when authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: true,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/']}>
<AppRouter />
</MemoryRouter>,
);
expect(screen.getByText('Dashboard Page')).toBeInTheDocument();
});
it('should redirect root path to login when not authenticated', () => {
vi.mocked(useAuthStore).mockReturnValue({
isAuthenticated: false,
isLoading: false,
} as any);
render(
<MemoryRouter initialEntries={['/']}>
<AppRouter />
</MemoryRouter>,
);
// Should redirect to dashboard, which then redirects to login
expect(screen.getByText('Login Page')).toBeInTheDocument();
});
});
});