@pellux/goodvibes-agent 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (398) hide show
  1. package/.goodvibes/GOODVIBES.md +35 -0
  2. package/.goodvibes/agents/reviewer.md +48 -0
  3. package/.goodvibes/skills/add-provider/SKILL.md +199 -0
  4. package/CHANGELOG.md +25 -0
  5. package/README.md +74 -0
  6. package/bin/goodvibes-agent.ts +2 -0
  7. package/docs/README.md +23 -0
  8. package/docs/deployment-and-services.md +57 -0
  9. package/docs/getting-started.md +53 -0
  10. package/docs/release-and-publishing.md +46 -0
  11. package/package.json +134 -0
  12. package/scripts/check-bun.sh +20 -0
  13. package/src/audio/player.ts +156 -0
  14. package/src/audio/spoken-turn-controller.ts +203 -0
  15. package/src/audio/spoken-turn-model-routing.ts +117 -0
  16. package/src/audio/spoken-turn-wiring.ts +44 -0
  17. package/src/audio/text-chunker.ts +110 -0
  18. package/src/cli/bundle-command.ts +227 -0
  19. package/src/cli/completion.ts +90 -0
  20. package/src/cli/config-overrides.ts +159 -0
  21. package/src/cli/endpoints.ts +63 -0
  22. package/src/cli/entrypoint.ts +172 -0
  23. package/src/cli/help.ts +299 -0
  24. package/src/cli/index.ts +11 -0
  25. package/src/cli/management-commands.ts +426 -0
  26. package/src/cli/management.ts +744 -0
  27. package/src/cli/network-posture.ts +46 -0
  28. package/src/cli/package-verification.ts +123 -0
  29. package/src/cli/parser.ts +369 -0
  30. package/src/cli/provider-auth-routes.ts +22 -0
  31. package/src/cli/provider-classification.ts +107 -0
  32. package/src/cli/redaction.ts +105 -0
  33. package/src/cli/service-command.ts +26 -0
  34. package/src/cli/service-posture.ts +482 -0
  35. package/src/cli/status.ts +383 -0
  36. package/src/cli/surface-command.ts +247 -0
  37. package/src/cli/tui-startup.ts +32 -0
  38. package/src/cli/types.ts +69 -0
  39. package/src/cli-flags.ts +21 -0
  40. package/src/config/goodvibes-home-audit.ts +465 -0
  41. package/src/config/index.ts +57 -0
  42. package/src/config/provider-model.ts +23 -0
  43. package/src/config/secret-config.ts +119 -0
  44. package/src/config/secrets.ts +71 -0
  45. package/src/config/surface.ts +1 -0
  46. package/src/core/composer-state.ts +61 -0
  47. package/src/core/conversation-rendering.ts +359 -0
  48. package/src/core/conversation.ts +551 -0
  49. package/src/core/history.ts +45 -0
  50. package/src/core/orchestrator.ts +7 -0
  51. package/src/core/system-message-router.ts +171 -0
  52. package/src/daemon/cli.ts +55 -0
  53. package/src/daemon/safe-serve.ts +61 -0
  54. package/src/input/agent-workspace.ts +428 -0
  55. package/src/input/autocomplete.ts +96 -0
  56. package/src/input/bookmark-modal.ts +115 -0
  57. package/src/input/command-args-hint.ts +36 -0
  58. package/src/input/command-registry.ts +329 -0
  59. package/src/input/commands/agent-externalized-tui.ts +73 -0
  60. package/src/input/commands/agent-workspace-runtime.ts +17 -0
  61. package/src/input/commands/branch-runtime.ts +72 -0
  62. package/src/input/commands/cloudflare-runtime.ts +370 -0
  63. package/src/input/commands/config.ts +18 -0
  64. package/src/input/commands/control-room-runtime.ts +255 -0
  65. package/src/input/commands/conversation-runtime.ts +207 -0
  66. package/src/input/commands/discovery-runtime.ts +52 -0
  67. package/src/input/commands/eval.ts +204 -0
  68. package/src/input/commands/experience-runtime.ts +278 -0
  69. package/src/input/commands/guidance-runtime.ts +106 -0
  70. package/src/input/commands/health-runtime.ts +434 -0
  71. package/src/input/commands/hooks-runtime.ts +148 -0
  72. package/src/input/commands/incident-runtime.ts +95 -0
  73. package/src/input/commands/integration-runtime.ts +394 -0
  74. package/src/input/commands/intelligence-runtime.ts +223 -0
  75. package/src/input/commands/knowledge.ts +531 -0
  76. package/src/input/commands/local-auth-runtime.ts +105 -0
  77. package/src/input/commands/local-provider-runtime.ts +170 -0
  78. package/src/input/commands/local-runtime.ts +392 -0
  79. package/src/input/commands/local-setup-review.ts +199 -0
  80. package/src/input/commands/local-setup-transfer.ts +135 -0
  81. package/src/input/commands/local-setup.ts +282 -0
  82. package/src/input/commands/managed-runtime.ts +209 -0
  83. package/src/input/commands/marketplace-runtime.ts +290 -0
  84. package/src/input/commands/mcp-runtime.ts +432 -0
  85. package/src/input/commands/memory-product-runtime.ts +111 -0
  86. package/src/input/commands/memory.ts +151 -0
  87. package/src/input/commands/notify-runtime.ts +83 -0
  88. package/src/input/commands/onboarding-runtime.ts +14 -0
  89. package/src/input/commands/operator-panel-runtime.ts +146 -0
  90. package/src/input/commands/operator-runtime.ts +392 -0
  91. package/src/input/commands/planning-runtime.ts +205 -0
  92. package/src/input/commands/platform-access-runtime.ts +422 -0
  93. package/src/input/commands/platform-services-runtime.ts +246 -0
  94. package/src/input/commands/policy-dispatch.ts +339 -0
  95. package/src/input/commands/policy.ts +17 -0
  96. package/src/input/commands/product-runtime.ts +351 -0
  97. package/src/input/commands/profile-sync-runtime.ts +99 -0
  98. package/src/input/commands/provider-accounts-runtime.ts +113 -0
  99. package/src/input/commands/provider.ts +363 -0
  100. package/src/input/commands/qrcode-runtime.ts +20 -0
  101. package/src/input/commands/quit-shared.ts +162 -0
  102. package/src/input/commands/recall-bundle.ts +132 -0
  103. package/src/input/commands/recall-capture.ts +152 -0
  104. package/src/input/commands/recall-query.ts +229 -0
  105. package/src/input/commands/recall-review.ts +98 -0
  106. package/src/input/commands/recall-shared.ts +22 -0
  107. package/src/input/commands/remote-runtime-pool.ts +106 -0
  108. package/src/input/commands/remote-runtime-setup.ts +199 -0
  109. package/src/input/commands/remote-runtime.ts +431 -0
  110. package/src/input/commands/replay-runtime.ts +18 -0
  111. package/src/input/commands/runtime-services.ts +291 -0
  112. package/src/input/commands/schedule-runtime.ts +91 -0
  113. package/src/input/commands/services-runtime.ts +209 -0
  114. package/src/input/commands/session-content.ts +408 -0
  115. package/src/input/commands/session-workflow.ts +464 -0
  116. package/src/input/commands/session.ts +375 -0
  117. package/src/input/commands/settings-sync-runtime.ts +174 -0
  118. package/src/input/commands/share-runtime.ts +119 -0
  119. package/src/input/commands/shell-core.ts +307 -0
  120. package/src/input/commands/skills-runtime.ts +221 -0
  121. package/src/input/commands/subscription-runtime.ts +434 -0
  122. package/src/input/commands/tasks-runtime.ts +230 -0
  123. package/src/input/commands/teamwork-runtime.ts +339 -0
  124. package/src/input/commands/teleport-runtime.ts +57 -0
  125. package/src/input/commands/tts-runtime.ts +29 -0
  126. package/src/input/commands/work-plan-runtime.ts +169 -0
  127. package/src/input/commands.ts +131 -0
  128. package/src/input/feed-context-factory.ts +254 -0
  129. package/src/input/file-picker.ts +192 -0
  130. package/src/input/handler-command-route.ts +180 -0
  131. package/src/input/handler-content-actions.ts +497 -0
  132. package/src/input/handler-feed-routes.ts +648 -0
  133. package/src/input/handler-feed.ts +452 -0
  134. package/src/input/handler-interactions.ts +281 -0
  135. package/src/input/handler-modal-routes.ts +418 -0
  136. package/src/input/handler-modal-stack.ts +263 -0
  137. package/src/input/handler-modal-token-routes.ts +329 -0
  138. package/src/input/handler-onboarding-cloudflare.ts +391 -0
  139. package/src/input/handler-onboarding.ts +620 -0
  140. package/src/input/handler-picker-routes.ts +472 -0
  141. package/src/input/handler-prompt-buffer.ts +320 -0
  142. package/src/input/handler-shortcuts.ts +213 -0
  143. package/src/input/handler-ui-state.ts +372 -0
  144. package/src/input/handler.ts +729 -0
  145. package/src/input/input-history.ts +297 -0
  146. package/src/input/keybindings.ts +292 -0
  147. package/src/input/mcp-workspace.ts +554 -0
  148. package/src/input/model-picker-provider-filter.ts +28 -0
  149. package/src/input/model-picker-types.ts +137 -0
  150. package/src/input/model-picker.ts +797 -0
  151. package/src/input/onboarding/handler-onboarding-routes.ts +125 -0
  152. package/src/input/onboarding/onboarding-runtime-status.ts +87 -0
  153. package/src/input/onboarding/onboarding-wizard-apply.ts +277 -0
  154. package/src/input/onboarding/onboarding-wizard-cloudflare-step.ts +494 -0
  155. package/src/input/onboarding/onboarding-wizard-cloudflare.ts +204 -0
  156. package/src/input/onboarding/onboarding-wizard-constants.ts +158 -0
  157. package/src/input/onboarding/onboarding-wizard-external-surface-extra-specs.ts +130 -0
  158. package/src/input/onboarding/onboarding-wizard-external-surfaces.ts +762 -0
  159. package/src/input/onboarding/onboarding-wizard-helpers.ts +167 -0
  160. package/src/input/onboarding/onboarding-wizard-rules.ts +256 -0
  161. package/src/input/onboarding/onboarding-wizard-state.ts +365 -0
  162. package/src/input/onboarding/onboarding-wizard-steps.ts +798 -0
  163. package/src/input/onboarding/onboarding-wizard-types.ts +195 -0
  164. package/src/input/onboarding/onboarding-wizard.ts +711 -0
  165. package/src/input/panel-integration-actions.ts +78 -0
  166. package/src/input/profile-picker-modal.ts +222 -0
  167. package/src/input/search.ts +100 -0
  168. package/src/input/selection-modal.ts +163 -0
  169. package/src/input/selection.ts +135 -0
  170. package/src/input/session-picker-modal.ts +136 -0
  171. package/src/input/settings-modal-behavior.ts +37 -0
  172. package/src/input/settings-modal-secrets.ts +41 -0
  173. package/src/input/settings-modal-subscriptions.ts +95 -0
  174. package/src/input/settings-modal-types.ts +91 -0
  175. package/src/input/settings-modal.ts +793 -0
  176. package/src/input/submission-intent.ts +17 -0
  177. package/src/input/submission-router.ts +59 -0
  178. package/src/input/tts-settings-actions.ts +100 -0
  179. package/src/main.ts +792 -0
  180. package/src/mcp/runtime-reload.ts +81 -0
  181. package/src/panels/agent-inspector-panel.ts +521 -0
  182. package/src/panels/agent-inspector-shared.ts +94 -0
  183. package/src/panels/agent-logs-panel.ts +559 -0
  184. package/src/panels/agent-logs-shared.ts +129 -0
  185. package/src/panels/approval-panel.ts +150 -0
  186. package/src/panels/automation-control-panel.ts +212 -0
  187. package/src/panels/base-panel.ts +254 -0
  188. package/src/panels/builtin/agent.ts +117 -0
  189. package/src/panels/builtin/development.ts +31 -0
  190. package/src/panels/builtin/knowledge.ts +26 -0
  191. package/src/panels/builtin/operations.ts +349 -0
  192. package/src/panels/builtin/session.ts +129 -0
  193. package/src/panels/builtin/shared.ts +274 -0
  194. package/src/panels/builtin-panels.ts +23 -0
  195. package/src/panels/cockpit-panel.ts +183 -0
  196. package/src/panels/communication-panel.ts +153 -0
  197. package/src/panels/confirm-state.ts +61 -0
  198. package/src/panels/context-visualizer-panel.ts +204 -0
  199. package/src/panels/control-plane-panel.ts +211 -0
  200. package/src/panels/cost-tracker-panel.ts +444 -0
  201. package/src/panels/debug-panel.ts +432 -0
  202. package/src/panels/diff-panel.ts +520 -0
  203. package/src/panels/docs-panel.ts +283 -0
  204. package/src/panels/eval-panel.ts +399 -0
  205. package/src/panels/file-explorer-panel.ts +584 -0
  206. package/src/panels/file-preview-panel.ts +434 -0
  207. package/src/panels/forensics-panel.ts +364 -0
  208. package/src/panels/git-panel.ts +638 -0
  209. package/src/panels/hooks-panel.ts +239 -0
  210. package/src/panels/incident-review-panel.ts +197 -0
  211. package/src/panels/index.ts +46 -0
  212. package/src/panels/intelligence-panel.ts +176 -0
  213. package/src/panels/knowledge-panel.ts +345 -0
  214. package/src/panels/local-auth-panel.ts +130 -0
  215. package/src/panels/marketplace-panel.ts +212 -0
  216. package/src/panels/memory-panel.ts +225 -0
  217. package/src/panels/ops-control-panel.ts +150 -0
  218. package/src/panels/ops-strategy-panel.ts +235 -0
  219. package/src/panels/orchestration-panel.ts +273 -0
  220. package/src/panels/panel-list-panel.ts +509 -0
  221. package/src/panels/panel-manager.ts +570 -0
  222. package/src/panels/panel-picker.ts +106 -0
  223. package/src/panels/plan-dashboard-panel.ts +274 -0
  224. package/src/panels/plugins-panel.ts +178 -0
  225. package/src/panels/policy-panel.ts +308 -0
  226. package/src/panels/polish.ts +717 -0
  227. package/src/panels/project-planning-panel.ts +711 -0
  228. package/src/panels/provider-account-snapshot.ts +259 -0
  229. package/src/panels/provider-accounts-panel.ts +218 -0
  230. package/src/panels/provider-health-domains.ts +215 -0
  231. package/src/panels/provider-health-panel.ts +727 -0
  232. package/src/panels/provider-health-tracker.ts +115 -0
  233. package/src/panels/provider-stats-panel.ts +366 -0
  234. package/src/panels/qr-panel.ts +182 -0
  235. package/src/panels/remote-panel.ts +449 -0
  236. package/src/panels/routes-panel.ts +178 -0
  237. package/src/panels/sandbox-panel.ts +283 -0
  238. package/src/panels/schedule-panel.ts +329 -0
  239. package/src/panels/scrollable-list-panel.ts +491 -0
  240. package/src/panels/search-focus.ts +32 -0
  241. package/src/panels/security-panel.ts +295 -0
  242. package/src/panels/services-panel.ts +231 -0
  243. package/src/panels/session-browser-panel.ts +400 -0
  244. package/src/panels/session-maintenance.ts +125 -0
  245. package/src/panels/settings-sync-panel.ts +120 -0
  246. package/src/panels/skills-panel.ts +431 -0
  247. package/src/panels/subscription-panel.ts +263 -0
  248. package/src/panels/symbol-outline-panel.ts +486 -0
  249. package/src/panels/system-messages-panel.ts +230 -0
  250. package/src/panels/tasks-panel.ts +399 -0
  251. package/src/panels/thinking-panel.ts +304 -0
  252. package/src/panels/token-budget-panel.ts +475 -0
  253. package/src/panels/tool-inspector-panel.ts +429 -0
  254. package/src/panels/types.ts +54 -0
  255. package/src/panels/watchers-panel.ts +193 -0
  256. package/src/panels/work-plan-panel.ts +175 -0
  257. package/src/panels/worktree-panel.ts +182 -0
  258. package/src/panels/wrfc-panel.ts +609 -0
  259. package/src/permissions/prompt.ts +165 -0
  260. package/src/planning/project-planning-coordinator.ts +543 -0
  261. package/src/plugins/loader.ts +15 -0
  262. package/src/renderer/agent-detail-modal.ts +331 -0
  263. package/src/renderer/agent-workspace.ts +238 -0
  264. package/src/renderer/ansi-sanitize.ts +76 -0
  265. package/src/renderer/autocomplete-overlay.ts +154 -0
  266. package/src/renderer/block-actions.ts +76 -0
  267. package/src/renderer/bookmark-modal.ts +101 -0
  268. package/src/renderer/bottom-bar.ts +58 -0
  269. package/src/renderer/buffer.ts +113 -0
  270. package/src/renderer/code-block.ts +373 -0
  271. package/src/renderer/compositor.ts +283 -0
  272. package/src/renderer/context-inspector.ts +219 -0
  273. package/src/renderer/conversation-layout.ts +67 -0
  274. package/src/renderer/conversation-overlays.ts +140 -0
  275. package/src/renderer/conversation-surface.ts +260 -0
  276. package/src/renderer/diff-view.ts +132 -0
  277. package/src/renderer/diff.ts +130 -0
  278. package/src/renderer/file-picker-overlay.ts +101 -0
  279. package/src/renderer/file-tree.ts +153 -0
  280. package/src/renderer/fullscreen-primitives.ts +130 -0
  281. package/src/renderer/fullscreen-workspace.ts +199 -0
  282. package/src/renderer/git-status.ts +89 -0
  283. package/src/renderer/help-overlay.ts +267 -0
  284. package/src/renderer/history-search-overlay.ts +73 -0
  285. package/src/renderer/layout-engine.ts +97 -0
  286. package/src/renderer/layout.ts +32 -0
  287. package/src/renderer/live-tail-modal.ts +156 -0
  288. package/src/renderer/markdown.ts +635 -0
  289. package/src/renderer/mcp-workspace.ts +237 -0
  290. package/src/renderer/modal-factory.ts +467 -0
  291. package/src/renderer/modal-utils.ts +24 -0
  292. package/src/renderer/model-picker-overlay.ts +473 -0
  293. package/src/renderer/model-workspace.ts +488 -0
  294. package/src/renderer/onboarding/onboarding-wizard.ts +615 -0
  295. package/src/renderer/overlay-box.ts +146 -0
  296. package/src/renderer/overlay-viewport.ts +104 -0
  297. package/src/renderer/panel-composite.ts +158 -0
  298. package/src/renderer/panel-picker-overlay.ts +202 -0
  299. package/src/renderer/panel-tab-bar.ts +69 -0
  300. package/src/renderer/panel-workspace-bar.ts +42 -0
  301. package/src/renderer/process-indicator.ts +96 -0
  302. package/src/renderer/process-modal.ts +656 -0
  303. package/src/renderer/process-summary.ts +67 -0
  304. package/src/renderer/profile-picker-modal.ts +129 -0
  305. package/src/renderer/progress.ts +98 -0
  306. package/src/renderer/qr-renderer.ts +120 -0
  307. package/src/renderer/search-overlay.ts +54 -0
  308. package/src/renderer/selection-modal-overlay.ts +214 -0
  309. package/src/renderer/semantic-diff.ts +369 -0
  310. package/src/renderer/session-picker-modal.ts +127 -0
  311. package/src/renderer/settings-modal-helpers.ts +193 -0
  312. package/src/renderer/settings-modal.ts +537 -0
  313. package/src/renderer/shell-surface.ts +88 -0
  314. package/src/renderer/status-glyphs.ts +21 -0
  315. package/src/renderer/status-token.ts +67 -0
  316. package/src/renderer/surface-layout.ts +101 -0
  317. package/src/renderer/syntax-highlighter.ts +542 -0
  318. package/src/renderer/system-message.ts +83 -0
  319. package/src/renderer/tab-strip.ts +108 -0
  320. package/src/renderer/text-layout.ts +31 -0
  321. package/src/renderer/thinking.ts +17 -0
  322. package/src/renderer/tool-call.ts +234 -0
  323. package/src/renderer/ui-factory.ts +524 -0
  324. package/src/renderer/ui-primitives.ts +96 -0
  325. package/src/runtime/bootstrap-command-context.ts +278 -0
  326. package/src/runtime/bootstrap-command-parts.ts +386 -0
  327. package/src/runtime/bootstrap-core.ts +540 -0
  328. package/src/runtime/bootstrap-hook-bridge.ts +112 -0
  329. package/src/runtime/bootstrap-shell.ts +283 -0
  330. package/src/runtime/bootstrap.ts +575 -0
  331. package/src/runtime/cloudflare-control-plane.ts +349 -0
  332. package/src/runtime/context.ts +142 -0
  333. package/src/runtime/diagnostics/panels/index.ts +24 -0
  334. package/src/runtime/diagnostics/panels/ops.ts +156 -0
  335. package/src/runtime/diagnostics/panels/panel-resources.ts +118 -0
  336. package/src/runtime/diagnostics/panels/policy.ts +177 -0
  337. package/src/runtime/index.ts +662 -0
  338. package/src/runtime/onboarding/apply.ts +642 -0
  339. package/src/runtime/onboarding/derivation.ts +534 -0
  340. package/src/runtime/onboarding/index.ts +7 -0
  341. package/src/runtime/onboarding/markers.ts +148 -0
  342. package/src/runtime/onboarding/snapshot.ts +406 -0
  343. package/src/runtime/onboarding/state.ts +141 -0
  344. package/src/runtime/onboarding/types.ts +404 -0
  345. package/src/runtime/onboarding/verify.ts +171 -0
  346. package/src/runtime/operator-token-cleanup.ts +27 -0
  347. package/src/runtime/perf/panel-contracts.ts +32 -0
  348. package/src/runtime/perf/panel-health-monitor.ts +18 -0
  349. package/src/runtime/sandbox-public-gaps.ts +358 -0
  350. package/src/runtime/services.ts +670 -0
  351. package/src/runtime/store/domains/domain-read-matrix.ts +15 -0
  352. package/src/runtime/store/domains/index.ts +222 -0
  353. package/src/runtime/store/domains/panels.ts +117 -0
  354. package/src/runtime/store/domains/ui-perf.ts +103 -0
  355. package/src/runtime/store/index.ts +305 -0
  356. package/src/runtime/store/selectors/index.ts +359 -0
  357. package/src/runtime/store/state.ts +145 -0
  358. package/src/runtime/surface-feature-flags.ts +65 -0
  359. package/src/runtime/terminal-output-guard.ts +228 -0
  360. package/src/runtime/ui/index.ts +39 -0
  361. package/src/runtime/ui/model-picker/data-provider.ts +182 -0
  362. package/src/runtime/ui/model-picker/health-enrichment.ts +228 -0
  363. package/src/runtime/ui/model-picker/index.ts +59 -0
  364. package/src/runtime/ui/model-picker/types.ts +149 -0
  365. package/src/runtime/ui/provider-health/data-provider.ts +244 -0
  366. package/src/runtime/ui/provider-health/fallback-visualizer.ts +71 -0
  367. package/src/runtime/ui/provider-health/index.ts +46 -0
  368. package/src/runtime/ui/provider-health/types.ts +146 -0
  369. package/src/runtime/ui-events.ts +1 -0
  370. package/src/runtime/ui-read-model-helpers.ts +1 -0
  371. package/src/runtime/ui-read-models-observability-maintenance.ts +1 -0
  372. package/src/runtime/ui-read-models-observability-options.ts +1 -0
  373. package/src/runtime/ui-read-models-observability-remote.ts +1 -0
  374. package/src/runtime/ui-read-models-observability-security.ts +1 -0
  375. package/src/runtime/ui-read-models-observability-system.ts +1 -0
  376. package/src/runtime/ui-read-models-observability.ts +1 -0
  377. package/src/runtime/ui-read-models.ts +61 -0
  378. package/src/runtime/ui-service-queries.ts +1 -0
  379. package/src/runtime/ui-services.ts +190 -0
  380. package/src/scripts/process-messages.ts +42 -0
  381. package/src/shell/blocking-input.ts +98 -0
  382. package/src/shell/service-settings-sync.ts +273 -0
  383. package/src/shell/ui-openers.ts +352 -0
  384. package/src/tools/index.ts +1 -0
  385. package/src/tools/wrfc-agent-guard.ts +49 -0
  386. package/src/types/grid.ts +48 -0
  387. package/src/types/sql-js.d.ts +15 -0
  388. package/src/utils/clipboard.ts +22 -0
  389. package/src/utils/splash-lines.ts +46 -0
  390. package/src/utils/terminal-width.ts +185 -0
  391. package/src/verification/live-verifier.ts +430 -0
  392. package/src/verification/verification-ledger.ts +242 -0
  393. package/src/version.ts +17 -0
  394. package/src/widget/index.ts +2 -0
  395. package/src/widget/types.ts +9 -0
  396. package/src/widget/widget.ts +8 -0
  397. package/src/work-plans/work-plan-store.ts +374 -0
  398. package/tsconfig.json +18 -0
@@ -0,0 +1,520 @@
1
+ // ---------------------------------------------------------------------------
2
+ // DiffPanel — unified diff view of agent file changes
3
+ // ---------------------------------------------------------------------------
4
+
5
+ import type { Line } from '../types/grid.ts';
6
+ import { createStyledCell, createEmptyLine } from '../types/grid.ts';
7
+ import { BasePanel } from './base-panel.ts';
8
+ import {
9
+ buildBodyText,
10
+ buildEmptyState,
11
+ buildPanelWorkspace,
12
+ resolveScrollablePanelSection,
13
+ buildStyledPanelLine,
14
+ type PanelWorkspaceSection,
15
+ DEFAULT_PANEL_PALETTE,
16
+ } from './polish.ts';
17
+
18
+ // ---------------------------------------------------------------------------
19
+ // Colour palette
20
+ // ---------------------------------------------------------------------------
21
+
22
+ const COLOR = {
23
+ addition: '#00ff88',
24
+ deletion: '#ff4444',
25
+ hunk: '#88aaff',
26
+ header: '#aaaaaa',
27
+ lineNum: '#555555',
28
+ lineNumAdd: '#00aa55',
29
+ lineNumDel: '#aa2222',
30
+ filename: '#ffffff',
31
+ tabActive: '#ffffff',
32
+ tabInactive: '#666666',
33
+ tabBg: '#222222',
34
+ context: '#888888',
35
+ statusBar: '#444444',
36
+ } as const;
37
+
38
+ // ---------------------------------------------------------------------------
39
+ // Types
40
+ // ---------------------------------------------------------------------------
41
+
42
+ interface DiffEntry {
43
+ filePath: string;
44
+ raw: string; // raw unified diff text
45
+ lines: ParsedLine[];
46
+ /** One-line semantic summary from computeSemanticDiff, if available. */
47
+ semanticSummary?: string;
48
+ }
49
+
50
+ type LineKind = 'addition' | 'deletion' | 'context' | 'hunk' | 'header';
51
+
52
+ interface ParsedLine {
53
+ kind: LineKind;
54
+ text: string;
55
+ beforeNum: number | null; // line number in original file
56
+ afterNum: number | null; // line number in new file
57
+ }
58
+
59
+ // ---------------------------------------------------------------------------
60
+ // Diff parser
61
+ // ---------------------------------------------------------------------------
62
+
63
+ function parseDiff(raw: string): ParsedLine[] {
64
+ const result: ParsedLine[] = [];
65
+ let before = 0;
66
+ let after = 0;
67
+
68
+ for (const line of raw.split('\n')) {
69
+ // Hunk header: @@ -a,b +c,d @@
70
+ const hunkMatch = line.match(/^@@\s+-([0-9]+)(?:,[0-9]+)?\s+\+([0-9]+)(?:,[0-9]+)?\s+@@/);
71
+ if (hunkMatch) {
72
+ before = parseInt(hunkMatch[1]!, 10);
73
+ after = parseInt(hunkMatch[2]!, 10);
74
+ result.push({ kind: 'hunk', text: line, beforeNum: null, afterNum: null });
75
+ continue;
76
+ }
77
+
78
+ if (line.startsWith('+++') || line.startsWith('---') ||
79
+ line.startsWith('diff ') || line.startsWith('index ') ||
80
+ line.startsWith('new file') || line.startsWith('old file') ||
81
+ line.startsWith('Binary')) {
82
+ result.push({ kind: 'header', text: line, beforeNum: null, afterNum: null });
83
+ continue;
84
+ }
85
+
86
+ if (line.startsWith('+')) {
87
+ result.push({ kind: 'addition', text: line.slice(1), beforeNum: null, afterNum: after });
88
+ after++;
89
+ } else if (line.startsWith('-')) {
90
+ result.push({ kind: 'deletion', text: line.slice(1), beforeNum: before, afterNum: null });
91
+ before++;
92
+ } else if (line.startsWith('\\')) {
93
+ // "No newline at end of file" note — treat as header
94
+ result.push({ kind: 'header', text: line, beforeNum: null, afterNum: null });
95
+ } else {
96
+ // context line (starts with space, or empty for blank context)
97
+ const text = line.startsWith(' ') ? line.slice(1) : line;
98
+ result.push({ kind: 'context', text, beforeNum: before, afterNum: after });
99
+ before++;
100
+ after++;
101
+ }
102
+ }
103
+
104
+ return result;
105
+ }
106
+
107
+ // ---------------------------------------------------------------------------
108
+ // Split a full `git diff` output into per-file entries
109
+ // ---------------------------------------------------------------------------
110
+
111
+ function splitIntoDiffEntries(raw: string): DiffEntry[] {
112
+ const entries: DiffEntry[] = [];
113
+ // Split on "diff --git" lines
114
+ const chunks = raw.split(/(?=^diff --git )/m);
115
+ for (const chunk of chunks) {
116
+ const trimmed = chunk.trim();
117
+ if (!trimmed) continue;
118
+
119
+ // Extract file path from "diff --git a/foo b/foo"
120
+ const match = trimmed.match(/^diff --git a\/.+? b\/(.+)$/m);
121
+ const filePath = match ? match[1]! : 'unknown';
122
+
123
+ entries.push({
124
+ filePath,
125
+ raw: chunk,
126
+ lines: parseDiff(chunk),
127
+ });
128
+ }
129
+ return entries;
130
+ }
131
+
132
+ // ---------------------------------------------------------------------------
133
+ // Rendering helpers
134
+ // ---------------------------------------------------------------------------
135
+
136
+ function makeLine(
137
+ width: number,
138
+ leftNum: string,
139
+ rightNum: string,
140
+ content: string,
141
+ fg: string,
142
+ bg: string,
143
+ numFg: string,
144
+ bold: boolean = false,
145
+ ): Line {
146
+ // Left line number (5 chars + space)
147
+ const LEFT_W = 5;
148
+ const usedForNums = LEFT_W + 1 + LEFT_W + 1 + 2; // 14
149
+ const contentWidth = Math.max(0, width - usedForNums);
150
+ const truncated = content.length > contentWidth
151
+ ? content.slice(0, contentWidth)
152
+ : content;
153
+ return buildStyledPanelLine(width, [
154
+ { text: leftNum.padStart(LEFT_W), fg: numFg, bg, dim: true },
155
+ { text: ' ', fg: '', bg },
156
+ { text: rightNum.padStart(LEFT_W), fg: numFg, bg, dim: true },
157
+ { text: ' ', fg: '', bg },
158
+ { text: '| ', fg: COLOR.lineNum, bg },
159
+ { text: truncated, fg, bg, bold },
160
+ ]);
161
+ }
162
+
163
+ function renderText(width: number, text: string, fg: string, bg: string, bold = false): Line {
164
+ const truncated = text.length > width ? text.slice(0, width) : text;
165
+ return buildStyledPanelLine(width, [{ text: truncated, fg, bg, bold }]);
166
+ }
167
+
168
+ // ---------------------------------------------------------------------------
169
+ // DiffPanel
170
+ // ---------------------------------------------------------------------------
171
+
172
+ export class DiffPanel extends BasePanel {
173
+ public override isTransient = true;
174
+
175
+ private readonly workingDirectory: string;
176
+ private entries: DiffEntry[] = [];
177
+ private selectedFile = 0;
178
+ private scrollOffset = 0;
179
+
180
+ constructor(workingDirectory: string) {
181
+ super('diff', 'Diff', 'D', 'development');
182
+ this.workingDirectory = workingDirectory;
183
+ }
184
+
185
+ // -------------------------------------------------------------------------
186
+ // Public API
187
+ // -------------------------------------------------------------------------
188
+
189
+ /** Show a unified diff for a specific file. Adds or replaces the entry. */
190
+ showDiff(filePath: string, diff: string): void {
191
+ const idx = this.entries.findIndex(e => e.filePath === filePath);
192
+ const entry: DiffEntry = { filePath, raw: diff, lines: parseDiff(diff), semanticSummary: this.entries[idx]?.semanticSummary };
193
+ if (idx >= 0) {
194
+ this.entries[idx] = entry;
195
+ // Stay on this file if it was already selected
196
+ if (this.selectedFile !== idx) {
197
+ this.selectedFile = idx;
198
+ this.scrollOffset = 0;
199
+ }
200
+ } else {
201
+ this.entries.push(entry);
202
+ this.selectedFile = this.entries.length - 1;
203
+ this.scrollOffset = 0;
204
+ }
205
+ this.markDirty();
206
+ }
207
+
208
+ /** Load a raw multi-file unified diff string directly. */
209
+ loadRawDiff(raw: string): void {
210
+ this.entries = splitIntoDiffEntries(raw);
211
+ this.selectedFile = 0;
212
+ this.scrollOffset = 0;
213
+ this.markDirty();
214
+ }
215
+
216
+ /** Run `git diff` against specific files and populate entries. */
217
+ async showFileDiffs(files: string[], ref?: string): Promise<void> {
218
+ const args = ['diff', ...(ref ? [ref] : []), '--', ...files];
219
+ const proc = Bun.spawn(['git', ...args], { stdout: 'pipe', cwd: this.workingDirectory });
220
+ const raw = await new Response(proc.stdout).text();
221
+ await proc.exited;
222
+ this.loadRawDiff(raw);
223
+ }
224
+
225
+ /** Run `git diff` and populate all changed files. */
226
+ async showGitDiff(ref?: string): Promise<void> {
227
+ const args = ['diff', ...(ref ? [ref] : [])];
228
+ const proc = Bun.spawn(['git', ...args], { stdout: 'pipe', stderr: 'pipe', cwd: this.workingDirectory });
229
+ const [raw, errText] = await Promise.all([
230
+ new Response(proc.stdout).text(),
231
+ new Response(proc.stderr).text(),
232
+ ]);
233
+ const exitCode = await proc.exited;
234
+ if (exitCode !== 0) {
235
+ const errorText = errText.trim() || 'git diff failed';
236
+ this.showDiff('(error)', `--- error\n+++ error\n@@ -0,0 +1,1 @@\n+${errorText}`);
237
+ return;
238
+ }
239
+ if (!raw.trim()) {
240
+ this.showDiff('(no changes)', '@@ -0,0 +0,0 @@\n No changes in working tree.');
241
+ return;
242
+ }
243
+ const newEntries = splitIntoDiffEntries(raw);
244
+ // Merge: update existing, append new
245
+ for (const entry of newEntries) {
246
+ const idx = this.entries.findIndex(e => e.filePath === entry.filePath);
247
+ if (idx >= 0) {
248
+ this.entries[idx] = { ...entry, semanticSummary: this.entries[idx]!.semanticSummary };
249
+ } else {
250
+ this.entries.push(entry);
251
+ }
252
+ }
253
+ this.selectedFile = Math.min(this.selectedFile, Math.max(0, this.entries.length - 1));
254
+ this.scrollOffset = 0;
255
+ this.markDirty();
256
+ }
257
+
258
+ /**
259
+ * Attach or update the semantic diff summary for a file entry.
260
+ * No-op if the file isn't currently loaded. Safe to call from an async
261
+ * callback after the entry has already been replaced.
262
+ */
263
+ setSemanticSummary(filePath: string, summary: string): void {
264
+ const entry = this.entries.find(e => e.filePath === filePath);
265
+ if (entry) {
266
+ entry.semanticSummary = summary;
267
+ this.markDirty();
268
+ }
269
+ }
270
+
271
+ /** Clear all diff entries. */
272
+ clear(): void {
273
+ this.entries = [];
274
+ this.selectedFile = 0;
275
+ this.scrollOffset = 0;
276
+ this.markDirty();
277
+ }
278
+
279
+ // -------------------------------------------------------------------------
280
+ // Lifecycle
281
+ // -------------------------------------------------------------------------
282
+
283
+ override onActivate(): void {
284
+ this.needsRender = true;
285
+ }
286
+
287
+ // -------------------------------------------------------------------------
288
+ // Input
289
+ // -------------------------------------------------------------------------
290
+
291
+ handleInput(key: string): boolean {
292
+ switch (key) {
293
+ case 'up': this.scrollUp(); return true;
294
+ case 'down': this.scrollDown(); return true;
295
+ case 'tab': this.nextFile(); return true;
296
+ case 'pageup': this.scrollPageUp(); return true;
297
+ case 'pagedown': this.scrollPageDown(); return true;
298
+ default: return false;
299
+ }
300
+ }
301
+
302
+ private scrollUp(): void {
303
+ if (this.scrollOffset > 0) {
304
+ this.scrollOffset--;
305
+ this.markDirty();
306
+ }
307
+ }
308
+
309
+ private scrollDown(): void {
310
+ const entry = this.currentEntry();
311
+ if (!entry) return;
312
+ const max = Math.max(0, entry.lines.length - 1);
313
+ if (this.scrollOffset < max) {
314
+ this.scrollOffset++;
315
+ this.markDirty();
316
+ }
317
+ }
318
+
319
+ private scrollPageUp(): void {
320
+ this.scrollOffset = Math.max(0, this.scrollOffset - 20);
321
+ this.markDirty();
322
+ }
323
+
324
+ private scrollPageDown(): void {
325
+ const entry = this.currentEntry();
326
+ if (!entry) return;
327
+ const max = Math.max(0, entry.lines.length - 1);
328
+ this.scrollOffset = Math.min(max, this.scrollOffset + 20);
329
+ this.markDirty();
330
+ }
331
+
332
+ private nextFile(): void {
333
+ if (this.entries.length === 0) return;
334
+ this.selectedFile = (this.selectedFile + 1) % this.entries.length;
335
+ this.scrollOffset = 0;
336
+ this.markDirty();
337
+ }
338
+
339
+ private currentEntry(): DiffEntry | null {
340
+ return this.entries[this.selectedFile] ?? null;
341
+ }
342
+
343
+ // -------------------------------------------------------------------------
344
+ // Render
345
+ // -------------------------------------------------------------------------
346
+
347
+ render(width: number, height: number): Line[] {
348
+ return this.trackedRender(() => {
349
+ if (height <= 0 || width <= 0) return [];
350
+
351
+ if (this.entries.length === 0) {
352
+ return buildPanelWorkspace(width, height, {
353
+ title: 'Diff Workspace',
354
+ palette: {
355
+ ...DEFAULT_PANEL_PALETTE,
356
+ info: COLOR.hunk,
357
+ dim: COLOR.context,
358
+ value: COLOR.filename,
359
+ },
360
+ sections: [{
361
+ title: 'Diff',
362
+ lines: buildEmptyState(
363
+ width,
364
+ ' No diff to display.',
365
+ 'Load a git diff or select a changed file to populate the workspace.',
366
+ [{ command: '/git diff', summary: 'load the current working-tree diff into the diff workspace' }],
367
+ {
368
+ ...DEFAULT_PANEL_PALETTE,
369
+ info: COLOR.hunk,
370
+ dim: COLOR.context,
371
+ value: COLOR.filename,
372
+ empty: COLOR.context,
373
+ },
374
+ ),
375
+ }],
376
+ });
377
+ }
378
+
379
+ const entry = this.currentEntry();
380
+ if (!entry) {
381
+ return Array.from({ length: height }, () => createEmptyLine(width));
382
+ }
383
+
384
+ const compact = height <= 12;
385
+ const summaryLines = entry.semanticSummary
386
+ ? buildBodyText(width, `Semantic summary: ${entry.semanticSummary}`, {
387
+ ...DEFAULT_PANEL_PALETTE,
388
+ dim: COLOR.context,
389
+ value: COLOR.filename,
390
+ }, COLOR.context)
391
+ : [];
392
+ const previewSection = resolveScrollablePanelSection(width, height, {
393
+ palette: {
394
+ ...DEFAULT_PANEL_PALETTE,
395
+ info: COLOR.hunk,
396
+ dim: COLOR.context,
397
+ value: COLOR.filename,
398
+ headerBg: COLOR.tabBg,
399
+ },
400
+ footerLines: [this.renderStatusBar(width, entry)],
401
+ beforeSections: [
402
+ {
403
+ title: compact ? undefined : 'Files',
404
+ lines: [
405
+ this.renderTabBar(width),
406
+ ...summaryLines,
407
+ ],
408
+ },
409
+ ],
410
+ section: {
411
+ title: compact ? undefined : 'Changes',
412
+ scrollableLines: entry.lines.map((pl) => this.renderParsedLine(pl, width)),
413
+ scrollOffset: this.scrollOffset,
414
+ minRows: 1,
415
+ },
416
+ });
417
+ this.scrollOffset = previewSection.scrollOffset;
418
+
419
+ const sections: PanelWorkspaceSection[] = [
420
+ {
421
+ title: compact ? undefined : 'Files',
422
+ lines: [
423
+ this.renderTabBar(width),
424
+ ...summaryLines,
425
+ ],
426
+ },
427
+ {
428
+ title: previewSection.section.title,
429
+ lines: previewSection.section.lines,
430
+ },
431
+ ];
432
+ return buildPanelWorkspace(width, height, {
433
+ title: 'Diff Workspace',
434
+ palette: {
435
+ ...DEFAULT_PANEL_PALETTE,
436
+ info: COLOR.hunk,
437
+ dim: COLOR.context,
438
+ value: COLOR.filename,
439
+ headerBg: COLOR.tabBg,
440
+ },
441
+ sections,
442
+ footerLines: [this.renderStatusBar(width, entry)],
443
+ });
444
+ });
445
+ }
446
+
447
+ // ── Tab bar ──────────────────────────────────────────────────────────────
448
+
449
+ private renderTabBar(width: number): Line {
450
+ const cells: Line = [];
451
+
452
+ for (let i = 0; i < this.entries.length; i++) {
453
+ const entry = this.entries[i]!;
454
+ const active = i === this.selectedFile;
455
+ const label = ` ${basename(entry.filePath)} `;
456
+ const fg = active ? COLOR.tabActive : COLOR.tabInactive;
457
+ const bg = active ? '#333333' : COLOR.tabBg;
458
+
459
+ for (const ch of label) {
460
+ if (cells.length >= width) break;
461
+ cells.push(createStyledCell(ch, { fg, bg, bold: active }));
462
+ }
463
+
464
+ if (cells.length < width) {
465
+ cells.push(createStyledCell('│', { fg: COLOR.lineNum, bg: COLOR.tabBg }));
466
+ }
467
+ }
468
+
469
+ // Fill remaining
470
+ while (cells.length < width) {
471
+ cells.push(createStyledCell(' ', { fg: '', bg: COLOR.tabBg }));
472
+ }
473
+
474
+ return cells.slice(0, width);
475
+ }
476
+
477
+ // ── Status bar ───────────────────────────────────────────────────────────
478
+
479
+ private renderStatusBar(width: number, entry: DiffEntry | null): Line {
480
+ const fileInfo = entry
481
+ ? `${entry.filePath} [${this.selectedFile + 1}/${this.entries.length}]`
482
+ : 'No file';
483
+ const scroll = entry
484
+ ? ` L${this.scrollOffset + 1}/${entry.lines.length} Tab: next file Up/Down: scroll`
485
+ : '';
486
+ const semantic = entry?.semanticSummary ? ` * ${entry.semanticSummary}` : '';
487
+ const text = ` ${fileInfo}${scroll}${semantic}`;
488
+ return renderText(width, text, COLOR.tabActive, COLOR.statusBar);
489
+ }
490
+
491
+ // ── Parsed line ──────────────────────────────────────────────────────────
492
+
493
+ private renderParsedLine(pl: ParsedLine, width: number): Line {
494
+ const left = pl.beforeNum !== null ? String(pl.beforeNum) : '';
495
+ const right = pl.afterNum !== null ? String(pl.afterNum) : '';
496
+
497
+ switch (pl.kind) {
498
+ case 'addition':
499
+ return makeLine(width, left, right, `+ ${pl.text}`, COLOR.addition, '#001a0d', COLOR.lineNumAdd, true);
500
+ case 'deletion':
501
+ return makeLine(width, left, right, `- ${pl.text}`, COLOR.deletion, '#1a0000', COLOR.lineNumDel, false);
502
+ case 'hunk':
503
+ return renderText(width, pl.text, COLOR.hunk, '#0a0a1a', false);
504
+ case 'header':
505
+ return renderText(width, pl.text, COLOR.header, '', false);
506
+ case 'context':
507
+ default:
508
+ return makeLine(width, left, right, ` ${pl.text}`, COLOR.context, '', COLOR.lineNum, false);
509
+ }
510
+ }
511
+ }
512
+
513
+ // ---------------------------------------------------------------------------
514
+ // Helpers
515
+ // ---------------------------------------------------------------------------
516
+
517
+ function basename(p: string): string {
518
+ const parts = p.replace(/\\/g, '/').split('/');
519
+ return parts[parts.length - 1] ?? p;
520
+ }