@ouro.bot/cli 0.1.0-alpha.46 → 0.1.0-alpha.461

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 (354) hide show
  1. package/README.md +127 -19
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +2928 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +832 -0
  16. package/dist/heart/agent-entry.js +58 -3
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +426 -0
  28. package/dist/heart/bridges/manager.js +358 -0
  29. package/dist/heart/bridges/state-machine.js +135 -0
  30. package/dist/heart/bridges/store.js +123 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +110 -128
  35. package/dist/heart/core.js +745 -227
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +490 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +216 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +631 -0
  44. package/dist/heart/daemon/cli-exec.js +6158 -0
  45. package/dist/heart/daemon/cli-help.js +457 -0
  46. package/dist/heart/daemon/cli-parse.js +1273 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +561 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/connect-bay.js +323 -0
  51. package/dist/heart/daemon/daemon-cli.js +29 -1688
  52. package/dist/heart/daemon/daemon-entry.js +345 -3
  53. package/dist/heart/daemon/daemon-health.js +141 -0
  54. package/dist/heart/daemon/daemon-runtime-sync.js +190 -12
  55. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  56. package/dist/heart/daemon/daemon.js +684 -58
  57. package/dist/heart/daemon/doctor-types.js +8 -0
  58. package/dist/heart/daemon/doctor.js +435 -0
  59. package/dist/heart/daemon/health-monitor.js +92 -1
  60. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  61. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  62. package/dist/heart/daemon/http-health-probe.js +80 -0
  63. package/dist/heart/daemon/human-command-screens.js +234 -0
  64. package/dist/heart/daemon/human-readiness.js +114 -0
  65. package/dist/heart/daemon/inner-status.js +89 -0
  66. package/dist/heart/daemon/interactive-repair.js +394 -0
  67. package/dist/heart/daemon/launchd.js +25 -5
  68. package/dist/heart/daemon/log-tailer.js +82 -12
  69. package/dist/heart/daemon/logs-prune.js +110 -0
  70. package/dist/heart/daemon/message-router.js +2 -2
  71. package/dist/heart/daemon/os-cron-deps.js +134 -0
  72. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  73. package/dist/heart/daemon/ouro-entry.js +3 -1
  74. package/dist/heart/daemon/process-manager.js +214 -0
  75. package/dist/heart/daemon/provider-discovery.js +137 -0
  76. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  77. package/dist/heart/daemon/pulse.js +475 -0
  78. package/dist/heart/daemon/readiness-repair.js +365 -0
  79. package/dist/heart/daemon/run-hooks.js +2 -0
  80. package/dist/heart/daemon/runtime-logging.js +67 -16
  81. package/dist/heart/daemon/runtime-metadata.js +73 -0
  82. package/dist/heart/daemon/runtime-mode.js +67 -0
  83. package/dist/heart/daemon/safe-mode.js +161 -0
  84. package/dist/heart/daemon/sense-manager.js +178 -37
  85. package/dist/heart/daemon/session-id-resolver.js +131 -0
  86. package/dist/heart/daemon/skill-management-installer.js +94 -0
  87. package/dist/heart/daemon/socket-client.js +307 -0
  88. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  89. package/dist/heart/daemon/startup-tui.js +264 -0
  90. package/dist/heart/daemon/task-scheduler.js +3 -25
  91. package/dist/heart/daemon/terminal-ui.js +499 -0
  92. package/dist/heart/daemon/thoughts.js +298 -13
  93. package/dist/heart/daemon/up-progress.js +366 -0
  94. package/dist/heart/delegation.js +62 -0
  95. package/dist/heart/habits/habit-migration.js +189 -0
  96. package/dist/heart/habits/habit-parser.js +140 -0
  97. package/dist/heart/habits/habit-runtime-state.js +100 -0
  98. package/dist/heart/habits/habit-scheduler.js +372 -0
  99. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  100. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  101. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  102. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  103. package/dist/heart/identity.js +201 -66
  104. package/dist/heart/kept-notes.js +357 -0
  105. package/dist/heart/kicks.js +1 -1
  106. package/dist/heart/machine-identity.js +161 -0
  107. package/dist/heart/mcp/mcp-server.js +653 -0
  108. package/dist/heart/migrate-config.js +100 -0
  109. package/dist/heart/model-capabilities.js +59 -0
  110. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  111. package/dist/heart/outlook/outlook-http-response.js +7 -0
  112. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  113. package/dist/heart/outlook/outlook-http-static.js +99 -0
  114. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  115. package/dist/heart/outlook/outlook-http.js +99 -0
  116. package/dist/heart/outlook/outlook-read.js +31 -0
  117. package/dist/heart/outlook/outlook-types.js +27 -0
  118. package/dist/heart/outlook/outlook-view.js +195 -0
  119. package/dist/heart/outlook/readers/agent-machine.js +359 -0
  120. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  121. package/dist/heart/outlook/readers/mail.js +273 -0
  122. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  123. package/dist/heart/outlook/readers/sessions.js +232 -0
  124. package/dist/heart/outlook/readers/shared.js +111 -0
  125. package/dist/heart/platform.js +81 -0
  126. package/dist/heart/progress-story.js +42 -0
  127. package/dist/heart/provider-attempt.js +134 -0
  128. package/dist/heart/provider-binding-resolver.js +255 -0
  129. package/dist/heart/provider-credentials.js +424 -0
  130. package/dist/heart/provider-failover.js +266 -0
  131. package/dist/heart/provider-models.js +81 -0
  132. package/dist/heart/provider-ping.js +262 -0
  133. package/dist/heart/provider-state.js +216 -0
  134. package/dist/heart/provider-visibility.js +188 -0
  135. package/dist/heart/providers/anthropic-token.js +131 -0
  136. package/dist/heart/providers/anthropic.js +193 -55
  137. package/dist/heart/providers/azure.js +103 -12
  138. package/dist/heart/providers/error-classification.js +63 -0
  139. package/dist/heart/providers/github-copilot.js +145 -0
  140. package/dist/heart/providers/minimax-vlm.js +189 -0
  141. package/dist/heart/providers/minimax.js +29 -7
  142. package/dist/heart/providers/openai-codex.js +62 -38
  143. package/dist/heart/runtime-capability-check.js +170 -0
  144. package/dist/heart/runtime-credentials.js +260 -0
  145. package/dist/heart/sense-truth.js +11 -4
  146. package/dist/heart/session-activity.js +190 -0
  147. package/dist/heart/session-events.js +855 -0
  148. package/dist/heart/session-transcript.js +167 -0
  149. package/dist/heart/start-of-turn-packet.js +345 -0
  150. package/dist/heart/streaming.js +36 -27
  151. package/dist/heart/sync.js +332 -0
  152. package/dist/heart/target-resolution.js +127 -0
  153. package/dist/heart/tempo.js +93 -0
  154. package/dist/heart/temporal-view.js +41 -0
  155. package/dist/heart/tool-activity-callbacks.js +36 -0
  156. package/dist/heart/tool-description.js +135 -0
  157. package/dist/heart/tool-friction.js +55 -0
  158. package/dist/heart/tool-loop.js +200 -0
  159. package/dist/heart/turn-context.js +361 -0
  160. package/dist/heart/turn-coordinator.js +28 -0
  161. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  162. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  163. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  164. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  165. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  166. package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
  167. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  168. package/dist/mailroom/attention.js +154 -0
  169. package/dist/mailroom/blob-store.js +302 -0
  170. package/dist/mailroom/core.js +538 -0
  171. package/dist/mailroom/entry.js +160 -0
  172. package/dist/mailroom/file-store.js +392 -0
  173. package/dist/mailroom/mbox-import.js +105 -0
  174. package/dist/mailroom/outbound.js +177 -0
  175. package/dist/mailroom/policy.js +263 -0
  176. package/dist/mailroom/reader.js +166 -0
  177. package/dist/mailroom/smtp-ingress.js +176 -0
  178. package/dist/mailroom/travel-extract.js +89 -0
  179. package/dist/mind/bundle-manifest.js +7 -1
  180. package/dist/mind/context.js +132 -93
  181. package/dist/mind/diary-integrity.js +60 -0
  182. package/dist/mind/{memory.js → diary.js} +74 -93
  183. package/dist/mind/embedding-provider.js +60 -0
  184. package/dist/mind/file-state.js +179 -0
  185. package/dist/mind/friends/channel.js +30 -0
  186. package/dist/mind/friends/group-context.js +144 -0
  187. package/dist/mind/friends/resolver.js +38 -1
  188. package/dist/mind/friends/store-file.js +39 -3
  189. package/dist/mind/friends/trust-explanation.js +74 -0
  190. package/dist/mind/friends/types.js +2 -2
  191. package/dist/mind/journal-index.js +161 -0
  192. package/dist/mind/note-search.js +268 -0
  193. package/dist/mind/obligation-steering.js +221 -0
  194. package/dist/mind/pending.js +66 -7
  195. package/dist/mind/prompt-refresh.js +3 -2
  196. package/dist/mind/prompt.js +976 -169
  197. package/dist/mind/provenance-trust.js +26 -0
  198. package/dist/mind/scrutiny.js +173 -0
  199. package/dist/nerves/cli-logging.js +7 -1
  200. package/dist/nerves/coverage/audit-rules.js +15 -6
  201. package/dist/nerves/coverage/audit.js +28 -2
  202. package/dist/nerves/coverage/cli.js +1 -1
  203. package/dist/nerves/coverage/contract.js +5 -5
  204. package/dist/nerves/coverage/file-completeness.js +83 -5
  205. package/dist/nerves/coverage/run-artifacts.js +1 -1
  206. package/dist/nerves/event-buffer.js +111 -0
  207. package/dist/nerves/index.js +224 -4
  208. package/dist/nerves/observation.js +20 -0
  209. package/dist/nerves/redact.js +79 -0
  210. package/dist/nerves/runtime.js +5 -1
  211. package/dist/outlook-ui/assets/index-BBM5EysT.js +61 -0
  212. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  213. package/dist/outlook-ui/index.html +15 -0
  214. package/dist/repertoire/ado-client.js +15 -56
  215. package/dist/repertoire/ado-semantic.js +11 -10
  216. package/dist/repertoire/api-client.js +97 -0
  217. package/dist/repertoire/bitwarden-store.js +774 -0
  218. package/dist/repertoire/bundle-templates.js +72 -0
  219. package/dist/repertoire/bw-installer.js +180 -0
  220. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  221. package/dist/repertoire/coding/context-pack.js +330 -0
  222. package/dist/repertoire/coding/feedback.js +197 -30
  223. package/dist/repertoire/coding/manager.js +158 -9
  224. package/dist/repertoire/coding/spawner.js +55 -9
  225. package/dist/repertoire/coding/tools.js +170 -7
  226. package/dist/repertoire/commerce-errors.js +109 -0
  227. package/dist/repertoire/commerce-self-test.js +156 -0
  228. package/dist/repertoire/credential-access.js +111 -0
  229. package/dist/repertoire/duffel-client.js +185 -0
  230. package/dist/repertoire/github-client.js +14 -55
  231. package/dist/repertoire/graph-client.js +11 -52
  232. package/dist/repertoire/guardrails.js +396 -0
  233. package/dist/repertoire/mcp-client.js +255 -0
  234. package/dist/repertoire/mcp-manager.js +305 -0
  235. package/dist/repertoire/mcp-tools.js +63 -0
  236. package/dist/repertoire/shell-sessions.js +133 -0
  237. package/dist/repertoire/skills.js +15 -24
  238. package/dist/repertoire/stripe-client.js +131 -0
  239. package/dist/repertoire/tasks/board.js +43 -5
  240. package/dist/repertoire/tasks/fix.js +182 -0
  241. package/dist/repertoire/tasks/index.js +37 -4
  242. package/dist/repertoire/tasks/lifecycle.js +2 -2
  243. package/dist/repertoire/tasks/parser.js +3 -2
  244. package/dist/repertoire/tasks/scanner.js +194 -37
  245. package/dist/repertoire/tasks/transitions.js +16 -78
  246. package/dist/repertoire/tool-results.js +29 -0
  247. package/dist/repertoire/tools-attachments.js +317 -0
  248. package/dist/repertoire/tools-base.js +44 -690
  249. package/dist/repertoire/tools-bluebubbles.js +1 -0
  250. package/dist/repertoire/tools-bridge.js +141 -0
  251. package/dist/repertoire/tools-bundle.js +984 -0
  252. package/dist/repertoire/tools-config.js +185 -0
  253. package/dist/repertoire/tools-continuity.js +248 -0
  254. package/dist/repertoire/tools-credential.js +361 -0
  255. package/dist/repertoire/tools-files.js +342 -0
  256. package/dist/repertoire/tools-flight.js +224 -0
  257. package/dist/repertoire/tools-flow.js +105 -0
  258. package/dist/repertoire/tools-github.js +1 -7
  259. package/dist/repertoire/tools-mail.js +762 -0
  260. package/dist/repertoire/tools-notes.js +376 -0
  261. package/dist/repertoire/tools-session.js +739 -0
  262. package/dist/repertoire/tools-shell.js +120 -0
  263. package/dist/repertoire/tools-stripe.js +180 -0
  264. package/dist/repertoire/tools-surface.js +243 -0
  265. package/dist/repertoire/tools-teams.js +9 -39
  266. package/dist/repertoire/tools-travel.js +125 -0
  267. package/dist/repertoire/tools-user-profile.js +144 -0
  268. package/dist/repertoire/tools-vault.js +40 -0
  269. package/dist/repertoire/tools.js +144 -113
  270. package/dist/repertoire/travel-api-client.js +360 -0
  271. package/dist/repertoire/user-profile.js +131 -0
  272. package/dist/repertoire/vault-setup.js +246 -0
  273. package/dist/repertoire/vault-unlock.js +561 -0
  274. package/dist/scripts/claude-code-hook.js +41 -0
  275. package/dist/scripts/claude-code-stop-hook.js +47 -0
  276. package/dist/senses/attention-queue.js +116 -0
  277. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  278. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  279. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  280. package/dist/senses/bluebubbles/entry.js +73 -0
  281. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +7 -3
  282. package/dist/senses/bluebubbles/index.js +1620 -0
  283. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  284. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  285. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  286. package/dist/senses/bluebubbles/replay.js +129 -0
  287. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +2 -2
  288. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  289. package/dist/senses/cli/bracketed-paste.js +82 -0
  290. package/dist/senses/cli/image-paste.js +287 -0
  291. package/dist/senses/cli/image-ref-navigation.js +75 -0
  292. package/dist/senses/cli/ink-app.js +156 -0
  293. package/dist/senses/cli/inline-diff.js +64 -0
  294. package/dist/senses/cli/input-keys.js +174 -0
  295. package/dist/senses/cli/kill-ring.js +86 -0
  296. package/dist/senses/cli/message-list.js +51 -0
  297. package/dist/senses/cli/ouro-tui.js +605 -0
  298. package/dist/senses/cli/spinner-imperative.js +135 -0
  299. package/dist/senses/cli/spinner.js +101 -0
  300. package/dist/senses/cli/status-line.js +60 -0
  301. package/dist/senses/cli/streaming-markdown.js +526 -0
  302. package/dist/senses/cli/tool-display.js +83 -0
  303. package/dist/senses/cli/tool-render.js +85 -0
  304. package/dist/senses/cli/tui-store.js +240 -0
  305. package/dist/senses/cli/virtual-list.js +35 -0
  306. package/dist/senses/cli-entry.js +60 -8
  307. package/dist/senses/cli-layout.js +187 -0
  308. package/dist/senses/cli.js +516 -211
  309. package/dist/senses/commands.js +66 -3
  310. package/dist/senses/habit-turn-message.js +108 -0
  311. package/dist/senses/inner-dialog-worker.js +97 -17
  312. package/dist/senses/inner-dialog.js +534 -106
  313. package/dist/senses/mail-entry.js +66 -0
  314. package/dist/senses/mail.js +224 -0
  315. package/dist/senses/pipeline.js +533 -72
  316. package/dist/senses/proactive-content-guard.js +51 -0
  317. package/dist/senses/shared-turn.js +205 -0
  318. package/dist/senses/surface-tool.js +68 -0
  319. package/dist/senses/teams-entry.js +60 -8
  320. package/dist/senses/teams.js +413 -163
  321. package/dist/senses/trust-gate.js +5 -5
  322. package/package.json +37 -7
  323. package/skills/agent-commerce.md +106 -0
  324. package/skills/browser-navigation.md +117 -0
  325. package/skills/commerce-setup-guide.md +116 -0
  326. package/skills/commerce-setup.md +84 -0
  327. package/skills/configure-dev-tools.md +101 -0
  328. package/skills/travel-planning.md +138 -0
  329. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  330. package/dist/heart/daemon/subagent-installer.js +0 -166
  331. package/dist/mind/associative-recall.js +0 -209
  332. package/dist/senses/bluebubbles-entry.js +0 -13
  333. package/dist/senses/bluebubbles.js +0 -1028
  334. package/dist/senses/debug-activity.js +0 -127
  335. package/subagents/README.md +0 -86
  336. package/subagents/work-doer.md +0 -237
  337. package/subagents/work-merger.md +0 -618
  338. package/subagents/work-planner.md +0 -390
  339. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  340. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  341. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  342. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  343. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  344. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  345. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  346. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  347. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  348. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  349. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  350. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  351. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  352. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  353. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  354. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -41,6 +41,9 @@ exports.readTaskFile = readTaskFile;
41
41
  exports.buildTaskTriggeredMessage = buildTaskTriggeredMessage;
42
42
  exports.deriveResumeCheckpoint = deriveResumeCheckpoint;
43
43
  exports.innerDialogSessionPath = innerDialogSessionPath;
44
+ exports.enrichDelegatedFromWithBridge = enrichDelegatedFromWithBridge;
45
+ exports.routeDelegatedCompletion = routeDelegatedCompletion;
46
+ exports.buildParseErrorNudge = buildParseErrorNudge;
44
47
  exports.runInnerDialogTurn = runInnerDialogTurn;
45
48
  const fs = __importStar(require("fs"));
46
49
  const path = __importStar(require("path"));
@@ -49,14 +52,28 @@ const core_1 = require("../heart/core");
49
52
  const identity_1 = require("../heart/identity");
50
53
  const context_1 = require("../mind/context");
51
54
  const prompt_1 = require("../mind/prompt");
55
+ const mcp_manager_1 = require("../repertoire/mcp-manager");
56
+ const tools_1 = require("../repertoire/tools");
52
57
  const bundle_manifest_1 = require("../mind/bundle-manifest");
53
58
  const pending_1 = require("../mind/pending");
59
+ const obligations_1 = require("../arc/obligations");
60
+ const attention_queue_1 = require("./attention-queue");
61
+ const packets_1 = require("../arc/packets");
54
62
  const channel_1 = require("../mind/friends/channel");
55
63
  const trust_gate_1 = require("./trust-gate");
56
64
  const tokens_1 = require("../mind/friends/tokens");
57
65
  const pipeline_1 = require("./pipeline");
58
66
  const nerves_1 = require("../nerves");
59
67
  const runtime_1 = require("../nerves/runtime");
68
+ const manager_1 = require("../heart/bridges/manager");
69
+ const session_activity_1 = require("../heart/session-activity");
70
+ const bluebubbles_1 = require("./bluebubbles");
71
+ const habit_turn_message_1 = require("./habit-turn-message");
72
+ const journal_index_1 = require("../mind/journal-index");
73
+ const habit_parser_1 = require("../heart/habits/habit-parser");
74
+ const habit_runtime_state_1 = require("../heart/habits/habit-runtime-state");
75
+ const cadence_1 = require("../heart/daemon/cadence");
76
+ const daemon_health_1 = require("../heart/daemon/daemon-health");
60
77
  const DEFAULT_INNER_DIALOG_INSTINCTS = [
61
78
  {
62
79
  id: "heartbeat_checkin",
@@ -95,23 +112,31 @@ function buildNonCanonicalCleanupNudge(nonCanonicalPaths) {
95
112
  }
96
113
  return [
97
114
  "## canonical cleanup nudge",
98
- "I found non-canonical files in my bundle. I should distill anything valuable into your memory system and remove these files.",
115
+ "I found non-canonical files in my bundle. I should distill anything valuable into my diary and remove these files.",
99
116
  ...listed,
100
117
  ].join("\n");
101
118
  }
119
+ function displayCheckpoint(checkpoint) {
120
+ const trimmed = checkpoint?.trim();
121
+ if (!trimmed || trimmed === "no prior checkpoint recorded") {
122
+ return undefined;
123
+ }
124
+ return trimmed;
125
+ }
102
126
  function buildInstinctUserMessage(instincts, _reason, state) {
103
127
  const active = instincts.find((instinct) => instinct.enabled !== false) ?? DEFAULT_INNER_DIALOG_INSTINCTS[0];
104
- const checkpoint = state.checkpoint?.trim();
128
+ const checkpoint = displayCheckpoint(state.checkpoint);
105
129
  const lines = [active.prompt];
106
130
  if (checkpoint) {
107
- lines.push(`\nlast i remember: ${checkpoint}`);
131
+ lines.push(`\nlast checkpoint: ${checkpoint}`);
108
132
  }
109
133
  return lines.join("\n");
110
134
  }
111
135
  function readTaskFile(agentRoot, taskId) {
112
- // Task files live in collection subdirectories (one-shots, ongoing, habits).
136
+ // Task files live in collection subdirectories (one-shots, ongoing).
113
137
  // Try each collection, then fall back to root tasks/ for legacy layout.
114
- const collections = ["one-shots", "ongoing", "habits", ""];
138
+ // Habits are no longer in tasks/ — they live at bundle root habits/.
139
+ const collections = ["one-shots", "ongoing", ""];
115
140
  for (const collection of collections) {
116
141
  try {
117
142
  return fs.readFileSync(path.join(agentRoot, "tasks", collection, `${taskId}.md`), "utf8").trim();
@@ -130,8 +155,9 @@ function buildTaskTriggeredMessage(taskId, taskContent, checkpoint) {
130
155
  else {
131
156
  lines.push("", `## task: ${taskId}`, "(task file not found)");
132
157
  }
133
- if (checkpoint) {
134
- lines.push("", `last i remember: ${checkpoint}`);
158
+ const renderedCheckpoint = displayCheckpoint(checkpoint);
159
+ if (renderedCheckpoint) {
160
+ lines.push("", `last checkpoint: ${renderedCheckpoint}`);
135
161
  }
136
162
  return lines.join("\n");
137
163
  }
@@ -219,6 +245,204 @@ function createInnerDialogCallbacks() {
219
245
  function innerDialogSessionPath() {
220
246
  return (0, config_1.sessionPath)(pending_1.INNER_DIALOG_PENDING.friendId, pending_1.INNER_DIALOG_PENDING.channel, pending_1.INNER_DIALOG_PENDING.key);
221
247
  }
248
+ function innerDialogRuntimeStatePath(sessionFilePath) {
249
+ return path.join(path.dirname(sessionFilePath), "runtime.json");
250
+ }
251
+ function writeInnerDialogRuntimeState(sessionFilePath, state) {
252
+ const filePath = innerDialogRuntimeStatePath(sessionFilePath);
253
+ try {
254
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
255
+ fs.writeFileSync(filePath, JSON.stringify(state, null, 2) + "\n", "utf8");
256
+ }
257
+ catch (error) {
258
+ (0, runtime_1.emitNervesEvent)({
259
+ level: "warn",
260
+ component: "senses",
261
+ event: "senses.inner_dialog_runtime_state_error",
262
+ message: "failed to write inner dialog runtime state",
263
+ meta: {
264
+ status: state.status,
265
+ reason: state.reason ?? null,
266
+ path: filePath,
267
+ /* v8 ignore next -- Node fs APIs throw Error objects for mkdirSync/writeFileSync failures @preserve */
268
+ error: error instanceof Error ? error.message : String(error),
269
+ },
270
+ });
271
+ }
272
+ }
273
+ /* v8 ignore start -- routing helpers: called from routing functions which are integration paths @preserve */
274
+ function writePendingEnvelope(pendingDir, message) {
275
+ fs.mkdirSync(pendingDir, { recursive: true });
276
+ const fileName = `${message.timestamp}-${Math.random().toString(36).slice(2, 10)}.json`;
277
+ const filePath = path.join(pendingDir, fileName);
278
+ fs.writeFileSync(filePath, JSON.stringify(message, null, 2), "utf8");
279
+ }
280
+ function sessionMatchesActivity(activity, session) {
281
+ return activity.friendId === session.friendId
282
+ && activity.channel === session.channel
283
+ && activity.key === session.key;
284
+ }
285
+ /* v8 ignore stop */
286
+ /* v8 ignore start -- routing: delivery now inline via surface tool; routing functions preserved for reuse @preserve */
287
+ function resolveBridgePreferredSession(delegatedFrom, sessionActivity) {
288
+ if (!delegatedFrom.bridgeId)
289
+ return null;
290
+ const bridge = (0, manager_1.createBridgeManager)().getBridge(delegatedFrom.bridgeId);
291
+ if (!bridge || bridge.lifecycle === "completed" || bridge.lifecycle === "cancelled") {
292
+ return null;
293
+ }
294
+ return sessionActivity.find((activity) => activity.friendId === delegatedFrom.friendId
295
+ && activity.channel !== "inner"
296
+ && bridge.attachedSessions.some((session) => sessionMatchesActivity(activity, session))) ?? null;
297
+ }
298
+ async function tryDeliverDelegatedCompletion(target, outboundEnvelope) {
299
+ if (target.channel !== "bluebubbles") {
300
+ return false;
301
+ }
302
+ const result = await (0, bluebubbles_1.sendProactiveBlueBubblesMessageToSession)({
303
+ friendId: target.friendId,
304
+ sessionKey: target.key,
305
+ text: outboundEnvelope.content,
306
+ intent: "explicit_cross_chat",
307
+ });
308
+ return result.delivered;
309
+ }
310
+ function enrichDelegatedFromWithBridge(delegatedFrom) {
311
+ if (delegatedFrom.bridgeId) {
312
+ return delegatedFrom;
313
+ }
314
+ const bridgeManager = (0, manager_1.createBridgeManager)();
315
+ const originBridges = bridgeManager.findBridgesForSession({
316
+ friendId: delegatedFrom.friendId,
317
+ channel: delegatedFrom.channel,
318
+ key: delegatedFrom.key,
319
+ });
320
+ const activeBridge = originBridges.find((b) => b.lifecycle === "active");
321
+ if (activeBridge) {
322
+ return { ...delegatedFrom, bridgeId: activeBridge.id };
323
+ }
324
+ return delegatedFrom;
325
+ }
326
+ function advanceObligationQuietly(agentName, obligationId, update) {
327
+ if (!obligationId)
328
+ return;
329
+ try {
330
+ (0, obligations_1.advanceReturnObligation)(agentName, obligationId, update);
331
+ /* v8 ignore start -- best-effort: obligation fs errors must never block return routing @preserve */
332
+ }
333
+ catch {
334
+ // swallowed
335
+ }
336
+ /* v8 ignore stop */
337
+ }
338
+ async function routeDelegatedCompletion(agentRoot, agentName, completion, drainedPending, timestamp) {
339
+ const delegated = (drainedPending ?? []).find((message) => message.delegatedFrom);
340
+ if (!delegated?.delegatedFrom || !completion?.answer?.trim()) {
341
+ return;
342
+ }
343
+ const delegatedFrom = enrichDelegatedFromWithBridge(delegated.delegatedFrom);
344
+ const obligationId = delegated.obligationId;
345
+ // Advance any inner return obligations from queued -> running (they were drained this turn).
346
+ // drainedPending is guaranteed non-null here (we found delegated above).
347
+ for (const msg of drainedPending) {
348
+ if (msg.obligationId) {
349
+ advanceObligationQuietly(agentName, msg.obligationId, {
350
+ status: "running",
351
+ startedAt: timestamp,
352
+ });
353
+ }
354
+ }
355
+ if (delegated.obligationStatus === "pending") {
356
+ // Fulfill the persistent obligation in the store
357
+ try {
358
+ const pending = (0, obligations_1.findPendingObligationForOrigin)(agentRoot, {
359
+ friendId: delegatedFrom.friendId,
360
+ channel: delegatedFrom.channel,
361
+ key: delegatedFrom.key,
362
+ });
363
+ /* v8 ignore next 2 -- obligation fulfillment tested via obligations.test.ts; integration requires real disk state @preserve */
364
+ if (pending) {
365
+ (0, obligations_1.fulfillObligation)(agentRoot, pending.id);
366
+ }
367
+ }
368
+ catch {
369
+ /* v8 ignore next -- defensive: obligation store read failure should not break delivery @preserve */
370
+ }
371
+ (0, runtime_1.emitNervesEvent)({
372
+ event: "senses.obligation_fulfilled",
373
+ component: "senses",
374
+ message: "obligation fulfilled via delegated completion",
375
+ meta: {
376
+ friendId: delegatedFrom.friendId,
377
+ channel: delegatedFrom.channel,
378
+ key: delegatedFrom.key,
379
+ },
380
+ });
381
+ }
382
+ const outboundEnvelope = {
383
+ from: agentName,
384
+ friendId: delegatedFrom.friendId,
385
+ channel: delegatedFrom.channel,
386
+ key: delegatedFrom.key,
387
+ content: completion.answer.trim(),
388
+ timestamp,
389
+ delegatedFrom,
390
+ ...(obligationId ? { obligationId } : {}),
391
+ };
392
+ const sessionActivity = (0, session_activity_1.listSessionActivity)({
393
+ sessionsDir: path.join(agentRoot, "state", "sessions"),
394
+ friendsDir: path.join(agentRoot, "friends"),
395
+ agentName,
396
+ });
397
+ // Priority 1: Bridge-preferred session (if delegation was within a bridge).
398
+ const bridgeTarget = resolveBridgePreferredSession(delegatedFrom, sessionActivity);
399
+ if (bridgeTarget) {
400
+ if (await tryDeliverDelegatedCompletion(bridgeTarget, outboundEnvelope)) {
401
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "bridge-session" });
402
+ return;
403
+ }
404
+ writePendingEnvelope((0, pending_1.getPendingDir)(agentName, bridgeTarget.friendId, bridgeTarget.channel, bridgeTarget.key), outboundEnvelope);
405
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "bridge-session" });
406
+ return;
407
+ }
408
+ // Priority 1.5: Direct return to originating session (ponder without bridge).
409
+ // When delegatedFrom has specific channel+key, route directly there instead of searching for freshest.
410
+ if (delegatedFrom.channel && delegatedFrom.key && delegatedFrom.channel !== "inner") {
411
+ const directTarget = sessionActivity.find((a) => a.friendId === delegatedFrom.friendId && a.channel === delegatedFrom.channel && a.key === delegatedFrom.key);
412
+ if (directTarget) {
413
+ if (await tryDeliverDelegatedCompletion(directTarget, outboundEnvelope)) {
414
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "direct-originator" });
415
+ return;
416
+ }
417
+ }
418
+ // Even if session isn't in activity list (might have ended), queue to its pending dir
419
+ writePendingEnvelope((0, pending_1.getPendingDir)(agentName, delegatedFrom.friendId, delegatedFrom.channel, delegatedFrom.key), outboundEnvelope);
420
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "direct-originator" });
421
+ return;
422
+ }
423
+ // Priority 2: Freshest active friend session.
424
+ // For BB, prefer DM sessions (;-;) over group chats (;+;) — proactive outreach should never land in groups.
425
+ const allFriendSessions = (0, session_activity_1.listSessionActivity)({
426
+ sessionsDir: path.join(agentRoot, "state", "sessions"),
427
+ friendsDir: path.join(agentRoot, "friends"),
428
+ agentName,
429
+ }).filter((s) => s.friendId === delegatedFrom.friendId && s.channel !== "inner");
430
+ const bbDm = allFriendSessions.find((s) => s.channel === "bluebubbles" && s.key.includes(";-;"));
431
+ const freshest = bbDm ?? allFriendSessions.find((s) => s.channel !== "bluebubbles" || s.key.includes(";-;")) ?? allFriendSessions[0];
432
+ if (freshest) {
433
+ if (await tryDeliverDelegatedCompletion(freshest, outboundEnvelope)) {
434
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "freshest-session" });
435
+ return;
436
+ }
437
+ writePendingEnvelope((0, pending_1.getPendingDir)(agentName, freshest.friendId, freshest.channel, freshest.key), outboundEnvelope);
438
+ advanceObligationQuietly(agentName, obligationId, { status: "returned", returnedAt: timestamp, returnTarget: "freshest-session" });
439
+ return;
440
+ }
441
+ // Priority 3: Deferred return queue.
442
+ writePendingEnvelope((0, pending_1.getDeferredReturnDir)(agentName, delegatedFrom.friendId), outboundEnvelope);
443
+ advanceObligationQuietly(agentName, obligationId, { status: "deferred", returnedAt: timestamp, returnTarget: "deferred" });
444
+ }
445
+ /* v8 ignore stop */
222
446
  // Self-referencing friend record for inner dialog (agent talking to itself).
223
447
  // No real friend to resolve -- this satisfies the pipeline's friend resolver contract.
224
448
  function createSelfFriend(agentName) {
@@ -245,110 +469,314 @@ function createNoOpFriendStore() {
245
469
  findByExternalId: async () => null,
246
470
  };
247
471
  }
472
+ function buildParseErrorNudge(parseErrors) {
473
+ if (parseErrors.length === 0)
474
+ return "";
475
+ const lines = parseErrors.map((e) => `I noticed my habit file \`${e.file}\` has invalid frontmatter — I should fix it. (${e.error})`);
476
+ return lines.join("\n");
477
+ }
478
+ function buildAlsoDueLine(agentRoot, currentHabitName, now) {
479
+ const habitsDir = path.join(agentRoot, "habits");
480
+ let files;
481
+ try {
482
+ files = fs.readdirSync(habitsDir);
483
+ }
484
+ catch {
485
+ return "";
486
+ }
487
+ const nowMs = now().getTime();
488
+ const alsoDue = [];
489
+ for (const file of files) {
490
+ if (!file.endsWith(".md"))
491
+ continue;
492
+ const stem = file.replace(/\.md$/, "");
493
+ if (stem === currentHabitName)
494
+ continue;
495
+ try {
496
+ const content = fs.readFileSync(path.join(habitsDir, file), "utf-8");
497
+ const habit = (0, habit_runtime_state_1.applyHabitRuntimeState)(agentRoot, (0, habit_parser_1.parseHabitFile)(content, path.join(habitsDir, file)));
498
+ if (habit.status !== "active" || !habit.cadence)
499
+ continue;
500
+ const cadenceMs = (0, cadence_1.parseCadenceToMs)(habit.cadence);
501
+ if (cadenceMs === null)
502
+ continue;
503
+ if (habit.lastRun === null) {
504
+ alsoDue.push(stem);
505
+ continue;
506
+ }
507
+ const lastRunMs = new Date(habit.lastRun).getTime();
508
+ if (nowMs - lastRunMs >= cadenceMs) {
509
+ alsoDue.push(stem);
510
+ }
511
+ }
512
+ catch {
513
+ // skip unreadable habits
514
+ }
515
+ }
516
+ if (alsoDue.length === 0)
517
+ return "";
518
+ return `also due: ${alsoDue.join(", ")}`;
519
+ }
248
520
  async function runInnerDialogTurn(options) {
249
521
  const now = options?.now ?? (() => new Date());
250
- const reason = options?.reason ?? "heartbeat";
522
+ const reason = options?.reason ?? "instinct";
251
523
  const sessionFilePath = innerDialogSessionPath();
252
- const loaded = (0, context_1.loadSession)(sessionFilePath);
253
- const existingMessages = loaded?.messages ? [...loaded.messages] : [];
254
- const instincts = options?.instincts ?? loadInnerDialogInstincts();
255
- const state = {
256
- cycleCount: 1,
257
- resting: false,
258
- lastHeartbeatAt: now().toISOString(),
259
- };
260
- // ── Adapter concern: build user message ──────────────────────────
261
- let userContent;
262
- if (existingMessages.length === 0) {
263
- // Fresh session: bootstrap message with non-canonical cleanup nudge
264
- const aspirations = readAspirations((0, identity_1.getAgentRoot)());
265
- const nonCanonical = (0, bundle_manifest_1.findNonCanonicalBundlePaths)((0, identity_1.getAgentRoot)());
266
- const cleanupNudge = buildNonCanonicalCleanupNudge(nonCanonical);
267
- userContent = [
268
- buildInnerDialogBootstrapMessage(aspirations, "No prior inner dialog session found."),
269
- cleanupNudge,
270
- ].filter(Boolean).join("\n\n");
271
- }
272
- else {
273
- // Resumed session: task-triggered or instinct message with checkpoint context
274
- const assistantTurns = existingMessages.filter((message) => message.role === "assistant").length;
275
- state.cycleCount = assistantTurns + 1;
276
- state.checkpoint = deriveResumeCheckpoint(existingMessages);
277
- if (options?.taskId) {
278
- const taskContent = readTaskFile((0, identity_1.getAgentRoot)(), options.taskId);
279
- userContent = buildTaskTriggeredMessage(options.taskId, taskContent, state.checkpoint);
524
+ const agentName = (0, identity_1.getAgentName)();
525
+ writeInnerDialogRuntimeState(sessionFilePath, {
526
+ status: "running",
527
+ reason,
528
+ startedAt: now().toISOString(),
529
+ });
530
+ try {
531
+ const loaded = (0, context_1.loadSession)(sessionFilePath);
532
+ const existingMessages = loaded?.messages ? [...loaded.messages] : [];
533
+ const instincts = options?.instincts ?? loadInnerDialogInstincts();
534
+ const state = {
535
+ cycleCount: 1,
536
+ resting: false,
537
+ lastHeartbeatAt: now().toISOString(),
538
+ };
539
+ // ── Adapter concern: build user message ──────────────────────────
540
+ let userContent;
541
+ let habitTools;
542
+ let habitParsedSuccessfully = false;
543
+ if (existingMessages.length === 0) {
544
+ // Fresh session: bootstrap message with non-canonical cleanup nudge
545
+ const aspirations = readAspirations((0, identity_1.getAgentRoot)());
546
+ const nonCanonical = (0, bundle_manifest_1.findNonCanonicalBundlePaths)((0, identity_1.getAgentRoot)());
547
+ const cleanupNudge = buildNonCanonicalCleanupNudge(nonCanonical);
548
+ userContent = [
549
+ buildInnerDialogBootstrapMessage(aspirations, "No prior inner dialog session found."),
550
+ cleanupNudge,
551
+ ].filter(Boolean).join("\n\n");
280
552
  }
281
553
  else {
282
- userContent = buildInstinctUserMessage(instincts, reason, state);
554
+ // Resumed session: task-triggered or instinct message with checkpoint context
555
+ const assistantTurns = existingMessages.filter((message) => message.role === "assistant").length;
556
+ state.cycleCount = assistantTurns + 1;
557
+ state.checkpoint = deriveResumeCheckpoint(existingMessages);
558
+ if (options?.taskId) {
559
+ const taskContent = readTaskFile((0, identity_1.getAgentRoot)(), options.taskId);
560
+ userContent = buildTaskTriggeredMessage(options.taskId, taskContent, state.checkpoint);
561
+ }
562
+ else if (reason === "habit" && options?.habitName) {
563
+ const agentRoot = (0, identity_1.getAgentRoot)();
564
+ const habitName = options.habitName;
565
+ const habitFilePath = path.join(agentRoot, "habits", `${habitName}.md`);
566
+ // Read and parse the habit file
567
+ let habitBody;
568
+ let habitTitle = habitName;
569
+ let habitLastRun = null;
570
+ try {
571
+ const habitContent = fs.readFileSync(habitFilePath, "utf-8");
572
+ const parsed = (0, habit_runtime_state_1.applyHabitRuntimeState)(agentRoot, (0, habit_parser_1.parseHabitFile)(habitContent, habitFilePath));
573
+ habitBody = parsed.body || undefined;
574
+ habitTitle = parsed.title || habitName;
575
+ habitLastRun = parsed.lastRun;
576
+ habitTools = parsed.tools;
577
+ }
578
+ catch {
579
+ // Habit file missing or unreadable
580
+ }
581
+ // If the habit file couldn't be read at all (no body, no title parsed), error message
582
+ if (habitBody === undefined && habitTitle === habitName) {
583
+ userContent = `habit "${habitName}" could not be read (file not found or unreadable). check habits/${habitName}.md exists.`;
584
+ }
585
+ else {
586
+ habitParsedSuccessfully = true;
587
+ // Unified path: gather context for ALL habits (heartbeat included)
588
+ const obligations = (0, obligations_1.listActiveReturnObligations)(agentName);
589
+ const nowMs = now().getTime();
590
+ const staleObligations = obligations.map((o) => ({
591
+ friendName: o.origin.friendId,
592
+ content: o.delegatedContent,
593
+ stalenessMs: nowMs - o.createdAt,
594
+ }));
595
+ const alsoDue = buildAlsoDueLine(agentRoot, habitName, now);
596
+ // Degraded state (best-effort: never crash)
597
+ let degradedComponents = [];
598
+ try {
599
+ const health = (0, daemon_health_1.readHealth)((0, daemon_health_1.getDefaultHealthPath)());
600
+ if (health && health.degraded.length > 0) {
601
+ degradedComponents = health.degraded.map((d) => ({ component: d.component, reason: d.reason }));
602
+ }
603
+ }
604
+ catch {
605
+ // Best-effort: missing file or parse error -> empty array, no crash
606
+ }
607
+ userContent = (0, habit_turn_message_1.buildHabitTurnMessage)({
608
+ habitName,
609
+ habitTitle,
610
+ habitBody,
611
+ lastRun: habitLastRun,
612
+ checkpoint: displayCheckpoint(state.checkpoint),
613
+ alsoDue: alsoDue || undefined,
614
+ staleObligations,
615
+ parseErrors: options?.parseErrors ?? [],
616
+ degradedComponents,
617
+ now,
618
+ });
619
+ // Piggyback journal embedding indexing (best-effort, fire-and-forget)
620
+ const journalDir = path.join(agentRoot, "journal");
621
+ /* v8 ignore start -- journal indexing piggyback: embedding provider may not be available; tested via journal-index unit tests @preserve */
622
+ void (0, journal_index_1.indexJournalFiles)(journalDir, path.join(journalDir, ".index.json"), {
623
+ embed: async () => [],
624
+ }).catch(() => {
625
+ // swallowed: indexing failure must never block habit turn
626
+ });
627
+ /* v8 ignore stop */
628
+ }
629
+ }
630
+ else {
631
+ userContent = buildInstinctUserMessage(instincts, reason, state);
632
+ }
633
+ }
634
+ const userMessage = { role: "user", content: userContent };
635
+ // ── Session loader: wraps existing session logic ──────────────────
636
+ const innerCapabilities = (0, channel_1.getChannelCapabilities)("inner");
637
+ const pendingDir = (0, pending_1.getInnerDialogPendingDir)(agentName);
638
+ const selfFriend = createSelfFriend(agentName);
639
+ const selfContext = { friend: selfFriend, channel: innerCapabilities };
640
+ const mcpManager = await (0, mcp_manager_1.getSharedMcpManager)() ?? undefined;
641
+ // ── Habit tool enforcement ───────────────────────────────────────
642
+ let habitToolsResolved;
643
+ if (habitTools !== undefined) {
644
+ const fullTools = (0, tools_1.getToolsForChannel)(innerCapabilities);
645
+ habitToolsResolved = fullTools.filter((t) => habitTools.includes(t.function.name));
646
+ (0, runtime_1.emitNervesEvent)({
647
+ event: "habit.tools_restricted",
648
+ component: "senses",
649
+ message: "habit running with restricted tools",
650
+ meta: {
651
+ habitName: options?.habitName,
652
+ declared: habitTools,
653
+ resolved: habitToolsResolved.map((t) => t.function.name),
654
+ },
655
+ });
283
656
  }
657
+ else if (reason === "habit" && options?.habitName && habitParsedSuccessfully) {
658
+ (0, runtime_1.emitNervesEvent)({
659
+ event: "habit.tools_unrestricted",
660
+ component: "senses",
661
+ message: "habit running with full tool repertoire",
662
+ meta: { habitName: options.habitName },
663
+ });
664
+ }
665
+ const sessionLoader = {
666
+ loadOrCreate: async () => {
667
+ if (existingMessages.length > 0) {
668
+ return {
669
+ messages: existingMessages,
670
+ sessionPath: sessionFilePath,
671
+ };
672
+ }
673
+ // Fresh session: build system prompt
674
+ const systemPrompt = await (0, prompt_1.buildSystem)("inner", { toolChoiceRequired: true });
675
+ return {
676
+ messages: [{ role: "system", content: (0, prompt_1.flattenSystemPrompt)(systemPrompt) }],
677
+ sessionPath: sessionFilePath,
678
+ };
679
+ },
680
+ };
681
+ // ── Call shared pipeline ──────────────────────────────────────────
682
+ const callbacks = createInnerDialogCallbacks();
683
+ const traceId = (0, nerves_1.createTraceId)();
684
+ // Attention queue: built when pending messages are drained, shared with tool context
685
+ let attentionQueue = [];
686
+ const result = await (0, pipeline_1.handleInboundTurn)({
687
+ channel: "inner",
688
+ sessionKey: "dialog",
689
+ capabilities: innerCapabilities,
690
+ messages: [userMessage],
691
+ continuityIngressTexts: [],
692
+ callbacks,
693
+ friendResolver: { resolve: () => Promise.resolve(selfContext) },
694
+ sessionLoader,
695
+ pendingDir,
696
+ friendStore: createNoOpFriendStore(),
697
+ enforceTrustGate: trust_gate_1.enforceTrustGate,
698
+ drainPending: pending_1.drainPending,
699
+ runAgent: core_1.runAgent,
700
+ postTurn: (turnMessages, sessionPathArg, usage, hooks, state) => {
701
+ const prepared = (0, context_1.postTurnTrim)(turnMessages, usage, hooks);
702
+ (0, context_1.deferPostTurnPersist)(sessionPathArg, prepared, usage, state);
703
+ },
704
+ accumulateFriendTokens: tokens_1.accumulateFriendTokens,
705
+ signal: options?.signal,
706
+ /* v8 ignore start -- attention queue: callback invoked by pipeline during pending drain; tested via attention-queue unit tests @preserve */
707
+ onPendingDrained: (drained) => {
708
+ const outstandingObligations = (0, obligations_1.listActiveReturnObligations)(agentName);
709
+ attentionQueue = (0, attention_queue_1.buildAttentionQueue)({
710
+ drainedPending: drained,
711
+ outstandingObligations,
712
+ friendNameResolver: (friendId) => {
713
+ try {
714
+ const raw = fs.readFileSync(path.join((0, identity_1.getAgentRoot)(agentName), "friends", friendId + ".json"), "utf-8");
715
+ const parsed = JSON.parse(raw);
716
+ return typeof parsed.name === "string" ? parsed.name : null;
717
+ }
718
+ catch {
719
+ return null;
720
+ }
721
+ },
722
+ packetResolver: (packetId) => {
723
+ try {
724
+ return (0, packets_1.readPonderPacket)((0, identity_1.getAgentRoot)(agentName), packetId);
725
+ }
726
+ catch {
727
+ return null;
728
+ }
729
+ },
730
+ });
731
+ const summary = (0, attention_queue_1.buildAttentionQueueSummary)(attentionQueue);
732
+ return summary ? [summary] : [];
733
+ },
734
+ /* v8 ignore stop */
735
+ runAgentOptions: {
736
+ traceId,
737
+ toolChoiceRequired: true,
738
+ mcpManager,
739
+ ...(habitToolsResolved !== undefined && { tools: habitToolsResolved }),
740
+ toolContext: {
741
+ signin: async () => undefined,
742
+ delegatedOrigins: attentionQueue,
743
+ },
744
+ },
745
+ });
746
+ // Post-turn routeDelegatedCompletion removed: delivery is now inline via surface tool.
747
+ // settle in inner dialog produces no CompletionMetadata, so routeDelegatedCompletion
748
+ // would be a no-op. The routing infrastructure is reused by the surface handler.
749
+ const resultMessages = result.messages ?? [];
750
+ const assistantPreview = extractAssistantPreview(resultMessages);
751
+ const toolCalls = extractToolCallNames(resultMessages);
752
+ (0, runtime_1.emitNervesEvent)({
753
+ component: "senses",
754
+ event: "senses.inner_dialog_turn",
755
+ message: "inner dialog turn completed",
756
+ meta: {
757
+ reason,
758
+ session: sessionFilePath,
759
+ ...(options?.taskId && { taskId: options.taskId }),
760
+ ...(assistantPreview && { assistantPreview }),
761
+ ...(toolCalls.length > 0 && { toolCalls }),
762
+ ...(result.usage && {
763
+ promptTokens: result.usage.input_tokens,
764
+ completionTokens: result.usage.output_tokens,
765
+ totalTokens: result.usage.total_tokens,
766
+ }),
767
+ },
768
+ });
769
+ return {
770
+ messages: resultMessages,
771
+ usage: result.usage,
772
+ sessionPath: result.sessionPath ?? sessionFilePath,
773
+ completion: result.completion,
774
+ };
775
+ }
776
+ finally {
777
+ writeInnerDialogRuntimeState(sessionFilePath, {
778
+ status: "idle",
779
+ lastCompletedAt: now().toISOString(),
780
+ });
284
781
  }
285
- const userMessage = { role: "user", content: userContent };
286
- // ── Session loader: wraps existing session logic ──────────────────
287
- const innerCapabilities = (0, channel_1.getChannelCapabilities)("inner");
288
- const pendingDir = (0, pending_1.getInnerDialogPendingDir)((0, identity_1.getAgentName)());
289
- const selfFriend = createSelfFriend((0, identity_1.getAgentName)());
290
- const selfContext = { friend: selfFriend, channel: innerCapabilities };
291
- const sessionLoader = {
292
- loadOrCreate: async () => {
293
- if (existingMessages.length > 0) {
294
- return { messages: existingMessages, sessionPath: sessionFilePath };
295
- }
296
- // Fresh session: build system prompt
297
- const systemPrompt = await (0, prompt_1.buildSystem)("inner", { toolChoiceRequired: true });
298
- return {
299
- messages: [{ role: "system", content: systemPrompt }],
300
- sessionPath: sessionFilePath,
301
- };
302
- },
303
- };
304
- // ── Call shared pipeline ──────────────────────────────────────────
305
- const callbacks = createInnerDialogCallbacks();
306
- const traceId = (0, nerves_1.createTraceId)();
307
- const result = await (0, pipeline_1.handleInboundTurn)({
308
- channel: "inner",
309
- capabilities: innerCapabilities,
310
- messages: [userMessage],
311
- continuityIngressTexts: [],
312
- callbacks,
313
- friendResolver: { resolve: () => Promise.resolve(selfContext) },
314
- sessionLoader,
315
- pendingDir,
316
- friendStore: createNoOpFriendStore(),
317
- enforceTrustGate: trust_gate_1.enforceTrustGate,
318
- drainPending: pending_1.drainPending,
319
- runAgent: core_1.runAgent,
320
- postTurn: context_1.postTurn,
321
- accumulateFriendTokens: tokens_1.accumulateFriendTokens,
322
- signal: options?.signal,
323
- runAgentOptions: {
324
- traceId,
325
- toolChoiceRequired: true,
326
- skipConfirmation: true,
327
- },
328
- });
329
- const resultMessages = result.messages ?? [];
330
- const assistantPreview = extractAssistantPreview(resultMessages);
331
- const toolCalls = extractToolCallNames(resultMessages);
332
- (0, runtime_1.emitNervesEvent)({
333
- component: "senses",
334
- event: "senses.inner_dialog_turn",
335
- message: "inner dialog turn completed",
336
- meta: {
337
- reason,
338
- session: sessionFilePath,
339
- ...(options?.taskId && { taskId: options.taskId }),
340
- ...(assistantPreview && { assistantPreview }),
341
- ...(toolCalls.length > 0 && { toolCalls }),
342
- ...(result.usage && {
343
- promptTokens: result.usage.input_tokens,
344
- completionTokens: result.usage.output_tokens,
345
- totalTokens: result.usage.total_tokens,
346
- }),
347
- },
348
- });
349
- return {
350
- messages: resultMessages,
351
- usage: result.usage,
352
- sessionPath: result.sessionPath ?? sessionFilePath,
353
- };
354
782
  }