1
0
Files
DiAL/tests/web/components/TargetDetailDrawer.test.tsx
lanyuanxiaoyao 7926514986 feat: 配置变量系统与 target id/name 双字段标识
- 新增顶层 variables 段支持 string/number/boolean 字面量
- target 字符串字段支持 、、{...} 转义语法
- 变量解析优先级: variables -> process.env -> 默认值 -> 报错
- 完整引用保留原始类型,部分引用拼接为字符串
- 变量替换在 YAML 解析后、AJV 校验前执行
- 替换仅作用于 targets,跳过 id/type 字段
- target 新增必填 id 字段作为唯一标识,name 改为可选展示名称
- 数据库存储/API/前端全面迁移到 id 标识
- 统一 checker 运行时类型检查为 es-toolkit predicates
- 同步 delta specs 到主 specs,归档 config-variables 变更
2026-05-17 00:37:54 +08:00

124 lines
3.8 KiB
TypeScript

import "../../../tests/web/test-utils";
import { render } from "@testing-library/react";
import { describe, expect, test, vi } from "bun:test";
import type { HistoryResponse, TargetMetricsResponse, TargetStatus } from "../../../src/shared/api";
import { TargetDetailDrawer } from "../../../src/web/components/TargetDetailDrawer";
describe("TargetDetailDrawer", () => {
const target: TargetStatus = {
currentStreak: null,
group: "default",
id: "1",
interval: "30s",
latestCheck: {
durationMs: 100,
failure: null,
matched: true,
statusDetail: "200 OK",
timestamp: "2025-01-15T10:00:00.000Z",
},
name: "test-target",
recentSamples: [],
stats: { availability: 100, downChecks: 0, totalChecks: 10, upChecks: 10 },
target: "https://example.com",
type: "http",
};
const metricsData: TargetMetricsResponse = {
stats: {
availability: 95,
avgDurationMs: 150,
currentStreak: null,
downChecks: 1,
incidentCount: 1,
longestOutage: null,
mttr: null,
p95DurationMs: 200,
p99DurationMs: 250,
totalChecks: 20,
upChecks: 19,
},
targetId: "1",
trend: [],
window: { bucket: "1h", from: "", to: "" },
};
const historyData: HistoryResponse = {
items: [],
page: 1,
pageSize: 20,
total: 0,
};
const defaultProps = {
activeTab: "overview",
historyData,
historyLoading: false,
metricsData,
metricsLoading: false,
onClose: vi.fn(),
onPageChange: vi.fn(),
onTabChange: vi.fn(),
onTimeChange: vi.fn(),
target,
timeFrom: "2025-01-15T00:00:00.000Z",
timeTo: "2025-01-15T23:59:59.999Z",
};
test("target 为 null 时不崩溃", () => {
const { container } = render(<TargetDetailDrawer {...defaultProps} target={null} />);
// When target is null, the drawer might not render, which is expected behavior
expect(container).not.toBeNull();
});
test("target 存在时不崩溃", () => {
const { asFragment } = render(<TargetDetailDrawer {...defaultProps} />);
// Just verify rendering doesn't throw
expect(asFragment()).not.toBeNull();
});
test("关闭按钮不崩溃", () => {
const onClose = vi.fn();
const { asFragment } = render(<TargetDetailDrawer {...defaultProps} onClose={onClose} />);
// Just verify rendering doesn't throw
expect(asFragment()).not.toBeNull();
});
test("Drawer 使用响应式默认宽度 CSS 变量", () => {
render(<TargetDetailDrawer {...defaultProps} />);
const wrapper = document.querySelector<HTMLElement>(".t-drawer__content-wrapper")!;
expect(wrapper).not.toBeNull();
expect(wrapper.style.width).toBe("var(--target-detail-drawer-width)");
});
test("Drawer 包含业务 className", () => {
render(<TargetDetailDrawer {...defaultProps} />);
const drawer = document.querySelector(".target-detail-drawer");
expect(drawer).not.toBeNull();
});
test("Drawer 启用 sizeDraggable 拖拽", () => {
render(<TargetDetailDrawer {...defaultProps} />);
const wrapper = document.querySelector<HTMLElement>(".t-drawer__content-wrapper")!;
expect(wrapper).not.toBeNull();
const dragLine = wrapper.querySelector('[style*="col-resize"]');
expect(dragLine).not.toBeNull();
});
test("Drawer 拖拽宽度不写入 localStorage", () => {
const keysBefore = window.localStorage.length;
render(<TargetDetailDrawer {...defaultProps} />);
expect(window.localStorage.length).toBe(keysBefore);
});
test("Drawer sizeDraggable 配置最小拖拽边界", () => {
render(<TargetDetailDrawer {...defaultProps} />);
const wrapper = document.querySelector<HTMLElement>(".t-drawer__content-wrapper")!;
expect(wrapper).not.toBeNull();
const dragLine = wrapper.querySelector('[style*="col-resize"]');
expect(dragLine).not.toBeNull();
});
});