import { parseEpisodes, filterEpisodes, attributeNodes, groupBySession, aggregate, inferConflicts } from './dashboard-core.js'; const AGD = window.AGD; let episodes = []; let skipped = 0; let network = null; // ── data loading ────────────────────────────────────────────── async function loadEpisodes() { const files = await fetch('/api/episodes').then((r) => r.json()); const all = []; let skip = 0; for (const f of files) { const url = '/docs/observer/' + f; const text = await fetch(url).then((r) => (r.ok ? r.text() : '')); const r = parseEpisodes(text); all.push(...r.episodes); skip += r.skipped; } all.sort((a, b) => String(a.startedAt).localeCompare(String(b.startedAt))); episodes = all; skipped = skip; document.getElementById('status').textContent = `${episodes.length} эпизодов · ${skipped} пропущено`; } // ── graph banner ────────────────────────────────────────────── function renderGraph() { const nodes = new vis.DataSet(AGD.NODES); const edges = new vis.DataSet(AGD.EDGES); network = new vis.Network( document.getElementById('network'), { nodes, edges }, { groups: AGD.GROUPS, nodes: { shape: 'dot', borderWidth: 2, font: { multi: 'html' } }, edges: { smooth: { type: 'continuous', roundness: 0.5 } }, physics: { enabled: false }, interaction: { hover: true, tooltipDelay: 400 }, } ); network.once('afterDrawing', () => network.fit()); return { nodes, edges }; } // ── view switching ──────────────────────────────────────────── const views = {}; let activeView = 'map'; views.map = function renderMapView() { // Plain mode: clear any overlay coloring applied by other views. window.__graph.nodes.update(AGD.NODES.map((n) => ({ id: n.id, color: undefined }))); // List the design-time conflict edges (dashed edges carry an emoji label). const conflicts = AGD.EDGES.filter((e) => e.dashes === true); const ul = document.getElementById('map-conflicts'); ul.innerHTML = ''; for (const c of conflicts) { const li = document.createElement('li'); li.textContent = `${c.label || '•'} ${c.from} ↔ ${c.to}: ${c.title || ''}`; ul.appendChild(li); } }; views.replay = function renderReplayView() { const filter = { classification: document.getElementById('f-classification').value || undefined, outcome: document.getElementById('f-outcome').value || undefined, withErrors: document.getElementById('f-errors').checked || undefined, }; const list = filterEpisodes(episodes, filter); const ul = document.getElementById('replay-episodes'); ul.innerHTML = ''; list.forEach((ep) => { const li = document.createElement('li'); li.textContent = `${ep.startedAt} · ${ep.taskClassification || '—'} · ${ep.outcome}` + (ep.errorCount ? ` · ⚠${ep.errorCount}` : ''); li.addEventListener('click', () => selectEpisode(ep)); ul.appendChild(li); }); }; function selectEpisode(ep) { const attr = attributeNodes(ep); window.__graph.nodes.update( AGD.NODES.map((n) => ({ id: n.id, color: attr.nodeIds.includes(n.id) ? { background: '#268bd2', border: '#93a1a1' } : { background: '#2a2a3a', border: '#444' }, })) ); const d = document.getElementById('replay-detail'); const prov = ep.decisionProvenance; const provLine = prov && prov.kind === 'user_directed_method' ? `перенаправление: выбран ${prov.node || '?'}, автономно был бы ${prov.claude_would_have_chosen || '?'}` : prov ? prov.kind : '—'; const env = ep.environment || {}; d.innerHTML = `
provenance: ${provLine}
hard-floor: ${ep.hardFloor.invoked ? (ep.hardFloor.rules || []).join(', ') : 'нет'}
окружение: economy=${env.economy_level ?? '—'} · ${env.model || '—'} · turn ${env.session_turn ?? '—'}${env.post_compaction ? ' · post-compaction' : ''}${env.parallel_session ? ' · parallel' : ''}
атрибутировано узлов: ${attr.attributed} из ${attr.signals} сигналов
${a.count}
${a.totalErrors} / ${a.totalRetries}
${(a.redirectRate * 100).toFixed(0)}%
${dist(a.pathType)}
${dist(a.outcome)}
${dist(a.classification)}
${dist(a.economy)}
${topNodes.map(([k, v]) => `${k}×${v}`).join(' · ') || '—'}
Дизайн-конфликты (факт): ${c.design.length} размеченных рёбер
Трение (инференс): ${top(c.friction)}
Корреляция (эвристика): ${c.correlation.length} ходов с ошибкой на паре конфликтующих узлов
`; }; function applyHeat(nodeHeat) { const max = Math.max(1, ...Object.values(nodeHeat)); window.__graph.nodes.update( AGD.NODES.map((n) => { const h = nodeHeat[n.id] || 0; const t = h / max; return { id: n.id, color: h ? { background: `rgba(38,139,210,${0.25 + 0.6 * t})`, border: '#93a1a1' } : { background: '#2a2a3a', border: '#444' }, }; }) ); } function feedCard(ep) { const dur = ep.durationMs != null ? Math.round(ep.durationMs / 1000) + 's' : '—'; const redirect = ep.decisionProvenance && ep.decisionProvenance.kind === 'user_directed_method' ? ' ↪' : ''; return `