@ouro.bot/cli 0.1.0-alpha.6 → 0.1.0-alpha.600

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