ESLint 升级到 recommended-type-checked + stylistic-type-checked, 引入 perfectionist 导入排序和 import 插件导入验证。 Prettier 显式声明全部格式化参数,消除跨环境差异。 TypeScript 启用 noUnusedLocals 和 noPropertyAccessFromIndexSignature。 完善 ignore 列表,排除 .agents/、bun.lock、data/ 等。 引入 husky + lint-staged(pre-commit)+ commitlint(commit-msg)。 更新 DEVELOPMENT.md 代码质量章节。 修复所有新增规则检测到的类型和风格违规。
148 lines
5.6 KiB
TypeScript
148 lines
5.6 KiB
TypeScript
import { describe, expect, test } from "bun:test";
|
|
|
|
import { checkHttpExpect } from "../../../../../src/server/checker/runner/http/expect";
|
|
|
|
function obs(
|
|
overrides: { body?: null | string; durationMs?: number; headers?: Record<string, string>; statusCode?: number } = {},
|
|
) {
|
|
return {
|
|
body: overrides.body ?? "",
|
|
durationMs: overrides.durationMs ?? 100,
|
|
headers: overrides.headers ?? {},
|
|
statusCode: overrides.statusCode ?? 200,
|
|
};
|
|
}
|
|
|
|
describe("checkHttpExpect", () => {
|
|
test("无 expect 配置时默认检查 status [200] 匹配成功", () => {
|
|
const r = checkHttpExpect(obs().statusCode, obs().headers, obs().body, obs().durationMs);
|
|
expect(r.matched).toBe(true);
|
|
expect(r.failure).toBeNull();
|
|
});
|
|
|
|
test("无 expect 配置时 status 非 200 匹配失败", () => {
|
|
const r = checkHttpExpect(500, {}, "", 100);
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure).not.toBeNull();
|
|
expect(r.failure!.phase).toBe("status");
|
|
expect(r.failure!.kind).toBe("mismatch");
|
|
});
|
|
|
|
test("status 匹配指定状态码", () => {
|
|
const cfg = { status: [200, 301] };
|
|
expect(checkHttpExpect(200, {}, "", 100, cfg).matched).toBe(true);
|
|
expect(checkHttpExpect(301, {}, "", 100, cfg).matched).toBe(true);
|
|
expect(checkHttpExpect(404, {}, "", 100, cfg).matched).toBe(false);
|
|
});
|
|
|
|
test("status 不匹配返回 phase=status 的失败", () => {
|
|
const r = checkHttpExpect(503, {}, "", 100, { status: [200] });
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("status");
|
|
expect(r.failure!.expected).toEqual([200]);
|
|
expect(r.failure!.actual).toBe(503);
|
|
});
|
|
|
|
test("duration 在限制内匹配成功", () => {
|
|
const r = checkHttpExpect(200, {}, "", 50, { maxDurationMs: 100 });
|
|
expect(r.matched).toBe(true);
|
|
});
|
|
|
|
test("duration 超过限制匹配失败", () => {
|
|
const r = checkHttpExpect(200, {}, "", 200, { maxDurationMs: 100 });
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("duration");
|
|
});
|
|
|
|
test("duration 恰好等于限制匹配成功", () => {
|
|
const r = checkHttpExpect(200, {}, "", 100, { maxDurationMs: 100 });
|
|
expect(r.matched).toBe(true);
|
|
});
|
|
|
|
test("headers 字符串格式检查(等于)", () => {
|
|
const h = { "content-type": "application/json", "x-api": "v1" };
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "content-type": "application/json" } }).matched).toBe(true);
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "content-type": "text/html" } }).matched).toBe(false);
|
|
});
|
|
|
|
test("headers 操作符格式检查", () => {
|
|
const h = { "content-type": "application/json" };
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "content-type": { contains: "json" } } }).matched).toBe(true);
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "content-type": { match: "^application/" } } }).matched).toBe(
|
|
true,
|
|
);
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "content-type": { contains: "xml" } } }).matched).toBe(false);
|
|
});
|
|
|
|
test("headers 大小写不敏感匹配", () => {
|
|
const h = { "content-type": "application/json" };
|
|
expect(checkHttpExpect(200, h, "", 100, { headers: { "Content-Type": "application/json" } }).matched).toBe(true);
|
|
});
|
|
|
|
test("headers 不存在时返回失败", () => {
|
|
const r = checkHttpExpect(200, {}, "", 100, { headers: { "x-missing": "value" } });
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("headers");
|
|
});
|
|
|
|
test("body 规则数组按顺序检查", () => {
|
|
const body = JSON.stringify({ count: 5, status: "ok" });
|
|
const r = checkHttpExpect(200, {}, body, 100, {
|
|
body: [{ contains: "ok" }, { json: { gte: 1, path: "$.count" } }],
|
|
});
|
|
expect(r.matched).toBe(true);
|
|
});
|
|
|
|
test("body 第一条规则失败立即返回", () => {
|
|
const r = checkHttpExpect(200, {}, "hello world", 100, {
|
|
body: [{ contains: "missing" }, { contains: "hello" }],
|
|
});
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.path).toBe("body[0]");
|
|
});
|
|
|
|
test("body 为 null 但有 body 规则时报错", () => {
|
|
const r = checkHttpExpect(200, {}, null, 100, { body: [{ contains: "test" }] });
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.kind).toBe("error");
|
|
});
|
|
|
|
test("完整流水线 status->duration->headers->body 全部通过", () => {
|
|
const r = checkHttpExpect(200, { "content-type": "application/json" }, JSON.stringify({ status: "healthy" }), 50, {
|
|
body: [{ json: { equals: "healthy", path: "$.status" } }],
|
|
headers: { "content-type": { contains: "json" } },
|
|
maxDurationMs: 100,
|
|
status: [200],
|
|
});
|
|
expect(r.matched).toBe(true);
|
|
expect(r.failure).toBeNull();
|
|
});
|
|
|
|
test("完整流水线 status 通过但 duration 失败", () => {
|
|
const r = checkHttpExpect(200, {}, "", 500, { maxDurationMs: 100, status: [200] });
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("duration");
|
|
});
|
|
|
|
test("完整流水线 status 和 duration 通过但 headers 失败", () => {
|
|
const r = checkHttpExpect(200, { "x-api": "v1" }, "", 50, {
|
|
headers: { "x-api": "v2" },
|
|
maxDurationMs: 100,
|
|
status: [200],
|
|
});
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("headers");
|
|
});
|
|
|
|
test("完整流水线 status/duration/headers 通过但 body 失败", () => {
|
|
const r = checkHttpExpect(200, { "content-type": "text/plain" }, "error occurred", 50, {
|
|
body: [{ contains: "success" }],
|
|
headers: { "content-type": "text/plain" },
|
|
maxDurationMs: 100,
|
|
status: [200],
|
|
});
|
|
expect(r.matched).toBe(false);
|
|
expect(r.failure!.phase).toBe("body");
|
|
});
|
|
});
|