使用方式
API 函数定义在 src/service/api/,按后端模块各一个文件,命名 fetchXxx。
一个完整流程
1. 后端先有接口
python
# app/business/hr/api/manage.py
@router.post("/employees/search", summary="搜索员工")
async def _(obj_in: EmployeeSearch): ...2. TS 类型定义
typescript
// web/src/typings/api/hr-manage.d.ts
declare namespace Api {
namespace HR {
interface EmployeeSearch extends Common.PaginatingCommonParams {
name?: string;
status?: 'pending' | 'onboarding' | 'active' | 'resigned';
}
interface Employee {
id: string; // sqid
employeeNo: string;
status: string;
departmentId: string;
departmentName: string;
tagIds: string[];
tagNames: string[];
createdAt: string;
// ...
}
type EmployeeListResp = Common.PaginatingQueryRecord<Employee>;
}
}3. API 函数
typescript
// web/src/service/api/hr-manage.ts
export function fetchEmployeeList(params: Api.HR.EmployeeSearch) {
return request.Post<Api.HR.EmployeeListResp>('/business/hr/employees/search', params);
}
export function fetchCreateEmployee(body: Api.HR.EmployeeCreate) {
return request.Post<Api.Common.CreatedId>('/business/hr/employees', body);
}
export function fetchUpdateEmployee(id: string, body: Api.HR.EmployeeUpdate) {
return request.Patch<Api.Common.UpdatedId>(`/business/hr/employees/${id}`, body);
}
export function fetchDeleteEmployee(id: string) {
return request.Delete<Api.Common.DeletedId>(`/business/hr/employees/${id}`);
}
export function fetchBatchDeleteEmployees(ids: string[]) {
return request.Delete<Api.Common.BatchDeletedIds>('/business/hr/employees', { ids });
}4. 在组件中调用
直接 await:
typescript
const { data, error } = await fetchEmployeeList({ current: 1, size: 10 });
if (error) {
// 网络错误(业务码失败已被 onBackendFail 统一处理)
return;
}
console.log(data.records);或者配合 useTable:
typescript
const { data, loading, columns, pagination, getData } = useNaivePaginatedTable({
apiFn: fetchEmployeeList,
apiParams: { current: 1, size: 10, name: '', status: '' },
columns: () => [...],
});命名规范
| 操作 | 函数名 | HTTP |
|---|---|---|
| 列表 / 搜索 | fetchXxxList(params) | POST /xxx/search |
| 单条 | fetchXxx(id) | GET /xxx/{id} |
| 创建 | fetchCreateXxx(body) | POST /xxx |
| 更新 | fetchUpdateXxx(id, body) | PATCH /xxx/{id} |
| 删除 | fetchDeleteXxx(id) | DELETE /xxx/{id} |
| 批量删除 | fetchBatchDeleteXxx(ids) | DELETE /xxx |
| 派生查询 | fetchGetXxxTree() / fetchGetXxxOptions() | GET /xxx/tree |
| 实例动作 | fetchXxxAction(id, body) | POST /xxx/{id}/action-name |
CLI 一键生成已经按这套约定产出;手写时也建议沿用。
资源 ID 是 sqid 字符串
后端所有资源 ID 是 sqid 字符串(如 Yc7vN3kE),不是数字。前端不要解码,原样传递即可:
typescript
const id: string = '...sqid...';
fetchUpdateEmployee(id, { name: 'X' });Api.Common.CreatedId / UpdatedId / DeletedId 等返回类型也是 string。
后端响应格式
后端统一返回 { code, msg, data }。请求工厂的 transformBackendResponse 已经把外层剥掉,业务里拿到的就是 data。
typescript
// 实际网络响应
{ "code": "0000", "msg": "OK", "data": { "records": [...], "total": 100 } }
// 业务里拿到的(typeof data)
{ records: [...], total: 100, current: 1, size: 10 }取消请求
typescript
import { abortRequestByMethodName } from '@/service/request';
// 在组件 onBeforeUnmount 等清理时机里
abortRequestByMethodName('fetchEmployeeList');