Files
lcbp3/frontend/components/transmittal/__tests__/transmittal-form.test.tsx
T
admin 7e8f4859cd
CI / CD Pipeline / build (push) Failing after 6m24s
CI / CD Pipeline / deploy (push) Has been skipped
feat(ai): add ADR-036 unified OCR architecture and frontend test coverage
- Add ADR-036 unified OCR architecture (typhoon-ocr via Ollama)
- Extend AI execution profiles for OCR sandbox configuration
- Add comprehensive frontend test coverage (components, hooks, services)
- Add backend test coverage for document-numbering services
- Update OCR sidecar with typhoon-ocr integration
- Add AI policy service and execution profile management
- Update AGENTS.md and architecture documentation
2026-06-14 06:34:07 +07:00

126 lines
5.0 KiB
TypeScript

// File: frontend/components/transmittal/__tests__/transmittal-form.test.tsx
// Change Log
// - 2026-06-13: Add coverage for transmittal form render, cancel, validation, and submit flows.
import { render, screen, waitFor } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { beforeEach, describe, expect, it, vi } from 'vitest';
import { toast } from 'sonner';
import { createTestQueryClient } from '@/lib/test-utils';
import { TransmittalForm } from '../transmittal-form';
import { transmittalService } from '@/lib/services/transmittal.service';
import { correspondenceService } from '@/lib/services/correspondence.service';
import { projectService } from '@/lib/services/project.service';
import { organizationService } from '@/lib/services/organization.service';
const push = vi.fn();
const back = vi.fn();
vi.mock('next/navigation', () => ({
useRouter: () => ({ push, back }),
}));
vi.mock('@/lib/services/transmittal.service', () => ({
transmittalService: {
create: vi.fn(),
},
}));
vi.mock('@/lib/services/correspondence.service', () => ({
correspondenceService: {
getAll: vi.fn(),
},
}));
vi.mock('@/lib/services/project.service', () => ({
projectService: {
getAll: vi.fn(),
},
}));
vi.mock('@/lib/services/organization.service', () => ({
organizationService: {
getAll: vi.fn(),
},
}));
function renderForm() {
const { wrapper } = createTestQueryClient();
return render(<TransmittalForm />, { wrapper });
}
async function chooseCombobox(label: string | RegExp, option: string): Promise<void> {
const user = userEvent.setup();
await user.click(screen.getByRole('combobox', { name: label }));
const matches = await screen.findAllByText(option);
await user.click(matches[matches.length - 1]);
}
describe('TransmittalForm', () => {
beforeEach(() => {
vi.clearAllMocks();
Element.prototype.scrollIntoView = vi.fn();
vi.mocked(projectService.getAll).mockResolvedValue({
data: [{ publicId: '019505a1-7c3e-7000-8000-abc123defc01', projectName: 'LCBP3' }],
});
vi.mocked(organizationService.getAll).mockResolvedValue({
data: [{ uuid: '019505a1-7c3e-7000-8000-abc123defc02', organizationName: 'TEAM Consulting' }],
});
vi.mocked(correspondenceService.getAll).mockResolvedValue({
data: [{ uuid: '019505a1-7c3e-7000-8000-abc123defc03', correspondenceNumber: 'COR-001' }],
});
vi.mocked(transmittalService.create).mockResolvedValue({
uuid: '019505a1-7c3e-7000-8000-abc123defc04',
correspondence: { uuid: '019505a1-7c3e-7000-8000-abc123defc05' },
});
});
it('renders main sections and supports cancel navigation', async () => {
const user = userEvent.setup();
renderForm();
expect(await screen.findByText('Transmittal Details')).toBeInTheDocument();
expect(screen.getByText('Transmittal Items')).toBeInTheDocument();
await user.click(screen.getByRole('button', { name: 'Cancel' }));
expect(back).toHaveBeenCalled();
});
it('shows validation errors when required fields are missing', async () => {
const user = userEvent.setup();
renderForm();
await user.click(await screen.findByRole('button', { name: 'Create Transmittal' }));
expect(await screen.findByText('Project is required')).toBeInTheDocument();
expect(screen.getByText('Recipient is required')).toBeInTheDocument();
expect(screen.getByText('Correspondence is required')).toBeInTheDocument();
expect(screen.getByText('Subject is required')).toBeInTheDocument();
});
it('submits cleaned transmittal payload and navigates to created record', async () => {
const user = userEvent.setup();
renderForm();
await screen.findByText('Transmittal Details');
await chooseCombobox(/project/i, 'LCBP3');
await chooseCombobox(/recipient organization/i, 'TEAM Consulting');
await user.click(screen.getByRole('combobox', { name: /reference document/i }));
await user.click(await screen.findByText('COR-001'));
await user.type(screen.getByPlaceholderText('Enter transmittal subject'), 'Weekly package');
await user.clear(screen.getByPlaceholderText('ID'));
await user.type(screen.getByPlaceholderText('ID'), '12');
await user.type(screen.getByPlaceholderText('Copies/Notes'), 'For record');
await user.type(screen.getByPlaceholderText('Additional notes...'), 'Submitted by test');
await user.click(screen.getByRole('button', { name: 'Create Transmittal' }));
await waitFor(() => {
expect(transmittalService.create).toHaveBeenCalledWith({
projectId: '019505a1-7c3e-7000-8000-abc123defc01',
recipientOrganizationId: '019505a1-7c3e-7000-8000-abc123defc02',
correspondenceId: '019505a1-7c3e-7000-8000-abc123defc03',
subject: 'Weekly package',
purpose: 'FOR_APPROVAL',
remarks: 'Submitted by test',
items: [{ itemType: 'DRAWING', itemId: 12, description: 'For record' }],
});
});
expect(toast.success).toHaveBeenCalledWith('Transmittal created successfully');
expect(push).toHaveBeenCalledWith('/transmittals/019505a1-7c3e-7000-8000-abc123defc05');
});
});