- enforce-catch-type: 增加 TSUnknownKeyword 判断,消除28个 TS6 假阳性 - no-empty-function: 统一为注释方案,移除测试/生产分支和 eslint-disable 引导 - logger.ts: 空函数体改为注释说明,删除无用 eslint-disable 指令 - 补充15处 catch 子句 : unknown 类型注解 - 清理7个测试文件失效 eslint-disable 指令 - chat/send.ts: 提取 getModelWithProvider DAO,消除直接 Drizzle 操作 - projects/update.ts: 修复死代码+条件逻辑 bug - providers/update.ts: 补充至少一个字段校验 - 前端: inline style → CSS className, ProviderFormModal whitespace 校验 - 开发文档: 更新 Zod 使用说明(AI SDK 框架级约束)
103 lines
2.8 KiB
TypeScript
103 lines
2.8 KiB
TypeScript
import { MutationCache, QueryCache, QueryClient, QueryClientProvider, useQuery } from "@tanstack/react-query";
|
|
import { render } from "@testing-library/react";
|
|
import { describe, expect, test } from "bun:test";
|
|
import { createElement, useRef } from "react";
|
|
|
|
import { useCreateProject } from "../../../src/web/hooks/use-projects";
|
|
import { installFetchMock, jsonResponse } from "../test-utils";
|
|
|
|
describe("QueryClient MutationCache onError", () => {
|
|
test("mutation 错误触发 MutationCache onError 回调", async () => {
|
|
installFetchMock(() => jsonResponse({ error: "项目名称已存在" }, { status: 409 }));
|
|
|
|
const errors: string[] = [];
|
|
const queryClient = new QueryClient({
|
|
defaultOptions: { queries: { retry: false } },
|
|
mutationCache: new MutationCache({
|
|
onError: (error: Error) => {
|
|
errors.push(error.message);
|
|
},
|
|
}),
|
|
});
|
|
|
|
function TestComponent({ onResult }: { onResult: (mutate: () => void) => void }) {
|
|
const { mutate } = useCreateProject();
|
|
const called = useRef(false);
|
|
|
|
if (!called.current) {
|
|
called.current = true;
|
|
onResult(() => {
|
|
mutate(
|
|
{ name: "test" },
|
|
{
|
|
onError: () => {},
|
|
},
|
|
);
|
|
});
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
render(
|
|
createElement(
|
|
QueryClientProvider,
|
|
{ client: queryClient },
|
|
createElement(TestComponent, { onResult: (fn) => fn() }),
|
|
),
|
|
);
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 200));
|
|
|
|
expect(errors.length).toBe(1);
|
|
expect(errors[0]).toBe("项目名称已存在");
|
|
});
|
|
});
|
|
|
|
describe("QueryClient QueryCache onError", () => {
|
|
test("query 错误触发 QueryCache onError 回调", async () => {
|
|
installFetchMock(() => new Response("broken", { status: 500 }));
|
|
|
|
const errors: string[] = [];
|
|
const queryClient = new QueryClient({
|
|
defaultOptions: { queries: { retry: false } },
|
|
queryCache: new QueryCache({
|
|
onError: (error: Error) => {
|
|
errors.push(error.message);
|
|
},
|
|
}),
|
|
});
|
|
|
|
function TestComponent({ onResult }: { onResult: (trigger: () => void) => void }) {
|
|
const called = useRef(false);
|
|
|
|
useQuery({
|
|
queryFn: () => Promise.reject(new Error("test query error")),
|
|
queryKey: ["test-query-error"],
|
|
});
|
|
|
|
if (!called.current) {
|
|
called.current = true;
|
|
onResult(() => {
|
|
// no-op trigger
|
|
});
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
render(
|
|
createElement(
|
|
QueryClientProvider,
|
|
{ client: queryClient },
|
|
createElement(TestComponent, { onResult: (fn) => fn() }),
|
|
),
|
|
);
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 200));
|
|
|
|
expect(errors.length).toBe(1);
|
|
expect(errors[0]).toBe("test query error");
|
|
});
|
|
});
|