@ouro.bot/cli 0.1.0-alpha.49 → 0.1.0-alpha.490

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 (365) hide show
  1. package/README.md +133 -19
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +3118 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +989 -0
  16. package/dist/heart/agent-entry.js +58 -3
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +426 -0
  28. package/dist/heart/background-operations.js +281 -0
  29. package/dist/heart/bridges/manager.js +37 -0
  30. package/dist/heart/bridges/state-machine.js +20 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +119 -129
  35. package/dist/heart/core.js +758 -227
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +490 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +216 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +640 -0
  44. package/dist/heart/daemon/cli-exec.js +7229 -0
  45. package/dist/heart/daemon/cli-help.js +493 -0
  46. package/dist/heart/daemon/cli-parse.js +1533 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +561 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/connect-bay.js +323 -0
  51. package/dist/heart/daemon/daemon-cli.js +29 -1616
  52. package/dist/heart/daemon/daemon-entry.js +345 -3
  53. package/dist/heart/daemon/daemon-health.js +141 -0
  54. package/dist/heart/daemon/daemon-runtime-sync.js +190 -12
  55. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  56. package/dist/heart/daemon/daemon.js +677 -58
  57. package/dist/heart/daemon/dns-workflow.js +394 -0
  58. package/dist/heart/daemon/doctor-types.js +8 -0
  59. package/dist/heart/daemon/doctor.js +486 -0
  60. package/dist/heart/daemon/health-monitor.js +92 -1
  61. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  62. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  63. package/dist/heart/daemon/http-health-probe.js +80 -0
  64. package/dist/heart/daemon/human-command-screens.js +234 -0
  65. package/dist/heart/daemon/human-readiness.js +114 -0
  66. package/dist/heart/daemon/inner-status.js +89 -0
  67. package/dist/heart/daemon/interactive-repair.js +394 -0
  68. package/dist/heart/daemon/launchd.js +25 -5
  69. package/dist/heart/daemon/log-tailer.js +82 -12
  70. package/dist/heart/daemon/logs-prune.js +110 -0
  71. package/dist/heart/daemon/message-router.js +2 -2
  72. package/dist/heart/daemon/os-cron-deps.js +134 -0
  73. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  74. package/dist/heart/daemon/ouro-entry.js +3 -1
  75. package/dist/heart/daemon/process-manager.js +214 -0
  76. package/dist/heart/daemon/provider-discovery.js +137 -0
  77. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  78. package/dist/heart/daemon/pulse.js +475 -0
  79. package/dist/heart/daemon/readiness-repair.js +365 -0
  80. package/dist/heart/daemon/run-hooks.js +2 -0
  81. package/dist/heart/daemon/runtime-logging.js +67 -16
  82. package/dist/heart/daemon/runtime-metadata.js +73 -0
  83. package/dist/heart/daemon/runtime-mode.js +67 -0
  84. package/dist/heart/daemon/safe-mode.js +161 -0
  85. package/dist/heart/daemon/sense-manager.js +178 -37
  86. package/dist/heart/daemon/session-id-resolver.js +131 -0
  87. package/dist/heart/daemon/skill-management-installer.js +94 -0
  88. package/dist/heart/daemon/socket-client.js +109 -4
  89. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  90. package/dist/heart/daemon/startup-tui.js +264 -0
  91. package/dist/heart/daemon/task-scheduler.js +3 -25
  92. package/dist/heart/daemon/terminal-ui.js +499 -0
  93. package/dist/heart/daemon/thoughts.js +162 -17
  94. package/dist/heart/daemon/up-progress.js +366 -0
  95. package/dist/heart/daemon/vault-items.js +56 -0
  96. package/dist/heart/delegation.js +62 -0
  97. package/dist/heart/habits/habit-migration.js +189 -0
  98. package/dist/heart/habits/habit-parser.js +140 -0
  99. package/dist/heart/habits/habit-runtime-state.js +100 -0
  100. package/dist/heart/habits/habit-scheduler.js +372 -0
  101. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  102. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  103. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  104. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  105. package/dist/heart/identity.js +201 -66
  106. package/dist/heart/kept-notes.js +357 -0
  107. package/dist/heart/kicks.js +1 -1
  108. package/dist/heart/machine-identity.js +161 -0
  109. package/dist/heart/mail-import-discovery.js +353 -0
  110. package/dist/heart/mcp/mcp-server.js +653 -0
  111. package/dist/heart/migrate-config.js +100 -0
  112. package/dist/heart/model-capabilities.js +59 -0
  113. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  114. package/dist/heart/outlook/outlook-http-response.js +7 -0
  115. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  116. package/dist/heart/outlook/outlook-http-static.js +103 -0
  117. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  118. package/dist/heart/outlook/outlook-http.js +99 -0
  119. package/dist/heart/outlook/outlook-read.js +31 -0
  120. package/dist/heart/outlook/outlook-types.js +27 -0
  121. package/dist/heart/outlook/outlook-view.js +195 -0
  122. package/dist/heart/outlook/readers/agent-machine.js +382 -0
  123. package/dist/heart/outlook/readers/continuity-readers.js +336 -0
  124. package/dist/heart/outlook/readers/mail.js +362 -0
  125. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  126. package/dist/heart/outlook/readers/sessions.js +232 -0
  127. package/dist/heart/outlook/readers/shared.js +111 -0
  128. package/dist/heart/platform.js +81 -0
  129. package/dist/heart/progress-story.js +42 -0
  130. package/dist/heart/provider-attempt.js +134 -0
  131. package/dist/heart/provider-binding-resolver.js +255 -0
  132. package/dist/heart/provider-credentials.js +424 -0
  133. package/dist/heart/provider-failover.js +301 -0
  134. package/dist/heart/provider-models.js +81 -0
  135. package/dist/heart/provider-ping.js +262 -0
  136. package/dist/heart/provider-state.js +216 -0
  137. package/dist/heart/provider-visibility.js +188 -0
  138. package/dist/heart/providers/anthropic-token.js +131 -0
  139. package/dist/heart/providers/anthropic.js +193 -55
  140. package/dist/heart/providers/azure.js +104 -13
  141. package/dist/heart/providers/error-classification.js +63 -0
  142. package/dist/heart/providers/github-copilot.js +145 -0
  143. package/dist/heart/providers/minimax-vlm.js +189 -0
  144. package/dist/heart/providers/minimax.js +29 -7
  145. package/dist/heart/providers/openai-codex.js +63 -39
  146. package/dist/heart/runtime-capability-check.js +170 -0
  147. package/dist/heart/runtime-credentials.js +260 -0
  148. package/dist/heart/sense-truth.js +11 -4
  149. package/dist/heart/session-activity.js +190 -0
  150. package/dist/heart/session-events.js +981 -0
  151. package/dist/heart/session-transcript.js +167 -0
  152. package/dist/heart/start-of-turn-packet.js +345 -0
  153. package/dist/heart/streaming.js +48 -28
  154. package/dist/heart/sync.js +332 -0
  155. package/dist/heart/target-resolution.js +127 -0
  156. package/dist/heart/tempo.js +93 -0
  157. package/dist/heart/temporal-view.js +41 -0
  158. package/dist/heart/tool-activity-callbacks.js +36 -0
  159. package/dist/heart/tool-description.js +135 -0
  160. package/dist/heart/tool-friction.js +55 -0
  161. package/dist/heart/tool-loop.js +200 -0
  162. package/dist/heart/turn-context.js +372 -0
  163. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  164. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  165. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  166. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  167. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  168. package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
  169. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  170. package/dist/mailroom/attention.js +167 -0
  171. package/dist/mailroom/autonomy.js +209 -0
  172. package/dist/mailroom/blob-store.js +600 -0
  173. package/dist/mailroom/core.js +658 -0
  174. package/dist/mailroom/entry.js +160 -0
  175. package/dist/mailroom/file-store.js +426 -0
  176. package/dist/mailroom/mbox-import.js +382 -0
  177. package/dist/mailroom/outbound.js +380 -0
  178. package/dist/mailroom/policy.js +263 -0
  179. package/dist/mailroom/reader.js +219 -0
  180. package/dist/mailroom/search-cache.js +182 -0
  181. package/dist/mailroom/search-relevance.js +319 -0
  182. package/dist/mailroom/smtp-ingress.js +176 -0
  183. package/dist/mailroom/source-state.js +176 -0
  184. package/dist/mailroom/travel-extract.js +89 -0
  185. package/dist/mind/bundle-manifest.js +7 -1
  186. package/dist/mind/context.js +164 -93
  187. package/dist/mind/diary-integrity.js +60 -0
  188. package/dist/mind/{memory.js → diary.js} +74 -93
  189. package/dist/mind/embedding-provider.js +60 -0
  190. package/dist/mind/file-state.js +179 -0
  191. package/dist/mind/friends/channel.js +30 -0
  192. package/dist/mind/friends/group-context.js +144 -0
  193. package/dist/mind/friends/resolver.js +54 -2
  194. package/dist/mind/friends/store-file.js +39 -3
  195. package/dist/mind/friends/trust-explanation.js +74 -0
  196. package/dist/mind/friends/types.js +2 -2
  197. package/dist/mind/journal-index.js +161 -0
  198. package/dist/mind/note-search.js +268 -0
  199. package/dist/mind/obligation-steering.js +221 -0
  200. package/dist/mind/pending.js +56 -8
  201. package/dist/mind/prompt-refresh.js +3 -2
  202. package/dist/mind/prompt.js +973 -168
  203. package/dist/mind/provenance-trust.js +26 -0
  204. package/dist/mind/scrutiny.js +173 -0
  205. package/dist/nerves/cli-logging.js +7 -1
  206. package/dist/nerves/coverage/audit-rules.js +15 -6
  207. package/dist/nerves/coverage/audit.js +28 -2
  208. package/dist/nerves/coverage/cli.js +1 -1
  209. package/dist/nerves/coverage/contract.js +5 -5
  210. package/dist/nerves/coverage/file-completeness.js +93 -5
  211. package/dist/nerves/coverage/run-artifacts.js +1 -1
  212. package/dist/nerves/event-buffer.js +111 -0
  213. package/dist/nerves/index.js +224 -4
  214. package/dist/nerves/observation.js +20 -0
  215. package/dist/nerves/redact.js +79 -0
  216. package/dist/nerves/runtime.js +5 -1
  217. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  218. package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
  219. package/dist/outlook-ui/index.html +15 -0
  220. package/dist/repertoire/ado-client.js +15 -56
  221. package/dist/repertoire/ado-semantic.js +11 -10
  222. package/dist/repertoire/api-client.js +97 -0
  223. package/dist/repertoire/bitwarden-store.js +774 -0
  224. package/dist/repertoire/bundle-templates.js +72 -0
  225. package/dist/repertoire/bw-installer.js +180 -0
  226. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  227. package/dist/repertoire/coding/context-pack.js +330 -0
  228. package/dist/repertoire/coding/feedback.js +197 -30
  229. package/dist/repertoire/coding/manager.js +158 -9
  230. package/dist/repertoire/coding/spawner.js +55 -9
  231. package/dist/repertoire/coding/tools.js +170 -7
  232. package/dist/repertoire/commerce-errors.js +109 -0
  233. package/dist/repertoire/commerce-self-test.js +156 -0
  234. package/dist/repertoire/credential-access.js +111 -0
  235. package/dist/repertoire/duffel-client.js +185 -0
  236. package/dist/repertoire/github-client.js +14 -55
  237. package/dist/repertoire/graph-client.js +11 -52
  238. package/dist/repertoire/guardrails.js +396 -0
  239. package/dist/repertoire/mcp-client.js +255 -0
  240. package/dist/repertoire/mcp-manager.js +305 -0
  241. package/dist/repertoire/mcp-tools.js +63 -0
  242. package/dist/repertoire/shell-sessions.js +133 -0
  243. package/dist/repertoire/skills.js +15 -24
  244. package/dist/repertoire/stripe-client.js +131 -0
  245. package/dist/repertoire/tasks/board.js +31 -5
  246. package/dist/repertoire/tasks/fix.js +182 -0
  247. package/dist/repertoire/tasks/index.js +16 -4
  248. package/dist/repertoire/tasks/lifecycle.js +2 -2
  249. package/dist/repertoire/tasks/parser.js +3 -2
  250. package/dist/repertoire/tasks/scanner.js +194 -37
  251. package/dist/repertoire/tasks/transitions.js +16 -78
  252. package/dist/repertoire/tool-results.js +29 -0
  253. package/dist/repertoire/tools-attachments.js +317 -0
  254. package/dist/repertoire/tools-base.js +46 -842
  255. package/dist/repertoire/tools-bluebubbles.js +1 -0
  256. package/dist/repertoire/tools-bridge.js +141 -0
  257. package/dist/repertoire/tools-bundle.js +984 -0
  258. package/dist/repertoire/tools-config.js +185 -0
  259. package/dist/repertoire/tools-continuity.js +248 -0
  260. package/dist/repertoire/tools-credential.js +381 -0
  261. package/dist/repertoire/tools-files.js +342 -0
  262. package/dist/repertoire/tools-flight.js +224 -0
  263. package/dist/repertoire/tools-flow.js +105 -0
  264. package/dist/repertoire/tools-github.js +1 -7
  265. package/dist/repertoire/tools-mail.js +1281 -0
  266. package/dist/repertoire/tools-notes.js +376 -0
  267. package/dist/repertoire/tools-session.js +749 -0
  268. package/dist/repertoire/tools-shell.js +120 -0
  269. package/dist/repertoire/tools-stripe.js +180 -0
  270. package/dist/repertoire/tools-surface.js +243 -0
  271. package/dist/repertoire/tools-teams.js +9 -39
  272. package/dist/repertoire/tools-travel.js +125 -0
  273. package/dist/repertoire/tools-trip.js +280 -0
  274. package/dist/repertoire/tools-user-profile.js +144 -0
  275. package/dist/repertoire/tools-vault.js +40 -0
  276. package/dist/repertoire/tools.js +144 -115
  277. package/dist/repertoire/travel-api-client.js +360 -0
  278. package/dist/repertoire/user-profile.js +131 -0
  279. package/dist/repertoire/vault-setup.js +246 -0
  280. package/dist/repertoire/vault-unlock.js +561 -0
  281. package/dist/scripts/claude-code-hook.js +41 -0
  282. package/dist/scripts/claude-code-stop-hook.js +47 -0
  283. package/dist/senses/attention-queue.js +116 -0
  284. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  285. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  286. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  287. package/dist/senses/bluebubbles/entry.js +73 -0
  288. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  289. package/dist/senses/bluebubbles/index.js +1835 -0
  290. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  291. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  292. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  293. package/dist/senses/bluebubbles/processed-log.js +111 -0
  294. package/dist/senses/bluebubbles/replay.js +129 -0
  295. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +2 -2
  296. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  297. package/dist/senses/cli/bracketed-paste.js +82 -0
  298. package/dist/senses/cli/image-paste.js +287 -0
  299. package/dist/senses/cli/image-ref-navigation.js +75 -0
  300. package/dist/senses/cli/ink-app.js +156 -0
  301. package/dist/senses/cli/inline-diff.js +64 -0
  302. package/dist/senses/cli/input-keys.js +174 -0
  303. package/dist/senses/cli/kill-ring.js +86 -0
  304. package/dist/senses/cli/message-list.js +51 -0
  305. package/dist/senses/cli/ouro-tui.js +605 -0
  306. package/dist/senses/cli/spinner-imperative.js +135 -0
  307. package/dist/senses/cli/spinner.js +101 -0
  308. package/dist/senses/cli/status-line.js +60 -0
  309. package/dist/senses/cli/streaming-markdown.js +526 -0
  310. package/dist/senses/cli/tool-display.js +83 -0
  311. package/dist/senses/cli/tool-render.js +85 -0
  312. package/dist/senses/cli/tui-store.js +240 -0
  313. package/dist/senses/cli/virtual-list.js +35 -0
  314. package/dist/senses/cli-entry.js +60 -8
  315. package/dist/senses/cli-layout.js +187 -0
  316. package/dist/senses/cli.js +515 -211
  317. package/dist/senses/commands.js +66 -3
  318. package/dist/senses/habit-turn-message.js +108 -0
  319. package/dist/senses/inner-dialog-worker.js +110 -20
  320. package/dist/senses/inner-dialog.js +408 -21
  321. package/dist/senses/mail-entry.js +66 -0
  322. package/dist/senses/mail.js +379 -0
  323. package/dist/senses/pipeline.js +588 -81
  324. package/dist/senses/proactive-content-guard.js +51 -0
  325. package/dist/senses/shared-turn.js +205 -0
  326. package/dist/senses/surface-tool.js +68 -0
  327. package/dist/senses/teams-entry.js +60 -8
  328. package/dist/senses/teams.js +412 -163
  329. package/dist/senses/trust-gate.js +100 -5
  330. package/dist/trips/core.js +138 -0
  331. package/dist/trips/store.js +146 -0
  332. package/package.json +37 -7
  333. package/skills/agent-commerce.md +106 -0
  334. package/skills/browser-navigation.md +117 -0
  335. package/skills/commerce-setup-guide.md +116 -0
  336. package/skills/commerce-setup.md +84 -0
  337. package/skills/configure-dev-tools.md +101 -0
  338. package/skills/travel-planning.md +138 -0
  339. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  340. package/dist/heart/daemon/subagent-installer.js +0 -166
  341. package/dist/heart/session-recall.js +0 -116
  342. package/dist/mind/associative-recall.js +0 -209
  343. package/dist/senses/bluebubbles-entry.js +0 -13
  344. package/dist/senses/bluebubbles.js +0 -1032
  345. package/dist/senses/debug-activity.js +0 -127
  346. package/subagents/README.md +0 -86
  347. package/subagents/work-doer.md +0 -237
  348. package/subagents/work-merger.md +0 -618
  349. package/subagents/work-planner.md +0 -390
  350. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  351. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  352. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  353. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  354. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  355. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  356. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  357. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  358. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  359. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  360. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  361. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  362. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  363. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  364. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  365. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,161 @@
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.SAFE_MODE_OVERRIDE_FILENAME = void 0;
37
+ exports.detectSafeMode = detectSafeMode;
38
+ exports.pruneOldCrashes = pruneOldCrashes;
39
+ const fs = __importStar(require("fs"));
40
+ const path = __importStar(require("path"));
41
+ const runtime_1 = require("../../nerves/runtime");
42
+ exports.SAFE_MODE_OVERRIDE_FILENAME = "safe-mode-override.json";
43
+ /** 3+ crashes within this window triggers safe mode */
44
+ const CRASH_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
45
+ const CRASH_THRESHOLD = 3;
46
+ /**
47
+ * Reads crash history from the tombstone file and determines if safe mode should be active.
48
+ * Returns active if 3+ crashes occurred within 5 minutes.
49
+ *
50
+ * Safe mode is bypassed when:
51
+ * - devMode is true
52
+ * - A safe-mode override file exists (written by `ouro up --force`)
53
+ */
54
+ function detectSafeMode(tombstonePath, options) {
55
+ const inactive = { active: false, reason: "", enteredAt: "" };
56
+ // Dev mode: never enter safe mode
57
+ if (options?.devMode) {
58
+ return inactive;
59
+ }
60
+ // Check for override file (--force)
61
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
62
+ try {
63
+ if (fs.existsSync(overridePath)) {
64
+ return inactive;
65
+ }
66
+ }
67
+ catch {
68
+ // Best-effort check
69
+ }
70
+ // Read tombstone
71
+ let parsed;
72
+ try {
73
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
74
+ parsed = JSON.parse(raw);
75
+ }
76
+ catch {
77
+ return inactive;
78
+ }
79
+ // Extract recentCrashes
80
+ if (!Array.isArray(parsed.recentCrashes)) {
81
+ return inactive;
82
+ }
83
+ const nowMs = options?.now ? options.now() : Date.now();
84
+ const windowStart = nowMs - CRASH_WINDOW_MS;
85
+ // Filter to valid string timestamps within the crash window
86
+ const recentInWindow = parsed.recentCrashes.filter((entry) => {
87
+ if (typeof entry !== "string")
88
+ return false;
89
+ const ts = new Date(entry).getTime();
90
+ if (isNaN(ts))
91
+ return false;
92
+ return ts >= windowStart;
93
+ });
94
+ if (recentInWindow.length < CRASH_THRESHOLD) {
95
+ return inactive;
96
+ }
97
+ const result = {
98
+ active: true,
99
+ reason: `crash loop detected: ${recentInWindow.length} crashes in last 5 minutes`,
100
+ enteredAt: new Date(nowMs).toISOString(),
101
+ };
102
+ (0, runtime_1.emitNervesEvent)({
103
+ level: "error",
104
+ component: "daemon",
105
+ event: "daemon.safe_mode_entered",
106
+ message: result.reason,
107
+ meta: {
108
+ crashCount: recentInWindow.length,
109
+ windowMs: CRASH_WINDOW_MS,
110
+ tombstonePath,
111
+ },
112
+ });
113
+ return result;
114
+ }
115
+ /**
116
+ * Prunes crash entries older than 5 minutes from the tombstone's recentCrashes.
117
+ * Also removes the safe-mode override file if present.
118
+ * Called after successful startup (uptime > stability threshold).
119
+ */
120
+ function pruneOldCrashes(tombstonePath, options) {
121
+ // Remove override file
122
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
123
+ try {
124
+ if (fs.existsSync(overridePath)) {
125
+ fs.unlinkSync(overridePath);
126
+ }
127
+ }
128
+ catch {
129
+ // Best-effort
130
+ }
131
+ // Read existing tombstone
132
+ let parsed;
133
+ try {
134
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
135
+ parsed = JSON.parse(raw);
136
+ }
137
+ catch {
138
+ return;
139
+ }
140
+ if (!Array.isArray(parsed.recentCrashes)) {
141
+ return;
142
+ }
143
+ const nowMs = options?.now ? options.now() : Date.now();
144
+ const windowStart = nowMs - CRASH_WINDOW_MS;
145
+ // Keep only entries within the window
146
+ const pruned = parsed.recentCrashes.filter((entry) => {
147
+ if (typeof entry !== "string")
148
+ return false;
149
+ const ts = new Date(entry).getTime();
150
+ if (isNaN(ts))
151
+ return false;
152
+ return ts >= windowStart;
153
+ });
154
+ parsed.recentCrashes = pruned;
155
+ try {
156
+ fs.writeFileSync(tombstonePath, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
157
+ }
158
+ catch {
159
+ // Best-effort
160
+ }
161
+ }
@@ -38,18 +38,22 @@ const fs = __importStar(require("fs"));
38
38
  const os = __importStar(require("os"));
39
39
  const path = __importStar(require("path"));
40
40
  const runtime_1 = require("../../nerves/runtime");
41
- const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
42
41
  const identity_1 = require("../identity");
42
+ const runtime_credentials_1 = require("../runtime-credentials");
43
43
  const sense_truth_1 = require("../sense-truth");
44
+ const machine_identity_1 = require("../machine-identity");
44
45
  const process_manager_1 = require("./process-manager");
46
+ const http_health_probe_1 = require("./http-health-probe");
45
47
  const DEFAULT_TEAMS_PORT = 3978;
46
48
  const DEFAULT_BLUEBUBBLES_PORT = 18790;
47
49
  const DEFAULT_BLUEBUBBLES_WEBHOOK_PATH = "/bluebubbles-webhook";
50
+ const BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS = 90_000;
48
51
  function defaultSenses() {
49
52
  return {
50
53
  cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
51
54
  teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
52
55
  bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
56
+ mail: { ...identity_1.DEFAULT_AGENT_SENSES.mail },
53
57
  };
54
58
  }
55
59
  function readAgentSenses(agentJsonPath) {
@@ -75,7 +79,7 @@ function readAgentSenses(agentJsonPath) {
75
79
  if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
76
80
  return defaults;
77
81
  }
78
- for (const sense of ["cli", "teams", "bluebubbles"]) {
82
+ for (const sense of ["cli", "teams", "bluebubbles", "mail"]) {
79
83
  const rawSense = rawSenses[sense];
80
84
  if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
81
85
  continue;
@@ -87,22 +91,6 @@ function readAgentSenses(agentJsonPath) {
87
91
  }
88
92
  return defaults;
89
93
  }
90
- function readSecretsPayload(secretsPath) {
91
- try {
92
- const raw = fs.readFileSync(secretsPath, "utf-8");
93
- const parsed = JSON.parse(raw);
94
- if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
95
- return { payload: {}, error: "invalid secrets.json object" };
96
- }
97
- return { payload: parsed, error: null };
98
- }
99
- catch (error) {
100
- return {
101
- payload: {},
102
- error: error instanceof Error ? error.message : String(error),
103
- };
104
- }
105
- }
106
94
  function textField(record, key) {
107
95
  const value = record?.[key];
108
96
  return typeof value === "string" ? value.trim() : "";
@@ -111,17 +99,36 @@ function numberField(record, key, fallback) {
111
99
  const value = record?.[key];
112
100
  return typeof value === "number" && Number.isFinite(value) ? value : fallback;
113
101
  }
114
- function senseFactsFromSecrets(agent, senses, secretsPath) {
102
+ function compactRuntimeConfigError(agent, error) {
103
+ const compact = error.replace(/\s+/g, " ").trim();
104
+ if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
105
+ return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
106
+ }
107
+ return compact || "unavailable";
108
+ }
109
+ function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
110
+ if (runtimeConfig.ok)
111
+ return "";
112
+ const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
113
+ if (runtimeConfig.reason === "missing")
114
+ return `missing vault ${itemName} (${agent})`;
115
+ return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
116
+ }
117
+ function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
115
118
  const base = {
116
119
  cli: { configured: true, detail: "local interactive terminal" },
117
120
  teams: { configured: false, detail: "not enabled in agent.json" },
118
121
  bluebubbles: { configured: false, detail: "not enabled in agent.json" },
122
+ mail: { configured: false, detail: "not enabled in agent.json" },
119
123
  };
120
- const { payload, error } = readSecretsPayload(secretsPath);
124
+ const payload = runtimeConfig.ok ? runtimeConfig.config : {};
125
+ const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
121
126
  const teams = payload.teams;
122
127
  const teamsChannel = payload.teamsChannel;
123
- const bluebubbles = payload.bluebubbles;
124
- const bluebubblesChannel = payload.bluebubblesChannel;
128
+ const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
129
+ const bluebubbles = machinePayload.bluebubbles;
130
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
131
+ const mailroom = payload.mailroom;
125
132
  if (senses.teams.enabled) {
126
133
  const missing = [];
127
134
  if (!textField(teams, "clientId"))
@@ -137,9 +144,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
137
144
  }
138
145
  : {
139
146
  configured: false,
140
- detail: error && !fs.existsSync(secretsPath)
141
- ? `missing secrets.json (${agent})`
142
- : `missing ${missing.join("/")}`,
147
+ detail: runtimeConfig.ok
148
+ ? `missing ${missing.join("/")}`
149
+ : unavailableDetail,
143
150
  };
144
151
  }
145
152
  if (senses.bluebubbles.enabled) {
@@ -155,19 +162,52 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
155
162
  }
156
163
  : {
157
164
  configured: false,
158
- detail: error && !fs.existsSync(secretsPath)
159
- ? `missing secrets.json (${agent})`
160
- : `missing ${missing.join("/")}`,
165
+ optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
166
+ detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
167
+ ? "not attached on this machine"
168
+ : machineRuntimeConfig.ok
169
+ ? `missing ${missing.join("/")}`
170
+ : runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
171
+ };
172
+ }
173
+ if (senses.mail.enabled) {
174
+ const privateKeys = mailroom?.privateKeys;
175
+ const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
176
+ const mailboxAddress = textField(mailroom, "mailboxAddress");
177
+ const missing = [];
178
+ if (!mailboxAddress)
179
+ missing.push("mailroom.mailboxAddress");
180
+ if (!hasPrivateKeys)
181
+ missing.push("mailroom.privateKeys");
182
+ base.mail = missing.length === 0
183
+ ? { configured: true, detail: mailboxAddress }
184
+ : {
185
+ configured: false,
186
+ detail: runtimeConfig.ok
187
+ ? `missing ${missing.join("/")}`
188
+ : unavailableDetail,
161
189
  };
162
190
  }
163
191
  return base;
164
192
  }
193
+ function senseRepairHint(agent, sense) {
194
+ if (sense === "teams") {
195
+ return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
196
+ }
197
+ if (sense === "mail") {
198
+ return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
199
+ }
200
+ return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
201
+ }
202
+ function currentMachineId() {
203
+ return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
204
+ }
165
205
  function parseSenseSnapshotName(name) {
166
206
  const parts = name.split(":");
167
207
  if (parts.length !== 2)
168
208
  return null;
169
209
  const [agent, sense] = parts;
170
- if (sense !== "teams" && sense !== "bluebubbles")
210
+ if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail")
171
211
  return null;
172
212
  return { agent, sense };
173
213
  }
@@ -176,20 +216,65 @@ function runtimeInfoFor(status) {
176
216
  return { runtime: "running" };
177
217
  return { runtime: "error" };
178
218
  }
219
+ function managedSenseEntry(sense) {
220
+ if (sense === "teams")
221
+ return "senses/teams-entry.js";
222
+ if (sense === "bluebubbles")
223
+ return "senses/bluebubbles/entry.js";
224
+ return "senses/mail-entry.js";
225
+ }
226
+ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
227
+ if (!lastCheckedAt) {
228
+ return false;
229
+ }
230
+ const checkedAt = Date.parse(lastCheckedAt);
231
+ if (!Number.isFinite(checkedAt)) {
232
+ return false;
233
+ }
234
+ return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
235
+ }
236
+ function readBlueBubblesRuntimeJson(runtimePath) {
237
+ try {
238
+ const raw = fs.readFileSync(runtimePath, "utf-8");
239
+ const parsed = JSON.parse(raw);
240
+ /* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
241
+ return {
242
+ upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
243
+ ? parsed.upstreamStatus
244
+ : "unknown",
245
+ detail: typeof parsed.detail === "string" && parsed.detail.trim()
246
+ ? parsed.detail
247
+ : "startup health probe pending",
248
+ lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
249
+ };
250
+ /* v8 ignore stop */
251
+ /* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
252
+ }
253
+ catch {
254
+ return { upstreamStatus: "unknown", detail: "startup health probe pending" };
255
+ }
256
+ /* v8 ignore stop */
257
+ }
179
258
  function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
180
259
  const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
181
260
  const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
182
- if (snapshot?.runtime !== "running" || !fs.existsSync(runtimePath)) {
261
+ if (!fs.existsSync(runtimePath)) {
262
+ return { runtime: snapshot?.runtime };
263
+ }
264
+ const state = readBlueBubblesRuntimeJson(runtimePath);
265
+ if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
183
266
  return { runtime: snapshot?.runtime };
184
267
  }
185
- const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
186
268
  if (state.upstreamStatus === "error") {
187
269
  return {
188
270
  runtime: "error",
189
271
  detail: state.detail,
190
272
  };
191
273
  }
192
- return { runtime: snapshot.runtime };
274
+ if (state.upstreamStatus === "ok") {
275
+ return { runtime: "running" };
276
+ }
277
+ return { runtime: snapshot?.runtime };
193
278
  }
194
279
  class DaemonSenseManager {
195
280
  processManager;
@@ -197,26 +282,53 @@ class DaemonSenseManager {
197
282
  bundlesRoot;
198
283
  constructor(options) {
199
284
  const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
200
- const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
201
285
  this.bundlesRoot = bundlesRoot;
202
286
  this.contexts = new Map(options.agents.map((agent) => {
203
287
  const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
204
- const facts = senseFactsFromSecrets(agent, senses, path.join(secretsRoot, agent, "secrets.json"));
288
+ const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
205
289
  return [agent, { senses, facts }];
206
290
  }));
207
291
  const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
208
- return ["teams", "bluebubbles"]
209
- .filter((sense) => context.senses[sense].enabled && context.facts[sense].configured)
292
+ return ["teams", "bluebubbles", "mail"]
293
+ .filter((sense) => context.senses[sense].enabled)
210
294
  .map((sense) => ({
211
295
  name: `${agent}:${sense}`,
212
296
  agentArg: agent,
213
- entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles-entry.js",
297
+ entry: managedSenseEntry(sense),
214
298
  channel: sense,
215
299
  autoStart: true,
216
300
  }));
217
301
  });
218
302
  this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
219
303
  agents: managedSenseAgents,
304
+ configCheck: async (name) => {
305
+ const parsed = parseSenseSnapshotName(name);
306
+ if (!parsed)
307
+ return { ok: true };
308
+ const context = this.contexts.get(parsed.agent);
309
+ if (!context)
310
+ return { ok: true };
311
+ const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
312
+ const machineRefreshed = parsed.sense === "bluebubbles"
313
+ ? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
314
+ : (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
315
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
316
+ const fact = context.facts[parsed.sense];
317
+ if (fact.configured)
318
+ return { ok: true };
319
+ if (fact.optional) {
320
+ return {
321
+ ok: false,
322
+ skip: true,
323
+ error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
324
+ };
325
+ }
326
+ return {
327
+ ok: false,
328
+ error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
329
+ fix: senseRepairHint(parsed.agent, parsed.sense),
330
+ };
331
+ },
220
332
  });
221
333
  (0, runtime_1.emitNervesEvent)({
222
334
  component: "channels",
@@ -234,6 +346,29 @@ class DaemonSenseManager {
234
346
  async stopAll() {
235
347
  await this.processManager.stopAll();
236
348
  }
349
+ /* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
350
+ listManagedPids() {
351
+ return this.processManager.listAgentSnapshots()
352
+ .map((s) => s.pid)
353
+ .filter((pid) => pid !== null && pid !== undefined);
354
+ }
355
+ /* v8 ignore stop */
356
+ listHealthProbes() {
357
+ const probes = [];
358
+ for (const [agent, context] of this.contexts.entries()) {
359
+ const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
360
+ const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
361
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
362
+ if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
363
+ continue;
364
+ }
365
+ const machinePayload = machineRuntimeConfig.config;
366
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
367
+ const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
368
+ probes.push((0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port));
369
+ }
370
+ return probes;
371
+ }
237
372
  listSenseRows() {
238
373
  const runtime = new Map();
239
374
  for (const snapshot of this.processManager.listAgentSnapshots()) {
@@ -245,6 +380,7 @@ class DaemonSenseManager {
245
380
  runtime.set(parsed.agent, current);
246
381
  }
247
382
  const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
383
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
248
384
  const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
249
385
  const runtimeInfo = {
250
386
  cli: { configured: true },
@@ -254,8 +390,13 @@ class DaemonSenseManager {
254
390
  },
255
391
  bluebubbles: {
256
392
  configured: context.facts.bluebubbles.configured,
393
+ optional: context.facts.bluebubbles.optional,
257
394
  ...blueBubblesRuntimeFacts,
258
395
  },
396
+ mail: {
397
+ configured: context.facts.mail.configured,
398
+ ...(runtime.get(agent)?.mail ?? {}),
399
+ },
259
400
  };
260
401
  const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
261
402
  return inventory.map((entry) => ({
@@ -0,0 +1,131 @@
1
+ "use strict";
2
+ // Pluggable session ID resolver for MCP conversations.
3
+ // Tries tool-specific methods first, falls back to UUID.
4
+ //
5
+ // Claude Code: walks parent PID chain -> reads ~/.claude/sessions/{pid}.json
6
+ // Codex: reads thread_id from env (future)
7
+ // Fallback: generates UUID
8
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
9
+ if (k2 === undefined) k2 = k;
10
+ var desc = Object.getOwnPropertyDescriptor(m, k);
11
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
12
+ desc = { enumerable: true, get: function() { return m[k]; } };
13
+ }
14
+ Object.defineProperty(o, k2, desc);
15
+ }) : (function(o, m, k, k2) {
16
+ if (k2 === undefined) k2 = k;
17
+ o[k2] = m[k];
18
+ }));
19
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
20
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
21
+ }) : function(o, v) {
22
+ o["default"] = v;
23
+ });
24
+ var __importStar = (this && this.__importStar) || (function () {
25
+ var ownKeys = function(o) {
26
+ ownKeys = Object.getOwnPropertyNames || function (o) {
27
+ var ar = [];
28
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
29
+ return ar;
30
+ };
31
+ return ownKeys(o);
32
+ };
33
+ return function (mod) {
34
+ if (mod && mod.__esModule) return mod;
35
+ var result = {};
36
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
37
+ __setModuleDefault(result, mod);
38
+ return result;
39
+ };
40
+ })();
41
+ Object.defineProperty(exports, "__esModule", { value: true });
42
+ exports.resolveSessionId = resolveSessionId;
43
+ const fs = __importStar(require("fs"));
44
+ const path = __importStar(require("path"));
45
+ const os = __importStar(require("os"));
46
+ const crypto_1 = require("crypto");
47
+ const runtime_1 = require("../../nerves/runtime");
48
+ const DEFAULT_CLAUDE_SESSIONS_DIR = path.join(os.homedir(), ".claude", "sessions");
49
+ const MAX_PID_WALK_DEPTH = 10;
50
+ /**
51
+ * Try to read a Claude Code session ID from a PID-keyed session file.
52
+ * Returns the sessionId if found, null otherwise.
53
+ */
54
+ function tryReadClaudeSession(sessionsDir, pid) {
55
+ const sessionFile = path.join(sessionsDir, `${pid}.json`);
56
+ if (!fs.existsSync(sessionFile))
57
+ return null;
58
+ try {
59
+ const raw = fs.readFileSync(sessionFile, "utf-8");
60
+ const data = JSON.parse(raw);
61
+ if (typeof data.sessionId === "string" && data.sessionId.length > 0) {
62
+ return data.sessionId;
63
+ }
64
+ return null;
65
+ }
66
+ catch {
67
+ return null;
68
+ }
69
+ }
70
+ /**
71
+ * Walk the parent PID chain looking for a Claude Code session file.
72
+ * Starts at the current process PID, walks up to parent, grandparent, etc.
73
+ * Returns the session ID from the first matching file, or null.
74
+ */
75
+ function walkPidChain(sessionsDir) {
76
+ let currentPid = process.pid;
77
+ for (let depth = 0; depth < MAX_PID_WALK_DEPTH; depth++) {
78
+ const sessionId = tryReadClaudeSession(sessionsDir, currentPid);
79
+ if (sessionId) {
80
+ (0, runtime_1.emitNervesEvent)({
81
+ component: "daemon",
82
+ event: "daemon.session_id_pid_walk_hit",
83
+ message: "found Claude session via PID walk",
84
+ meta: { pid: currentPid, depth, sessionId },
85
+ });
86
+ return sessionId;
87
+ }
88
+ // Walk to parent PID
89
+ // On macOS/Linux, process.ppid gives the parent. For deeper ancestry,
90
+ // we'd need /proc/{pid}/stat or ps -o ppid=. For now, we check current + parent.
91
+ if (depth === 0) {
92
+ currentPid = process.ppid;
93
+ }
94
+ else {
95
+ // Cannot walk further without OS-specific process tree APIs
96
+ break;
97
+ }
98
+ }
99
+ return null;
100
+ }
101
+ /**
102
+ * Resolve a session ID for the current MCP connection.
103
+ * Returns a stable identifier that ties MCP tool calls to a conversation session.
104
+ *
105
+ * Resolution order:
106
+ * 1. Claude Code PID walk: check ~/.claude/sessions/{pid}.json for current + parent PID
107
+ * 2. UUID fallback: generate a random UUID
108
+ */
109
+ function resolveSessionId(options) {
110
+ const sessionsDir = options?.claudeSessionsDir ?? DEFAULT_CLAUDE_SESSIONS_DIR;
111
+ // Try Claude Code PID walk
112
+ const claudeSessionId = walkPidChain(sessionsDir);
113
+ if (claudeSessionId) {
114
+ (0, runtime_1.emitNervesEvent)({
115
+ component: "daemon",
116
+ event: "daemon.session_id_resolved",
117
+ message: "session ID resolved via Claude Code PID walk",
118
+ meta: { sessionId: claudeSessionId, method: "claude-pid-walk" },
119
+ });
120
+ return claudeSessionId;
121
+ }
122
+ // Fallback: UUID
123
+ const sessionId = (0, crypto_1.randomUUID)();
124
+ (0, runtime_1.emitNervesEvent)({
125
+ component: "daemon",
126
+ event: "daemon.session_id_resolved",
127
+ message: "session ID resolved via UUID fallback",
128
+ meta: { sessionId, method: "uuid-fallback" },
129
+ });
130
+ return sessionId;
131
+ }