Files
Alfred/tests/web/hooks/use-providers.test.ts
lanyuanxiaoyao 48c76e6180 fix: 模型管理审查修复与归档
- 修复 registry 测试 ai mock 缺失 createProviderRegistry 导出
- 新增 POST /api/providers/test 支持未保存供应商配置连通性测试
- 供应商表单新增测试连接按钮,新建默认 openai-compatible
- 连通性测试按 ok 展示成功/失败,不再统一 success 样式
- 模型表单新建时也可测试供应商连接
- 模型页使用独立 provider 列表避免分页/搜索影响
- 移除模型管理组件内联 style
- 新增 ProviderTestResultResponse 共享响应类型
- 新增 bun run format:check 脚本
- 补充关键测试覆盖(删除关联、连通性、默认类型、表单测试)
- 更新 docs/user/usage.md、docs/development/*、design.md、tasks.md
- 归档 change 至 openspec/changes/archive/2026-05-29-add-model-management
2026-05-29 14:05:01 +08:00

129 lines
4.2 KiB
TypeScript

import { describe, expect, test } from "bun:test";
import {
createProvider,
deleteProvider,
disableProvider,
enableProvider,
fetchProvider,
fetchProviderList,
testProviderConfig,
testProviderConnection,
updateProvider,
} from "../../../src/web/hooks/use-providers";
import { installFetchMock, jsonResponse } from "../test-utils";
const PROVIDER = {
apiKey: "sk-test",
baseUrl: "https://api.openai.com/v1",
createdAt: "2024-01-01T00:00:00.000Z",
enabled: true,
id: "pv1",
name: "OpenAI",
type: "openai" as const,
updatedAt: "2024-01-01T00:00:00.000Z",
};
async function expectRejectsWithMessage(action: () => Promise<unknown>, message: string) {
try {
await action();
throw new Error("expected rejection");
} catch (error) {
expect(error).toBeInstanceOf(Error);
expect((error as Error).message).toBe(message);
}
}
function jsonBody(body: BodyInit | null | undefined): unknown {
return JSON.parse(typeof body === "string" ? body : "{}");
}
describe("use-providers request helpers", () => {
test("fetchProviderList builds correct query params", async () => {
const calls = installFetchMock(() => jsonResponse({ items: [PROVIDER], page: 1, pageSize: 20, total: 1 }));
const result = await fetchProviderList({ keyword: "OpenAI", page: 1, pageSize: 20 });
expect(result.items).toHaveLength(1);
expect(calls[0]?.method).toBe("GET");
expect(calls[0]?.url).toBe("/api/providers?page=1&pageSize=20&keyword=OpenAI");
});
test("CRUD and enable/disable use correct method, URL and body", async () => {
const calls = installFetchMock((call) => {
if (call.method === "DELETE") return new Response(null, { status: 204 });
return jsonResponse(
{ provider: PROVIDER },
{ status: call.method === "POST" && call.url === "/api/providers" ? 201 : 200 },
);
});
await createProvider({ apiKey: "sk-test", baseUrl: "https://api.openai.com/v1", name: "OpenAI", type: "openai" });
await updateProvider("pv1", { name: "New OpenAI" });
await enableProvider("pv1");
await disableProvider("pv1");
await deleteProvider("pv1");
await fetchProvider("pv1");
expect(calls.map((c) => c.method + " " + c.url)).toEqual([
"POST /api/providers",
"PATCH /api/providers/pv1",
"POST /api/providers/pv1/enable",
"POST /api/providers/pv1/disable",
"DELETE /api/providers/pv1",
"GET /api/providers/pv1",
]);
expect(jsonBody(calls[0]?.body)).toEqual({
apiKey: "sk-test",
baseUrl: "https://api.openai.com/v1",
name: "OpenAI",
type: "openai",
});
expect(jsonBody(calls[1]?.body)).toEqual({ name: "New OpenAI" });
});
test("testProviderConnection uses correct URL and parses response", async () => {
installFetchMock(() => jsonResponse({ providerTestResponse: { message: "ok", ok: true } }));
const result = await testProviderConnection("pv1");
expect(result).toEqual({ message: "ok", ok: true });
});
test("testProviderConfig posts form config and parses response", async () => {
const calls = installFetchMock(() => jsonResponse({ providerTestResponse: { message: "ok", ok: true } }));
const result = await testProviderConfig({
apiKey: "sk-test",
baseUrl: "https://api.openai.com/v1",
name: "OpenAI",
type: "openai-compatible",
});
expect(result).toEqual({ message: "ok", ok: true });
expect(calls[0]?.method).toBe("POST");
expect(calls[0]?.url).toBe("/api/providers/test");
expect(jsonBody(calls[0]?.body)).toEqual({
apiKey: "sk-test",
baseUrl: "https://api.openai.com/v1",
name: "OpenAI",
type: "openai-compatible",
});
});
test("error response uses backend error field", async () => {
installFetchMock(() => jsonResponse({ error: "dup" }, { status: 409 }));
await expectRejectsWithMessage(
() => createProvider({ apiKey: "sk", baseUrl: "https://x.com", name: "dup", type: "openai-compatible" }),
"dup",
);
});
test("non-JSON error falls back to HTTP status", async () => {
installFetchMock(() => new Response("broken", { status: 500 }));
await expectRejectsWithMessage(() => fetchProvider("missing"), "HTTP 500");
});
});