@raquezha/notrace 0.1.1 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +18 -0
- package/dist/notrace/index.d.ts +34 -0
- package/dist/notrace/index.js +144 -118
- package/dist/notrace/report-app/__tests__/analytics.test.d.ts +1 -0
- package/dist/notrace/report-app/__tests__/analytics.test.js +35 -0
- package/dist/notrace/report-app/__tests__/card.test.d.ts +1 -0
- package/dist/notrace/report-app/__tests__/card.test.js +26 -0
- package/dist/notrace/report-app/__tests__/event.test.d.ts +1 -0
- package/dist/notrace/report-app/__tests__/event.test.js +20 -0
- package/dist/notrace/report-app/__tests__/format.test.d.ts +1 -0
- package/dist/notrace/report-app/__tests__/format.test.js +41 -0
- package/dist/notrace/report-app/__tests__/report.test.d.ts +1 -0
- package/dist/notrace/report-app/__tests__/report.test.js +31 -0
- package/dist/notrace/report-app/analytics.d.ts +3 -0
- package/dist/notrace/report-app/analytics.js +78 -0
- package/dist/notrace/report-app/client.d.ts +2 -0
- package/dist/notrace/report-app/client.js +105 -0
- package/dist/notrace/report-app/components/card.d.ts +4 -0
- package/dist/notrace/report-app/components/card.js +36 -0
- package/dist/notrace/report-app/components/dashboard.d.ts +1 -0
- package/dist/notrace/report-app/components/dashboard.js +16 -0
- package/dist/notrace/report-app/components/event.d.ts +5 -0
- package/dist/notrace/report-app/components/event.js +42 -0
- package/dist/notrace/report-app/components/message.d.ts +2 -0
- package/dist/notrace/report-app/components/message.js +43 -0
- package/dist/notrace/report-app/dashboard-report.d.ts +1 -0
- package/dist/notrace/report-app/dashboard-report.js +6 -0
- package/dist/notrace/report-app/escape.d.ts +1 -0
- package/dist/notrace/report-app/escape.js +10 -0
- package/dist/notrace/report-app/format.d.ts +13 -0
- package/dist/notrace/report-app/format.js +102 -0
- package/dist/notrace/report-app/report.d.ts +1 -0
- package/dist/notrace/report-app/report.js +29 -0
- package/dist/notrace/report-app/shell.d.ts +5 -0
- package/dist/notrace/report-app/shell.js +19 -0
- package/dist/notrace/report-app/styles.d.ts +1 -0
- package/dist/notrace/report-app/styles.js +431 -0
- package/dist/notrace/report-app/types.d.ts +28 -0
- package/dist/notrace/report-app/types.js +1 -0
- package/extensions/notrace/__tests__/ghost-session.test.ts +103 -0
- package/extensions/notrace/__tests__/helpers.ts +11 -0
- package/extensions/notrace/__tests__/lock-race.test.ts +176 -0
- package/extensions/notrace/__tests__/usage-normalization.test.ts +80 -0
- package/extensions/notrace/index.ts +160 -124
- package/extensions/notrace/report-app/__tests__/analytics.test.ts +41 -0
- package/extensions/notrace/report-app/__tests__/card.test.ts +29 -0
- package/extensions/notrace/report-app/__tests__/event.test.ts +23 -0
- package/extensions/notrace/report-app/__tests__/format.test.ts +46 -0
- package/extensions/notrace/report-app/__tests__/report.test.ts +33 -0
- package/extensions/notrace/report-app/analytics.ts +79 -0
- package/extensions/notrace/report-app/client.ts +106 -0
- package/extensions/notrace/report-app/components/card.ts +38 -0
- package/extensions/notrace/report-app/components/dashboard.ts +17 -0
- package/extensions/notrace/report-app/components/event.ts +39 -0
- package/extensions/notrace/report-app/components/message.ts +39 -0
- package/extensions/notrace/report-app/dashboard-report.ts +7 -0
- package/extensions/notrace/report-app/escape.ts +10 -0
- package/extensions/notrace/report-app/format.ts +107 -0
- package/extensions/notrace/report-app/report.ts +33 -0
- package/extensions/notrace/report-app/shell.ts +24 -0
- package/extensions/notrace/report-app/styles.ts +431 -0
- package/extensions/notrace/report-app/types.ts +35 -0
- package/package.json +4 -2
- package/templates/dashboard.sample.html +103 -63
- package/templates/dashboard.sample.json +73 -10
- package/templates/render-samples.mjs +119 -1
- package/templates/session.sample.html +125 -168
- package/templates/session.sample.json +66 -7
- package/templates/sessions/019ed2ee-1000-76ee-b353-000000000001/notrace.html +125 -163
- package/templates/sessions/019ed2ee-1000-76ee-b353-000000000001/notrace.json +50 -0
- package/templates/sessions/019ed2ee-1001-76ee-b353-000000000002/notrace.html +125 -162
- package/templates/sessions/019ed2ee-1001-76ee-b353-000000000002/notrace.json +50 -0
- package/templates/sessions/019ed2ee-1002-76ee-b353-000000000003/notrace.html +125 -163
- package/templates/sessions/019ed2ee-1002-76ee-b353-000000000003/notrace.json +50 -0
- package/templates/sessions/019ed2ee-massive/notrace.html +498 -0
- package/templates/sessions/019ed2ee-massive/notrace.json +14660 -0
- package/tsconfig.json +1 -1
- package/dist/notrace/renderer.d.ts +0 -4
- package/dist/notrace/renderer.js +0 -800
- package/extensions/notrace/renderer.ts +0 -810
package/dist/notrace/renderer.js
DELETED
|
@@ -1,800 +0,0 @@
|
|
|
1
|
-
export function escapeHtml(v) {
|
|
2
|
-
const map = {
|
|
3
|
-
"&": "&",
|
|
4
|
-
"<": "<",
|
|
5
|
-
">": ">",
|
|
6
|
-
"'": "'",
|
|
7
|
-
'"': """
|
|
8
|
-
};
|
|
9
|
-
return String(v ?? "").replace(/[&<>'"]/g, c => map[c]);
|
|
10
|
-
}
|
|
11
|
-
export function safeJsonForScript(v) {
|
|
12
|
-
const map = {
|
|
13
|
-
"<": "\\u003c",
|
|
14
|
-
">": "\\u003e",
|
|
15
|
-
"&": "\\u0026",
|
|
16
|
-
"\u2028": "\\u2028",
|
|
17
|
-
"\u2029": "\\u2029"
|
|
18
|
-
};
|
|
19
|
-
return JSON.stringify(v).replace(/[<>&\u2028\u2029]/g, c => map[c]);
|
|
20
|
-
}
|
|
21
|
-
function parseDate(value) {
|
|
22
|
-
const date = new Date(value);
|
|
23
|
-
return Number.isNaN(date.getTime()) ? null : date;
|
|
24
|
-
}
|
|
25
|
-
function formatDateCell(value) {
|
|
26
|
-
const date = parseDate(value);
|
|
27
|
-
if (!date)
|
|
28
|
-
return `<span>${escapeHtml(value)}</span>`;
|
|
29
|
-
const day = date.toISOString().slice(0, 10);
|
|
30
|
-
const time = date.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit", hour12: false });
|
|
31
|
-
return `<div class="date-cell"><strong>${escapeHtml(day)}</strong><span>${escapeHtml(time)}</span></div>`;
|
|
32
|
-
}
|
|
33
|
-
function formatDateLong(value) {
|
|
34
|
-
const date = parseDate(value);
|
|
35
|
-
if (!date)
|
|
36
|
-
return escapeHtml(value);
|
|
37
|
-
const day = date.toISOString().slice(0, 10);
|
|
38
|
-
const time = date.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit", hour12: false });
|
|
39
|
-
return `${escapeHtml(day)} ${escapeHtml(time)}`;
|
|
40
|
-
}
|
|
41
|
-
function formatTime(value) {
|
|
42
|
-
const date = parseDate(value);
|
|
43
|
-
if (!date)
|
|
44
|
-
return escapeHtml(value);
|
|
45
|
-
return escapeHtml(date.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit", second: "2-digit", hour12: false }));
|
|
46
|
-
}
|
|
47
|
-
function workflowDisplayName(workflow) {
|
|
48
|
-
switch (workflow) {
|
|
49
|
-
case "norpiv":
|
|
50
|
-
return "RPIV";
|
|
51
|
-
case "research":
|
|
52
|
-
return "Research";
|
|
53
|
-
case "generic":
|
|
54
|
-
default:
|
|
55
|
-
return "Generic";
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
function workflowClassName(workflow) {
|
|
59
|
-
switch (workflow) {
|
|
60
|
-
case "norpiv":
|
|
61
|
-
return "workflow-rpiv";
|
|
62
|
-
case "research":
|
|
63
|
-
return "workflow-research";
|
|
64
|
-
case "generic":
|
|
65
|
-
default:
|
|
66
|
-
return "workflow-generic";
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
function taskDisplay(taskish) {
|
|
70
|
-
const task = taskish?.task || taskish;
|
|
71
|
-
const workflow = task?.workflow || taskish?.workflow || "generic";
|
|
72
|
-
const taskId = task?.id ?? taskish?.taskId;
|
|
73
|
-
if (taskId) {
|
|
74
|
-
if (workflow === "research" && String(taskId).startsWith("branch:")) {
|
|
75
|
-
return `Branch ${String(taskId).slice(7)}`;
|
|
76
|
-
}
|
|
77
|
-
return String(taskId);
|
|
78
|
-
}
|
|
79
|
-
if (workflow === "research")
|
|
80
|
-
return "Open research";
|
|
81
|
-
if (workflow === "generic")
|
|
82
|
-
return "General session";
|
|
83
|
-
return "No active task";
|
|
84
|
-
}
|
|
85
|
-
function resolveRepoName(data) {
|
|
86
|
-
const name = data?.repository?.name || data?.repositoryName || data?.repoName || "Repository";
|
|
87
|
-
const branch = data?.repository?.branch;
|
|
88
|
-
return branch ? `${name} @ ${branch}` : name;
|
|
89
|
-
}
|
|
90
|
-
function formatUsd(value) {
|
|
91
|
-
const num = Number(value || 0);
|
|
92
|
-
if (num === 0)
|
|
93
|
-
return "$0";
|
|
94
|
-
return `$${num.toFixed(5)}`;
|
|
95
|
-
}
|
|
96
|
-
function formatTelemetryStatus(value) {
|
|
97
|
-
switch (value) {
|
|
98
|
-
case "active": return "Active";
|
|
99
|
-
case "loaded-disabled": return "Loaded disabled";
|
|
100
|
-
case "loaded-inactive": return "Loaded inactive";
|
|
101
|
-
case "absent": return "Absent";
|
|
102
|
-
case "unknown":
|
|
103
|
-
default:
|
|
104
|
-
return "Unknown";
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
function summarizeEventCount(data) {
|
|
108
|
-
return Array.isArray(data?.events)
|
|
109
|
-
? data.events.filter((ev) => ev?.type !== "session_start" && ev?.type !== "turn_start").length
|
|
110
|
-
: 0;
|
|
111
|
-
}
|
|
112
|
-
function wordmarkSvg(className = "wordmark") {
|
|
113
|
-
return `<svg class="${escapeHtml(className)}" viewBox="0 0 420 138" aria-label="notrace" role="img" xmlns="http://www.w3.org/2000/svg">
|
|
114
|
-
<defs>
|
|
115
|
-
<linearGradient id="fadeGrad" x1="0%" y1="0%" x2="100%" y2="0%">
|
|
116
|
-
<stop offset="0%" stop-color="#E2754A"/>
|
|
117
|
-
<stop offset="100%" stop-color="#EDE2D2"/>
|
|
118
|
-
</linearGradient>
|
|
119
|
-
</defs>
|
|
120
|
-
<g id="trace-icon" transform="translate(1 -18) scale(0.93)">
|
|
121
|
-
<path d="M6,50 C16,18 26,18 36,50 C46,82 54,82 60,50 C64,30 68,30 71,50"
|
|
122
|
-
fill="none" stroke="url(#fadeGrad)" stroke-width="4" stroke-linecap="round"/>
|
|
123
|
-
<line x1="74" y1="50" x2="79" y2="50" stroke="#D9C9B5" stroke-width="4" stroke-linecap="round" stroke-opacity="0.6"/>
|
|
124
|
-
<circle cx="85" cy="50" r="2.2" fill="#D9C9B5" opacity="0.5"/>
|
|
125
|
-
<circle cx="90" cy="50" r="1.4" fill="#EDE2D2" opacity="0.32"/>
|
|
126
|
-
<circle cx="94" cy="50" r="0.9" fill="#EDE2D2" opacity="0.15"/>
|
|
127
|
-
</g>
|
|
128
|
-
<text x="0" y="114" fill="#ECE3DA" style="fill:#ECE3DA" font-family="Inter, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, sans-serif" font-size="96" font-weight="900" letter-spacing="-7">no</text>
|
|
129
|
-
<text x="82" y="114" fill="#d88462" font-family="Inter, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, sans-serif" font-size="96" font-weight="900" letter-spacing="-7">trace</text>
|
|
130
|
-
</svg>`;
|
|
131
|
-
}
|
|
132
|
-
function faviconHref() {
|
|
133
|
-
const svg = `<svg width="100" height="100" viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg" role="img"><title>notrace logo mark</title><desc>A wave that smooths into a flat line, then fades into dots — color shifts from trace orange to no cream as it dissolves.</desc><defs><linearGradient id="fadeGrad" x1="0%" y1="0%" x2="100%" y2="0%"><stop offset="0%" stop-color="#E2754A"/><stop offset="100%" stop-color="#EDE2D2"/></linearGradient></defs><g id="trace-icon"><path d="M6,50 C16,18 26,18 36,50 C46,82 54,82 60,50 C64,30 68,30 71,50" fill="none" stroke="url(#fadeGrad)" stroke-width="4" stroke-linecap="round"/><line x1="74" y1="50" x2="79" y2="50" stroke="#D9C9B5" stroke-width="4" stroke-linecap="round" stroke-opacity="0.6"/><circle cx="85" cy="50" r="2.2" fill="#D9C9B5" opacity="0.5"/><circle cx="90" cy="50" r="1.4" fill="#EDE2D2" opacity="0.32"/><circle cx="94" cy="50" r="0.9" fill="#EDE2D2" opacity="0.15"/></g></svg>`;
|
|
134
|
-
return `data:image/svg+xml,${encodeURIComponent(svg)}`;
|
|
135
|
-
}
|
|
136
|
-
function shell(title, body, script = "") {
|
|
137
|
-
return `<!DOCTYPE html>
|
|
138
|
-
<html lang="en">
|
|
139
|
-
<head>
|
|
140
|
-
<meta charset="UTF-8">
|
|
141
|
-
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
142
|
-
<title>${escapeHtml(title)}</title>
|
|
143
|
-
<link rel="icon" href="${faviconHref()}">
|
|
144
|
-
<style>
|
|
145
|
-
:root {
|
|
146
|
-
--bg: #0c0b0a;
|
|
147
|
-
--panel: rgba(255,255,255,0.04);
|
|
148
|
-
--panel-strong: rgba(255,255,255,0.06);
|
|
149
|
-
--text: #ece3da;
|
|
150
|
-
--muted: rgba(236,227,218,0.68);
|
|
151
|
-
--accent: #d88462;
|
|
152
|
-
--accent-soft: rgba(216,132,98,0.12);
|
|
153
|
-
--border: rgba(255,255,255,0.08);
|
|
154
|
-
--shadow: 0 20px 50px rgba(0,0,0,0.45);
|
|
155
|
-
--code: #090807;
|
|
156
|
-
--err: #ef7f7f;
|
|
157
|
-
--rpiv-fg: #f3be8a;
|
|
158
|
-
--rpiv-bg: rgba(243,190,138,0.12);
|
|
159
|
-
--rpiv-border: rgba(243,190,138,0.26);
|
|
160
|
-
--research-fg: #8ec5ff;
|
|
161
|
-
--research-bg: rgba(142,197,255,0.12);
|
|
162
|
-
--research-border: rgba(142,197,255,0.24);
|
|
163
|
-
--generic-fg: #b9b4ae;
|
|
164
|
-
--generic-bg: rgba(185,180,174,0.12);
|
|
165
|
-
--generic-border: rgba(185,180,174,0.2);
|
|
166
|
-
}
|
|
167
|
-
* { box-sizing: border-box; }
|
|
168
|
-
html { color-scheme: dark; }
|
|
169
|
-
body {
|
|
170
|
-
margin: 0;
|
|
171
|
-
font-family: Inter, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, sans-serif;
|
|
172
|
-
background: var(--bg);
|
|
173
|
-
color: var(--text);
|
|
174
|
-
line-height: 1.5;
|
|
175
|
-
background-image: radial-gradient(circle at 50% -10%, rgba(216,132,98,0.14), transparent 45%);
|
|
176
|
-
background-attachment: fixed;
|
|
177
|
-
}
|
|
178
|
-
a { color: inherit; }
|
|
179
|
-
.container { max-width: 1120px; margin: 0 auto; padding: 32px 20px 64px; }
|
|
180
|
-
.hero, .panel {
|
|
181
|
-
background: var(--panel);
|
|
182
|
-
border: 1px solid var(--border);
|
|
183
|
-
border-radius: 24px;
|
|
184
|
-
box-shadow: var(--shadow);
|
|
185
|
-
backdrop-filter: blur(10px);
|
|
186
|
-
}
|
|
187
|
-
.hero { padding: 28px; margin-bottom: 24px; }
|
|
188
|
-
.hero-top {
|
|
189
|
-
display: grid;
|
|
190
|
-
grid-template-columns: minmax(0, 1fr) auto;
|
|
191
|
-
gap: 16px;
|
|
192
|
-
align-items: start;
|
|
193
|
-
}
|
|
194
|
-
.brand { margin-bottom: 10px; }
|
|
195
|
-
.brand-link {
|
|
196
|
-
display: inline-flex;
|
|
197
|
-
align-items: flex-start;
|
|
198
|
-
text-decoration: none;
|
|
199
|
-
}
|
|
200
|
-
.wordmark {
|
|
201
|
-
width: 340px;
|
|
202
|
-
height: 112px;
|
|
203
|
-
display: block;
|
|
204
|
-
overflow: visible;
|
|
205
|
-
}
|
|
206
|
-
.subtitle { margin: 10px 0 0; color: var(--muted); }
|
|
207
|
-
.session-subtitle {
|
|
208
|
-
display: flex;
|
|
209
|
-
align-items: center;
|
|
210
|
-
gap: 10px;
|
|
211
|
-
flex-wrap: wrap;
|
|
212
|
-
}
|
|
213
|
-
.session-id-chip {
|
|
214
|
-
display: inline-flex;
|
|
215
|
-
align-items: center;
|
|
216
|
-
gap: 8px;
|
|
217
|
-
max-width: 100%;
|
|
218
|
-
padding: 6px 8px 6px 10px;
|
|
219
|
-
border: 1px solid var(--border);
|
|
220
|
-
border-radius: 999px;
|
|
221
|
-
background: rgba(0,0,0,0.18);
|
|
222
|
-
color: var(--text);
|
|
223
|
-
font-family: "SFMono-Regular", ui-monospace, Menlo, Monaco, Consolas, monospace;
|
|
224
|
-
font-size: 0.78rem;
|
|
225
|
-
word-break: break-all;
|
|
226
|
-
}
|
|
227
|
-
.copy-btn {
|
|
228
|
-
display: inline-flex;
|
|
229
|
-
align-items: center;
|
|
230
|
-
justify-content: center;
|
|
231
|
-
width: 26px;
|
|
232
|
-
height: 26px;
|
|
233
|
-
border: 1px solid rgba(255,255,255,0.12);
|
|
234
|
-
border-radius: 999px;
|
|
235
|
-
background: rgba(255,255,255,0.04);
|
|
236
|
-
color: var(--muted);
|
|
237
|
-
cursor: pointer;
|
|
238
|
-
transition: color 120ms ease, border-color 120ms ease, background 120ms ease;
|
|
239
|
-
}
|
|
240
|
-
.copy-btn:hover, .copy-btn.copied {
|
|
241
|
-
color: var(--text);
|
|
242
|
-
border-color: rgba(216,132,98,0.45);
|
|
243
|
-
background: var(--accent-soft);
|
|
244
|
-
}
|
|
245
|
-
.meta {
|
|
246
|
-
display: flex;
|
|
247
|
-
gap: 8px;
|
|
248
|
-
flex-wrap: wrap;
|
|
249
|
-
justify-content: flex-end;
|
|
250
|
-
align-items: center;
|
|
251
|
-
margin-top: 16px;
|
|
252
|
-
}
|
|
253
|
-
.pill, .workflow-pill, .sort-btn, .export-btn {
|
|
254
|
-
display: inline-flex;
|
|
255
|
-
align-items: center;
|
|
256
|
-
gap: 6px;
|
|
257
|
-
text-decoration: none;
|
|
258
|
-
padding: 8px 12px;
|
|
259
|
-
border: 1px solid var(--border);
|
|
260
|
-
border-radius: 999px;
|
|
261
|
-
background: rgba(255,255,255,0.03);
|
|
262
|
-
color: var(--muted);
|
|
263
|
-
font-size: 0.86rem;
|
|
264
|
-
font-weight: 600;
|
|
265
|
-
}
|
|
266
|
-
.metrics { display: grid; grid-template-columns: repeat(auto-fit, minmax(135px, 1fr)); gap: 16px; margin: 24px 0; }
|
|
267
|
-
.metric-card {
|
|
268
|
-
background: var(--panel-strong);
|
|
269
|
-
border: 1px solid var(--border);
|
|
270
|
-
border-radius: 18px;
|
|
271
|
-
padding: 18px;
|
|
272
|
-
}
|
|
273
|
-
.metric-card small { display: block; color: var(--accent); text-transform: uppercase; letter-spacing: 0.08em; font-size: 0.72rem; font-weight: 700; }
|
|
274
|
-
.metric-card strong { display: block; margin-top: 8px; font-size: 1.55rem; }
|
|
275
|
-
.panel { padding: 0; overflow: hidden; }
|
|
276
|
-
.section-title { margin: 0; padding: 20px 22px; border-bottom: 1px solid var(--border); font-size: 1rem; }
|
|
277
|
-
.empty { padding: 32px 22px; color: var(--muted); }
|
|
278
|
-
table { width: 100%; border-collapse: collapse; }
|
|
279
|
-
th, td { padding: 14px 18px; text-align: left; border-bottom: 1px solid var(--border); vertical-align: top; }
|
|
280
|
-
th { color: var(--muted); font-size: 0.8rem; text-transform: uppercase; letter-spacing: 0.08em; }
|
|
281
|
-
.num-cell { text-align: right; }
|
|
282
|
-
tr:last-child td { border-bottom: 0; }
|
|
283
|
-
.col-index { width: 64px; }
|
|
284
|
-
.sortable-head { padding: 10px 18px; }
|
|
285
|
-
.sort-btn {
|
|
286
|
-
padding: 0;
|
|
287
|
-
border: 0;
|
|
288
|
-
border-radius: 0;
|
|
289
|
-
background: transparent;
|
|
290
|
-
font: inherit;
|
|
291
|
-
text-transform: inherit;
|
|
292
|
-
letter-spacing: inherit;
|
|
293
|
-
cursor: pointer;
|
|
294
|
-
}
|
|
295
|
-
.sort-label { color: inherit; }
|
|
296
|
-
.sort-state { color: var(--accent); font-size: 0.9rem; min-width: 16px; text-align: left; line-height: 1; }
|
|
297
|
-
.index-cell { color: var(--muted); font-variant-numeric: tabular-nums; }
|
|
298
|
-
.session-link { text-decoration: none; }
|
|
299
|
-
.session-link strong { display: block; }
|
|
300
|
-
.session-sub { display: block; margin-top: 2px; color: var(--muted); font-size: 0.8rem; }
|
|
301
|
-
.workflow-pill { padding: 6px 10px; font-size: 0.78rem; }
|
|
302
|
-
.workflow-rpiv { color: var(--rpiv-fg); background: var(--rpiv-bg); border-color: var(--rpiv-border); }
|
|
303
|
-
.workflow-research { color: var(--research-fg); background: var(--research-bg); border-color: var(--research-border); }
|
|
304
|
-
.workflow-generic { color: var(--generic-fg); background: var(--generic-bg); border-color: var(--generic-border); }
|
|
305
|
-
.date-cell { display: grid; gap: 2px; }
|
|
306
|
-
.date-cell strong { font-size: 0.92rem; }
|
|
307
|
-
.date-cell span { color: var(--muted); font-size: 0.82rem; }
|
|
308
|
-
.timeline { display: grid; gap: 14px; }
|
|
309
|
-
.event {
|
|
310
|
-
background: var(--panel);
|
|
311
|
-
border: 1px solid var(--border);
|
|
312
|
-
border-radius: 18px;
|
|
313
|
-
overflow: hidden;
|
|
314
|
-
}
|
|
315
|
-
.event summary {
|
|
316
|
-
list-style: none;
|
|
317
|
-
cursor: pointer;
|
|
318
|
-
display: flex;
|
|
319
|
-
justify-content: space-between;
|
|
320
|
-
gap: 14px;
|
|
321
|
-
align-items: center;
|
|
322
|
-
padding: 16px 18px;
|
|
323
|
-
}
|
|
324
|
-
.event summary::-webkit-details-marker { display: none; }
|
|
325
|
-
.event summary:hover { background: rgba(255,255,255,0.02); }
|
|
326
|
-
.event-main { display: flex; gap: 10px; align-items: center; flex-wrap: wrap; }
|
|
327
|
-
.badge {
|
|
328
|
-
display: inline-flex;
|
|
329
|
-
align-items: center;
|
|
330
|
-
padding: 4px 8px;
|
|
331
|
-
border-radius: 999px;
|
|
332
|
-
font-size: 0.72rem;
|
|
333
|
-
font-weight: 800;
|
|
334
|
-
text-transform: uppercase;
|
|
335
|
-
letter-spacing: 0.06em;
|
|
336
|
-
border: 1px solid var(--border);
|
|
337
|
-
background: rgba(255,255,255,0.03);
|
|
338
|
-
}
|
|
339
|
-
.badge-llm { color: var(--accent); background: var(--accent-soft); border-color: rgba(216,132,98,0.24); }
|
|
340
|
-
.badge-tool { color: #8ec5ff; background: rgba(142,197,255,0.1); border-color: rgba(142,197,255,0.22); }
|
|
341
|
-
.badge-system { color: var(--muted); }
|
|
342
|
-
.badge-error { color: var(--err); background: rgba(239,127,127,0.12); border-color: rgba(239,127,127,0.24); }
|
|
343
|
-
.event-title { font-weight: 700; }
|
|
344
|
-
.event-time { color: var(--muted); font-size: 0.9rem; white-space: nowrap; }
|
|
345
|
-
.event-body { padding: 0 18px 18px; }
|
|
346
|
-
.stack { display: grid; gap: 12px; }
|
|
347
|
-
.block {
|
|
348
|
-
background: rgba(0,0,0,0.18);
|
|
349
|
-
border: 1px solid var(--border);
|
|
350
|
-
border-radius: 14px;
|
|
351
|
-
overflow: hidden;
|
|
352
|
-
}
|
|
353
|
-
.block h4 {
|
|
354
|
-
margin: 0;
|
|
355
|
-
padding: 10px 12px;
|
|
356
|
-
border-bottom: 1px solid var(--border);
|
|
357
|
-
color: var(--muted);
|
|
358
|
-
font-size: 0.8rem;
|
|
359
|
-
text-transform: uppercase;
|
|
360
|
-
letter-spacing: 0.08em;
|
|
361
|
-
}
|
|
362
|
-
pre {
|
|
363
|
-
margin: 0;
|
|
364
|
-
padding: 14px;
|
|
365
|
-
overflow-x: auto;
|
|
366
|
-
white-space: pre-wrap;
|
|
367
|
-
word-break: break-word;
|
|
368
|
-
font-family: "SFMono-Regular", ui-monospace, Menlo, Monaco, Consolas, monospace;
|
|
369
|
-
font-size: 0.84rem;
|
|
370
|
-
background: var(--code);
|
|
371
|
-
}
|
|
372
|
-
.msg { border-bottom: 1px solid var(--border); }
|
|
373
|
-
.msg:last-child { border-bottom: 0; }
|
|
374
|
-
.msg-head {
|
|
375
|
-
display: flex;
|
|
376
|
-
justify-content: space-between;
|
|
377
|
-
gap: 12px;
|
|
378
|
-
padding: 10px 12px;
|
|
379
|
-
border-bottom: 1px solid var(--border);
|
|
380
|
-
background: rgba(255,255,255,0.02);
|
|
381
|
-
}
|
|
382
|
-
.msg-role { font-size: 0.78rem; font-weight: 800; letter-spacing: 0.08em; text-transform: uppercase; }
|
|
383
|
-
.msg.user .msg-role { color: #8ec5ff; }
|
|
384
|
-
.msg.assistant .msg-role { color: var(--accent); }
|
|
385
|
-
.msg-content { padding: 14px; }
|
|
386
|
-
.chat-text {
|
|
387
|
-
white-space: pre-wrap;
|
|
388
|
-
word-break: break-word;
|
|
389
|
-
font-size: 0.95rem;
|
|
390
|
-
line-height: 1.6;
|
|
391
|
-
margin-bottom: 12px;
|
|
392
|
-
}
|
|
393
|
-
.chat-text:last-child { margin-bottom: 0; }
|
|
394
|
-
.chat-tool-use {
|
|
395
|
-
background: rgba(0,0,0,0.3);
|
|
396
|
-
border: 1px solid var(--border);
|
|
397
|
-
border-radius: 8px;
|
|
398
|
-
overflow: hidden;
|
|
399
|
-
margin-bottom: 12px;
|
|
400
|
-
}
|
|
401
|
-
.chat-tool-use:last-child { margin-bottom: 0; }
|
|
402
|
-
.chat-tool-header {
|
|
403
|
-
background: rgba(255,255,255,0.04);
|
|
404
|
-
padding: 8px 12px;
|
|
405
|
-
font-size: 0.8rem;
|
|
406
|
-
font-family: "SFMono-Regular", ui-monospace, Menlo, Monaco, Consolas, monospace;
|
|
407
|
-
color: #8ec5ff;
|
|
408
|
-
border-bottom: 1px solid var(--border);
|
|
409
|
-
display: flex;
|
|
410
|
-
align-items: center;
|
|
411
|
-
gap: 8px;
|
|
412
|
-
}
|
|
413
|
-
.chat-tool-body {
|
|
414
|
-
padding: 12px;
|
|
415
|
-
margin: 0;
|
|
416
|
-
background: transparent;
|
|
417
|
-
border: none;
|
|
418
|
-
max-height: 400px;
|
|
419
|
-
overflow-y: auto;
|
|
420
|
-
}
|
|
421
|
-
.footer-note {
|
|
422
|
-
margin-top: 22px;
|
|
423
|
-
color: var(--muted);
|
|
424
|
-
text-align: center;
|
|
425
|
-
padding: 10px 0 0;
|
|
426
|
-
font-family: inherit;
|
|
427
|
-
}
|
|
428
|
-
.footer-note.minimal {
|
|
429
|
-
font-size: 0.84rem;
|
|
430
|
-
font-variant-caps: all-small-caps;
|
|
431
|
-
letter-spacing: 0.14em;
|
|
432
|
-
line-height: 1;
|
|
433
|
-
}
|
|
434
|
-
.footer-note.stack {
|
|
435
|
-
display: grid;
|
|
436
|
-
gap: 6px;
|
|
437
|
-
line-height: 1.2;
|
|
438
|
-
}
|
|
439
|
-
.footer-brand {
|
|
440
|
-
color: var(--text);
|
|
441
|
-
font-size: 0.88rem;
|
|
442
|
-
font-weight: 700;
|
|
443
|
-
font-variant-caps: all-small-caps;
|
|
444
|
-
letter-spacing: 0.16em;
|
|
445
|
-
}
|
|
446
|
-
.footer-tagline {
|
|
447
|
-
font-size: 0.78rem;
|
|
448
|
-
letter-spacing: 0.08em;
|
|
449
|
-
font-variant-caps: all-small-caps;
|
|
450
|
-
}
|
|
451
|
-
.footer-meta {
|
|
452
|
-
font-size: 0.76rem;
|
|
453
|
-
font-variant-caps: all-small-caps;
|
|
454
|
-
letter-spacing: 0.14em;
|
|
455
|
-
}
|
|
456
|
-
.footer-meta a {
|
|
457
|
-
color: inherit;
|
|
458
|
-
text-decoration: none;
|
|
459
|
-
border-bottom: 1px solid rgba(236,227,218,0.22);
|
|
460
|
-
}
|
|
461
|
-
.footer-meta a:hover {
|
|
462
|
-
color: var(--text);
|
|
463
|
-
border-bottom-color: rgba(236,227,218,0.45);
|
|
464
|
-
}
|
|
465
|
-
.export-btn {
|
|
466
|
-
cursor: pointer;
|
|
467
|
-
transition: color 120ms ease, border-color 120ms ease, background 120ms ease;
|
|
468
|
-
}
|
|
469
|
-
.export-btn:hover, .export-btn.copied {
|
|
470
|
-
color: var(--text);
|
|
471
|
-
border-color: rgba(216,132,98,0.45);
|
|
472
|
-
background: var(--accent-soft);
|
|
473
|
-
}
|
|
474
|
-
.container { padding: 20px 14px 48px; }
|
|
475
|
-
.hero { padding: 20px; }
|
|
476
|
-
.hero-top { grid-template-columns: 1fr; }
|
|
477
|
-
.meta { justify-content: flex-start; margin-top: 8px; }
|
|
478
|
-
.wordmark { width: 280px; height: 92px; }
|
|
479
|
-
th:nth-child(5), td:nth-child(5) { display: none; }
|
|
480
|
-
.event summary { align-items: flex-start; }
|
|
481
|
-
}
|
|
482
|
-
</style>
|
|
483
|
-
</head>
|
|
484
|
-
<body>${body}${script ? `<script>${script}</script>` : ""}</body>
|
|
485
|
-
</html>`;
|
|
486
|
-
}
|
|
487
|
-
function copyButton(value, label, className = "copy-btn") {
|
|
488
|
-
return `<button class="${escapeHtml(className)}" type="button" data-copy-value="${escapeHtml(value)}" aria-label="Copy ${escapeHtml(label)}" title="Copy ${escapeHtml(label)}"><svg width="14" height="14" viewBox="0 0 24 24" aria-hidden="true" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path></svg></button>`;
|
|
489
|
-
}
|
|
490
|
-
function exportButton(data) {
|
|
491
|
-
const payload = JSON.stringify({
|
|
492
|
-
kind: "notrace-export",
|
|
493
|
-
traceId: data.traceId,
|
|
494
|
-
repository: data.repository?.name,
|
|
495
|
-
branch: data.repository?.branch,
|
|
496
|
-
task: data.task,
|
|
497
|
-
metrics: data.activity?.totals,
|
|
498
|
-
summary: data.telemetry?.extensions?.noheadroom?.summary,
|
|
499
|
-
events: (data.events || []).filter((e) => e.type === "llm_completion").map((e) => ({
|
|
500
|
-
model: e.model,
|
|
501
|
-
input: e.inputPayload?.messages,
|
|
502
|
-
output: e.outputContent
|
|
503
|
-
}))
|
|
504
|
-
});
|
|
505
|
-
return `<button class="export-btn" type="button" data-copy-value="${escapeHtml(payload)}" title="Copy session data for LLM/Agent context"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path><polyline points="7 10 12 15 17 10"></polyline><line x1="12" y1="15" x2="12" y2="3"></line></svg><span>Export</span></button>`;
|
|
506
|
-
}
|
|
507
|
-
function copyScript() {
|
|
508
|
-
return `(() => {
|
|
509
|
-
document.querySelectorAll('[data-copy-value]').forEach((button) => {
|
|
510
|
-
button.addEventListener('click', async () => {
|
|
511
|
-
const value = button.getAttribute('data-copy-value') || '';
|
|
512
|
-
try {
|
|
513
|
-
if (navigator.clipboard?.writeText) {
|
|
514
|
-
await navigator.clipboard.writeText(value);
|
|
515
|
-
} else {
|
|
516
|
-
const textarea = document.createElement('textarea');
|
|
517
|
-
textarea.value = value;
|
|
518
|
-
textarea.style.position = 'fixed';
|
|
519
|
-
textarea.style.opacity = '0';
|
|
520
|
-
document.body.appendChild(textarea);
|
|
521
|
-
textarea.focus();
|
|
522
|
-
textarea.select();
|
|
523
|
-
document.execCommand('copy');
|
|
524
|
-
textarea.remove();
|
|
525
|
-
}
|
|
526
|
-
const previous = button.innerHTML;
|
|
527
|
-
button.classList.add('copied');
|
|
528
|
-
button.innerHTML = '<svg width="14" height="14" viewBox="0 0 24 24" aria-hidden="true" fill="none" stroke="currentColor" stroke-width="2.4" stroke-linecap="round" stroke-linejoin="round"><path d="M20 6 9 17l-5-5"></path></svg>';
|
|
529
|
-
setTimeout(() => {
|
|
530
|
-
button.classList.remove('copied');
|
|
531
|
-
button.innerHTML = previous;
|
|
532
|
-
}, 1400);
|
|
533
|
-
} catch {
|
|
534
|
-
button.textContent = 'ERR';
|
|
535
|
-
}
|
|
536
|
-
});
|
|
537
|
-
});
|
|
538
|
-
})();`;
|
|
539
|
-
}
|
|
540
|
-
function renderJsonBlock(title, value) {
|
|
541
|
-
return `<section class="block"><h4>${escapeHtml(title)}</h4><pre>${escapeHtml(typeof value === "string" ? value : JSON.stringify(value, null, 2))}</pre></section>`;
|
|
542
|
-
}
|
|
543
|
-
function renderToolUseHtml(name, input) {
|
|
544
|
-
const parsedInput = typeof input === "string" ? (() => { try {
|
|
545
|
-
return JSON.parse(input);
|
|
546
|
-
}
|
|
547
|
-
catch {
|
|
548
|
-
return input;
|
|
549
|
-
} })() : input;
|
|
550
|
-
return `<div class="chat-tool-use"><div class="chat-tool-header"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14.7 6.3a1 1 0 0 0 0 1.4l1.6 1.6a1 1 0 0 0 1.4 0l3.77-3.77a6 6 0 0 1-7.94 7.94l-6.91 6.91a2.12 2.12 0 0 1-3-3l6.91-6.91a6 6 0 0 1 7.94-7.94l-3.76 3.76z"></path></svg> ${escapeHtml(name)}</div><pre class="chat-tool-body">${escapeHtml(typeof parsedInput === 'string' ? parsedInput : JSON.stringify(parsedInput, null, 2))}</pre></div>`;
|
|
551
|
-
}
|
|
552
|
-
function renderToolResultHtml(id, content) {
|
|
553
|
-
return `<div class="chat-tool-use"><div class="chat-tool-header" style="color: var(--muted);"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 10 4 15 9 20"></polyline><path d="M20 4v7a4 4 0 0 1-4 4H4"></path></svg> Tool Result: ${escapeHtml(id)}</div><pre class="chat-tool-body">${escapeHtml(typeof content === 'string' ? content : JSON.stringify(content, null, 2))}</pre></div>`;
|
|
554
|
-
}
|
|
555
|
-
function renderUniversalMessageContent(m) {
|
|
556
|
-
if (!m)
|
|
557
|
-
return "";
|
|
558
|
-
let html = "";
|
|
559
|
-
// 1. Handle string content or Anthropic/Pi blocks
|
|
560
|
-
if (typeof m.content === "string" && m.content.trim()) {
|
|
561
|
-
html += `<div class="chat-text">${escapeHtml(m.content)}</div>`;
|
|
562
|
-
}
|
|
563
|
-
else if (Array.isArray(m.content)) {
|
|
564
|
-
html += m.content.map((block) => {
|
|
565
|
-
if (!block)
|
|
566
|
-
return "";
|
|
567
|
-
if (block.type === "text")
|
|
568
|
-
return `<div class="chat-text">${escapeHtml(block.text)}</div>`;
|
|
569
|
-
if (block.type === "tool_use")
|
|
570
|
-
return renderToolUseHtml(block.name, block.input);
|
|
571
|
-
if (block.type === "tool_result")
|
|
572
|
-
return renderToolResultHtml(block.tool_use_id || "unknown", block.content);
|
|
573
|
-
return `<pre class="chat-tool-body">${escapeHtml(JSON.stringify(block, null, 2))}</pre>`;
|
|
574
|
-
}).join("");
|
|
575
|
-
}
|
|
576
|
-
else if (m.content && typeof m.content === "object") {
|
|
577
|
-
html += `<pre class="chat-tool-body">${escapeHtml(JSON.stringify(m.content, null, 2))}</pre>`;
|
|
578
|
-
}
|
|
579
|
-
// 2. Handle OpenAI/Codex tool_calls (attached to message, not in content)
|
|
580
|
-
if (Array.isArray(m.tool_calls)) {
|
|
581
|
-
html += m.tool_calls.map((tc) => {
|
|
582
|
-
if (tc.type === "function" && tc.function) {
|
|
583
|
-
return renderToolUseHtml(tc.function.name, tc.function.arguments);
|
|
584
|
-
}
|
|
585
|
-
return "";
|
|
586
|
-
}).join("");
|
|
587
|
-
}
|
|
588
|
-
// 3. Handle OpenAI legacy function_call
|
|
589
|
-
if (m.function_call) {
|
|
590
|
-
html += renderToolUseHtml(m.function_call.name, m.function_call.arguments);
|
|
591
|
-
}
|
|
592
|
-
// 4. Handle OpenAI tool result (message role is "tool")
|
|
593
|
-
if (m.role === "tool" && !html.includes("chat-tool-result")) {
|
|
594
|
-
// If it was just a string, it rendered above. Wrap it in a tool result block instead.
|
|
595
|
-
html = renderToolResultHtml(m.tool_call_id || m.name || "unknown", m.content);
|
|
596
|
-
}
|
|
597
|
-
return html || `<div class="empty">Empty message</div>`;
|
|
598
|
-
}
|
|
599
|
-
function renderMessages(messages) {
|
|
600
|
-
if (!messages?.length)
|
|
601
|
-
return "";
|
|
602
|
-
return `<section class="block"><h4>Input Messages</h4>${messages.map(m => `<div class="msg ${escapeHtml(m?.role || "unknown")}"><div class="msg-head"><span class="msg-role">${escapeHtml(m?.role || "unknown")}</span></div><div class="msg-content">${renderUniversalMessageContent(m)}</div></div>`).join("")}</section>`;
|
|
603
|
-
}
|
|
604
|
-
function eventBadgeClass(ev) {
|
|
605
|
-
if (ev.type === "llm_completion")
|
|
606
|
-
return "badge badge-llm";
|
|
607
|
-
if (ev.type === "tool_start" || ev.type === "tool_end")
|
|
608
|
-
return ev.isError ? "badge badge-error" : "badge badge-tool";
|
|
609
|
-
return "badge badge-system";
|
|
610
|
-
}
|
|
611
|
-
function eventTitle(ev) {
|
|
612
|
-
return ev.model || ev.toolName || ev.type;
|
|
613
|
-
}
|
|
614
|
-
function renderEventCard(ev) {
|
|
615
|
-
const sections = [];
|
|
616
|
-
if (ev.type === "llm_completion") {
|
|
617
|
-
sections.push(renderMessages(ev.inputPayload?.messages));
|
|
618
|
-
if (ev.stopReason && ev.stopReason !== "stop" && ev.stopReason !== "toolUse") {
|
|
619
|
-
sections.push(renderJsonBlock("Stop Reason", ev.stopReason));
|
|
620
|
-
}
|
|
621
|
-
if (ev.errorMessage) {
|
|
622
|
-
sections.push(renderJsonBlock("Error Message", ev.errorMessage));
|
|
623
|
-
}
|
|
624
|
-
sections.push(`<section class="block"><h4>Output</h4><div class="msg-content">${renderUniversalMessageContent({ content: ev.outputContent })}</div></section>`);
|
|
625
|
-
if (ev.usage)
|
|
626
|
-
sections.push(renderJsonBlock("Usage", ev.usage));
|
|
627
|
-
}
|
|
628
|
-
else if (ev.type === "tool_start") {
|
|
629
|
-
sections.push(`<section class="block"><h4>Arguments</h4><div class="msg-content"><div class="chat-tool-use"><div class="chat-tool-header"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14.7 6.3a1 1 0 0 0 0 1.4l1.6 1.6a1 1 0 0 0 1.4 0l3.77-3.77a6 6 0 0 1-7.94 7.94l-6.91 6.91a2.12 2.12 0 0 1-3-3l6.91-6.91a6 6 0 0 1 7.94-7.94l-3.76 3.76z"></path></svg> Execution Input</div><pre class="chat-tool-body">${escapeHtml(typeof ev.args === 'string' ? ev.args : JSON.stringify(ev.args, null, 2))}</pre></div></div></section>`);
|
|
630
|
-
}
|
|
631
|
-
else if (ev.type === "tool_end") {
|
|
632
|
-
sections.push(`<section class="block"><h4>${ev.isError ? "Error Result" : "Result"}</h4><div class="msg-content"><div class="chat-tool-use" style="${ev.isError ? 'border-color: rgba(239,127,127,0.3);' : ''}"><div class="chat-tool-header" style="${ev.isError ? 'color: var(--err);' : 'color: var(--muted);'}"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 10 4 15 9 20"></polyline><path d="M20 4v7a4 4 0 0 1-4 4H4"></path></svg> Execution Output</div><pre class="chat-tool-body">${escapeHtml(typeof ev.result === 'string' ? ev.result : JSON.stringify(ev.result, null, 2))}</pre></div></div></section>`);
|
|
633
|
-
}
|
|
634
|
-
else {
|
|
635
|
-
sections.push(renderJsonBlock("Event", ev));
|
|
636
|
-
}
|
|
637
|
-
return `<details class="event">
|
|
638
|
-
<summary>
|
|
639
|
-
<div class="event-main">
|
|
640
|
-
<span class="${eventBadgeClass(ev)}">${escapeHtml(ev.type)}</span>
|
|
641
|
-
<span class="event-title">${escapeHtml(eventTitle(ev))}</span>
|
|
642
|
-
</div>
|
|
643
|
-
<span class="event-time">${formatTime(ev.timestamp)}</span>
|
|
644
|
-
</summary>
|
|
645
|
-
<div class="event-body"><div class="stack">${sections.join("")}</div></div>
|
|
646
|
-
</details>`;
|
|
647
|
-
}
|
|
648
|
-
function dashboardSortScript() {
|
|
649
|
-
return `(() => {
|
|
650
|
-
const table = document.querySelector('[data-dashboard-table]');
|
|
651
|
-
if (!table) return;
|
|
652
|
-
const tbody = table.querySelector('tbody');
|
|
653
|
-
if (!tbody) return;
|
|
654
|
-
const buttons = Array.from(document.querySelectorAll('[data-sort-key]'));
|
|
655
|
-
let currentKey = 'index';
|
|
656
|
-
let currentDir = 'desc';
|
|
657
|
-
|
|
658
|
-
function icon(dir) {
|
|
659
|
-
return dir === 'asc' ? '↑' : '↓';
|
|
660
|
-
}
|
|
661
|
-
|
|
662
|
-
function updateState() {
|
|
663
|
-
buttons.forEach(btn => {
|
|
664
|
-
const key = btn.getAttribute('data-sort-key');
|
|
665
|
-
const state = btn.querySelector('.sort-state');
|
|
666
|
-
if (!state) return;
|
|
667
|
-
state.textContent = key === currentKey ? icon(currentDir) : '';
|
|
668
|
-
});
|
|
669
|
-
}
|
|
670
|
-
|
|
671
|
-
function compare(a, b, key) {
|
|
672
|
-
if (key === 'index' || key === 'started' || key === 'tokens' || key === 'cost') {
|
|
673
|
-
return Number(a.dataset[key] || 0) - Number(b.dataset[key] || 0);
|
|
674
|
-
}
|
|
675
|
-
return String(a.dataset[key] || '').localeCompare(String(b.dataset[key] || ''));
|
|
676
|
-
}
|
|
677
|
-
|
|
678
|
-
function sortBy(key) {
|
|
679
|
-
const rows = Array.from(tbody.querySelectorAll('tr'));
|
|
680
|
-
rows.sort((a, b) => {
|
|
681
|
-
const result = compare(a, b, key);
|
|
682
|
-
return currentDir === 'asc' ? result : -result;
|
|
683
|
-
});
|
|
684
|
-
rows.forEach(row => tbody.appendChild(row));
|
|
685
|
-
updateState();
|
|
686
|
-
}
|
|
687
|
-
|
|
688
|
-
buttons.forEach(btn => {
|
|
689
|
-
btn.addEventListener('click', () => {
|
|
690
|
-
const key = btn.getAttribute('data-sort-key') || 'index';
|
|
691
|
-
if (currentKey === key) currentDir = currentDir === 'asc' ? 'desc' : 'asc';
|
|
692
|
-
else {
|
|
693
|
-
currentKey = key;
|
|
694
|
-
currentDir = key === 'workflow' ? 'asc' : 'desc';
|
|
695
|
-
}
|
|
696
|
-
sortBy(currentKey);
|
|
697
|
-
});
|
|
698
|
-
});
|
|
699
|
-
|
|
700
|
-
sortBy(currentKey);
|
|
701
|
-
})();`;
|
|
702
|
-
}
|
|
703
|
-
export function generateDashboardHtml(sessions, options = {}) {
|
|
704
|
-
const reversed = sessions.slice().reverse();
|
|
705
|
-
const totalCost = sessions.reduce((sum, s) => sum + Number(s.activity?.totals?.totalCostUsd || 0), 0);
|
|
706
|
-
const totalTokens = sessions.reduce((sum, s) => sum + Number(s.activity?.totals?.totalTokens || 0), 0);
|
|
707
|
-
const homeHref = options?.indexHref || "index.html";
|
|
708
|
-
const body = `<div class="container">
|
|
709
|
-
<section class="hero">
|
|
710
|
-
<div class="hero-split">
|
|
711
|
-
<a class="brand-link" href="${escapeHtml(homeHref)}">${wordmarkSvg()}</a>
|
|
712
|
-
<div class="hero-right">
|
|
713
|
-
<div class="hero-session">
|
|
714
|
-
<strong style="color: var(--text); font-weight: 500;">Global Index</strong>
|
|
715
|
-
<span style="color: var(--muted);">Machine-wide session evidence.</span>
|
|
716
|
-
</div>
|
|
717
|
-
<div class="hero-meta">
|
|
718
|
-
<span class="hero-pill">${sessions.length} sessions</span>
|
|
719
|
-
</div>
|
|
720
|
-
</div>
|
|
721
|
-
</div>
|
|
722
|
-
<div class="metrics">
|
|
723
|
-
<div class="metric-card"><small>Sessions</small><strong>${sessions.length}</strong></div>
|
|
724
|
-
<div class="metric-card"><small>Total Tokens</small><strong>${totalTokens.toLocaleString()}</strong></div>
|
|
725
|
-
<div class="metric-card"><small>Total Cost</small><strong>${formatUsd(totalCost)}</strong></div>
|
|
726
|
-
</div>
|
|
727
|
-
</section>
|
|
728
|
-
<section class="panel">
|
|
729
|
-
<h2 class="section-title">Session Reports</h2>
|
|
730
|
-
${reversed.length ? `<table data-dashboard-table><thead><tr><th class="col-index sortable-head"><button class="sort-btn" data-sort-key="index"><span class="sort-label">#</span><span class="sort-state">↓</span></button></th><th>Session</th><th>Project</th><th class="sortable-head"><button class="sort-btn" data-sort-key="workflow"><span class="sort-label">Workflow</span><span class="sort-state"></span></button></th><th class="sortable-head"><button class="sort-btn" data-sort-key="started"><span class="sort-label">Started</span><span class="sort-state"></span></button></th><th>Task</th><th class="sortable-head num-cell"><button class="sort-btn" data-sort-key="tokens"><span class="sort-label">Tokens</span><span class="sort-state"></span></button></th><th class="sortable-head num-cell"><button class="sort-btn" data-sort-key="cost"><span class="sort-label">Cost</span><span class="sort-state"></span></button></th></tr></thead><tbody>
|
|
731
|
-
${reversed.map((s, index) => {
|
|
732
|
-
const link = s.artifacts?.html ? (s.artifacts.html.startsWith(".notrace/") ? s.artifacts.html.substring(9) : s.artifacts.html) : "#";
|
|
733
|
-
const workflow = s.task?.workflow || "generic";
|
|
734
|
-
const workflowLabel = workflowDisplayName(workflow);
|
|
735
|
-
const tokens = Number(s.activity?.totals?.totalTokens || 0);
|
|
736
|
-
const cost = Number(s.activity?.totals?.totalCostUsd || 0);
|
|
737
|
-
return `<tr data-index="${reversed.length - index}" data-workflow="${escapeHtml(workflowLabel)}" data-started="${parseDate(s.startedAt)?.getTime() || 0}" data-tokens="${tokens}" data-cost="${cost}"><td class="index-cell">${reversed.length - index}</td><td><a class="session-link" href="${escapeHtml(link)}"><strong>${escapeHtml(String(s.sessionId).slice(0, 8))}</strong><span class="session-sub">${escapeHtml(String(s.sessionId))}</span></a></td><td><span class="hero-pill">${escapeHtml(s.repositoryName || "Unknown")}</span></td><td><span class="workflow-pill ${workflowClassName(workflow)}">${escapeHtml(workflowLabel)}</span></td><td>${formatDateCell(s.startedAt)}</td><td>${escapeHtml(taskDisplay(s))}</td><td class="num-cell">${tokens.toLocaleString()}</td><td class="num-cell">${formatUsd(cost)}</td></tr>`;
|
|
738
|
-
}).join("")}
|
|
739
|
-
</tbody></table>` : `<div class="empty">No sessions yet. Run Pi with notrace enabled. New reports appear here.</div>`}
|
|
740
|
-
</section>
|
|
741
|
-
<footer class="footer-note minimal">notrace • raquezha 2026</footer>
|
|
742
|
-
</div>`;
|
|
743
|
-
return shell("notrace", body, dashboardSortScript());
|
|
744
|
-
}
|
|
745
|
-
export function generateHtmlReport(data) {
|
|
746
|
-
const visibleEvents = (data.events || []).filter((ev) => ev.type !== "session_start" && ev.type !== "turn_start");
|
|
747
|
-
const indexHref = data?.navigation?.indexHref || "../../index.html";
|
|
748
|
-
const repositoryName = resolveRepoName(data);
|
|
749
|
-
const task = data.task;
|
|
750
|
-
const body = `<div class="container">
|
|
751
|
-
<section class="hero">
|
|
752
|
-
<div class="hero-top">
|
|
753
|
-
<div>
|
|
754
|
-
<div class="brand"><a class="brand-link" href="${escapeHtml(indexHref)}" onclick="if (window.history.length > 1) { window.history.back(); return false; }">${wordmarkSvg()}</a><p class="subtitle session-subtitle"><span>Session retrospective</span><span class="session-id-chip"><span>${escapeHtml(data.traceId)}</span>${copyButton(String(data.traceId || ""), "session ID")}</span></p></div>
|
|
755
|
-
</div>
|
|
756
|
-
<div class="meta">
|
|
757
|
-
<span class="pill">${escapeHtml(resolveRepoName(data))}</span>
|
|
758
|
-
<span class="pill">Started ${formatDateLong(data.session?.startedAt)}</span>
|
|
759
|
-
<span class="pill">Mode: ${escapeHtml(data.captureMode || "full")}</span>
|
|
760
|
-
${exportButton(data)}
|
|
761
|
-
</div>
|
|
762
|
-
</div>
|
|
763
|
-
<div class="metrics">
|
|
764
|
-
<div class="metric-card"><small>Total Cost</small><strong>${formatUsd(data.activity?.totals?.totalCostUsd)}</strong></div>
|
|
765
|
-
<div class="metric-card"><small>Total Tokens</small><strong>${Number(data.activity?.totals?.totalTokens || 0).toLocaleString()}</strong></div>
|
|
766
|
-
<div class="metric-card"><small>Input Tokens</small><strong>${Number(data.activity?.totals?.inputTokens || 0).toLocaleString()}</strong></div>
|
|
767
|
-
<div class="metric-card"><small>Output Tokens</small><strong>${Number(data.activity?.totals?.outputTokens || 0).toLocaleString()}</strong></div>
|
|
768
|
-
<div class="metric-card"><small>Tool Calls</small><strong>${Number(data.activity?.toolCallCount || 0)}</strong></div>
|
|
769
|
-
<div class="metric-card"><small>Events</small><strong>${visibleEvents.length}</strong></div>
|
|
770
|
-
</div>
|
|
771
|
-
</section>
|
|
772
|
-
<section class="panel">
|
|
773
|
-
<h2 class="section-title">Run Summary</h2>
|
|
774
|
-
<div style="padding: 16px;" class="stack">
|
|
775
|
-
${renderJsonBlock("Session", data.session || {})}
|
|
776
|
-
${renderJsonBlock("Task", task || { workflow: "generic", id: null })}
|
|
777
|
-
${renderJsonBlock("Conditions", data.conditions || {})}
|
|
778
|
-
${renderJsonBlock("Activity", data.activity || {})}
|
|
779
|
-
</div>
|
|
780
|
-
</section>
|
|
781
|
-
<section class="panel" style="margin-top: 24px;">
|
|
782
|
-
<h2 class="section-title">Dynamic Extension Telemetry</h2>
|
|
783
|
-
<div style="padding: 16px;" class="stack">
|
|
784
|
-
${Object.entries(data.telemetry?.extensions || {}).length ? Object.entries(data.telemetry.extensions).map(([name, ext]) => renderJsonBlock(`${name} (${formatTelemetryStatus(ext?.status)})`, { summary: ext?.summary || null, ...ext?.details })).join("") : `<div class="empty">No extension telemetry captured for this run.</div>`}
|
|
785
|
-
</div>
|
|
786
|
-
</section>
|
|
787
|
-
<section class="panel" style="margin-top: 24px;">
|
|
788
|
-
<h2 class="section-title">Timeline</h2>
|
|
789
|
-
<div style="padding: 16px;">
|
|
790
|
-
<div class="timeline">${visibleEvents.map(renderEventCard).join("") || `<div class="empty">No visible events captured.</div>`}</div>
|
|
791
|
-
</div>
|
|
792
|
-
</section>
|
|
793
|
-
<footer class="footer-note stack">
|
|
794
|
-
<div class="footer-brand">notrace</div>
|
|
795
|
-
<div class="footer-tagline">Local-first retrospective engine</div>
|
|
796
|
-
<div class="footer-meta"><a href="https://opensource.org/licenses/MIT">MIT</a></div>
|
|
797
|
-
</footer>
|
|
798
|
-
</div>`;
|
|
799
|
-
return shell(`notrace - ${data.traceId}`, body, copyScript());
|
|
800
|
-
}
|