@ouro.bot/cli 0.1.0-alpha.61 → 0.1.0-alpha.611

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 (416) hide show
  1. package/README.md +127 -23
  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/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  10. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  12. package/changelog.json +3902 -0
  13. package/dist/arc/attention-types.js +8 -0
  14. package/dist/arc/cares.js +140 -0
  15. package/dist/arc/episodes.js +117 -0
  16. package/dist/arc/intentions.js +133 -0
  17. package/dist/arc/json-store.js +117 -0
  18. package/dist/arc/obligations.js +254 -0
  19. package/dist/arc/packets.js +193 -0
  20. package/dist/arc/presence.js +185 -0
  21. package/dist/arc/task-lifecycle.js +65 -0
  22. package/dist/heart/active-work.js +837 -26
  23. package/dist/heart/agent-entry.js +69 -3
  24. package/dist/heart/attachments/image-normalize.js +194 -0
  25. package/dist/heart/attachments/materialize.js +97 -0
  26. package/dist/heart/attachments/originals.js +88 -0
  27. package/dist/heart/attachments/render.js +29 -0
  28. package/dist/heart/attachments/sources/adapter.js +2 -0
  29. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  30. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  31. package/dist/heart/attachments/sources/index.js +16 -0
  32. package/dist/heart/attachments/store.js +103 -0
  33. package/dist/heart/attachments/types.js +93 -0
  34. package/dist/heart/auth/auth-flow.js +479 -0
  35. package/dist/heart/awaiting/await-alert.js +146 -0
  36. package/dist/heart/awaiting/await-expiry.js +108 -0
  37. package/dist/heart/awaiting/await-loader.js +91 -0
  38. package/dist/heart/awaiting/await-parser.js +141 -0
  39. package/dist/heart/awaiting/await-runtime-state.js +97 -0
  40. package/dist/heart/awaiting/await-scheduler.js +377 -0
  41. package/dist/heart/background-operations.js +281 -0
  42. package/dist/heart/bundle-state.js +168 -0
  43. package/dist/heart/commitments.js +142 -0
  44. package/dist/heart/config-registry.js +322 -0
  45. package/dist/heart/config.js +114 -119
  46. package/dist/heart/core.js +909 -246
  47. package/dist/heart/cross-chat-delivery.js +3 -18
  48. package/dist/heart/daemon/agent-config-check.js +419 -0
  49. package/dist/heart/daemon/agent-discovery.js +102 -3
  50. package/dist/heart/daemon/agent-service.js +522 -0
  51. package/dist/heart/daemon/agentic-repair.js +547 -0
  52. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  53. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  54. package/dist/heart/daemon/cadence.js +70 -0
  55. package/dist/heart/daemon/cli-defaults.js +776 -0
  56. package/dist/heart/daemon/cli-exec.js +7579 -0
  57. package/dist/heart/daemon/cli-help.js +498 -0
  58. package/dist/heart/daemon/cli-parse.js +1599 -0
  59. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  60. package/dist/heart/daemon/cli-render.js +763 -0
  61. package/dist/heart/daemon/cli-types.js +8 -0
  62. package/dist/heart/daemon/connect-bay.js +323 -0
  63. package/dist/heart/daemon/daemon-cli.js +29 -1700
  64. package/dist/heart/daemon/daemon-entry.js +485 -2
  65. package/dist/heart/daemon/daemon-health.js +176 -0
  66. package/dist/heart/daemon/daemon-rollup.js +57 -0
  67. package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
  68. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  69. package/dist/heart/daemon/daemon.js +857 -70
  70. package/dist/heart/daemon/dns-workflow.js +394 -0
  71. package/dist/heart/daemon/doctor-types.js +8 -0
  72. package/dist/heart/daemon/doctor.js +873 -0
  73. package/dist/heart/daemon/health-monitor.js +122 -1
  74. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  75. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  76. package/dist/heart/daemon/http-health-probe.js +80 -0
  77. package/dist/heart/daemon/human-command-screens.js +234 -0
  78. package/dist/heart/daemon/human-readiness.js +114 -0
  79. package/dist/heart/daemon/inner-status.js +89 -0
  80. package/dist/heart/daemon/interactive-repair.js +394 -0
  81. package/dist/heart/daemon/launchd.js +37 -8
  82. package/dist/heart/daemon/log-tailer.js +78 -9
  83. package/dist/heart/daemon/logs-prune.js +110 -0
  84. package/dist/heart/daemon/mcp-canary.js +297 -0
  85. package/dist/heart/daemon/os-cron-deps.js +135 -0
  86. package/dist/heart/daemon/os-cron.js +14 -12
  87. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  88. package/dist/heart/daemon/ouro-entry.js +3 -1
  89. package/dist/heart/daemon/process-manager.js +450 -34
  90. package/dist/heart/daemon/provider-discovery.js +137 -0
  91. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  92. package/dist/heart/daemon/pulse.js +475 -0
  93. package/dist/heart/daemon/readiness-repair.js +365 -0
  94. package/dist/heart/daemon/run-hooks.js +2 -0
  95. package/dist/heart/daemon/runtime-logging.js +10 -2
  96. package/dist/heart/daemon/runtime-metadata.js +2 -30
  97. package/dist/heart/daemon/safe-mode.js +161 -0
  98. package/dist/heart/daemon/sense-manager.js +481 -38
  99. package/dist/heart/daemon/session-id-resolver.js +131 -0
  100. package/dist/heart/daemon/skill-management-installer.js +28 -7
  101. package/dist/heart/daemon/socket-client.js +158 -11
  102. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  103. package/dist/heart/daemon/startup-tui.js +330 -0
  104. package/dist/heart/daemon/task-scheduler.js +3 -25
  105. package/dist/heart/daemon/terminal-ui.js +499 -0
  106. package/dist/heart/daemon/thoughts.js +229 -17
  107. package/dist/heart/daemon/up-progress.js +366 -0
  108. package/dist/heart/daemon/vault-items.js +56 -0
  109. package/dist/heart/delegation.js +1 -1
  110. package/dist/heart/habits/habit-migration.js +189 -0
  111. package/dist/heart/habits/habit-parser.js +140 -0
  112. package/dist/heart/habits/habit-runtime-state.js +100 -0
  113. package/dist/heart/habits/habit-scheduler.js +372 -0
  114. package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -56
  115. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  116. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  117. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  118. package/dist/heart/identity.js +166 -55
  119. package/dist/heart/kept-notes.js +357 -0
  120. package/dist/heart/kicks.js +1 -1
  121. package/dist/heart/machine-identity.js +161 -0
  122. package/dist/heart/mail-import-discovery.js +353 -0
  123. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  124. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  125. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  126. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  127. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  128. package/dist/heart/mailbox/mailbox-http.js +99 -0
  129. package/dist/heart/mailbox/mailbox-read.js +31 -0
  130. package/dist/heart/mailbox/mailbox-types.js +27 -0
  131. package/dist/heart/mailbox/mailbox-view.js +197 -0
  132. package/dist/heart/mailbox/readers/agent-machine.js +425 -0
  133. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  134. package/dist/heart/mailbox/readers/mail.js +375 -0
  135. package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
  136. package/dist/heart/mailbox/readers/sessions.js +232 -0
  137. package/dist/heart/mailbox/readers/shared.js +111 -0
  138. package/dist/heart/mcp/mcp-server.js +656 -0
  139. package/dist/heart/migrate-config.js +100 -0
  140. package/dist/heart/model-capabilities.js +19 -0
  141. package/dist/heart/platform.js +81 -0
  142. package/dist/heart/provider-attempt.js +134 -0
  143. package/dist/heart/provider-binding-resolver.js +267 -0
  144. package/dist/heart/provider-credentials.js +425 -0
  145. package/dist/heart/provider-failover.js +301 -0
  146. package/dist/heart/provider-models.js +81 -0
  147. package/dist/heart/provider-ping.js +262 -0
  148. package/dist/heart/provider-readiness-cache.js +40 -0
  149. package/dist/heart/provider-visibility.js +188 -0
  150. package/dist/heart/providers/anthropic-token.js +131 -0
  151. package/dist/heart/providers/anthropic.js +139 -52
  152. package/dist/heart/providers/azure.js +23 -11
  153. package/dist/heart/providers/error-classification.js +127 -0
  154. package/dist/heart/providers/github-copilot.js +145 -0
  155. package/dist/heart/providers/minimax-vlm.js +189 -0
  156. package/dist/heart/providers/minimax.js +26 -8
  157. package/dist/heart/providers/openai-codex.js +55 -40
  158. package/dist/heart/runtime-capability-check.js +170 -0
  159. package/dist/heart/runtime-credentials.js +367 -0
  160. package/dist/heart/runtime-cwd.js +87 -0
  161. package/dist/heart/sense-truth.js +13 -4
  162. package/dist/heart/session-activity.js +48 -24
  163. package/dist/heart/session-events.js +1149 -0
  164. package/dist/heart/session-playback-cli-main.js +5 -0
  165. package/dist/heart/session-playback-cli.js +36 -0
  166. package/dist/heart/session-playback.js +231 -0
  167. package/dist/heart/session-stats-cli-main.js +5 -0
  168. package/dist/heart/session-stats.js +182 -0
  169. package/dist/heart/session-transcript.js +243 -0
  170. package/dist/heart/start-of-turn-packet.js +345 -0
  171. package/dist/heart/streaming.js +44 -27
  172. package/dist/heart/sync-classification.js +176 -0
  173. package/dist/heart/sync.js +449 -0
  174. package/dist/heart/target-resolution.js +9 -5
  175. package/dist/heart/tempo.js +93 -0
  176. package/dist/heart/temporal-view.js +41 -0
  177. package/dist/heart/timeouts.js +101 -0
  178. package/dist/heart/tool-activity-callbacks.js +59 -0
  179. package/dist/heart/tool-description.js +143 -0
  180. package/dist/heart/tool-friction.js +55 -0
  181. package/dist/heart/tool-loop.js +200 -0
  182. package/dist/heart/turn-context.js +421 -0
  183. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  184. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  185. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  186. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  187. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  188. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  189. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  190. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  191. package/dist/mailbox-ui/assets/index-CtUWEo-S.js +61 -0
  192. package/dist/mailbox-ui/index.html +15 -0
  193. package/dist/mailroom/attention.js +167 -0
  194. package/dist/mailroom/autonomy.js +209 -0
  195. package/dist/mailroom/blob-store.js +712 -0
  196. package/dist/mailroom/body-cache.js +61 -0
  197. package/dist/mailroom/core.js +788 -0
  198. package/dist/mailroom/entry.js +160 -0
  199. package/dist/mailroom/file-store.js +460 -0
  200. package/dist/mailroom/mbox-import.js +393 -0
  201. package/dist/mailroom/migration.js +164 -0
  202. package/dist/mailroom/outbound.js +380 -0
  203. package/dist/mailroom/policy.js +263 -0
  204. package/dist/mailroom/reader.js +233 -0
  205. package/dist/mailroom/search-cache.js +268 -0
  206. package/dist/mailroom/search-relevance.js +319 -0
  207. package/dist/mailroom/smtp-ingress.js +176 -0
  208. package/dist/mailroom/source-state.js +176 -0
  209. package/dist/mailroom/thread.js +109 -0
  210. package/dist/mailroom/travel-extract.js +89 -0
  211. package/dist/mind/bundle-manifest.js +7 -1
  212. package/dist/mind/context.js +250 -101
  213. package/dist/mind/diary-integrity.js +60 -0
  214. package/dist/mind/{memory.js → diary.js} +62 -75
  215. package/dist/mind/embedding-provider.js +60 -0
  216. package/dist/mind/file-state.js +179 -0
  217. package/dist/mind/friends/channel.js +39 -0
  218. package/dist/mind/friends/resolver.js +54 -2
  219. package/dist/mind/friends/store-file.js +39 -3
  220. package/dist/mind/friends/types.js +2 -2
  221. package/dist/mind/journal-index.js +161 -0
  222. package/dist/mind/note-search.js +268 -0
  223. package/dist/mind/obligation-steering.js +221 -0
  224. package/dist/mind/pending.js +4 -0
  225. package/dist/mind/prompt-refresh.js +3 -2
  226. package/dist/mind/prompt.js +1050 -135
  227. package/dist/mind/provenance-trust.js +26 -0
  228. package/dist/mind/scrutiny.js +173 -0
  229. package/dist/nerves/cli-logging.js +7 -1
  230. package/dist/nerves/coverage/audit-rules.js +15 -6
  231. package/dist/nerves/coverage/audit.js +28 -2
  232. package/dist/nerves/coverage/cli.js +1 -1
  233. package/dist/nerves/coverage/contract.js +5 -5
  234. package/dist/nerves/coverage/file-completeness.js +129 -5
  235. package/dist/nerves/event-buffer.js +111 -0
  236. package/dist/nerves/index.js +224 -4
  237. package/dist/nerves/observation.js +20 -0
  238. package/dist/nerves/redact.js +79 -0
  239. package/dist/nerves/review/cli-main.js +5 -0
  240. package/dist/nerves/review/cli.js +156 -0
  241. package/dist/nerves/review/core.js +152 -0
  242. package/dist/nerves/runtime.js +5 -1
  243. package/dist/repertoire/ado-client.js +15 -56
  244. package/dist/repertoire/ado-semantic.js +11 -10
  245. package/dist/repertoire/api-client.js +97 -0
  246. package/dist/repertoire/bitwarden-store.js +997 -0
  247. package/dist/repertoire/bundle-templates.js +72 -0
  248. package/dist/repertoire/bw-installer.js +180 -0
  249. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  250. package/dist/repertoire/coding/context-pack.js +330 -0
  251. package/dist/repertoire/coding/feedback.js +197 -30
  252. package/dist/repertoire/coding/manager.js +158 -9
  253. package/dist/repertoire/coding/spawner.js +55 -9
  254. package/dist/repertoire/coding/tools.js +170 -7
  255. package/dist/repertoire/commerce-errors.js +109 -0
  256. package/dist/repertoire/commerce-self-test.js +156 -0
  257. package/dist/repertoire/credential-access.js +178 -0
  258. package/dist/repertoire/duffel-client.js +185 -0
  259. package/dist/repertoire/github-client.js +14 -55
  260. package/dist/repertoire/graph-client.js +11 -52
  261. package/dist/repertoire/guardrails.js +396 -0
  262. package/dist/repertoire/mcp-client.js +295 -0
  263. package/dist/repertoire/mcp-manager.js +362 -0
  264. package/dist/repertoire/mcp-tools.js +63 -0
  265. package/dist/repertoire/shell-sessions.js +133 -0
  266. package/dist/repertoire/skills.js +18 -4
  267. package/dist/repertoire/stripe-client.js +131 -0
  268. package/dist/repertoire/tasks/board.js +31 -5
  269. package/dist/repertoire/tasks/fix.js +182 -0
  270. package/dist/repertoire/tasks/index.js +16 -4
  271. package/dist/repertoire/tasks/lifecycle.js +2 -2
  272. package/dist/repertoire/tasks/parser.js +3 -2
  273. package/dist/repertoire/tasks/scanner.js +194 -37
  274. package/dist/repertoire/tasks/transitions.js +16 -78
  275. package/dist/repertoire/tool-results.js +29 -0
  276. package/dist/repertoire/tools-attachments.js +317 -0
  277. package/dist/repertoire/tools-awaiting.js +360 -0
  278. package/dist/repertoire/tools-base.js +53 -1082
  279. package/dist/repertoire/tools-bluebubbles.js +1 -0
  280. package/dist/repertoire/tools-bridge.js +142 -0
  281. package/dist/repertoire/tools-bundle.js +984 -0
  282. package/dist/repertoire/tools-config.js +185 -0
  283. package/dist/repertoire/tools-continuity.js +248 -0
  284. package/dist/repertoire/tools-credential.js +381 -0
  285. package/dist/repertoire/tools-files.js +342 -0
  286. package/dist/repertoire/tools-flight.js +224 -0
  287. package/dist/repertoire/tools-flow.js +119 -0
  288. package/dist/repertoire/tools-github.js +1 -7
  289. package/dist/repertoire/tools-mail.js +1916 -0
  290. package/dist/repertoire/tools-notes.js +421 -0
  291. package/dist/repertoire/tools-obligations.js +142 -0
  292. package/dist/repertoire/tools-runtime.js +61 -0
  293. package/dist/repertoire/tools-session.js +809 -0
  294. package/dist/repertoire/tools-shell.js +120 -0
  295. package/dist/repertoire/tools-stripe.js +180 -0
  296. package/dist/repertoire/tools-surface.js +345 -0
  297. package/dist/repertoire/tools-teams.js +9 -39
  298. package/dist/repertoire/tools-travel.js +125 -0
  299. package/dist/repertoire/tools-trip.js +604 -0
  300. package/dist/repertoire/tools-user-profile.js +144 -0
  301. package/dist/repertoire/tools-vault.js +40 -0
  302. package/dist/repertoire/tools-voice.js +144 -0
  303. package/dist/repertoire/tools.js +115 -103
  304. package/dist/repertoire/travel-api-client.js +360 -0
  305. package/dist/repertoire/user-profile.js +131 -0
  306. package/dist/repertoire/vault-setup.js +246 -0
  307. package/dist/repertoire/vault-unlock.js +594 -0
  308. package/dist/scripts/claude-code-hook.js +41 -0
  309. package/dist/scripts/claude-code-stop-hook.js +47 -0
  310. package/dist/senses/attention-queue.js +116 -0
  311. package/dist/senses/await-turn-message.js +58 -0
  312. package/dist/senses/bluebubbles/active-turns.js +216 -0
  313. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  314. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  315. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  316. package/dist/senses/bluebubbles/entry.js +77 -0
  317. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  318. package/dist/senses/bluebubbles/index.js +2613 -0
  319. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
  320. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  321. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  322. package/dist/senses/bluebubbles/processed-log.js +133 -0
  323. package/dist/senses/bluebubbles/replay.js +137 -0
  324. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
  325. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  326. package/dist/senses/bluebubbles-meta-guard.js +40 -0
  327. package/dist/senses/cli/bracketed-paste.js +82 -0
  328. package/dist/senses/cli/image-paste.js +287 -0
  329. package/dist/senses/cli/image-ref-navigation.js +75 -0
  330. package/dist/senses/cli/ink-app.js +156 -0
  331. package/dist/senses/cli/inline-diff.js +64 -0
  332. package/dist/senses/cli/input-keys.js +174 -0
  333. package/dist/senses/cli/kill-ring.js +86 -0
  334. package/dist/senses/cli/message-list.js +51 -0
  335. package/dist/senses/cli/ouro-tui.js +607 -0
  336. package/dist/senses/cli/spinner-imperative.js +135 -0
  337. package/dist/senses/cli/spinner.js +101 -0
  338. package/dist/senses/cli/status-line.js +60 -0
  339. package/dist/senses/cli/streaming-markdown.js +526 -0
  340. package/dist/senses/cli/tool-display.js +85 -0
  341. package/dist/senses/cli/tool-render.js +85 -0
  342. package/dist/senses/cli/tui-store.js +240 -0
  343. package/dist/senses/cli/virtual-list.js +35 -0
  344. package/dist/senses/cli-entry.js +60 -8
  345. package/dist/senses/cli-layout.js +100 -0
  346. package/dist/senses/cli.js +516 -204
  347. package/dist/senses/commands.js +66 -3
  348. package/dist/senses/habit-turn-message.js +108 -0
  349. package/dist/senses/inner-dialog-worker.js +185 -21
  350. package/dist/senses/inner-dialog.js +469 -39
  351. package/dist/senses/mail-entry.js +66 -0
  352. package/dist/senses/mail.js +379 -0
  353. package/dist/senses/pipeline.js +654 -181
  354. package/dist/senses/proactive-content-guard.js +51 -0
  355. package/dist/senses/shared-turn.js +392 -0
  356. package/dist/senses/surface-tool.js +70 -0
  357. package/dist/senses/teams-entry.js +60 -8
  358. package/dist/senses/teams.js +387 -98
  359. package/dist/senses/trust-gate.js +100 -5
  360. package/dist/senses/voice/audio-playback.js +237 -0
  361. package/dist/senses/voice/audio-routing.js +119 -0
  362. package/dist/senses/voice/elevenlabs.js +202 -0
  363. package/dist/senses/voice/floor-control.js +431 -0
  364. package/dist/senses/voice/floor-controller.js +115 -0
  365. package/dist/senses/voice/golden-path.js +116 -0
  366. package/dist/senses/voice/index.js +29 -0
  367. package/dist/senses/voice/meeting.js +113 -0
  368. package/dist/senses/voice/outbound.js +190 -0
  369. package/dist/senses/voice/phone.js +33 -0
  370. package/dist/senses/voice/playback.js +139 -0
  371. package/dist/senses/voice/realtime-eval.js +496 -0
  372. package/dist/senses/voice/realtime-trace.js +531 -0
  373. package/dist/senses/voice/transcript.js +70 -0
  374. package/dist/senses/voice/turn.js +191 -0
  375. package/dist/senses/voice/twilio-phone-runtime.js +807 -0
  376. package/dist/senses/voice/twilio-phone.js +5079 -0
  377. package/dist/senses/voice/types.js +2 -0
  378. package/dist/senses/voice/whisper.js +161 -0
  379. package/dist/senses/voice-entry.js +81 -0
  380. package/dist/senses/voice-realtime-eval-command.js +99 -0
  381. package/dist/senses/voice-realtime-eval-entry.js +21 -0
  382. package/dist/senses/voice-twilio-entry.js +87 -0
  383. package/dist/trips/core.js +138 -0
  384. package/dist/trips/store.js +265 -0
  385. package/package.json +41 -7
  386. package/skills/agent-commerce.md +106 -0
  387. package/skills/browser-navigation.md +117 -0
  388. package/skills/commerce-setup-guide.md +116 -0
  389. package/skills/commerce-setup.md +84 -0
  390. package/skills/configure-dev-tools.md +99 -0
  391. package/skills/travel-planning.md +138 -0
  392. package/dist/heart/daemon/auth-flow.js +0 -351
  393. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  394. package/dist/heart/safe-workspace.js +0 -228
  395. package/dist/heart/session-recall.js +0 -116
  396. package/dist/mind/associative-recall.js +0 -209
  397. package/dist/senses/bluebubbles-entry.js +0 -13
  398. package/dist/senses/bluebubbles.js +0 -1177
  399. package/dist/senses/debug-activity.js +0 -148
  400. package/subagents/README.md +0 -7
  401. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  402. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  403. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  404. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  405. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  406. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  407. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  408. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  409. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  410. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  411. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  412. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  413. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  414. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  415. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  416. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -33,7 +33,12 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.HARNESS_CANONICAL_REPO_URL = exports.DEFAULT_AGENT_SENSES = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = void 0;
36
+ exports.HARNESS_CANONICAL_REPO_URL = exports.DEFAULT_AGENT_SENSES = exports.LEGACY_VAULT_SERVER_URL_ALIASES = exports.DEFAULT_VAULT_SERVER_URL = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = exports.PROVIDER_CREDENTIALS = void 0;
37
+ exports.normalizeVaultServerUrl = normalizeVaultServerUrl;
38
+ exports.getVaultServerUrlCandidates = getVaultServerUrlCandidates;
39
+ exports.defaultStableVaultEmail = defaultStableVaultEmail;
40
+ exports.resolveVaultConfig = resolveVaultConfig;
41
+ exports.normalizeSenses = normalizeSenses;
37
42
  exports.buildDefaultAgentTemplate = buildDefaultAgentTemplate;
38
43
  exports.getAgentName = getAgentName;
39
44
  exports.getRepoRoot = getRepoRoot;
@@ -45,8 +50,8 @@ exports.getAgentDaemonStateRoot = getAgentDaemonStateRoot;
45
50
  exports.getAgentDaemonLogsDir = getAgentDaemonLogsDir;
46
51
  exports.getAgentDaemonLoggingConfigPath = getAgentDaemonLoggingConfigPath;
47
52
  exports.getAgentMessagesRoot = getAgentMessagesRoot;
53
+ exports.getAgentMailroomRoot = getAgentMailroomRoot;
48
54
  exports.getAgentToolsRoot = getAgentToolsRoot;
49
- exports.getAgentSecretsPath = getAgentSecretsPath;
50
55
  exports.loadAgentConfig = loadAgentConfig;
51
56
  exports.setAgentName = setAgentName;
52
57
  exports.setAgentConfigOverride = setAgentConfigOverride;
@@ -56,6 +61,15 @@ const fs = __importStar(require("fs"));
56
61
  const os = __importStar(require("os"));
57
62
  const path = __importStar(require("path"));
58
63
  const runtime_1 = require("../nerves/runtime");
64
+ const migrate_config_1 = require("./migrate-config");
65
+ /** Single source of truth for per-provider credential field names, env var mappings, and prompt labels. */
66
+ exports.PROVIDER_CREDENTIALS = {
67
+ anthropic: { required: ["setupToken"], envVars: { ANTHROPIC_API_KEY: "setupToken" }, promptLabels: { setupToken: "Anthropic setup-token" } },
68
+ "openai-codex": { required: ["oauthAccessToken"], envVars: { OPENAI_API_KEY: "oauthAccessToken" }, promptLabels: { oauthAccessToken: "OpenAI Codex OAuth token" } },
69
+ azure: { required: ["apiKey", "endpoint", "deployment"], envVars: { AZURE_OPENAI_API_KEY: "apiKey", AZURE_OPENAI_KEY: "apiKey", AZURE_OPENAI_ENDPOINT: "endpoint", AZURE_OPENAI_DEPLOYMENT: "deployment" }, promptLabels: { apiKey: "Azure API key", endpoint: "Azure endpoint", deployment: "Azure deployment" } },
70
+ minimax: { required: ["apiKey"], envVars: { MINIMAX_API_KEY: "apiKey" }, promptLabels: { apiKey: "MiniMax API key" } },
71
+ "github-copilot": { required: ["githubToken", "baseUrl"], envVars: { GH_TOKEN: "githubToken", GITHUB_TOKEN: "githubToken" }, promptLabels: { githubToken: "GitHub token" } },
72
+ };
59
73
  exports.DEFAULT_AGENT_CONTEXT = {
60
74
  maxTokens: 80000,
61
75
  contextMargin: 20,
@@ -65,16 +79,72 @@ exports.DEFAULT_AGENT_PHRASES = {
65
79
  tool: ["running tool"],
66
80
  followup: ["processing"],
67
81
  };
82
+ exports.DEFAULT_VAULT_SERVER_URL = "https://vault.ouro.bot";
83
+ exports.LEGACY_VAULT_SERVER_URL_ALIASES = [
84
+ "https://vault.ouroboros.bot",
85
+ "https://ouro-vault.gentleflower-74452a1e.eastus2.azurecontainerapps.io",
86
+ ];
87
+ function normalizeVaultServerUrl(serverUrl) {
88
+ const trimmed = serverUrl.trim();
89
+ const withoutTrailingSlash = trimmed.replace(/\/+$/, "");
90
+ if (!withoutTrailingSlash) {
91
+ return exports.DEFAULT_VAULT_SERVER_URL;
92
+ }
93
+ if (exports.LEGACY_VAULT_SERVER_URL_ALIASES.includes(withoutTrailingSlash)) {
94
+ return exports.DEFAULT_VAULT_SERVER_URL;
95
+ }
96
+ return withoutTrailingSlash;
97
+ }
98
+ function getVaultServerUrlCandidates(serverUrl) {
99
+ const raw = serverUrl.trim();
100
+ const withoutTrailingSlash = raw.replace(/\/+$/, "");
101
+ const normalized = normalizeVaultServerUrl(serverUrl);
102
+ const candidates = [normalized];
103
+ for (const candidate of [withoutTrailingSlash, raw]) {
104
+ if (candidate && !candidates.includes(candidate)) {
105
+ candidates.push(candidate);
106
+ }
107
+ }
108
+ if (normalized === exports.DEFAULT_VAULT_SERVER_URL) {
109
+ for (const alias of exports.LEGACY_VAULT_SERVER_URL_ALIASES) {
110
+ if (!candidates.includes(alias)) {
111
+ candidates.push(alias);
112
+ }
113
+ }
114
+ }
115
+ return candidates;
116
+ }
117
+ function defaultStableVaultEmail(agentName) {
118
+ const local = agentName
119
+ .toLowerCase()
120
+ .replace(/[^a-z0-9._-]+/g, "-")
121
+ .replace(/^-+|-+$/g, "") || "agent";
122
+ return `${local}@ouro.bot`;
123
+ }
124
+ /**
125
+ * Resolve the vault config for an agent, applying defaults.
126
+ * If vault is not configured in agent.json, returns default values.
127
+ */
128
+ function resolveVaultConfig(agentName, config) {
129
+ return {
130
+ email: config?.email ?? defaultStableVaultEmail(agentName),
131
+ serverUrl: normalizeVaultServerUrl(config?.serverUrl ?? exports.DEFAULT_VAULT_SERVER_URL),
132
+ };
133
+ }
68
134
  exports.DEFAULT_AGENT_SENSES = {
69
135
  cli: { enabled: true },
70
136
  teams: { enabled: false },
71
137
  bluebubbles: { enabled: false },
138
+ mail: { enabled: false },
139
+ voice: { enabled: false },
72
140
  };
73
141
  function normalizeSenses(value, configFile) {
74
142
  const defaults = {
75
143
  cli: { ...exports.DEFAULT_AGENT_SENSES.cli },
76
144
  teams: { ...exports.DEFAULT_AGENT_SENSES.teams },
77
145
  bluebubbles: { ...exports.DEFAULT_AGENT_SENSES.bluebubbles },
146
+ mail: { ...exports.DEFAULT_AGENT_SENSES.mail },
147
+ voice: { ...exports.DEFAULT_AGENT_SENSES.voice },
78
148
  };
79
149
  if (value === undefined) {
80
150
  return defaults;
@@ -90,7 +160,7 @@ function normalizeSenses(value, configFile) {
90
160
  throw new Error(`agent.json at ${configFile} must include senses as an object when present.`);
91
161
  }
92
162
  const raw = value;
93
- const senseNames = ["cli", "teams", "bluebubbles"];
163
+ const senseNames = ["cli", "teams", "bluebubbles", "mail", "voice"];
94
164
  for (const senseName of senseNames) {
95
165
  const rawSense = raw[senseName];
96
166
  if (rawSense === undefined) {
@@ -123,14 +193,17 @@ function normalizeSenses(value, configFile) {
123
193
  }
124
194
  function buildDefaultAgentTemplate(_agentName) {
125
195
  return {
126
- version: 1,
196
+ version: 2,
127
197
  enabled: true,
128
- provider: "anthropic",
198
+ humanFacing: { provider: "anthropic", model: "claude-opus-4-6" },
199
+ agentFacing: { provider: "anthropic", model: "claude-opus-4-6" },
129
200
  context: { ...exports.DEFAULT_AGENT_CONTEXT },
130
201
  senses: {
131
202
  cli: { ...exports.DEFAULT_AGENT_SENSES.cli },
132
203
  teams: { ...exports.DEFAULT_AGENT_SENSES.teams },
133
204
  bluebubbles: { ...exports.DEFAULT_AGENT_SENSES.bluebubbles },
205
+ mail: { ...exports.DEFAULT_AGENT_SENSES.mail },
206
+ voice: { ...exports.DEFAULT_AGENT_SENSES.voice },
134
207
  },
135
208
  phrases: {
136
209
  thinking: [...exports.DEFAULT_AGENT_PHRASES.thinking],
@@ -148,12 +221,6 @@ let _agentConfigOverride = null;
148
221
  */
149
222
  function getAgentName() {
150
223
  if (_cachedAgentName) {
151
- (0, runtime_1.emitNervesEvent)({
152
- event: "identity.resolve",
153
- component: "config/identity",
154
- message: "resolved agent name from cache",
155
- meta: { source: "cache" },
156
- });
157
224
  return _cachedAgentName;
158
225
  }
159
226
  const idx = process.argv.indexOf("--agent");
@@ -222,26 +289,17 @@ function getAgentDaemonLoggingConfigPath(agentName) {
222
289
  function getAgentMessagesRoot(agentName) {
223
290
  return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "messages");
224
291
  }
292
+ function getAgentMailroomRoot(agentName) {
293
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "mailroom");
294
+ }
225
295
  function getAgentToolsRoot(agentName) {
226
296
  return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "tools");
227
297
  }
228
- /**
229
- * Returns the conventional secrets path: `~/.agentsecrets/<agentName>/secrets.json`
230
- */
231
- function getAgentSecretsPath(agentName = getAgentName()) {
232
- return path.join(os.homedir(), ".agentsecrets", agentName, "secrets.json");
298
+ const VALID_PROVIDERS = ["azure", "minimax", "anthropic", "openai-codex", "github-copilot"];
299
+ function isValidProvider(value) {
300
+ return typeof value === "string" && VALID_PROVIDERS.includes(value);
233
301
  }
234
- /**
235
- * Load and parse `<agentRoot>/agent.json`.
236
- * Reads the file fresh on each call unless an override is set.
237
- * Throws descriptive error if file is missing or contains invalid JSON.
238
- */
239
- function loadAgentConfig() {
240
- if (_agentConfigOverride) {
241
- return _agentConfigOverride;
242
- }
243
- const agentRoot = getAgentRoot();
244
- const configFile = path.join(agentRoot, "agent.json");
302
+ function readAndParseAgentJson(configFile) {
245
303
  let raw;
246
304
  try {
247
305
  raw = fs.readFileSync(configFile, "utf-8");
@@ -259,9 +317,8 @@ function loadAgentConfig() {
259
317
  });
260
318
  throw new Error(`Cannot read agent.json at ${configFile}. Does the agent directory exist?`);
261
319
  }
262
- let parsed;
263
320
  try {
264
- parsed = JSON.parse(raw);
321
+ return JSON.parse(raw);
265
322
  }
266
323
  catch (error) {
267
324
  (0, runtime_1.emitNervesEvent)({
@@ -276,6 +333,62 @@ function loadAgentConfig() {
276
333
  });
277
334
  throw new Error(`Invalid JSON in agent.json at ${configFile}. Check syntax.`);
278
335
  }
336
+ }
337
+ function validateFacingConfig(parsed, facingName, configFile) {
338
+ const raw = parsed[facingName];
339
+ if (!raw || typeof raw !== "object" || Array.isArray(raw)) {
340
+ (0, runtime_1.emitNervesEvent)({
341
+ level: "error",
342
+ event: "config_identity.error",
343
+ component: "config/identity",
344
+ message: `agent config missing or invalid ${facingName}`,
345
+ meta: { path: configFile, [facingName]: raw ?? null },
346
+ });
347
+ throw new Error(`agent.json at ${configFile} must include ${facingName} as { provider, model }.`);
348
+ }
349
+ const facing = raw;
350
+ if (!isValidProvider(facing.provider)) {
351
+ (0, runtime_1.emitNervesEvent)({
352
+ level: "error",
353
+ event: "config_identity.error",
354
+ component: "config/identity",
355
+ message: `agent config has invalid provider in ${facingName}`,
356
+ meta: { path: configFile, provider: facing.provider ?? null },
357
+ });
358
+ throw new Error(`agent.json at ${configFile} ${facingName}.provider must be one of: ${VALID_PROVIDERS.join(", ")}.`);
359
+ }
360
+ if (typeof facing.model !== "string") {
361
+ (0, runtime_1.emitNervesEvent)({
362
+ level: "error",
363
+ event: "config_identity.error",
364
+ component: "config/identity",
365
+ message: `agent config has invalid model in ${facingName}`,
366
+ meta: { path: configFile, model: facing.model ?? null },
367
+ });
368
+ throw new Error(`agent.json at ${configFile} ${facingName}.model must be a string.`);
369
+ }
370
+ return { provider: facing.provider, model: facing.model };
371
+ }
372
+ /**
373
+ * Load and parse `<agentRoot>/agent.json`.
374
+ * Reads the file fresh on each call unless an override is set.
375
+ * If the config is v1, auto-migrates to v2 via migrateAgentConfigV1ToV2 and re-reads.
376
+ * Throws descriptive error if file is missing or contains invalid JSON.
377
+ */
378
+ function loadAgentConfig() {
379
+ if (_agentConfigOverride) {
380
+ return _agentConfigOverride;
381
+ }
382
+ const agentRoot = getAgentRoot();
383
+ const configFile = path.join(agentRoot, "agent.json");
384
+ let parsed = readAndParseAgentJson(configFile);
385
+ // Inline migration: v1 -> v2
386
+ const rawVersion = parsed.version;
387
+ const initialVersion = typeof rawVersion === "number" ? rawVersion : 1;
388
+ if (initialVersion < 2) {
389
+ (0, migrate_config_1.migrateAgentConfigV1ToV2)(agentRoot);
390
+ parsed = readAndParseAgentJson(configFile);
391
+ }
279
392
  const existingPhrases = parsed.phrases;
280
393
  const needsFill = !existingPhrases ||
281
394
  !existingPhrases.thinking ||
@@ -297,28 +410,11 @@ function loadAgentConfig() {
297
410
  });
298
411
  fs.writeFileSync(configFile, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
299
412
  }
300
- const rawProvider = parsed.provider;
301
- if (rawProvider !== "azure" &&
302
- rawProvider !== "minimax" &&
303
- rawProvider !== "anthropic" &&
304
- rawProvider !== "openai-codex") {
305
- (0, runtime_1.emitNervesEvent)({
306
- level: "error",
307
- event: "config_identity.error",
308
- component: "config/identity",
309
- message: "agent config missing or invalid provider",
310
- meta: {
311
- path: configFile,
312
- provider: rawProvider,
313
- },
314
- });
315
- throw new Error(`agent.json at ${configFile} must include provider: "azure", "minimax", "anthropic", or "openai-codex".`);
316
- }
317
- const provider = rawProvider;
318
- const rawVersion = parsed.version;
319
- const version = rawVersion === undefined ? 1 : rawVersion;
320
- if (typeof version !== "number" ||
321
- !Number.isInteger(version) ||
413
+ // Validate v2 facing configs
414
+ const humanFacing = validateFacingConfig(parsed, "humanFacing", configFile);
415
+ const agentFacing = validateFacingConfig(parsed, "agentFacing", configFile);
416
+ const version = typeof parsed.version === "number" ? parsed.version : 1;
417
+ if (!Number.isInteger(version) ||
322
418
  version < 1) {
323
419
  (0, runtime_1.emitNervesEvent)({
324
420
  level: "error",
@@ -327,7 +423,7 @@ function loadAgentConfig() {
327
423
  message: "agent config missing or invalid version",
328
424
  meta: {
329
425
  path: configFile,
330
- version: rawVersion,
426
+ version: parsed.version,
331
427
  },
332
428
  });
333
429
  throw new Error(`agent.json at ${configFile} must include version as integer >= 1.`);
@@ -347,15 +443,30 @@ function loadAgentConfig() {
347
443
  });
348
444
  throw new Error(`agent.json at ${configFile} must include enabled as boolean.`);
349
445
  }
446
+ // Tolerate deprecated provider field for backward compatibility
447
+ const rawProvider = parsed.provider;
448
+ const provider = isValidProvider(rawProvider) ? rawProvider : undefined;
449
+ // Spread parsed first so any field present in AgentConfig is carried
450
+ // through by default, then explicitly override the fields that need
451
+ // validation or normalization. This eliminates the field-drop bug class
452
+ // that caused the `sync` block (and previously `shell`) to be silently
453
+ // omitted from the returned config. Regression-guarded by the
454
+ // Required<AgentConfig> contract test in identity-contract.test.ts.
350
455
  const config = {
456
+ ...parsed,
351
457
  version,
352
458
  enabled,
353
- provider,
354
- context: parsed.context,
355
- logging: parsed.logging,
459
+ humanFacing,
460
+ agentFacing,
356
461
  senses: normalizeSenses(parsed.senses, configFile),
357
462
  phrases: parsed.phrases,
358
463
  };
464
+ if (provider !== undefined) {
465
+ config.provider = provider;
466
+ }
467
+ else {
468
+ delete config.provider;
469
+ }
359
470
  (0, runtime_1.emitNervesEvent)({
360
471
  event: "identity.resolve",
361
472
  component: "config/identity",
@@ -0,0 +1,357 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.gatherKeptNotesCandidates = gatherKeptNotesCandidates;
37
+ exports.renderKeptNotesOutcome = renderKeptNotesOutcome;
38
+ exports.injectKeptNotes = injectKeptNotes;
39
+ exports.createKeptNotesJudge = createKeptNotesJudge;
40
+ const fs = __importStar(require("fs"));
41
+ const path = __importStar(require("path"));
42
+ const diary_1 = require("../mind/diary");
43
+ const runtime_1 = require("../nerves/runtime");
44
+ const DEFAULT_TIMEOUT_MS = 2500;
45
+ const MAX_CANDIDATES = 8;
46
+ const NOOP_CALLBACKS = {
47
+ onModelStart() { },
48
+ onModelStreamStart() { },
49
+ onTextChunk() { },
50
+ onReasoningChunk() { },
51
+ onToolStart() { },
52
+ onToolEnd() { },
53
+ onError() { },
54
+ };
55
+ function elapsedSince(startedAt) {
56
+ return Math.max(0, Date.now() - startedAt);
57
+ }
58
+ function latestUserText(messages) {
59
+ for (let i = messages.length - 1; i >= 0; i--) {
60
+ const message = messages[i];
61
+ if (message.role !== "user" || typeof message.content !== "string")
62
+ continue;
63
+ const text = message.content.trim();
64
+ if (text)
65
+ return text;
66
+ }
67
+ return "";
68
+ }
69
+ function tokenize(text) {
70
+ return new Set(text
71
+ .toLowerCase()
72
+ .split(/[^a-z0-9]+/g)
73
+ .map((token) => token.trim())
74
+ .filter((token) => token.length > 2));
75
+ }
76
+ function scoreText(queryTerms, text) {
77
+ const textTerms = tokenize(text);
78
+ let matches = 0;
79
+ for (const term of queryTerms) {
80
+ if (textTerms.has(term))
81
+ matches += 1;
82
+ }
83
+ return matches / queryTerms.size;
84
+ }
85
+ function readJournalIndex(journalDir) {
86
+ try {
87
+ const parsed = JSON.parse(fs.readFileSync(path.join(journalDir, ".index.json"), "utf8"));
88
+ if (!Array.isArray(parsed))
89
+ return [];
90
+ return parsed
91
+ .filter((entry) => (typeof entry === "object" &&
92
+ entry !== null &&
93
+ typeof entry.filename === "string" &&
94
+ typeof entry.preview === "string"));
95
+ }
96
+ catch {
97
+ return [];
98
+ }
99
+ }
100
+ function journalDirForDiaryRoot(diaryRoot, explicitJournalDir) {
101
+ if (explicitJournalDir)
102
+ return explicitJournalDir;
103
+ return path.join(path.dirname(diaryRoot), "journal");
104
+ }
105
+ function diaryCandidate(fact) {
106
+ return {
107
+ text: fact.text,
108
+ source: { kind: "diary", label: "diary", ref: fact.id },
109
+ };
110
+ }
111
+ function friendNoteCandidates(friend, queryTerms) {
112
+ if (!friend)
113
+ return [];
114
+ return Object.entries(friend.notes ?? {})
115
+ .map(([key, note]) => {
116
+ const text = `${friend.name} / ${key}: ${note.value}`;
117
+ return {
118
+ candidate: {
119
+ text,
120
+ source: { kind: "friend-note", label: `friend note: ${friend.name}`, ref: key },
121
+ },
122
+ score: scoreText(queryTerms, `${key} ${note.value}`),
123
+ };
124
+ })
125
+ .filter((entry) => entry.score > 0);
126
+ }
127
+ function gatherKeptNotesCandidates(query, options = {}) {
128
+ const queryTerms = tokenize(query);
129
+ if (queryTerms.size === 0)
130
+ return [];
131
+ const diaryRoot = (0, diary_1.resolveDiaryRoot)(options.diaryRoot);
132
+ const diaryEntries = (0, diary_1.readDiaryEntries)(diaryRoot);
133
+ const diaryCandidates = diaryEntries
134
+ .map((fact) => ({ candidate: diaryCandidate(fact), score: scoreText(queryTerms, fact.text) }))
135
+ .filter((entry) => entry.score > 0);
136
+ const journalDir = journalDirForDiaryRoot(diaryRoot, options.journalDir);
137
+ const journalCandidates = readJournalIndex(journalDir)
138
+ .map((entry) => ({
139
+ candidate: {
140
+ text: `${entry.filename}: ${entry.preview}`,
141
+ source: { kind: "journal", label: "journal", ref: entry.filename },
142
+ },
143
+ score: scoreText(queryTerms, `${entry.filename} ${entry.preview}`),
144
+ }))
145
+ .filter((entry) => entry.score > 0);
146
+ return [...diaryCandidates, ...journalCandidates, ...friendNoteCandidates(options.friend, queryTerms)]
147
+ .sort((left, right) => right.score - left.score)
148
+ .slice(0, MAX_CANDIDATES)
149
+ .map((entry) => entry.candidate);
150
+ }
151
+ function selectedSources(candidates, indexes) {
152
+ if (!indexes || indexes.length === 0)
153
+ return [];
154
+ const sources = [];
155
+ for (const index of indexes) {
156
+ if (!Number.isInteger(index))
157
+ continue;
158
+ const candidate = candidates[index];
159
+ if (candidate)
160
+ sources.push(candidate.source);
161
+ }
162
+ return sources;
163
+ }
164
+ function isFirstPerson(text) {
165
+ return /^(i|i'm|i’ve|i've|my|me)\b/i.test(text.trim());
166
+ }
167
+ function foundLine(text) {
168
+ const trimmed = text.trim();
169
+ if (isFirstPerson(trimmed))
170
+ return trimmed;
171
+ return `I kept this: ${trimmed}`;
172
+ }
173
+ function fuzzyLine(text) {
174
+ const trimmed = text.trim();
175
+ if (isFirstPerson(trimmed))
176
+ return trimmed;
177
+ return `I may have kept something related: ${trimmed}`;
178
+ }
179
+ const SOURCE_KIND_ORDER = ["diary", "journal", "friend-note"];
180
+ const SOURCE_KIND_LABELS = {
181
+ diary: "my diary",
182
+ journal: "my journal",
183
+ "friend-note": "my friend notes",
184
+ };
185
+ function joinLabels(labels) {
186
+ if (labels.length === 1)
187
+ return labels[0];
188
+ if (labels.length === 2)
189
+ return `${labels[0]} and ${labels[1]}`;
190
+ return `${labels.slice(0, -1).join(", ")}, and ${labels[labels.length - 1]}`;
191
+ }
192
+ function sourceHeading(sources) {
193
+ const sourceKinds = new Set(sources.map((source) => source.kind));
194
+ const labels = SOURCE_KIND_ORDER
195
+ .filter((kind) => sourceKinds.has(kind))
196
+ .map((kind) => SOURCE_KIND_LABELS[kind]);
197
+ if (labels.length === 0)
198
+ return "## from notes i chose to keep";
199
+ return `## from ${joinLabels(labels)}`;
200
+ }
201
+ function renderKeptNotesOutcome(outcome) {
202
+ if (outcome.status === "found") {
203
+ return `${sourceHeading(outcome.sources)}\nThis may matter now:\n${foundLine(outcome.note)}`;
204
+ }
205
+ if (outcome.status === "fuzzy") {
206
+ return `${sourceHeading(outcome.sources)}\nThis is only a possible match; I should verify it before relying on it:\n${fuzzyLine(outcome.hint)}`;
207
+ }
208
+ return null;
209
+ }
210
+ function finish(outcome, traceId) {
211
+ (0, runtime_1.emitNervesEvent)({
212
+ component: "mind",
213
+ event: "mind.kept_notes_end",
214
+ trace_id: traceId,
215
+ message: "kept notes completed",
216
+ meta: { status: outcome.status, elapsedMs: outcome.elapsedMs },
217
+ });
218
+ return outcome;
219
+ }
220
+ async function withTimeout(promise, timeoutMs) {
221
+ return await new Promise((resolve, reject) => {
222
+ const timeout = setTimeout(() => resolve("timeout"), timeoutMs);
223
+ promise
224
+ .then((value) => {
225
+ clearTimeout(timeout);
226
+ resolve(value);
227
+ })
228
+ .catch((error) => {
229
+ clearTimeout(timeout);
230
+ reject(error);
231
+ });
232
+ });
233
+ }
234
+ async function injectKeptNotes(messages, options = {}) {
235
+ const startedAt = Date.now();
236
+ const systemMessage = messages[0];
237
+ const query = latestUserText(messages);
238
+ if (systemMessage?.role !== "system" || typeof systemMessage.content !== "string" || !query) {
239
+ return { status: "none", elapsedMs: elapsedSince(startedAt), pressure: [] };
240
+ }
241
+ (0, runtime_1.emitNervesEvent)({
242
+ component: "mind",
243
+ event: "mind.kept_notes_start",
244
+ trace_id: options.traceId,
245
+ message: "kept notes started",
246
+ meta: { channel: options.channel ?? "unknown" },
247
+ });
248
+ try {
249
+ const candidates = gatherKeptNotesCandidates(query, options);
250
+ if (candidates.length === 0 || !options.judge) {
251
+ return finish({ status: "none", elapsedMs: elapsedSince(startedAt), pressure: [] }, options.traceId);
252
+ }
253
+ const judged = await withTimeout(options.judge({ query, candidates }), options.timeoutMs ?? DEFAULT_TIMEOUT_MS);
254
+ if (judged === "timeout") {
255
+ return finish({ status: "timeout", elapsedMs: elapsedSince(startedAt) }, options.traceId);
256
+ }
257
+ const elapsedMs = elapsedSince(startedAt);
258
+ const outcome = judged.status === "found"
259
+ ? { status: "found", note: judged.note, sources: selectedSources(candidates, judged.sourceIndexes), elapsedMs }
260
+ : judged.status === "fuzzy"
261
+ ? { status: "fuzzy", hint: judged.hint, sources: selectedSources(candidates, judged.sourceIndexes), elapsedMs }
262
+ : { status: "none", pressure: judged.pressure, elapsedMs };
263
+ if (outcome.status === "found" || outcome.status === "fuzzy") {
264
+ const rendered = renderKeptNotesOutcome(outcome);
265
+ messages[0] = { role: "system", content: `${systemMessage.content}\n\n${rendered}` };
266
+ (0, runtime_1.emitNervesEvent)({
267
+ component: "mind",
268
+ event: "mind.kept_notes_injected",
269
+ trace_id: options.traceId,
270
+ message: "kept notes injected",
271
+ meta: { status: outcome.status, sourceCount: outcome.sources.length },
272
+ });
273
+ }
274
+ return finish(outcome, options.traceId);
275
+ }
276
+ catch (error) {
277
+ const reason = error instanceof Error ? error.message : String(error);
278
+ const outcome = { status: "error", reason, elapsedMs: elapsedSince(startedAt) };
279
+ (0, runtime_1.emitNervesEvent)({
280
+ level: "warn",
281
+ component: "mind",
282
+ event: "mind.kept_notes_error",
283
+ trace_id: options.traceId,
284
+ message: "kept notes failed",
285
+ meta: { reason },
286
+ });
287
+ return finish(outcome, options.traceId);
288
+ }
289
+ }
290
+ function parseJudgeResult(content) {
291
+ try {
292
+ const parsed = JSON.parse(content);
293
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
294
+ return { status: "none", pressure: ["invalid kept notes judge output"] };
295
+ }
296
+ const record = parsed;
297
+ if (record.status === "found" && typeof record.note === "string" && record.note.trim()) {
298
+ return {
299
+ status: "found",
300
+ note: record.note.trim(),
301
+ sourceIndexes: Array.isArray(record.sourceIndexes) ? record.sourceIndexes.filter((index) => typeof index === "number") : undefined,
302
+ };
303
+ }
304
+ if (record.status === "fuzzy" && typeof record.hint === "string" && record.hint.trim()) {
305
+ return {
306
+ status: "fuzzy",
307
+ hint: record.hint.trim(),
308
+ sourceIndexes: Array.isArray(record.sourceIndexes) ? record.sourceIndexes.filter((index) => typeof index === "number") : undefined,
309
+ };
310
+ }
311
+ if (record.status === "none") {
312
+ return {
313
+ status: "none",
314
+ pressure: Array.isArray(record.pressure) ? record.pressure.filter((value) => typeof value === "string") : [],
315
+ };
316
+ }
317
+ return { status: "none", pressure: ["invalid kept notes judge output"] };
318
+ }
319
+ catch {
320
+ return { status: "none", pressure: ["invalid kept notes judge output"] };
321
+ }
322
+ }
323
+ function createJudgePrompt(input) {
324
+ const candidates = input.candidates
325
+ .map((candidate, index) => `${index}. [${candidate.source.kind}] ${candidate.text}`)
326
+ .join("\n");
327
+ return [
328
+ {
329
+ role: "system",
330
+ content: [
331
+ "Decide whether these intentionally kept notes matter to the user's current turn.",
332
+ "Return only JSON.",
333
+ "Use found when a note clearly helps, fuzzy when a note is close but uncertain, and none when nothing should be surfaced.",
334
+ "Shapes: {\"status\":\"found\",\"note\":\"...\",\"sourceIndexes\":[0]}, {\"status\":\"fuzzy\",\"hint\":\"...\",\"sourceIndexes\":[0]}, or {\"status\":\"none\",\"pressure\":[]}.",
335
+ ].join("\n"),
336
+ },
337
+ {
338
+ role: "user",
339
+ content: `Current turn:\n${input.query}\n\nCandidates:\n${candidates}`,
340
+ },
341
+ ];
342
+ }
343
+ function createKeptNotesJudge(runtime, signal) {
344
+ return async (input) => {
345
+ const messages = createJudgePrompt(input);
346
+ runtime.resetTurnState?.(messages);
347
+ const result = await runtime.streamTurn({
348
+ messages,
349
+ activeTools: [],
350
+ callbacks: NOOP_CALLBACKS,
351
+ signal,
352
+ toolChoiceRequired: false,
353
+ reasoningEffort: "low",
354
+ });
355
+ return parseJudgeResult(result.content ?? "");
356
+ };
357
+ }