Some checks failed
CI / Lint → Typecheck → Test → Build (22) (push) Failing after 29s
CI / E2E Tests (push) Has been skipped
CodeQL Analysis / CodeQL (javascript-typescript) (push) Failing after 2m42s
Deploy / Build Web Image (push) Failing after 27s
Deploy / Build AI Services Image (push) Failing after 29s
E2E Tests / Playwright E2E (push) Failing after 43s
Deploy / Build API Image (push) Failing after 1m31s
Security Scanning / Dependency Audit (pnpm) (push) Failing after 6s
Security Scanning / Trivy Scan — API Image (push) Failing after 5m35s
Security Scanning / Trivy Scan — AI Services Image (push) Failing after 3m45s
Deploy / Deploy to Staging (push) Has been skipped
Deploy / Smoke Test Staging (push) Has been skipped
Deploy / Deploy to Production (push) Has been skipped
Deploy / Smoke Test Production (push) Has been skipped
Deploy / Rollback Staging (push) Has been skipped
Deploy / Rollback Production (push) Has been skipped
Security Scanning / Trivy Scan — Web Image (push) Failing after 13m51s
Security Scanning / Trivy Filesystem Scan (push) Failing after 14m46s
Security Scanning / Security Gate (push) Has been cancelled
307 lines
11 KiB
Markdown
307 lines
11 KiB
Markdown
# Tài Liệu Độ Phủ Kiểm Thử
|
||
|
||
## Tổng Quan
|
||
|
||
Thư mục này chứa tài liệu toàn diện hướng dẫn viết kiểm thử cho **17 tệp nguồn chưa được kiểm thử** thuộc các module inquiries, leads và reviews của GoodGo Platform API.
|
||
|
||
## Các Tệp Bao Gồm
|
||
|
||
### 1. **TEST_COVERAGE_ANALYSIS.md** (1.841 dòng, 55KB)
|
||
- **Triển khai đầy đủ** của tất cả 17 tệp chưa được kiểm thử
|
||
- Mã nguồn hoàn chỉnh cho từng tệp
|
||
- Giải thích chi tiết chức năng của từng phương thức
|
||
- Các kịch bản kiểm thử cho từng thành phần
|
||
- Tham chiếu các mẫu kiểm thử từ các tệp kiểm thử hiện có
|
||
- Mẫu và ví dụ kiểm thử toàn diện
|
||
|
||
### 2. **TEST_COVERAGE_QUICK_REFERENCE.md** (301 dòng, 9,1KB)
|
||
- Hướng dẫn tra cứu nhanh cho tất cả 17 tệp
|
||
- Danh sách kiểm tra theo loại tệp (repositories, VOs, DTOs, controllers)
|
||
- Các kịch bản kiểm thử theo module
|
||
- Ma trận ưu tiên (critical, high, medium)
|
||
- Thứ tự thực thi kiểm thử được khuyến nghị
|
||
- Mẫu thiết lập mock
|
||
- Các công thức quan trọng cần xác minh
|
||
|
||
### 3. **TEST_TEMPLATES.md** (500+ dòng)
|
||
- Mẫu kiểm thử sẵn dùng cho:
|
||
- Kiểm thử Repository
|
||
- Kiểm thử Value Object
|
||
- Kiểm thử DTO
|
||
- Kiểm thử Controller
|
||
- Kiểm thử helper cho tính toán phân trang
|
||
- Kiểm thử helper cho công thức tổng hợp
|
||
- Kiểm thử helper xác thực DTO
|
||
|
||
## Bắt Đầu Nhanh
|
||
|
||
### Để có cái nhìn tổng quan nhanh:
|
||
1. Đọc **Quick Reference** trước (5–10 phút)
|
||
2. Xác định loại tệp bạn muốn kiểm thử
|
||
3. Chuyển đến mục tương ứng trong **TEST_TEMPLATES.md**
|
||
4. Sao chép mẫu và điều chỉnh cho tệp của bạn
|
||
|
||
### Để hiểu sâu hơn:
|
||
1. Bắt đầu với tổng quan **Quick Reference**
|
||
2. Đọc phần module liên quan trong **TEST_COVERAGE_ANALYSIS.md**
|
||
3. Sử dụng **TEST_TEMPLATES.md** làm hướng dẫn triển khai
|
||
4. Tham chiếu các mẫu kiểm thử ví dụ trong tài liệu Analysis
|
||
|
||
## Tổ Chức Tệp Theo Module
|
||
|
||
### Module Inquiries (4 tệp)
|
||
```
|
||
✗ prisma-inquiry.repository.ts — 6 phương thức, phân trang, quan hệ
|
||
✗ inquiries.controller.ts — 4 endpoint, guards
|
||
✗ create-inquiry.dto.ts — 3 validation
|
||
✗ list-inquiries.dto.ts — 2 validation (phân trang)
|
||
```
|
||
|
||
### Module Leads (6 tệp)
|
||
```
|
||
✗ prisma-lead.repository.ts — 6 phương thức, tổng hợp thống kê
|
||
✗ lead-score.vo.ts — Xác thực khoảng giá trị (0-100)
|
||
✗ leads.controller.ts — 5 endpoint, role guards
|
||
✗ create-lead.dto.ts — 6 validation
|
||
✗ list-leads.dto.ts — Enum trạng thái, phân trang
|
||
✗ update-lead-status.dto.ts — Enum trạng thái
|
||
```
|
||
|
||
### Module Reviews (5 tệp)
|
||
```
|
||
✗ prisma-review.repository.ts — 7 phương thức, thống kê phân phối
|
||
✗ rating.vo.ts — Xác thực khoảng số nguyên (1-5)
|
||
✗ reviews.controller.ts — 5 endpoint, xác thực hỗn hợp
|
||
✗ create-review.dto.ts — 4 validation
|
||
✗ list-reviews.dto.ts — 2 lớp DTO, phân trang
|
||
```
|
||
|
||
## Mức Độ Ưu Tiên Kiểm Thử
|
||
|
||
### 🔴 CRITICAL (Logic Nghiệp Vụ)
|
||
1. **PrismaReviewRepository.getStats()** — Tính toán phân phối phức tạp
|
||
2. **PrismaLeadRepository.getStatsByAgent()** — Công thức tỷ lệ chuyển đổi
|
||
3. **Rating.vo** — Phải xác thực chỉ số nguyên từ 1–5
|
||
4. **LeadScore.vo** — Phải xác thực khoảng 0–100
|
||
|
||
**Bắt đầu từ đây** — Đây là các validation quan trọng nhất cho nghiệp vụ
|
||
|
||
### 🟡 HIGH (Tính Toàn Vẹn Dữ Liệu)
|
||
1. Các thao tác CRUD của Repository
|
||
2. Tính toán phân trang
|
||
3. Ánh xạ quan hệ (user joins, duyệt listing)
|
||
4. Điều phối command/query từ Controller
|
||
|
||
**Kiểm thử tiếp theo** — Đảm bảo tính nhất quán của dữ liệu
|
||
|
||
### 🟢 MEDIUM (Validation & Guards)
|
||
1. Xác thực trường DTO
|
||
2. Ràng buộc Enum
|
||
3. Xử lý trường tùy chọn
|
||
4. Guards xác thực/phân quyền
|
||
|
||
**Kiểm thử sau cùng** — Đây là các mối quan tâm ở tầng framework
|
||
|
||
## Lộ Trình Thực Thi Kiểm Thử
|
||
|
||
### Tuần 1: Value Objects & DTOs (8–10 tệp kiểm thử)
|
||
- [ ] LeadScore.vo.ts (1 tệp, 4–5 test case)
|
||
- [ ] Rating.vo.ts (1 tệp, 4–5 test case)
|
||
- [ ] Tất cả 10 tệp DTO (dùng mẫu, biến thể tối thiểu)
|
||
|
||
**Thời gian ước tính:** 3–5 giờ tổng cộng
|
||
|
||
### Tuần 2: Controllers (2 tệp kiểm thử)
|
||
- [ ] InquiriesController (4 endpoint + guards)
|
||
- [ ] LeadsController (5 endpoint + guards)
|
||
- [ ] ReviewsController (5 endpoint + xác thực hỗn hợp)
|
||
|
||
**Thời gian ước tính:** 2–3 giờ tổng cộng
|
||
|
||
### Tuần 3: Repositories (3 tệp kiểm thử)
|
||
- [ ] PrismaInquiryRepository (6 phương thức)
|
||
- [ ] PrismaLeadRepository (6 phương thức + tổng hợp)
|
||
- [ ] PrismaReviewRepository (7 phương thức + phân phối)
|
||
|
||
**Thời gian ước tính:** 4–6 giờ tổng cộng
|
||
|
||
### Tổng Thời Gian Ước Tính: 9–14 giờ
|
||
|
||
## Các Mẫu Kiểm Thử Quan Trọng
|
||
|
||
### Repositories
|
||
```typescript
|
||
// Mock tất cả phương thức Prisma trong beforeEach
|
||
mockPrisma = {
|
||
[model]: {
|
||
findUnique: vi.fn(),
|
||
findMany: vi.fn(),
|
||
create: vi.fn(),
|
||
update: vi.fn(),
|
||
count: vi.fn()
|
||
}
|
||
};
|
||
|
||
// Kiểm thử luồng thành công và trường hợp lỗi
|
||
// Xác minh tính toán phân trang
|
||
// Xác minh chuyển đổi dữ liệu (toDomain)
|
||
```
|
||
|
||
### Value Objects
|
||
```typescript
|
||
// Kiểm thử các trường hợp hợp lệ
|
||
const result = ValueObject.create(validValue);
|
||
expect(result.isOk()).toBe(true);
|
||
|
||
// Kiểm thử các trường hợp không hợp lệ
|
||
const result = ValueObject.create(invalidValue);
|
||
expect(result.isErr()).toBe(true);
|
||
expect(result.unwrapErr()).toBe('error message');
|
||
```
|
||
|
||
### DTOs
|
||
```typescript
|
||
// Sử dụng validate() từ class-validator
|
||
const errors = await validate(dto);
|
||
expect(errors).toHaveLength(0); // hoặc > 0 cho trường hợp không hợp lệ
|
||
|
||
// Kiểm thử chuyển đổi kiểu với class-transformer
|
||
// Kiểm thử xử lý trường tùy chọn
|
||
```
|
||
|
||
### Controllers
|
||
```typescript
|
||
// Mock CommandBus và QueryBus
|
||
mockCommandBus.execute.mockResolvedValue(expectedResult);
|
||
|
||
// Xác minh cấu trúc command/query
|
||
const command = mockCommandBus.execute.mock.calls[0][0];
|
||
expect(command.userId).toBe(expectedUserId);
|
||
```
|
||
|
||
## Các Công Thức Quan Trọng Cần Kiểm Thử
|
||
|
||
### Phân Trang
|
||
```
|
||
skip = (page - 1) * take
|
||
totalPages = Math.ceil(total / take)
|
||
take = Math.min(limit, 100) // Giới hạn tối đa 100
|
||
```
|
||
|
||
### Thống Kê Lead
|
||
```
|
||
conversionRate = (CONVERTED_count / total_leads) * 100 // 2 chữ số thập phân
|
||
avgScore = (sum_of_scores / non_null_count) // 1 chữ số thập phân, null nếu không có điểm
|
||
```
|
||
|
||
### Thống Kê Review
|
||
```
|
||
averageRating = (sum_of_ratings / total_reviews) // 1 chữ số thập phân
|
||
distribution = { 1: count, 2: count, 3: count, 4: count, 5: count }
|
||
```
|
||
|
||
## Các Câu Lệnh Import Cần Dùng
|
||
|
||
```typescript
|
||
// Tiện ích kiểm thử
|
||
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
||
import { validate } from 'class-validator';
|
||
import { plainToClass } from 'class-transformer';
|
||
|
||
// Tiện ích NestJS (dùng cho kiểm thử controller)
|
||
import type { EventBus, CommandBus, QueryBus } from '@nestjs/cqrs';
|
||
|
||
// Các lớp domain
|
||
import { InquiryEntity } from './domain/entities/inquiry.entity';
|
||
import { LeadEntity } from './domain/entities/lead.entity';
|
||
import { ReviewEntity } from './domain/entities/review.entity';
|
||
import { LeadScore } from './domain/value-objects/lead-score.vo';
|
||
import { Rating } from './domain/value-objects/rating.vo';
|
||
```
|
||
|
||
## Dữ Liệu Kiểm Thử Thông Dụng
|
||
|
||
### ID Người Dùng
|
||
```
|
||
user-1, user-2, user-3, user-agent-1
|
||
```
|
||
|
||
### Trạng Thái Lead
|
||
```
|
||
NEW, CONTACTED, QUALIFIED, NEGOTIATING, CONVERTED, LOST
|
||
```
|
||
|
||
### Giá Trị Mặc Định Phân Trang
|
||
```
|
||
page: 1, limit: 20 (tối đa 100)
|
||
```
|
||
|
||
### Khoảng Đánh Giá
|
||
```
|
||
1-5 sao (chỉ số nguyên)
|
||
```
|
||
|
||
### Khoảng Điểm Lead
|
||
```
|
||
0-100 (bất kỳ số nào)
|
||
```
|
||
|
||
## Tài Liệu Tham Khảo
|
||
|
||
### Các Tệp Kiểm Thử Hiện Có (Dùng Làm Tham Chiếu)
|
||
- `src/modules/inquiries/application/__tests__/create-inquiry.handler.spec.ts` (mẫu 20 dòng)
|
||
- `src/modules/leads/application/__tests__/create-lead.handler.spec.ts` (mẫu 20 dòng)
|
||
- `src/modules/reviews/presentation/__tests__/reviews.controller.spec.ts` (mẫu 135 dòng)
|
||
|
||
### Triển Khai Đầy Đủ
|
||
Tất cả 17 triển khai tệp có trong `TEST_COVERAGE_ANALYSIS.md` với giải thích từng dòng.
|
||
|
||
## Xử Lý Sự Cố
|
||
|
||
### Vấn đề: "Cannot find module" khi mock
|
||
**Giải pháp:** Sử dụng ép kiểu `as any` cho các dependency được mock
|
||
```typescript
|
||
const repo = new PrismaInquiryRepository(mockPrisma as any);
|
||
```
|
||
|
||
### Vấn đề: Kiểm thử validation không hoạt động
|
||
**Giải pháp:** Đảm bảo dùng `plainToClass` cho các DTO có transformer
|
||
```typescript
|
||
const dto = plainToClass(CreateLeadDto, plainObject);
|
||
```
|
||
|
||
### Vấn đề: Tính toán phân trang không khớp
|
||
**Giải pháp:** Nhớ rằng limit được giới hạn ở 100
|
||
```typescript
|
||
const take = Math.min(limit, 100); // Luôn thực hiện bước này
|
||
```
|
||
|
||
### Vấn đề: Tính toán thống kê có lỗi làm tròn
|
||
**Giải pháp:** Kiểm tra yêu cầu số chữ số thập phân:
|
||
- conversionRate: 2 chữ số thập phân (nhân với 10000, chia cho 100)
|
||
- avgScore: 1 chữ số thập phân (nhân với 10, chia cho 10)
|
||
- averageRating: 1 chữ số thập phân (nhân với 10, chia cho 10)
|
||
|
||
## Các Bước Tiếp Theo
|
||
|
||
1. **Chọn tệp bắt đầu** từ Tuần 1 (khuyến nghị LeadScore.vo trước — đơn giản nhất)
|
||
2. **Mở mẫu** tương ứng với loại tệp đó trong TEST_TEMPLATES.md
|
||
3. **Sao chép mẫu** vào vị trí tệp kiểm thử của bạn
|
||
4. **Điều chỉnh mẫu** với các import và dữ liệu cụ thể của bạn
|
||
5. **Chạy kiểm thử** và xác minh chúng đạt
|
||
6. **Chuyển sang tệp tiếp theo** theo lộ trình
|
||
|
||
## Tài Nguyên Hỗ Trợ
|
||
|
||
- Triển khai đầy đủ: Xem **TEST_COVERAGE_ANALYSIS.md**
|
||
- Tra cứu nhanh: Xem **TEST_COVERAGE_QUICK_REFERENCE.md**
|
||
- Mẫu mã: Xem **TEST_TEMPLATES.md**
|
||
- Kiểm thử tham chiếu: Xem các tệp spec hiện có trong các module
|
||
|
||
---
|
||
|
||
**Cập Nhật Lần Cuối:** 2026-04-11
|
||
**Tổng Số Tệp Được Ghi Lại:** 17 tệp nguồn chưa được kiểm thử
|
||
**Tổng Tài Liệu:** 2.142 dòng phân tích và mẫu
|
||
**Thời Gian Kiểm Thử Ước Tính:** 9–14 giờ tổng cộng
|