1462 lines
64 KiB
JavaScript
1462 lines
64 KiB
JavaScript
// SPEC: PJ2026-01060508 Web哨兵 draft-2026-06-27-p12-cadence-scheduler-monitor-web.
|
|
// Responsibility: Vue monitor-web runtime for sentinel trend, timeline, detail and finding observability.
|
|
import { createApp, computed, nextTick, onMounted, ref, watch } from "./vendor/vue.esm-browser.prod.js";
|
|
|
|
const bootstrap = readBootstrap();
|
|
const internalTextPattern = /水合|投影|Trace|trace|Shell|API|DOM|Console|console|Runner|runner|JSONL|steer|facts|分页|HTTP|http|requestfailed|pageerror|Final Response|Code Agent|web-probe|observe|analyzer|终态/u;
|
|
const memoryChartFrame = Object.freeze({
|
|
left: 64,
|
|
right: 696,
|
|
top: 20,
|
|
bottom: 132,
|
|
yLabelX: 56,
|
|
xLabelY: 154,
|
|
xTitleY: 172,
|
|
});
|
|
|
|
createApp({
|
|
setup() {
|
|
const initialDeepLink = readMonitorDeepLink();
|
|
const loading = ref(true);
|
|
const error = ref("");
|
|
const overview = ref(null);
|
|
const runs = ref([]);
|
|
const findings = ref([]);
|
|
const runCheckSummaries = ref({});
|
|
const selectedRunId = ref("");
|
|
const selectedDetail = ref(null);
|
|
const runFilter = ref("");
|
|
const severityFilter = ref("");
|
|
const checkScope = ref("run");
|
|
const checkRunId = ref("");
|
|
const checkTimeWindow = ref("24h");
|
|
const checkSeverityFilter = ref("alert");
|
|
const findingFilter = ref("");
|
|
const activeCheckItem = ref(null);
|
|
const autoRefresh = ref(true);
|
|
const refreshSeconds = ref(30);
|
|
const lastLoadedAt = ref("");
|
|
const hoveredTrendDot = ref(null);
|
|
const deepLinkRunId = ref(initialDeepLink.runId);
|
|
const deepLinkFocus = ref(initialDeepLink.focus);
|
|
let pendingInitialFocus = initialDeepLink.focus;
|
|
let lastAutoRefreshAt = 0;
|
|
|
|
const sentinels = computed(() => {
|
|
const rows = Array.isArray(bootstrap.sentinels) ? bootstrap.sentinels : [];
|
|
return rows.length > 0 ? rows : [{ id: bootstrap.sentinelId, enabled: true }];
|
|
});
|
|
|
|
const currentStatus = computed(() => String(overview.value?.status || "idle"));
|
|
const latestRun = computed(() => overview.value?.latestRun || null);
|
|
const severityTotals = computed(() => overview.value?.severityCounts || {});
|
|
const filteredRuns = computed(() => {
|
|
const needle = runFilter.value.trim().toLowerCase();
|
|
return runs.value.filter((run) => {
|
|
if (severityFilter.value && severityClass(run) !== severityFilter.value) return false;
|
|
if (!needle) return true;
|
|
return [run.id, run.scenarioId, run.status, run.observerId].some((item) => String(item || "").toLowerCase().includes(needle));
|
|
});
|
|
});
|
|
const selectedRun = computed(() => runs.value.find((run) => run.id === selectedRunId.value) || (selectedDetail.value?.run?.id === selectedRunId.value ? selectedDetail.value.run : null) || latestRun.value);
|
|
const trendRows = computed(() => runs.value.slice().sort((a, b) => Date.parse(a.updatedAt || a.createdAt || "") - Date.parse(b.updatedAt || b.createdAt || "")).slice(-48));
|
|
const latestTrendRun = computed(() => trendRows.value.length > 0 ? trendRows.value[trendRows.value.length - 1] : latestRun.value);
|
|
const trendDurationMax = computed(() => Math.max(0, ...trendRows.value.map((run) => trendDurationMinutes(run))));
|
|
const trendErrorMax = computed(() => Math.max(0, ...trendRows.value.map((run) => trendErrorCount(run))));
|
|
const trendWarningMax = computed(() => Math.max(0, ...trendRows.value.map((run) => trendWarningCount(run))));
|
|
const trendMax = computed(() => Math.max(1, trendDurationMax.value, trendErrorMax.value, trendWarningMax.value));
|
|
const trendPolylines = computed(() => ({
|
|
duration: trendPolyline((run) => trendDurationMinutes(run)),
|
|
red: trendPolyline((run) => trendErrorCount(run)),
|
|
warning: trendPolyline((run) => trendWarningCount(run)),
|
|
}));
|
|
const trendDots = computed(() => trendRows.value.map((run, index) => {
|
|
const red = trendErrorCount(run);
|
|
const warning = trendWarningCount(run);
|
|
const total = trendTotalCount(run);
|
|
const duration = trendDurationMinutes(run);
|
|
const x = trendX(index, trendRows.value.length);
|
|
const durationY = trendY(duration);
|
|
const redY = trendY(red);
|
|
const warningY = trendY(warning);
|
|
const rawTime = run.updatedAt || run.createdAt || "";
|
|
const durationText = runDurationText(run);
|
|
const configTimingText = runTimingConfigText(run);
|
|
const tooltipY = Math.min(durationY, redY, warningY);
|
|
return {
|
|
id: run.id || String(index),
|
|
runId: run.id || "",
|
|
x,
|
|
durationY,
|
|
redY,
|
|
warningY,
|
|
duration,
|
|
tooltipLeft: `${clamp((x / 720) * 100, 16, 84)}%`,
|
|
tooltipTop: `${clamp(((tooltipY + 18) / 142) * 100, 24, 76)}%`,
|
|
red,
|
|
warning,
|
|
total,
|
|
status: run.status || "-",
|
|
severity: severityClass(run),
|
|
rawTime,
|
|
timeLabel: formatDate(rawTime),
|
|
absoluteTime: formatAbsoluteDate(rawTime),
|
|
durationText,
|
|
configTimingText,
|
|
reportSha: shortHash(run.reportJsonSha256 || run.report_json_sha256 || run.reportSha256 || ""),
|
|
title: `${shortId(run.id)} ${formatAbsoluteDate(rawTime)} 错误 ${red} 告警 ${warning} 合计 ${total} 运行 ${durationText} ${configTimingText}`,
|
|
};
|
|
}));
|
|
const timelineRuns = computed(() => runs.value.slice(0, 16));
|
|
const historicalCheckFindings = computed(() => {
|
|
const rows = findings.value.filter((item) => item.checkRegistered !== false || item.rootCause || item.nextAction || ["red", "warning", "info"].includes(severityClass(item)));
|
|
return rows.slice(0, 200);
|
|
});
|
|
const runDetailCheckFindings = computed(() => {
|
|
const rows = selectedDetail.value?.findings;
|
|
return Array.isArray(rows) ? rows : [];
|
|
});
|
|
const detailMemory = computed(() => selectedDetail.value?.memory && typeof selectedDetail.value.memory === "object" ? selectedDetail.value.memory : {});
|
|
const detailMemorySeries = computed(() => {
|
|
const rows = Array.isArray(detailMemory.value.pageSeries) ? detailMemory.value.pageSeries : [];
|
|
return rows.map((series) => ({
|
|
...series,
|
|
points: Array.isArray(series.points) ? series.points.filter((point) => Number.isFinite(Number(point?.memoryMb))) : [],
|
|
})).filter((series) => series.points.length > 0);
|
|
});
|
|
const detailMemoryMax = computed(() => Math.max(1, ...detailMemorySeries.value.flatMap((series) => series.points.map((point) => number(point.memoryMb)))));
|
|
const detailMemoryDurationMax = computed(() => Math.max(1, ...detailMemorySeries.value.flatMap((series) => series.points.map((point) => number(point.elapsedMinutes)))));
|
|
const detailMemoryYAxisTicks = computed(() => memoryTickValues(detailMemoryMax.value).map((value) => ({
|
|
key: `memory-y-${value}`,
|
|
value,
|
|
y: memoryYValue(value),
|
|
label: formatMb(value),
|
|
})));
|
|
const detailMemoryXAxisTicks = computed(() => memoryTickValues(detailMemoryDurationMax.value).map((value) => ({
|
|
key: `memory-x-${value}`,
|
|
value,
|
|
x: memoryXValue(value),
|
|
label: formatMinuteTick(value),
|
|
})));
|
|
const scopedCheckFindings = computed(() => {
|
|
if (checkScope.value === "history") return historicalCheckFindings.value;
|
|
return runDetailCheckFindings.value;
|
|
});
|
|
const scopedCheckSummary = computed(() => summarizeCheckRows(scopedCheckFindings.value));
|
|
const visibleCheckFindings = computed(() => {
|
|
const needle = findingFilter.value.trim().toLowerCase();
|
|
const rows = scopedCheckFindings.value.filter((item) => checkMatchesLevel(item, checkSeverityFilter.value));
|
|
const filtered = needle.length === 0 ? rows : rows.filter((item) => findingSearchText(item).includes(needle));
|
|
return filtered.slice(0, 64);
|
|
});
|
|
const visibleCheckSummary = computed(() => summarizeCheckRows(visibleCheckFindings.value));
|
|
const checkScopeRun = computed(() => runs.value.find((run) => run.id === selectedRunId.value) || selectedRun.value || latestRun.value);
|
|
const checkScopeText = computed(() => {
|
|
if (checkScope.value === "history") return `历史聚合 · ${timeWindowLabel(checkTimeWindow.value)}`;
|
|
return `运行记录 · ${checkScopeRun.value?.id || "未选择"}`;
|
|
});
|
|
const cadence = computed(() => {
|
|
const intervalMs = Number(overview.value?.scheduler?.intervalMs || 0);
|
|
const latestAge = Number(overview.value?.freshness?.latestRunAgeSeconds ?? -1);
|
|
const heartbeatAge = Number(overview.value?.freshness?.schedulerHeartbeatAgeSeconds ?? -1);
|
|
const intervalSeconds = intervalMs > 0 ? Math.round(intervalMs / 1000) : 0;
|
|
const stale = intervalSeconds > 0 && latestAge > intervalSeconds * 2;
|
|
return {
|
|
intervalSeconds,
|
|
latestAge,
|
|
heartbeatAge,
|
|
stale,
|
|
label: intervalSeconds > 0 ? `${formatDuration(intervalSeconds)} 间隔` : "未配置",
|
|
alert: stale ? `最近运行 ${formatDuration(latestAge)} 前,超过预设间隔 2 倍;按 SPEC 作为非阻塞报警展示。` : "运行新鲜度在预设窗口内",
|
|
};
|
|
});
|
|
const healthChecks = computed(() => {
|
|
const checks = overview.value?.health?.checks || {};
|
|
return Object.entries(checks).map(([key, value]) => ({
|
|
key,
|
|
ok: value?.ok !== false,
|
|
text: `${key} ${value?.ok === false ? "异常" : "ok"}`,
|
|
}));
|
|
});
|
|
|
|
async function loadAll(options = {}) {
|
|
if (!options.silent) loading.value = true;
|
|
error.value = "";
|
|
setReady(false);
|
|
try {
|
|
const overviewPayload = await fetchJson("/api/overview");
|
|
const [runsResult, findingsResult] = await Promise.allSettled([
|
|
fetchJson("/api/runs?limit=30&sort=updated"),
|
|
fetchJson(findingsApiPath(checkTimeWindow.value)),
|
|
]);
|
|
overview.value = overviewPayload;
|
|
if (runsResult.status === "fulfilled") {
|
|
const runsPayload = runsResult.value;
|
|
runs.value = Array.isArray(runsPayload.runs) ? runsPayload.runs : Array.isArray(runsPayload.items) ? runsPayload.items : [];
|
|
} else if (runs.value.length === 0) {
|
|
throw runsResult.reason;
|
|
}
|
|
if (findingsResult.status === "fulfilled") {
|
|
const findingsPayload = findingsResult.value;
|
|
findings.value = Array.isArray(findingsPayload.findings) ? findingsPayload.findings : Array.isArray(findingsPayload.groups) ? findingsPayload.groups : [];
|
|
} else {
|
|
console.warn("monitor-web findings refresh failed", findingsResult.reason);
|
|
}
|
|
await refreshRunCheckSummaries(runs.value);
|
|
lastLoadedAt.value = new Date().toISOString();
|
|
lastAutoRefreshAt = Date.now();
|
|
const deepLinkedRun = deepLinkRunId.value ? runs.value.find((run) => run.id === deepLinkRunId.value) : null;
|
|
const keepSelected = runs.value.find((run) => run.id === selectedRunId.value);
|
|
const nextRun = deepLinkedRun || keepSelected || runs.value[0] || latestRun.value;
|
|
const nextRunRef = deepLinkRunId.value && !deepLinkedRun && !keepSelected ? deepLinkRunId.value : nextRun;
|
|
if ((typeof nextRunRef === "string" && nextRunRef) || nextRunRef?.id) await selectRun(nextRunRef, true, { focus: pendingInitialFocus });
|
|
pendingInitialFocus = "";
|
|
} catch (cause) {
|
|
const message = String(cause?.message || cause);
|
|
if (!options.silent || runs.value.length === 0) error.value = message;
|
|
else console.warn("monitor-web silent refresh failed", message);
|
|
} finally {
|
|
loading.value = false;
|
|
setReady(true);
|
|
}
|
|
}
|
|
|
|
async function selectRun(run, silent = false, options = {}) {
|
|
const runId = typeof run === "string" ? run : run?.id;
|
|
if (!runId) return;
|
|
selectedRunId.value = runId;
|
|
checkRunId.value = runId;
|
|
deepLinkRunId.value = runId;
|
|
const nextFocus = normalizeDeepLinkFocus(options.focus || deepLinkFocus.value || "");
|
|
deepLinkFocus.value = nextFocus;
|
|
syncMonitorDeepLink(runId, nextFocus);
|
|
if (!silent) selectedDetail.value = null;
|
|
try {
|
|
selectedDetail.value = await fetchJson(`/api/runs/${encodeURIComponent(runId)}`);
|
|
} catch (cause) {
|
|
selectedDetail.value = { ok: false, error: String(cause?.message || cause), runId };
|
|
}
|
|
await focusDeepLinkedPanel(nextFocus);
|
|
}
|
|
|
|
function refreshNow() {
|
|
void loadAll();
|
|
}
|
|
|
|
function selectCheckRun() {
|
|
const run = runs.value.find((item) => item.id === checkRunId.value);
|
|
if (run) void selectRun(run);
|
|
}
|
|
|
|
async function focusDeepLinkedPanel(focus) {
|
|
if (!isMemoryFocus(focus)) return;
|
|
await nextTick();
|
|
const card = document.querySelector("[data-run-memory-chart='true']");
|
|
if (card instanceof HTMLElement) {
|
|
card.scrollIntoView({ block: "center", inline: "nearest", behavior: "auto" });
|
|
card.setAttribute("data-deep-link-focused", "memory");
|
|
document.querySelector("#monitor-web-root")?.setAttribute("data-monitor-deep-link-focus", "memory");
|
|
}
|
|
}
|
|
|
|
async function refreshRunCheckSummaries(rows) {
|
|
const targets = Array.isArray(rows) ? rows.slice(0, 48) : [];
|
|
const next = { ...runCheckSummaries.value };
|
|
const results = await Promise.allSettled(targets.map(async (run) => {
|
|
const runId = run?.id || run?.runId;
|
|
if (!runId) return null;
|
|
const payload = await fetchJson(`/api/runs/${encodeURIComponent(runId)}`);
|
|
const detailRows = Array.isArray(payload.findings) ? payload.findings : [];
|
|
return [runId, summarizeCheckRows(detailRows)];
|
|
}));
|
|
const failures = results.filter((item) => item.status === "rejected");
|
|
if (failures.length > 0) throw new Error(`监测项详情加载失败: ${failures.length}`);
|
|
for (const result of results) {
|
|
if (result.status !== "fulfilled" || result.value === null) continue;
|
|
const [runId, summary] = result.value;
|
|
next[runId] = summary;
|
|
}
|
|
runCheckSummaries.value = next;
|
|
}
|
|
|
|
function trendSummary(run) {
|
|
const runId = run?.id || run?.runId || "";
|
|
return runId ? runCheckSummaries.value[runId] || emptyCheckSummary() : emptyCheckSummary();
|
|
}
|
|
|
|
function runCheckErrorCount(run) {
|
|
return trendSummary(run).errorTypeCount;
|
|
}
|
|
|
|
function runCheckWarningCount(run) {
|
|
return trendSummary(run).warningTypeCount;
|
|
}
|
|
|
|
function runCheckAlertCount(run) {
|
|
return trendSummary(run).alertTypeCount;
|
|
}
|
|
|
|
function trendErrorCount(run) {
|
|
return runCheckErrorCount(run);
|
|
}
|
|
|
|
function trendWarningCount(run) {
|
|
return runCheckWarningCount(run);
|
|
}
|
|
|
|
function trendTotalCount(run) {
|
|
return runCheckAlertCount(run);
|
|
}
|
|
|
|
function trendDurationMinutes(run) {
|
|
return number(run?.runDurationMinutes ?? run?.durationMinutes ?? run?.timing?.durationMinutes);
|
|
}
|
|
|
|
function openCheckDetail(item) {
|
|
activeCheckItem.value = item || null;
|
|
}
|
|
|
|
function closeCheckDetail() {
|
|
activeCheckItem.value = null;
|
|
}
|
|
|
|
async function refreshHistoricalFindings() {
|
|
try {
|
|
const findingsPayload = await fetchJson(findingsApiPath(checkTimeWindow.value));
|
|
findings.value = Array.isArray(findingsPayload.findings) ? findingsPayload.findings : Array.isArray(findingsPayload.groups) ? findingsPayload.groups : [];
|
|
} catch (cause) {
|
|
console.warn("monitor-web historical findings refresh failed", cause);
|
|
}
|
|
}
|
|
|
|
function findingsApiPath(windowValue) {
|
|
if (windowValue === "all") return "/api/findings?limit=200";
|
|
return `/api/findings?limit=200&window=${encodeURIComponent(windowValue || "24h")}`;
|
|
}
|
|
|
|
function currentHref(item) {
|
|
if (!item || item.id === bootstrap.sentinelId) return bootstrap.basePath || "/";
|
|
if (item.monitorRoot === true) return "/";
|
|
return `/sentinels/${encodeURIComponent(item.id)}/`;
|
|
}
|
|
|
|
function trendPolyline(accessor) {
|
|
if (trendRows.value.length < 2) return "";
|
|
return trendRows.value.map((run, index) => `${trendX(index, trendRows.value.length)},${trendY(accessor(run))}`).join(" ");
|
|
}
|
|
|
|
function trendX(index, total) {
|
|
if (total <= 1) return 24;
|
|
return Math.round(24 + index * (672 / (total - 1)));
|
|
}
|
|
|
|
function trendY(value) {
|
|
return Math.round(126 - (Number(value || 0) / trendMax.value) * 102);
|
|
}
|
|
|
|
function memoryPolyline(series) {
|
|
const points = Array.isArray(series?.points) ? series.points : [];
|
|
if (points.length < 2) return "";
|
|
return points.map((point) => `${memoryX(point)},${memoryY(point)}`).join(" ");
|
|
}
|
|
|
|
function memoryX(point) {
|
|
const elapsed = number(point?.elapsedMinutes);
|
|
return memoryXValue(elapsed);
|
|
}
|
|
|
|
function memoryY(point) {
|
|
const memory = number(point?.memoryMb);
|
|
return memoryYValue(memory);
|
|
}
|
|
|
|
function memoryXValue(elapsedMinutes) {
|
|
const ratio = clamp(number(elapsedMinutes) / detailMemoryDurationMax.value, 0, 1);
|
|
return Math.round(memoryChartFrame.left + ratio * (memoryChartFrame.right - memoryChartFrame.left));
|
|
}
|
|
|
|
function memoryYValue(memoryMb) {
|
|
const ratio = clamp(number(memoryMb) / detailMemoryMax.value, 0, 1);
|
|
return Math.round(memoryChartFrame.bottom - ratio * (memoryChartFrame.bottom - memoryChartFrame.top));
|
|
}
|
|
|
|
function memoryLineClass(index) {
|
|
return `memory-line-${(index % 6) + 1}`;
|
|
}
|
|
|
|
function memorySeriesLabel(series) {
|
|
const sampleCount = Number.isFinite(Number(series?.sampleCount)) ? Number(series.sampleCount) : Array.isArray(series?.points) ? series.points.length : 0;
|
|
const latest = Array.isArray(series?.points) && series.points.length > 0 ? series.points[series.points.length - 1] : null;
|
|
return `${series?.label || series?.pageRole || series?.pageId || "page"} · 最新 ${formatMb(latest?.memoryMb)} · 样本 ${sampleCount}`;
|
|
}
|
|
|
|
function showTrendTooltip(dot) {
|
|
hoveredTrendDot.value = dot;
|
|
}
|
|
|
|
function hideTrendTooltip() {
|
|
hoveredTrendDot.value = null;
|
|
}
|
|
|
|
onMounted(() => {
|
|
void loadAll();
|
|
window.setInterval(() => {
|
|
if (!autoRefresh.value) return;
|
|
if (Date.now() - lastAutoRefreshAt >= Math.max(5, Number(refreshSeconds.value || 30)) * 1000) void loadAll({ silent: true });
|
|
}, 1000);
|
|
});
|
|
|
|
watch(checkTimeWindow, () => {
|
|
if (checkScope.value === "history") void refreshHistoricalFindings();
|
|
});
|
|
|
|
watch(checkScope, () => {
|
|
if (checkScope.value === "history") void refreshHistoricalFindings();
|
|
});
|
|
|
|
return {
|
|
bootstrap,
|
|
loading,
|
|
error,
|
|
overview,
|
|
runs,
|
|
findings,
|
|
runCheckSummaries,
|
|
selectedRunId,
|
|
selectedDetail,
|
|
runFilter,
|
|
severityFilter,
|
|
checkScope,
|
|
checkRunId,
|
|
checkTimeWindow,
|
|
checkSeverityFilter,
|
|
findingFilter,
|
|
activeCheckItem,
|
|
autoRefresh,
|
|
refreshSeconds,
|
|
lastLoadedAt,
|
|
hoveredTrendDot,
|
|
deepLinkRunId,
|
|
deepLinkFocus,
|
|
sentinels,
|
|
currentStatus,
|
|
latestRun,
|
|
severityTotals,
|
|
filteredRuns,
|
|
selectedRun,
|
|
trendRows,
|
|
latestTrendRun,
|
|
trendDurationMax,
|
|
trendErrorMax,
|
|
trendWarningMax,
|
|
trendPolylines,
|
|
trendDots,
|
|
timelineRuns,
|
|
historicalCheckFindings,
|
|
runDetailCheckFindings,
|
|
detailMemory,
|
|
detailMemorySeries,
|
|
detailMemoryMax,
|
|
detailMemoryDurationMax,
|
|
detailMemoryYAxisTicks,
|
|
detailMemoryXAxisTicks,
|
|
memoryFrame: memoryChartFrame,
|
|
scopedCheckFindings,
|
|
scopedCheckSummary,
|
|
visibleCheckFindings,
|
|
visibleCheckSummary,
|
|
checkScopeRun,
|
|
checkScopeText,
|
|
cadence,
|
|
healthChecks,
|
|
loadAll,
|
|
selectRun,
|
|
selectCheckRun,
|
|
openCheckDetail,
|
|
closeCheckDetail,
|
|
refreshNow,
|
|
currentHref,
|
|
showTrendTooltip,
|
|
hideTrendTooltip,
|
|
redCount,
|
|
warningCount,
|
|
findingCount,
|
|
findingSampleCount,
|
|
alertSampleCount,
|
|
runCheckErrorCount,
|
|
runCheckWarningCount,
|
|
runCheckAlertCount,
|
|
trendTotalCount,
|
|
trendErrorCount,
|
|
trendWarningCount,
|
|
trendDurationMinutes,
|
|
memoryPolyline,
|
|
memoryLineClass,
|
|
memorySeriesLabel,
|
|
runDurationText,
|
|
runTimingConfigText,
|
|
severityClass,
|
|
formatDate,
|
|
formatAbsoluteDate,
|
|
formatDuration,
|
|
formatMinutes,
|
|
formatMb,
|
|
shortId,
|
|
rootCauseText,
|
|
findingTitle,
|
|
findingCode,
|
|
checkRowKey,
|
|
checkRunText,
|
|
checkTimeText,
|
|
checkActionText,
|
|
checkDetailRows,
|
|
checkEvidenceRows,
|
|
levelLabel,
|
|
findingGroupCountLabel,
|
|
timeWindowLabel,
|
|
detailSummaryText,
|
|
detailSummaryRows,
|
|
commandSummary,
|
|
statusLabel,
|
|
};
|
|
},
|
|
template: `
|
|
<div class="monitor-shell" data-monitor-shell="true" :data-monitor-status="currentStatus">
|
|
<header class="topbar">
|
|
<div class="title-block">
|
|
<div class="mark" aria-hidden="true"></div>
|
|
<div>
|
|
<h1>HWLAB Web哨兵</h1>
|
|
<p class="subtitle">
|
|
<span>{{ bootstrap.node }} / {{ bootstrap.lane }}</span>
|
|
<span class="mono">{{ bootstrap.sentinelId }}</span>
|
|
<span>Vue monitor-web</span>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div class="toolbar" aria-label="观察面板控制">
|
|
<span class="pill" :class="currentStatus">{{ statusLabel(currentStatus) }}</span>
|
|
<select v-model.number="refreshSeconds" aria-label="刷新间隔">
|
|
<option :value="5">5s</option>
|
|
<option :value="10">10s</option>
|
|
<option :value="30">30s</option>
|
|
</select>
|
|
<button class="toolbar-button" type="button" @click="autoRefresh = !autoRefresh">{{ autoRefresh ? "自动" : "手动" }}</button>
|
|
<button class="toolbar-button" type="button" @click="refreshNow">刷新</button>
|
|
</div>
|
|
</header>
|
|
|
|
<section class="entry-strip" aria-label="哨兵入口">
|
|
<div class="entry-copy">
|
|
<strong>哨兵入口</strong>
|
|
<span>{{ sentinels.length }} 个实例,当前入口固定在第一屏</span>
|
|
</div>
|
|
<div class="entry-links">
|
|
<a
|
|
v-for="item in sentinels"
|
|
:key="item.id"
|
|
class="entry-link"
|
|
:class="{ current: item.id === bootstrap.sentinelId, disabled: item.enabled === false }"
|
|
:href="currentHref(item)"
|
|
>
|
|
<span>{{ item.id }}</span>
|
|
<small>{{ item.id === bootstrap.sentinelId ? "当前" : item.enabled === false ? "停用" : "查看" }}</small>
|
|
</a>
|
|
</div>
|
|
</section>
|
|
|
|
<section class="trend-stage" aria-label="运行趋势与时间线">
|
|
<section class="trend-panel" aria-labelledby="trend-heading">
|
|
<div class="panel-header">
|
|
<div>
|
|
<h2 id="trend-heading">运行趋势曲线</h2>
|
|
<p>按运行更新时间展示最近 {{ trendRows.length }} 次巡检的耗时、独立错误类型和独立告警类型</p>
|
|
</div>
|
|
<span class="pill" :class="cadence.stale ? 'warning' : 'healthy'">{{ cadence.stale ? "非阻塞报警" : "新鲜" }}</span>
|
|
</div>
|
|
<div class="trend-chart-wrap">
|
|
<svg class="trend-chart" viewBox="0 0 720 142" role="img" aria-label="运行耗时、独立错误和独立告警趋势" data-monitor-trend-curve="true">
|
|
<line class="trend-grid-line" x1="24" x2="696" y1="24" y2="24"></line>
|
|
<line class="trend-grid-line" x1="24" x2="696" y1="75" y2="75"></line>
|
|
<line class="trend-grid-line" x1="24" x2="696" y1="126" y2="126"></line>
|
|
<polyline v-if="trendPolylines.duration" class="trend-duration" :points="trendPolylines.duration"></polyline>
|
|
<polyline v-if="trendPolylines.red" class="trend-red" :points="trendPolylines.red" data-monitor-trend-error-curve="true"></polyline>
|
|
<polyline v-if="trendPolylines.warning" class="trend-warning" :points="trendPolylines.warning" data-monitor-trend-warning-curve="true"></polyline>
|
|
<g
|
|
v-for="dot in trendDots"
|
|
:key="dot.id"
|
|
class="trend-dot-hit"
|
|
tabindex="0"
|
|
role="button"
|
|
:aria-label="dot.title"
|
|
@mouseenter="showTrendTooltip(dot)"
|
|
@focusin="showTrendTooltip(dot)"
|
|
@mouseleave="hideTrendTooltip"
|
|
@focusout="hideTrendTooltip"
|
|
>
|
|
<circle class="trend-hit-area" :cx="dot.x" :cy="dot.durationY" r="12"></circle>
|
|
<circle class="trend-dot-red" :cx="dot.x" :cy="dot.redY" r="3"></circle>
|
|
<circle class="trend-dot-warning" :cx="dot.x" :cy="dot.warningY" r="3"></circle>
|
|
<circle class="trend-dot-duration" :cx="dot.x" :cy="dot.durationY" r="4">
|
|
<title>{{ dot.title }}</title>
|
|
</circle>
|
|
</g>
|
|
</svg>
|
|
<div
|
|
v-if="hoveredTrendDot"
|
|
class="trend-tooltip"
|
|
data-monitor-trend-tooltip="true"
|
|
:style="{ left: hoveredTrendDot.tooltipLeft, top: hoveredTrendDot.tooltipTop }"
|
|
>
|
|
<strong>{{ shortId(hoveredTrendDot.runId) }}</strong>
|
|
<span>{{ hoveredTrendDot.absoluteTime }}</span>
|
|
<span>状态 {{ hoveredTrendDot.status }}</span>
|
|
<span>运行 {{ hoveredTrendDot.durationText }}</span>
|
|
<span v-if="hoveredTrendDot.configTimingText !== '-'">配置 {{ hoveredTrendDot.configTimingText }}</span>
|
|
<span>错误 {{ hoveredTrendDot.red }} / 告警 {{ hoveredTrendDot.warning }} / 合计 {{ hoveredTrendDot.total }}</span>
|
|
<span v-if="hoveredTrendDot.reportSha">report {{ hoveredTrendDot.reportSha }}</span>
|
|
</div>
|
|
<div v-if="trendRows.length === 0" class="trend-empty">暂无运行数据</div>
|
|
</div>
|
|
<div class="trend-legend">
|
|
<span class="legend-item"><span class="legend-swatch duration"></span>最新运行耗时 {{ runDurationText(latestTrendRun) }}</span>
|
|
<span class="legend-item"><span class="legend-swatch red"></span>独立错误 {{ trendErrorCount(latestTrendRun) }} / 最高 {{ trendErrorMax }}</span>
|
|
<span class="legend-item"><span class="legend-swatch warning"></span>独立告警 {{ trendWarningCount(latestTrendRun) }} / 最高 {{ trendWarningMax }}</span>
|
|
<span class="legend-item">最近最高耗时 {{ formatMinutes(trendDurationMax) }}</span>
|
|
<span class="legend-item">历史样本累计 错误 {{ redCount({ severityCounts: severityTotals }) }} / 告警 {{ warningCount({ severityCounts: severityTotals }) }}</span>
|
|
<span class="legend-item">{{ cadence.alert }}</span>
|
|
</div>
|
|
</section>
|
|
|
|
<section class="timeline-panel" aria-labelledby="timeline-heading" data-monitor-timeline="true">
|
|
<div class="panel-header">
|
|
<div>
|
|
<h2 id="timeline-heading">运行时间线</h2>
|
|
<p>位于哨兵入口下方,方便首屏确认最近运行</p>
|
|
</div>
|
|
<span class="tag">{{ cadence.label }}</span>
|
|
</div>
|
|
<div class="timeline-list">
|
|
<button
|
|
v-for="run in timelineRuns"
|
|
:key="run.id"
|
|
class="timeline-item"
|
|
:class="severityClass(run)"
|
|
type="button"
|
|
@click="selectRun(run)"
|
|
>
|
|
<span class="muted">{{ formatDate(run.updatedAt || run.createdAt) }}</span>
|
|
<span class="severity-line"><span class="timeline-marker"></span><strong>{{ run.scenarioId || shortId(run.id) }}</strong></span>
|
|
<span class="run-alert-tags">
|
|
<span v-if="runCheckErrorCount(run) > 0" class="tag red">错误 {{ runCheckErrorCount(run) }}</span>
|
|
<span v-if="runCheckWarningCount(run) > 0" class="tag warning">告警 {{ runCheckWarningCount(run) }}</span>
|
|
<span v-if="runCheckAlertCount(run) === 0" class="tag healthy">正常</span>
|
|
<span class="tag">运行 {{ runDurationText(run) }}</span>
|
|
</span>
|
|
</button>
|
|
<div v-if="timelineRuns.length === 0" class="empty">暂无时间线记录</div>
|
|
</div>
|
|
</section>
|
|
</section>
|
|
|
|
<section class="status-strip" aria-label="状态指标">
|
|
<div class="metric">
|
|
<span>最近运行</span>
|
|
<strong>{{ latestRun ? formatDate(latestRun.updatedAt || latestRun.createdAt) : "-" }}</strong>
|
|
</div>
|
|
<div class="metric" :class="{ warning: cadence.stale }">
|
|
<span>调度新鲜度</span>
|
|
<strong>{{ cadence.latestAge >= 0 ? formatDuration(cadence.latestAge) : "-" }}</strong>
|
|
</div>
|
|
<div class="metric">
|
|
<span>历史错误样本</span>
|
|
<strong>{{ redCount({ severityCounts: severityTotals }) }}</strong>
|
|
</div>
|
|
<div class="metric">
|
|
<span>历史告警样本</span>
|
|
<strong>{{ warningCount({ severityCounts: severityTotals }) }}</strong>
|
|
</div>
|
|
<div class="metric">
|
|
<span>最后刷新</span>
|
|
<strong>{{ lastLoadedAt ? formatDate(lastLoadedAt) : "-" }}</strong>
|
|
</div>
|
|
</section>
|
|
|
|
<section v-if="error" class="banner error" id="monitor-web-error">{{ error }}</section>
|
|
|
|
<section class="workspace-grid" aria-label="哨兵工作区" data-monitor-independent-scroll="true">
|
|
<aside class="pane pane-runs" aria-labelledby="runs-heading">
|
|
<div class="pane-header">
|
|
<div>
|
|
<h2 id="runs-heading">运行记录</h2>
|
|
<p class="muted">{{ filteredRuns.length }} / {{ runs.length }}</p>
|
|
</div>
|
|
<span v-if="loading" class="tag">加载中</span>
|
|
</div>
|
|
<div class="filter-row">
|
|
<input v-model="runFilter" type="search" placeholder="搜索 run / scenario" aria-label="搜索运行">
|
|
<select v-model="severityFilter" aria-label="严重级别筛选">
|
|
<option value="">全部</option>
|
|
<option value="red">错误</option>
|
|
<option value="warning">告警</option>
|
|
<option value="info">信息</option>
|
|
<option value="healthy">正常</option>
|
|
</select>
|
|
</div>
|
|
<div class="run-list">
|
|
<button
|
|
v-for="run in filteredRuns"
|
|
:key="run.id"
|
|
class="run-row"
|
|
:class="[severityClass(run), { selected: run.id === selectedRunId }]"
|
|
:data-run-id="run.id"
|
|
:data-run-error-count="runCheckErrorCount(run)"
|
|
:data-run-warning-count="runCheckWarningCount(run)"
|
|
:data-run-alert-count="runCheckAlertCount(run)"
|
|
type="button"
|
|
@click="selectRun(run)"
|
|
>
|
|
<span class="row-line">
|
|
<span class="severity-line"><span class="severity-dot"></span><strong>{{ run.scenarioId || shortId(run.id) }}</strong></span>
|
|
<span v-if="runCheckErrorCount(run) > 0" class="tag red" data-run-error-tag="true">错误 {{ runCheckErrorCount(run) }}</span>
|
|
<span v-if="runCheckWarningCount(run) > 0" class="tag warning" data-run-warning-tag="true">告警 {{ runCheckWarningCount(run) }}</span>
|
|
<span v-if="runCheckAlertCount(run) === 0" class="tag healthy">正常</span>
|
|
</span>
|
|
<span class="row-line muted">
|
|
<span>{{ run.status || "-" }}</span>
|
|
<span>运行 {{ runDurationText(run) }}</span>
|
|
<span>{{ formatDate(run.updatedAt || run.createdAt) }}</span>
|
|
</span>
|
|
</button>
|
|
<div v-if="filteredRuns.length === 0" class="empty">没有匹配的运行记录</div>
|
|
</div>
|
|
</aside>
|
|
|
|
<main class="pane pane-detail" aria-labelledby="detail-heading">
|
|
<div class="pane-header">
|
|
<div>
|
|
<h2 id="detail-heading">运行详情</h2>
|
|
<p class="muted">{{ selectedRun ? selectedRun.id : "未选择" }}</p>
|
|
</div>
|
|
<span v-if="selectedRun" class="tag" :class="severityClass(selectedRun)">{{ severityClass(selectedRun) }}</span>
|
|
</div>
|
|
<div v-if="selectedRun" class="detail-stack">
|
|
<section class="detail-card">
|
|
<h3>摘要</h3>
|
|
<div class="detail-grid">
|
|
<div class="metric"><span>状态</span><strong>{{ selectedRun.status || "-" }}</strong></div>
|
|
<div class="metric"><span>运行分钟</span><strong>{{ runDurationText(selectedRun) }}</strong></div>
|
|
<div class="metric"><span>配置周期</span><strong>{{ runTimingConfigText(selectedRun) }}</strong></div>
|
|
<div class="metric"><span>监测项类型</span><strong>{{ findingCount(selectedRun) }}</strong></div>
|
|
<div class="metric"><span>错误/告警样本</span><strong>{{ alertSampleCount(selectedRun) }}</strong></div>
|
|
<div class="metric"><span>全部样本</span><strong>{{ findingSampleCount(selectedRun) }}</strong></div>
|
|
<div class="metric"><span>Observer</span><strong>{{ selectedRun.observerId || "-" }}</strong></div>
|
|
<div class="metric"><span>更新时间</span><strong>{{ formatDate(selectedRun.updatedAt || selectedRun.createdAt) }}</strong></div>
|
|
</div>
|
|
</section>
|
|
<section class="detail-card">
|
|
<h3>健康检查</h3>
|
|
<div class="check-grid">
|
|
<span v-for="check in healthChecks" :key="check.key" class="check-chip" :class="check.ok ? 'ok' : 'bad'">{{ check.text }}</span>
|
|
<span v-if="healthChecks.length === 0" class="check-chip">无检查数据</span>
|
|
</div>
|
|
</section>
|
|
<section class="detail-card">
|
|
<h3>报告摘要</h3>
|
|
<div class="summary-list">
|
|
<div v-for="row in detailSummaryRows(selectedDetail)" :key="row.key" class="summary-row">
|
|
<span>{{ row.label }}</span>
|
|
<strong>{{ row.value }}</strong>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section
|
|
class="detail-card detail-card-wide memory-card"
|
|
data-run-memory-chart="true"
|
|
:data-memory-run-id="selectedRunId"
|
|
:data-memory-page-count="detailMemorySeries.length"
|
|
:data-memory-sample-count="detailMemory.sampleCount || 0"
|
|
:data-memory-source="detailMemory.source || ''"
|
|
>
|
|
<div class="detail-card-heading">
|
|
<h3>页面内存曲线</h3>
|
|
<span class="tag">每个页面一条线 · 峰值 {{ formatMb(detailMemory.maxMemoryMb || detailMemoryMax) }}</span>
|
|
</div>
|
|
<div v-if="detailMemorySeries.length > 0" class="memory-chart-wrap">
|
|
<svg class="memory-chart" viewBox="0 0 720 178" role="img" aria-label="运行详情页面内存曲线,纵轴为 MB,横轴为运行分钟">
|
|
<g data-memory-axis-y="true">
|
|
<g v-for="tick in detailMemoryYAxisTicks" :key="tick.key" data-memory-axis-y-tick="true">
|
|
<line class="memory-grid-line" :x1="memoryFrame.left" :x2="memoryFrame.right" :y1="tick.y" :y2="tick.y"></line>
|
|
<text class="memory-axis-label memory-axis-label-y" :x="memoryFrame.yLabelX" :y="tick.y + 4">{{ tick.label }}</text>
|
|
</g>
|
|
<line class="memory-axis-line" :x1="memoryFrame.left" :x2="memoryFrame.left" :y1="memoryFrame.top" :y2="memoryFrame.bottom"></line>
|
|
<text class="memory-axis-title memory-axis-title-y" x="18" y="78" transform="rotate(-90 18 78)">内存 MB</text>
|
|
</g>
|
|
<g data-memory-axis-x="true">
|
|
<line class="memory-axis-line" :x1="memoryFrame.left" :x2="memoryFrame.right" :y1="memoryFrame.bottom" :y2="memoryFrame.bottom"></line>
|
|
<g v-for="tick in detailMemoryXAxisTicks" :key="tick.key" data-memory-axis-x-tick="true">
|
|
<line class="memory-tick-line" :x1="tick.x" :x2="tick.x" :y1="memoryFrame.bottom" :y2="memoryFrame.bottom + 6"></line>
|
|
<text class="memory-axis-label memory-axis-label-x" :x="tick.x" :y="memoryFrame.xLabelY">{{ tick.label }}</text>
|
|
</g>
|
|
<text class="memory-axis-title memory-axis-title-x" :x="(memoryFrame.left + memoryFrame.right) / 2" :y="memoryFrame.xTitleY">运行分钟</text>
|
|
</g>
|
|
<polyline
|
|
v-for="(series, index) in detailMemorySeries"
|
|
:key="series.key || series.label || index"
|
|
class="memory-line"
|
|
:class="memoryLineClass(index)"
|
|
:points="memoryPolyline(series)"
|
|
></polyline>
|
|
</svg>
|
|
<div class="memory-legend">
|
|
<span
|
|
v-for="(series, index) in detailMemorySeries"
|
|
:key="(series.key || series.label || index) + '-legend'"
|
|
class="legend-item"
|
|
>
|
|
<span class="legend-swatch memory" :class="memoryLineClass(index)"></span>{{ memorySeriesLabel(series) }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<div v-else class="empty">暂无页面级内存样本</div>
|
|
</section>
|
|
<section class="detail-card">
|
|
<h3>复现命令</h3>
|
|
<pre>{{ commandSummary(selectedDetail) }}</pre>
|
|
</section>
|
|
</div>
|
|
<div v-else class="empty">选择一条运行记录查看详情</div>
|
|
</main>
|
|
</section>
|
|
|
|
<section
|
|
class="pane pane-findings checks-panel"
|
|
aria-labelledby="findings-heading"
|
|
data-monitor-checks="true"
|
|
:data-check-scope="checkScope"
|
|
:data-check-run-id="checkScopeRun ? checkScopeRun.id : ''"
|
|
:data-check-type-count="scopedCheckSummary.typeCount"
|
|
:data-check-error-type-count="scopedCheckSummary.errorTypeCount"
|
|
:data-check-warning-type-count="scopedCheckSummary.warningTypeCount"
|
|
:data-check-alert-type-count="scopedCheckSummary.alertTypeCount"
|
|
:data-check-error-samples="scopedCheckSummary.errorSamples"
|
|
:data-check-warning-samples="scopedCheckSummary.warningSamples"
|
|
:data-check-alert-samples="scopedCheckSummary.alertSamples"
|
|
:data-visible-check-count="visibleCheckFindings.length"
|
|
:data-visible-check-alert-samples="visibleCheckSummary.alertSamples"
|
|
>
|
|
<div class="pane-header">
|
|
<div>
|
|
<h2 id="findings-heading">监测项</h2>
|
|
<p class="muted">{{ checkScopeText }}</p>
|
|
</div>
|
|
<span class="tag">错误/告警样本 {{ scopedCheckSummary.alertSamples }}</span>
|
|
</div>
|
|
<div class="filter-row">
|
|
<select v-model="checkScope" aria-label="监测项作用域">
|
|
<option value="run">按运行记录</option>
|
|
<option value="history">按时间聚合</option>
|
|
</select>
|
|
<select v-model="checkRunId" :disabled="checkScope === 'history'" aria-label="选择运行记录" @change="selectCheckRun">
|
|
<option v-for="run in runs" :key="run.id" :value="run.id">{{ shortId(run.id) }} · {{ formatDate(run.updatedAt || run.createdAt) }}</option>
|
|
</select>
|
|
<select v-model="checkTimeWindow" :disabled="checkScope !== 'history'" aria-label="监测项时间窗口">
|
|
<option value="1h">最近 1 小时</option>
|
|
<option value="24h">最近 24 小时</option>
|
|
<option value="7d">最近 7 天</option>
|
|
<option value="all">全部历史</option>
|
|
</select>
|
|
<select v-model="checkSeverityFilter" aria-label="监测项等级筛选">
|
|
<option value="alert">错误+告警</option>
|
|
<option value="red">错误</option>
|
|
<option value="warning">告警</option>
|
|
<option value="info">信息</option>
|
|
<option value="all">全部</option>
|
|
</select>
|
|
<input v-model="findingFilter" type="search" placeholder="搜索编号 / 标题 / 处理建议" aria-label="搜索监测项">
|
|
</div>
|
|
<div class="check-summary" aria-label="监测项摘要">
|
|
<div class="metric"><span>当前作用域</span><strong>{{ checkScope === "history" ? timeWindowLabel(checkTimeWindow) : shortId(checkScopeRun && checkScopeRun.id) }}</strong></div>
|
|
<div class="metric"><span>监测项类型</span><strong>{{ scopedCheckSummary.typeCount }}</strong></div>
|
|
<div class="metric"><span>错误/告警类型</span><strong>{{ scopedCheckSummary.alertTypeCount }}</strong></div>
|
|
<div class="metric"><span>错误样本</span><strong>{{ scopedCheckSummary.errorSamples }}</strong></div>
|
|
<div class="metric"><span>告警样本</span><strong>{{ scopedCheckSummary.warningSamples }}</strong></div>
|
|
<div class="metric"><span>错误/告警样本</span><strong>{{ scopedCheckSummary.alertSamples }}</strong></div>
|
|
</div>
|
|
<div class="finding-list check-table-wrap">
|
|
<table v-if="visibleCheckFindings.length > 0" class="check-table" aria-label="监测项列表">
|
|
<thead>
|
|
<tr>
|
|
<th scope="col">编号</th>
|
|
<th scope="col">等级</th>
|
|
<th scope="col">标题</th>
|
|
<th scope="col">样本</th>
|
|
<th scope="col">运行记录</th>
|
|
<th scope="col">时间</th>
|
|
<th scope="col">处理建议</th>
|
|
<th scope="col">详情</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr
|
|
v-for="(item, index) in visibleCheckFindings"
|
|
:key="checkRowKey(item, index)"
|
|
class="check-row"
|
|
:class="severityClass(item)"
|
|
data-check-row="true"
|
|
tabindex="0"
|
|
role="button"
|
|
:aria-label="'查看监测项详情 ' + findingCode(item) + ' ' + findingTitle(item)"
|
|
:data-check-code="findingCode(item)"
|
|
@click="openCheckDetail(item)"
|
|
@keydown.enter.prevent="openCheckDetail(item)"
|
|
@keydown.space.prevent="openCheckDetail(item)"
|
|
>
|
|
<td><span class="check-code">{{ findingCode(item) }}</span></td>
|
|
<td><span class="tag" :class="severityClass(item)">{{ levelLabel(item) }}</span></td>
|
|
<td class="check-title-cell"><strong>{{ findingTitle(item) }}</strong><span>{{ rootCauseText(item) }}</span></td>
|
|
<td>{{ findingGroupCountLabel(item) }}</td>
|
|
<td class="mono">{{ checkRunText(item) }}</td>
|
|
<td>{{ checkTimeText(item) }}</td>
|
|
<td>{{ checkActionText(item) }}</td>
|
|
<td><span class="detail-link">查看</span></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
<div v-if="visibleCheckFindings.length === 0" class="empty">暂无匹配监测项</div>
|
|
</div>
|
|
</section>
|
|
|
|
<div
|
|
v-if="activeCheckItem"
|
|
class="check-dialog-backdrop"
|
|
data-check-dialog="true"
|
|
@click.self="closeCheckDetail"
|
|
@keydown.esc="closeCheckDetail"
|
|
>
|
|
<section class="check-dialog" role="dialog" aria-modal="true" aria-labelledby="check-dialog-title" tabindex="-1">
|
|
<header class="check-dialog-header">
|
|
<div>
|
|
<span class="check-code">{{ findingCode(activeCheckItem) }}</span>
|
|
<h2 id="check-dialog-title">{{ findingTitle(activeCheckItem) }}</h2>
|
|
<p class="muted">{{ checkScopeText }}</p>
|
|
</div>
|
|
<div class="check-dialog-actions">
|
|
<span class="tag" :class="severityClass(activeCheckItem)">{{ levelLabel(activeCheckItem) }} · {{ findingGroupCountLabel(activeCheckItem) }}</span>
|
|
<button class="dialog-close" type="button" aria-label="关闭监测项详情" @click="closeCheckDetail">关闭</button>
|
|
</div>
|
|
</header>
|
|
<div class="check-dialog-body">
|
|
<section class="detail-card">
|
|
<h3>基本信息</h3>
|
|
<div class="summary-list">
|
|
<div v-for="row in checkDetailRows(activeCheckItem)" :key="row.key" class="summary-row">
|
|
<span>{{ row.label }}</span>
|
|
<strong>{{ row.value }}</strong>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section class="detail-card">
|
|
<h3>用户影响</h3>
|
|
<p>{{ rootCauseText(activeCheckItem) }}</p>
|
|
<p class="muted">处理建议: {{ checkActionText(activeCheckItem) }}</p>
|
|
</section>
|
|
<section class="detail-card detail-card-wide">
|
|
<h3>关联证据</h3>
|
|
<div class="summary-list">
|
|
<div v-for="row in checkEvidenceRows(activeCheckItem)" :key="row.key" class="summary-row">
|
|
<span>{{ row.label }}</span>
|
|
<strong>{{ row.value }}</strong>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
</div>
|
|
`,
|
|
}).mount("#monitor-web-root");
|
|
|
|
function readBootstrap() {
|
|
const root = document.querySelector("#monitor-web-root");
|
|
const script = document.querySelector("#monitor-web-bootstrap");
|
|
let parsed = {};
|
|
try {
|
|
parsed = script?.textContent ? JSON.parse(script.textContent) : {};
|
|
} catch {
|
|
parsed = {};
|
|
}
|
|
return {
|
|
node: root?.getAttribute("data-node") || parsed.node || "",
|
|
lane: root?.getAttribute("data-lane") || parsed.lane || "",
|
|
sentinelId: root?.getAttribute("data-sentinel-id") || parsed.sentinelId || "",
|
|
basePath: root?.getAttribute("data-base-path") || parsed.basePath || "",
|
|
publicOrigin: root?.getAttribute("data-public-origin") || parsed.publicOrigin || "",
|
|
contractVersion: root?.getAttribute("data-contract-version") || parsed.contractVersion || "",
|
|
sentinels: Array.isArray(parsed.sentinels) ? parsed.sentinels : [],
|
|
};
|
|
}
|
|
|
|
function readMonitorDeepLink() {
|
|
const params = new URLSearchParams(window.location.search || "");
|
|
const runId = String(params.get("run") || params.get("runId") || "").trim();
|
|
const focus = normalizeDeepLinkFocus(params.get("focus") || params.get("panel") || window.location.hash.replace(/^#/u, ""));
|
|
return { runId, focus };
|
|
}
|
|
|
|
function syncMonitorDeepLink(runId, focus) {
|
|
if (!window.history?.replaceState) return;
|
|
const nextFocus = normalizeDeepLinkFocus(focus);
|
|
const url = new URL(window.location.href);
|
|
url.searchParams.set("run", runId);
|
|
if (nextFocus) url.searchParams.set("focus", nextFocus);
|
|
else url.searchParams.delete("focus");
|
|
window.history.replaceState(null, "", url.toString());
|
|
}
|
|
|
|
function normalizeDeepLinkFocus(value) {
|
|
const text = String(value || "").trim().toLowerCase();
|
|
if (["memory", "memory-chart", "run-memory", "page-memory"].includes(text)) return "memory";
|
|
return "";
|
|
}
|
|
|
|
function isMemoryFocus(value) {
|
|
return normalizeDeepLinkFocus(value) === "memory";
|
|
}
|
|
|
|
async function fetchJson(path) {
|
|
const response = await fetch(apiUrl(path), { cache: "no-store" });
|
|
if (!response.ok) throw new Error(`${path} HTTP ${response.status}`);
|
|
return await response.json();
|
|
}
|
|
|
|
function apiUrl(path) {
|
|
const prefix = bootstrap.basePath || "";
|
|
const suffix = path.startsWith("/") ? path : `/${path}`;
|
|
return `${prefix}${suffix}`;
|
|
}
|
|
|
|
function setReady(value) {
|
|
document.querySelector("#monitor-web-root")?.setAttribute("data-monitor-ready", value ? "true" : "false");
|
|
}
|
|
|
|
function redCount(item) {
|
|
const counts = item?.severityCounts || item || {};
|
|
return number(counts.red) + number(counts.critical) + number(counts.error);
|
|
}
|
|
|
|
function warningCount(item) {
|
|
const counts = item?.severityCounts || item || {};
|
|
return number(counts.warning) + number(counts.warn) + number(counts.amber);
|
|
}
|
|
|
|
function findingCount(item) {
|
|
if (Number.isFinite(Number(item?.findingTypeCount))) return Number(item.findingTypeCount);
|
|
if (Number.isFinite(Number(item?.findingCount))) return Number(item.findingCount);
|
|
if (Number.isFinite(Number(item?.finding_count))) return Number(item.finding_count);
|
|
if (Number.isFinite(Number(item?.count))) return Number(item.count);
|
|
const counts = item?.severityCounts;
|
|
if (counts && typeof counts === "object") return Object.keys(counts).length;
|
|
return Object.values(item || {}).reduce((sum, value) => sum + number(value), 0);
|
|
}
|
|
|
|
function findingSampleCount(item) {
|
|
if (Number.isFinite(Number(item?.findingSampleCount))) return Number(item.findingSampleCount);
|
|
const counts = item?.severityCounts;
|
|
if (counts && typeof counts === "object") return Object.values(counts).reduce((sum, value) => sum + number(value), 0);
|
|
return 0;
|
|
}
|
|
|
|
function alertSampleCount(item) {
|
|
if (Number.isFinite(Number(item?.findingAlertSampleCount))) return Number(item.findingAlertSampleCount);
|
|
return redCount(item) + warningCount(item);
|
|
}
|
|
|
|
function summarizeCheckRows(rows) {
|
|
const items = Array.isArray(rows) ? rows : [];
|
|
const errorItems = items.filter((item) => severityBucket(item) === "red");
|
|
const warningItems = items.filter((item) => severityBucket(item) === "warning");
|
|
const sampleCount = (entry) => Number.isFinite(Number(entry?.count)) ? Number(entry.count) : 1;
|
|
const sumSamples = (entries) => entries.reduce((sum, entry) => sum + sampleCount(entry), 0);
|
|
return {
|
|
typeCount: items.length,
|
|
errorTypeCount: errorItems.length,
|
|
warningTypeCount: warningItems.length,
|
|
alertTypeCount: errorItems.length + warningItems.length,
|
|
errorSamples: sumSamples(errorItems),
|
|
warningSamples: sumSamples(warningItems),
|
|
alertSamples: sumSamples(errorItems) + sumSamples(warningItems),
|
|
allSamples: sumSamples(items),
|
|
};
|
|
}
|
|
|
|
function emptyCheckSummary() {
|
|
return {
|
|
typeCount: 0,
|
|
errorTypeCount: 0,
|
|
warningTypeCount: 0,
|
|
alertTypeCount: 0,
|
|
errorSamples: 0,
|
|
warningSamples: 0,
|
|
alertSamples: 0,
|
|
allSamples: 0,
|
|
};
|
|
}
|
|
|
|
function checkMatchesLevel(item, filter) {
|
|
const value = String(filter || "alert");
|
|
const bucket = severityBucket(item);
|
|
if (value === "all") return true;
|
|
if (value === "alert") return bucket === "red" || bucket === "warning";
|
|
return bucket === value;
|
|
}
|
|
|
|
function severityBucket(item) {
|
|
const explicit = String(item?.maxSeverity || item?.checkLevel || item?.severity || item?.level || "").toLowerCase();
|
|
if (["red", "critical", "error", "blocked", "failed"].includes(explicit)) return "red";
|
|
if (["warning", "warn", "amber"].includes(explicit)) return "warning";
|
|
if (["info", "notice"].includes(explicit)) return "info";
|
|
if (redCount(item) > 0) return "red";
|
|
if (warningCount(item) > 0) return "warning";
|
|
return "healthy";
|
|
}
|
|
|
|
function severityClass(item) {
|
|
return severityBucket(item);
|
|
}
|
|
|
|
function formatDate(value) {
|
|
if (!value) return "-";
|
|
const date = new Date(value);
|
|
if (Number.isNaN(date.getTime())) return String(value);
|
|
const now = Date.now();
|
|
const seconds = Math.max(0, Math.round((now - date.getTime()) / 1000));
|
|
if (seconds < 90) return `${seconds}s前`;
|
|
if (seconds < 7200) return `${Math.round(seconds / 60)}m前`;
|
|
if (seconds < 172800) return `${Math.round(seconds / 3600)}h前`;
|
|
return date.toISOString().slice(5, 16).replace("T", " ");
|
|
}
|
|
|
|
function formatAbsoluteDate(value) {
|
|
if (!value) return "-";
|
|
const date = new Date(value);
|
|
if (Number.isNaN(date.getTime())) return String(value);
|
|
return `${date.toISOString().slice(0, 19).replace("T", " ")} UTC`;
|
|
}
|
|
|
|
function formatDuration(seconds) {
|
|
const value = Math.max(0, Number(seconds || 0));
|
|
if (value < 90) return `${Math.round(value)}s`;
|
|
if (value < 7200) return `${Math.round(value / 60)}m`;
|
|
if (value < 172800) return `${Math.round(value / 3600)}h`;
|
|
return `${Math.round(value / 86400)}d`;
|
|
}
|
|
|
|
function formatMinutes(value) {
|
|
const numberValue = optionalNumber(value);
|
|
if (numberValue === null) return "-";
|
|
const rounded = Math.round(numberValue * 100) / 100;
|
|
return `${Number.isInteger(rounded) ? String(rounded) : String(rounded).replace(/0+$/u, "").replace(/\.$/u, "")} 分钟`;
|
|
}
|
|
|
|
function formatMinuteTick(value) {
|
|
const numberValue = optionalNumber(value);
|
|
if (numberValue === null) return "-";
|
|
const rounded = Math.round(numberValue * 100) / 100;
|
|
const text = Number.isInteger(rounded) ? String(rounded) : String(rounded).replace(/0+$/u, "").replace(/\.$/u, "");
|
|
return `${text}m`;
|
|
}
|
|
|
|
function formatMb(value) {
|
|
const numberValue = optionalNumber(value);
|
|
if (numberValue === null) return "-";
|
|
const rounded = Math.round(numberValue * 100) / 100;
|
|
const text = rounded >= 100 ? String(Math.round(rounded)) : rounded >= 10 ? rounded.toFixed(1).replace(/\.0$/u, "") : rounded.toFixed(2).replace(/0+$/u, "").replace(/\.$/u, "");
|
|
return `${text} MB`;
|
|
}
|
|
|
|
function memoryTickValues(maxValue) {
|
|
const max = Math.max(1, number(maxValue));
|
|
return [0, max / 2, max];
|
|
}
|
|
|
|
function runDurationText(run) {
|
|
const timing = run?.timing || {};
|
|
return formatMinutes(optionalNumber(run?.runDurationMinutes, run?.durationMinutes, timing.runDurationMinutes, timing.durationMinutes));
|
|
}
|
|
|
|
function runTimingConfigText(run) {
|
|
const timing = run?.timing || {};
|
|
const cadence = optionalNumber(run?.scenarioCadenceMinutes, timing.scenarioCadenceMinutes);
|
|
const maxRun = optionalNumber(run?.scenarioMaxRunMinutes, timing.scenarioMaxRunMinutes);
|
|
const scheduler = optionalNumber(run?.schedulerIntervalMinutes, timing.schedulerIntervalMinutes);
|
|
const parts = [];
|
|
if (cadence !== null) parts.push(`场景周期 ${formatMinutes(cadence)}`);
|
|
if (maxRun !== null) parts.push(`上限 ${formatMinutes(maxRun)}`);
|
|
if (scheduler !== null) parts.push(`调度 ${formatMinutes(scheduler)}`);
|
|
return parts.length > 0 ? parts.join(" / ") : "-";
|
|
}
|
|
|
|
function runTimingSourceText(run) {
|
|
const timing = run?.timing || {};
|
|
return safeDetailValue(run?.durationSource || timing.durationSource || timing.sourceOfTruth);
|
|
}
|
|
|
|
function timeWindowLabel(value) {
|
|
if (value === "1h") return "最近 1 小时";
|
|
if (value === "24h") return "最近 24 小时";
|
|
if (value === "7d") return "最近 7 天";
|
|
if (value === "all") return "全部历史";
|
|
return String(value || "未配置");
|
|
}
|
|
|
|
function shortId(value) {
|
|
const text = String(value || "");
|
|
return text.length > 18 ? `${text.slice(0, 10)}...${text.slice(-6)}` : text || "-";
|
|
}
|
|
|
|
function shortHash(value) {
|
|
const text = String(value || "");
|
|
if (text.length === 0) return "";
|
|
return text.length > 12 ? text.slice(0, 12) : text;
|
|
}
|
|
|
|
function clamp(value, min, max) {
|
|
return Math.max(min, Math.min(max, value));
|
|
}
|
|
|
|
function rootCauseText(item) {
|
|
const display = checkDisplay(item);
|
|
if (display.summary) return display.summary;
|
|
const safeSummary = safeUserText(item?.checkSummaryZh || item?.check?.summaryZh);
|
|
if (safeSummary) return safeSummary;
|
|
if (item?.rootCause) return "已记录内部根因,见报告详情。";
|
|
return "尚未记录用户可见问题摘要。";
|
|
}
|
|
|
|
function findingTitle(item) {
|
|
const display = checkDisplay(item);
|
|
if (display.title) return display.title;
|
|
return safeUserText(item?.checkTitleZh || item?.check?.titleZh) || "未登记监测项";
|
|
}
|
|
|
|
function findingCode(item) {
|
|
return checkDisplay(item).code;
|
|
}
|
|
|
|
function levelLabel(item) {
|
|
const value = String(item?.checkLevel || item?.severity || item?.level || "").toLowerCase();
|
|
if (["critical", "red"].includes(value)) return "严重";
|
|
if (["error", "blocked", "failed"].includes(value)) return "错误";
|
|
if (["warning", "warn", "amber"].includes(value)) return "告警";
|
|
if (["info", "notice"].includes(value)) return "信息";
|
|
return "未知";
|
|
}
|
|
|
|
function findingGroupCountLabel(item) {
|
|
const runCount = Number.isFinite(Number(item?.runCount)) ? Number(item.runCount) : null;
|
|
const sampleCount = Number.isFinite(Number(item?.count)) ? Number(item.count) : null;
|
|
if (runCount !== null && sampleCount !== null) return `${runCount} 次运行 / ${sampleCount} 样本`;
|
|
if (sampleCount !== null) return `${sampleCount} 样本`;
|
|
if (runCount !== null) return `${runCount} 次运行`;
|
|
return "1 样本";
|
|
}
|
|
|
|
function findingSearchText(item) {
|
|
return [
|
|
item?.checkCode,
|
|
item?.check?.code,
|
|
item?.checkTitleZh,
|
|
item?.check?.titleZh,
|
|
item?.checkLevel,
|
|
item?.severity,
|
|
item?.findingId,
|
|
item?.code,
|
|
item?.summary,
|
|
item?.checkSummaryZh,
|
|
item?.rootCause,
|
|
item?.evidenceSummary,
|
|
item?.nextAction,
|
|
item?.timingStatus,
|
|
item?.timingSourceOfTruth,
|
|
item?.scenarioId,
|
|
item?.latestRunId,
|
|
].filter((value) => value !== null && value !== undefined).join(" ").toLowerCase();
|
|
}
|
|
|
|
function checkRowKey(item, index) {
|
|
return [
|
|
findingCode(item),
|
|
item?.latestRunId || item?.runId || item?.run?.id || "",
|
|
item?.sampleSeq ?? item?.count ?? index,
|
|
].join(":");
|
|
}
|
|
|
|
function checkRunText(item) {
|
|
return shortId(item?.latestRunId || item?.runId || item?.run?.id || item?.scenarioId || "");
|
|
}
|
|
|
|
function checkTimeText(item) {
|
|
const value = item?.latestRunUpdatedAt || item?.updatedAt || item?.updated_at || item?.createdAt || item?.created_at || item?.timestamp || "";
|
|
return value ? formatDate(value) : "-";
|
|
}
|
|
|
|
function checkActionText(item) {
|
|
const display = checkDisplay(item);
|
|
return display.action || safeUserText(item?.checkActionZh || item?.check?.actionZh) || "查看详情后处理";
|
|
}
|
|
|
|
function checkDetailRows(item) {
|
|
if (!item) return [{ key: "empty", label: "状态", value: "未选择监测项" }];
|
|
return [
|
|
{ key: "code", label: "编号", value: findingCode(item) },
|
|
{ key: "level", label: "等级", value: levelLabel(item) },
|
|
{ key: "samples", label: "样本", value: findingGroupCountLabel(item) },
|
|
{ key: "run", label: "运行记录", value: checkRunText(item) },
|
|
{ key: "time", label: "时间", value: checkTimeText(item) },
|
|
{ key: "scenario", label: "场景", value: safeDetailValue(item?.scenarioId || item?.scenario_id) },
|
|
{ key: "observer", label: "观察任务", value: safeDetailValue(item?.observerId || item?.observer_id) },
|
|
{ key: "timing", label: "计时来源", value: checkTimingText(item) },
|
|
{ key: "report", label: "报告", value: shortHash(item?.reportJsonSha256 || item?.report_json_sha256 || item?.reportSha256 || "") || "-" },
|
|
].filter((row) => row.value !== "");
|
|
}
|
|
|
|
function checkEvidenceRows(item) {
|
|
if (!item) return [{ key: "empty", label: "状态", value: "未选择监测项" }];
|
|
const evidence = item?.evidence && typeof item.evidence === "object" && !Array.isArray(item.evidence) ? item.evidence : {};
|
|
const rows = [
|
|
{ key: "summary", label: "证据摘要", value: rootCauseText(item) || safeUserText(item?.evidenceSummary || evidence.summary) },
|
|
{ key: "sample", label: "样本序号", value: safeDetailValue(item?.sampleSeq ?? evidence.sampleSeq) },
|
|
{ key: "page", label: "页面", value: safeDetailValue(item?.pageRole || evidence.pageRole) },
|
|
{ key: "command", label: "命令编号", value: safeDetailValue(item?.commandId || evidence.commandId) },
|
|
{ key: "range", label: "采集范围", value: safeDetailValue(item?.sentinelRange || evidence.sentinelRange) },
|
|
{ key: "timing", label: "计时状态", value: checkTimingText(item) },
|
|
{ key: "blocking", label: "阻塞状态", value: item?.blocking === true ? "阻塞" : "非阻塞" },
|
|
].filter((row) => row.value !== "" && row.value !== "-");
|
|
return rows.length > 0 ? rows : [{ key: "none", label: "证据摘要", value: "已记录到报告详情。" }];
|
|
}
|
|
|
|
function checkTimingText(item) {
|
|
const status = item?.timingStatus || "";
|
|
const source = item?.timingSourceOfTruth || item?.expectedElapsedSource || item?.evidenceKind || "";
|
|
if (!status && !source) return "";
|
|
return [
|
|
status ? `status=${safeDetailValue(status)}` : "",
|
|
source ? `source=${safeDetailValue(source)}` : "",
|
|
item?.timingAlert === true ? "alert=true" : "",
|
|
].filter(Boolean).join(" ");
|
|
}
|
|
|
|
function safeDetailValue(value) {
|
|
if (value === null || value === undefined || value === "") return "-";
|
|
const text = String(value).replace(/\s+/g, " ").trim();
|
|
return text.length > 0 ? text : "-";
|
|
}
|
|
|
|
function checkDisplay(item) {
|
|
const rawCode = rawCheckCode(item);
|
|
const rawId = rawFindingId(item);
|
|
const serviceCode = publicCheckCode(item?.checkCode || item?.check?.code);
|
|
return {
|
|
code: serviceCode || publicCheckCode(rawId || rawCode) || stableCheckCode(rawCode || rawId),
|
|
title: safeUserText(item?.checkTitleZh || item?.check?.titleZh) || "未登记监测项",
|
|
summary: safeUserText(item?.checkSummaryZh || item?.check?.summaryZh) || "已记录监测项详情,见报告原文。",
|
|
action: safeUserText(item?.checkActionZh || item?.check?.actionZh) || "查看详情后处理",
|
|
};
|
|
}
|
|
|
|
function rawCheckCode(item) {
|
|
return String(item?.checkCode || item?.check?.code || rawFindingId(item) || "unknown");
|
|
}
|
|
|
|
function rawFindingId(item) {
|
|
const value = item?.findingId || item?.finding_id || item?.id || item?.kind || item?.code;
|
|
return value === null || value === undefined || value === "" ? "" : String(value);
|
|
}
|
|
|
|
function displayCheckCode(value) {
|
|
const text = String(value || "").replace(/\s+/g, " ").trim();
|
|
if (text.length === 0 || text === "unknown") return "";
|
|
return text;
|
|
}
|
|
|
|
function publicCheckCode(value) {
|
|
const text = displayCheckCode(value);
|
|
return /^(?:CHECK|WBC)-\d+$/u.test(text) ? text : "";
|
|
}
|
|
|
|
function stableCheckCode(value) {
|
|
const text = String(value || "unknown");
|
|
let hash = 0;
|
|
for (let index = 0; index < text.length; index += 1) hash = (hash * 31 + text.charCodeAt(index)) >>> 0;
|
|
return `CHECK-${String(hash % 10000).padStart(4, "0")}`;
|
|
}
|
|
|
|
function safeUserText(value) {
|
|
const text = String(value || "").replace(/\s+/g, " ").trim();
|
|
if (text.length === 0 || internalTextPattern.test(text)) return "";
|
|
return text;
|
|
}
|
|
|
|
function detailSummaryText(detail) {
|
|
return detailSummaryRows(detail).map((row) => `${row.label}: ${row.value}`).join("\n");
|
|
}
|
|
|
|
function detailSummaryRows(detail) {
|
|
if (!detail) return [{ key: "loading", label: "状态", value: "加载详情中" }];
|
|
if (detail.ok === false) return [{ key: "error", label: "状态", value: detail.error || "详情不可用" }];
|
|
const run = detail.run || {};
|
|
const summary = detail.summary || {};
|
|
const artifacts = detail.artifacts || {};
|
|
const counts = run.severityCounts || {};
|
|
const rows = [
|
|
{ key: "run", label: "运行", value: shortId(run.id || run.runId || detail.runId) },
|
|
{ key: "scenario", label: "场景", value: run.scenarioId || run.scenario_id || "-" },
|
|
{ key: "status", label: "状态", value: run.status || "-" },
|
|
{ key: "duration", label: "运行分钟", value: runDurationText(run) },
|
|
{ key: "timing", label: "周期/上限", value: runTimingConfigText(run) },
|
|
{ key: "durationSource", label: "计时来源", value: runTimingSourceText(run) },
|
|
{ key: "checks", label: "监测项类型", value: String(findingCount(run)) },
|
|
{ key: "alertSamples", label: "错误/告警样本", value: String(alertSampleCount(run)) },
|
|
{ key: "allSamples", label: "全部样本", value: String(findingSampleCount(run)) },
|
|
{ key: "error", label: "错误样本", value: String(redCount({ severityCounts: counts })) },
|
|
{ key: "warning", label: "告警样本", value: String(warningCount({ severityCounts: counts })) },
|
|
{ key: "observer", label: "Observer", value: run.observerId || run.observer_id || "-" },
|
|
{ key: "updated", label: "更新时间", value: formatAbsoluteDate(run.updatedAt || run.updated_at || run.createdAt || run.created_at) },
|
|
{ key: "report", label: "报告", value: shortHash(artifacts.reportJsonSha256 || run.reportJsonSha256 || run.report_json_sha256 || "") || "-" },
|
|
];
|
|
const windowStart = summary.analysisWindow?.start || summary.window?.start;
|
|
const windowEnd = summary.analysisWindow?.end || summary.window?.end;
|
|
if (windowStart || windowEnd) rows.push({ key: "window", label: "窗口", value: `${formatDate(windowStart)} - ${formatDate(windowEnd)}` });
|
|
return rows;
|
|
}
|
|
|
|
function commandSummary(detail) {
|
|
const commands = detail?.commands || {};
|
|
const lines = Object.entries(commands).map(([key, value]) => `${key}: ${value}`);
|
|
return lines.length > 0 ? lines.join("\n") : "暂无命令";
|
|
}
|
|
|
|
function statusLabel(status) {
|
|
const value = String(status || "");
|
|
if (value === "blocked") return "阻塞";
|
|
if (value === "degraded") return "降级";
|
|
if (value === "warning") return "告警";
|
|
if (value === "healthy") return "健康";
|
|
return "空闲";
|
|
}
|
|
|
|
function number(value) {
|
|
const parsed = Number(value);
|
|
return Number.isFinite(parsed) ? parsed : 0;
|
|
}
|
|
|
|
function optionalNumber(...values) {
|
|
for (const value of values) {
|
|
if (value === null || value === undefined || value === "") continue;
|
|
const parsed = Number(value);
|
|
if (Number.isFinite(parsed)) return parsed;
|
|
}
|
|
return null;
|
|
}
|