@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500

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 (380) hide show
  1. package/README.md +226 -183
  2. package/SerpentGuide.ouro/agent.json +82 -0
  3. package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
  5. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  6. package/assets/ouroboros.png +0 -0
  7. package/changelog.json +3418 -0
  8. package/dist/arc/attention-types.js +8 -0
  9. package/dist/arc/cares.js +140 -0
  10. package/dist/arc/episodes.js +117 -0
  11. package/dist/arc/intentions.js +133 -0
  12. package/dist/arc/json-store.js +117 -0
  13. package/dist/arc/obligations.js +237 -0
  14. package/dist/arc/packets.js +193 -0
  15. package/dist/arc/presence.js +185 -0
  16. package/dist/arc/task-lifecycle.js +65 -0
  17. package/dist/heart/active-work.js +989 -0
  18. package/dist/heart/agent-entry.js +58 -3
  19. package/dist/heart/attachments/image-normalize.js +194 -0
  20. package/dist/heart/attachments/materialize.js +97 -0
  21. package/dist/heart/attachments/originals.js +88 -0
  22. package/dist/heart/attachments/render.js +29 -0
  23. package/dist/heart/attachments/sources/adapter.js +2 -0
  24. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  25. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  26. package/dist/heart/attachments/sources/index.js +16 -0
  27. package/dist/heart/attachments/store.js +103 -0
  28. package/dist/heart/attachments/types.js +93 -0
  29. package/dist/heart/auth/auth-flow.js +426 -0
  30. package/dist/heart/background-operations.js +281 -0
  31. package/dist/heart/bridges/manager.js +358 -0
  32. package/dist/heart/bridges/state-machine.js +135 -0
  33. package/dist/heart/bridges/store.js +123 -0
  34. package/dist/heart/bundle-state.js +168 -0
  35. package/dist/heart/commitments.js +111 -0
  36. package/dist/heart/config-registry.js +304 -0
  37. package/dist/heart/config.js +193 -130
  38. package/dist/heart/core.js +1010 -261
  39. package/dist/heart/cross-chat-delivery.js +131 -0
  40. package/dist/heart/daemon/agent-config-check.js +490 -0
  41. package/dist/heart/daemon/agent-discovery.js +157 -0
  42. package/dist/heart/daemon/agent-service.js +360 -0
  43. package/dist/heart/daemon/agentic-repair.js +216 -0
  44. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  45. package/dist/heart/daemon/cadence.js +70 -0
  46. package/dist/heart/daemon/cli-defaults.js +640 -0
  47. package/dist/heart/daemon/cli-exec.js +7239 -0
  48. package/dist/heart/daemon/cli-help.js +493 -0
  49. package/dist/heart/daemon/cli-parse.js +1533 -0
  50. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  51. package/dist/heart/daemon/cli-render.js +561 -0
  52. package/dist/heart/daemon/cli-types.js +8 -0
  53. package/dist/heart/daemon/connect-bay.js +323 -0
  54. package/dist/heart/daemon/daemon-cli.js +30 -697
  55. package/dist/heart/daemon/daemon-entry.js +359 -8
  56. package/dist/heart/daemon/daemon-health.js +141 -0
  57. package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
  58. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  59. package/dist/heart/daemon/daemon.js +813 -19
  60. package/dist/heart/daemon/dns-workflow.js +394 -0
  61. package/dist/heart/daemon/doctor-types.js +8 -0
  62. package/dist/heart/daemon/doctor.js +615 -0
  63. package/dist/heart/daemon/health-monitor.js +92 -1
  64. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  65. package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
  66. package/dist/heart/daemon/http-health-probe.js +80 -0
  67. package/dist/heart/daemon/human-command-screens.js +234 -0
  68. package/dist/heart/daemon/human-readiness.js +114 -0
  69. package/dist/heart/daemon/inner-status.js +89 -0
  70. package/dist/heart/daemon/interactive-repair.js +394 -0
  71. package/dist/heart/daemon/launchd.js +171 -0
  72. package/dist/heart/daemon/log-tailer.js +82 -12
  73. package/dist/heart/daemon/logs-prune.js +110 -0
  74. package/dist/heart/daemon/message-router.js +17 -8
  75. package/dist/heart/daemon/os-cron-deps.js +134 -0
  76. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  77. package/dist/heart/daemon/ouro-entry.js +3 -1
  78. package/dist/heart/daemon/process-manager.js +215 -1
  79. package/dist/heart/daemon/provider-discovery.js +137 -0
  80. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  81. package/dist/heart/daemon/pulse.js +475 -0
  82. package/dist/heart/daemon/readiness-repair.js +365 -0
  83. package/dist/heart/daemon/run-hooks.js +39 -0
  84. package/dist/heart/daemon/runtime-logging.js +67 -16
  85. package/dist/heart/daemon/runtime-metadata.js +191 -0
  86. package/dist/heart/daemon/runtime-mode.js +67 -0
  87. package/dist/heart/daemon/safe-mode.js +161 -0
  88. package/dist/heart/daemon/sense-manager.js +431 -0
  89. package/dist/heart/daemon/session-id-resolver.js +131 -0
  90. package/dist/heart/daemon/skill-management-installer.js +94 -0
  91. package/dist/heart/daemon/socket-client.js +307 -0
  92. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  93. package/dist/heart/daemon/startup-tui.js +264 -0
  94. package/dist/heart/daemon/task-scheduler.js +3 -25
  95. package/dist/heart/daemon/terminal-ui.js +499 -0
  96. package/dist/heart/daemon/thoughts.js +524 -0
  97. package/dist/heart/daemon/up-progress.js +366 -0
  98. package/dist/heart/daemon/vault-items.js +56 -0
  99. package/dist/heart/delegation.js +62 -0
  100. package/dist/heart/habits/habit-migration.js +189 -0
  101. package/dist/heart/habits/habit-parser.js +140 -0
  102. package/dist/heart/habits/habit-runtime-state.js +100 -0
  103. package/dist/heart/habits/habit-scheduler.js +372 -0
  104. package/dist/heart/{daemon → hatch}/hatch-animation.js +10 -3
  105. package/dist/heart/{daemon → hatch}/hatch-flow.js +54 -136
  106. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  107. package/dist/heart/hatch/specialist-orchestrator.js +129 -0
  108. package/dist/heart/hatch/specialist-prompt.js +102 -0
  109. package/dist/heart/hatch/specialist-tools.js +306 -0
  110. package/dist/heart/identity.js +274 -61
  111. package/dist/heart/kept-notes.js +357 -0
  112. package/dist/heart/kicks.js +2 -20
  113. package/dist/heart/machine-identity.js +161 -0
  114. package/dist/heart/mail-import-discovery.js +353 -0
  115. package/dist/heart/mcp/mcp-server.js +653 -0
  116. package/dist/heart/migrate-config.js +100 -0
  117. package/dist/heart/model-capabilities.js +59 -0
  118. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  119. package/dist/heart/outlook/outlook-http-response.js +7 -0
  120. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  121. package/dist/heart/outlook/outlook-http-static.js +103 -0
  122. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  123. package/dist/heart/outlook/outlook-http.js +99 -0
  124. package/dist/heart/outlook/outlook-read.js +31 -0
  125. package/dist/heart/outlook/outlook-types.js +27 -0
  126. package/dist/heart/outlook/outlook-view.js +195 -0
  127. package/dist/heart/outlook/readers/agent-machine.js +382 -0
  128. package/dist/heart/outlook/readers/continuity-readers.js +336 -0
  129. package/dist/heart/outlook/readers/mail.js +362 -0
  130. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  131. package/dist/heart/outlook/readers/sessions.js +232 -0
  132. package/dist/heart/outlook/readers/shared.js +111 -0
  133. package/dist/heart/platform.js +81 -0
  134. package/dist/heart/progress-story.js +42 -0
  135. package/dist/heart/provider-attempt.js +134 -0
  136. package/dist/heart/provider-binding-resolver.js +255 -0
  137. package/dist/heart/provider-credentials.js +424 -0
  138. package/dist/heart/provider-failover.js +301 -0
  139. package/dist/heart/provider-models.js +81 -0
  140. package/dist/heart/provider-ping.js +262 -0
  141. package/dist/heart/provider-state.js +216 -0
  142. package/dist/heart/provider-visibility.js +188 -0
  143. package/dist/heart/providers/anthropic-token.js +131 -0
  144. package/dist/heart/providers/anthropic.js +202 -50
  145. package/dist/heart/providers/azure.js +104 -13
  146. package/dist/heart/providers/error-classification.js +63 -0
  147. package/dist/heart/providers/github-copilot.js +145 -0
  148. package/dist/heart/providers/minimax-vlm.js +189 -0
  149. package/dist/heart/providers/minimax.js +29 -7
  150. package/dist/heart/providers/openai-codex.js +63 -39
  151. package/dist/heart/runtime-capability-check.js +170 -0
  152. package/dist/heart/runtime-credentials.js +260 -0
  153. package/dist/heart/sense-truth.js +68 -0
  154. package/dist/heart/session-activity.js +190 -0
  155. package/dist/heart/session-events.js +1089 -0
  156. package/dist/heart/session-playback-cli-main.js +5 -0
  157. package/dist/heart/session-playback-cli.js +36 -0
  158. package/dist/heart/session-playback.js +231 -0
  159. package/dist/heart/session-transcript.js +167 -0
  160. package/dist/heart/start-of-turn-packet.js +345 -0
  161. package/dist/heart/streaming.js +129 -34
  162. package/dist/heart/sync.js +332 -0
  163. package/dist/heart/target-resolution.js +127 -0
  164. package/dist/heart/tempo.js +93 -0
  165. package/dist/heart/temporal-view.js +41 -0
  166. package/dist/heart/tool-activity-callbacks.js +36 -0
  167. package/dist/heart/tool-description.js +135 -0
  168. package/dist/heart/tool-friction.js +55 -0
  169. package/dist/heart/tool-loop.js +200 -0
  170. package/dist/heart/turn-context.js +372 -0
  171. package/dist/heart/turn-coordinator.js +28 -0
  172. package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
  173. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  174. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  175. package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
  176. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  177. package/dist/heart/versioning/staged-restart.js +146 -0
  178. package/dist/heart/versioning/update-checker.js +115 -0
  179. package/dist/heart/versioning/update-hooks.js +142 -0
  180. package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
  181. package/dist/mailroom/attention.js +167 -0
  182. package/dist/mailroom/autonomy.js +209 -0
  183. package/dist/mailroom/blob-store.js +606 -0
  184. package/dist/mailroom/core.js +672 -0
  185. package/dist/mailroom/entry.js +160 -0
  186. package/dist/mailroom/file-store.js +426 -0
  187. package/dist/mailroom/mbox-import.js +382 -0
  188. package/dist/mailroom/outbound.js +380 -0
  189. package/dist/mailroom/policy.js +263 -0
  190. package/dist/mailroom/reader.js +219 -0
  191. package/dist/mailroom/search-cache.js +182 -0
  192. package/dist/mailroom/search-relevance.js +319 -0
  193. package/dist/mailroom/smtp-ingress.js +176 -0
  194. package/dist/mailroom/source-state.js +176 -0
  195. package/dist/mailroom/thread.js +109 -0
  196. package/dist/mailroom/travel-extract.js +89 -0
  197. package/dist/mind/bundle-manifest.js +77 -1
  198. package/dist/mind/context.js +173 -94
  199. package/dist/mind/diary-integrity.js +60 -0
  200. package/dist/mind/{memory.js → diary.js} +84 -96
  201. package/dist/mind/embedding-provider.js +60 -0
  202. package/dist/mind/file-state.js +179 -0
  203. package/dist/mind/first-impressions.js +16 -2
  204. package/dist/mind/friends/channel.js +73 -0
  205. package/dist/mind/friends/group-context.js +144 -0
  206. package/dist/mind/friends/resolver.js +54 -2
  207. package/dist/mind/friends/store-file.js +58 -3
  208. package/dist/mind/friends/trust-explanation.js +74 -0
  209. package/dist/mind/friends/types.js +10 -2
  210. package/dist/mind/journal-index.js +161 -0
  211. package/dist/mind/note-search.js +268 -0
  212. package/dist/mind/obligation-steering.js +221 -0
  213. package/dist/mind/pending.js +76 -9
  214. package/dist/mind/phrases.js +1 -0
  215. package/dist/mind/prompt-refresh.js +3 -2
  216. package/dist/mind/prompt.js +1144 -117
  217. package/dist/mind/provenance-trust.js +26 -0
  218. package/dist/mind/scrutiny.js +173 -0
  219. package/dist/mind/token-estimate.js +8 -12
  220. package/dist/nerves/cli-logging.js +22 -3
  221. package/dist/nerves/coverage/audit-rules.js +15 -6
  222. package/dist/nerves/coverage/audit.js +28 -2
  223. package/dist/nerves/coverage/cli.js +1 -1
  224. package/dist/nerves/coverage/contract.js +5 -5
  225. package/dist/nerves/coverage/file-completeness.js +101 -5
  226. package/dist/nerves/coverage/run-artifacts.js +1 -1
  227. package/dist/nerves/event-buffer.js +111 -0
  228. package/dist/nerves/index.js +224 -4
  229. package/dist/nerves/observation.js +20 -0
  230. package/dist/nerves/redact.js +79 -0
  231. package/dist/nerves/runtime.js +5 -1
  232. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  233. package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
  234. package/dist/outlook-ui/index.html +15 -0
  235. package/dist/repertoire/ado-client.js +17 -56
  236. package/dist/repertoire/ado-semantic.js +11 -10
  237. package/dist/repertoire/api-client.js +97 -0
  238. package/dist/repertoire/bitwarden-store.js +774 -0
  239. package/dist/repertoire/bundle-templates.js +72 -0
  240. package/dist/repertoire/bw-installer.js +180 -0
  241. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  242. package/dist/repertoire/coding/context-pack.js +330 -0
  243. package/dist/repertoire/coding/feedback.js +301 -0
  244. package/dist/repertoire/coding/index.js +4 -1
  245. package/dist/repertoire/coding/manager.js +220 -13
  246. package/dist/repertoire/coding/spawner.js +58 -12
  247. package/dist/repertoire/coding/tools.js +209 -7
  248. package/dist/repertoire/commerce-errors.js +109 -0
  249. package/dist/repertoire/commerce-self-test.js +156 -0
  250. package/dist/repertoire/credential-access.js +111 -0
  251. package/dist/repertoire/data/ado-endpoints.json +188 -0
  252. package/dist/repertoire/duffel-client.js +185 -0
  253. package/dist/repertoire/github-client.js +14 -55
  254. package/dist/repertoire/graph-client.js +11 -52
  255. package/dist/repertoire/guardrails.js +396 -0
  256. package/dist/repertoire/mcp-client.js +255 -0
  257. package/dist/repertoire/mcp-manager.js +305 -0
  258. package/dist/repertoire/mcp-tools.js +63 -0
  259. package/dist/repertoire/shell-sessions.js +133 -0
  260. package/dist/repertoire/skills.js +15 -24
  261. package/dist/repertoire/stripe-client.js +131 -0
  262. package/dist/repertoire/tasks/board.js +43 -5
  263. package/dist/repertoire/tasks/fix.js +182 -0
  264. package/dist/repertoire/tasks/index.js +39 -13
  265. package/dist/repertoire/tasks/lifecycle.js +2 -2
  266. package/dist/repertoire/tasks/parser.js +3 -2
  267. package/dist/repertoire/tasks/scanner.js +194 -37
  268. package/dist/repertoire/tasks/transitions.js +16 -79
  269. package/dist/repertoire/tool-results.js +29 -0
  270. package/dist/repertoire/tools-attachments.js +317 -0
  271. package/dist/repertoire/tools-base.js +49 -707
  272. package/dist/repertoire/tools-bluebubbles.js +94 -0
  273. package/dist/repertoire/tools-bridge.js +141 -0
  274. package/dist/repertoire/tools-bundle.js +984 -0
  275. package/dist/repertoire/tools-config.js +185 -0
  276. package/dist/repertoire/tools-continuity.js +248 -0
  277. package/dist/repertoire/tools-credential.js +381 -0
  278. package/dist/repertoire/tools-files.js +342 -0
  279. package/dist/repertoire/tools-flight.js +224 -0
  280. package/dist/repertoire/tools-flow.js +105 -0
  281. package/dist/repertoire/tools-github.js +1 -7
  282. package/dist/repertoire/tools-mail.js +1377 -0
  283. package/dist/repertoire/tools-notes.js +376 -0
  284. package/dist/repertoire/tools-session.js +749 -0
  285. package/dist/repertoire/tools-shell.js +120 -0
  286. package/dist/repertoire/tools-stripe.js +180 -0
  287. package/dist/repertoire/tools-surface.js +243 -0
  288. package/dist/repertoire/tools-teams.js +64 -61
  289. package/dist/repertoire/tools-travel.js +125 -0
  290. package/dist/repertoire/tools-trip.js +356 -0
  291. package/dist/repertoire/tools-user-profile.js +144 -0
  292. package/dist/repertoire/tools-vault.js +40 -0
  293. package/dist/repertoire/tools.js +149 -98
  294. package/dist/repertoire/travel-api-client.js +360 -0
  295. package/dist/repertoire/user-profile.js +131 -0
  296. package/dist/repertoire/vault-setup.js +246 -0
  297. package/dist/repertoire/vault-unlock.js +561 -0
  298. package/dist/scripts/claude-code-hook.js +41 -0
  299. package/dist/scripts/claude-code-stop-hook.js +47 -0
  300. package/dist/senses/attention-queue.js +116 -0
  301. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  302. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  303. package/dist/senses/bluebubbles/client.js +685 -0
  304. package/dist/senses/bluebubbles/entry.js +73 -0
  305. package/dist/senses/bluebubbles/inbound-log.js +126 -0
  306. package/dist/senses/bluebubbles/index.js +1881 -0
  307. package/dist/senses/bluebubbles/media.js +389 -0
  308. package/dist/senses/bluebubbles/model.js +282 -0
  309. package/dist/senses/bluebubbles/mutation-log.js +116 -0
  310. package/dist/senses/bluebubbles/processed-log.js +111 -0
  311. package/dist/senses/bluebubbles/replay.js +129 -0
  312. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  313. package/dist/senses/bluebubbles/session-cleanup.js +72 -0
  314. package/dist/senses/cli/bracketed-paste.js +82 -0
  315. package/dist/senses/cli/image-paste.js +287 -0
  316. package/dist/senses/cli/image-ref-navigation.js +75 -0
  317. package/dist/senses/cli/ink-app.js +156 -0
  318. package/dist/senses/cli/inline-diff.js +64 -0
  319. package/dist/senses/cli/input-keys.js +174 -0
  320. package/dist/senses/cli/kill-ring.js +86 -0
  321. package/dist/senses/cli/message-list.js +51 -0
  322. package/dist/senses/cli/ouro-tui.js +605 -0
  323. package/dist/senses/cli/spinner-imperative.js +135 -0
  324. package/dist/senses/cli/spinner.js +101 -0
  325. package/dist/senses/cli/status-line.js +60 -0
  326. package/dist/senses/cli/streaming-markdown.js +526 -0
  327. package/dist/senses/cli/tool-display.js +83 -0
  328. package/dist/senses/cli/tool-render.js +85 -0
  329. package/dist/senses/cli/tui-store.js +240 -0
  330. package/dist/senses/cli/virtual-list.js +35 -0
  331. package/dist/senses/cli-entry.js +60 -8
  332. package/dist/senses/cli-layout.js +187 -0
  333. package/dist/senses/cli.js +768 -264
  334. package/dist/senses/commands.js +66 -3
  335. package/dist/senses/continuity.js +94 -0
  336. package/dist/senses/habit-turn-message.js +108 -0
  337. package/dist/senses/inner-dialog-worker.js +199 -16
  338. package/dist/senses/inner-dialog.js +640 -91
  339. package/dist/senses/mail-entry.js +66 -0
  340. package/dist/senses/mail.js +379 -0
  341. package/dist/senses/pipeline.js +665 -0
  342. package/dist/senses/proactive-content-guard.js +51 -0
  343. package/dist/senses/shared-turn.js +248 -0
  344. package/dist/senses/surface-tool.js +68 -0
  345. package/dist/senses/teams-entry.js +60 -8
  346. package/dist/senses/teams.js +844 -197
  347. package/dist/senses/trust-gate.js +207 -2
  348. package/dist/trips/core.js +138 -0
  349. package/dist/trips/store.js +146 -0
  350. package/package.json +47 -6
  351. package/skills/agent-commerce.md +106 -0
  352. package/skills/browser-navigation.md +117 -0
  353. package/skills/commerce-setup-guide.md +116 -0
  354. package/skills/commerce-setup.md +84 -0
  355. package/skills/configure-dev-tools.md +101 -0
  356. package/skills/travel-planning.md +138 -0
  357. package/AdoptionSpecialist.ouro/agent.json +0 -20
  358. package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
  359. package/dist/heart/daemon/specialist-orchestrator.js +0 -160
  360. package/dist/heart/daemon/specialist-prompt.js +0 -40
  361. package/dist/heart/daemon/specialist-session.js +0 -142
  362. package/dist/heart/daemon/specialist-tools.js +0 -128
  363. package/dist/heart/daemon/subagent-installer.js +0 -125
  364. package/dist/inner-worker-entry.js +0 -4
  365. package/dist/mind/associative-recall.js +0 -197
  366. package/subagents/README.md +0 -73
  367. package/subagents/work-doer.md +0 -233
  368. package/subagents/work-merger.md +0 -624
  369. package/subagents/work-planner.md +0 -373
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
@@ -33,11 +33,20 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.MarkdownStreamer = exports.InputController = exports.Spinner = void 0;
36
+ exports.InputQueue = exports.MarkdownStreamer = exports.InputController = exports.Spinner = exports.StreamingWordWrapper = exports.wrapCliText = exports.formatEchoedInputSummary = void 0;
37
+ exports.formatPendingPrefix = formatPendingPrefix;
38
+ exports.getCliContinuityIngressTexts = getCliContinuityIngressTexts;
39
+ exports.formatTimeAgo = formatTimeAgo;
40
+ exports.writeCliAsyncAssistantMessage = writeCliAsyncAssistantMessage;
41
+ exports.pauseActiveSpinner = pauseActiveSpinner;
42
+ exports.resumeActiveSpinner = resumeActiveSpinner;
43
+ exports.setActiveSpinner = setActiveSpinner;
37
44
  exports.handleSigint = handleSigint;
38
45
  exports.addHistory = addHistory;
39
46
  exports.renderMarkdown = renderMarkdown;
40
47
  exports.createCliCallbacks = createCliCallbacks;
48
+ exports.createDebouncedLines = createDebouncedLines;
49
+ exports.runCliSession = runCliSession;
41
50
  exports.main = main;
42
51
  const readline = __importStar(require("readline"));
43
52
  const os = __importStar(require("os"));
@@ -47,11 +56,12 @@ const prompt_1 = require("../mind/prompt");
47
56
  const phrases_1 = require("../mind/phrases");
48
57
  const format_1 = require("../mind/format");
49
58
  const config_1 = require("../heart/config");
59
+ const session_events_1 = require("../heart/session-events");
50
60
  const context_1 = require("../mind/context");
51
61
  const pending_1 = require("../mind/pending");
52
- const prompt_refresh_1 = require("../mind/prompt-refresh");
53
62
  const commands_1 = require("./commands");
54
63
  const identity_1 = require("../heart/identity");
64
+ const mcp_manager_1 = require("../repertoire/mcp-manager");
55
65
  const nerves_1 = require("../nerves");
56
66
  const store_file_1 = require("../mind/friends/store-file");
57
67
  const resolver_1 = require("../mind/friends/resolver");
@@ -59,59 +69,78 @@ const tokens_1 = require("../mind/friends/tokens");
59
69
  const cli_logging_1 = require("../nerves/cli-logging");
60
70
  const runtime_1 = require("../nerves/runtime");
61
71
  const trust_gate_1 = require("./trust-gate");
72
+ const pipeline_1 = require("./pipeline");
73
+ const channel_1 = require("../mind/friends/channel");
62
74
  const session_lock_1 = require("./session-lock");
63
- // spinner that only touches stderr, cleans up after itself
64
- // exported for direct testability (stop-without-start branch)
65
- class Spinner {
66
- frames = ["\u280B", "\u2819", "\u2839", "\u2838", "\u283C", "\u2834", "\u2826", "\u2827", "\u2807", "\u280F"];
67
- i = 0;
68
- iv = null;
69
- piv = null;
70
- msg = "";
71
- phrases = null;
72
- lastPhrase = "";
73
- constructor(m = "working", phrases) {
74
- this.msg = m;
75
- if (phrases && phrases.length > 0)
76
- this.phrases = phrases;
77
- }
78
- start() {
79
- process.stderr.write("\r\x1b[K");
80
- this.spin();
81
- this.iv = setInterval(() => this.spin(), 80);
82
- if (this.phrases) {
83
- this.piv = setInterval(() => this.rotatePhrase(), 1500);
84
- }
85
- }
86
- spin() {
87
- process.stderr.write(`\r${this.frames[this.i]} ${this.msg}... `);
88
- this.i = (this.i + 1) % this.frames.length;
89
- }
90
- rotatePhrase() {
91
- const next = (0, phrases_1.pickPhrase)(this.phrases, this.lastPhrase);
92
- this.lastPhrase = next;
93
- this.msg = next;
94
- }
95
- stop(ok) {
96
- if (this.iv) {
97
- clearInterval(this.iv);
98
- this.iv = null;
99
- }
100
- if (this.piv) {
101
- clearInterval(this.piv);
102
- this.piv = null;
103
- }
104
- process.stderr.write("\r\x1b[K");
105
- /* v8 ignore next -- ok parameter currently unused by callers @preserve */
106
- if (ok)
107
- process.stderr.write(`\x1b[32m\u2713\x1b[0m ${ok}\n`);
108
- }
109
- fail(msg) {
110
- this.stop();
111
- process.stderr.write(`\x1b[31m\u2717\x1b[0m ${msg}\n`);
75
+ const update_hooks_1 = require("../heart/versioning/update-hooks");
76
+ const bundle_meta_1 = require("../heart/daemon/hooks/bundle-meta");
77
+ const agent_config_v2_1 = require("../heart/daemon/hooks/agent-config-v2");
78
+ const bundle_manifest_1 = require("../mind/bundle-manifest");
79
+ const cli_layout_1 = require("./cli-layout");
80
+ const image_paste_1 = require("./cli/image-paste");
81
+ const spinner_imperative_1 = require("./cli/spinner-imperative");
82
+ const tool_display_1 = require("./cli/tool-display");
83
+ var cli_layout_2 = require("./cli-layout");
84
+ Object.defineProperty(exports, "formatEchoedInputSummary", { enumerable: true, get: function () { return cli_layout_2.formatEchoedInputSummary; } });
85
+ Object.defineProperty(exports, "wrapCliText", { enumerable: true, get: function () { return cli_layout_2.wrapCliText; } });
86
+ Object.defineProperty(exports, "StreamingWordWrapper", { enumerable: true, get: function () { return cli_layout_2.StreamingWordWrapper; } });
87
+ /**
88
+ * Format pending messages as content-prefix strings for injection into
89
+ * the next user message. Self-messages (from === agentName) become
90
+ * `[inner thought: {content}]`, inter-agent messages become
91
+ * `[message from {name}: {content}]`.
92
+ */
93
+ function formatPendingPrefix(messages, agentName) {
94
+ return messages
95
+ .map((msg) => msg.from === agentName
96
+ ? `[inner thought: ${msg.content}]`
97
+ : `[message from ${msg.from}: ${msg.content}]`)
98
+ .join("\n");
99
+ }
100
+ function getCliContinuityIngressTexts(input) {
101
+ const trimmed = input.trim();
102
+ return trimmed ? [trimmed] : [];
103
+ }
104
+ /* v8 ignore start -- cosmetic time formatting @preserve */
105
+ function formatTimeAgo(date) {
106
+ const seconds = Math.floor((Date.now() - date.getTime()) / 1000);
107
+ if (seconds < 60)
108
+ return "just now";
109
+ const minutes = Math.floor(seconds / 60);
110
+ if (minutes < 60)
111
+ return `${minutes}m ago`;
112
+ const hours = Math.floor(minutes / 60);
113
+ if (hours < 24)
114
+ return `${hours}h ago`;
115
+ const days = Math.floor(hours / 24);
116
+ return `${days}d ago`;
117
+ }
118
+ const CLI_PROMPT = "\x1b[36m) \x1b[0m";
119
+ function writeCliAsyncAssistantMessage(rl, message, stdout = process.stdout) {
120
+ const rlInt = rl;
121
+ const currentLine = rlInt.line ?? "";
122
+ const currentCursor = rlInt.cursor ?? currentLine.length;
123
+ stdout.write("\r\x1b[K");
124
+ stdout.write(`${renderMarkdown(message)}\n`);
125
+ stdout.write(CLI_PROMPT);
126
+ if (!currentLine)
127
+ return;
128
+ stdout.write(currentLine);
129
+ if (currentCursor < currentLine.length) {
130
+ readline.cursorTo(process.stdout, 2 + currentCursor);
112
131
  }
113
132
  }
114
- exports.Spinner = Spinner;
133
+ // Module-level active spinner for log coordination.
134
+ // The terminal log sink calls these to avoid interleaving with spinner output.
135
+ let _activeSpinner = null;
136
+ /* v8 ignore start -- spinner coordination: exercised at runtime, not unit-testable without real terminal @preserve */
137
+ function pauseActiveSpinner() { _activeSpinner?.pause(); }
138
+ function resumeActiveSpinner() { _activeSpinner?.resume(); }
139
+ /* v8 ignore stop */
140
+ function setActiveSpinner(s) { _activeSpinner = s; }
141
+ // Re-export ImperativeSpinner as Spinner for backward compatibility
142
+ var spinner_imperative_2 = require("./cli/spinner-imperative");
143
+ Object.defineProperty(exports, "Spinner", { enumerable: true, get: function () { return spinner_imperative_2.ImperativeSpinner; } });
115
144
  // Input controller: pauses readline during model/tool execution.
116
145
  // Does NOT touch raw mode — readline with terminal:true manages raw mode
117
146
  // internally. Touching it causes ^C to be echoed by the terminal driver.
@@ -279,41 +308,54 @@ function createCliCallbacks() {
279
308
  meta: {},
280
309
  });
281
310
  let currentSpinner = null;
282
- let hadReasoning = false;
311
+ function setSpinner(s) { currentSpinner = s; setActiveSpinner(s); }
283
312
  let hadToolRun = false;
284
313
  let textDirty = false; // true when text/reasoning was written without a trailing newline
285
314
  const streamer = new MarkdownStreamer();
315
+ const wrapper = new cli_layout_1.StreamingWordWrapper();
286
316
  return {
287
317
  onModelStart: () => {
288
318
  currentSpinner?.stop();
289
- currentSpinner = null;
290
- hadReasoning = false;
319
+ setSpinner(null);
291
320
  textDirty = false;
292
321
  streamer.reset();
322
+ wrapper.reset();
293
323
  const phrases = (0, phrases_1.getPhrases)();
294
324
  const pool = hadToolRun ? phrases.followup : phrases.thinking;
295
325
  const first = (0, phrases_1.pickPhrase)(pool);
296
- currentSpinner = new Spinner(first, pool);
326
+ setSpinner(new spinner_imperative_1.ImperativeSpinner(first, pool));
297
327
  currentSpinner.start();
298
328
  },
299
329
  onModelStreamStart: () => {
300
- currentSpinner?.stop();
301
- currentSpinner = null;
330
+ // No-op: content callbacks (onTextChunk, onReasoningChunk) handle
331
+ // stopping the spinner. onModelStreamStart fires too early and
332
+ // doesn't fire at all for settle tool streaming.
333
+ },
334
+ onClearText: () => {
335
+ streamer.reset();
336
+ wrapper.reset();
302
337
  },
303
338
  onTextChunk: (text) => {
304
- if (hadReasoning) {
305
- process.stdout.write("\n\n");
306
- hadReasoning = false;
339
+ // Stop spinner if still running — settle streaming and Anthropic
340
+ // tool-only responses bypass onModelStreamStart, so the spinner would
341
+ // otherwise keep running (and its \r writes overwrite response text).
342
+ if (currentSpinner) {
343
+ currentSpinner.stop();
344
+ setSpinner(null);
307
345
  }
308
346
  const rendered = streamer.push(text);
309
- if (rendered)
310
- process.stdout.write(rendered);
347
+ /* v8 ignore start -- wrapper integration: tested via cli.test.ts onTextChunk tests @preserve */
348
+ if (rendered) {
349
+ const wrapped = wrapper.push(rendered);
350
+ if (wrapped)
351
+ process.stdout.write(wrapped);
352
+ }
353
+ /* v8 ignore stop */
311
354
  textDirty = text.length > 0 && !text.endsWith("\n");
312
355
  },
313
- onReasoningChunk: (text) => {
314
- hadReasoning = true;
315
- process.stdout.write(`\x1b[2m${text}\x1b[0m`);
316
- textDirty = text.length > 0 && !text.endsWith("\n");
356
+ onReasoningChunk: (_text) => {
357
+ // Keep reasoning private in the CLI surface. The spinner continues to
358
+ // represent active thinking until actual tool or answer output arrives.
317
359
  },
318
360
  onToolStart: (_name, _args) => {
319
361
  // Stop the model-start spinner: when the model returns only tool calls
@@ -328,31 +370,29 @@ function createCliCallbacks() {
328
370
  }
329
371
  const toolPhrases = (0, phrases_1.getPhrases)().tool;
330
372
  const first = (0, phrases_1.pickPhrase)(toolPhrases);
331
- currentSpinner = new Spinner(first, toolPhrases);
373
+ setSpinner(new spinner_imperative_1.ImperativeSpinner(first, toolPhrases));
332
374
  currentSpinner.start();
333
375
  hadToolRun = true;
334
376
  },
335
377
  onToolEnd: (name, argSummary, success) => {
336
378
  currentSpinner?.stop();
337
- currentSpinner = null;
338
- const msg = (0, format_1.formatToolResult)(name, argSummary, success);
339
- const color = success ? "\x1b[32m" : "\x1b[31m";
340
- process.stderr.write(`${color}${msg}\x1b[0m\n`);
379
+ setSpinner(null);
380
+ (0, tool_display_1.writeToolEnd)(name, argSummary, success);
341
381
  },
342
382
  onError: (error, severity) => {
343
383
  if (severity === "transient") {
344
384
  currentSpinner?.fail(error.message);
345
- currentSpinner = null;
385
+ setSpinner(null);
346
386
  }
347
387
  else {
348
388
  currentSpinner?.stop();
349
- currentSpinner = null;
389
+ setSpinner(null);
350
390
  process.stderr.write(`\x1b[31m${(0, format_1.formatError)(error)}\x1b[0m\n`);
351
391
  }
352
392
  },
353
393
  onKick: () => {
354
394
  currentSpinner?.stop();
355
- currentSpinner = null;
395
+ setSpinner(null);
356
396
  if (textDirty) {
357
397
  process.stdout.write("\n");
358
398
  textDirty = false;
@@ -361,27 +401,547 @@ function createCliCallbacks() {
361
401
  },
362
402
  flushMarkdown: () => {
363
403
  currentSpinner?.stop();
364
- currentSpinner = null;
404
+ setSpinner(null);
405
+ /* v8 ignore start -- wrapper flush: tested via cli.test.ts flushMarkdown tests @preserve */
365
406
  const remaining = streamer.flush();
366
- if (remaining)
367
- process.stdout.write(remaining);
407
+ if (remaining) {
408
+ const wrapped = wrapper.push(remaining);
409
+ if (wrapped)
410
+ process.stdout.write(wrapped);
411
+ }
412
+ const tail = wrapper.flush();
413
+ if (tail)
414
+ process.stdout.write(tail);
415
+ /* v8 ignore stop */
416
+ },
417
+ };
418
+ }
419
+ // Debounced line iterator: collects rapid-fire lines (paste) into a single input.
420
+ // When the debounce timeout wins the race, the pending iter.next() is saved
421
+ // and reused in the next iteration to prevent it from silently consuming input.
422
+ async function* createDebouncedLines(source, debounceMs) {
423
+ if (debounceMs <= 0) {
424
+ yield* source;
425
+ return;
426
+ }
427
+ const iter = source[Symbol.asyncIterator]();
428
+ let pending = null;
429
+ while (true) {
430
+ const first = pending ? await pending : await iter.next();
431
+ pending = null;
432
+ if (first.done)
433
+ break;
434
+ const lines = [first.value];
435
+ let more = true;
436
+ while (more) {
437
+ const nextPromise = iter.next();
438
+ const raced = await Promise.race([
439
+ nextPromise.then((r) => ({ kind: "line", result: r })),
440
+ new Promise((r) => setTimeout(() => r({ kind: "timeout" }), debounceMs)),
441
+ ]);
442
+ if (raced.kind === "timeout") {
443
+ pending = nextPromise;
444
+ more = false;
445
+ }
446
+ else if (raced.result.done) {
447
+ more = false;
448
+ }
449
+ else {
450
+ lines.push(raced.result.value);
451
+ }
452
+ }
453
+ yield lines.join("\n");
454
+ }
455
+ }
456
+ /**
457
+ * Async queue that bridges push-based Ink input to pull-based async iteration.
458
+ * Input from Ink's onSubmit callback is pushed; the business logic loop awaits via for-await.
459
+ */
460
+ class InputQueue {
461
+ queue = [];
462
+ resolve = null;
463
+ done = false;
464
+ push(input) {
465
+ if (this.done)
466
+ return;
467
+ if (this.resolve) {
468
+ const r = this.resolve;
469
+ this.resolve = null;
470
+ r({ value: input, done: false });
471
+ }
472
+ else {
473
+ this.queue.push(input);
474
+ }
475
+ }
476
+ close() {
477
+ this.done = true;
478
+ if (this.resolve) {
479
+ const r = this.resolve;
480
+ this.resolve = null;
481
+ r({ value: undefined, done: true });
482
+ }
483
+ }
484
+ /** Drain all buffered items, leaving any pending async awaiter untouched. */
485
+ drainAll() {
486
+ const items = [...this.queue];
487
+ this.queue = [];
488
+ return items;
489
+ }
490
+ [Symbol.asyncIterator]() {
491
+ return {
492
+ next: () => {
493
+ if (this.queue.length > 0) {
494
+ return Promise.resolve({ value: this.queue.shift(), done: false });
495
+ }
496
+ if (this.done) {
497
+ return Promise.resolve({ value: undefined, done: true });
498
+ }
499
+ return new Promise((resolve) => {
500
+ this.resolve = resolve;
501
+ });
502
+ },
503
+ };
504
+ }
505
+ }
506
+ exports.InputQueue = InputQueue;
507
+ async function runCliSession(options) {
508
+ /* v8 ignore start -- integration: runCliSession is interactive, tested via E2E @preserve */
509
+ const registry = (0, commands_1.createCommandRegistry)();
510
+ if (!options.disableCommands) {
511
+ (0, commands_1.registerDefaultCommands)(registry);
512
+ }
513
+ const messages = options.messages
514
+ ?? [{ role: "system", content: (0, prompt_1.flattenSystemPrompt)(await (0, prompt_1.buildSystem)("cli")) }];
515
+ // ─── Rendering: TUI (Ink + Static) for TTY, imperative for tests/pipes ───
516
+ const useTui = !options._testInputSource && process.stdin.isTTY === true;
517
+ let currentAbort = null;
518
+ let closed = false;
519
+ // eslint-disable-next-line prefer-const -- set by onImageMap callback during input
520
+ let pendingImages = null;
521
+ let cliCallbacks;
522
+ let tuiStore = null;
523
+ let inkRef = null;
524
+ const inputQueue = useTui ? new InputQueue() : null;
525
+ let rl = null;
526
+ if (useTui) {
527
+ try {
528
+ const [ink, React, tuiMod, storeMod] = await Promise.all([
529
+ Promise.resolve().then(() => __importStar(require("ink"))),
530
+ Promise.resolve().then(() => __importStar(require("react"))),
531
+ Promise.resolve().then(() => __importStar(require("./cli/ouro-tui"))),
532
+ Promise.resolve().then(() => __importStar(require("./cli/tui-store"))),
533
+ ]);
534
+ const { OuroTui } = tuiMod;
535
+ const { TuiStore, createTuiCallbacks } = storeMod;
536
+ tuiStore = new TuiStore();
537
+ cliCallbacks = createTuiCallbacks(tuiStore);
538
+ // Seed input history from previous session (for up/down arrows) — NOT display
539
+ const prevUserMsgs = messages
540
+ .filter((msg) => msg.role === "user" && typeof msg.content === "string")
541
+ .map(msg => msg.content);
542
+ tuiStore.seedHistory(prevUserMsgs);
543
+ // Show session resume context: last 2 exchanges as normal messages
544
+ if (messages.length > 1) {
545
+ const userAssistantMsgs = messages.filter((m) => (m.role === "user" || m.role === "assistant") && typeof m.content === "string" && m.content.trim().length > 0);
546
+ // Extract last 2 exchanges (up to 4 messages)
547
+ const lastExchanges = [];
548
+ for (let i = userAssistantMsgs.length - 1; i >= 0 && lastExchanges.length < 4; i--) {
549
+ lastExchanges.unshift({ role: userAssistantMsgs[i].role, content: userAssistantMsgs[i].content });
550
+ }
551
+ tuiStore.addResumeMessages(lastExchanges);
552
+ }
553
+ // Compute resumeInfo for header banner
554
+ const resumeInfo = messages.length > 1
555
+ ? {
556
+ messageCount: messages.filter(m => m.role === "user" || m.role === "assistant").length,
557
+ timeAgo: options.lastActivityAt ? formatTimeAgo(new Date(options.lastActivityAt)) : "unknown",
558
+ }
559
+ : undefined;
560
+ // Ctrl-C state machine (Claude Code behavior):
561
+ // During generation: abort current request
562
+ // Idle with text: clear input
563
+ // Idle empty (first): warn
564
+ // Idle empty (second): exit
565
+ let ctrlCWarned = false;
566
+ let ctrlCTimer = null;
567
+ const handleCtrlC = (hasInput) => {
568
+ if (currentAbort) {
569
+ currentAbort.abort();
570
+ ctrlCWarned = false;
571
+ return "abort";
572
+ }
573
+ if (hasInput) {
574
+ ctrlCWarned = false;
575
+ return "clear";
576
+ }
577
+ if (ctrlCWarned) {
578
+ ctrlCWarned = false;
579
+ if (ctrlCTimer) {
580
+ clearTimeout(ctrlCTimer);
581
+ ctrlCTimer = null;
582
+ }
583
+ closed = true;
584
+ inputQueue.close();
585
+ return "exit";
586
+ }
587
+ ctrlCWarned = true;
588
+ // Reset after 2 seconds — must press twice within window
589
+ ctrlCTimer = setTimeout(() => { ctrlCWarned = false; }, 2000);
590
+ return "warn";
591
+ };
592
+ // TUI root: subscribes to store, passes props to OuroTui
593
+ // Elapsed timer is local React state (no store.notify overhead)
594
+ const storeRef = tuiStore;
595
+ function TuiRoot() {
596
+ const [, forceUpdate] = React.useState(0);
597
+ const [elapsed, setElapsed] = React.useState(0);
598
+ React.useEffect(() => storeRef.subscribe(() => {
599
+ forceUpdate((n) => n + 1);
600
+ // Reset ctrlC warning on any state change (new turn, etc.)
601
+ }), []);
602
+ React.useEffect(() => {
603
+ const iv = setInterval(() => setElapsed(storeRef.getElapsed()), 1000);
604
+ return () => clearInterval(iv);
605
+ }, []);
606
+ return React.createElement(OuroTui, {
607
+ agentName: options.agentName,
608
+ model: (0, identity_1.loadAgentConfig)().humanFacing?.model ?? "",
609
+ completedMessages: storeRef.completedMessages,
610
+ inputHistory: storeRef.inputHistory,
611
+ queuedInputs: storeRef.queuedInputs,
612
+ live: storeRef.live,
613
+ elapsedSeconds: elapsed,
614
+ contextPercent: 0,
615
+ onSubmit: (text) => { ctrlCWarned = false; inputQueue.push(text); storeRef.enqueueInput(text); },
616
+ onCtrlC: handleCtrlC,
617
+ onPopQueue: () => { const items = storeRef.popAllQueuedForEditing(); inputQueue.drainAll(); return items; },
618
+ headerShown: storeRef.headerShown,
619
+ cwd: process.cwd().replace(process.env.HOME ?? "", "~"),
620
+ resumeInfo,
621
+ onImageMap: (images) => { pendingImages = images; },
622
+ onHistoryAdd: (text) => { storeRef.addToHistoryOnly(text); },
623
+ });
624
+ }
625
+ inkRef = ink.render(React.createElement(TuiRoot), { exitOnCtrlC: false, patchConsole: false });
626
+ }
627
+ catch (err) {
628
+ // Ink failed to load (CJS compat, missing deps, etc.) — fall through to imperative
629
+ (0, runtime_1.emitNervesEvent)({
630
+ component: "senses",
631
+ event: "senses.tui_fallback",
632
+ message: `TUI failed to load, falling back to imperative: ${err instanceof Error ? err.message : String(err)}`,
633
+ meta: {},
634
+ });
635
+ }
636
+ }
637
+ // Fallback to imperative callbacks if TUI didn't initialize
638
+ if (!tuiStore) {
639
+ cliCallbacks = createCliCallbacks();
640
+ if (options.banner !== false) {
641
+ const bannerText = typeof options.banner === "string"
642
+ ? options.banner
643
+ : `${options.agentName} (type /commands for help)`;
644
+ // eslint-disable-next-line no-console -- terminal UX: startup banner
645
+ console.log(`\n${bannerText}\n`);
646
+ }
647
+ }
648
+ // Display helpers: route to TUI store or imperative stderr/stdout
649
+ const display = {
650
+ error: (msg) => {
651
+ if (tuiStore)
652
+ tuiStore.setError(msg);
653
+ else
654
+ process.stderr.write(`\x1b[31m${msg}\x1b[0m\n`);
655
+ },
656
+ warn: (msg) => {
657
+ if (tuiStore)
658
+ tuiStore.setError(msg); // TUI shows warnings as errors (amber color handled in component)
659
+ else
660
+ process.stderr.write(`\x1b[33m${msg}\x1b[0m\n`);
661
+ },
662
+ text: (msg) => {
663
+ if (tuiStore)
664
+ tuiStore.appendText(msg);
665
+ else
666
+ process.stdout.write(`${msg}\n`);
667
+ },
668
+ suppressInput: () => {
669
+ if (tuiStore)
670
+ tuiStore.suppressInput();
671
+ },
672
+ restoreInput: () => {
673
+ if (tuiStore)
674
+ tuiStore.restoreInput();
368
675
  },
369
676
  };
677
+ const effectiveToolContext = {
678
+ signin: options.toolContext?.signin ?? (async () => undefined),
679
+ ...options.toolContext,
680
+ codingFeedback: {
681
+ send: async (message) => {
682
+ const assistantMessage = {
683
+ role: "assistant",
684
+ content: message,
685
+ };
686
+ messages.push(assistantMessage);
687
+ await options.onAsyncAssistantMessage?.(messages, assistantMessage);
688
+ display.text(message);
689
+ await options.toolContext?.codingFeedback?.send(message);
690
+ },
691
+ },
692
+ };
693
+ // exitOnToolCall machinery: wrap execTool to detect target tool
694
+ let exitToolResult;
695
+ let exitToolFired = false;
696
+ const resolvedExecTool = options.execTool;
697
+ const wrappedExecTool = options.exitOnToolCall && resolvedExecTool
698
+ ? async (name, args, ctx) => {
699
+ const result = await resolvedExecTool(name, args, ctx);
700
+ if (name === options.exitOnToolCall) {
701
+ exitToolResult = result;
702
+ exitToolFired = true;
703
+ currentAbort?.abort();
704
+ }
705
+ return result;
706
+ }
707
+ : resolvedExecTool;
708
+ // Resolve toolChoiceRequired: use explicit option if set, else fall back to toggle
709
+ const getEffectiveToolChoiceRequired = () => options.toolChoiceRequired !== undefined ? options.toolChoiceRequired : (0, commands_1.getToolChoiceRequired)();
710
+ (0, runtime_1.emitNervesEvent)({
711
+ component: "senses",
712
+ event: "senses.cli_session_start",
713
+ message: "runCliSession started",
714
+ meta: { agentName: options.agentName, hasExitOnToolCall: !!options.exitOnToolCall },
715
+ });
716
+ let exitReason = "user_quit";
717
+ // Auto-first-turn: process the last user message immediately so the agent
718
+ // speaks first (e.g. specialist greeting). Only triggers when explicitly opted in.
719
+ if (options.autoFirstTurn && messages.length > 0 && messages[messages.length - 1]?.role === "user") {
720
+ currentAbort = new AbortController();
721
+ const traceId = (0, nerves_1.createTraceId)();
722
+ display.suppressInput();
723
+ let result;
724
+ try {
725
+ result = await (0, core_1.runAgent)(messages, cliCallbacks, options.skipSystemPromptRefresh ? undefined : "cli", currentAbort.signal, {
726
+ toolChoiceRequired: getEffectiveToolChoiceRequired(),
727
+ traceId,
728
+ tools: options.tools,
729
+ execTool: wrappedExecTool,
730
+ toolContext: effectiveToolContext,
731
+ });
732
+ }
733
+ catch (err) {
734
+ if (!(err instanceof DOMException && err.name === "AbortError")) {
735
+ display.error(err instanceof Error ? err.message : String(err));
736
+ }
737
+ }
738
+ cliCallbacks.flushMarkdown();
739
+ display.restoreInput();
740
+ currentAbort = null;
741
+ if (exitToolFired) {
742
+ exitReason = "tool_exit";
743
+ closed = true;
744
+ }
745
+ else {
746
+ const lastMsg = messages[messages.length - 1];
747
+ if (lastMsg?.role === "assistant" && !(typeof lastMsg.content === "string" ? lastMsg.content : "").trim()) {
748
+ display.warn("(empty response)");
749
+ }
750
+ if (options.onTurnEnd) {
751
+ await options.onTurnEnd(messages, result ?? { usage: undefined });
752
+ }
753
+ }
754
+ }
755
+ try {
756
+ // Input source: TUI queue for Ink, test source for tests, readline fallback
757
+ let inputSource;
758
+ let inputCtrl = null;
759
+ if (tuiStore && inputQueue) {
760
+ // TUI path: input comes from Ink's onSubmit → InputQueue
761
+ inputSource = inputQueue;
762
+ }
763
+ else if (options._testInputSource) {
764
+ inputSource = options._testInputSource;
765
+ }
766
+ else {
767
+ // Imperative fallback: readline
768
+ const isTTY = process.stdin.isTTY === true;
769
+ rl = readline.createInterface({ input: process.stdin, output: process.stdout, terminal: isTTY });
770
+ inputCtrl = new InputController(rl);
771
+ inputSource = rl;
772
+ process.stdout.write(CLI_PROMPT);
773
+ }
774
+ for await (const input of inputSource) {
775
+ if (closed)
776
+ break;
777
+ if (!input.trim())
778
+ continue;
779
+ // Remove from TUI queue display as the agent picks up this message
780
+ tuiStore?.dequeueInput(input);
781
+ // Optional input gate (e.g. trust gate in main)
782
+ if (options.onInput) {
783
+ const gate = options.onInput(input);
784
+ if (!gate.allowed) {
785
+ if (gate.reply) {
786
+ display.text(gate.reply);
787
+ }
788
+ if (closed)
789
+ break;
790
+ continue;
791
+ }
792
+ }
793
+ // Check for slash commands (legacy path only — pipeline handles commands for runTurn path)
794
+ if (!options.runTurn) {
795
+ const parsed = (0, commands_1.parseSlashCommand)(input);
796
+ if (parsed) {
797
+ const dispatchResult = registry.dispatch(parsed.command, { channel: "cli" });
798
+ if (dispatchResult.handled && dispatchResult.result) {
799
+ if (dispatchResult.result.action === "exit") {
800
+ break;
801
+ }
802
+ else if (dispatchResult.result.action === "response") {
803
+ display.text(dispatchResult.result.message || "");
804
+ continue;
805
+ }
806
+ }
807
+ }
808
+ }
809
+ // Track user message in TUI for display
810
+ if (tuiStore)
811
+ tuiStore.addUserMessage(input);
812
+ currentAbort = new AbortController();
813
+ if (tuiStore)
814
+ tuiStore.suppressInput();
815
+ inputCtrl?.suppress(() => { currentAbort?.abort(); });
816
+ // Resolve pending image content before the turn executes
817
+ let contentParts = null;
818
+ const currentImages = pendingImages;
819
+ if (currentImages && currentImages.size > 0) {
820
+ contentParts = await (0, image_paste_1.resolveImageContent)(input, currentImages);
821
+ pendingImages = null;
822
+ }
823
+ let result;
824
+ try {
825
+ if (options.runTurn) {
826
+ // Pipeline-based turn: the runTurn callback handles user message assembly,
827
+ // pending drain, trust gate, runAgent, postTurn, and token accumulation.
828
+ result = await options.runTurn(messages, input, cliCallbacks, currentAbort.signal, effectiveToolContext, contentParts ?? undefined);
829
+ // Handle pipeline-intercepted commands with loop-control side effects
830
+ if (result?.turnOutcome === "command") {
831
+ if (result.commandAction === "exit") {
832
+ break;
833
+ }
834
+ // For "response" commands: the pipeline already emitted the response via onTextChunk
835
+ cliCallbacks.flushMarkdown();
836
+ continue;
837
+ }
838
+ }
839
+ else {
840
+ // Legacy path: inline runAgent (used by serpent guide and tests)
841
+ const prefix = options.getContentPrefix?.();
842
+ const userContent = contentParts
843
+ ? contentParts
844
+ : (prefix ? `${prefix}\n\n${input}` : input);
845
+ const userMsg = { role: "user", content: userContent };
846
+ (0, session_events_1.stampIngressTime)(userMsg);
847
+ messages.push(userMsg);
848
+ const traceId = (0, nerves_1.createTraceId)();
849
+ result = await (0, core_1.runAgent)(messages, cliCallbacks, options.skipSystemPromptRefresh ? undefined : "cli", currentAbort.signal, {
850
+ toolChoiceRequired: getEffectiveToolChoiceRequired(),
851
+ traceId,
852
+ tools: options.tools,
853
+ execTool: wrappedExecTool,
854
+ toolContext: effectiveToolContext,
855
+ });
856
+ }
857
+ }
858
+ catch (err) {
859
+ if (!(err instanceof DOMException && err.name === "AbortError")) {
860
+ display.error(err instanceof Error ? err.message : String(err));
861
+ }
862
+ }
863
+ cliCallbacks.flushMarkdown();
864
+ if (!tuiStore)
865
+ process.stdout.write("\n"); // ensure response ends with newline before prompt (imperative only)
866
+ if (tuiStore)
867
+ tuiStore.restoreInput();
868
+ inputCtrl?.restore();
869
+ currentAbort = null;
870
+ // Check if exit tool was fired during this turn
871
+ if (exitToolFired) {
872
+ exitReason = "tool_exit";
873
+ break;
874
+ }
875
+ // Safety net: never silently swallow an empty response
876
+ const lastMsg = messages[messages.length - 1];
877
+ if (lastMsg?.role === "assistant" && !(typeof lastMsg.content === "string" ? lastMsg.content : "").trim()) {
878
+ display.warn("(empty response)");
879
+ }
880
+ // Post-turn hook (session persistence, pending drain, prompt refresh, etc.)
881
+ if (options.onTurnEnd) {
882
+ await options.onTurnEnd(messages, result ?? { usage: undefined });
883
+ }
884
+ if (closed)
885
+ break;
886
+ }
887
+ }
888
+ finally {
889
+ rl?.close();
890
+ if (inkRef) {
891
+ // Suppress React "state update on unmounted component" warnings during cleanup.
892
+ // Ink's useInput hook fires after unmount — this is harmless but noisy.
893
+ // This also covers the SerpentGuide exitOnToolCall path, which aborts the
894
+ // current request and breaks out of the loop into this finally block.
895
+ // eslint-disable-next-line no-console -- intentional console.warn/error override for cleanup
896
+ const origWarn = console.warn;
897
+ const origError = console.error; // eslint-disable-line no-console
898
+ // eslint-disable-next-line no-console -- suppress React unmount warnings
899
+ console.warn = (...args) => { if (typeof args[0] === "string" && args[0].includes("Can't perform a React state update"))
900
+ return; origWarn.apply(console, args); };
901
+ // eslint-disable-next-line no-console -- suppress React unmount warnings
902
+ console.error = (...args) => { if (typeof args[0] === "string" && args[0].includes("Can't perform a React state update"))
903
+ return; origError.apply(console, args); };
904
+ inkRef.unmount();
905
+ setTimeout(() => { console.warn = origWarn; console.error = origError; }, 100); // eslint-disable-line no-console
906
+ }
907
+ if (options.banner !== false) {
908
+ // eslint-disable-next-line no-console -- terminal UX: goodbye
909
+ console.log("bye");
910
+ }
911
+ }
912
+ /* v8 ignore stop */
913
+ return { exitReason, toolResult: exitToolResult };
370
914
  }
371
915
  async function main(agentName, options) {
372
916
  if (agentName)
373
917
  (0, identity_1.setAgentName)(agentName);
374
918
  const pasteDebounceMs = options?.pasteDebounceMs ?? 50;
919
+ // Safety net: process-level SIGINT handler ensures Ctrl+C always exits,
920
+ // even when Ink's event loop is blocked by expensive renders.
921
+ /* v8 ignore start -- process signal handler @preserve */
922
+ let sigintCount = 0;
923
+ const sigintHandler = () => {
924
+ sigintCount++;
925
+ if (sigintCount >= 2)
926
+ process.exit(1);
927
+ };
928
+ if (!options?._testInputSource) {
929
+ process.on("SIGINT", sigintHandler);
930
+ }
931
+ /* v8 ignore stop */
932
+ // Register spinner hooks so log output clears the spinner before printing
933
+ (0, nerves_1.registerSpinnerHooks)(pauseActiveSpinner, resumeActiveSpinner);
934
+ // Fallback: apply pending updates for daemon-less direct CLI usage
935
+ (0, update_hooks_1.registerUpdateHook)(bundle_meta_1.bundleMetaHook);
936
+ (0, update_hooks_1.registerUpdateHook)(agent_config_v2_1.agentConfigV2Hook);
937
+ await (0, update_hooks_1.applyPendingUpdates)((0, identity_1.getAgentBundlesRoot)(), (0, bundle_manifest_1.getPackageVersion)());
375
938
  // Fail fast if provider is misconfigured (triggers human-readable error + exit)
376
- (0, core_1.getProvider)();
377
- const registry = (0, commands_1.createCommandRegistry)();
378
- (0, commands_1.registerDefaultCommands)(registry);
939
+ (0, core_1.getProvider)("human");
379
940
  // Resolve context kernel (identity + channel) for CLI
380
941
  const friendsPath = path.join((0, identity_1.getAgentRoot)(), "friends");
381
942
  const friendStore = new store_file_1.FileFriendStore(friendsPath);
382
943
  const username = os.userInfo().username;
383
- const hostname = os.hostname();
384
- const localExternalId = `${username}@${hostname}`;
944
+ const localExternalId = username;
385
945
  const resolver = new resolver_1.FriendResolver(friendStore, {
386
946
  provider: "local",
387
947
  externalId: localExternalId,
@@ -389,13 +949,6 @@ async function main(agentName, options) {
389
949
  channel: "cli",
390
950
  });
391
951
  const resolvedContext = await resolver.resolve();
392
- const cliToolContext = {
393
- /* v8 ignore next -- CLI has no OAuth sign-in; this no-op satisfies the interface @preserve */
394
- signin: async () => undefined,
395
- context: resolvedContext,
396
- friendStore,
397
- summarize: (0, core_1.createSummarize)(),
398
- };
399
952
  const friendId = resolvedContext.friend.id;
400
953
  const agentConfig = (0, identity_1.loadAgentConfig)();
401
954
  (0, cli_logging_1.configureCliRuntimeLogger)(friendId, {
@@ -418,187 +971,138 @@ async function main(agentName, options) {
418
971
  }
419
972
  // Load existing session or start fresh
420
973
  const existing = (0, context_1.loadSession)(sessPath);
421
- const messages = existing?.messages && existing.messages.length > 0
974
+ let sessionState = existing?.state;
975
+ let sessionEvents = existing?.events ?? [];
976
+ const mcpManager = await (0, mcp_manager_1.getSharedMcpManager)() ?? undefined;
977
+ const sessionMessages = existing?.messages && existing.messages.length > 0
422
978
  ? existing.messages
423
- : [{ role: "system", content: await (0, prompt_1.buildSystem)("cli", undefined, resolvedContext) }];
424
- // Pending queue drain: inject pending messages as harness-context + assistant-content pairs
425
- const pendingDir = (0, pending_1.getPendingDir)((0, identity_1.getAgentName)(), friendId, "cli", "session");
426
- const drainToMessages = () => {
427
- const pending = (0, pending_1.drainPending)(pendingDir);
428
- if (pending.length === 0)
429
- return 0;
430
- for (const msg of pending) {
431
- messages.push({ role: "user", name: "harness", content: `[proactive message from ${msg.from}]` });
432
- messages.push({ role: "assistant", content: msg.content });
433
- }
434
- return pending.length;
435
- };
436
- // Startup drain: deliver offline messages
437
- const startupCount = drainToMessages();
438
- if (startupCount > 0) {
439
- (0, context_1.saveSession)(sessPath, messages);
440
- }
441
- const rl = readline.createInterface({ input: process.stdin, output: process.stdout, terminal: true });
442
- const ctrl = new InputController(rl);
443
- let currentAbort = null;
444
- const history = [];
445
- let closed = false;
446
- rl.on("close", () => { closed = true; });
447
- // eslint-disable-next-line no-console -- terminal UX: startup banner
448
- console.log(`\n${(0, identity_1.getAgentName)()} (type /commands for help)\n`);
449
- const cliCallbacks = createCliCallbacks();
450
- process.stdout.write("\x1b[36m> \x1b[0m");
451
- // Ctrl-C at the input prompt: clear line or warn/exit
452
- // readline with terminal:true catches Ctrl-C in raw mode (no ^C echo)
453
- rl.on("SIGINT", () => {
454
- const rlInt = rl;
455
- const currentLine = rlInt.line || "";
456
- const result = handleSigint(rl, currentLine);
457
- if (result === "clear") {
458
- rlInt.line = "";
459
- rlInt.cursor = 0;
460
- process.stdout.write("\r\x1b[K\x1b[36m> \x1b[0m");
461
- }
462
- else if (result === "warn") {
463
- rlInt.line = "";
464
- rlInt.cursor = 0;
465
- process.stdout.write("\r\x1b[K");
466
- process.stderr.write("press Ctrl-C again to exit\n");
467
- process.stdout.write("\x1b[36m> \x1b[0m");
468
- }
469
- else {
470
- rl.close();
471
- }
472
- });
473
- // Debounced line iterator: collects rapid-fire lines (paste) into a single input
474
- async function* debouncedLines(source) {
475
- if (pasteDebounceMs <= 0) {
476
- yield* source;
477
- return;
478
- }
479
- const iter = source[Symbol.asyncIterator]();
480
- while (true) {
481
- const first = await iter.next();
482
- if (first.done)
483
- break;
484
- // Collect any lines that arrive within the debounce window (paste detection)
485
- const lines = [first.value];
486
- let more = true;
487
- while (more) {
488
- const raced = await Promise.race([
489
- iter.next().then((r) => ({ kind: "line", result: r })),
490
- new Promise((r) => setTimeout(() => r({ kind: "timeout" }), pasteDebounceMs)),
491
- ]);
492
- if (raced.kind === "timeout") {
493
- more = false;
494
- }
495
- else if (raced.result.done) {
496
- more = false;
497
- }
498
- else {
499
- lines.push(raced.result.value);
500
- }
501
- }
502
- yield lines.join("\n");
503
- }
504
- }
979
+ : [{ role: "system", content: (0, prompt_1.flattenSystemPrompt)(await (0, prompt_1.buildSystem)("cli", {}, resolvedContext)) }];
980
+ // Repair any orphaned tool calls from a crash mid-turn
981
+ (0, core_1.repairOrphanedToolCalls)(sessionMessages);
982
+ // Per-turn pipeline input: CLI capabilities and pending dir
983
+ const cliCapabilities = (0, channel_1.getChannelCapabilities)("cli");
984
+ const currentAgentName = (0, identity_1.getAgentName)();
985
+ const pendingDir = (0, pending_1.getPendingDir)(currentAgentName, friendId, "cli", "session");
986
+ const summarize = (0, core_1.createSummarize)("human");
987
+ const cliFailoverState = { pending: null };
505
988
  try {
506
- for await (const input of debouncedLines(rl)) {
507
- if (closed)
508
- break;
509
- if (!input.trim()) {
510
- process.stdout.write("\x1b[36m> \x1b[0m");
511
- continue;
512
- }
513
- const trustGate = (0, trust_gate_1.enforceTrustGate)({
514
- friend: resolvedContext.friend,
515
- provider: "local",
516
- externalId: localExternalId,
517
- channel: "cli",
518
- });
519
- if (!trustGate.allowed) {
520
- if (trustGate.reason === "stranger_first_reply") {
521
- process.stdout.write(`${trustGate.autoReply}\n`);
989
+ await runCliSession({
990
+ agentName: currentAgentName,
991
+ pasteDebounceMs,
992
+ messages: sessionMessages,
993
+ lastActivityAt: sessionState?.lastFriendActivityAt,
994
+ _testInputSource: options?._testInputSource,
995
+ onAsyncAssistantMessage: async (messages, _assistantMessage) => {
996
+ const prepared = (0, context_1.postTurnTrim)(messages);
997
+ const events = (0, context_1.postTurnPersist)(sessPath, prepared, undefined, sessionState);
998
+ /* v8 ignore next -- defensive: postTurnPersist always returns events in practice @preserve */
999
+ sessionEvents = events.length > 0 ? events : sessionEvents;
1000
+ },
1001
+ runTurn: async (messages, userInput, callbacks, signal, toolContext, userContent) => {
1002
+ // Run the full per-turn pipeline: resolve -> gate -> session -> drain -> runAgent -> postTurn -> tokens
1003
+ // User message passed via input.messages so the pipeline can prepend pending messages to it.
1004
+ const failoverState = cliFailoverState;
1005
+ // Capture terminal errors instead of displaying immediately — the failover
1006
+ // message replaces the raw error if failover triggers successfully.
1007
+ let capturedTerminalError = null;
1008
+ /* v8 ignore start -- failover-aware callback wrapper: tested via pipeline integration @preserve */
1009
+ const failoverAwareCallbacks = {
1010
+ ...callbacks,
1011
+ // Save session after each tool result for crash recovery (deferred to avoid blocking)
1012
+ onToolResult: (turnMessages) => {
1013
+ const prepared = (0, context_1.postTurnTrim)(turnMessages);
1014
+ (0, context_1.deferPostTurnPersist)(sessPath, prepared, undefined, sessionState);
1015
+ },
1016
+ onError: (error, severity) => {
1017
+ if (severity === "terminal" && failoverState) {
1018
+ capturedTerminalError = error;
1019
+ callbacks.onError(new Error(""), "transient");
1020
+ return;
1021
+ }
1022
+ callbacks.onError(error, severity);
1023
+ },
1024
+ };
1025
+ /* v8 ignore stop */
1026
+ const result = await (0, pipeline_1.handleInboundTurn)({
1027
+ channel: "cli",
1028
+ sessionKey: "session",
1029
+ capabilities: cliCapabilities,
1030
+ messages: [{ role: "user", content: userContent ?? userInput }],
1031
+ continuityIngressTexts: getCliContinuityIngressTexts(userInput),
1032
+ callbacks: failoverAwareCallbacks,
1033
+ friendResolver: { resolve: () => Promise.resolve(resolvedContext) },
1034
+ sessionLoader: {
1035
+ loadOrCreate: () => Promise.resolve({
1036
+ messages,
1037
+ sessionPath: sessPath,
1038
+ state: sessionState,
1039
+ events: sessionEvents,
1040
+ }),
1041
+ },
1042
+ pendingDir,
1043
+ friendStore,
1044
+ provider: "local",
1045
+ externalId: localExternalId,
1046
+ enforceTrustGate: trust_gate_1.enforceTrustGate,
1047
+ drainPending: pending_1.drainPending,
1048
+ drainDeferredReturns: (deferredFriendId) => (0, pending_1.drainDeferredReturns)(currentAgentName, deferredFriendId),
1049
+ runAgent: (msgs, cb, channel, sig, opts) => (0, core_1.runAgent)(msgs, cb, channel, sig, {
1050
+ ...opts,
1051
+ toolContext: {
1052
+ /* v8 ignore next -- default no-op signin; pipeline provides the real one @preserve */
1053
+ signin: async () => undefined,
1054
+ ...opts?.toolContext,
1055
+ summarize,
1056
+ },
1057
+ }),
1058
+ postTurn: (turnMessages, sessionPathArg, usage, hooks, state) => {
1059
+ // Trim synchronously (mutates turnMessages for next turn),
1060
+ // then defer envelope build + disk I/O to avoid blocking the TUI.
1061
+ const prepared = (0, context_1.postTurnTrim)(turnMessages, usage, hooks);
1062
+ sessionState = state;
1063
+ (0, context_1.deferPostTurnPersist)(sessionPathArg, prepared, usage, state).then((events) => {
1064
+ /* v8 ignore next -- defensive: deferPostTurnPersist always resolves events in practice @preserve */
1065
+ sessionEvents = events.length > 0 ? events : sessionEvents;
1066
+ });
1067
+ },
1068
+ accumulateFriendTokens: tokens_1.accumulateFriendTokens,
1069
+ signal,
1070
+ runAgentOptions: {
1071
+ toolChoiceRequired: (0, commands_1.getToolChoiceRequired)(),
1072
+ traceId: (0, nerves_1.createTraceId)(),
1073
+ mcpManager,
1074
+ toolContext,
1075
+ },
1076
+ failoverState,
1077
+ });
1078
+ /* v8 ignore start -- failover display: tested via pipeline integration tests @preserve */
1079
+ if (result.failoverMessage) {
1080
+ // Failover handled it — show the actionable message instead of the raw error
1081
+ process.stdout.write(`\x1b[33m${result.failoverMessage}\x1b[0m\n`);
522
1082
  }
523
- if (closed)
524
- break;
525
- process.stdout.write("\x1b[36m> \x1b[0m");
526
- continue;
527
- }
528
- // Check for slash commands
529
- const parsed = (0, commands_1.parseSlashCommand)(input);
530
- if (parsed) {
531
- const dispatchResult = registry.dispatch(parsed.command, { channel: "cli" });
532
- if (dispatchResult.handled && dispatchResult.result) {
533
- if (dispatchResult.result.action === "exit") {
534
- break;
535
- }
536
- else if (dispatchResult.result.action === "new") {
537
- messages.length = 0;
538
- messages.push({ role: "system", content: await (0, prompt_1.buildSystem)("cli") });
539
- (0, context_1.deleteSession)(sessPath);
540
- // eslint-disable-next-line no-console -- terminal UX: session cleared
541
- console.log("session cleared");
542
- process.stdout.write("\x1b[36m> \x1b[0m");
543
- continue;
544
- }
545
- else if (dispatchResult.result.action === "response") {
546
- // eslint-disable-next-line no-console -- terminal UX: command dispatch result
547
- console.log(dispatchResult.result.message || "");
548
- process.stdout.write("\x1b[36m> \x1b[0m");
549
- continue;
1083
+ else if (capturedTerminalError) {
1084
+ // Failover didn't trigger (no failoverState, or sequence failed) — show the raw error
1085
+ process.stderr.write(`\x1b[31m${(0, format_1.formatError)(capturedTerminalError)}\x1b[0m\n`);
1086
+ }
1087
+ /* v8 ignore stop */
1088
+ // Handle gate rejection: display auto-reply if present
1089
+ if (!result.gateResult.allowed) {
1090
+ if ("autoReply" in result.gateResult && result.gateResult.autoReply) {
1091
+ process.stdout.write(`${result.gateResult.autoReply}\n`);
550
1092
  }
551
1093
  }
552
- }
553
- // Re-style the echoed input lines (readline terminal:true echoes each line)
554
- // For multiline paste, each line was echoed separately — erase them all
555
- const cols = process.stdout.columns || 80;
556
- const inputLines = input.split("\n");
557
- let echoRows = 0;
558
- for (const line of inputLines) {
559
- echoRows += Math.ceil((2 + line.length) / cols); // "> " prefix + line content
560
- }
561
- process.stdout.write(`\x1b[${echoRows}A\x1b[K` + `\x1b[1m> ${inputLines[0]}${inputLines.length > 1 ? ` (+${inputLines.length - 1} lines)` : ""}\x1b[0m\n\n`);
562
- messages.push({ role: "user", content: input });
563
- addHistory(history, input);
564
- currentAbort = new AbortController();
565
- const traceId = (0, nerves_1.createTraceId)();
566
- ctrl.suppress(() => currentAbort.abort());
567
- let result;
568
- try {
569
- result = await (0, core_1.runAgent)(messages, cliCallbacks, "cli", currentAbort.signal, {
570
- toolChoiceRequired: (0, commands_1.getToolChoiceRequired)(),
571
- toolContext: cliToolContext,
572
- traceId,
573
- });
574
- }
575
- catch {
576
- // AbortError — silently return to prompt
577
- }
578
- cliCallbacks.flushMarkdown();
579
- ctrl.restore();
580
- currentAbort = null;
581
- // Safety net: never silently swallow an empty response
582
- const lastMsg = messages[messages.length - 1];
583
- if (lastMsg?.role === "assistant" && !(typeof lastMsg.content === "string" ? lastMsg.content : "").trim()) {
584
- process.stderr.write("\x1b[33m(empty response)\x1b[0m\n");
585
- }
586
- process.stdout.write("\n\n");
587
- (0, context_1.postTurn)(messages, sessPath, result?.usage);
588
- await (0, tokens_1.accumulateFriendTokens)(friendStore, resolvedContext.friend.id, result?.usage);
589
- // Post-turn: drain any pending messages that arrived during runAgent
590
- drainToMessages();
591
- // Post-turn: refresh system prompt so active sessions metadata is current
592
- await (0, prompt_refresh_1.refreshSystemPrompt)(messages, "cli", undefined, resolvedContext);
593
- if (closed)
594
- break;
595
- process.stdout.write("\x1b[36m> \x1b[0m");
596
- }
1094
+ return { usage: result.usage, turnOutcome: result.turnOutcome, commandAction: result.commandAction };
1095
+ },
1096
+ });
597
1097
  }
598
1098
  finally {
599
1099
  sessionLock?.release();
600
- rl.close();
601
- // eslint-disable-next-line no-console -- terminal UX: goodbye
602
- console.log("bye");
603
1100
  }
1101
+ // Force exit: lingering handles (Ink cleanup timers, MCP connections) keep the
1102
+ // event loop alive after the interactive session ends. This is safe because all
1103
+ // session persistence has already completed in the finally block above.
1104
+ /* v8 ignore next -- process.exit not callable in vitest @preserve */
1105
+ if (!options?._testInputSource)
1106
+ process.exit(0);
604
1107
  }
1108
+ // CI trigger