1
0
Files
DiAL/tests/server/checker/runner/detail.test.ts
lanyuanxiaoyao 375dd3492b feat: 结构化 observation 替代 statusDetail,API 层动态构造 detail
- CheckResult: statusDetail -> observation (持久化) + detail (API 动态派生)
- 存储: status_detail 列 -> observation TEXT (JSON)
- CheckerDefinition: 新增 buildDetail(observation) 方法
- 各 checker 返回结构化 observation,API 层通过 registry 调用 buildDetail
- HTTP: bodyPreview 在 status/header 失败时也提前采集
- UDP: observation 包含 durationMs,未响应归为 error failure
- CMD: 超时/输出超限时保留已收集 observation
- TCP: connectTimeMs 仅含连接建立耗时,不含 banner 等待
- 新增 buildDetail 单测和 mapCheckResult 覆盖测试
- 同步 openspec 主规范,归档 checker-observation 变更
2026-05-19 22:49:00 +08:00

68 lines
2.6 KiB
TypeScript

import { describe, expect, test } from "bun:test";
import { CommandChecker } from "../../../../src/server/checker/runner/cmd/execute";
import { DbChecker } from "../../../../src/server/checker/runner/db/execute";
import { HttpChecker } from "../../../../src/server/checker/runner/http/execute";
import { IcmpChecker } from "../../../../src/server/checker/runner/icmp/execute";
import { LlmChecker } from "../../../../src/server/checker/runner/llm/execute";
import { TcpChecker } from "../../../../src/server/checker/runner/tcp/execute";
import { UdpChecker } from "../../../../src/server/checker/runner/udp/execute";
describe("Checker buildDetail", () => {
test("HTTP detail", () => {
expect(new HttpChecker().buildDetail({ statusCode: 200 })).toBe("HTTP 200");
});
test("TCP detail", () => {
const detail = new TcpChecker().buildDetail({
banner: "220 smtp.example.com ESMTP",
connected: true,
connectTimeMs: 12,
});
expect(detail).toContain("connected in 12ms");
expect(detail).toContain("banner:");
});
test("UDP detail", () => {
const checker = new UdpChecker();
expect(checker.buildDetail({ durationMs: 12, responded: true, responsePreview: "PONG", responseSize: 4 })).toBe(
"responded in 12ms, 4 bytes, response: PONG",
);
expect(checker.buildDetail({ durationMs: 200, responded: false })).toBe("no response in 200ms");
});
test("Ping detail", () => {
const checker = new IcmpChecker();
expect(checker.buildDetail({ alive: true, avgLatencyMs: 12, packetLoss: 0, received: 3, transmitted: 3 })).toBe(
"alive, avg 12ms, loss 0% (3/3)",
);
expect(checker.buildDetail({ alive: false, received: 0, transmitted: 3 })).toBe("unreachable (0/3 received)");
});
test("DB detail", () => {
const checker = new DbChecker();
expect(checker.buildDetail({ connected: true, rowCount: 3 })).toBe("3 rows");
expect(checker.buildDetail({ connected: true, rowCount: null })).toBe("connected");
});
test("CMD detail", () => {
expect(new CommandChecker().buildDetail({ exitCode: 0 })).toBe("exitCode=0");
});
test("LLM detail", () => {
const detail = new LlmChecker().buildDetail({
finishReason: "stop",
http: { status: 200 },
mode: "http",
outputLength: 2,
provider: "openai",
usage: { inputTokens: 12, outputTokens: 2, totalTokens: 14 },
});
expect(detail).toContain("LLM openai http");
expect(detail).toContain("200");
expect(detail).toContain("finish=stop");
expect(detail).toContain("output=2 chars");
expect(detail).toContain("usage=12/2 tokens");
});
});