Web 端 (Playwright): - fixtures: test-data 工厂 + API Client (乐观锁 version) + 增强 auth fixture - pages: LoginPage, PatientListPage, PatientDetailPage, HealthDataPage, AppointmentPage - flows: 患者全流程, 体征数据链路, 预约排班链路, 随访管理链路, 告警处理链路 - smoke tests 迁移到 smoke/ 目录,import 路径更新 - playwright.config.ts 更新: globalSetup 环境检查, 60s timeout, video retain 小程序端 (Vitest + miniprogram-automator): - helpers: AutomatorClient, MpApiClient, MpAuthHelper, MpNavigator - flows: 患者健康数据查看, 体征数据录入, 积分签到兑换, 积分商城浏览 - vitest.config.ts + check-readiness.ts - vitest 4.1.5 依赖安装 Playwright 发现 15 个测试 (5 flow + 10 smoke),全部就绪
189 lines
6.7 KiB
TypeScript
189 lines
6.7 KiB
TypeScript
// apps/web/e2e/fixtures/api-client.ts
|
|
|
|
import type {
|
|
PatientData, DoctorData, VitalSignsData, ScheduleData,
|
|
AppointmentData, FollowUpTemplateData, FollowUpTaskData, AlertRuleData,
|
|
} from './test-data';
|
|
|
|
const API_BASE = process.env.E2E_API_URL || 'http://localhost:3000/api/v1';
|
|
|
|
interface ApiResponse<T> { success: boolean; data: T }
|
|
interface Versioned { id: string; version: number }
|
|
type VEntity<T> = T & Versioned;
|
|
|
|
interface LoginResponse {
|
|
access_token: string;
|
|
refresh_token: string;
|
|
expires_in: number;
|
|
user: { id: string; username: string; display_name: string; roles: string[] };
|
|
}
|
|
|
|
export class ApiClient {
|
|
private token = '';
|
|
|
|
async login(username?: string, password?: string): Promise<LoginResponse> {
|
|
const res = await this.rawPost<{ success: boolean; data: LoginResponse }>(
|
|
'/auth/login',
|
|
{
|
|
username: username || process.env.E2E_ADMIN_USER || 'admin',
|
|
password: password || process.env.E2E_ADMIN_PASS || 'Admin@2026',
|
|
},
|
|
);
|
|
this.token = res.data.access_token;
|
|
return res.data;
|
|
}
|
|
|
|
async loginAsAdmin(): Promise<LoginResponse> {
|
|
return this.login();
|
|
}
|
|
|
|
getToken(): string { return this.token; }
|
|
|
|
async createPatient(overrides?: Partial<PatientData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/patients', overrides ?? {});
|
|
}
|
|
|
|
async updatePatient(id: string, version: number, data: Partial<PatientData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.put(`/health/patients/${id}`, { ...data, version });
|
|
}
|
|
|
|
async deletePatient(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/patients/${id}`, { version });
|
|
}
|
|
|
|
async createDoctor(overrides?: Partial<DoctorData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/doctors', overrides ?? {});
|
|
}
|
|
|
|
async deleteDoctor(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/doctors/${id}`, { version });
|
|
}
|
|
|
|
async createVitalSigns(patientId: string, overrides?: Partial<VitalSignsData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post(`/health/patients/${patientId}/vital-signs`, overrides ?? {});
|
|
}
|
|
|
|
async deleteVitalSigns(patientId: string, id: string, version: number): Promise<void> {
|
|
await this.del(`/health/patients/${patientId}/vital-signs/${id}`, { version });
|
|
}
|
|
|
|
async createSchedule(overrides: ScheduleData): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/doctor-schedules', overrides);
|
|
}
|
|
|
|
async deleteSchedule(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/doctor-schedules/${id}`, { version });
|
|
}
|
|
|
|
async createAppointment(overrides: AppointmentData): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/appointments', overrides);
|
|
}
|
|
|
|
async updateAppointmentStatus(id: string, version: number, status: string): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.put(`/health/appointments/${id}/status`, { status, version });
|
|
}
|
|
|
|
async deleteAppointment(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/appointments/${id}`, { version });
|
|
}
|
|
|
|
async createFollowUpTemplate(overrides?: Partial<FollowUpTemplateData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/follow-up-templates', overrides ?? {});
|
|
}
|
|
|
|
async deleteFollowUpTemplate(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/follow-up-templates/${id}`, { version });
|
|
}
|
|
|
|
async createFollowUpTask(overrides: FollowUpTaskData): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/follow-up-tasks', overrides);
|
|
}
|
|
|
|
async deleteFollowUpTask(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/follow-up-tasks/${id}`, { version });
|
|
}
|
|
|
|
async createAlertRule(overrides?: Partial<AlertRuleData>): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.post('/health/alert-rules', overrides ?? {});
|
|
}
|
|
|
|
async deleteAlertRule(id: string, version: number): Promise<void> {
|
|
await this.del(`/health/alert-rules/${id}`, { version });
|
|
}
|
|
|
|
async listAlerts(): Promise<VEntity<Record<string, unknown>>[]> {
|
|
const res = await this.get<{ items: VEntity<Record<string, unknown>>[] }>('/health/alerts');
|
|
return res.items ?? [];
|
|
}
|
|
|
|
async acknowledgeAlert(id: string, version: number): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.put(`/health/alerts/${id}/acknowledge`, { version });
|
|
}
|
|
|
|
async resolveAlert(id: string, version: number): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.put(`/health/alerts/${id}/resolve`, { version });
|
|
}
|
|
|
|
async dismissAlert(id: string, version: number): Promise<VEntity<Record<string, unknown>>> {
|
|
return this.put(`/health/alerts/${id}/dismiss`, { version });
|
|
}
|
|
|
|
private async headers(): Promise<Record<string, string>> {
|
|
return {
|
|
'Content-Type': 'application/json',
|
|
...(this.token ? { Authorization: `Bearer ${this.token}` } : {}),
|
|
};
|
|
}
|
|
|
|
private async get<T>(path: string): Promise<T> {
|
|
const res = await fetch(`${API_BASE}${path}`, { headers: await this.headers() });
|
|
const json = await res.json();
|
|
if (!json.success) throw new Error(`GET ${path} failed: ${json.error ?? res.status}`);
|
|
return json.data as T;
|
|
}
|
|
|
|
private async post<T>(path: string, body: unknown): Promise<T> {
|
|
const res = await fetch(`${API_BASE}${path}`, {
|
|
method: 'POST',
|
|
headers: await this.headers(),
|
|
body: JSON.stringify(body),
|
|
});
|
|
const json = await res.json();
|
|
if (!json.success) throw new Error(`POST ${path} failed: ${json.error ?? res.status}`);
|
|
return json.data as T;
|
|
}
|
|
|
|
private async put<T>(path: string, body: unknown): Promise<T> {
|
|
const res = await fetch(`${API_BASE}${path}`, {
|
|
method: 'PUT',
|
|
headers: await this.headers(),
|
|
body: JSON.stringify(body),
|
|
});
|
|
const json = await res.json();
|
|
if (!json.success) throw new Error(`PUT ${path} failed: ${json.error ?? res.status}`);
|
|
return json.data as T;
|
|
}
|
|
|
|
private async del(path: string, body?: unknown): Promise<void> {
|
|
const res = await fetch(`${API_BASE}${path}`, {
|
|
method: 'DELETE',
|
|
headers: await this.headers(),
|
|
body: body ? JSON.stringify(body) : undefined,
|
|
});
|
|
if (res.status === 204) return;
|
|
const json = await res.json();
|
|
if (!json.success) throw new Error(`DELETE ${path} failed: ${json.error ?? res.status}`);
|
|
}
|
|
|
|
private async rawPost<T>(path: string, body: unknown): Promise<T> {
|
|
const res = await fetch(`${API_BASE}${path}`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(body),
|
|
});
|
|
const json = await res.json();
|
|
if (!json.success) throw new Error(`POST ${path} failed: ${json.error ?? res.status}`);
|
|
return json as T;
|
|
}
|
|
}
|