- 修复 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
228 lines
7.7 KiB
TypeScript
228 lines
7.7 KiB
TypeScript
import { fireEvent, screen, waitFor } from "@testing-library/react";
|
|
import { describe, expect, mock, test } from "bun:test";
|
|
import { createElement } from "react";
|
|
|
|
import type { Model, Provider } from "../../../src/shared/api";
|
|
|
|
import { ModelFormModal } from "../../../src/web/pages/models/components/ModelFormModal";
|
|
import { ProviderFormModal } from "../../../src/web/pages/models/components/ProviderFormModal";
|
|
import { renderWithProviders } from "../test-utils";
|
|
|
|
const ENABLED_PROVIDER: 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",
|
|
updatedAt: "2024-01-01T00:00:00.000Z",
|
|
};
|
|
|
|
const DISABLED_PROVIDER: Provider = {
|
|
apiKey: "sk-off",
|
|
baseUrl: "https://api.deepseek.com/v1",
|
|
createdAt: "2024-01-01T00:00:00.000Z",
|
|
enabled: false,
|
|
id: "pv2",
|
|
name: "DeepSeek",
|
|
type: "openai-compatible",
|
|
updatedAt: "2024-01-01T00:00:00.000Z",
|
|
};
|
|
|
|
const ENABLED_MODEL: Model = {
|
|
capabilities: ["text", "reasoning"],
|
|
contextLength: 128000,
|
|
createdAt: "2024-01-01T00:00:00.000Z",
|
|
enabled: true,
|
|
id: "m1",
|
|
maxOutputTokens: 4096,
|
|
modelId: "gpt-4o",
|
|
name: "GPT-4o",
|
|
providerId: "pv1",
|
|
updatedAt: "2024-01-01T00:00:00.000Z",
|
|
};
|
|
|
|
function clickLatestConfirmButton() {
|
|
const buttons = screen.getAllByRole("button", { name: /OK|确\s*定/ });
|
|
fireEvent.click(buttons[buttons.length - 1]!);
|
|
}
|
|
|
|
describe("ProviderFormModal", () => {
|
|
test("编辑供应商表单只提交变更字段", async () => {
|
|
const updateCalls: unknown[] = [];
|
|
|
|
renderWithProviders(
|
|
createElement(ProviderFormModal, {
|
|
editingProvider: ENABLED_PROVIDER,
|
|
onCancel: () => undefined,
|
|
onCreate: () => Promise.resolve(),
|
|
onOpenChange: () => undefined,
|
|
onTest: () => Promise.resolve({ message: "连接成功", ok: true }),
|
|
onUpdate: (args: unknown) => {
|
|
updateCalls.push(args);
|
|
return Promise.resolve();
|
|
},
|
|
open: true,
|
|
submitting: false,
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByPlaceholderText("请输入供应商名称")).not.toBeNull());
|
|
fireEvent.change(screen.getByPlaceholderText("请输入供应商名称"), { target: { value: "New OpenAI" } });
|
|
clickLatestConfirmButton();
|
|
|
|
await waitFor(() => expect(updateCalls.length).toBe(1));
|
|
expect(updateCalls[0]).toEqual({ data: { name: "New OpenAI" }, id: "pv1" });
|
|
});
|
|
|
|
test("新建供应商默认使用 openai-compatible 类型", async () => {
|
|
const createCalls: unknown[] = [];
|
|
|
|
renderWithProviders(
|
|
createElement(ProviderFormModal, {
|
|
editingProvider: null,
|
|
onCancel: () => undefined,
|
|
onCreate: (data: unknown) => {
|
|
createCalls.push(data);
|
|
return Promise.resolve();
|
|
},
|
|
onOpenChange: () => undefined,
|
|
onTest: () => Promise.resolve({ message: "连接成功", ok: true }),
|
|
onUpdate: () => Promise.resolve(),
|
|
open: true,
|
|
submitting: false,
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByPlaceholderText("请输入供应商名称")).not.toBeNull());
|
|
fireEvent.change(screen.getByPlaceholderText("请输入供应商名称"), { target: { value: "兼容供应商" } });
|
|
fireEvent.change(screen.getByPlaceholderText("https://api.openai.com/v1"), {
|
|
target: { value: "https://api.test.com/v1" },
|
|
});
|
|
fireEvent.change(screen.getByPlaceholderText("请输入 API Key"), { target: { value: "sk-test" } });
|
|
clickLatestConfirmButton();
|
|
|
|
await waitFor(() => expect(createCalls.length).toBe(1));
|
|
expect(createCalls[0]).toEqual({
|
|
apiKey: "sk-test",
|
|
baseUrl: "https://api.test.com/v1",
|
|
name: "兼容供应商",
|
|
type: "openai-compatible",
|
|
});
|
|
});
|
|
|
|
test("供应商表单可使用当前表单配置测试连接", async () => {
|
|
const testCalls: unknown[] = [];
|
|
|
|
renderWithProviders(
|
|
createElement(ProviderFormModal, {
|
|
editingProvider: null,
|
|
onCancel: () => undefined,
|
|
onCreate: () => Promise.resolve(),
|
|
onOpenChange: () => undefined,
|
|
onTest: (data: unknown) => {
|
|
testCalls.push(data);
|
|
return Promise.resolve({ message: "连接成功", ok: true });
|
|
},
|
|
onUpdate: () => Promise.resolve(),
|
|
open: true,
|
|
submitting: false,
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByPlaceholderText("请输入供应商名称")).not.toBeNull());
|
|
fireEvent.change(screen.getByPlaceholderText("请输入供应商名称"), { target: { value: "兼容供应商" } });
|
|
fireEvent.change(screen.getByPlaceholderText("https://api.openai.com/v1"), {
|
|
target: { value: "https://api.test.com/v1" },
|
|
});
|
|
fireEvent.change(screen.getByPlaceholderText("请输入 API Key"), { target: { value: "sk-test" } });
|
|
fireEvent.click(screen.getByRole("button", { name: "测试连接" }));
|
|
|
|
await waitFor(() => expect(testCalls.length).toBe(1));
|
|
expect(testCalls[0]).toEqual({
|
|
apiKey: "sk-test",
|
|
baseUrl: "https://api.test.com/v1",
|
|
name: "兼容供应商",
|
|
type: "openai-compatible",
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("ModelFormModal", () => {
|
|
test("编辑模型表单只提交变更字段", async () => {
|
|
const updateCalls: unknown[] = [];
|
|
|
|
renderWithProviders(
|
|
createElement(ModelFormModal, {
|
|
editingModel: ENABLED_MODEL,
|
|
onCancel: () => undefined,
|
|
onCreate: () => Promise.resolve(),
|
|
onOpenChange: () => undefined,
|
|
onUpdate: (args: unknown) => {
|
|
updateCalls.push(args);
|
|
return Promise.resolve();
|
|
},
|
|
open: true,
|
|
providers: [ENABLED_PROVIDER, DISABLED_PROVIDER],
|
|
submitting: false,
|
|
testConnection: () => Promise.resolve({ message: "连接成功", ok: true }),
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByPlaceholderText("请输入模型名称")).not.toBeNull());
|
|
fireEvent.change(screen.getByPlaceholderText("请输入模型名称"), { target: { value: "GPT-4o Mini" } });
|
|
clickLatestConfirmButton();
|
|
|
|
await waitFor(() => expect(updateCalls.length).toBe(1));
|
|
expect(updateCalls[0]).toEqual({ data: { name: "GPT-4o Mini" }, id: "m1" });
|
|
});
|
|
|
|
test("模型表单校验失败不会提交", async () => {
|
|
const onCreate = mock(() => Promise.resolve());
|
|
|
|
renderWithProviders(
|
|
createElement(ModelFormModal, {
|
|
editingModel: null,
|
|
onCancel: () => undefined,
|
|
onCreate,
|
|
onOpenChange: () => undefined,
|
|
onUpdate: () => Promise.resolve(),
|
|
open: true,
|
|
providers: [ENABLED_PROVIDER],
|
|
submitting: false,
|
|
testConnection: () => Promise.resolve({ message: "连接成功", ok: true }),
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByPlaceholderText("请输入模型名称")).not.toBeNull());
|
|
clickLatestConfirmButton();
|
|
expect(onCreate).not.toHaveBeenCalled();
|
|
});
|
|
|
|
test("新建模型时可测试所选供应商连接", async () => {
|
|
const testConnection = mock(() => Promise.resolve({ message: "连接成功", ok: true }));
|
|
|
|
renderWithProviders(
|
|
createElement(ModelFormModal, {
|
|
editingModel: null,
|
|
onCancel: () => undefined,
|
|
onCreate: () => Promise.resolve(),
|
|
onOpenChange: () => undefined,
|
|
onUpdate: () => Promise.resolve(),
|
|
open: true,
|
|
providers: [ENABLED_PROVIDER],
|
|
submitting: false,
|
|
testConnection,
|
|
}),
|
|
);
|
|
|
|
await waitFor(() => expect(screen.getByText("测试连接")).not.toBeNull());
|
|
fireEvent.mouseDown(screen.getByRole("combobox"));
|
|
fireEvent.click(await screen.findByText("OpenAI"));
|
|
fireEvent.click(screen.getByRole("button", { name: "测试连接" }));
|
|
|
|
await waitFor(() => expect(testConnection).toHaveBeenCalledWith("pv1"));
|
|
});
|
|
});
|