92f20ed978
- 桌面端(≥1024px)三栏 master-detail:Runs | Detail | Findings,点击 run 无需滚动 - 状态摘要条替换 4 metric cards,单行 ~48px,合并 status-pill/overall 重复值 - overview checks 全绿折叠为单 pill,点击展开明细 - runs/findings filter 默认折叠为单行摘要 chip,展开功能不变 - Detail 分 5 tab(概要/发现项/多轮摘要/Trace/证据与命令),URL 支持 ?run=&tab= - Finding 默认显示聚合,点击 chip drill-down 到列表;item 默认折叠只显示 title+severity+count - Runs 行级 severity 色条(red/amber/info 左 border),一眼定位问题行 - Timeline 升级为底部紧凑条,加相对时间刻度,可折叠 - auto-refresh 交互暂停:hover runs/detail 时暂停刷新 - CLI 命令改可复制按钮+toast,detailCommands 不再纯文本 - 快捷键 j/k 上下选 run、Enter 打开、r 刷新、/ 聚焦搜索 - 降权标签:减少 uppercase/font-weight:800 滥用,只在 red/blocked 强调 - 去冗余 box-shadow:panel 用 1px border,只在 modal/overlay 用阴影 - contract version 升级 draft-2026-06-26-p9-desktop-view-density - 保持纯 vanilla JS + 原生 CSS,无框架/无构建步骤,API 契约不变
1325 lines
59 KiB
JavaScript
1325 lines
59 KiB
JavaScript
// SPEC: PJ2026-01060508 Web哨兵 draft-2026-06-26-p9-desktop-view-density.
|
|
// Responsibility: Browser-side API client, formatting, auto refresh, and base dashboard rendering.
|
|
// Desktop view redesign (issue #1025): 三栏 master-detail、低噪声、渐进披露、排查直觉化。
|
|
// 保持纯 vanilla JS + 原生 CSS,无框架、无构建步骤,所有渲染通过 innerHTML 拼接。
|
|
|
|
/**
|
|
* @typedef {{runId?: string|null, observerId?: string|null, stateDir?: string|null, reportJsonSha256?: string|null, stateRoot?: string|null, source?: string|null}} Traceability
|
|
* @typedef {{id?: string|null, runId?: string|null, scenarioId?: string|null, status?: string|null, observerId?: string|null, stateDir?: string|null, reportJsonSha256?: string|null, findingCount?: number, artifactCount?: number, updatedAt?: string|null, maxSeverity?: string|null, traceability?: Traceability}} SentinelRun
|
|
* @typedef {{ok: boolean, status?: string, node?: string, lane?: string, publicOrigin?: string|null, latestRun?: SentinelRun|null, runCounts?: Record<string, number>, severityCounts?: Record<string, number>, freshness?: Record<string, unknown>, targetValidation?: {maxSeconds?: number, scenarioId?: string|null}, valuesRedacted?: boolean}} SentinelOverview
|
|
* @typedef {{code?: string|null, findingId?: string|null, severity?: string|null, scenarioId?: string|null, count?: number, runCount?: number, latestAt?: string|null, latestRunId?: string|null, latestReportJsonSha256?: string|null, summary?: string|null, traceability?: Traceability|null}} SentinelFindingGroup
|
|
* @typedef {{groups?: SentinelFindingGroup[], findings?: SentinelFindingGroup[], page?: {limit?: number, hasMore?: boolean}}} SentinelFindingsResponse
|
|
* @typedef {{items?: SentinelRun[], runs?: SentinelRun[], page?: {limit?: number, nextCursor?: string|null, hasMore?: boolean}}} SentinelRunsResponse
|
|
* @typedef {{ok?: boolean, view?: string, renderedText?: string, renderedTextBytes?: number, truncated?: boolean, error?: string|null, finalResponse?: Record<string, unknown>|null}} SentinelRunView
|
|
* @typedef {{ok?: boolean, run?: SentinelRun, summary?: Record<string, unknown>, findings?: Record<string, unknown>[], artifacts?: Record<string, unknown>, commands?: Record<string, string>, redaction?: Record<string, unknown>, traceability?: Traceability, viewsAvailable?: string[], valuesRedacted?: boolean}} SentinelRunDetail
|
|
*/
|
|
|
|
const root = document.getElementById("sentinel-dashboard");
|
|
const refs = {
|
|
statusPill: document.getElementById("status-pill"),
|
|
loadingBanner: document.getElementById("loading-banner"),
|
|
errorBanner: document.getElementById("error-banner"),
|
|
manualRefresh: document.getElementById("manual-refresh"),
|
|
autoRefreshEnabled: document.getElementById("auto-refresh-enabled"),
|
|
autoRefreshInterval: document.getElementById("auto-refresh-interval"),
|
|
latestRun: document.getElementById("latest-run"),
|
|
filterRed: document.getElementById("filter-red"),
|
|
originNote: document.getElementById("sentinel-origin-note"),
|
|
statusSummary: document.getElementById("status-summary"),
|
|
summaryStatus: document.getElementById("summary-status"),
|
|
summaryLatest: document.getElementById("summary-latest"),
|
|
summaryLatestAge: document.getElementById("summary-latest-age"),
|
|
summaryFindings: document.getElementById("summary-findings"),
|
|
summaryFindingsNote: document.getElementById("summary-findings-note"),
|
|
summaryScheduler: document.getElementById("summary-scheduler"),
|
|
summaryBudget: document.getElementById("summary-budget"),
|
|
summaryChecks: document.getElementById("summary-checks"),
|
|
overviewChecks: document.getElementById("overview-checks"),
|
|
checkSummaryPill: document.getElementById("check-summary-pill"),
|
|
checkConfig: document.getElementById("check-config"),
|
|
checkPvc: document.getElementById("check-pvc"),
|
|
checkAnalyzer: document.getElementById("check-analyzer"),
|
|
checkPublic: document.getElementById("check-public"),
|
|
checkMaintenance: document.getElementById("check-maintenance"),
|
|
timeline: document.getElementById("run-timeline"),
|
|
timelineCount: document.getElementById("timeline-count"),
|
|
timelineToggle: document.getElementById("timeline-toggle"),
|
|
runsFilterSummary: document.getElementById("runs-filter-summary"),
|
|
filterForm: document.getElementById("runs-filter"),
|
|
filterStatus: document.getElementById("filter-status"),
|
|
filterSeverity: document.getElementById("filter-severity"),
|
|
filterWindow: document.getElementById("filter-window"),
|
|
filterSort: document.getElementById("filter-sort"),
|
|
filterSearch: document.getElementById("filter-search"),
|
|
clearFilters: document.getElementById("clear-filters"),
|
|
runsBody: document.getElementById("runs-body"),
|
|
runsCount: document.getElementById("runs-count"),
|
|
findingsFilterSummary: document.getElementById("findings-filter-summary"),
|
|
findingsFilterForm: document.getElementById("findings-filter"),
|
|
findingFilterSeverity: document.getElementById("finding-filter-severity"),
|
|
findingFilterWindow: document.getElementById("finding-filter-window"),
|
|
findingFilterCode: document.getElementById("finding-filter-code"),
|
|
findingFilterScenario: document.getElementById("finding-filter-scenario"),
|
|
findingClearFilters: document.getElementById("finding-clear-filters"),
|
|
findingAggregation: document.getElementById("finding-aggregation"),
|
|
findingsList: document.getElementById("findings-list"),
|
|
findingsCount: document.getElementById("findings-count"),
|
|
findingsDrilldown: document.getElementById("findings-drilldown"),
|
|
detailTabs: document.getElementById("detail-tabs"),
|
|
detailSubtitle: document.getElementById("detail-subtitle"),
|
|
detailContent: document.getElementById("detail-content"),
|
|
copyToast: document.getElementById("copy-toast"),
|
|
};
|
|
|
|
const state = {
|
|
loading: false,
|
|
selectedRunId: null,
|
|
/** @type {SentinelOverview|null} */
|
|
overview: null,
|
|
/** @type {SentinelRun[]} */
|
|
runs: [],
|
|
/** @type {SentinelFindingGroup[]} */
|
|
findings: [],
|
|
/** @type {SentinelRunDetail|null} */
|
|
runDetail: null,
|
|
runViews: null,
|
|
selectedTraceChoiceIndex: 0,
|
|
lastUpdatedAt: null,
|
|
filters: readFiltersFromLocation(),
|
|
findingFilters: readFindingFiltersFromLocation(),
|
|
selectedTab: readTabFromLocation(),
|
|
selectedFindingCode: new URLSearchParams(window.location.search).get("finding") || "",
|
|
findingDrilldownKey: "",
|
|
runsFilterOpen: false,
|
|
findingsFilterOpen: false,
|
|
checksExpanded: false,
|
|
timelineCollapsed: false,
|
|
pausedByInteraction: false,
|
|
};
|
|
|
|
const DETAIL_TABS = ["overview", "findings", "turn", "trace", "evidence"];
|
|
|
|
const dashboardLimits = {
|
|
timeline: { mobile: 6, tablet: 9, desktop: 16 },
|
|
runs: { mobile: 8, tablet: 12, desktop: 30 },
|
|
findingsRed: { mobile: 4, tablet: 6, desktop: 8 },
|
|
findingsOther: { mobile: 2, tablet: 3, desktop: 5 },
|
|
};
|
|
|
|
const dashboardApi = createDashboardApi();
|
|
const autoRefresh = createAutoRefresh({
|
|
storageKey: "hwlab.webProbeSentinel.dashboard.autoRefresh",
|
|
intervals: [5, 10, 30],
|
|
defaultInterval: 10,
|
|
onRefresh: () => loadDashboard({ silent: true }),
|
|
shouldPause: () => document.hidden || state.loading || state.pausedByInteraction,
|
|
});
|
|
|
|
refs.manualRefresh.addEventListener("click", () => loadDashboard({ silent: false }));
|
|
refs.autoRefreshEnabled.addEventListener("change", () => autoRefresh.setEnabled(refs.autoRefreshEnabled.checked));
|
|
refs.autoRefreshInterval.addEventListener("change", () => autoRefresh.setInterval(Number(refs.autoRefreshInterval.value)));
|
|
refs.latestRun.addEventListener("click", () => jumpToLatestRun());
|
|
refs.filterRed.addEventListener("click", () => toggleRedOnly());
|
|
refs.runsFilterSummary.addEventListener("click", () => toggleRunsFilter());
|
|
refs.findingsFilterSummary.addEventListener("click", () => toggleFindingsFilter());
|
|
refs.checkSummaryPill.addEventListener("click", () => toggleChecks());
|
|
refs.timelineToggle.addEventListener("click", () => toggleTimeline());
|
|
refs.filterForm.addEventListener("submit", (event) => event.preventDefault());
|
|
for (const control of [refs.filterStatus, refs.filterSeverity, refs.filterWindow, refs.filterSort]) {
|
|
control.addEventListener("change", () => applyFilterControls());
|
|
}
|
|
refs.filterSearch.addEventListener("input", debounce(() => applyFilterControls(), 250));
|
|
refs.clearFilters.addEventListener("click", () => {
|
|
state.filters = { status: "", severity: "", window: "", search: "", sort: "updated" };
|
|
writeFiltersToControls();
|
|
updateRunsFilterSummary();
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
});
|
|
refs.findingsFilterForm.addEventListener("submit", (event) => event.preventDefault());
|
|
for (const control of [refs.findingFilterSeverity, refs.findingFilterWindow]) {
|
|
control.addEventListener("change", () => applyFindingFilterControls());
|
|
}
|
|
for (const control of [refs.findingFilterCode, refs.findingFilterScenario]) {
|
|
control.addEventListener("input", debounce(() => applyFindingFilterControls(), 250));
|
|
}
|
|
refs.findingClearFilters.addEventListener("click", () => {
|
|
state.findingFilters = { severity: "", window: "24h", code: "", scenario: "" };
|
|
state.findingDrilldownKey = "";
|
|
writeFindingFiltersToControls();
|
|
updateFindingsFilterSummary();
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
});
|
|
refs.detailTabs.addEventListener("click", (event) => {
|
|
const button = event.target.closest("[data-detail-tab]");
|
|
if (button) selectTab(button.dataset.detailTab);
|
|
});
|
|
document.addEventListener("visibilitychange", () => {
|
|
if (!document.hidden && autoRefresh.enabled()) loadDashboard({ silent: true });
|
|
});
|
|
document.addEventListener("keydown", handleKeyboardShortcut);
|
|
window.addEventListener("resize", debounce(updateViewportClass, 150));
|
|
|
|
// 交互暂停:hover runs 表格或在 detail tab 阅读时暂停 auto-refresh
|
|
refs.runsBody.addEventListener("mouseenter", () => { state.pausedByInteraction = true; }, true);
|
|
refs.runsBody.addEventListener("mouseleave", () => { state.pausedByInteraction = false; }, true);
|
|
refs.detailContent.addEventListener("mouseenter", () => { state.pausedByInteraction = true; }, true);
|
|
refs.detailContent.addEventListener("mouseleave", () => { state.pausedByInteraction = false; }, true);
|
|
|
|
hydrateControls();
|
|
writeFiltersToControls();
|
|
writeFindingFiltersToControls();
|
|
updateRunsFilterSummary();
|
|
updateFindingsFilterSummary();
|
|
updateViewportClass();
|
|
loadDashboard({ silent: false }).catch((error) => renderError(error));
|
|
|
|
function createDashboardApi() {
|
|
return {
|
|
/** @returns {Promise<SentinelOverview>} */
|
|
overview: () => getJson("/api/overview"),
|
|
/** @returns {Promise<SentinelRunsResponse>} */
|
|
runs: (filters) => getJson(`/api/runs?${runsQuery(filters)}`),
|
|
/** @returns {Promise<SentinelFindingsResponse>} */
|
|
findings: (filters) => getJson(`/api/findings?${findingsQuery(filters)}`),
|
|
/** @returns {Promise<SentinelRunDetail>} */
|
|
runDetail: (runId) => getJson(`/api/runs/${encodeURIComponent(runId)}`),
|
|
runViews: (runId, view = null) => {
|
|
const query = new URLSearchParams({ maxBytes: "24000" });
|
|
if (view) query.set("view", view);
|
|
return getJson(`/api/runs/${encodeURIComponent(runId)}/views?${query.toString()}`);
|
|
},
|
|
};
|
|
}
|
|
|
|
async function getJson(path) {
|
|
const response = await fetch(path, { headers: { accept: "application/json" } });
|
|
const body = await response.json().catch(() => ({}));
|
|
if (!response.ok || body.ok === false) {
|
|
const message = body.error || `${response.status} ${response.statusText}`;
|
|
throw new Error(message);
|
|
}
|
|
return body;
|
|
}
|
|
|
|
function createAutoRefresh(options) {
|
|
let enabled = false;
|
|
let intervalSeconds = options.defaultInterval;
|
|
let countdown = intervalSeconds;
|
|
let timer = null;
|
|
|
|
try {
|
|
const saved = JSON.parse(localStorage.getItem(options.storageKey) || "{}");
|
|
if (saved.enabled === true) enabled = true;
|
|
if (options.intervals.includes(Number(saved.intervalSeconds))) intervalSeconds = Number(saved.intervalSeconds);
|
|
} catch {
|
|
enabled = false;
|
|
}
|
|
|
|
function persist() {
|
|
try {
|
|
localStorage.setItem(options.storageKey, JSON.stringify({ enabled, intervalSeconds }));
|
|
} catch {
|
|
/* ignore */
|
|
}
|
|
}
|
|
|
|
async function tick() {
|
|
if (!enabled || options.shouldPause()) return;
|
|
if (countdown > 0) {
|
|
countdown -= 1;
|
|
return;
|
|
}
|
|
countdown = intervalSeconds;
|
|
await options.onRefresh();
|
|
}
|
|
|
|
function start() {
|
|
if (timer !== null) return;
|
|
timer = window.setInterval(tick, 1000);
|
|
}
|
|
|
|
function stop() {
|
|
if (timer === null) return;
|
|
window.clearInterval(timer);
|
|
timer = null;
|
|
}
|
|
|
|
return {
|
|
enabled: () => enabled,
|
|
intervalSeconds: () => intervalSeconds,
|
|
setEnabled(value) {
|
|
enabled = value;
|
|
countdown = intervalSeconds;
|
|
persist();
|
|
if (enabled) start();
|
|
else stop();
|
|
},
|
|
setInterval(value) {
|
|
if (!options.intervals.includes(value)) return;
|
|
intervalSeconds = value;
|
|
countdown = value;
|
|
persist();
|
|
},
|
|
start,
|
|
};
|
|
}
|
|
|
|
function hydrateControls() {
|
|
refs.autoRefreshEnabled.checked = autoRefresh.enabled();
|
|
refs.autoRefreshInterval.value = String(autoRefresh.intervalSeconds());
|
|
if (autoRefresh.enabled()) autoRefresh.start();
|
|
}
|
|
|
|
async function loadDashboard(options) {
|
|
if (state.loading) return;
|
|
state.loading = true;
|
|
renderLoading(!options.silent);
|
|
try {
|
|
const [overview, runs, findings] = await Promise.all([
|
|
dashboardApi.overview(),
|
|
dashboardApi.runs(state.filters),
|
|
dashboardApi.findings(state.findingFilters),
|
|
]);
|
|
state.overview = overview;
|
|
state.runs = Array.isArray(runs.items) ? runs.items : Array.isArray(runs.runs) ? runs.runs : [];
|
|
state.findings = Array.isArray(findings.groups) ? findings.groups : Array.isArray(findings.findings) ? findings.findings : [];
|
|
state.lastUpdatedAt = new Date();
|
|
const queryRun = new URLSearchParams(window.location.search).get("run");
|
|
if (queryRun && state.selectedRunId === null) state.selectedRunId = queryRun;
|
|
if (state.selectedRunId && !state.runs.some((run) => run.runId === state.selectedRunId || run.id === state.selectedRunId)) {
|
|
state.selectedRunId = null;
|
|
state.runDetail = null;
|
|
state.runViews = null;
|
|
state.selectedTraceChoiceIndex = 0;
|
|
}
|
|
renderDashboard();
|
|
if (state.selectedRunId && state.runDetail === null) await selectRun(state.selectedRunId);
|
|
} catch (error) {
|
|
renderError(error);
|
|
} finally {
|
|
state.loading = false;
|
|
renderLoading(false);
|
|
}
|
|
}
|
|
|
|
async function selectRun(runId) {
|
|
if (!runId) return;
|
|
state.selectedRunId = runId;
|
|
state.runDetail = null;
|
|
state.runViews = null;
|
|
state.selectedTraceChoiceIndex = 0;
|
|
syncLocationQuery();
|
|
renderRuns();
|
|
refs.detailSubtitle.textContent = runId;
|
|
refs.detailTabs.hidden = false;
|
|
refs.detailContent.innerHTML = '<div class="empty-state">加载中</div>';
|
|
try {
|
|
const [detail, views] = await Promise.all([
|
|
dashboardApi.runDetail(runId),
|
|
loadRunViews(runId),
|
|
]);
|
|
state.runDetail = detail;
|
|
state.runViews = views;
|
|
renderDetail();
|
|
} catch (error) {
|
|
refs.detailContent.innerHTML = `<div class="empty-state">${escapeHtml(error.message || String(error))}</div>`;
|
|
}
|
|
}
|
|
|
|
async function loadRunViews(runId, view = null) {
|
|
try {
|
|
return await dashboardApi.runViews(runId, view);
|
|
} catch (error) {
|
|
return { ok: false, error: error?.message || String(error), valuesRedacted: true };
|
|
}
|
|
}
|
|
|
|
function renderDashboard() {
|
|
refs.errorBanner.hidden = true;
|
|
renderOverview();
|
|
renderTimeline();
|
|
renderRuns();
|
|
renderFindings();
|
|
renderDetail();
|
|
}
|
|
|
|
function renderOverview() {
|
|
const overview = state.overview || {};
|
|
const status = overview.status || (overview.ok ? "healthy" : "degraded");
|
|
const latest = overview.latestRun || null;
|
|
const latestStatus = latest?.status || status;
|
|
refs.statusPill.textContent = displayStatus(latestStatus);
|
|
refs.statusPill.className = `status-pill ${statusClass(latestStatus)}`;
|
|
|
|
// 状态摘要条合并 status-pill 信息(B1 去重:不再单独显示 metric-overall 重复值)
|
|
refs.statusSummary.hidden = false;
|
|
refs.summaryStatus.textContent = displayStatus(latestStatus);
|
|
refs.summaryStatus.className = statusClass(latestStatus);
|
|
refs.originNote.textContent = overview.publicOrigin || root.dataset.publicOrigin || "";
|
|
|
|
refs.summaryLatest.textContent = latest?.runId || latest?.id || "-";
|
|
refs.summaryLatestAge.textContent = latest?.updatedAt ? `${formatRelative(latest.updatedAt)} 更新` : "-";
|
|
|
|
const severityCounts = overview.severityCounts || {};
|
|
const totalFindings = Object.values(severityCounts).reduce((sum, value) => sum + Number(value || 0), 0);
|
|
refs.summaryFindings.textContent = formatNumber(totalFindings);
|
|
refs.summaryFindingsNote.textContent = formatSeveritySummary(severityCounts);
|
|
|
|
const scheduler = overview.scheduler || {};
|
|
const heartbeat = scheduler.heartbeat || {};
|
|
refs.summaryScheduler.textContent = heartbeat.at ? formatRelative(heartbeat.at) : "未知";
|
|
const maxSeconds = overview.targetValidation?.maxSeconds ?? 120;
|
|
refs.summaryBudget.textContent = `预算 ${maxSeconds}s`;
|
|
|
|
const checks = overview.health?.checks || {};
|
|
const checkResults = [
|
|
{ label: "配置", ok: checks.config?.ok, detail: checks.config?.status },
|
|
{ label: "PVC", ok: checks.pvc?.ok, detail: checks.pvc?.stateRoot },
|
|
{ label: "分析器", ok: checks.analyzer?.ok, detail: "observe analyze" },
|
|
{ label: "公开入口", ok: Boolean(overview.publicOrigin), detail: overview.publicOrigin || "-" },
|
|
];
|
|
const maintenance = overview.maintenance || {};
|
|
checkResults.push({ label: "维护窗口", ok: maintenance.active !== true, detail: maintenance.active ? "生效中" : "未生效" });
|
|
|
|
const allOk = checkResults.every((item) => item.ok);
|
|
const okCount = checkResults.filter((item) => item.ok).length;
|
|
refs.summaryChecks.hidden = false;
|
|
refs.checkSummaryPill.textContent = allOk ? `✓ ${okCount}/${checkResults.length} 检查通过` : `${okCount}/${checkResults.length} 检查`;
|
|
refs.checkSummaryPill.className = `check-summary-pill ${allOk ? "check-ok" : "check-blocked"}`;
|
|
refs.overviewChecks.hidden = false;
|
|
|
|
renderCheckChip(refs.checkConfig, "配置", checks.config?.ok, checks.config?.status);
|
|
renderCheckChip(refs.checkPvc, "PVC", checks.pvc?.ok, checks.pvc?.stateRoot);
|
|
renderCheckChip(refs.checkAnalyzer, "分析器", checks.analyzer?.ok, "observe analyze");
|
|
renderCheckChip(refs.checkPublic, "公开入口", Boolean(overview.publicOrigin), overview.publicOrigin || "-");
|
|
renderCheckChip(refs.checkMaintenance, "维护窗口", maintenance.active !== true, maintenance.active ? "生效中" : "未生效");
|
|
|
|
// B3: 全绿时默认折叠 overview checks 明细
|
|
refs.overviewChecks.classList.toggle("overview-checks-collapsed", !state.checksExpanded);
|
|
}
|
|
|
|
function renderTimeline() {
|
|
const visibleRuns = state.runs.slice(0, responsiveLimit(dashboardLimits.timeline));
|
|
refs.timelineCount.textContent = `最近 ${formatNumber(visibleRuns.length)} / ${formatNumber(state.runs.length)}`;
|
|
if (state.runs.length === 0) {
|
|
refs.timeline.innerHTML = '<div class="empty-state compact">暂无时间线</div>';
|
|
return;
|
|
}
|
|
refs.timeline.innerHTML = visibleRuns.map((run) => {
|
|
const runId = run.runId || run.id || "-";
|
|
const age = run.updatedAt ? formatRelative(run.updatedAt) : "-";
|
|
const title = `${displayStatus(run.status)} · ${run.findingCount ?? 0} 个发现 · ${age}`;
|
|
return `<button class="timeline-node ${statusClass(run.status)}" type="button" data-run-id="${escapeAttr(runId)}" title="${escapeAttr(title)}">
|
|
<span class="timeline-label">${escapeHtml(displayStatus(run.status))}</span>
|
|
<span class="timeline-age">${escapeHtml(age)}</span>
|
|
</button>`;
|
|
}).join("");
|
|
for (const node of refs.timeline.querySelectorAll("[data-run-id]")) {
|
|
node.addEventListener("click", () => selectRun(node.dataset.runId));
|
|
}
|
|
}
|
|
|
|
function renderRuns() {
|
|
const visibleLimit = responsiveLimit(dashboardLimits.runs);
|
|
const visibleRuns = state.runs.slice(0, visibleLimit);
|
|
const hiddenCount = Math.max(0, state.runs.length - visibleRuns.length);
|
|
refs.runsCount.textContent = `${formatNumber(visibleRuns.length)} / ${formatNumber(state.runs.length)} 条`;
|
|
if (state.runs.length === 0) {
|
|
refs.runsBody.innerHTML = '<tr><td class="empty-state" colspan="5">暂无运行</td></tr>';
|
|
return;
|
|
}
|
|
refs.runsBody.innerHTML = visibleRuns.map((run) => {
|
|
const runId = run.runId || run.id || "-";
|
|
const selected = state.selectedRunId === runId ? " selected-row" : "";
|
|
const severityRowClass = run.maxSeverity ? ` ${severityClass(run.maxSeverity)}` : "";
|
|
return `<tr class="${selected}${severityRowClass}" data-run-id="${escapeAttr(runId)}">
|
|
<td data-label="运行"><div class="run-identity"><code>${escapeHtml(shortText(runId, 24))}</code><small>${escapeHtml(run.observerId ? shortText(run.observerId, 28) : "-")}</small></div></td>
|
|
<td data-label="状态"><span class="status-pill ${statusClass(run.status)}">${escapeHtml(displayStatus(run.status))}</span></td>
|
|
<td data-label="场景">${escapeHtml(run.scenarioId || "-")}</td>
|
|
<td data-label="发现">${escapeHtml(String(run.findingCount ?? 0))}${run.maxSeverity ? ` <span class="severity-pill ${severityClass(run.maxSeverity)}">${escapeHtml(displaySeverity(run.maxSeverity))}</span>` : ""}</td>
|
|
<td data-label="更新">${escapeHtml(run.updatedAt ? formatRelative(run.updatedAt) : "-")}${run.maintenance ? `<small>维护窗口</small>` : ""}</td>
|
|
</tr>`;
|
|
}).join("") + (hiddenCount > 0 ? `<tr class="limit-row"><td colspan="5">其余 ${formatNumber(hiddenCount)} 条通过筛选、搜索或 API drill-down 查看</td></tr>` : "");
|
|
for (const row of refs.runsBody.querySelectorAll("tr[data-run-id]")) {
|
|
row.addEventListener("click", () => selectRun(row.dataset.runId));
|
|
}
|
|
}
|
|
|
|
function renderFindings() {
|
|
renderFindingAggregation();
|
|
refs.findingsCount.textContent = `${formatNumber(state.findings.length)} 组`;
|
|
if (state.findingDrilldownKey) {
|
|
renderFindingsDrilldown();
|
|
return;
|
|
}
|
|
refs.findingsList.hidden = false;
|
|
refs.findingsDrilldown.hidden = true;
|
|
if (state.findings.length === 0) {
|
|
refs.findingsList.innerHTML = '<div class="empty-state">暂无发现项</div>';
|
|
return;
|
|
}
|
|
// B8: 默认只显示聚合,list 折叠为 severity group summary(点击展开明细)
|
|
const groups = groupedFindingsBySeverity(state.findings);
|
|
refs.findingsList.innerHTML = groups.map((group) => {
|
|
const visibleItems = group.items.slice(0, findingVisibleLimit(group.key));
|
|
const hiddenCount = Math.max(0, group.items.length - visibleItems.length);
|
|
return `<details class="finding-group finding-group-${escapeAttr(group.key)}">
|
|
<summary>
|
|
<span>${escapeHtml(displaySeverity(group.key))}</span>
|
|
<strong>${formatNumber(group.totalCount)}</strong>
|
|
<small>${formatNumber(group.items.length)} 组${hiddenCount > 0 ? ` · 其余 ${formatNumber(hiddenCount)} 组` : ""}</small>
|
|
</summary>
|
|
<div class="finding-group-list">${visibleItems.map((item) => renderFindingItemCollapsed(item)).join("")}</div>
|
|
</details>`;
|
|
}).join("");
|
|
bindFindingItemEvents();
|
|
}
|
|
|
|
function renderFindingsDrilldown() {
|
|
refs.findingsList.hidden = true;
|
|
refs.findingsDrilldown.hidden = false;
|
|
const [filterKey, filterValue] = state.findingDrilldownKey.split(":", 2);
|
|
const matched = state.findings.filter((item) => {
|
|
if (filterKey === "severity") return String(item.severity || "unknown").toLowerCase() === filterValue;
|
|
if (filterKey === "code") return (item.code || item.findingId || "unknown") === filterValue;
|
|
if (filterKey === "scenario") return (item.scenarioId || "unknown") === filterValue;
|
|
return false;
|
|
});
|
|
const label = drilldownLabel(filterKey, filterValue);
|
|
refs.findingsDrilldown.innerHTML = `<button type="button" class="link-button findings-drilldown-back" id="findings-drilldown-back">← 返回聚合</button>
|
|
<div class="view-note">drill-down: ${escapeHtml(label)} · ${formatNumber(matched.length)} 组</div>
|
|
<div class="finding-group-list">${matched.map((item) => renderFindingItemCollapsed(item)).join("")}</div>`;
|
|
document.getElementById("findings-drilldown-back")?.addEventListener("click", () => {
|
|
state.findingDrilldownKey = "";
|
|
syncLocationQuery();
|
|
renderFindings();
|
|
});
|
|
bindFindingItemEvents(refs.findingsDrilldown);
|
|
}
|
|
|
|
function drilldownLabel(filterKey, filterValue) {
|
|
if (filterKey === "severity") return `严重级别=${displaySeverity(filterValue)}`;
|
|
if (filterKey === "code") return `代码=${displayFindingCode(filterValue)}`;
|
|
if (filterKey === "scenario") return `场景=${filterValue}`;
|
|
return filterValue;
|
|
}
|
|
|
|
function renderFindingItemCollapsed(item) {
|
|
const code = item.code || item.findingId || "finding";
|
|
const latestRunId = item.latestRunId || "-";
|
|
const hasLatestRun = latestRunId !== "-";
|
|
const codeLabel = displayFindingCode(code);
|
|
return `<article class="finding-item is-collapsed" data-finding-run-id="${escapeAttr(latestRunId)}">
|
|
<div class="finding-row" data-finding-toggle>
|
|
<span class="finding-title">${escapeHtml(codeLabel)}${codeLabel === code ? "" : ` <small class="mono">${escapeHtml(code)}</small>`}<small> · ${formatNumber(item.count ?? 0)} 次</small></span>
|
|
<span class="severity-pill ${severityClass(item.severity)}">${escapeHtml(displaySeverity(item.severity))}</span>
|
|
</div>
|
|
<div class="finding-detail">
|
|
<div class="finding-summary">${escapeHtml(shortText(displayFindingSummary(code, item.summary || ""), 220))}</div>
|
|
<div class="finding-actions">
|
|
<button type="button" class="filter-chip" data-finding-filter="code" data-filter-value="${escapeAttr(code)}">${escapeHtml(code)}</button>
|
|
<button type="button" class="filter-chip" data-finding-filter="scenario" data-filter-value="${escapeAttr(item.scenarioId || "")}">${escapeHtml(item.scenarioId || "-")}</button>
|
|
</div>
|
|
<div class="finding-meta">次数=${escapeHtml(String(item.count ?? 0))} · 运行=${escapeHtml(String(item.runCount ?? 0))} · 最近=${escapeHtml(item.latestAt ? formatRelative(item.latestAt) : "-")}</div>
|
|
<div class="finding-meta mono">run=${escapeHtml(latestRunId)} report=${escapeHtml(item.latestReportJsonSha256 || "-")}</div>
|
|
<div class="finding-meta">${escapeHtml(findingNextAction(code))}</div>
|
|
<button type="button" class="link-button" data-open-finding-run="${escapeAttr(latestRunId)}"${hasLatestRun ? "" : " disabled"}>打开最近运行</button>
|
|
</div>
|
|
</article>`;
|
|
}
|
|
|
|
function bindFindingItemEvents(scope) {
|
|
const container = scope || refs.findingsList;
|
|
for (const row of container.querySelectorAll("[data-finding-toggle]")) {
|
|
row.addEventListener("click", (event) => {
|
|
if (event.target.closest("[data-open-finding-run]") || event.target.closest("[data-finding-filter]")) return;
|
|
const item = row.closest(".finding-item");
|
|
if (item) item.classList.toggle("is-collapsed");
|
|
});
|
|
}
|
|
for (const button of container.querySelectorAll("[data-open-finding-run]")) {
|
|
button.addEventListener("click", () => selectRun(button.dataset.openFindingRun));
|
|
}
|
|
for (const button of container.querySelectorAll("[data-finding-filter]")) {
|
|
button.addEventListener("click", () => applyFindingFilter(button.dataset.findingFilter, button.dataset.filterValue || ""));
|
|
}
|
|
}
|
|
|
|
function responsiveLimit(limits) {
|
|
if (window.matchMedia("(max-width: 560px)").matches) return limits.mobile;
|
|
if (window.matchMedia("(max-width: 980px)").matches) return limits.tablet;
|
|
return limits.desktop;
|
|
}
|
|
|
|
function findingVisibleLimit(severity) {
|
|
const key = String(severity || "").toLowerCase();
|
|
return responsiveLimit(key === "red" || key === "critical" || key === "error" ? dashboardLimits.findingsRed : dashboardLimits.findingsOther);
|
|
}
|
|
|
|
function groupedFindingsBySeverity(findings) {
|
|
const severityOrder = ["red", "critical", "error", "warning", "amber", "info", "unknown"];
|
|
const groups = new Map();
|
|
for (const item of findings) {
|
|
const key = String(item.severity || "unknown").toLowerCase();
|
|
if (!groups.has(key)) groups.set(key, []);
|
|
groups.get(key).push(item);
|
|
}
|
|
return Array.from(groups.entries())
|
|
.map(([key, items]) => ({
|
|
key,
|
|
items: items.sort((a, b) => Number(b.count || 0) - Number(a.count || 0)),
|
|
totalCount: items.reduce((sum, item) => sum + Number(item.count || 0), 0),
|
|
}))
|
|
.sort((a, b) => severityRank(a.key, severityOrder) - severityRank(b.key, severityOrder));
|
|
}
|
|
|
|
function severityRank(key, order) {
|
|
const index = order.indexOf(key);
|
|
return index >= 0 ? index : order.length;
|
|
}
|
|
|
|
function renderFindingAggregation() {
|
|
if (state.findings.length === 0) {
|
|
refs.findingAggregation.innerHTML = '<div class="empty-state compact">暂无发现聚合</div>';
|
|
return;
|
|
}
|
|
const severityEntries = aggregateFindings((item) => item.severity || "unknown").slice(0, 5);
|
|
const codeEntries = aggregateFindings((item) => item.code || item.findingId || "unknown").slice(0, 5);
|
|
const scenarioEntries = aggregateFindings((item) => item.scenarioId || "unknown").slice(0, 5);
|
|
refs.findingAggregation.innerHTML = [
|
|
aggregationGroup("严重级别", severityEntries.map((item) => ({ ...item, label: displaySeverity(item.key) })), "severity"),
|
|
aggregationGroup("代码", codeEntries.map((item) => ({ ...item, label: displayFindingCode(item.key) })), "code"),
|
|
aggregationGroup("场景", scenarioEntries, "scenario"),
|
|
`<div class="aggregation-group"><span>窗口</span><button type="button" class="filter-chip active">${escapeHtml(state.findingFilters.window || "全部")}</button></div>`,
|
|
].join("");
|
|
for (const button of refs.findingAggregation.querySelectorAll("[data-finding-filter]")) {
|
|
button.addEventListener("click", () => openFindingsDrilldown(button.dataset.findingFilter, button.dataset.filterValue || ""));
|
|
}
|
|
}
|
|
|
|
function aggregationGroup(label, entries, filterKey) {
|
|
const chips = entries.length === 0
|
|
? '<span class="finding-meta">-</span>'
|
|
: entries.map((entry) => `<button type="button" class="filter-chip" data-finding-filter="${escapeAttr(filterKey)}" data-filter-value="${escapeAttr(entry.key)}">${escapeHtml(entry.label || entry.key)} ${formatNumber(entry.count)}</button>`).join("");
|
|
return `<div class="aggregation-group"><span>${escapeHtml(label)}</span>${chips}</div>`;
|
|
}
|
|
|
|
function aggregateFindings(keyFn) {
|
|
const counts = new Map();
|
|
for (const item of state.findings) {
|
|
const key = keyFn(item);
|
|
counts.set(key, (counts.get(key) || 0) + Number(item.count || 0));
|
|
}
|
|
return Array.from(counts.entries())
|
|
.map(([key, count]) => ({ key, count }))
|
|
.sort((a, b) => b.count - a.count || a.key.localeCompare(b.key));
|
|
}
|
|
|
|
function openFindingsDrilldown(filterKey, filterValue) {
|
|
if (!filterValue || filterValue === "-") return;
|
|
state.findingDrilldownKey = `${filterKey}:${filterValue}`;
|
|
syncLocationQuery();
|
|
renderFindings();
|
|
}
|
|
|
|
function applyFindingFilter(key, value) {
|
|
if (!value || value === "-") return;
|
|
if (key === "severity") state.findingFilters.severity = value;
|
|
if (key === "code") state.findingFilters.code = value;
|
|
if (key === "scenario") state.findingFilters.scenario = value;
|
|
writeFindingFiltersToControls();
|
|
updateFindingsFilterSummary();
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
}
|
|
|
|
function selectTab(tab) {
|
|
if (!DETAIL_TABS.includes(tab)) return;
|
|
state.selectedTab = tab;
|
|
for (const button of refs.detailTabs.querySelectorAll("[data-detail-tab]")) {
|
|
button.classList.toggle("active", button.dataset.detailTab === tab);
|
|
}
|
|
syncLocationQuery();
|
|
renderDetail();
|
|
}
|
|
|
|
function renderDetail() {
|
|
if (!state.selectedRunId || !state.runDetail) {
|
|
refs.detailSubtitle.textContent = "未选择运行";
|
|
refs.detailTabs.hidden = true;
|
|
refs.detailContent.innerHTML = '<div class="empty-state">请选择一条运行</div>';
|
|
return;
|
|
}
|
|
const tab = state.selectedTab || "overview";
|
|
for (const button of refs.detailTabs.querySelectorAll("[data-detail-tab]")) {
|
|
button.classList.toggle("active", button.dataset.detailTab === tab);
|
|
}
|
|
const detail = state.runDetail;
|
|
const run = detail.run || {};
|
|
const findings = Array.isArray(detail.findings) ? detail.findings : [];
|
|
const artifacts = detail.artifacts || {};
|
|
const commands = detail.commands || {};
|
|
const turnSummaryView = selectedView(state.runViews, "turn-summary");
|
|
refs.detailSubtitle.textContent = run.runId || state.selectedRunId;
|
|
refs.detailContent.innerHTML = renderDetailTab(tab, detail, run, findings, artifacts, commands, turnSummaryView);
|
|
bindDetailControls();
|
|
}
|
|
|
|
function renderDetailTab(tab, detail, run, findings, artifacts, commands, turnSummaryView) {
|
|
if (tab === "findings") {
|
|
return detailFindings(findings);
|
|
}
|
|
if (tab === "turn") {
|
|
return detailTurnSummary(turnSummaryView);
|
|
}
|
|
if (tab === "trace") {
|
|
return detailTraceReader(state.runViews, commands);
|
|
}
|
|
if (tab === "evidence") {
|
|
return [
|
|
detailArtifacts(artifacts),
|
|
detailCommands(commands),
|
|
detailEvidence(detail, artifacts),
|
|
detailBlock("脱敏", [
|
|
["values", detail.valuesRedacted === true ? "已脱敏" : "-"],
|
|
["prompt", detail.redaction?.prompt || "-"],
|
|
["assistant", detail.redaction?.assistantFinal || "-"],
|
|
]),
|
|
].join("");
|
|
}
|
|
// overview
|
|
return [
|
|
detailBlock("追溯信息", [
|
|
["run", run.runId || "-"],
|
|
["observer", run.observerId || "-"],
|
|
["stateDir", run.stateDir || "-"],
|
|
["report", run.reportJsonSha256 || "-"],
|
|
]),
|
|
detailBlock("摘要", [
|
|
["status", displayStatus(run.status)],
|
|
["scenario", run.scenarioId || "-"],
|
|
["findings", String(run.findingCount ?? 0)],
|
|
["artifacts", String(run.artifactCount ?? 0)],
|
|
["updated", run.updatedAt || "-"],
|
|
["views", Array.isArray(detail.viewsAvailable) ? detail.viewsAvailable.join(", ") : "-"],
|
|
]),
|
|
detailBlock("报告摘要", safeSummaryRows(detail.summary)),
|
|
].join("");
|
|
}
|
|
|
|
function detailBlock(title, rows, extraClass) {
|
|
const body = rows.map(([key, value]) => `<div><span>${escapeHtml(key)}</span><code>${escapeHtml(String(value ?? "-"))}</code></div>`).join("");
|
|
return `<article class="detail-block${extraClass ? ` ${extraClass}` : ""}"><strong>${escapeHtml(title)}</strong><div class="detail-grid">${body}</div></article>`;
|
|
}
|
|
|
|
function detailFindings(findings) {
|
|
if (!Array.isArray(findings) || findings.length === 0) {
|
|
return '<article class="detail-block"><strong>运行发现项</strong><div class="empty-state compact">暂无发现项</div></article>';
|
|
}
|
|
return `<article class="detail-block"><strong>运行发现项 · ${formatNumber(findings.length)} 条</strong>
|
|
<div class="detail-table-frame"><table class="detail-table"><thead><tr>
|
|
<th>严重级别</th><th>代码</th><th>次数</th><th>摘要</th><th>报告</th>
|
|
</tr></thead><tbody>${findings.map((item) => `<tr>
|
|
<td><span class="severity-pill ${severityClass(item.severity)}">${escapeHtml(displaySeverity(item.severity))}</span></td>
|
|
<td class="mono">${escapeHtml(item.finding_id || item.findingId || "-")}</td>
|
|
<td>${escapeHtml(String(item.count ?? 0))}</td>
|
|
<td>${escapeHtml(shortText(displayFindingSummary(item.finding_id || item.findingId || "", item.summary || ""), 220))}</td>
|
|
<td class="mono">${escapeHtml(shortText(item.report_json_sha256 || item.reportJsonSha256 || "-", 24))}</td>
|
|
</tr>`).join("")}</tbody></table></div>
|
|
</article>`;
|
|
}
|
|
|
|
function detailArtifacts(artifacts) {
|
|
const screenshot = artifacts.screenshot || {};
|
|
return detailBlock("产物", [
|
|
["artifactCount", String(artifacts.artifactCount ?? "-")],
|
|
["reportJsonSha256", artifacts.reportJsonSha256 || "-"],
|
|
["screenshotPath", screenshot.path || "-"],
|
|
["screenshotSha256", screenshot.sha256 || "-"],
|
|
["publicOrigin", artifacts.publicOrigin || "-"],
|
|
]);
|
|
}
|
|
|
|
function detailCommands(commands) {
|
|
const rows = [
|
|
["summary", commands.summary || "-"],
|
|
["turn-summary", commands.turnSummary || "-"],
|
|
["trace-frame", commands.traceFrame || "-"],
|
|
];
|
|
// D4: CLI 命令改可复制按钮
|
|
return `<article class="detail-block"><strong>CLI 对照命令</strong>
|
|
<div class="command-list">${rows.map(([label, command]) => `<div class="command-row">
|
|
<span class="command-label">${escapeHtml(label)}</span>
|
|
<code class="command-code">${escapeHtml(command)}</code>
|
|
<button type="button" class="copy-button" data-copy-command="${escapeAttr(command)}">复制</button>
|
|
</div>`).join("")}</div>
|
|
</article>`;
|
|
}
|
|
|
|
function detailTurnSummary(view) {
|
|
if (!view) return detailBlock("多轮摘要 - 第一层", [["status", "未索引"]]);
|
|
if (view.ok === false) return detailBlock("多轮摘要 - 第一层", [["status", view.error || "不可用"]]);
|
|
const text = redactDisplayText(view.renderedText || "");
|
|
const note = `${formatNumber(view.renderedTextBytes || text.length)} bytes${view.truncated ? " 已截断" : ""}`;
|
|
return `<article class="detail-block"><strong>多轮摘要 - 第一层</strong>
|
|
<div class="view-note">${escapeHtml(note)}</div>
|
|
<pre class="detail-pre">${escapeHtml(text || "-")}</pre>
|
|
</article>`;
|
|
}
|
|
|
|
function detailTraceReader(response, commands) {
|
|
const turnView = selectedView(response, "turn-summary");
|
|
const traceView = selectedView(response, "trace-frame");
|
|
const choices = traceChoices(turnView?.renderedText || "", traceView?.renderedText || "");
|
|
const selectedIndex = Math.min(state.selectedTraceChoiceIndex, Math.max(0, choices.length - 1));
|
|
const selected = choices[selectedIndex] || { label: "已保存 trace-frame", meta: "当前运行", key: "stored" };
|
|
const traceText = traceView?.ok === false ? "" : redactDisplayText(traceView?.renderedText || "");
|
|
const traceNote = traceView
|
|
? `${formatNumber(traceView.renderedTextBytes || traceText.length)} bytes${traceView.truncated ? " 已截断" : ""}`
|
|
: "trace-frame 未索引";
|
|
return `<article class="detail-block trace-reader"><strong>Trace Frame - 第二层</strong>
|
|
<div class="trace-reader-grid">
|
|
<section class="trace-turn-picker" aria-label="turn、trace、sample 选择">
|
|
<div class="view-note">选择 turn / trace / sample</div>
|
|
<div class="trace-choice-list">${choices.map((choice, index) => `<button type="button" class="trace-choice ${index === selectedIndex ? "active" : ""}" data-trace-choice-index="${escapeAttr(String(index))}">
|
|
<span>${escapeHtml(choice.label)}</span>
|
|
<small>${escapeHtml(choice.meta)}</small>
|
|
</button>`).join("")}</div>
|
|
</section>
|
|
<section class="trace-frame-view" aria-label="trace-frame 文字视图">
|
|
<div class="view-note">已选择: ${escapeHtml(selected.label)} · ${escapeHtml(traceNote)}</div>
|
|
<pre class="detail-pre trace-frame-pre">${escapeHtml(traceText || (traceView?.ok === false ? traceView.error || "trace-frame 不可用" : "trace-frame 未索引"))}</pre>
|
|
${finalResponseBlock(traceView)}
|
|
<div class="trace-source-note mono">source=${escapeHtml(commands.traceFrame || "-")} · analyzer finding 不改写此文字证据</div>
|
|
</section>
|
|
</div>
|
|
</article>`;
|
|
}
|
|
|
|
function finalResponseBlock(traceView) {
|
|
if (!traceView) {
|
|
return `<section class="final-response-block unavailable"><strong>Final Response</strong><div>trace-frame 未索引</div></section>`;
|
|
}
|
|
if (traceView.ok === false) {
|
|
return `<section class="final-response-block unavailable"><strong>Final Response</strong><div>${escapeHtml(traceView.error || "trace-frame 不可用")}</div></section>`;
|
|
}
|
|
const block = traceView.finalResponse || {};
|
|
const text = block.empty === true ? "(空内容)" : redactDisplayText(block.text || "");
|
|
const bytes = Number(block.byteCount || 0);
|
|
return `<section class="final-response-block ${block.empty === true ? "empty" : ""}">
|
|
<strong>Final Response</strong>
|
|
<div class="view-note">${block.empty === true ? "空内容" : "有内容"} · ${formatNumber(bytes)} bytes · 已脱敏</div>
|
|
<pre class="final-response-text">${escapeHtml(text || "(空内容)")}</pre>
|
|
</section>`;
|
|
}
|
|
|
|
function traceChoices(turnSummaryText, traceFrameText) {
|
|
const sourceLines = String(turnSummaryText || "")
|
|
.split("\n")
|
|
.map((line) => line.trim())
|
|
.filter((line) => /(turn|trace|sample|final response|轮次|用户消息)/iu.test(line))
|
|
.slice(0, 12);
|
|
const lines = sourceLines.length > 0 ? sourceLines : String(traceFrameText || "")
|
|
.split("\n")
|
|
.map((line) => line.trim())
|
|
.filter((line) => /(trace|sample|total=|final response)/iu.test(line))
|
|
.slice(0, 12);
|
|
if (lines.length === 0) return [{ label: "已保存 trace-frame", meta: "当前运行", key: "stored" }];
|
|
return lines.map((line, index) => {
|
|
const trace = line.match(/trace(?:Id)?[=: ]+([A-Za-z0-9_.:-]+)/u)?.[1] || line.match(/\btrc_[A-Za-z0-9_.:-]+/u)?.[0] || null;
|
|
const sample = line.match(/sample(?:Seq)?[=: ]+([0-9]+)/u)?.[1] || null;
|
|
const turn = line.match(/turn[=: #]+([0-9A-Za-z_.:-]+)/iu)?.[1] || null;
|
|
const meta = [turn ? `turn ${turn}` : null, trace ? `trace ${trace}` : null, sample ? `sample ${sample}` : null].filter(Boolean).join(" · ") || `第 ${index + 1} 行`;
|
|
return { label: shortText(redactDisplayText(line), 120), meta, key: `${trace || "line"}-${sample || index}` };
|
|
});
|
|
}
|
|
|
|
function detailEvidence(detail, artifacts) {
|
|
const traceability = detail.traceability || {};
|
|
return detailBlock("证据", [
|
|
["source", traceability.source || "-"],
|
|
["stateRoot", traceability.stateRoot || "-"],
|
|
["stateDir", traceability.stateDir || "-"],
|
|
["observerId", traceability.observerId || "-"],
|
|
["runId", traceability.runId || "-"],
|
|
["reportJsonSha256", traceability.reportJsonSha256 || artifacts.reportJsonSha256 || "-"],
|
|
]);
|
|
}
|
|
|
|
function selectedView(response, viewName) {
|
|
if (response?.ok === false) return { ok: false, error: response.error || "不可用", view: viewName };
|
|
const views = Array.isArray(response?.views) ? response.views : [];
|
|
return views.find((item) => item.view === viewName) || null;
|
|
}
|
|
|
|
function bindDetailControls() {
|
|
for (const button of refs.detailContent.querySelectorAll("[data-trace-choice-index]")) {
|
|
button.addEventListener("click", () => {
|
|
state.selectedTraceChoiceIndex = Number(button.dataset.traceChoiceIndex || 0);
|
|
renderDetail();
|
|
});
|
|
}
|
|
for (const button of refs.detailContent.querySelectorAll("[data-copy-command]")) {
|
|
button.addEventListener("click", () => copyCommand(button));
|
|
}
|
|
}
|
|
|
|
async function copyCommand(button) {
|
|
const command = button.dataset.copyCommand || "";
|
|
try {
|
|
await navigator.clipboard.writeText(command);
|
|
} catch {
|
|
// 降级:用临时 textarea
|
|
const textarea = document.createElement("textarea");
|
|
textarea.value = command;
|
|
textarea.style.position = "fixed";
|
|
textarea.style.opacity = "0";
|
|
document.body.appendChild(textarea);
|
|
textarea.select();
|
|
try { document.execCommand("copy"); } catch { /* ignore */ }
|
|
document.body.removeChild(textarea);
|
|
}
|
|
button.classList.add("copied");
|
|
button.textContent = "已复制";
|
|
showCopyToast();
|
|
window.setTimeout(() => {
|
|
button.classList.remove("copied");
|
|
button.textContent = "复制";
|
|
}, 1500);
|
|
}
|
|
|
|
function showCopyToast() {
|
|
refs.copyToast.hidden = false;
|
|
window.setTimeout(() => { refs.copyToast.hidden = true; }, 1200);
|
|
}
|
|
|
|
function renderLoading(show) {
|
|
refs.loadingBanner.hidden = !show;
|
|
refs.manualRefresh.disabled = state.loading;
|
|
}
|
|
|
|
function renderCheckChip(element, label, ok, detail) {
|
|
const status = ok ? "正常" : "阻塞";
|
|
element.textContent = `${label} ${status}${detail ? ` · ${shortText(detail, 34)}` : ""}`;
|
|
element.className = `check-chip ${ok ? "check-ok" : "check-blocked"}`;
|
|
}
|
|
|
|
function toggleChecks() {
|
|
state.checksExpanded = !state.checksExpanded;
|
|
refs.overviewChecks.classList.toggle("overview-checks-collapsed", !state.checksExpanded);
|
|
}
|
|
|
|
function toggleTimeline() {
|
|
state.timelineCollapsed = !state.timelineCollapsed;
|
|
const panel = refs.timeline.closest(".timeline-panel");
|
|
panel.classList.toggle("collapsed", state.timelineCollapsed);
|
|
refs.timelineToggle.textContent = state.timelineCollapsed ? "展开" : "折叠";
|
|
refs.timelineToggle.setAttribute("aria-expanded", String(!state.timelineCollapsed));
|
|
}
|
|
|
|
function toggleRunsFilter() {
|
|
state.runsFilterOpen = !state.runsFilterOpen;
|
|
refs.filterForm.hidden = !state.runsFilterOpen;
|
|
refs.runsFilterSummary.setAttribute("aria-expanded", String(state.runsFilterOpen));
|
|
updateRunsFilterSummary();
|
|
}
|
|
|
|
function toggleFindingsFilter() {
|
|
state.findingsFilterOpen = !state.findingsFilterOpen;
|
|
refs.findingsFilterForm.hidden = !state.findingsFilterOpen;
|
|
refs.findingsFilterSummary.setAttribute("aria-expanded", String(state.findingsFilterOpen));
|
|
updateFindingsFilterSummary();
|
|
}
|
|
|
|
function updateRunsFilterSummary() {
|
|
const f = state.filters;
|
|
const parts = [];
|
|
if (f.status) parts.push(`状态=${displayStatus(f.status)}`);
|
|
if (f.severity) parts.push(`严重=${displaySeverity(f.severity)}`);
|
|
if (f.window) parts.push(`时间=${f.window}`);
|
|
if (f.search) parts.push(`搜索="${shortText(f.search, 16)}"`);
|
|
if (f.sort && f.sort !== "updated") parts.push(`排序=${f.sort}`);
|
|
refs.runsFilterSummary.textContent = `筛选: ${parts.length ? parts.join(" · ") : "全部"}`;
|
|
}
|
|
|
|
function updateFindingsFilterSummary() {
|
|
const f = state.findingFilters;
|
|
const parts = [];
|
|
if (f.severity) parts.push(`严重=${displaySeverity(f.severity)}`);
|
|
parts.push(`窗口=${f.window || "全部"}`);
|
|
if (f.code) parts.push(`代码="${shortText(f.code, 16)}"`);
|
|
if (f.scenario) parts.push(`场景="${shortText(f.scenario, 16)}"`);
|
|
refs.findingsFilterSummary.textContent = `筛选: ${parts.join(" · ")}`;
|
|
}
|
|
|
|
function jumpToLatestRun() {
|
|
const latest = state.runs[0];
|
|
if (latest) selectRun(latest.runId || latest.id);
|
|
}
|
|
|
|
function toggleRedOnly() {
|
|
const active = refs.filterRed.classList.toggle("active");
|
|
if (active) {
|
|
state.filters.severity = "red";
|
|
} else {
|
|
state.filters.severity = "";
|
|
}
|
|
refs.filterSeverity.value = state.filters.severity;
|
|
updateRunsFilterSummary();
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
}
|
|
|
|
function handleKeyboardShortcut(event) {
|
|
if (event.target instanceof HTMLInputElement || event.target instanceof HTMLSelectElement || event.target instanceof HTMLTextAreaElement) return;
|
|
if (event.metaKey || event.ctrlKey || event.altKey) return;
|
|
const key = event.key.toLowerCase();
|
|
if (key === "r") { event.preventDefault(); loadDashboard({ silent: false }); return; }
|
|
if (key === "/") { event.preventDefault(); state.runsFilterOpen = true; refs.filterForm.hidden = false; refs.runsFilterSummary.setAttribute("aria-expanded", "true"); refs.filterSearch.focus(); return; }
|
|
if (key === "j" || key === "k") { event.preventDefault(); navigateRun(key === "j" ? 1 : -1); return; }
|
|
if (key === "enter") {
|
|
event.preventDefault();
|
|
if (state.selectedRunId) { selectTab("overview"); return; }
|
|
const first = state.runs[0];
|
|
if (first) selectRun(first.runId || first.id);
|
|
return;
|
|
}
|
|
}
|
|
|
|
function navigateRun(direction) {
|
|
const ids = state.runs.map((run) => run.runId || run.id).filter(Boolean);
|
|
if (ids.length === 0) return;
|
|
const currentIndex = state.selectedRunId ? ids.indexOf(state.selectedRunId) : -1;
|
|
let nextIndex = currentIndex + direction;
|
|
if (nextIndex < 0) nextIndex = 0;
|
|
if (nextIndex >= ids.length) nextIndex = ids.length - 1;
|
|
const nextId = ids[nextIndex];
|
|
if (nextId && nextId !== state.selectedRunId) selectRun(nextId);
|
|
}
|
|
|
|
function updateViewportClass() {
|
|
const width = window.innerWidth;
|
|
const viewport = width >= 1024 ? "desktop" : width >= 981 ? "tablet" : "mobile";
|
|
root.dataset.viewport = viewport;
|
|
if (state.overview) {
|
|
renderTimeline();
|
|
renderRuns();
|
|
renderFindings();
|
|
}
|
|
}
|
|
|
|
function applyFilterControls() {
|
|
state.filters = {
|
|
status: refs.filterStatus.value,
|
|
severity: refs.filterSeverity.value,
|
|
window: refs.filterWindow.value,
|
|
search: refs.filterSearch.value.trim(),
|
|
sort: refs.filterSort.value || "updated",
|
|
};
|
|
updateRunsFilterSummary();
|
|
refs.filterRed.classList.toggle("active", state.filters.severity === "red");
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
}
|
|
|
|
function applyFindingFilterControls() {
|
|
state.findingFilters = {
|
|
severity: refs.findingFilterSeverity.value,
|
|
window: refs.findingFilterWindow.value,
|
|
code: refs.findingFilterCode.value.trim(),
|
|
scenario: refs.findingFilterScenario.value.trim(),
|
|
};
|
|
updateFindingsFilterSummary();
|
|
syncLocationQuery();
|
|
loadDashboard({ silent: false });
|
|
}
|
|
|
|
function writeFiltersToControls() {
|
|
refs.filterStatus.value = state.filters.status || "";
|
|
refs.filterSeverity.value = state.filters.severity || "";
|
|
refs.filterWindow.value = state.filters.window || "";
|
|
refs.filterSearch.value = state.filters.search || "";
|
|
refs.filterSort.value = state.filters.sort || "updated";
|
|
refs.filterRed.classList.toggle("active", state.filters.severity === "red");
|
|
}
|
|
|
|
function writeFindingFiltersToControls() {
|
|
refs.findingFilterSeverity.value = state.findingFilters.severity || "";
|
|
refs.findingFilterWindow.value = state.findingFilters.window || "24h";
|
|
refs.findingFilterCode.value = state.findingFilters.code || "";
|
|
refs.findingFilterScenario.value = state.findingFilters.scenario || "";
|
|
}
|
|
|
|
function readFiltersFromLocation() {
|
|
const query = new URLSearchParams(window.location.search);
|
|
return {
|
|
status: query.get("status") || "",
|
|
severity: query.get("severity") || "",
|
|
window: query.get("window") || "",
|
|
search: query.get("search") || "",
|
|
sort: query.get("sort") || "updated",
|
|
};
|
|
}
|
|
|
|
function readFindingFiltersFromLocation() {
|
|
const query = new URLSearchParams(window.location.search);
|
|
return {
|
|
severity: query.get("fseverity") || "",
|
|
window: query.get("fwindow") || "24h",
|
|
code: query.get("fcode") || "",
|
|
scenario: query.get("fscenario") || "",
|
|
};
|
|
}
|
|
|
|
function readTabFromLocation() {
|
|
const tab = new URLSearchParams(window.location.search).get("tab");
|
|
return DETAIL_TABS.includes(tab) ? tab : "overview";
|
|
}
|
|
|
|
function syncLocationQuery() {
|
|
const query = new URLSearchParams();
|
|
for (const [key, value] of Object.entries(state.filters)) {
|
|
if (value && !(key === "sort" && value === "updated")) query.set(key, value);
|
|
}
|
|
const findingQueryKeys = {
|
|
severity: "fseverity",
|
|
window: "fwindow",
|
|
code: "fcode",
|
|
scenario: "fscenario",
|
|
};
|
|
for (const [key, value] of Object.entries(state.findingFilters)) {
|
|
if (value && !(key === "window" && value === "24h")) query.set(findingQueryKeys[key], value);
|
|
}
|
|
if (state.selectedRunId) query.set("run", state.selectedRunId);
|
|
// C3: URL 支持 tab deep-link
|
|
if (state.selectedRunId && state.selectedTab && state.selectedTab !== "overview") query.set("tab", state.selectedTab);
|
|
if (state.findingDrilldownKey) query.set("finding", state.findingDrilldownKey);
|
|
const next = `${window.location.pathname}${query.toString() ? `?${query.toString()}` : ""}`;
|
|
window.history.replaceState(null, "", next);
|
|
}
|
|
|
|
function runsQuery(filters) {
|
|
const query = new URLSearchParams({ limit: "40", sort: filters.sort || "updated", direction: "desc" });
|
|
if (filters.status) query.set("status", filters.status);
|
|
if (filters.severity) query.set("severity", filters.severity);
|
|
if (filters.search) query.set("search", filters.search);
|
|
const from = windowFrom(filters.window);
|
|
if (from) query.set("from", from);
|
|
return query.toString();
|
|
}
|
|
|
|
function findingsQuery(filters) {
|
|
const query = new URLSearchParams({ limit: "30" });
|
|
if (filters.severity) query.set("severity", filters.severity);
|
|
if (filters.window) query.set("window", filters.window);
|
|
if (filters.code) query.set("code", filters.code);
|
|
if (filters.scenario) query.set("scenario", filters.scenario);
|
|
return query.toString();
|
|
}
|
|
|
|
function windowFrom(value) {
|
|
if (!value) return "";
|
|
const match = /^(\d+)(h|d)$/u.exec(value);
|
|
if (!match) return "";
|
|
const amount = Number(match[1]);
|
|
const ms = match[2] === "h" ? amount * 3_600_000 : amount * 86_400_000;
|
|
return new Date(Date.now() - ms).toISOString();
|
|
}
|
|
|
|
function debounce(fn, waitMs) {
|
|
let timer = null;
|
|
return (...args) => {
|
|
if (timer !== null) window.clearTimeout(timer);
|
|
timer = window.setTimeout(() => fn(...args), waitMs);
|
|
};
|
|
}
|
|
|
|
function safeSummaryRows(summary) {
|
|
const source = summary && typeof summary === "object" ? summary : {};
|
|
const rows = [];
|
|
for (const [key, value] of Object.entries(source)) {
|
|
if (!safeDisplayKey(key)) continue;
|
|
if (rows.length >= 8) break;
|
|
rows.push([key, safeDisplayValue(value)]);
|
|
}
|
|
return rows.length === 0 ? [["status", "未索引"]] : rows;
|
|
}
|
|
|
|
function safeDisplayKey(key) {
|
|
return !/(prompt|provider|cookie|token|authorization|payload|stdout|stderr|assistant|secret|password)/iu.test(key);
|
|
}
|
|
|
|
function safeDisplayValue(value) {
|
|
if (value === null || typeof value === "undefined") return "-";
|
|
if (typeof value === "string" || typeof value === "number" || typeof value === "boolean") return String(value);
|
|
if (Array.isArray(value)) return `${value.length} 项`;
|
|
const entries = Object.entries(value).filter(([key]) => safeDisplayKey(key)).slice(0, 4);
|
|
if (entries.length === 0) return "已脱敏对象";
|
|
return entries.map(([key, item]) => `${key}=${primitiveValue(item)}`).join(", ");
|
|
}
|
|
|
|
function primitiveValue(value) {
|
|
if (value === null || typeof value === "undefined") return "-";
|
|
if (typeof value === "string" || typeof value === "number" || typeof value === "boolean") return shortText(String(value), 80);
|
|
if (Array.isArray(value)) return `${value.length} 项`;
|
|
return "对象";
|
|
}
|
|
|
|
function redactDisplayText(value) {
|
|
return String(value || "")
|
|
.split("\n")
|
|
.map((line) => {
|
|
if (/(prompt|cookie|token|authorization|provider|payload|api[_-]?key|password|secret|stdout|stderr)/iu.test(line)) return "[已脱敏]";
|
|
if (/assistant/iu.test(line) && line.length > 180) return "[assistant 正文已脱敏]";
|
|
return shortText(line, 360);
|
|
})
|
|
.join("\n");
|
|
}
|
|
|
|
function renderError(error) {
|
|
refs.errorBanner.hidden = false;
|
|
refs.errorBanner.textContent = error?.message || String(error);
|
|
}
|
|
|
|
function statusClass(value) {
|
|
const normalized = String(value || "idle").toLowerCase();
|
|
if (["healthy", "observed", "analyzed", "pass", "ok"].includes(normalized)) return "status-healthy";
|
|
if (["warning", "degraded", "planned", "running", "queued"].includes(normalized)) return "status-warning";
|
|
if (["blocked", "failed", "error", "timeout", "interrupted"].includes(normalized)) return "status-blocked";
|
|
return "status-idle";
|
|
}
|
|
|
|
function severityClass(value) {
|
|
const normalized = String(value || "unknown").toLowerCase();
|
|
if (["red", "critical", "error"].includes(normalized)) return `severity-${normalized}`;
|
|
if (["warning", "amber"].includes(normalized)) return `severity-${normalized}`;
|
|
if (["info"].includes(normalized)) return "severity-info";
|
|
return "severity-unknown";
|
|
}
|
|
|
|
function displayStatus(value) {
|
|
const normalized = String(value || "idle").toLowerCase();
|
|
const labels = {
|
|
healthy: "健康",
|
|
observed: "已观察",
|
|
analyzed: "已分析",
|
|
pass: "通过",
|
|
ok: "正常",
|
|
warning: "警告",
|
|
degraded: "降级",
|
|
planned: "已计划",
|
|
running: "运行中",
|
|
queued: "排队中",
|
|
blocked: "阻塞",
|
|
failed: "失败",
|
|
error: "错误",
|
|
timeout: "超时",
|
|
interrupted: "已中断",
|
|
idle: "空闲",
|
|
};
|
|
return labels[normalized] || String(value || "-");
|
|
}
|
|
|
|
function displaySeverity(value) {
|
|
const normalized = String(value || "unknown").toLowerCase();
|
|
const labels = {
|
|
red: "红色",
|
|
critical: "严重",
|
|
error: "错误",
|
|
warning: "警告",
|
|
amber: "警告",
|
|
info: "信息",
|
|
unknown: "未知",
|
|
};
|
|
return labels[normalized] || String(value || "-");
|
|
}
|
|
|
|
function displayFindingCode(code) {
|
|
const normalized = String(code || "").toLowerCase();
|
|
const labels = {
|
|
"quick-verify-no-business-turn": "quick verify 未触达业务 turn",
|
|
"observer-command-failed": "观察器控制命令失败",
|
|
"runtime-requestfailed": "运行时请求失败",
|
|
"runtime-console-alerts": "运行时控制台告警",
|
|
"browser-console-or-page-errors": "浏览器控制台或页面错误",
|
|
"browser-timeout": "页面 readiness 超时",
|
|
"target-page-readiness-timeout": "页面 readiness 超时",
|
|
"network-or-api-fetch-bug": "网络或 API 请求异常",
|
|
};
|
|
return labels[normalized] || String(code || "finding");
|
|
}
|
|
|
|
function displayFindingSummary(code, summary) {
|
|
const normalized = String(code || "").toLowerCase();
|
|
const summaries = {
|
|
"quick-verify-no-business-turn": "quick verify 没有形成 sendPrompt、session、trace rows 或 Final Response;不能把公开 dashboard 200 当作 HWLAB 恢复证据。",
|
|
"observer-command-failed": "observe control command 失败,需要查看 observer timeline 和 failed command 文件确认是 readiness、session API、超时还是 runner shutdown。",
|
|
"runtime-requestfailed": "页面运行时存在请求失败,需要按路径聚合确认是 asset/provenance 噪声、public origin、auth/session 还是 Workbench API。",
|
|
"runtime-console-alerts": "页面控制台出现告警,需要结合 run detail 的 network/console 证据判断是否影响业务 turn。",
|
|
"browser-console-or-page-errors": "浏览器页面报错,需要先看 finalUrl、readiness、session create 和 API 证据,再判断是否是前端缺陷。",
|
|
"browser-timeout": "页面或 Workbench readiness 超时,不应默认判断为浏览器安装环境问题。",
|
|
"target-page-readiness-timeout": "页面或 Workbench readiness 超时,不应默认判断为浏览器安装环境问题。",
|
|
};
|
|
const translated = summaries[normalized];
|
|
if (!translated) return String(summary || "");
|
|
const source = String(summary || "").trim();
|
|
return source && !source.includes(translated) ? `${translated} 原始摘要: ${source}` : translated;
|
|
}
|
|
|
|
function findingNextAction(code) {
|
|
const normalized = String(code || "").toLowerCase();
|
|
const actions = {
|
|
"quick-verify-no-business-turn": "下一步: 打开该 run 的 turn-summary/trace-frame,并用 CLI 对照命令确认没有业务 turn。",
|
|
"observer-command-failed": "下一步: 查看 observe collect timeline 和 failed command 文件,定位失败阶段。",
|
|
"runtime-requestfailed": "下一步: 按请求路径聚合失败,区分网络、auth/session、API 或静态资源问题。",
|
|
"runtime-console-alerts": "下一步: 结合 console 样本与业务 trace 判断是否为阻塞级。",
|
|
"browser-console-or-page-errors": "下一步: 先查 Workbench readiness 和 session create,再决定是否修前端。",
|
|
"browser-timeout": "下一步: 查 finalUrl、loading、session create、network requestfailed 和 reportPath,不先改浏览器安装。",
|
|
"target-page-readiness-timeout": "下一步: 查 finalUrl、loading、session create、network requestfailed 和 reportPath,不先改浏览器安装。",
|
|
};
|
|
return actions[normalized] || "下一步: 打开最近运行详情,并用 CLI 对照命令复核同一 run/observer/report。";
|
|
}
|
|
|
|
function formatSeveritySummary(counts) {
|
|
const entries = Object.entries(counts || {}).filter(([, value]) => Number(value || 0) > 0);
|
|
if (entries.length === 0) return "无";
|
|
return entries.map(([key, value]) => `${displaySeverity(key)} ${formatNumber(Number(value || 0))}`).join(" · ");
|
|
}
|
|
|
|
function formatNumber(value) {
|
|
return new Intl.NumberFormat("zh-CN").format(Number(value || 0));
|
|
}
|
|
|
|
function shortText(value, limit) {
|
|
const text = String(value || "");
|
|
return text.length <= limit ? text : `${text.slice(0, Math.max(0, limit - 3))}...`;
|
|
}
|
|
|
|
function formatRelative(iso) {
|
|
const ms = Date.parse(String(iso || ""));
|
|
if (!Number.isFinite(ms)) return "-";
|
|
const seconds = Math.max(0, Math.floor((Date.now() - ms) / 1000));
|
|
if (seconds < 5) return "刚刚";
|
|
if (seconds < 60) return `${seconds} 秒前`;
|
|
const minutes = Math.floor(seconds / 60);
|
|
if (minutes < 60) return `${minutes} 分前`;
|
|
const hours = Math.floor(minutes / 60);
|
|
if (hours < 24) return `${hours} 时前`;
|
|
return `${Math.floor(hours / 24)} 天前`;
|
|
}
|
|
|
|
function escapeHtml(value) {
|
|
return String(value ?? "")
|
|
.replace(/&/gu, "&")
|
|
.replace(/</gu, "<")
|
|
.replace(/>/gu, ">")
|
|
.replace(/"/gu, """)
|
|
.replace(/'/gu, "'");
|
|
}
|
|
|
|
function escapeAttr(value) {
|
|
return escapeHtml(value);
|
|
}
|