gsd-pi 2.33.0-dev.bafba33 → 2.33.1-dev.235d83a

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.
Files changed (751) hide show
  1. package/README.md +13 -18
  2. package/dist/cli.js +16 -0
  3. package/dist/resource-loader.d.ts +2 -0
  4. package/dist/resource-loader.js +36 -3
  5. package/dist/resources/extensions/ask-user-questions.js +217 -0
  6. package/dist/resources/extensions/async-jobs/async-bash-tool.js +180 -0
  7. package/dist/resources/extensions/async-jobs/await-tool.js +90 -0
  8. package/dist/resources/extensions/async-jobs/cancel-job-tool.js +28 -0
  9. package/dist/resources/extensions/async-jobs/index.js +119 -0
  10. package/dist/resources/extensions/async-jobs/job-manager.js +159 -0
  11. package/dist/resources/extensions/aws-auth/index.js +138 -0
  12. package/dist/resources/extensions/bg-shell/bg-shell-command.js +182 -0
  13. package/dist/resources/extensions/bg-shell/bg-shell-lifecycle.js +343 -0
  14. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +831 -0
  15. package/dist/resources/extensions/bg-shell/index.js +41 -0
  16. package/dist/resources/extensions/bg-shell/interaction.js +160 -0
  17. package/dist/resources/extensions/bg-shell/output-formatter.js +245 -0
  18. package/dist/resources/extensions/bg-shell/overlay.js +378 -0
  19. package/dist/resources/extensions/bg-shell/process-manager.js +413 -0
  20. package/dist/resources/extensions/bg-shell/readiness-detector.js +109 -0
  21. package/dist/resources/extensions/bg-shell/types.js +96 -0
  22. package/dist/resources/extensions/bg-shell/utilities.js +50 -0
  23. package/dist/resources/extensions/browser-tools/capture.js +179 -0
  24. package/dist/resources/extensions/browser-tools/core.js +899 -0
  25. package/dist/resources/extensions/browser-tools/{evaluate-helpers.ts → evaluate-helpers.js} +0 -1
  26. package/dist/resources/extensions/browser-tools/index.js +123 -0
  27. package/dist/resources/extensions/browser-tools/lifecycle.js +222 -0
  28. package/dist/resources/extensions/browser-tools/refs.js +254 -0
  29. package/dist/resources/extensions/browser-tools/settle.js +173 -0
  30. package/dist/resources/extensions/browser-tools/state.js +126 -0
  31. package/dist/resources/extensions/browser-tools/tools/action-cache.js +179 -0
  32. package/dist/resources/extensions/browser-tools/tools/assertions.js +320 -0
  33. package/dist/resources/extensions/browser-tools/tools/codegen.js +242 -0
  34. package/dist/resources/extensions/browser-tools/tools/device.js +162 -0
  35. package/dist/resources/extensions/browser-tools/tools/extract.js +191 -0
  36. package/dist/resources/extensions/browser-tools/tools/forms.js +710 -0
  37. package/dist/resources/extensions/browser-tools/tools/injection-detect.js +178 -0
  38. package/dist/resources/extensions/browser-tools/tools/inspection.js +426 -0
  39. package/dist/resources/extensions/browser-tools/tools/intent.js +556 -0
  40. package/dist/resources/extensions/browser-tools/tools/interaction.js +776 -0
  41. package/dist/resources/extensions/browser-tools/tools/navigation.js +208 -0
  42. package/dist/resources/extensions/browser-tools/tools/network-mock.js +194 -0
  43. package/dist/resources/extensions/browser-tools/tools/pages.js +280 -0
  44. package/dist/resources/extensions/browser-tools/tools/pdf.js +74 -0
  45. package/dist/resources/extensions/browser-tools/tools/refs.js +485 -0
  46. package/dist/resources/extensions/browser-tools/tools/screenshot.js +87 -0
  47. package/dist/resources/extensions/browser-tools/tools/session.js +375 -0
  48. package/dist/resources/extensions/browser-tools/tools/state-persistence.js +180 -0
  49. package/dist/resources/extensions/browser-tools/tools/visual-diff.js +174 -0
  50. package/dist/resources/extensions/browser-tools/tools/wait.js +201 -0
  51. package/dist/resources/extensions/browser-tools/tools/zoom.js +90 -0
  52. package/dist/resources/extensions/browser-tools/utils.js +490 -0
  53. package/dist/resources/extensions/context7/index.js +337 -0
  54. package/dist/resources/extensions/get-secrets-from-user.js +492 -0
  55. package/dist/resources/extensions/google-search/index.js +373 -0
  56. package/dist/resources/extensions/gsd/activity-log.js +146 -0
  57. package/dist/resources/extensions/gsd/atomic-write.js +38 -0
  58. package/dist/resources/extensions/gsd/auto/session.js +187 -0
  59. package/dist/resources/extensions/gsd/auto-budget.js +30 -0
  60. package/dist/resources/extensions/gsd/{auto-constants.ts → auto-constants.js} +0 -1
  61. package/dist/resources/extensions/gsd/auto-dashboard.js +509 -0
  62. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +180 -0
  63. package/dist/resources/extensions/gsd/auto-dispatch.js +382 -0
  64. package/dist/resources/extensions/gsd/auto-idempotency.js +106 -0
  65. package/dist/resources/extensions/gsd/auto-model-selection.js +133 -0
  66. package/dist/resources/extensions/gsd/auto-observability.js +54 -0
  67. package/dist/resources/extensions/gsd/auto-post-unit.js +495 -0
  68. package/dist/resources/extensions/gsd/auto-prompts.js +1121 -0
  69. package/dist/resources/extensions/gsd/auto-recovery.js +550 -0
  70. package/dist/resources/extensions/gsd/auto-start.js +393 -0
  71. package/dist/resources/extensions/gsd/auto-stuck-detection.js +165 -0
  72. package/dist/resources/extensions/gsd/{auto-supervisor.ts → auto-supervisor.js} +29 -32
  73. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +183 -0
  74. package/dist/resources/extensions/gsd/auto-timers.js +181 -0
  75. package/dist/resources/extensions/gsd/auto-tool-tracking.js +50 -0
  76. package/dist/resources/extensions/gsd/auto-unit-closeout.js +30 -0
  77. package/dist/resources/extensions/gsd/auto-verification.js +209 -0
  78. package/dist/resources/extensions/gsd/auto-worktree.js +594 -0
  79. package/dist/resources/extensions/gsd/auto.js +1485 -0
  80. package/dist/resources/extensions/gsd/{cache.ts → cache.js} +5 -7
  81. package/dist/resources/extensions/gsd/captures.js +348 -0
  82. package/dist/resources/extensions/gsd/claude-import.js +540 -0
  83. package/dist/resources/extensions/gsd/collision-diagnostics.js +226 -0
  84. package/dist/resources/extensions/gsd/commands-bootstrap.js +223 -0
  85. package/dist/resources/extensions/gsd/commands-config.js +89 -0
  86. package/dist/resources/extensions/gsd/commands-extensions.js +259 -0
  87. package/dist/resources/extensions/gsd/commands-handlers.js +310 -0
  88. package/dist/resources/extensions/gsd/commands-inspect.js +70 -0
  89. package/dist/resources/extensions/gsd/commands-logs.js +468 -0
  90. package/dist/resources/extensions/gsd/commands-maintenance.js +185 -0
  91. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +660 -0
  92. package/dist/resources/extensions/gsd/commands-workflow-templates.js +408 -0
  93. package/dist/resources/extensions/gsd/commands.js +1024 -0
  94. package/dist/resources/extensions/gsd/complexity-classifier.js +257 -0
  95. package/dist/resources/extensions/gsd/{constants.ts → constants.js} +0 -6
  96. package/dist/resources/extensions/gsd/context-budget.js +187 -0
  97. package/dist/resources/extensions/gsd/context-store.js +165 -0
  98. package/dist/resources/extensions/gsd/crash-recovery.js +110 -0
  99. package/dist/resources/extensions/gsd/dashboard-overlay.js +562 -0
  100. package/dist/resources/extensions/gsd/db-writer.js +298 -0
  101. package/dist/resources/extensions/gsd/debug-logger.js +161 -0
  102. package/dist/resources/extensions/gsd/detection.js +373 -0
  103. package/dist/resources/extensions/gsd/diff-context.js +168 -0
  104. package/dist/resources/extensions/gsd/dispatch-guard.js +76 -0
  105. package/dist/resources/extensions/gsd/doctor-checks.js +563 -0
  106. package/dist/resources/extensions/gsd/doctor-environment.js +429 -0
  107. package/dist/resources/extensions/gsd/doctor-format.js +71 -0
  108. package/dist/resources/extensions/gsd/doctor-proactive.js +239 -0
  109. package/dist/resources/extensions/gsd/doctor-providers.js +292 -0
  110. package/dist/resources/extensions/gsd/doctor-types.js +12 -0
  111. package/dist/resources/extensions/gsd/doctor.js +672 -0
  112. package/dist/resources/extensions/gsd/error-utils.js +6 -0
  113. package/dist/resources/extensions/gsd/{errors.ts → errors.js} +6 -11
  114. package/dist/resources/extensions/gsd/exit-command.js +11 -0
  115. package/dist/resources/extensions/gsd/{export-html.ts → export-html.js} +482 -614
  116. package/dist/resources/extensions/gsd/export.js +268 -0
  117. package/dist/resources/extensions/gsd/file-watcher.js +76 -0
  118. package/dist/resources/extensions/gsd/files.js +937 -0
  119. package/dist/resources/extensions/gsd/forensics.js +511 -0
  120. package/dist/resources/extensions/gsd/{git-constants.ts → git-constants.js} +4 -5
  121. package/dist/resources/extensions/gsd/git-self-heal.js +113 -0
  122. package/dist/resources/extensions/gsd/git-service.js +460 -0
  123. package/dist/resources/extensions/gsd/{gitignore.ts → gitignore.js} +98 -125
  124. package/dist/resources/extensions/gsd/gsd-db.js +590 -0
  125. package/dist/resources/extensions/gsd/guided-flow-queue.js +366 -0
  126. package/dist/resources/extensions/gsd/guided-flow.js +1149 -0
  127. package/dist/resources/extensions/gsd/health-widget.js +141 -0
  128. package/dist/resources/extensions/gsd/history.js +118 -0
  129. package/dist/resources/extensions/gsd/index.js +1250 -0
  130. package/dist/resources/extensions/gsd/init-wizard.js +479 -0
  131. package/dist/resources/extensions/gsd/json-persistence.js +62 -0
  132. package/dist/resources/extensions/gsd/{jsonl-utils.ts → jsonl-utils.js} +10 -7
  133. package/dist/resources/extensions/gsd/key-manager.js +829 -0
  134. package/dist/resources/extensions/gsd/marketplace-discovery.js +356 -0
  135. package/dist/resources/extensions/gsd/md-importer.js +440 -0
  136. package/dist/resources/extensions/gsd/mechanical-completion.js +351 -0
  137. package/dist/resources/extensions/gsd/memory-extractor.js +295 -0
  138. package/dist/resources/extensions/gsd/memory-store.js +351 -0
  139. package/dist/resources/extensions/gsd/metrics.js +377 -0
  140. package/dist/resources/extensions/gsd/migrate/command.js +157 -0
  141. package/dist/resources/extensions/gsd/migrate/index.js +7 -0
  142. package/dist/resources/extensions/gsd/migrate/parser.js +268 -0
  143. package/dist/resources/extensions/gsd/migrate/parsers.js +477 -0
  144. package/dist/resources/extensions/gsd/migrate/preview.js +47 -0
  145. package/dist/resources/extensions/gsd/migrate/transformer.js +278 -0
  146. package/dist/resources/extensions/gsd/migrate/types.js +4 -0
  147. package/dist/resources/extensions/gsd/migrate/validator.js +41 -0
  148. package/dist/resources/extensions/gsd/migrate/writer.js +477 -0
  149. package/dist/resources/extensions/gsd/migrate-external.js +130 -0
  150. package/dist/resources/extensions/gsd/milestone-actions.js +111 -0
  151. package/dist/resources/extensions/gsd/{milestone-ids.ts → milestone-ids.js} +50 -63
  152. package/dist/resources/extensions/gsd/model-cost-table.js +48 -0
  153. package/dist/resources/extensions/gsd/model-router.js +187 -0
  154. package/dist/resources/extensions/gsd/namespaced-registry.js +322 -0
  155. package/dist/resources/extensions/gsd/namespaced-resolver.js +176 -0
  156. package/dist/resources/extensions/gsd/native-git-bridge.js +842 -0
  157. package/dist/resources/extensions/gsd/native-parser-bridge.js +156 -0
  158. package/dist/resources/extensions/gsd/notifications.js +58 -0
  159. package/dist/resources/extensions/gsd/observability-validator.js +398 -0
  160. package/dist/resources/extensions/gsd/parallel-eligibility.js +182 -0
  161. package/dist/resources/extensions/gsd/parallel-merge.js +121 -0
  162. package/dist/resources/extensions/gsd/parallel-orchestrator.js +687 -0
  163. package/dist/resources/extensions/gsd/paths.js +414 -0
  164. package/dist/resources/extensions/gsd/plugin-importer.js +254 -0
  165. package/dist/resources/extensions/gsd/post-unit-hooks.js +433 -0
  166. package/dist/resources/extensions/gsd/preferences-models.js +294 -0
  167. package/dist/resources/extensions/gsd/preferences-skills.js +154 -0
  168. package/dist/resources/extensions/gsd/preferences-types.js +73 -0
  169. package/dist/resources/extensions/gsd/preferences-validation.js +607 -0
  170. package/dist/resources/extensions/gsd/preferences.js +325 -0
  171. package/dist/resources/extensions/gsd/progress-score.js +197 -0
  172. package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +150 -0
  173. package/dist/resources/extensions/gsd/prompt-compressor.js +393 -0
  174. package/dist/resources/extensions/gsd/prompt-loader.js +119 -0
  175. package/dist/resources/extensions/gsd/prompt-ordering.js +170 -0
  176. package/dist/resources/extensions/gsd/provider-error-pause.js +60 -0
  177. package/dist/resources/extensions/gsd/queue-order.js +178 -0
  178. package/dist/resources/extensions/gsd/queue-reorder-ui.js +234 -0
  179. package/dist/resources/extensions/gsd/quick.js +230 -0
  180. package/dist/resources/extensions/gsd/repo-identity.js +187 -0
  181. package/dist/resources/extensions/gsd/{reports.ts → reports.js} +146 -241
  182. package/dist/resources/extensions/gsd/{resource-version.ts → resource-version.js} +50 -53
  183. package/dist/resources/extensions/gsd/roadmap-slices.js +130 -0
  184. package/dist/resources/extensions/gsd/routing-history.js +210 -0
  185. package/dist/resources/extensions/gsd/safe-fs.js +52 -0
  186. package/dist/resources/extensions/gsd/semantic-chunker.js +254 -0
  187. package/dist/resources/extensions/gsd/session-forensics.js +427 -0
  188. package/dist/resources/extensions/gsd/session-lock.js +376 -0
  189. package/dist/resources/extensions/gsd/session-status-io.js +134 -0
  190. package/dist/resources/extensions/gsd/skill-discovery.js +121 -0
  191. package/dist/resources/extensions/gsd/skill-health.js +324 -0
  192. package/dist/resources/extensions/gsd/{skill-telemetry.ts → skill-telemetry.js} +58 -74
  193. package/dist/resources/extensions/gsd/state.js +653 -0
  194. package/dist/resources/extensions/gsd/structured-data-formatter.js +97 -0
  195. package/dist/resources/extensions/gsd/summary-distiller.js +212 -0
  196. package/dist/resources/extensions/gsd/token-counter.js +54 -0
  197. package/dist/resources/extensions/gsd/triage-resolution.js +217 -0
  198. package/dist/resources/extensions/gsd/triage-ui.js +125 -0
  199. package/dist/resources/extensions/gsd/types.js +4 -0
  200. package/dist/resources/extensions/gsd/undo.js +210 -0
  201. package/dist/resources/extensions/gsd/unit-id.js +7 -0
  202. package/dist/resources/extensions/gsd/unit-runtime.js +130 -0
  203. package/dist/resources/extensions/gsd/validate-directory.js +143 -0
  204. package/dist/resources/extensions/gsd/verification-evidence.js +123 -0
  205. package/dist/resources/extensions/gsd/verification-gate.js +543 -0
  206. package/dist/resources/extensions/gsd/visualizer-data.js +612 -0
  207. package/dist/resources/extensions/gsd/visualizer-overlay.js +501 -0
  208. package/dist/resources/extensions/gsd/visualizer-views.js +893 -0
  209. package/dist/resources/extensions/gsd/workflow-templates.js +188 -0
  210. package/dist/resources/extensions/gsd/workspace-index.js +139 -0
  211. package/dist/resources/extensions/gsd/worktree-command-bootstrap.js +40 -0
  212. package/dist/resources/extensions/gsd/worktree-command.js +676 -0
  213. package/dist/resources/extensions/gsd/worktree-manager.js +359 -0
  214. package/dist/resources/extensions/gsd/worktree.js +236 -0
  215. package/dist/resources/extensions/mac-tools/index.js +768 -0
  216. package/dist/resources/extensions/mcp-client/index.js +363 -0
  217. package/dist/resources/extensions/package.json +3 -0
  218. package/dist/resources/extensions/remote-questions/config.js +70 -0
  219. package/dist/resources/extensions/remote-questions/discord-adapter.js +134 -0
  220. package/dist/resources/extensions/remote-questions/format.js +234 -0
  221. package/dist/resources/extensions/remote-questions/http-client.js +43 -0
  222. package/dist/resources/extensions/remote-questions/manager.js +156 -0
  223. package/dist/resources/extensions/remote-questions/{mod.ts → mod.js} +1 -10
  224. package/dist/resources/extensions/remote-questions/notify.js +89 -0
  225. package/dist/resources/extensions/remote-questions/remote-command.js +453 -0
  226. package/dist/resources/extensions/remote-questions/slack-adapter.js +123 -0
  227. package/dist/resources/extensions/remote-questions/status.js +25 -0
  228. package/dist/resources/extensions/remote-questions/store.js +70 -0
  229. package/dist/resources/extensions/remote-questions/telegram-adapter.js +123 -0
  230. package/dist/resources/extensions/remote-questions/types.js +5 -0
  231. package/dist/resources/extensions/search-the-web/cache.js +74 -0
  232. package/dist/resources/extensions/search-the-web/command-search-provider.js +79 -0
  233. package/dist/resources/extensions/search-the-web/format.js +161 -0
  234. package/dist/resources/extensions/search-the-web/http.js +178 -0
  235. package/dist/resources/extensions/search-the-web/index.js +41 -0
  236. package/dist/resources/extensions/search-the-web/native-search.js +166 -0
  237. package/dist/resources/extensions/search-the-web/provider.js +143 -0
  238. package/dist/resources/extensions/search-the-web/tavily.js +82 -0
  239. package/dist/resources/extensions/search-the-web/tool-fetch-page.js +452 -0
  240. package/dist/resources/extensions/search-the-web/tool-llm-context.js +455 -0
  241. package/dist/resources/extensions/search-the-web/tool-search.js +482 -0
  242. package/dist/resources/extensions/search-the-web/url-utils.js +121 -0
  243. package/dist/resources/extensions/shared/confirm-ui.js +96 -0
  244. package/dist/resources/extensions/shared/{format-utils.ts → format-utils.js} +85 -91
  245. package/dist/resources/extensions/shared/frontmatter.js +109 -0
  246. package/dist/resources/extensions/shared/interview-ui.js +569 -0
  247. package/dist/resources/extensions/shared/{mod.ts → mod.js} +2 -24
  248. package/dist/resources/extensions/shared/next-action-ui.js +168 -0
  249. package/dist/resources/extensions/shared/{path-display.ts → path-display.js} +2 -3
  250. package/dist/resources/extensions/shared/sanitize.js +17 -0
  251. package/dist/resources/extensions/shared/terminal.js +21 -0
  252. package/dist/resources/extensions/shared/ui.js +245 -0
  253. package/dist/resources/extensions/shared/wizard-ui.js +478 -0
  254. package/dist/resources/extensions/slash-commands/audit.js +72 -0
  255. package/dist/resources/extensions/slash-commands/clear.js +8 -0
  256. package/dist/resources/extensions/slash-commands/create-extension.js +264 -0
  257. package/dist/resources/extensions/slash-commands/create-slash-command.js +208 -0
  258. package/dist/resources/extensions/slash-commands/index.js +10 -0
  259. package/dist/resources/extensions/subagent/agents.js +103 -0
  260. package/dist/resources/extensions/subagent/index.js +905 -0
  261. package/dist/resources/extensions/subagent/isolation.js +384 -0
  262. package/dist/resources/extensions/subagent/worker-registry.js +73 -0
  263. package/dist/resources/extensions/ttsr/index.js +144 -0
  264. package/dist/resources/extensions/ttsr/rule-loader.js +70 -0
  265. package/dist/resources/extensions/ttsr/ttsr-manager.js +380 -0
  266. package/dist/resources/extensions/universal-config/discovery.js +94 -0
  267. package/dist/resources/extensions/universal-config/format.js +178 -0
  268. package/dist/resources/extensions/universal-config/index.js +99 -0
  269. package/dist/resources/extensions/universal-config/scanners.js +574 -0
  270. package/dist/resources/extensions/universal-config/tools.js +57 -0
  271. package/dist/resources/extensions/universal-config/types.js +8 -0
  272. package/dist/resources/extensions/voice/index.js +247 -0
  273. package/dist/startup-timings.d.ts +2 -0
  274. package/dist/startup-timings.js +22 -0
  275. package/package.json +1 -1
  276. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
  277. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
  278. package/packages/pi-coding-agent/dist/core/extensions/index.js +1 -1
  279. package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
  280. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +1 -0
  281. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  282. package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
  283. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  284. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  285. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  286. package/packages/pi-coding-agent/dist/index.js +1 -1
  287. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  288. package/packages/pi-coding-agent/package.json +1 -1
  289. package/packages/pi-coding-agent/src/core/extensions/index.ts +1 -0
  290. package/packages/pi-coding-agent/src/core/extensions/loader.ts +35 -4
  291. package/packages/pi-coding-agent/src/index.ts +1 -0
  292. package/pkg/package.json +1 -1
  293. package/src/resources/extensions/bg-shell/index.ts +41 -46
  294. package/src/resources/extensions/browser-tools/index.ts +156 -67
  295. package/src/resources/extensions/gsd/auto-supervisor.ts +10 -5
  296. package/src/resources/extensions/gsd/auto-verification.ts +20 -2
  297. package/src/resources/extensions/gsd/auto-worktree.ts +136 -1
  298. package/src/resources/extensions/gsd/commands-bootstrap.ts +252 -0
  299. package/src/resources/extensions/gsd/commands.ts +53 -33
  300. package/src/resources/extensions/gsd/git-service.ts +24 -11
  301. package/src/resources/extensions/gsd/index.ts +106 -34
  302. package/src/resources/extensions/gsd/quick.ts +52 -3
  303. package/src/resources/extensions/gsd/repo-identity.ts +56 -22
  304. package/src/resources/extensions/gsd/session-lock.ts +80 -16
  305. package/src/resources/extensions/gsd/tests/auto-dispatch-loop.test.ts +691 -0
  306. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +317 -0
  307. package/src/resources/extensions/gsd/tests/git-service.test.ts +16 -7
  308. package/src/resources/extensions/gsd/tests/loop-regression.test.ts +40 -5
  309. package/src/resources/extensions/gsd/tests/quick-branch-lifecycle.test.ts +281 -0
  310. package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +71 -0
  311. package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +358 -0
  312. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +216 -0
  313. package/src/resources/extensions/gsd/tests/session-lock.test.ts +119 -0
  314. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +65 -0
  315. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +206 -0
  316. package/src/resources/extensions/gsd/types.ts +4 -0
  317. package/src/resources/extensions/gsd/verification-evidence.ts +1 -1
  318. package/src/resources/extensions/gsd/verification-gate.ts +23 -2
  319. package/src/resources/extensions/gsd/worktree-command-bootstrap.ts +46 -0
  320. package/src/resources/extensions/gsd/worktree-command.ts +17 -11
  321. package/src/resources/extensions/package.json +3 -0
  322. package/src/resources/extensions/search-the-web/command-search-provider.ts +1 -1
  323. package/src/resources/extensions/search-the-web/index.ts +35 -52
  324. package/src/resources/extensions/search-the-web/tavily.ts +1 -1
  325. package/dist/resources/extensions/ask-user-questions.ts +0 -290
  326. package/dist/resources/extensions/async-jobs/async-bash-tool.ts +0 -212
  327. package/dist/resources/extensions/async-jobs/await-tool.ts +0 -103
  328. package/dist/resources/extensions/async-jobs/cancel-job-tool.ts +0 -35
  329. package/dist/resources/extensions/async-jobs/index.ts +0 -141
  330. package/dist/resources/extensions/async-jobs/job-manager.ts +0 -211
  331. package/dist/resources/extensions/aws-auth/index.ts +0 -144
  332. package/dist/resources/extensions/bg-shell/bg-shell-command.ts +0 -219
  333. package/dist/resources/extensions/bg-shell/bg-shell-lifecycle.ts +0 -400
  334. package/dist/resources/extensions/bg-shell/bg-shell-tool.ts +0 -985
  335. package/dist/resources/extensions/bg-shell/index.ts +0 -59
  336. package/dist/resources/extensions/bg-shell/interaction.ts +0 -198
  337. package/dist/resources/extensions/bg-shell/output-formatter.ts +0 -279
  338. package/dist/resources/extensions/bg-shell/overlay.ts +0 -437
  339. package/dist/resources/extensions/bg-shell/process-manager.ts +0 -464
  340. package/dist/resources/extensions/bg-shell/readiness-detector.ts +0 -126
  341. package/dist/resources/extensions/bg-shell/types.ts +0 -303
  342. package/dist/resources/extensions/bg-shell/utilities.ts +0 -57
  343. package/dist/resources/extensions/browser-tools/capture.ts +0 -199
  344. package/dist/resources/extensions/browser-tools/core.ts +0 -1196
  345. package/dist/resources/extensions/browser-tools/index.ts +0 -71
  346. package/dist/resources/extensions/browser-tools/lifecycle.ts +0 -270
  347. package/dist/resources/extensions/browser-tools/refs.ts +0 -264
  348. package/dist/resources/extensions/browser-tools/settle.ts +0 -197
  349. package/dist/resources/extensions/browser-tools/state.ts +0 -408
  350. package/dist/resources/extensions/browser-tools/tools/action-cache.ts +0 -216
  351. package/dist/resources/extensions/browser-tools/tools/assertions.ts +0 -342
  352. package/dist/resources/extensions/browser-tools/tools/codegen.ts +0 -274
  353. package/dist/resources/extensions/browser-tools/tools/device.ts +0 -183
  354. package/dist/resources/extensions/browser-tools/tools/extract.ts +0 -229
  355. package/dist/resources/extensions/browser-tools/tools/forms.ts +0 -801
  356. package/dist/resources/extensions/browser-tools/tools/injection-detect.ts +0 -221
  357. package/dist/resources/extensions/browser-tools/tools/inspection.ts +0 -492
  358. package/dist/resources/extensions/browser-tools/tools/intent.ts +0 -614
  359. package/dist/resources/extensions/browser-tools/tools/interaction.ts +0 -865
  360. package/dist/resources/extensions/browser-tools/tools/navigation.ts +0 -232
  361. package/dist/resources/extensions/browser-tools/tools/network-mock.ts +0 -244
  362. package/dist/resources/extensions/browser-tools/tools/pages.ts +0 -303
  363. package/dist/resources/extensions/browser-tools/tools/pdf.ts +0 -92
  364. package/dist/resources/extensions/browser-tools/tools/refs.ts +0 -541
  365. package/dist/resources/extensions/browser-tools/tools/screenshot.ts +0 -101
  366. package/dist/resources/extensions/browser-tools/tools/session.ts +0 -400
  367. package/dist/resources/extensions/browser-tools/tools/state-persistence.ts +0 -202
  368. package/dist/resources/extensions/browser-tools/tools/visual-diff.ts +0 -209
  369. package/dist/resources/extensions/browser-tools/tools/wait.ts +0 -247
  370. package/dist/resources/extensions/browser-tools/tools/zoom.ts +0 -104
  371. package/dist/resources/extensions/browser-tools/utils.ts +0 -660
  372. package/dist/resources/extensions/context7/index.ts +0 -428
  373. package/dist/resources/extensions/get-secrets-from-user.ts +0 -607
  374. package/dist/resources/extensions/google-search/index.ts +0 -466
  375. package/dist/resources/extensions/gsd/activity-log.ts +0 -162
  376. package/dist/resources/extensions/gsd/atomic-write.ts +0 -35
  377. package/dist/resources/extensions/gsd/auto/session.ts +0 -236
  378. package/dist/resources/extensions/gsd/auto-budget.ts +0 -32
  379. package/dist/resources/extensions/gsd/auto-dashboard.ts +0 -626
  380. package/dist/resources/extensions/gsd/auto-direct-dispatch.ts +0 -224
  381. package/dist/resources/extensions/gsd/auto-dispatch.ts +0 -409
  382. package/dist/resources/extensions/gsd/auto-idempotency.ts +0 -151
  383. package/dist/resources/extensions/gsd/auto-model-selection.ts +0 -179
  384. package/dist/resources/extensions/gsd/auto-observability.ts +0 -72
  385. package/dist/resources/extensions/gsd/auto-post-unit.ts +0 -618
  386. package/dist/resources/extensions/gsd/auto-prompts.ts +0 -1273
  387. package/dist/resources/extensions/gsd/auto-recovery.ts +0 -578
  388. package/dist/resources/extensions/gsd/auto-start.ts +0 -483
  389. package/dist/resources/extensions/gsd/auto-stuck-detection.ts +0 -221
  390. package/dist/resources/extensions/gsd/auto-timeout-recovery.ts +0 -263
  391. package/dist/resources/extensions/gsd/auto-timers.ts +0 -224
  392. package/dist/resources/extensions/gsd/auto-tool-tracking.ts +0 -54
  393. package/dist/resources/extensions/gsd/auto-unit-closeout.ts +0 -48
  394. package/dist/resources/extensions/gsd/auto-verification.ts +0 -229
  395. package/dist/resources/extensions/gsd/auto-worktree.ts +0 -524
  396. package/dist/resources/extensions/gsd/auto.ts +0 -1834
  397. package/dist/resources/extensions/gsd/captures.ts +0 -427
  398. package/dist/resources/extensions/gsd/claude-import.ts +0 -656
  399. package/dist/resources/extensions/gsd/collision-diagnostics.ts +0 -332
  400. package/dist/resources/extensions/gsd/commands-config.ts +0 -102
  401. package/dist/resources/extensions/gsd/commands-extensions.ts +0 -328
  402. package/dist/resources/extensions/gsd/commands-handlers.ts +0 -395
  403. package/dist/resources/extensions/gsd/commands-inspect.ts +0 -91
  404. package/dist/resources/extensions/gsd/commands-logs.ts +0 -536
  405. package/dist/resources/extensions/gsd/commands-maintenance.ts +0 -206
  406. package/dist/resources/extensions/gsd/commands-prefs-wizard.ts +0 -780
  407. package/dist/resources/extensions/gsd/commands-workflow-templates.ts +0 -543
  408. package/dist/resources/extensions/gsd/commands.ts +0 -1137
  409. package/dist/resources/extensions/gsd/complexity-classifier.ts +0 -320
  410. package/dist/resources/extensions/gsd/context-budget.ts +0 -266
  411. package/dist/resources/extensions/gsd/context-store.ts +0 -195
  412. package/dist/resources/extensions/gsd/crash-recovery.ts +0 -121
  413. package/dist/resources/extensions/gsd/dashboard-overlay.ts +0 -681
  414. package/dist/resources/extensions/gsd/db-writer.ts +0 -360
  415. package/dist/resources/extensions/gsd/debug-logger.ts +0 -178
  416. package/dist/resources/extensions/gsd/detection.ts +0 -470
  417. package/dist/resources/extensions/gsd/diff-context.ts +0 -214
  418. package/dist/resources/extensions/gsd/dispatch-guard.ts +0 -81
  419. package/dist/resources/extensions/gsd/doctor-checks.ts +0 -612
  420. package/dist/resources/extensions/gsd/doctor-environment.ts +0 -497
  421. package/dist/resources/extensions/gsd/doctor-format.ts +0 -78
  422. package/dist/resources/extensions/gsd/doctor-proactive.ts +0 -292
  423. package/dist/resources/extensions/gsd/doctor-providers.ts +0 -343
  424. package/dist/resources/extensions/gsd/doctor-types.ts +0 -87
  425. package/dist/resources/extensions/gsd/doctor.ts +0 -722
  426. package/dist/resources/extensions/gsd/error-utils.ts +0 -6
  427. package/dist/resources/extensions/gsd/exit-command.ts +0 -18
  428. package/dist/resources/extensions/gsd/export.ts +0 -317
  429. package/dist/resources/extensions/gsd/file-watcher.ts +0 -97
  430. package/dist/resources/extensions/gsd/files.ts +0 -1058
  431. package/dist/resources/extensions/gsd/forensics.ts +0 -629
  432. package/dist/resources/extensions/gsd/git-self-heal.ts +0 -127
  433. package/dist/resources/extensions/gsd/git-service.ts +0 -580
  434. package/dist/resources/extensions/gsd/gsd-db.ts +0 -685
  435. package/dist/resources/extensions/gsd/guided-flow-queue.ts +0 -440
  436. package/dist/resources/extensions/gsd/guided-flow.ts +0 -1303
  437. package/dist/resources/extensions/gsd/health-widget.ts +0 -167
  438. package/dist/resources/extensions/gsd/history.ts +0 -143
  439. package/dist/resources/extensions/gsd/index.ts +0 -1390
  440. package/dist/resources/extensions/gsd/init-wizard.ts +0 -587
  441. package/dist/resources/extensions/gsd/json-persistence.ts +0 -67
  442. package/dist/resources/extensions/gsd/key-manager.ts +0 -996
  443. package/dist/resources/extensions/gsd/marketplace-discovery.ts +0 -508
  444. package/dist/resources/extensions/gsd/md-importer.ts +0 -527
  445. package/dist/resources/extensions/gsd/mechanical-completion.ts +0 -430
  446. package/dist/resources/extensions/gsd/memory-extractor.ts +0 -352
  447. package/dist/resources/extensions/gsd/memory-store.ts +0 -441
  448. package/dist/resources/extensions/gsd/metrics.ts +0 -532
  449. package/dist/resources/extensions/gsd/migrate/command.ts +0 -219
  450. package/dist/resources/extensions/gsd/migrate/index.ts +0 -42
  451. package/dist/resources/extensions/gsd/migrate/parser.ts +0 -323
  452. package/dist/resources/extensions/gsd/migrate/parsers.ts +0 -539
  453. package/dist/resources/extensions/gsd/migrate/preview.ts +0 -48
  454. package/dist/resources/extensions/gsd/migrate/transformer.ts +0 -346
  455. package/dist/resources/extensions/gsd/migrate/types.ts +0 -370
  456. package/dist/resources/extensions/gsd/migrate/validator.ts +0 -55
  457. package/dist/resources/extensions/gsd/migrate/writer.ts +0 -579
  458. package/dist/resources/extensions/gsd/migrate-external.ts +0 -140
  459. package/dist/resources/extensions/gsd/milestone-actions.ts +0 -126
  460. package/dist/resources/extensions/gsd/model-cost-table.ts +0 -65
  461. package/dist/resources/extensions/gsd/model-router.ts +0 -256
  462. package/dist/resources/extensions/gsd/namespaced-registry.ts +0 -467
  463. package/dist/resources/extensions/gsd/namespaced-resolver.ts +0 -307
  464. package/dist/resources/extensions/gsd/native-git-bridge.ts +0 -1041
  465. package/dist/resources/extensions/gsd/native-parser-bridge.ts +0 -267
  466. package/dist/resources/extensions/gsd/notifications.ts +0 -87
  467. package/dist/resources/extensions/gsd/observability-validator.ts +0 -429
  468. package/dist/resources/extensions/gsd/parallel-eligibility.ts +0 -233
  469. package/dist/resources/extensions/gsd/parallel-merge.ts +0 -157
  470. package/dist/resources/extensions/gsd/parallel-orchestrator.ts +0 -826
  471. package/dist/resources/extensions/gsd/paths.ts +0 -449
  472. package/dist/resources/extensions/gsd/plugin-importer.ts +0 -411
  473. package/dist/resources/extensions/gsd/post-unit-hooks.ts +0 -520
  474. package/dist/resources/extensions/gsd/preferences-models.ts +0 -329
  475. package/dist/resources/extensions/gsd/preferences-skills.ts +0 -169
  476. package/dist/resources/extensions/gsd/preferences-types.ts +0 -229
  477. package/dist/resources/extensions/gsd/preferences-validation.ts +0 -590
  478. package/dist/resources/extensions/gsd/preferences.ts +0 -416
  479. package/dist/resources/extensions/gsd/progress-score.ts +0 -273
  480. package/dist/resources/extensions/gsd/prompt-cache-optimizer.ts +0 -213
  481. package/dist/resources/extensions/gsd/prompt-compressor.ts +0 -508
  482. package/dist/resources/extensions/gsd/prompt-loader.ts +0 -130
  483. package/dist/resources/extensions/gsd/prompt-ordering.ts +0 -200
  484. package/dist/resources/extensions/gsd/provider-error-pause.ts +0 -88
  485. package/dist/resources/extensions/gsd/queue-order.ts +0 -230
  486. package/dist/resources/extensions/gsd/queue-reorder-ui.ts +0 -276
  487. package/dist/resources/extensions/gsd/quick.ts +0 -212
  488. package/dist/resources/extensions/gsd/repo-identity.ts +0 -169
  489. package/dist/resources/extensions/gsd/roadmap-slices.ts +0 -149
  490. package/dist/resources/extensions/gsd/routing-history.ts +0 -286
  491. package/dist/resources/extensions/gsd/safe-fs.ts +0 -47
  492. package/dist/resources/extensions/gsd/semantic-chunker.ts +0 -336
  493. package/dist/resources/extensions/gsd/session-forensics.ts +0 -537
  494. package/dist/resources/extensions/gsd/session-lock.ts +0 -362
  495. package/dist/resources/extensions/gsd/session-status-io.ts +0 -179
  496. package/dist/resources/extensions/gsd/skill-discovery.ts +0 -139
  497. package/dist/resources/extensions/gsd/skill-health.ts +0 -417
  498. package/dist/resources/extensions/gsd/state.ts +0 -727
  499. package/dist/resources/extensions/gsd/structured-data-formatter.ts +0 -144
  500. package/dist/resources/extensions/gsd/summary-distiller.ts +0 -258
  501. package/dist/resources/extensions/gsd/tests/activity-log.test.ts +0 -213
  502. package/dist/resources/extensions/gsd/tests/agent-end-retry.test.ts +0 -107
  503. package/dist/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +0 -200
  504. package/dist/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +0 -50
  505. package/dist/resources/extensions/gsd/tests/auto-dashboard.test.ts +0 -166
  506. package/dist/resources/extensions/gsd/tests/auto-lock-creation.test.ts +0 -186
  507. package/dist/resources/extensions/gsd/tests/auto-preflight.test.ts +0 -40
  508. package/dist/resources/extensions/gsd/tests/auto-recovery.test.ts +0 -640
  509. package/dist/resources/extensions/gsd/tests/auto-reentrancy-guard.test.ts +0 -127
  510. package/dist/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +0 -302
  511. package/dist/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +0 -257
  512. package/dist/resources/extensions/gsd/tests/auto-skip-loop.test.ts +0 -123
  513. package/dist/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +0 -340
  514. package/dist/resources/extensions/gsd/tests/auto-worktree.test.ts +0 -167
  515. package/dist/resources/extensions/gsd/tests/budget-prediction.test.ts +0 -220
  516. package/dist/resources/extensions/gsd/tests/captures.test.ts +0 -438
  517. package/dist/resources/extensions/gsd/tests/claude-import-tui.test.ts +0 -351
  518. package/dist/resources/extensions/gsd/tests/collect-from-manifest.test.ts +0 -469
  519. package/dist/resources/extensions/gsd/tests/collision-diagnostics.test.ts +0 -705
  520. package/dist/resources/extensions/gsd/tests/commands-logs.test.ts +0 -241
  521. package/dist/resources/extensions/gsd/tests/complete-milestone.test.ts +0 -209
  522. package/dist/resources/extensions/gsd/tests/complexity-classifier.test.ts +0 -181
  523. package/dist/resources/extensions/gsd/tests/context-budget.test.ts +0 -352
  524. package/dist/resources/extensions/gsd/tests/context-compression.test.ts +0 -193
  525. package/dist/resources/extensions/gsd/tests/context-store.test.ts +0 -462
  526. package/dist/resources/extensions/gsd/tests/continue-here.test.ts +0 -285
  527. package/dist/resources/extensions/gsd/tests/cost-projection.test.ts +0 -134
  528. package/dist/resources/extensions/gsd/tests/crash-recovery.test.ts +0 -134
  529. package/dist/resources/extensions/gsd/tests/dashboard-budget.test.ts +0 -346
  530. package/dist/resources/extensions/gsd/tests/db-writer.test.ts +0 -602
  531. package/dist/resources/extensions/gsd/tests/debug-logger.test.ts +0 -185
  532. package/dist/resources/extensions/gsd/tests/derive-state-db.test.ts +0 -405
  533. package/dist/resources/extensions/gsd/tests/derive-state-deps.test.ts +0 -421
  534. package/dist/resources/extensions/gsd/tests/derive-state-draft.test.ts +0 -308
  535. package/dist/resources/extensions/gsd/tests/derive-state.test.ts +0 -788
  536. package/dist/resources/extensions/gsd/tests/detection.test.ts +0 -398
  537. package/dist/resources/extensions/gsd/tests/diff-context.test.ts +0 -136
  538. package/dist/resources/extensions/gsd/tests/discuss-prompt.test.ts +0 -15
  539. package/dist/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -128
  540. package/dist/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +0 -132
  541. package/dist/resources/extensions/gsd/tests/dispatch-stall-guard.test.ts +0 -126
  542. package/dist/resources/extensions/gsd/tests/doctor-environment.test.ts +0 -314
  543. package/dist/resources/extensions/gsd/tests/doctor-fixlevel.test.ts +0 -245
  544. package/dist/resources/extensions/gsd/tests/doctor-git.test.ts +0 -344
  545. package/dist/resources/extensions/gsd/tests/doctor-proactive.test.ts +0 -278
  546. package/dist/resources/extensions/gsd/tests/doctor-providers.test.ts +0 -298
  547. package/dist/resources/extensions/gsd/tests/doctor-runtime.test.ts +0 -302
  548. package/dist/resources/extensions/gsd/tests/doctor.test.ts +0 -652
  549. package/dist/resources/extensions/gsd/tests/draft-promotion.test.ts +0 -169
  550. package/dist/resources/extensions/gsd/tests/exit-command.test.ts +0 -50
  551. package/dist/resources/extensions/gsd/tests/export-html-all.test.ts +0 -105
  552. package/dist/resources/extensions/gsd/tests/export-html-enhancements.test.ts +0 -378
  553. package/dist/resources/extensions/gsd/tests/extension-selector-separator.test.ts +0 -144
  554. package/dist/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +0 -424
  555. package/dist/resources/extensions/gsd/tests/git-self-heal.test.ts +0 -131
  556. package/dist/resources/extensions/gsd/tests/git-service.test.ts +0 -1172
  557. package/dist/resources/extensions/gsd/tests/gsd-db.test.ts +0 -353
  558. package/dist/resources/extensions/gsd/tests/gsd-inspect.test.ts +0 -125
  559. package/dist/resources/extensions/gsd/tests/gsd-tools.test.ts +0 -326
  560. package/dist/resources/extensions/gsd/tests/headless-answers.test.ts +0 -340
  561. package/dist/resources/extensions/gsd/tests/headless-query.test.ts +0 -162
  562. package/dist/resources/extensions/gsd/tests/idle-recovery.test.ts +0 -485
  563. package/dist/resources/extensions/gsd/tests/in-flight-tool-tracking.test.ts +0 -32
  564. package/dist/resources/extensions/gsd/tests/init-wizard.test.ts +0 -197
  565. package/dist/resources/extensions/gsd/tests/integration/headless-command.ts +0 -534
  566. package/dist/resources/extensions/gsd/tests/integration-edge.test.ts +0 -228
  567. package/dist/resources/extensions/gsd/tests/integration-lifecycle.test.ts +0 -277
  568. package/dist/resources/extensions/gsd/tests/integration-mixed-milestones.test.ts +0 -523
  569. package/dist/resources/extensions/gsd/tests/key-manager.test.ts +0 -414
  570. package/dist/resources/extensions/gsd/tests/knowledge.test.ts +0 -161
  571. package/dist/resources/extensions/gsd/tests/loop-regression.test.ts +0 -839
  572. package/dist/resources/extensions/gsd/tests/manifest-status.test.ts +0 -283
  573. package/dist/resources/extensions/gsd/tests/marketplace-test-fixtures.ts +0 -91
  574. package/dist/resources/extensions/gsd/tests/md-importer.test.ts +0 -410
  575. package/dist/resources/extensions/gsd/tests/mechanical-completion.test.ts +0 -356
  576. package/dist/resources/extensions/gsd/tests/memory-extractor.test.ts +0 -180
  577. package/dist/resources/extensions/gsd/tests/memory-leak-guards.test.ts +0 -91
  578. package/dist/resources/extensions/gsd/tests/memory-store.test.ts +0 -345
  579. package/dist/resources/extensions/gsd/tests/metrics.test.ts +0 -253
  580. package/dist/resources/extensions/gsd/tests/migrate-command.test.ts +0 -369
  581. package/dist/resources/extensions/gsd/tests/migrate-parser.test.ts +0 -757
  582. package/dist/resources/extensions/gsd/tests/migrate-transformer.test.ts +0 -635
  583. package/dist/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +0 -414
  584. package/dist/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +0 -303
  585. package/dist/resources/extensions/gsd/tests/migrate-writer.test.ts +0 -398
  586. package/dist/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +0 -147
  587. package/dist/resources/extensions/gsd/tests/model-cost-table.test.ts +0 -69
  588. package/dist/resources/extensions/gsd/tests/model-isolation.test.ts +0 -157
  589. package/dist/resources/extensions/gsd/tests/model-router.test.ts +0 -167
  590. package/dist/resources/extensions/gsd/tests/must-have-parser.test.ts +0 -291
  591. package/dist/resources/extensions/gsd/tests/namespaced-registry.test.ts +0 -1027
  592. package/dist/resources/extensions/gsd/tests/namespaced-resolver.test.ts +0 -671
  593. package/dist/resources/extensions/gsd/tests/native-has-changes-cache.test.ts +0 -61
  594. package/dist/resources/extensions/gsd/tests/next-milestone-id.test.ts +0 -23
  595. package/dist/resources/extensions/gsd/tests/none-mode-gates.test.ts +0 -114
  596. package/dist/resources/extensions/gsd/tests/notifications.test.ts +0 -67
  597. package/dist/resources/extensions/gsd/tests/overrides.test.ts +0 -131
  598. package/dist/resources/extensions/gsd/tests/parallel-budget-atomicity.test.ts +0 -331
  599. package/dist/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +0 -298
  600. package/dist/resources/extensions/gsd/tests/parallel-merge.test.ts +0 -468
  601. package/dist/resources/extensions/gsd/tests/parallel-orchestration.test.ts +0 -685
  602. package/dist/resources/extensions/gsd/tests/parallel-worker-monitoring.test.ts +0 -171
  603. package/dist/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +0 -354
  604. package/dist/resources/extensions/gsd/tests/park-edge-cases.test.ts +0 -276
  605. package/dist/resources/extensions/gsd/tests/park-milestone.test.ts +0 -401
  606. package/dist/resources/extensions/gsd/tests/parsers.test.ts +0 -1704
  607. package/dist/resources/extensions/gsd/tests/plan-milestone.test.ts +0 -133
  608. package/dist/resources/extensions/gsd/tests/plan-quality-validator.test.ts +0 -363
  609. package/dist/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +0 -42
  610. package/dist/resources/extensions/gsd/tests/plugin-importer-live.test.ts +0 -481
  611. package/dist/resources/extensions/gsd/tests/plugin-importer.test.ts +0 -1383
  612. package/dist/resources/extensions/gsd/tests/post-unit-hooks.test.ts +0 -337
  613. package/dist/resources/extensions/gsd/tests/preferences.test.ts +0 -276
  614. package/dist/resources/extensions/gsd/tests/progress-score.test.ts +0 -206
  615. package/dist/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +0 -464
  616. package/dist/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +0 -314
  617. package/dist/resources/extensions/gsd/tests/prompt-compressor.test.ts +0 -529
  618. package/dist/resources/extensions/gsd/tests/prompt-db.test.ts +0 -385
  619. package/dist/resources/extensions/gsd/tests/prompt-ordering.test.ts +0 -296
  620. package/dist/resources/extensions/gsd/tests/provider-errors.test.ts +0 -338
  621. package/dist/resources/extensions/gsd/tests/queue-draft-detection.test.ts +0 -126
  622. package/dist/resources/extensions/gsd/tests/queue-order.test.ts +0 -204
  623. package/dist/resources/extensions/gsd/tests/queue-reorder-e2e.test.ts +0 -282
  624. package/dist/resources/extensions/gsd/tests/reassess-detection.test.ts +0 -154
  625. package/dist/resources/extensions/gsd/tests/reassess-prompt.test.ts +0 -145
  626. package/dist/resources/extensions/gsd/tests/regex-hardening.test.ts +0 -281
  627. package/dist/resources/extensions/gsd/tests/remote-questions.test.ts +0 -642
  628. package/dist/resources/extensions/gsd/tests/remote-status.test.ts +0 -99
  629. package/dist/resources/extensions/gsd/tests/replan-slice.test.ts +0 -538
  630. package/dist/resources/extensions/gsd/tests/requirements.test.ts +0 -106
  631. package/dist/resources/extensions/gsd/tests/roadmap-slices.test.ts +0 -66
  632. package/dist/resources/extensions/gsd/tests/routing-history.test.ts +0 -240
  633. package/dist/resources/extensions/gsd/tests/run-uat.test.ts +0 -416
  634. package/dist/resources/extensions/gsd/tests/secure-env-collect.test.ts +0 -185
  635. package/dist/resources/extensions/gsd/tests/semantic-chunker.test.ts +0 -426
  636. package/dist/resources/extensions/gsd/tests/session-lock.test.ts +0 -315
  637. package/dist/resources/extensions/gsd/tests/skill-lifecycle.test.ts +0 -126
  638. package/dist/resources/extensions/gsd/tests/smart-entry-draft.test.ts +0 -123
  639. package/dist/resources/extensions/gsd/tests/stale-worktree-cwd.test.ts +0 -142
  640. package/dist/resources/extensions/gsd/tests/stop-auto-remote.test.ts +0 -156
  641. package/dist/resources/extensions/gsd/tests/structured-data-formatter.test.ts +0 -365
  642. package/dist/resources/extensions/gsd/tests/summary-distiller.test.ts +0 -323
  643. package/dist/resources/extensions/gsd/tests/test-helpers.ts +0 -61
  644. package/dist/resources/extensions/gsd/tests/token-counter.test.ts +0 -129
  645. package/dist/resources/extensions/gsd/tests/token-optimization-benchmark.test.ts +0 -1272
  646. package/dist/resources/extensions/gsd/tests/token-optimization-prefs.test.ts +0 -164
  647. package/dist/resources/extensions/gsd/tests/token-profile.test.ts +0 -268
  648. package/dist/resources/extensions/gsd/tests/token-savings.test.ts +0 -366
  649. package/dist/resources/extensions/gsd/tests/triage-dispatch.test.ts +0 -340
  650. package/dist/resources/extensions/gsd/tests/triage-resolution.test.ts +0 -416
  651. package/dist/resources/extensions/gsd/tests/undo.test.ts +0 -136
  652. package/dist/resources/extensions/gsd/tests/unique-milestone-ids.test.ts +0 -219
  653. package/dist/resources/extensions/gsd/tests/unit-runtime.test.ts +0 -258
  654. package/dist/resources/extensions/gsd/tests/update-command.test.ts +0 -67
  655. package/dist/resources/extensions/gsd/tests/validate-directory.test.ts +0 -222
  656. package/dist/resources/extensions/gsd/tests/validate-milestone.test.ts +0 -375
  657. package/dist/resources/extensions/gsd/tests/verification-evidence.test.ts +0 -745
  658. package/dist/resources/extensions/gsd/tests/verification-gate.test.ts +0 -1208
  659. package/dist/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +0 -145
  660. package/dist/resources/extensions/gsd/tests/visualizer-data.test.ts +0 -446
  661. package/dist/resources/extensions/gsd/tests/visualizer-overlay.test.ts +0 -237
  662. package/dist/resources/extensions/gsd/tests/visualizer-views.test.ts +0 -718
  663. package/dist/resources/extensions/gsd/tests/worker-registry.test.ts +0 -148
  664. package/dist/resources/extensions/gsd/tests/workflow-templates.test.ts +0 -173
  665. package/dist/resources/extensions/gsd/tests/workspace-index.test.ts +0 -38
  666. package/dist/resources/extensions/gsd/tests/worktree-bugfix.test.ts +0 -120
  667. package/dist/resources/extensions/gsd/tests/worktree-e2e.test.ts +0 -244
  668. package/dist/resources/extensions/gsd/tests/worktree-integration.test.ts +0 -207
  669. package/dist/resources/extensions/gsd/tests/worktree-manager.test.ts +0 -141
  670. package/dist/resources/extensions/gsd/tests/worktree-post-create-hook.test.ts +0 -165
  671. package/dist/resources/extensions/gsd/tests/worktree.test.ts +0 -171
  672. package/dist/resources/extensions/gsd/tests/write-gate.test.ts +0 -211
  673. package/dist/resources/extensions/gsd/token-counter.ts +0 -65
  674. package/dist/resources/extensions/gsd/triage-resolution.ts +0 -284
  675. package/dist/resources/extensions/gsd/triage-ui.ts +0 -175
  676. package/dist/resources/extensions/gsd/types.ts +0 -425
  677. package/dist/resources/extensions/gsd/undo.ts +0 -223
  678. package/dist/resources/extensions/gsd/unit-id.ts +0 -14
  679. package/dist/resources/extensions/gsd/unit-runtime.ts +0 -192
  680. package/dist/resources/extensions/gsd/validate-directory.ts +0 -164
  681. package/dist/resources/extensions/gsd/verification-evidence.ts +0 -188
  682. package/dist/resources/extensions/gsd/verification-gate.ts +0 -643
  683. package/dist/resources/extensions/gsd/visualizer-data.ts +0 -866
  684. package/dist/resources/extensions/gsd/visualizer-overlay.ts +0 -566
  685. package/dist/resources/extensions/gsd/visualizer-views.ts +0 -1171
  686. package/dist/resources/extensions/gsd/workflow-templates.ts +0 -241
  687. package/dist/resources/extensions/gsd/workspace-index.ts +0 -217
  688. package/dist/resources/extensions/gsd/worktree-command.ts +0 -807
  689. package/dist/resources/extensions/gsd/worktree-manager.ts +0 -449
  690. package/dist/resources/extensions/gsd/worktree.ts +0 -257
  691. package/dist/resources/extensions/mac-tools/index.ts +0 -852
  692. package/dist/resources/extensions/mcp-client/index.ts +0 -459
  693. package/dist/resources/extensions/mcporter/extension-manifest.json +0 -12
  694. package/dist/resources/extensions/remote-questions/config.ts +0 -83
  695. package/dist/resources/extensions/remote-questions/discord-adapter.ts +0 -148
  696. package/dist/resources/extensions/remote-questions/format.ts +0 -315
  697. package/dist/resources/extensions/remote-questions/http-client.ts +0 -76
  698. package/dist/resources/extensions/remote-questions/manager.ts +0 -184
  699. package/dist/resources/extensions/remote-questions/notify.ts +0 -90
  700. package/dist/resources/extensions/remote-questions/remote-command.ts +0 -457
  701. package/dist/resources/extensions/remote-questions/slack-adapter.ts +0 -141
  702. package/dist/resources/extensions/remote-questions/status.ts +0 -31
  703. package/dist/resources/extensions/remote-questions/store.ts +0 -81
  704. package/dist/resources/extensions/remote-questions/telegram-adapter.ts +0 -149
  705. package/dist/resources/extensions/remote-questions/types.ts +0 -102
  706. package/dist/resources/extensions/search-the-web/cache.ts +0 -78
  707. package/dist/resources/extensions/search-the-web/command-search-provider.ts +0 -101
  708. package/dist/resources/extensions/search-the-web/format.ts +0 -258
  709. package/dist/resources/extensions/search-the-web/http.ts +0 -238
  710. package/dist/resources/extensions/search-the-web/index.ts +0 -65
  711. package/dist/resources/extensions/search-the-web/native-search.ts +0 -193
  712. package/dist/resources/extensions/search-the-web/provider.ts +0 -148
  713. package/dist/resources/extensions/search-the-web/tavily.ts +0 -116
  714. package/dist/resources/extensions/search-the-web/tool-fetch-page.ts +0 -589
  715. package/dist/resources/extensions/search-the-web/tool-llm-context.ts +0 -608
  716. package/dist/resources/extensions/search-the-web/tool-search.ts +0 -649
  717. package/dist/resources/extensions/search-the-web/url-utils.ts +0 -125
  718. package/dist/resources/extensions/shared/confirm-ui.ts +0 -126
  719. package/dist/resources/extensions/shared/frontmatter.ts +0 -117
  720. package/dist/resources/extensions/shared/interview-ui.ts +0 -613
  721. package/dist/resources/extensions/shared/next-action-ui.ts +0 -212
  722. package/dist/resources/extensions/shared/sanitize.ts +0 -19
  723. package/dist/resources/extensions/shared/terminal.ts +0 -23
  724. package/dist/resources/extensions/shared/tests/format-utils.test.ts +0 -153
  725. package/dist/resources/extensions/shared/ui.ts +0 -400
  726. package/dist/resources/extensions/shared/wizard-ui.ts +0 -551
  727. package/dist/resources/extensions/slash-commands/audit.ts +0 -88
  728. package/dist/resources/extensions/slash-commands/clear.ts +0 -10
  729. package/dist/resources/extensions/slash-commands/create-extension.ts +0 -297
  730. package/dist/resources/extensions/slash-commands/create-slash-command.ts +0 -234
  731. package/dist/resources/extensions/slash-commands/index.ts +0 -12
  732. package/dist/resources/extensions/subagent/agents.ts +0 -126
  733. package/dist/resources/extensions/subagent/index.ts +0 -1121
  734. package/dist/resources/extensions/subagent/isolation.ts +0 -501
  735. package/dist/resources/extensions/subagent/worker-registry.ts +0 -99
  736. package/dist/resources/extensions/ttsr/index.ts +0 -168
  737. package/dist/resources/extensions/ttsr/rule-loader.ts +0 -74
  738. package/dist/resources/extensions/ttsr/ttsr-manager.ts +0 -456
  739. package/dist/resources/extensions/universal-config/discovery.ts +0 -104
  740. package/dist/resources/extensions/universal-config/format.ts +0 -191
  741. package/dist/resources/extensions/universal-config/index.ts +0 -120
  742. package/dist/resources/extensions/universal-config/scanners.ts +0 -642
  743. package/dist/resources/extensions/universal-config/tests/discovery.test.ts +0 -119
  744. package/dist/resources/extensions/universal-config/tests/format.test.ts +0 -127
  745. package/dist/resources/extensions/universal-config/tests/scanners.test.ts +0 -456
  746. package/dist/resources/extensions/universal-config/tools.ts +0 -60
  747. package/dist/resources/extensions/universal-config/types.ts +0 -135
  748. package/dist/resources/extensions/voice/index.ts +0 -272
  749. package/dist/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +0 -51
  750. package/dist/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +0 -143
  751. package/src/resources/extensions/mcporter/extension-manifest.json +0 -12
@@ -19,56 +19,31 @@
19
19
  *
20
20
  * Design: Linear-inspired — restrained palette, geometric status, no emoji.
21
21
  */
22
-
23
- import type {
24
- VisualizerData,
25
- VisualizerMilestone,
26
- VisualizerSlice,
27
- } from './visualizer-data.js';
28
22
  import { formatDateShort, formatDuration } from '../shared/format-utils.js';
29
23
  import { formatCost, formatTokenCount } from './metrics.js';
30
- import type { UnitMetrics } from './metrics.js';
31
-
32
- // ─── Public API ────────────────────────────────────────────────────────────────
33
-
34
- export interface HtmlReportOptions {
35
- projectName: string;
36
- projectPath: string;
37
- gsdVersion: string;
38
- milestoneId?: string;
39
- indexRelPath?: string;
40
- }
41
-
42
- export function generateHtmlReport(
43
- data: VisualizerData,
44
- opts: HtmlReportOptions,
45
- ): string {
46
- const generated = new Date().toISOString();
47
-
48
- const sections = [
49
- buildSummarySection(data, opts, generated),
50
- buildBlockersSection(data),
51
- buildProgressSection(data),
52
- buildTimelineSection(data),
53
- buildDepGraphSection(data),
54
- buildMetricsSection(data),
55
- buildHealthSection(data),
56
- buildChangelogSection(data),
57
- buildKnowledgeSection(data),
58
- buildCapturesSection(data),
59
- buildStatsSection(data),
60
- buildDiscussionSection(data),
61
- ];
62
-
63
- const milestoneTag = opts.milestoneId
64
- ? ` <span class="sep">/</span> <span class="mono accent">${esc(opts.milestoneId)}</span>`
65
- : '';
66
-
67
- const backLink = opts.indexRelPath
68
- ? `<a class="back-link" href="${esc(opts.indexRelPath)}">All Reports</a>`
69
- : '';
70
-
71
- return `<!DOCTYPE html>
24
+ export function generateHtmlReport(data, opts) {
25
+ const generated = new Date().toISOString();
26
+ const sections = [
27
+ buildSummarySection(data, opts, generated),
28
+ buildBlockersSection(data),
29
+ buildProgressSection(data),
30
+ buildTimelineSection(data),
31
+ buildDepGraphSection(data),
32
+ buildMetricsSection(data),
33
+ buildHealthSection(data),
34
+ buildChangelogSection(data),
35
+ buildKnowledgeSection(data),
36
+ buildCapturesSection(data),
37
+ buildStatsSection(data),
38
+ buildDiscussionSection(data),
39
+ ];
40
+ const milestoneTag = opts.milestoneId
41
+ ? ` <span class="sep">/</span> <span class="mono accent">${esc(opts.milestoneId)}</span>`
42
+ : '';
43
+ const backLink = opts.indexRelPath
44
+ ? `<a class="back-link" href="${esc(opts.indexRelPath)}">All Reports</a>`
45
+ : '';
46
+ return `<!DOCTYPE html>
72
47
  <html lang="en">
73
48
  <head>
74
49
  <meta charset="UTF-8">
@@ -126,61 +101,51 @@ ${sections.join('\n')}
126
101
  </body>
127
102
  </html>`;
128
103
  }
129
-
130
104
  // ─── Section: Summary ─────────────────────────────────────────────────────────
131
-
132
- function buildSummarySection(
133
- data: VisualizerData,
134
- opts: HtmlReportOptions,
135
- _generated: string,
136
- ): string {
137
- const t = data.totals;
138
- const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
139
- const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
140
- const doneMilestones = data.milestones.filter(m => m.status === 'complete').length;
141
- const activeMilestone = data.milestones.find(m => m.status === 'active');
142
- const pct = totalSlices > 0 ? Math.round((doneSlices / totalSlices) * 100) : 0;
143
-
144
- const act = data.agentActivity;
145
- const kv = [
146
- kvi('Milestones', `${doneMilestones}/${data.milestones.length}`),
147
- kvi('Slices', `${doneSlices}/${totalSlices}`),
148
- kvi('Phase', data.phase),
149
- t ? kvi('Cost', formatCost(t.cost)) : '',
150
- t ? kvi('Tokens', formatTokenCount(t.tokens.total)) : '',
151
- t ? kvi('Duration', formatDuration(t.duration)) : '',
152
- t ? kvi('Tool calls', String(t.toolCalls)) : '',
153
- t ? kvi('Units', String(t.units)) : '',
154
- data.remainingSliceCount > 0 ? kvi('Remaining', String(data.remainingSliceCount)) : '',
155
- act ? kvi('Rate', `${act.completionRate.toFixed(1)}/hr`) : '',
156
- t && doneSlices > 0 ? kvi('Cost/slice', formatCost(t.cost / doneSlices)) : '',
157
- t && t.toolCalls > 0 ? kvi('Tokens/tool', formatTokenCount(t.tokens.total / t.toolCalls)) : '',
158
- t && (t.tokens.input + t.tokens.cacheRead) > 0
159
- ? kvi('Cache hit', ((t.tokens.cacheRead / (t.tokens.input + t.tokens.cacheRead)) * 100).toFixed(1) + '%')
160
- : '',
161
- opts.milestoneId ? kvi('Scope', opts.milestoneId) : '',
162
- ].filter(Boolean).join('');
163
-
164
- const activeInfo = activeMilestone ? (() => {
165
- const active = activeMilestone.slices.find(s => s.active);
166
- if (!active) return '';
167
- return `<div class="active-info">
105
+ function buildSummarySection(data, opts, _generated) {
106
+ const t = data.totals;
107
+ const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
108
+ const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
109
+ const doneMilestones = data.milestones.filter(m => m.status === 'complete').length;
110
+ const activeMilestone = data.milestones.find(m => m.status === 'active');
111
+ const pct = totalSlices > 0 ? Math.round((doneSlices / totalSlices) * 100) : 0;
112
+ const act = data.agentActivity;
113
+ const kv = [
114
+ kvi('Milestones', `${doneMilestones}/${data.milestones.length}`),
115
+ kvi('Slices', `${doneSlices}/${totalSlices}`),
116
+ kvi('Phase', data.phase),
117
+ t ? kvi('Cost', formatCost(t.cost)) : '',
118
+ t ? kvi('Tokens', formatTokenCount(t.tokens.total)) : '',
119
+ t ? kvi('Duration', formatDuration(t.duration)) : '',
120
+ t ? kvi('Tool calls', String(t.toolCalls)) : '',
121
+ t ? kvi('Units', String(t.units)) : '',
122
+ data.remainingSliceCount > 0 ? kvi('Remaining', String(data.remainingSliceCount)) : '',
123
+ act ? kvi('Rate', `${act.completionRate.toFixed(1)}/hr`) : '',
124
+ t && doneSlices > 0 ? kvi('Cost/slice', formatCost(t.cost / doneSlices)) : '',
125
+ t && t.toolCalls > 0 ? kvi('Tokens/tool', formatTokenCount(t.tokens.total / t.toolCalls)) : '',
126
+ t && (t.tokens.input + t.tokens.cacheRead) > 0
127
+ ? kvi('Cache hit', ((t.tokens.cacheRead / (t.tokens.input + t.tokens.cacheRead)) * 100).toFixed(1) + '%')
128
+ : '',
129
+ opts.milestoneId ? kvi('Scope', opts.milestoneId) : '',
130
+ ].filter(Boolean).join('');
131
+ const activeInfo = activeMilestone ? (() => {
132
+ const active = activeMilestone.slices.find(s => s.active);
133
+ if (!active)
134
+ return '';
135
+ return `<div class="active-info">
168
136
  Executing <span class="mono">${esc(activeMilestone.id)}/${esc(active.id)}</span> — ${esc(active.title)}
169
137
  </div>`;
170
- })() : '';
171
-
172
- const activityHtml = act?.active ? `
138
+ })() : '';
139
+ const activityHtml = act?.active ? `
173
140
  <div class="activity-line">
174
141
  <span class="dot dot-active"></span>
175
142
  <span class="mono">${esc(act.currentUnit?.type ?? '')}</span>
176
143
  <span class="mono muted">${esc(act.currentUnit?.id ?? '')}</span>
177
144
  <span class="muted">${formatDuration(act.elapsed)} elapsed</span>
178
145
  </div>` : '';
179
-
180
- const execSummary = buildExecutiveSummary(data, opts);
181
- const etaLine = buildEtaLine(data);
182
-
183
- return section('summary', 'Summary', `
146
+ const execSummary = buildExecutiveSummary(data, opts);
147
+ const etaLine = buildEtaLine(data);
148
+ return section('summary', 'Summary', `
184
149
  ${execSummary}
185
150
  <div class="kv-grid">${kv}</div>
186
151
  <div class="progress-wrap">
@@ -192,134 +157,104 @@ function buildSummarySection(
192
157
  ${etaLine}
193
158
  `);
194
159
  }
195
-
196
- function buildExecutiveSummary(data: VisualizerData, opts: HtmlReportOptions): string {
197
- const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
198
- const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
199
- const pct = totalSlices > 0 ? Math.round((doneSlices / totalSlices) * 100) : 0;
200
- const spent = data.totals?.cost ?? 0;
201
- const activeMilestone = data.milestones.find(m => m.status === 'active');
202
- const activeSlice = activeMilestone?.slices.find(s => s.active);
203
- const currentExec = activeMilestone && activeSlice
204
- ? ` Currently executing ${esc(activeMilestone.id)}/${esc(activeSlice.id)}.`
205
- : '';
206
- const budgetCtx = data.health.budgetCeiling
207
- ? ` Budget: ${formatCost(spent)} of ${formatCost(data.health.budgetCeiling)} ceiling (${((spent / data.health.budgetCeiling) * 100).toFixed(0)}% used).`
208
- : '';
209
- return `<p class="exec-summary">${esc(opts.projectName)} is ${pct}% complete across ${data.milestones.length} milestones. ${formatCost(spent)} spent.${currentExec}${budgetCtx}</p>`;
160
+ function buildExecutiveSummary(data, opts) {
161
+ const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
162
+ const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
163
+ const pct = totalSlices > 0 ? Math.round((doneSlices / totalSlices) * 100) : 0;
164
+ const spent = data.totals?.cost ?? 0;
165
+ const activeMilestone = data.milestones.find(m => m.status === 'active');
166
+ const activeSlice = activeMilestone?.slices.find(s => s.active);
167
+ const currentExec = activeMilestone && activeSlice
168
+ ? ` Currently executing ${esc(activeMilestone.id)}/${esc(activeSlice.id)}.`
169
+ : '';
170
+ const budgetCtx = data.health.budgetCeiling
171
+ ? ` Budget: ${formatCost(spent)} of ${formatCost(data.health.budgetCeiling)} ceiling (${((spent / data.health.budgetCeiling) * 100).toFixed(0)}% used).`
172
+ : '';
173
+ return `<p class="exec-summary">${esc(opts.projectName)} is ${pct}% complete across ${data.milestones.length} milestones. ${formatCost(spent)} spent.${currentExec}${budgetCtx}</p>`;
210
174
  }
211
-
212
- function buildEtaLine(data: VisualizerData): string {
213
- const act = data.agentActivity;
214
- if (!act || act.completionRate <= 0 || data.remainingSliceCount <= 0) return '';
215
- const hoursRemaining = data.remainingSliceCount / act.completionRate;
216
- const formatted = formatDuration(hoursRemaining * 3_600_000);
217
- return `<div class="eta-line">ETA: ~${formatted} remaining (${data.remainingSliceCount} slices at ${act.completionRate.toFixed(1)}/hr)</div>`;
175
+ function buildEtaLine(data) {
176
+ const act = data.agentActivity;
177
+ if (!act || act.completionRate <= 0 || data.remainingSliceCount <= 0)
178
+ return '';
179
+ const hoursRemaining = data.remainingSliceCount / act.completionRate;
180
+ const formatted = formatDuration(hoursRemaining * 3_600_000);
181
+ return `<div class="eta-line">ETA: ~${formatted} remaining (${data.remainingSliceCount} slices at ${act.completionRate.toFixed(1)}/hr)</div>`;
218
182
  }
219
-
220
183
  // ─── Section: Blockers ────────────────────────────────────────────────────────
221
-
222
- function buildBlockersSection(data: VisualizerData): string {
223
- const blockers = data.sliceVerifications.filter(v => v.blockerDiscovered === true);
224
- const highRisk: { msId: string; slId: string }[] = [];
225
- for (const ms of data.milestones) {
226
- for (const sl of ms.slices) {
227
- if (!sl.done && sl.risk?.toLowerCase() === 'high') {
228
- highRisk.push({ msId: ms.id, slId: sl.id });
229
- }
184
+ function buildBlockersSection(data) {
185
+ const blockers = data.sliceVerifications.filter(v => v.blockerDiscovered === true);
186
+ const highRisk = [];
187
+ for (const ms of data.milestones) {
188
+ for (const sl of ms.slices) {
189
+ if (!sl.done && sl.risk?.toLowerCase() === 'high') {
190
+ highRisk.push({ msId: ms.id, slId: sl.id });
191
+ }
192
+ }
230
193
  }
231
- }
232
-
233
- if (blockers.length === 0 && highRisk.length === 0) {
234
- return section('blockers', 'Blockers', '<p class="empty">No blockers or high-risk items found.</p>');
235
- }
236
-
237
- const blockerCards = blockers.map(v => `
194
+ if (blockers.length === 0 && highRisk.length === 0) {
195
+ return section('blockers', 'Blockers', '<p class="empty">No blockers or high-risk items found.</p>');
196
+ }
197
+ const blockerCards = blockers.map(v => `
238
198
  <div class="blocker-card">
239
199
  <div class="blocker-id">${esc(v.milestoneId)}/${esc(v.sliceId)}</div>
240
200
  <div class="blocker-text">${esc(v.verificationResult ?? 'Blocker discovered')}</div>
241
201
  </div>`).join('');
242
-
243
- const riskCards = highRisk
244
- .filter(hr => !blockers.some(b => b.milestoneId === hr.msId && b.sliceId === hr.slId))
245
- .map(hr => `
202
+ const riskCards = highRisk
203
+ .filter(hr => !blockers.some(b => b.milestoneId === hr.msId && b.sliceId === hr.slId))
204
+ .map(hr => `
246
205
  <div class="blocker-card">
247
206
  <div class="blocker-id">${esc(hr.msId)}/${esc(hr.slId)}</div>
248
207
  <div class="blocker-text">High risk — incomplete</div>
249
208
  </div>`).join('');
250
-
251
- return section('blockers', 'Blockers', `${blockerCards}${riskCards}`);
209
+ return section('blockers', 'Blockers', `${blockerCards}${riskCards}`);
252
210
  }
253
-
254
211
  // ─── Section: Health ──────────────────────────────────────────────────────────
255
-
256
- function buildHealthSection(data: VisualizerData): string {
257
- const h = data.health;
258
- const t = data.totals;
259
-
260
- const rows: string[] = [];
261
- rows.push(hRow('Token profile', h.tokenProfile));
262
- if (h.budgetCeiling !== undefined) {
263
- const spent = t?.cost ?? 0;
264
- const pct = (spent / h.budgetCeiling) * 100;
265
- const status = pct > 90 ? 'warn' : pct > 75 ? 'caution' : 'ok';
266
- rows.push(hRow(
267
- 'Budget ceiling',
268
- `${formatCost(h.budgetCeiling)} (${formatCost(spent)} spent, ${pct.toFixed(0)}% used)`,
269
- status,
270
- ));
271
- }
272
- rows.push(hRow(
273
- 'Truncation rate',
274
- `${h.truncationRate.toFixed(1)}% per unit (${t?.totalTruncationSections ?? 0} total)`,
275
- h.truncationRate > 20 ? 'warn' : h.truncationRate > 10 ? 'caution' : 'ok',
276
- ));
277
- rows.push(hRow(
278
- 'Continue-here rate',
279
- `${h.continueHereRate.toFixed(1)}% per unit (${t?.continueHereFiredCount ?? 0} total)`,
280
- h.continueHereRate > 15 ? 'warn' : h.continueHereRate > 8 ? 'caution' : 'ok',
281
- ));
282
- if (h.tierSavingsLine) rows.push(hRow('Routing savings', h.tierSavingsLine));
283
- rows.push(hRow('Tool calls', String(h.toolCalls)));
284
- rows.push(hRow('Messages', `${h.assistantMessages} assistant / ${h.userMessages} user`));
285
-
286
- const tierRows = h.tierBreakdown.length > 0 ? `
212
+ function buildHealthSection(data) {
213
+ const h = data.health;
214
+ const t = data.totals;
215
+ const rows = [];
216
+ rows.push(hRow('Token profile', h.tokenProfile));
217
+ if (h.budgetCeiling !== undefined) {
218
+ const spent = t?.cost ?? 0;
219
+ const pct = (spent / h.budgetCeiling) * 100;
220
+ const status = pct > 90 ? 'warn' : pct > 75 ? 'caution' : 'ok';
221
+ rows.push(hRow('Budget ceiling', `${formatCost(h.budgetCeiling)} (${formatCost(spent)} spent, ${pct.toFixed(0)}% used)`, status));
222
+ }
223
+ rows.push(hRow('Truncation rate', `${h.truncationRate.toFixed(1)}% per unit (${t?.totalTruncationSections ?? 0} total)`, h.truncationRate > 20 ? 'warn' : h.truncationRate > 10 ? 'caution' : 'ok'));
224
+ rows.push(hRow('Continue-here rate', `${h.continueHereRate.toFixed(1)}% per unit (${t?.continueHereFiredCount ?? 0} total)`, h.continueHereRate > 15 ? 'warn' : h.continueHereRate > 8 ? 'caution' : 'ok'));
225
+ if (h.tierSavingsLine)
226
+ rows.push(hRow('Routing savings', h.tierSavingsLine));
227
+ rows.push(hRow('Tool calls', String(h.toolCalls)));
228
+ rows.push(hRow('Messages', `${h.assistantMessages} assistant / ${h.userMessages} user`));
229
+ const tierRows = h.tierBreakdown.length > 0 ? `
287
230
  <h3>Tier breakdown</h3>
288
231
  <table class="tbl">
289
232
  <thead><tr><th>Tier</th><th>Units</th><th>Cost</th><th>Tokens</th></tr></thead>
290
233
  <tbody>
291
- ${h.tierBreakdown.map(tb =>
292
- `<tr><td class="mono">${esc(tb.tier)}</td>
234
+ ${h.tierBreakdown.map(tb => `<tr><td class="mono">${esc(tb.tier)}</td>
293
235
  <td>${tb.units}</td><td>${formatCost(tb.cost)}</td>
294
- <td>${formatTokenCount(tb.tokens.total)}</td></tr>`
295
- ).join('')}
236
+ <td>${formatTokenCount(tb.tokens.total)}</td></tr>`).join('')}
296
237
  </tbody>
297
238
  </table>` : '';
298
-
299
- return section('health', 'Health', `
239
+ return section('health', 'Health', `
300
240
  <table class="tbl tbl-kv"><tbody>${rows.join('')}</tbody></table>
301
241
  ${tierRows}
302
242
  `);
303
243
  }
304
-
305
244
  // ─── Section: Progress ────────────────────────────────────────────────────────
306
-
307
- function buildProgressSection(data: VisualizerData): string {
308
- if (data.milestones.length === 0) {
309
- return section('progress', 'Progress', '<p class="empty">No milestones found.</p>');
310
- }
311
-
312
- const critMS = new Set(data.criticalPath.milestonePath);
313
- const critSL = new Set(data.criticalPath.slicePath);
314
-
315
- const msHtml = data.milestones.map(ms => {
316
- const doneCount = ms.slices.filter(s => s.done).length;
317
- const onCrit = critMS.has(ms.id);
318
- const sliceHtml = ms.slices.length > 0
319
- ? ms.slices.map(sl => buildSliceRow(sl, critSL, data)).join('')
320
- : '<p class="empty indent">No slices in roadmap yet.</p>';
321
-
322
- return `
245
+ function buildProgressSection(data) {
246
+ if (data.milestones.length === 0) {
247
+ return section('progress', 'Progress', '<p class="empty">No milestones found.</p>');
248
+ }
249
+ const critMS = new Set(data.criticalPath.milestonePath);
250
+ const critSL = new Set(data.criticalPath.slicePath);
251
+ const msHtml = data.milestones.map(ms => {
252
+ const doneCount = ms.slices.filter(s => s.done).length;
253
+ const onCrit = critMS.has(ms.id);
254
+ const sliceHtml = ms.slices.length > 0
255
+ ? ms.slices.map(sl => buildSliceRow(sl, critSL, data)).join('')
256
+ : '<p class="empty indent">No slices in roadmap yet.</p>';
257
+ return `
323
258
  <details class="ms-block" ${ms.status !== 'pending' && ms.status !== 'parked' ? 'open' : ''}>
324
259
  <summary class="ms-summary ms-${ms.status}">
325
260
  <span class="dot dot-${ms.status}"></span>
@@ -331,18 +266,15 @@ function buildProgressSection(data: VisualizerData): string {
331
266
  </summary>
332
267
  <div class="ms-body">${sliceHtml}</div>
333
268
  </details>`;
334
- }).join('');
335
-
336
- return section('progress', 'Progress', msHtml);
269
+ }).join('');
270
+ return section('progress', 'Progress', msHtml);
337
271
  }
338
-
339
- function buildSliceRow(sl: VisualizerSlice, critSL: Set<string>, data: VisualizerData): string {
340
- const onCrit = critSL.has(sl.id);
341
- const ver = data.sliceVerifications.find(v => v.sliceId === sl.id);
342
- const slack = data.criticalPath.sliceSlack.get(sl.id);
343
- const status = sl.done ? 'complete' : sl.active ? 'active' : 'pending';
344
-
345
- const taskHtml = sl.tasks.length > 0 ? `
272
+ function buildSliceRow(sl, critSL, data) {
273
+ const onCrit = critSL.has(sl.id);
274
+ const ver = data.sliceVerifications.find(v => v.sliceId === sl.id);
275
+ const slack = data.criticalPath.sliceSlack.get(sl.id);
276
+ const status = sl.done ? 'complete' : sl.active ? 'active' : 'pending';
277
+ const taskHtml = sl.tasks.length > 0 ? `
346
278
  <ul class="task-list">
347
279
  ${sl.tasks.map(t => `
348
280
  <li class="task-row">
@@ -352,27 +284,22 @@ function buildSliceRow(sl: VisualizerSlice, critSL: Set<string>, data: Visualize
352
284
  ${t.estimate ? `<span class="muted">${esc(t.estimate)}</span>` : ''}
353
285
  </li>`).join('')}
354
286
  </ul>` : '';
355
-
356
- const tags = [
357
- ...(ver?.provides ?? []).map(p => `<span class="tag">provides: ${esc(p)}</span>`),
358
- ...(ver?.requires ?? []).map(r => `<span class="tag">requires: ${esc(r.provides)}</span>`),
359
- ].join('');
360
-
361
- const keyDecisions = ver?.keyDecisions?.length
362
- ? `<div class="detail-block"><span class="detail-label">Decisions</span><ul>${ver.keyDecisions.map(d => `<li>${esc(d)}</li>`).join('')}</ul></div>`
363
- : '';
364
-
365
- const patterns = ver?.patternsEstablished?.length
366
- ? `<div class="detail-block"><span class="detail-label">Patterns</span><ul>${ver.patternsEstablished.map(p => `<li>${esc(p)}</li>`).join('')}</ul></div>`
367
- : '';
368
-
369
- const verifBadge = ver?.verificationResult
370
- ? `<div class="verif ${ver.blockerDiscovered ? 'verif-blocker' : ''}">
287
+ const tags = [
288
+ ...(ver?.provides ?? []).map(p => `<span class="tag">provides: ${esc(p)}</span>`),
289
+ ...(ver?.requires ?? []).map(r => `<span class="tag">requires: ${esc(r.provides)}</span>`),
290
+ ].join('');
291
+ const keyDecisions = ver?.keyDecisions?.length
292
+ ? `<div class="detail-block"><span class="detail-label">Decisions</span><ul>${ver.keyDecisions.map(d => `<li>${esc(d)}</li>`).join('')}</ul></div>`
293
+ : '';
294
+ const patterns = ver?.patternsEstablished?.length
295
+ ? `<div class="detail-block"><span class="detail-label">Patterns</span><ul>${ver.patternsEstablished.map(p => `<li>${esc(p)}</li>`).join('')}</ul></div>`
296
+ : '';
297
+ const verifBadge = ver?.verificationResult
298
+ ? `<div class="verif ${ver.blockerDiscovered ? 'verif-blocker' : ''}">
371
299
  ${ver.blockerDiscovered ? 'Blocker: ' : ''}${esc(ver.verificationResult)}
372
300
  </div>`
373
- : '';
374
-
375
- return `
301
+ : '';
302
+ return `
376
303
  <details class="sl-block">
377
304
  <summary class="sl-summary ${onCrit ? 'sl-crit' : ''}">
378
305
  <span class="dot dot-${status} dot-sm"></span>
@@ -392,112 +319,113 @@ function buildSliceRow(sl: VisualizerSlice, critSL: Set<string>, data: Visualize
392
319
  </div>
393
320
  </details>`;
394
321
  }
395
-
396
322
  // ─── Section: Dependency Graph ────────────────────────────────────────────────
397
-
398
- function buildDepGraphSection(data: VisualizerData): string {
399
- const hasSlices = data.milestones.some(ms => ms.slices.length > 0);
400
- if (!hasSlices) return section('depgraph', 'Dependencies', '<p class="empty">No slices to graph.</p>');
401
-
402
- const hasDeps = data.milestones.some(ms => ms.slices.some(s => s.depends.length > 0));
403
- if (!hasDeps) return section('depgraph', 'Dependencies', '<p class="empty">No dependencies defined.</p>');
404
-
405
- const svgs = data.milestones
406
- .filter(ms => ms.slices.length > 0)
407
- .map(ms => buildMilestoneDepSVG(ms, data))
408
- .filter(Boolean)
409
- .join('');
410
-
411
- return section('depgraph', 'Dependencies', svgs);
323
+ function buildDepGraphSection(data) {
324
+ const hasSlices = data.milestones.some(ms => ms.slices.length > 0);
325
+ if (!hasSlices)
326
+ return section('depgraph', 'Dependencies', '<p class="empty">No slices to graph.</p>');
327
+ const hasDeps = data.milestones.some(ms => ms.slices.some(s => s.depends.length > 0));
328
+ if (!hasDeps)
329
+ return section('depgraph', 'Dependencies', '<p class="empty">No dependencies defined.</p>');
330
+ const svgs = data.milestones
331
+ .filter(ms => ms.slices.length > 0)
332
+ .map(ms => buildMilestoneDepSVG(ms, data))
333
+ .filter(Boolean)
334
+ .join('');
335
+ return section('depgraph', 'Dependencies', svgs);
412
336
  }
413
-
414
- function buildMilestoneDepSVG(ms: VisualizerMilestone, data: VisualizerData): string {
415
- const slices = ms.slices;
416
- if (slices.length === 0) return '';
417
-
418
- const critSL = new Set(data.criticalPath.slicePath);
419
- const slMap = new Map(slices.map(s => [s.id, s]));
420
-
421
- const layerMap = new Map<string, number>();
422
- const inDeg = new Map<string, number>();
423
- for (const s of slices) inDeg.set(s.id, 0);
424
- for (const s of slices) {
425
- for (const dep of s.depends) {
426
- if (slMap.has(dep)) inDeg.set(s.id, (inDeg.get(s.id) ?? 0) + 1);
427
- }
428
- }
429
-
430
- const visited = new Set<string>();
431
- const q: string[] = [];
432
- for (const [id, d] of inDeg) {
433
- if (d === 0) { q.push(id); visited.add(id); layerMap.set(id, 0); }
434
- }
435
-
436
- while (q.length > 0) {
437
- const node = q.shift()!;
337
+ function buildMilestoneDepSVG(ms, data) {
338
+ const slices = ms.slices;
339
+ if (slices.length === 0)
340
+ return '';
341
+ const critSL = new Set(data.criticalPath.slicePath);
342
+ const slMap = new Map(slices.map(s => [s.id, s]));
343
+ const layerMap = new Map();
344
+ const inDeg = new Map();
345
+ for (const s of slices)
346
+ inDeg.set(s.id, 0);
438
347
  for (const s of slices) {
439
- if (!s.depends.includes(node)) continue;
440
- const newDeg = (inDeg.get(s.id) ?? 1) - 1;
441
- inDeg.set(s.id, newDeg);
442
- layerMap.set(s.id, Math.max(layerMap.get(s.id) ?? 0, (layerMap.get(node) ?? 0) + 1));
443
- if (newDeg === 0 && !visited.has(s.id)) { visited.add(s.id); q.push(s.id); }
348
+ for (const dep of s.depends) {
349
+ if (slMap.has(dep))
350
+ inDeg.set(s.id, (inDeg.get(s.id) ?? 0) + 1);
351
+ }
444
352
  }
445
- }
446
- for (const s of slices) if (!layerMap.has(s.id)) layerMap.set(s.id, 0);
447
-
448
- const maxLayer = Math.max(...[...layerMap.values()]);
449
- const byLayer = new Map<number, string[]>();
450
- for (const [id, layer] of layerMap) {
451
- const arr = byLayer.get(layer) ?? [];
452
- arr.push(id);
453
- byLayer.set(layer, arr);
454
- }
455
-
456
- const NW = 130, NH = 40, CGAP = 56, RGAP = 14, PAD = 20;
457
- let maxRows = 0;
458
- for (let c = 0; c <= maxLayer; c++) maxRows = Math.max(maxRows, (byLayer.get(c) ?? []).length);
459
- const totalH = PAD * 2 + maxRows * NH + Math.max(0, maxRows - 1) * RGAP;
460
- const totalW = PAD * 2 + (maxLayer + 1) * NW + maxLayer * CGAP;
461
-
462
- const pos = new Map<string, { x: number; y: number }>();
463
- for (let col = 0; col <= maxLayer; col++) {
464
- const ids = byLayer.get(col) ?? [];
465
- const colH = ids.length * NH + Math.max(0, ids.length - 1) * RGAP;
466
- const startY = (totalH - colH) / 2;
467
- ids.forEach((id, i) => pos.set(id, { x: PAD + col * (NW + CGAP), y: startY + i * (NH + RGAP) }));
468
- }
469
-
470
- const edges = slices.flatMap(sl => sl.depends.flatMap(dep => {
471
- if (!pos.has(dep) || !pos.has(sl.id)) return [];
472
- const f = pos.get(dep)!, t = pos.get(sl.id)!;
473
- const x1 = f.x + NW, y1 = f.y + NH / 2;
474
- const x2 = t.x, y2 = t.y + NH / 2;
475
- const mx = (x1 + x2) / 2;
476
- const crit = critSL.has(sl.id) && critSL.has(dep);
477
- return [`<path d="M${x1},${y1} C${mx},${y1} ${mx},${y2} ${x2},${y2}" class="edge${crit ? ' edge-crit' : ''}" marker-end="url(#arr${crit ? '-crit' : ''})"/>`];
478
- }));
479
-
480
- const nodes = slices.map(sl => {
481
- const p = pos.get(sl.id);
482
- if (!p) return '';
483
- const crit = critSL.has(sl.id);
484
- const sc = sl.done ? 'n-done' : sl.active ? 'n-active' : 'n-pending';
485
- return `<g class="node ${sc}${crit ? ' n-crit' : ''}" transform="translate(${p.x},${p.y})">
353
+ const visited = new Set();
354
+ const q = [];
355
+ for (const [id, d] of inDeg) {
356
+ if (d === 0) {
357
+ q.push(id);
358
+ visited.add(id);
359
+ layerMap.set(id, 0);
360
+ }
361
+ }
362
+ while (q.length > 0) {
363
+ const node = q.shift();
364
+ for (const s of slices) {
365
+ if (!s.depends.includes(node))
366
+ continue;
367
+ const newDeg = (inDeg.get(s.id) ?? 1) - 1;
368
+ inDeg.set(s.id, newDeg);
369
+ layerMap.set(s.id, Math.max(layerMap.get(s.id) ?? 0, (layerMap.get(node) ?? 0) + 1));
370
+ if (newDeg === 0 && !visited.has(s.id)) {
371
+ visited.add(s.id);
372
+ q.push(s.id);
373
+ }
374
+ }
375
+ }
376
+ for (const s of slices)
377
+ if (!layerMap.has(s.id))
378
+ layerMap.set(s.id, 0);
379
+ const maxLayer = Math.max(...[...layerMap.values()]);
380
+ const byLayer = new Map();
381
+ for (const [id, layer] of layerMap) {
382
+ const arr = byLayer.get(layer) ?? [];
383
+ arr.push(id);
384
+ byLayer.set(layer, arr);
385
+ }
386
+ const NW = 130, NH = 40, CGAP = 56, RGAP = 14, PAD = 20;
387
+ let maxRows = 0;
388
+ for (let c = 0; c <= maxLayer; c++)
389
+ maxRows = Math.max(maxRows, (byLayer.get(c) ?? []).length);
390
+ const totalH = PAD * 2 + maxRows * NH + Math.max(0, maxRows - 1) * RGAP;
391
+ const totalW = PAD * 2 + (maxLayer + 1) * NW + maxLayer * CGAP;
392
+ const pos = new Map();
393
+ for (let col = 0; col <= maxLayer; col++) {
394
+ const ids = byLayer.get(col) ?? [];
395
+ const colH = ids.length * NH + Math.max(0, ids.length - 1) * RGAP;
396
+ const startY = (totalH - colH) / 2;
397
+ ids.forEach((id, i) => pos.set(id, { x: PAD + col * (NW + CGAP), y: startY + i * (NH + RGAP) }));
398
+ }
399
+ const edges = slices.flatMap(sl => sl.depends.flatMap(dep => {
400
+ if (!pos.has(dep) || !pos.has(sl.id))
401
+ return [];
402
+ const f = pos.get(dep), t = pos.get(sl.id);
403
+ const x1 = f.x + NW, y1 = f.y + NH / 2;
404
+ const x2 = t.x, y2 = t.y + NH / 2;
405
+ const mx = (x1 + x2) / 2;
406
+ const crit = critSL.has(sl.id) && critSL.has(dep);
407
+ return [`<path d="M${x1},${y1} C${mx},${y1} ${mx},${y2} ${x2},${y2}" class="edge${crit ? ' edge-crit' : ''}" marker-end="url(#arr${crit ? '-crit' : ''})"/>`];
408
+ }));
409
+ const nodes = slices.map(sl => {
410
+ const p = pos.get(sl.id);
411
+ if (!p)
412
+ return '';
413
+ const crit = critSL.has(sl.id);
414
+ const sc = sl.done ? 'n-done' : sl.active ? 'n-active' : 'n-pending';
415
+ return `<g class="node ${sc}${crit ? ' n-crit' : ''}" transform="translate(${p.x},${p.y})">
486
416
  <rect width="${NW}" height="${NH}" rx="4"/>
487
- <text x="${NW/2}" y="16" class="n-id">${esc(truncStr(sl.id, 18))}</text>
488
- <text x="${NW/2}" y="30" class="n-title">${esc(truncStr(sl.title, 18))}</text>
417
+ <text x="${NW / 2}" y="16" class="n-id">${esc(truncStr(sl.id, 18))}</text>
418
+ <text x="${NW / 2}" y="30" class="n-title">${esc(truncStr(sl.title, 18))}</text>
489
419
  <title>${esc(sl.id)}: ${esc(sl.title)}</title>
490
420
  </g>`;
491
- });
492
-
493
- const legend = `<div class="dep-legend">
421
+ });
422
+ const legend = `<div class="dep-legend">
494
423
  <span><span class="dot dot-complete dot-sm"></span> done</span>
495
424
  <span><span class="dot dot-active dot-sm"></span> active</span>
496
425
  <span><span class="dot dot-pending dot-sm"></span> pending</span>
497
426
  <span><span class="dot dot-parked dot-sm"></span> parked</span>
498
427
  </div>`;
499
-
500
- return `
428
+ return `
501
429
  <div class="dep-block">
502
430
  <h3>${esc(ms.id)}: ${esc(ms.title)}</h3>
503
431
  ${legend}
@@ -517,59 +445,52 @@ function buildMilestoneDepSVG(ms: VisualizerMilestone, data: VisualizerData): st
517
445
  </div>
518
446
  </div>`;
519
447
  }
520
-
521
448
  // ─── Section: Metrics ─────────────────────────────────────────────────────────
522
-
523
- function buildMetricsSection(data: VisualizerData): string {
524
- if (!data.totals) return section('metrics', 'Metrics', '<p class="empty">No metrics data yet.</p>');
525
- const t = data.totals;
526
-
527
- const grid = [
528
- kvi('Total cost', formatCost(t.cost)),
529
- kvi('Total tokens', formatTokenCount(t.tokens.total)),
530
- kvi('Input', formatTokenCount(t.tokens.input)),
531
- kvi('Output', formatTokenCount(t.tokens.output)),
532
- kvi('Cache read', formatTokenCount(t.tokens.cacheRead)),
533
- kvi('Cache write', formatTokenCount(t.tokens.cacheWrite)),
534
- kvi('Duration', formatDuration(t.duration)),
535
- kvi('Units', String(t.units)),
536
- kvi('Tool calls', String(t.toolCalls)),
537
- kvi('Truncations', String(t.totalTruncationSections)),
538
- ].join('');
539
-
540
- const tokenBreakdown = buildTokenBreakdown(t.tokens);
541
-
542
- const phaseRow = data.byPhase.length > 0 ? `
449
+ function buildMetricsSection(data) {
450
+ if (!data.totals)
451
+ return section('metrics', 'Metrics', '<p class="empty">No metrics data yet.</p>');
452
+ const t = data.totals;
453
+ const grid = [
454
+ kvi('Total cost', formatCost(t.cost)),
455
+ kvi('Total tokens', formatTokenCount(t.tokens.total)),
456
+ kvi('Input', formatTokenCount(t.tokens.input)),
457
+ kvi('Output', formatTokenCount(t.tokens.output)),
458
+ kvi('Cache read', formatTokenCount(t.tokens.cacheRead)),
459
+ kvi('Cache write', formatTokenCount(t.tokens.cacheWrite)),
460
+ kvi('Duration', formatDuration(t.duration)),
461
+ kvi('Units', String(t.units)),
462
+ kvi('Tool calls', String(t.toolCalls)),
463
+ kvi('Truncations', String(t.totalTruncationSections)),
464
+ ].join('');
465
+ const tokenBreakdown = buildTokenBreakdown(t.tokens);
466
+ const phaseRow = data.byPhase.length > 0 ? `
543
467
  <div class="chart-row">
544
468
  ${buildBarChart('Cost by phase', data.byPhase.map(p => ({
545
469
  label: p.phase, value: p.cost, display: formatCost(p.cost), sub: `${p.units} units`,
546
- })))}
470
+ })))}
547
471
  ${buildBarChart('Tokens by phase', data.byPhase.map(p => ({
548
472
  label: p.phase, value: p.tokens.total, display: formatTokenCount(p.tokens.total), sub: formatCost(p.cost),
549
- })))}
473
+ })))}
550
474
  </div>` : '';
551
-
552
- const sliceModelRow = (data.bySlice.length > 0 || data.byModel.length > 0) ? `
475
+ const sliceModelRow = (data.bySlice.length > 0 || data.byModel.length > 0) ? `
553
476
  <div class="chart-row">
554
477
  ${data.bySlice.length > 0 ? buildBarChart('Cost by slice', data.bySlice.map(s => ({
555
478
  label: s.sliceId, value: s.cost, display: formatCost(s.cost),
556
479
  sub: `${s.units} units`,
557
- }))) : ''}
480
+ }))) : ''}
558
481
  ${data.byModel.length > 0 ? buildBarChart('Cost by model', data.byModel.map(m => ({
559
482
  label: shortModel(m.model), value: m.cost, display: formatCost(m.cost),
560
483
  sub: `${m.units} units`,
561
- }))) : ''}
484
+ }))) : ''}
562
485
  ${data.bySlice.length > 0 ? buildBarChart('Duration by slice', data.bySlice.map(s => ({
563
486
  label: s.sliceId, value: s.duration, display: formatDuration(s.duration),
564
487
  sub: formatCost(s.cost),
565
- }))) : ''}
488
+ }))) : ''}
566
489
  </div>` : '';
567
-
568
- const costOverTime = buildCostOverTimeChart(data.units);
569
- const budgetBurndown = buildBudgetBurndown(data);
570
- const gantt = buildSliceGantt(data);
571
-
572
- return section('metrics', 'Metrics', `
490
+ const costOverTime = buildCostOverTimeChart(data.units);
491
+ const budgetBurndown = buildBudgetBurndown(data);
492
+ const gantt = buildSliceGantt(data);
493
+ return section('metrics', 'Metrics', `
573
494
  <div class="kv-grid">${grid}</div>
574
495
  ${budgetBurndown}
575
496
  ${tokenBreakdown}
@@ -579,42 +500,37 @@ function buildMetricsSection(data: VisualizerData): string {
579
500
  ${gantt}
580
501
  `);
581
502
  }
582
-
583
- function buildCostOverTimeChart(units: UnitMetrics[]): string {
584
- if (units.length < 2) return '';
585
- const sorted = [...units].sort((a, b) => a.startedAt - b.startedAt);
586
- const cumulative: number[] = [];
587
- let running = 0;
588
- for (const u of sorted) {
589
- running += u.cost;
590
- cumulative.push(running);
591
- }
592
-
593
- const padL = 50, padR = 30, padT = 20, padB = 30;
594
- const w = 600, h = 200;
595
- const plotW = w - padL - padR;
596
- const plotH = h - padT - padB;
597
- const maxCost = cumulative[cumulative.length - 1] || 1;
598
- const n = cumulative.length;
599
-
600
- const points = cumulative.map((c, i) => {
601
- const x = padL + (i / (n - 1)) * plotW;
602
- const y = padT + plotH - (c / maxCost) * plotH;
603
- return { x, y };
604
- });
605
-
606
- const linePath = points.map((p, i) => `${i === 0 ? 'M' : 'L'}${p.x.toFixed(1)},${p.y.toFixed(1)}`).join(' ');
607
- const areaPath = `${linePath} L${points[points.length - 1].x.toFixed(1)},${(padT + plotH).toFixed(1)} L${points[0].x.toFixed(1)},${(padT + plotH).toFixed(1)} Z`;
608
-
609
- const gridLines: string[] = [];
610
- for (let i = 0; i <= 4; i++) {
611
- const y = padT + (plotH / 4) * i;
612
- const val = formatCost(maxCost * (1 - i / 4));
613
- gridLines.push(`<line x1="${padL}" y1="${y}" x2="${w - padR}" y2="${y}" class="cost-grid"/>`);
614
- gridLines.push(`<text x="${padL - 4}" y="${y + 3}" class="cost-axis" text-anchor="end">${val}</text>`);
615
- }
616
-
617
- return `
503
+ function buildCostOverTimeChart(units) {
504
+ if (units.length < 2)
505
+ return '';
506
+ const sorted = [...units].sort((a, b) => a.startedAt - b.startedAt);
507
+ const cumulative = [];
508
+ let running = 0;
509
+ for (const u of sorted) {
510
+ running += u.cost;
511
+ cumulative.push(running);
512
+ }
513
+ const padL = 50, padR = 30, padT = 20, padB = 30;
514
+ const w = 600, h = 200;
515
+ const plotW = w - padL - padR;
516
+ const plotH = h - padT - padB;
517
+ const maxCost = cumulative[cumulative.length - 1] || 1;
518
+ const n = cumulative.length;
519
+ const points = cumulative.map((c, i) => {
520
+ const x = padL + (i / (n - 1)) * plotW;
521
+ const y = padT + plotH - (c / maxCost) * plotH;
522
+ return { x, y };
523
+ });
524
+ const linePath = points.map((p, i) => `${i === 0 ? 'M' : 'L'}${p.x.toFixed(1)},${p.y.toFixed(1)}`).join(' ');
525
+ const areaPath = `${linePath} L${points[points.length - 1].x.toFixed(1)},${(padT + plotH).toFixed(1)} L${points[0].x.toFixed(1)},${(padT + plotH).toFixed(1)} Z`;
526
+ const gridLines = [];
527
+ for (let i = 0; i <= 4; i++) {
528
+ const y = padT + (plotH / 4) * i;
529
+ const val = formatCost(maxCost * (1 - i / 4));
530
+ gridLines.push(`<line x1="${padL}" y1="${y}" x2="${w - padR}" y2="${y}" class="cost-grid"/>`);
531
+ gridLines.push(`<text x="${padL - 4}" y="${y + 3}" class="cost-axis" text-anchor="end">${val}</text>`);
532
+ }
533
+ return `
618
534
  <div class="token-block">
619
535
  <h3>Cost over time</h3>
620
536
  <svg class="cost-svg" viewBox="0 0 ${w} ${h}" width="${w}" height="${h}">
@@ -626,30 +542,27 @@ function buildCostOverTimeChart(units: UnitMetrics[]): string {
626
542
  </svg>
627
543
  </div>`;
628
544
  }
629
-
630
- function buildBudgetBurndown(data: VisualizerData): string {
631
- if (!data.health.budgetCeiling) return '';
632
- const ceiling = data.health.budgetCeiling;
633
- const spent = data.totals?.cost ?? 0;
634
- const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
635
- const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
636
- const avgCostPerSlice = doneSlices > 0 ? spent / doneSlices : 0;
637
- const projected = avgCostPerSlice > 0 ? avgCostPerSlice * data.remainingSliceCount + spent : spent;
638
- const maxVal = Math.max(ceiling, projected, spent);
639
-
640
- const spentPct = (spent / maxVal) * 100;
641
- const projectedRemPct = Math.max(0, ((projected - spent) / maxVal) * 100);
642
- const overshoot = projected > ceiling ? ((projected - ceiling) / maxVal) * 100 : 0;
643
- const projectedClean = projectedRemPct - overshoot;
644
-
645
- const legend = [
646
- `<span><span class="burndown-dot" style="background:var(--accent)"></span> Spent: ${formatCost(spent)}</span>`,
647
- `<span><span class="burndown-dot" style="background:var(--caution)"></span> Projected remaining: ${formatCost(Math.max(0, projected - spent))}</span>`,
648
- `<span><span class="burndown-dot" style="background:var(--border-2)"></span> Ceiling: ${formatCost(ceiling)}</span>`,
649
- overshoot > 0 ? `<span><span class="burndown-dot" style="background:var(--warn)"></span> Overshoot: ${formatCost(projected - ceiling)}</span>` : '',
650
- ].filter(Boolean).join('');
651
-
652
- return `
545
+ function buildBudgetBurndown(data) {
546
+ if (!data.health.budgetCeiling)
547
+ return '';
548
+ const ceiling = data.health.budgetCeiling;
549
+ const spent = data.totals?.cost ?? 0;
550
+ const totalSlices = data.milestones.reduce((s, m) => s + m.slices.length, 0);
551
+ const doneSlices = data.milestones.reduce((s, m) => s + m.slices.filter(sl => sl.done).length, 0);
552
+ const avgCostPerSlice = doneSlices > 0 ? spent / doneSlices : 0;
553
+ const projected = avgCostPerSlice > 0 ? avgCostPerSlice * data.remainingSliceCount + spent : spent;
554
+ const maxVal = Math.max(ceiling, projected, spent);
555
+ const spentPct = (spent / maxVal) * 100;
556
+ const projectedRemPct = Math.max(0, ((projected - spent) / maxVal) * 100);
557
+ const overshoot = projected > ceiling ? ((projected - ceiling) / maxVal) * 100 : 0;
558
+ const projectedClean = projectedRemPct - overshoot;
559
+ const legend = [
560
+ `<span><span class="burndown-dot" style="background:var(--accent)"></span> Spent: ${formatCost(spent)}</span>`,
561
+ `<span><span class="burndown-dot" style="background:var(--caution)"></span> Projected remaining: ${formatCost(Math.max(0, projected - spent))}</span>`,
562
+ `<span><span class="burndown-dot" style="background:var(--border-2)"></span> Ceiling: ${formatCost(ceiling)}</span>`,
563
+ overshoot > 0 ? `<span><span class="burndown-dot" style="background:var(--warn)"></span> Overshoot: ${formatCost(projected - ceiling)}</span>` : '',
564
+ ].filter(Boolean).join('');
565
+ return `
653
566
  <div class="burndown-wrap">
654
567
  <h3>Budget burndown</h3>
655
568
  <div class="burndown-bar">
@@ -660,61 +573,56 @@ function buildBudgetBurndown(data: VisualizerData): string {
660
573
  <div class="burndown-legend">${legend}</div>
661
574
  </div>`;
662
575
  }
663
-
664
- function buildSliceGantt(data: VisualizerData): string {
665
- const sliceTimings = new Map<string, { min: number; max: number }>();
666
- for (const u of data.units) {
667
- const parts = u.id.split('/');
668
- const sliceKey = parts.length >= 2 ? `${parts[0]}/${parts[1]}` : u.id;
669
- if (u.startedAt <= 0) continue;
670
- const existing = sliceTimings.get(sliceKey);
671
- const end = u.finishedAt > 0 ? u.finishedAt : Date.now();
672
- if (existing) {
673
- existing.min = Math.min(existing.min, u.startedAt);
674
- existing.max = Math.max(existing.max, end);
675
- } else {
676
- sliceTimings.set(sliceKey, { min: u.startedAt, max: end });
576
+ function buildSliceGantt(data) {
577
+ const sliceTimings = new Map();
578
+ for (const u of data.units) {
579
+ const parts = u.id.split('/');
580
+ const sliceKey = parts.length >= 2 ? `${parts[0]}/${parts[1]}` : u.id;
581
+ if (u.startedAt <= 0)
582
+ continue;
583
+ const existing = sliceTimings.get(sliceKey);
584
+ const end = u.finishedAt > 0 ? u.finishedAt : Date.now();
585
+ if (existing) {
586
+ existing.min = Math.min(existing.min, u.startedAt);
587
+ existing.max = Math.max(existing.max, end);
588
+ }
589
+ else {
590
+ sliceTimings.set(sliceKey, { min: u.startedAt, max: end });
591
+ }
677
592
  }
678
- }
679
-
680
- if (sliceTimings.size < 2) return '';
681
-
682
- const sliceEntries = [...sliceTimings.entries()].sort((a, b) => a[1].min - b[1].min);
683
- const globalMin = Math.min(...sliceEntries.map(e => e[1].min));
684
- const globalMax = Math.max(...sliceEntries.map(e => e[1].max));
685
- const range = globalMax - globalMin || 1;
686
-
687
- const sliceCount = sliceEntries.length;
688
- const barH = 18, rowH = 30, padL = 140, padR = 20, padT = 30, padB = 30;
689
- const plotW = 700 - padL - padR;
690
- const svgH = sliceCount * rowH + padT + padB;
691
-
692
- // Build a lookup of slice status
693
- const sliceStatusMap = new Map<string, string>();
694
- for (const ms of data.milestones) {
695
- for (const sl of ms.slices) {
696
- const key = `${ms.id}/${sl.id}`;
697
- sliceStatusMap.set(key, sl.done ? 'done' : sl.active ? 'active' : 'pending');
593
+ if (sliceTimings.size < 2)
594
+ return '';
595
+ const sliceEntries = [...sliceTimings.entries()].sort((a, b) => a[1].min - b[1].min);
596
+ const globalMin = Math.min(...sliceEntries.map(e => e[1].min));
597
+ const globalMax = Math.max(...sliceEntries.map(e => e[1].max));
598
+ const range = globalMax - globalMin || 1;
599
+ const sliceCount = sliceEntries.length;
600
+ const barH = 18, rowH = 30, padL = 140, padR = 20, padT = 30, padB = 30;
601
+ const plotW = 700 - padL - padR;
602
+ const svgH = sliceCount * rowH + padT + padB;
603
+ // Build a lookup of slice status
604
+ const sliceStatusMap = new Map();
605
+ for (const ms of data.milestones) {
606
+ for (const sl of ms.slices) {
607
+ const key = `${ms.id}/${sl.id}`;
608
+ sliceStatusMap.set(key, sl.done ? 'done' : sl.active ? 'active' : 'pending');
609
+ }
698
610
  }
699
- }
700
-
701
- const bars = sliceEntries.map(([sliceId, timing], i) => {
702
- const x = padL + ((timing.min - globalMin) / range) * plotW;
703
- const w = Math.max(2, ((timing.max - timing.min) / range) * plotW);
704
- const y = padT + i * rowH + (rowH - barH) / 2;
705
- const status = sliceStatusMap.get(sliceId) ?? 'pending';
706
- return `<text x="${padL - 6}" y="${y + barH / 2 + 4}" class="gantt-label" text-anchor="end">${esc(truncStr(sliceId, 18))}</text>
611
+ const bars = sliceEntries.map(([sliceId, timing], i) => {
612
+ const x = padL + ((timing.min - globalMin) / range) * plotW;
613
+ const w = Math.max(2, ((timing.max - timing.min) / range) * plotW);
614
+ const y = padT + i * rowH + (rowH - barH) / 2;
615
+ const status = sliceStatusMap.get(sliceId) ?? 'pending';
616
+ return `<text x="${padL - 6}" y="${y + barH / 2 + 4}" class="gantt-label" text-anchor="end">${esc(truncStr(sliceId, 18))}</text>
707
617
  <rect x="${x.toFixed(1)}" y="${y.toFixed(1)}" width="${w.toFixed(1)}" height="${barH}" rx="2" class="gantt-bar-${status}"><title>${esc(sliceId)}: ${formatDuration(timing.max - timing.min)}</title></rect>`;
708
- }).join('\n');
709
-
710
- // Time axis labels
711
- const axisLabels = [0, 0.25, 0.5, 0.75, 1].map(frac => {
712
- const t = globalMin + frac * range;
713
- const x = padL + frac * plotW;
714
- return `<text x="${x.toFixed(1)}" y="${svgH - 8}" class="gantt-axis" text-anchor="middle">${formatDateShort(new Date(t).toISOString())}</text>`;
715
- }).join('');
716
-
717
- return `
618
+ }).join('\n');
619
+ // Time axis labels
620
+ const axisLabels = [0, 0.25, 0.5, 0.75, 1].map(frac => {
621
+ const t = globalMin + frac * range;
622
+ const x = padL + frac * plotW;
623
+ return `<text x="${x.toFixed(1)}" y="${svgH - 8}" class="gantt-axis" text-anchor="middle">${formatDateShort(new Date(t).toISOString())}</text>`;
624
+ }).join('');
625
+ return `
718
626
  <div class="gantt-wrap">
719
627
  <h3>Slice timeline</h3>
720
628
  <svg class="gantt-svg" viewBox="0 0 700 ${svgH}" width="700" height="${svgH}">
@@ -723,69 +631,60 @@ function buildSliceGantt(data: VisualizerData): string {
723
631
  </svg>
724
632
  </div>`;
725
633
  }
726
-
727
- function buildTokenBreakdown(tokens: { input: number; output: number; cacheRead: number; cacheWrite: number; total: number }): string {
728
- if (tokens.total === 0) return '';
729
- const segs = [
730
- { label: 'Input', value: tokens.input, cls: 'seg-1' },
731
- { label: 'Output', value: tokens.output, cls: 'seg-2' },
732
- { label: 'Cache read', value: tokens.cacheRead, cls: 'seg-3' },
733
- { label: 'Cache write', value: tokens.cacheWrite, cls: 'seg-4' },
734
- ].filter(s => s.value > 0);
735
-
736
- const bars = segs.map(s => {
737
- const pct = (s.value / tokens.total) * 100;
738
- return `<div class="tseg ${s.cls}" style="width:${pct.toFixed(2)}%" title="${s.label}: ${formatTokenCount(s.value)} (${pct.toFixed(1)}%)"></div>`;
739
- }).join('');
740
-
741
- const legend = segs.map(s => {
742
- const pct = ((s.value / tokens.total) * 100).toFixed(1);
743
- return `<span class="leg-item"><span class="leg-dot ${s.cls}"></span>${s.label}: ${formatTokenCount(s.value)} (${pct}%)</span>`;
744
- }).join('');
745
-
746
- return `
634
+ function buildTokenBreakdown(tokens) {
635
+ if (tokens.total === 0)
636
+ return '';
637
+ const segs = [
638
+ { label: 'Input', value: tokens.input, cls: 'seg-1' },
639
+ { label: 'Output', value: tokens.output, cls: 'seg-2' },
640
+ { label: 'Cache read', value: tokens.cacheRead, cls: 'seg-3' },
641
+ { label: 'Cache write', value: tokens.cacheWrite, cls: 'seg-4' },
642
+ ].filter(s => s.value > 0);
643
+ const bars = segs.map(s => {
644
+ const pct = (s.value / tokens.total) * 100;
645
+ return `<div class="tseg ${s.cls}" style="width:${pct.toFixed(2)}%" title="${s.label}: ${formatTokenCount(s.value)} (${pct.toFixed(1)}%)"></div>`;
646
+ }).join('');
647
+ const legend = segs.map(s => {
648
+ const pct = ((s.value / tokens.total) * 100).toFixed(1);
649
+ return `<span class="leg-item"><span class="leg-dot ${s.cls}"></span>${s.label}: ${formatTokenCount(s.value)} (${pct}%)</span>`;
650
+ }).join('');
651
+ return `
747
652
  <div class="token-block">
748
653
  <h3>Token breakdown</h3>
749
654
  <div class="token-bar">${bars}</div>
750
655
  <div class="token-legend">${legend}</div>
751
656
  </div>`;
752
657
  }
753
-
754
- interface BarEntry { label: string; value: number; display: string; sub?: string; color?: number }
755
-
756
658
  const CHART_COLORS = 6;
757
-
758
- function buildBarChart(title: string, entries: BarEntry[]): string {
759
- if (entries.length === 0) return '';
760
- const max = Math.max(...entries.map(e => e.value), 1);
761
- const rows = entries.map((e, i) => {
762
- const pct = (e.value / max) * 100;
763
- const ci = e.color ?? i;
764
- return `
659
+ function buildBarChart(title, entries) {
660
+ if (entries.length === 0)
661
+ return '';
662
+ const max = Math.max(...entries.map(e => e.value), 1);
663
+ const rows = entries.map((e, i) => {
664
+ const pct = (e.value / max) * 100;
665
+ const ci = e.color ?? i;
666
+ return `
765
667
  <div class="bar-row">
766
668
  <div class="bar-lbl">${esc(truncStr(e.label, 22))}</div>
767
669
  <div class="bar-track"><div class="bar-fill bar-c${ci % CHART_COLORS}" style="width:${pct.toFixed(1)}%"></div></div>
768
670
  <div class="bar-val">${esc(e.display)}</div>
769
671
  </div>
770
672
  ${e.sub ? `<div class="bar-sub">${esc(e.sub)}</div>` : ''}`;
771
- }).join('');
772
- return `<div class="chart-block"><h3>${esc(title)}</h3>${rows}</div>`;
673
+ }).join('');
674
+ return `<div class="chart-block"><h3>${esc(title)}</h3>${rows}</div>`;
773
675
  }
774
-
775
676
  // ─── Section: Timeline ────────────────────────────────────────────────────────
776
-
777
- function buildTimelineSection(data: VisualizerData): string {
778
- if (data.units.length === 0) return section('timeline', 'Timeline', '<p class="empty">No units executed yet.</p>');
779
-
780
- const sorted = [...data.units].sort((a, b) => a.startedAt - b.startedAt);
781
- const maxCost = Math.max(...sorted.map(u => u.cost), 0.01);
782
-
783
- const rows = sorted.map((u, i) => {
784
- const dur = u.finishedAt > 0 ? formatDuration(u.finishedAt - u.startedAt) : 'running';
785
- // Cost heatmap: subtle red background for expensive rows
786
- const intensity = Math.min(u.cost / maxCost, 1);
787
- const heatStyle = intensity > 0.15 ? ` style="background:rgba(239,68,68,${(intensity * 0.15).toFixed(3)})"` : '';
788
- return `
677
+ function buildTimelineSection(data) {
678
+ if (data.units.length === 0)
679
+ return section('timeline', 'Timeline', '<p class="empty">No units executed yet.</p>');
680
+ const sorted = [...data.units].sort((a, b) => a.startedAt - b.startedAt);
681
+ const maxCost = Math.max(...sorted.map(u => u.cost), 0.01);
682
+ const rows = sorted.map((u, i) => {
683
+ const dur = u.finishedAt > 0 ? formatDuration(u.finishedAt - u.startedAt) : 'running';
684
+ // Cost heatmap: subtle red background for expensive rows
685
+ const intensity = Math.min(u.cost / maxCost, 1);
686
+ const heatStyle = intensity > 0.15 ? ` style="background:rgba(239,68,68,${(intensity * 0.15).toFixed(3)})"` : '';
687
+ return `
789
688
  <tr${heatStyle}>
790
689
  <td class="muted">${i + 1}</td>
791
690
  <td class="mono">${esc(u.type)}</td>
@@ -801,9 +700,8 @@ function buildTimelineSection(data: VisualizerData): string {
801
700
  <td class="num">${(u.truncationSections ?? 0) > 0 ? u.truncationSections : ''}</td>
802
701
  <td>${u.continueHereFired ? 'yes' : ''}</td>
803
702
  </tr>`;
804
- }).join('');
805
-
806
- return section('timeline', 'Timeline', `
703
+ }).join('');
704
+ return section('timeline', 'Timeline', `
807
705
  <div class="table-scroll">
808
706
  <table class="tbl">
809
707
  <thead><tr>
@@ -815,28 +713,24 @@ function buildTimelineSection(data: VisualizerData): string {
815
713
  </table>
816
714
  </div>`);
817
715
  }
818
-
819
716
  // ─── Section: Changelog ───────────────────────────────────────────────────────
820
-
821
- function buildChangelogSection(data: VisualizerData): string {
822
- if (data.changelog.entries.length === 0) return section('changelog', 'Changelog', '<p class="empty">No completed slices yet.</p>');
823
-
824
- const entries = data.changelog.entries.map(e => {
825
- const filesHtml = e.filesModified.length > 0 ? `
717
+ function buildChangelogSection(data) {
718
+ if (data.changelog.entries.length === 0)
719
+ return section('changelog', 'Changelog', '<p class="empty">No completed slices yet.</p>');
720
+ const entries = data.changelog.entries.map(e => {
721
+ const filesHtml = e.filesModified.length > 0 ? `
826
722
  <details class="files-detail">
827
723
  <summary class="muted">${e.filesModified.length} file${e.filesModified.length !== 1 ? 's' : ''} modified</summary>
828
724
  <ul class="file-list">
829
725
  ${e.filesModified.map(f => `<li><code>${esc(f.path)}</code>${f.description ? ` — ${esc(f.description)}` : ''}</li>`).join('')}
830
726
  </ul>
831
727
  </details>` : '';
832
-
833
- const ver = data.sliceVerifications.find(v => v.sliceId === e.sliceId);
834
- const decisionsHtml = ver?.keyDecisions?.length ? `
728
+ const ver = data.sliceVerifications.find(v => v.sliceId === e.sliceId);
729
+ const decisionsHtml = ver?.keyDecisions?.length ? `
835
730
  <div class="detail-block"><span class="detail-label">Decisions</span>
836
731
  <ul>${ver.keyDecisions.map(d => `<li>${esc(d)}</li>`).join('')}</ul>
837
732
  </div>` : '';
838
-
839
- return `
733
+ return `
840
734
  <div class="cl-entry">
841
735
  <div class="cl-header">
842
736
  <span class="mono muted">${esc(e.milestoneId)}/${esc(e.sliceId)}</span>
@@ -847,54 +741,46 @@ function buildChangelogSection(data: VisualizerData): string {
847
741
  ${decisionsHtml}
848
742
  ${filesHtml}
849
743
  </div>`;
850
- }).join('');
851
-
852
- return section('changelog', `Changelog <span class="count">${data.changelog.entries.length}</span>`, entries);
744
+ }).join('');
745
+ return section('changelog', `Changelog <span class="count">${data.changelog.entries.length}</span>`, entries);
853
746
  }
854
-
855
747
  // ─── Section: Knowledge ───────────────────────────────────────────────────────
856
-
857
- function buildKnowledgeSection(data: VisualizerData): string {
858
- const k = data.knowledge;
859
- if (!k.exists) return section('knowledge', 'Knowledge', '<p class="empty">No KNOWLEDGE.md found.</p>');
860
- const total = k.rules.length + k.patterns.length + k.lessons.length;
861
- if (total === 0) return section('knowledge', 'Knowledge', '<p class="empty">KNOWLEDGE.md exists but no entries parsed.</p>');
862
-
863
- const rulesHtml = k.rules.length > 0 ? `
748
+ function buildKnowledgeSection(data) {
749
+ const k = data.knowledge;
750
+ if (!k.exists)
751
+ return section('knowledge', 'Knowledge', '<p class="empty">No KNOWLEDGE.md found.</p>');
752
+ const total = k.rules.length + k.patterns.length + k.lessons.length;
753
+ if (total === 0)
754
+ return section('knowledge', 'Knowledge', '<p class="empty">KNOWLEDGE.md exists but no entries parsed.</p>');
755
+ const rulesHtml = k.rules.length > 0 ? `
864
756
  <h3>Rules <span class="count">${k.rules.length}</span></h3>
865
757
  <table class="tbl">
866
758
  <thead><tr><th>ID</th><th>Scope</th><th>Rule</th></tr></thead>
867
759
  <tbody>${k.rules.map(r => `<tr><td class="mono">${esc(r.id)}</td><td>${esc(r.scope)}</td><td>${esc(r.content)}</td></tr>`).join('')}</tbody>
868
760
  </table>` : '';
869
-
870
- const patternsHtml = k.patterns.length > 0 ? `
761
+ const patternsHtml = k.patterns.length > 0 ? `
871
762
  <h3>Patterns <span class="count">${k.patterns.length}</span></h3>
872
763
  <table class="tbl">
873
764
  <thead><tr><th>ID</th><th>Pattern</th></tr></thead>
874
765
  <tbody>${k.patterns.map(p => `<tr><td class="mono">${esc(p.id)}</td><td>${esc(p.content)}</td></tr>`).join('')}</tbody>
875
766
  </table>` : '';
876
-
877
- const lessonsHtml = k.lessons.length > 0 ? `
767
+ const lessonsHtml = k.lessons.length > 0 ? `
878
768
  <h3>Lessons <span class="count">${k.lessons.length}</span></h3>
879
769
  <table class="tbl">
880
770
  <thead><tr><th>ID</th><th>Lesson</th></tr></thead>
881
771
  <tbody>${k.lessons.map(l => `<tr><td class="mono">${esc(l.id)}</td><td>${esc(l.content)}</td></tr>`).join('')}</tbody>
882
772
  </table>` : '';
883
-
884
- return section('knowledge', `Knowledge <span class="count">${total}</span>`, `${rulesHtml}${patternsHtml}${lessonsHtml}`);
773
+ return section('knowledge', `Knowledge <span class="count">${total}</span>`, `${rulesHtml}${patternsHtml}${lessonsHtml}`);
885
774
  }
886
-
887
775
  // ─── Section: Captures ────────────────────────────────────────────────────────
888
-
889
- function buildCapturesSection(data: VisualizerData): string {
890
- const c = data.captures;
891
- if (c.totalCount === 0) return section('captures', 'Captures', '<p class="empty">No captures recorded.</p>');
892
-
893
- const badge = c.pendingCount > 0
894
- ? `<span class="count count-warn">${c.pendingCount} pending</span>`
895
- : `<span class="count">all triaged</span>`;
896
-
897
- const rows = c.entries.map(e => `
776
+ function buildCapturesSection(data) {
777
+ const c = data.captures;
778
+ if (c.totalCount === 0)
779
+ return section('captures', 'Captures', '<p class="empty">No captures recorded.</p>');
780
+ const badge = c.pendingCount > 0
781
+ ? `<span class="count count-warn">${c.pendingCount} pending</span>`
782
+ : `<span class="count">all triaged</span>`;
783
+ const rows = c.entries.map(e => `
898
784
  <tr>
899
785
  <td class="muted">${formatDateShort(new Date(e.timestamp).toISOString())}</td>
900
786
  <td class="mono">${esc(e.status)}</td>
@@ -905,8 +791,7 @@ function buildCapturesSection(data: VisualizerData): string {
905
791
  <td class="muted">${e.resolvedAt ? formatDateShort(e.resolvedAt) : ''}</td>
906
792
  <td>${e.executed !== undefined ? (e.executed ? 'yes' : 'no') : ''}</td>
907
793
  </tr>`).join('');
908
-
909
- return section('captures', `Captures ${badge}`, `
794
+ return section('captures', `Captures ${badge}`, `
910
795
  <div class="table-scroll">
911
796
  <table class="tbl">
912
797
  <thead><tr><th>Captured</th><th>Status</th><th>Class</th><th>Resolution</th><th>Text</th><th>Rationale</th><th>Resolved</th><th>Executed</th></tr></thead>
@@ -914,25 +799,21 @@ function buildCapturesSection(data: VisualizerData): string {
914
799
  </table>
915
800
  </div>`);
916
801
  }
917
-
918
802
  // ─── Section: Stats ───────────────────────────────────────────────────────────
919
-
920
- function buildStatsSection(data: VisualizerData): string {
921
- const s = data.stats;
922
-
923
- const missingHtml = s.missingCount > 0 ? `
803
+ function buildStatsSection(data) {
804
+ const s = data.stats;
805
+ const missingHtml = s.missingCount > 0 ? `
924
806
  <h3>Missing changelogs <span class="count">${s.missingCount}</span></h3>
925
807
  <table class="tbl">
926
808
  <thead><tr><th>Milestone</th><th>Slice</th><th>Title</th></tr></thead>
927
809
  <tbody>
928
810
  ${s.missingSlices.map(sl => `<tr><td class="mono">${esc(sl.milestoneId)}</td><td class="mono">${esc(sl.sliceId)}</td><td>${esc(sl.title)}</td></tr>`).join('')}
929
811
  ${s.missingCount > s.missingSlices.length
930
- ? `<tr><td colspan="3" class="muted">and ${s.missingCount - s.missingSlices.length} more</td></tr>`
931
- : ''}
812
+ ? `<tr><td colspan="3" class="muted">and ${s.missingCount - s.missingSlices.length} more</td></tr>`
813
+ : ''}
932
814
  </tbody>
933
815
  </table>` : '';
934
-
935
- const updatedHtml = s.updatedCount > 0 ? `
816
+ const updatedHtml = s.updatedCount > 0 ? `
936
817
  <h3>Recently completed <span class="count">${s.updatedCount}</span></h3>
937
818
  <table class="tbl">
938
819
  <thead><tr><th>Milestone</th><th>Slice</th><th>Title</th><th>Completed</th></tr></thead>
@@ -940,20 +821,16 @@ function buildStatsSection(data: VisualizerData): string {
940
821
  <tr><td class="mono">${esc(sl.milestoneId)}</td><td class="mono">${esc(sl.sliceId)}</td><td>${esc(sl.title)}</td><td class="muted">${sl.completedAt ? formatDateShort(sl.completedAt) : ''}</td></tr>`).join('')}
941
822
  </tbody>
942
823
  </table>` : '';
943
-
944
- if (!missingHtml && !updatedHtml) {
945
- return section('stats', 'Artifacts', '<p class="empty">All artifacts accounted for.</p>');
946
- }
947
-
948
- return section('stats', 'Artifacts', `${missingHtml}${updatedHtml}`);
824
+ if (!missingHtml && !updatedHtml) {
825
+ return section('stats', 'Artifacts', '<p class="empty">All artifacts accounted for.</p>');
826
+ }
827
+ return section('stats', 'Artifacts', `${missingHtml}${updatedHtml}`);
949
828
  }
950
-
951
829
  // ─── Section: Discussion ──────────────────────────────────────────────────────
952
-
953
- function buildDiscussionSection(data: VisualizerData): string {
954
- if (data.discussion.length === 0) return section('discussion', 'Planning', '<p class="empty">No milestones.</p>');
955
-
956
- const rows = data.discussion.map(d => `
830
+ function buildDiscussionSection(data) {
831
+ if (data.discussion.length === 0)
832
+ return section('discussion', 'Planning', '<p class="empty">No milestones.</p>');
833
+ const rows = data.discussion.map(d => `
957
834
  <tr>
958
835
  <td class="mono">${esc(d.milestoneId)}</td>
959
836
  <td>${esc(d.title)}</td>
@@ -962,48 +839,41 @@ function buildDiscussionSection(data: VisualizerData): string {
962
839
  <td>${d.hasDraft ? 'draft' : ''}</td>
963
840
  <td class="muted">${d.lastUpdated ? formatDateShort(d.lastUpdated) : ''}</td>
964
841
  </tr>`).join('');
965
-
966
- return section('discussion', 'Planning', `
842
+ return section('discussion', 'Planning', `
967
843
  <table class="tbl">
968
844
  <thead><tr><th>ID</th><th>Milestone</th><th>State</th><th>Context</th><th>Draft</th><th>Updated</th></tr></thead>
969
845
  <tbody>${rows}</tbody>
970
846
  </table>`);
971
847
  }
972
-
973
848
  // ─── Primitives ────────────────────────────────────────────────────────────────
974
-
975
- function section(id: string, title: string, body: string): string {
976
- return `\n<section id="${id}">\n <h2>${title}</h2>\n ${body}\n</section>`;
849
+ function section(id, title, body) {
850
+ return `\n<section id="${id}">\n <h2>${title}</h2>\n ${body}\n</section>`;
977
851
  }
978
-
979
- function kvi(label: string, value: string): string {
980
- return `<div class="kv"><span class="kv-val">${esc(value)}</span><span class="kv-lbl">${esc(label)}</span></div>`;
852
+ function kvi(label, value) {
853
+ return `<div class="kv"><span class="kv-val">${esc(value)}</span><span class="kv-lbl">${esc(label)}</span></div>`;
981
854
  }
982
-
983
- function hRow(label: string, value: string, status?: 'ok' | 'caution' | 'warn'): string {
984
- const cls = status ? ` class="h-${status}"` : '';
985
- return `<tr${cls}><td>${esc(label)}</td><td>${esc(value)}</td></tr>`;
855
+ function hRow(label, value, status) {
856
+ const cls = status ? ` class="h-${status}"` : '';
857
+ return `<tr${cls}><td>${esc(label)}</td><td>${esc(value)}</td></tr>`;
986
858
  }
987
-
988
- function shortModel(m: string) { return m.replace(/^claude-/, '').replace(/^anthropic\//, ''); }
989
- function truncStr(s: string, n: number) { return s.length > n ? s.slice(0, n - 1) + '\u2026' : s; }
990
-
991
- function formatDateLong(iso: string): string {
992
- try {
993
- const d = new Date(iso);
994
- return d.toLocaleString('en-US', { weekday: 'short', month: 'short', day: 'numeric', year: 'numeric', hour: '2-digit', minute: '2-digit', timeZoneName: 'short' });
995
- } catch { return iso; }
859
+ function shortModel(m) { return m.replace(/^claude-/, '').replace(/^anthropic\//, ''); }
860
+ function truncStr(s, n) { return s.length > n ? s.slice(0, n - 1) + '\u2026' : s; }
861
+ function formatDateLong(iso) {
862
+ try {
863
+ const d = new Date(iso);
864
+ return d.toLocaleString('en-US', { weekday: 'short', month: 'short', day: 'numeric', year: 'numeric', hour: '2-digit', minute: '2-digit', timeZoneName: 'short' });
865
+ }
866
+ catch {
867
+ return iso;
868
+ }
996
869
  }
997
-
998
-
999
- function esc(s: string | undefined | null): string {
1000
- if (s == null) return '';
1001
- return String(s).replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;').replace(/'/g, '&#39;');
870
+ function esc(s) {
871
+ if (s == null)
872
+ return '';
873
+ return String(s).replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;').replace(/'/g, '&#39;');
1002
874
  }
1003
-
1004
875
  // ─── CSS ───────────────────────────────────────────────────────────────────────
1005
876
  // Linear-inspired: restrained palette, one accent, no emoji, no gradients.
1006
-
1007
877
  const CSS = `
1008
878
  *,*::before,*::after{box-sizing:border-box;margin:0;padding:0}
1009
879
  :root{
@@ -1276,9 +1146,7 @@ footer{border-top:1px solid var(--border-1);padding:20px 32px;margin-top:40px}
1276
1146
  .table-scroll{overflow:visible}
1277
1147
  }
1278
1148
  `;
1279
-
1280
1149
  // ─── JS ────────────────────────────────────────────────────────────────────────
1281
-
1282
1150
  const JS = `
1283
1151
  (function(){
1284
1152
  const sections=document.querySelectorAll('section[id]');