@ouro.bot/cli 0.1.0-alpha.66 → 0.1.0-alpha.660

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 (449) 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 +4209 -13
  13. package/dist/a2a/card.js +56 -0
  14. package/dist/a2a/client.js +143 -0
  15. package/dist/a2a/config.js +50 -0
  16. package/dist/a2a/onboarding.js +111 -0
  17. package/dist/a2a/server.js +498 -0
  18. package/dist/a2a/task-store.js +69 -0
  19. package/dist/a2a/types.js +3 -0
  20. package/dist/arc/attention-types.js +8 -0
  21. package/dist/arc/cares.js +144 -0
  22. package/dist/arc/episodes.js +118 -0
  23. package/dist/arc/evolution.js +487 -0
  24. package/dist/arc/flight-recorder.js +369 -0
  25. package/dist/arc/intentions.js +134 -0
  26. package/dist/arc/json-store.js +117 -0
  27. package/dist/arc/obligations.js +292 -0
  28. package/dist/arc/packets.js +288 -0
  29. package/dist/arc/presence.js +185 -0
  30. package/dist/arc/task-lifecycle.js +57 -0
  31. package/dist/commerce/store.js +755 -0
  32. package/dist/commerce/types.js +3 -0
  33. package/dist/heart/active-work.js +860 -43
  34. package/dist/heart/agent-entry.js +69 -3
  35. package/dist/heart/attachments/image-normalize.js +194 -0
  36. package/dist/heart/attachments/materialize.js +97 -0
  37. package/dist/heart/attachments/originals.js +88 -0
  38. package/dist/heart/attachments/render.js +29 -0
  39. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  40. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  41. package/dist/heart/attachments/sources/index.js +16 -0
  42. package/dist/heart/attachments/store.js +103 -0
  43. package/dist/heart/attachments/types.js +93 -0
  44. package/dist/heart/auth/auth-flow.js +479 -0
  45. package/dist/heart/awaiting/await-alert.js +146 -0
  46. package/dist/heart/awaiting/await-expiry.js +108 -0
  47. package/dist/heart/awaiting/await-loader.js +91 -0
  48. package/dist/heart/awaiting/await-parser.js +141 -0
  49. package/dist/heart/awaiting/await-runtime-state.js +100 -0
  50. package/dist/heart/awaiting/await-scheduler.js +377 -0
  51. package/dist/heart/background-operations.js +281 -0
  52. package/dist/heart/bridges/manager.js +137 -17
  53. package/dist/heart/bridges/store.js +14 -2
  54. package/dist/heart/bundle-state.js +168 -0
  55. package/dist/heart/commitments.js +135 -0
  56. package/dist/heart/config-registry.js +331 -0
  57. package/dist/heart/config.js +118 -119
  58. package/dist/heart/context-loss-gauntlet.js +354 -0
  59. package/dist/heart/core.js +1123 -247
  60. package/dist/heart/cross-chat-delivery.js +3 -18
  61. package/dist/heart/daemon/agent-config-check.js +419 -0
  62. package/dist/heart/daemon/agent-discovery.js +102 -3
  63. package/dist/heart/daemon/agent-service.js +523 -0
  64. package/dist/heart/daemon/agentic-repair.js +547 -0
  65. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  66. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  67. package/dist/heart/daemon/cadence.js +70 -0
  68. package/dist/heart/daemon/cli-defaults.js +780 -0
  69. package/dist/heart/daemon/cli-desk.js +322 -0
  70. package/dist/heart/daemon/cli-exec.js +7767 -0
  71. package/dist/heart/daemon/cli-help.js +558 -0
  72. package/dist/heart/daemon/cli-parse.js +1688 -0
  73. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  74. package/dist/heart/daemon/cli-render.js +763 -0
  75. package/dist/heart/daemon/cli-types.js +8 -0
  76. package/dist/heart/daemon/connect-bay.js +323 -0
  77. package/dist/heart/daemon/daemon-cli.js +29 -1750
  78. package/dist/heart/daemon/daemon-entry.js +485 -2
  79. package/dist/heart/daemon/daemon-health.js +176 -0
  80. package/dist/heart/daemon/daemon-rollup.js +57 -0
  81. package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
  82. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  83. package/dist/heart/daemon/daemon.js +937 -74
  84. package/dist/heart/daemon/dns-workflow.js +394 -0
  85. package/dist/heart/daemon/doctor-types.js +8 -0
  86. package/dist/heart/daemon/doctor.js +873 -0
  87. package/dist/heart/daemon/health-monitor.js +122 -1
  88. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  89. package/dist/heart/daemon/hooks/bundle-meta.js +135 -1
  90. package/dist/heart/daemon/http-health-probe.js +80 -0
  91. package/dist/heart/daemon/human-command-screens.js +234 -0
  92. package/dist/heart/daemon/human-readiness.js +114 -0
  93. package/dist/heart/daemon/inner-status.js +78 -0
  94. package/dist/heart/daemon/interactive-repair.js +394 -0
  95. package/dist/heart/daemon/launchd.js +37 -8
  96. package/dist/heart/daemon/log-tailer.js +79 -10
  97. package/dist/heart/daemon/logs-prune.js +110 -0
  98. package/dist/heart/daemon/mcp-canary.js +297 -0
  99. package/dist/heart/daemon/message-router.js +6 -2
  100. package/dist/heart/daemon/migrate-to-desk.js +848 -0
  101. package/dist/heart/daemon/os-cron-deps.js +135 -0
  102. package/dist/heart/daemon/os-cron.js +14 -12
  103. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  104. package/dist/heart/daemon/ouro-entry.js +3 -1
  105. package/dist/heart/daemon/plugin-cli.js +432 -0
  106. package/dist/heart/daemon/process-manager.js +511 -40
  107. package/dist/heart/daemon/provider-discovery.js +137 -0
  108. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  109. package/dist/heart/daemon/pulse.js +475 -0
  110. package/dist/heart/daemon/readiness-repair.js +365 -0
  111. package/dist/heart/daemon/run-hooks.js +2 -0
  112. package/dist/heart/daemon/runtime-logging.js +35 -14
  113. package/dist/heart/daemon/runtime-metadata.js +2 -30
  114. package/dist/heart/daemon/safe-mode.js +161 -0
  115. package/dist/heart/daemon/sense-manager.js +564 -38
  116. package/dist/heart/daemon/session-id-resolver.js +131 -0
  117. package/dist/heart/daemon/skill-management-installer.js +1 -1
  118. package/dist/heart/daemon/socket-client.js +158 -11
  119. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  120. package/dist/heart/daemon/startup-tui.js +330 -0
  121. package/dist/heart/daemon/task-scheduler.js +117 -39
  122. package/dist/heart/daemon/terminal-ui.js +499 -0
  123. package/dist/heart/daemon/thoughts.js +229 -17
  124. package/dist/heart/daemon/up-progress.js +366 -0
  125. package/dist/heart/daemon/vault-items.js +56 -0
  126. package/dist/heart/delegation.js +1 -4
  127. package/dist/heart/habits/habit-migration.js +189 -0
  128. package/dist/heart/habits/habit-parser.js +203 -0
  129. package/dist/heart/habits/habit-runtime-state.js +100 -0
  130. package/dist/heart/habits/habit-scheduler.js +372 -0
  131. package/dist/heart/{daemon → hatch}/hatch-flow.js +40 -56
  132. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  133. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  134. package/dist/heart/{daemon → hatch}/specialist-tools.js +45 -18
  135. package/dist/heart/identity.js +174 -57
  136. package/dist/heart/kept-notes.js +289 -0
  137. package/dist/heart/kicks.js +1 -1
  138. package/dist/heart/machine-identity.js +161 -0
  139. package/dist/heart/mail-import-discovery.js +353 -0
  140. package/dist/heart/mailbox/mailbox-http-hooks.js +67 -0
  141. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  142. package/dist/heart/mailbox/mailbox-http-routes.js +250 -0
  143. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  144. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  145. package/dist/heart/mailbox/mailbox-http.js +99 -0
  146. package/dist/heart/mailbox/mailbox-read.js +32 -0
  147. package/dist/heart/mailbox/mailbox-types.js +27 -0
  148. package/dist/heart/mailbox/mailbox-view.js +197 -0
  149. package/dist/heart/mailbox/readers/agent-machine.js +418 -0
  150. package/dist/heart/mailbox/readers/continuity-readers.js +324 -0
  151. package/dist/heart/mailbox/readers/mail.js +375 -0
  152. package/dist/heart/mailbox/readers/runtime-readers.js +728 -0
  153. package/dist/heart/mailbox/readers/sessions.js +232 -0
  154. package/dist/heart/mailbox/readers/shared.js +111 -0
  155. package/dist/heart/mcp/mcp-server.js +696 -0
  156. package/dist/heart/migrate-config.js +100 -0
  157. package/dist/heart/model-capabilities.js +19 -0
  158. package/dist/heart/orientation-frame.js +217 -0
  159. package/dist/heart/platform.js +81 -0
  160. package/dist/heart/provider-attempt.js +134 -0
  161. package/dist/heart/provider-binding-resolver.js +272 -0
  162. package/dist/heart/provider-credentials.js +425 -0
  163. package/dist/heart/provider-failover.js +311 -0
  164. package/dist/heart/provider-models.js +81 -0
  165. package/dist/heart/provider-ping.js +262 -0
  166. package/dist/heart/provider-readiness-cache.js +40 -0
  167. package/dist/heart/provider-visibility.js +188 -0
  168. package/dist/heart/providers/anthropic-token.js +131 -0
  169. package/dist/heart/providers/anthropic.js +139 -52
  170. package/dist/heart/providers/azure.js +23 -11
  171. package/dist/heart/providers/error-classification.js +127 -0
  172. package/dist/heart/providers/github-copilot.js +145 -0
  173. package/dist/heart/providers/minimax-vlm.js +189 -0
  174. package/dist/heart/providers/minimax.js +26 -8
  175. package/dist/heart/providers/openai-codex-token.js +349 -0
  176. package/dist/heart/providers/openai-codex.js +55 -40
  177. package/dist/heart/runtime-capability-check.js +170 -0
  178. package/dist/heart/runtime-credentials.js +367 -0
  179. package/dist/heart/runtime-cwd.js +87 -0
  180. package/dist/heart/sense-truth.js +17 -4
  181. package/dist/heart/session-activity.js +48 -24
  182. package/dist/heart/session-events.js +1133 -0
  183. package/dist/heart/session-playback-cli-main.js +5 -0
  184. package/dist/heart/session-playback-cli.js +36 -0
  185. package/dist/heart/session-playback.js +231 -0
  186. package/dist/heart/session-stats-cli-main.js +5 -0
  187. package/dist/heart/session-stats.js +182 -0
  188. package/dist/heart/session-transcript.js +133 -0
  189. package/dist/heart/start-of-turn-packet.js +351 -0
  190. package/dist/heart/streaming.js +44 -27
  191. package/dist/heart/structured-output.js +196 -0
  192. package/dist/heart/sync-classification.js +176 -0
  193. package/dist/heart/sync.js +449 -0
  194. package/dist/heart/target-resolution.js +9 -5
  195. package/dist/heart/tempo.js +93 -0
  196. package/dist/heart/temporal-view.js +41 -0
  197. package/dist/heart/timeouts.js +101 -0
  198. package/dist/heart/tool-activity-callbacks.js +59 -0
  199. package/dist/heart/tool-description.js +155 -0
  200. package/dist/heart/tool-friction.js +55 -0
  201. package/dist/heart/tool-loop.js +200 -0
  202. package/dist/heart/turn-context.js +430 -0
  203. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  204. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  205. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  206. package/dist/heart/versioning/ouro-version-manager.js +409 -0
  207. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  208. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  209. package/dist/heart/versioning/update-hooks.js +154 -0
  210. package/dist/heart/work-card.js +386 -0
  211. package/dist/mailbox-ui/assets/index-B-V9vRQ0.js +61 -0
  212. package/dist/mailbox-ui/assets/index-BOZbGbkL.css +1 -0
  213. package/dist/mailbox-ui/index.html +15 -0
  214. package/dist/mailroom/attention.js +167 -0
  215. package/dist/mailroom/autonomy.js +209 -0
  216. package/dist/mailroom/blob-store.js +715 -0
  217. package/dist/mailroom/body-cache.js +61 -0
  218. package/dist/mailroom/core.js +788 -0
  219. package/dist/mailroom/entry.js +160 -0
  220. package/dist/mailroom/file-store.js +568 -0
  221. package/dist/mailroom/mbox-import.js +393 -0
  222. package/dist/mailroom/migration.js +164 -0
  223. package/dist/mailroom/outbound.js +380 -0
  224. package/dist/mailroom/policy.js +263 -0
  225. package/dist/mailroom/reader.js +233 -0
  226. package/dist/mailroom/search-cache.js +334 -0
  227. package/dist/mailroom/search-relevance.js +319 -0
  228. package/dist/mailroom/smtp-ingress.js +176 -0
  229. package/dist/mailroom/source-state.js +176 -0
  230. package/dist/mailroom/thread.js +109 -0
  231. package/dist/mailroom/travel-extract.js +89 -0
  232. package/dist/mind/bundle-manifest.js +21 -2
  233. package/dist/mind/context.js +250 -101
  234. package/dist/mind/desk-section.js +362 -0
  235. package/dist/mind/diary-integrity.js +60 -0
  236. package/dist/mind/{memory.js → diary.js} +68 -77
  237. package/dist/mind/embedding-provider.js +60 -0
  238. package/dist/mind/file-state.js +179 -0
  239. package/dist/mind/friends/channel.js +48 -0
  240. package/dist/mind/friends/resolver.js +67 -4
  241. package/dist/mind/friends/store-file.js +61 -4
  242. package/dist/mind/friends/types.js +2 -2
  243. package/dist/mind/{associative-recall.js → note-search.js} +47 -58
  244. package/dist/mind/obligation-steering.js +221 -0
  245. package/dist/mind/pending.js +6 -1
  246. package/dist/mind/prompt-refresh.js +3 -2
  247. package/dist/mind/prompt.js +1015 -140
  248. package/dist/mind/provenance-trust.js +26 -0
  249. package/dist/mind/record-paths.js +312 -0
  250. package/dist/mind/scrutiny.js +173 -0
  251. package/dist/nerves/cli-logging.js +7 -1
  252. package/dist/nerves/coverage/audit-rules.js +15 -6
  253. package/dist/nerves/coverage/audit.js +28 -2
  254. package/dist/nerves/coverage/cli.js +1 -1
  255. package/dist/nerves/coverage/contract.js +5 -5
  256. package/dist/nerves/coverage/file-completeness.js +139 -5
  257. package/dist/nerves/event-buffer.js +111 -0
  258. package/dist/nerves/index.js +224 -4
  259. package/dist/nerves/observation.js +20 -0
  260. package/dist/nerves/redact.js +79 -0
  261. package/dist/nerves/review/cli-main.js +5 -0
  262. package/dist/nerves/review/cli.js +156 -0
  263. package/dist/nerves/review/core.js +152 -0
  264. package/dist/nerves/runtime.js +5 -1
  265. package/dist/repertoire/ado-client.js +15 -56
  266. package/dist/repertoire/ado-semantic.js +16 -10
  267. package/dist/repertoire/api-client.js +97 -0
  268. package/dist/repertoire/bitwarden-store.js +1041 -0
  269. package/dist/repertoire/bundle-templates.js +71 -0
  270. package/dist/repertoire/bw-installer.js +180 -0
  271. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  272. package/dist/repertoire/coding/context-pack.js +331 -0
  273. package/dist/repertoire/coding/feedback.js +197 -30
  274. package/dist/repertoire/coding/manager.js +166 -10
  275. package/dist/repertoire/coding/spawner.js +55 -9
  276. package/dist/repertoire/coding/tools.js +219 -7
  277. package/dist/repertoire/commerce-errors.js +109 -0
  278. package/dist/repertoire/commerce-self-test.js +156 -0
  279. package/dist/repertoire/credential-access.js +178 -0
  280. package/dist/repertoire/desk/classifier.js +362 -0
  281. package/dist/repertoire/duffel-client.js +185 -0
  282. package/dist/repertoire/github-client.js +14 -55
  283. package/dist/repertoire/graph-client.js +11 -52
  284. package/dist/repertoire/guardrails.js +159 -25
  285. package/dist/repertoire/mcp-client.js +295 -0
  286. package/dist/repertoire/mcp-manager.js +434 -0
  287. package/dist/repertoire/mcp-tools.js +83 -0
  288. package/dist/repertoire/plugin-mcp.js +175 -0
  289. package/dist/repertoire/plugins.js +253 -0
  290. package/dist/repertoire/shell-sessions.js +133 -0
  291. package/dist/repertoire/skills.js +48 -4
  292. package/dist/repertoire/stripe-client.js +131 -0
  293. package/dist/repertoire/tool-results.js +29 -0
  294. package/dist/repertoire/tools-a2a.js +283 -0
  295. package/dist/repertoire/tools-attachments.js +317 -0
  296. package/dist/repertoire/tools-awaiting.js +372 -0
  297. package/dist/repertoire/tools-base.js +63 -1082
  298. package/dist/repertoire/tools-bluebubbles.js +2 -0
  299. package/dist/repertoire/tools-bridge.js +144 -0
  300. package/dist/repertoire/tools-bundle.js +993 -0
  301. package/dist/repertoire/tools-commerce.js +253 -0
  302. package/dist/repertoire/tools-config.js +186 -0
  303. package/dist/repertoire/tools-continuity.js +252 -0
  304. package/dist/repertoire/tools-credential.js +383 -0
  305. package/dist/repertoire/tools-evolution.js +527 -0
  306. package/dist/repertoire/tools-files.js +344 -0
  307. package/dist/repertoire/tools-flight.js +290 -0
  308. package/dist/repertoire/tools-flow.js +119 -0
  309. package/dist/repertoire/tools-github.js +3 -8
  310. package/dist/repertoire/tools-mail.js +1975 -0
  311. package/dist/repertoire/tools-notes.js +418 -0
  312. package/dist/repertoire/tools-obligations.js +143 -0
  313. package/dist/repertoire/tools-orientation.js +31 -0
  314. package/dist/repertoire/tools-record.js +469 -0
  315. package/dist/repertoire/tools-runtime.js +150 -0
  316. package/dist/repertoire/tools-session.js +766 -0
  317. package/dist/repertoire/tools-shell.js +120 -0
  318. package/dist/repertoire/tools-stripe.js +224 -0
  319. package/dist/repertoire/tools-surface.js +344 -0
  320. package/dist/repertoire/tools-teams.js +12 -39
  321. package/dist/repertoire/tools-travel.js +125 -0
  322. package/dist/repertoire/tools-trip.js +982 -0
  323. package/dist/repertoire/tools-user-profile.js +146 -0
  324. package/dist/repertoire/tools-vault.js +40 -0
  325. package/dist/repertoire/tools-voice.js +145 -0
  326. package/dist/repertoire/tools.js +243 -79
  327. package/dist/repertoire/travel-api-client.js +360 -0
  328. package/dist/repertoire/user-profile.js +131 -0
  329. package/dist/repertoire/vault-setup.js +246 -0
  330. package/dist/repertoire/vault-unlock.js +594 -0
  331. package/dist/scripts/claude-code-hook.js +41 -0
  332. package/dist/scripts/claude-code-stop-hook.js +47 -0
  333. package/dist/senses/a2a-entry.js +78 -0
  334. package/dist/senses/attention-queue.js +186 -0
  335. package/dist/senses/await-turn-message.js +58 -0
  336. package/dist/senses/bluebubbles/active-turns.js +216 -0
  337. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  338. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  339. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  340. package/dist/senses/bluebubbles/entry.js +77 -0
  341. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  342. package/dist/senses/bluebubbles/index.js +2737 -0
  343. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
  344. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  345. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  346. package/dist/senses/bluebubbles/processed-log.js +133 -0
  347. package/dist/senses/bluebubbles/replay.js +137 -0
  348. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
  349. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  350. package/dist/senses/bluebubbles-meta-guard.js +40 -0
  351. package/dist/senses/cli/bracketed-paste.js +82 -0
  352. package/dist/senses/cli/image-paste.js +287 -0
  353. package/dist/senses/cli/image-ref-navigation.js +75 -0
  354. package/dist/senses/cli/ink-app.js +156 -0
  355. package/dist/senses/cli/inline-diff.js +64 -0
  356. package/dist/senses/cli/input-keys.js +174 -0
  357. package/dist/senses/cli/kill-ring.js +86 -0
  358. package/dist/senses/cli/message-list.js +51 -0
  359. package/dist/senses/cli/ouro-tui.js +607 -0
  360. package/dist/senses/cli/spinner-imperative.js +135 -0
  361. package/dist/senses/cli/spinner.js +101 -0
  362. package/dist/senses/cli/status-line.js +60 -0
  363. package/dist/senses/cli/streaming-markdown.js +526 -0
  364. package/dist/senses/cli/tool-display.js +85 -0
  365. package/dist/senses/cli/tool-render.js +85 -0
  366. package/dist/senses/cli/tui-store.js +240 -0
  367. package/dist/senses/cli/virtual-list.js +35 -0
  368. package/dist/senses/cli-entry.js +60 -8
  369. package/dist/senses/cli-layout.js +100 -0
  370. package/dist/senses/cli.js +517 -204
  371. package/dist/senses/commands.js +66 -3
  372. package/dist/senses/habit-turn-message.js +122 -0
  373. package/dist/senses/inner-dialog-worker.js +303 -22
  374. package/dist/senses/inner-dialog.js +525 -41
  375. package/dist/senses/mail-entry.js +66 -0
  376. package/dist/senses/mail.js +379 -0
  377. package/dist/senses/pipeline.js +857 -180
  378. package/dist/senses/proactive-content-guard.js +51 -0
  379. package/dist/senses/shared-turn.js +419 -0
  380. package/dist/senses/surface-tool.js +108 -0
  381. package/dist/senses/teams-entry.js +60 -8
  382. package/dist/senses/teams.js +390 -98
  383. package/dist/senses/trust-gate.js +100 -5
  384. package/dist/senses/voice/audio-playback.js +237 -0
  385. package/dist/senses/voice/audio-routing.js +119 -0
  386. package/dist/senses/voice/elevenlabs.js +202 -0
  387. package/dist/senses/voice/floor-control.js +431 -0
  388. package/dist/senses/voice/floor-controller.js +115 -0
  389. package/dist/senses/voice/golden-path.js +116 -0
  390. package/dist/senses/voice/index.js +29 -0
  391. package/dist/senses/voice/meeting.js +113 -0
  392. package/dist/senses/voice/outbound.js +190 -0
  393. package/dist/senses/voice/phone.js +33 -0
  394. package/dist/senses/voice/playback.js +139 -0
  395. package/dist/senses/voice/realtime-eval.js +496 -0
  396. package/dist/senses/voice/realtime-trace.js +531 -0
  397. package/dist/senses/voice/transcript.js +70 -0
  398. package/dist/senses/voice/turn.js +191 -0
  399. package/dist/senses/voice/twilio-phone-runtime.js +807 -0
  400. package/dist/senses/voice/twilio-phone.js +5079 -0
  401. package/dist/senses/voice/types.js +2 -0
  402. package/dist/senses/voice/whisper.js +161 -0
  403. package/dist/senses/voice-entry.js +81 -0
  404. package/dist/senses/voice-realtime-eval-command.js +99 -0
  405. package/dist/senses/voice-realtime-eval-entry.js +21 -0
  406. package/dist/senses/voice-twilio-entry.js +87 -0
  407. package/dist/trips/core.js +138 -0
  408. package/dist/trips/store.js +265 -0
  409. package/dist/util/frontmatter.js +69 -0
  410. package/package.json +55 -12
  411. package/skills/agent-commerce.md +113 -0
  412. package/skills/browser-navigation.md +117 -0
  413. package/skills/commerce-setup-guide.md +116 -0
  414. package/skills/commerce-setup.md +84 -0
  415. package/skills/configure-dev-tools.md +99 -0
  416. package/skills/travel-planning.md +138 -0
  417. package/dist/heart/daemon/auth-flow.js +0 -351
  418. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  419. package/dist/heart/daemon/update-hooks.js +0 -138
  420. package/dist/heart/safe-workspace.js +0 -228
  421. package/dist/heart/session-recall.js +0 -116
  422. package/dist/repertoire/tasks/board.js +0 -134
  423. package/dist/repertoire/tasks/index.js +0 -224
  424. package/dist/repertoire/tasks/lifecycle.js +0 -80
  425. package/dist/repertoire/tasks/middleware.js +0 -65
  426. package/dist/repertoire/tasks/parser.js +0 -173
  427. package/dist/repertoire/tasks/scanner.js +0 -132
  428. package/dist/repertoire/tasks/transitions.js +0 -144
  429. package/dist/senses/bluebubbles-entry.js +0 -13
  430. package/dist/senses/bluebubbles.js +0 -1177
  431. package/dist/senses/debug-activity.js +0 -148
  432. package/subagents/README.md +0 -7
  433. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  434. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  435. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  436. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  437. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  438. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  439. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  440. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  441. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  442. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  443. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  444. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  445. /package/dist/{repertoire/tasks/types.js → heart/attachments/sources/adapter.js} +0 -0
  446. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  447. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  448. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  449. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,728 @@
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.readCodingDeep = readCodingDeep;
37
+ exports.readAttentionView = readAttentionView;
38
+ exports.readBridgeInventory = readBridgeInventory;
39
+ exports.readDaemonHealthDeep = readDaemonHealthDeep;
40
+ exports.readNotesView = readNotesView;
41
+ exports.readFriendView = readFriendView;
42
+ exports.readLogView = readLogView;
43
+ exports.readHabitView = readHabitView;
44
+ exports.readNeedsMeView = readNeedsMeView;
45
+ exports.readDeskPrefs = readDeskPrefs;
46
+ const fs = __importStar(require("fs"));
47
+ const path = __importStar(require("path"));
48
+ const runtime_1 = require("../../../nerves/runtime");
49
+ const habit_parser_1 = require("../../habits/habit-parser");
50
+ const habit_runtime_state_1 = require("../../habits/habit-runtime-state");
51
+ const identity_1 = require("../../identity");
52
+ const daemon_health_1 = require("../../daemon/daemon-health");
53
+ const shared_1 = require("./shared");
54
+ const agent_machine_1 = require("./agent-machine");
55
+ const sessions_1 = require("./sessions");
56
+ const record_paths_1 = require("../../../mind/record-paths");
57
+ const NOTES_VIEW_LIMIT = 20;
58
+ /* v8 ignore start — defensive parsing of on-disk JSON, fallback branches are safety nets */
59
+ function readCodingDeep(agentRoot) {
60
+ const stateFilePath = path.join(agentRoot, "state", "coding", "sessions.json");
61
+ if (!fs.existsSync(stateFilePath)) {
62
+ return { totalCount: 0, activeCount: 0, blockedCount: 0, items: [] };
63
+ }
64
+ let parsed;
65
+ try {
66
+ parsed = JSON.parse(fs.readFileSync(stateFilePath, "utf-8"));
67
+ }
68
+ catch {
69
+ return { totalCount: 0, activeCount: 0, blockedCount: 0, items: [] };
70
+ }
71
+ const items = Array.isArray(parsed.records)
72
+ ? parsed.records.flatMap((record) => {
73
+ const s = record?.session;
74
+ if (!s || typeof s.id !== "string" || typeof s.status !== "string")
75
+ return [];
76
+ const checkpoint = typeof s.checkpoint === "string" ? s.checkpoint
77
+ : typeof s.stderrTail === "string" && s.stderrTail.trim().length > 0 ? s.stderrTail.trim()
78
+ : typeof s.stdoutTail === "string" && s.stdoutTail.trim().length > 0 ? s.stdoutTail.trim()
79
+ : null;
80
+ const originSession = s.originSession;
81
+ const normalizedOrigin = originSession
82
+ && typeof originSession.friendId === "string"
83
+ && typeof originSession.channel === "string"
84
+ && typeof originSession.key === "string"
85
+ ? { friendId: originSession.friendId, channel: originSession.channel, key: originSession.key }
86
+ : null;
87
+ const failure = s.failure;
88
+ const normalizedFailure = failure && typeof failure === "object"
89
+ ? {
90
+ command: typeof failure.command === "string" ? failure.command : "",
91
+ args: Array.isArray(failure.args) ? failure.args.map(String) : [],
92
+ code: typeof failure.code === "number" ? failure.code : null,
93
+ signal: typeof failure.signal === "string" ? failure.signal : null,
94
+ stdoutTail: typeof failure.stdoutTail === "string" ? failure.stdoutTail : "",
95
+ stderrTail: typeof failure.stderrTail === "string" ? failure.stderrTail : "",
96
+ }
97
+ : null;
98
+ return [{
99
+ id: s.id,
100
+ runner: (typeof s.runner === "string" ? s.runner : "claude"),
101
+ status: s.status,
102
+ checkpoint,
103
+ taskRef: typeof s.taskRef === "string" ? s.taskRef : null,
104
+ workdir: typeof s.workdir === "string" ? s.workdir : "",
105
+ originSession: normalizedOrigin,
106
+ obligationId: typeof s.obligationId === "string" ? s.obligationId : null,
107
+ scopeFile: typeof s.scopeFile === "string" ? s.scopeFile : null,
108
+ stateFile: typeof s.stateFile === "string" ? s.stateFile : null,
109
+ artifactPath: typeof s.artifactPath === "string" ? s.artifactPath : null,
110
+ pid: typeof s.pid === "number" ? s.pid : null,
111
+ startedAt: typeof s.startedAt === "string" ? s.startedAt : "",
112
+ lastActivityAt: typeof s.lastActivityAt === "string" ? s.lastActivityAt : "",
113
+ endedAt: typeof s.endedAt === "string" ? s.endedAt : null,
114
+ restartCount: typeof s.restartCount === "number" ? s.restartCount : 0,
115
+ lastExitCode: typeof s.lastExitCode === "number" ? s.lastExitCode : null,
116
+ lastSignal: typeof s.lastSignal === "string" ? s.lastSignal : null,
117
+ stdoutTail: typeof s.stdoutTail === "string" ? s.stdoutTail : "",
118
+ stderrTail: typeof s.stderrTail === "string" ? s.stderrTail : "",
119
+ failure: normalizedFailure,
120
+ }];
121
+ })
122
+ : [];
123
+ return {
124
+ totalCount: items.length,
125
+ activeCount: items.filter((item) => shared_1.ACTIVE_CODING_STATUSES.has(item.status)).length,
126
+ blockedCount: items.filter((item) => shared_1.BLOCKED_CODING_STATUSES.has(item.status)).length,
127
+ items,
128
+ };
129
+ }
130
+ /* v8 ignore stop */
131
+ function scanPendingChannels(agentRoot) {
132
+ const pendingRoot = path.join(agentRoot, "state", "pending");
133
+ const channels = [];
134
+ for (const friendId of (0, shared_1.safeReaddir)(pendingRoot)) {
135
+ if (friendId === "self")
136
+ continue;
137
+ const friendDir = path.join(pendingRoot, friendId);
138
+ if (!(0, shared_1.safeIsDirectory)(friendDir))
139
+ continue;
140
+ for (const channel of (0, shared_1.safeReaddir)(friendDir)) {
141
+ const channelDir = path.join(friendDir, channel);
142
+ if (!(0, shared_1.safeIsDirectory)(channelDir))
143
+ continue;
144
+ for (const key of (0, shared_1.safeReaddir)(channelDir)) {
145
+ const keyDir = path.join(channelDir, key);
146
+ if (!(0, shared_1.safeIsDirectory)(keyDir))
147
+ continue;
148
+ const files = (0, shared_1.safeReaddir)(keyDir).filter((f) => f.endsWith(".json") || f.endsWith(".json.processing"));
149
+ if (files.length > 0) {
150
+ channels.push({ friendId, channel, key, messageCount: files.length });
151
+ }
152
+ }
153
+ }
154
+ }
155
+ return channels;
156
+ }
157
+ function readPendingMessagesNonDestructive(pendingDir) {
158
+ const files = (0, shared_1.safeReaddir)(pendingDir).filter((f) => f.endsWith(".json") || f.endsWith(".json.processing"));
159
+ const messages = [];
160
+ for (const file of files.sort()) {
161
+ try {
162
+ const raw = fs.readFileSync(path.join(pendingDir, file), "utf-8");
163
+ messages.push(JSON.parse(raw));
164
+ }
165
+ catch {
166
+ // skip unparseable pending messages
167
+ }
168
+ }
169
+ return messages;
170
+ }
171
+ function readAttentionView(agentName, options = {}) {
172
+ const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
173
+ const agentRoot = path.join(bundlesRoot, `${agentName}.ouro`);
174
+ const friendsDir = path.join(agentRoot, "friends");
175
+ const pendingChannels = scanPendingChannels(agentRoot);
176
+ const queueItems = [];
177
+ const pendingRoot = path.join(agentRoot, "state", "pending");
178
+ for (const pending of pendingChannels) {
179
+ const pendingDir = path.join(pendingRoot, pending.friendId, pending.channel, pending.key);
180
+ const messages = readPendingMessagesNonDestructive(pendingDir);
181
+ for (const msg of messages) {
182
+ const delegatedFrom = msg.delegatedFrom;
183
+ queueItems.push({
184
+ id: typeof msg.timestamp === "number" ? `${msg.timestamp}-${pending.friendId}` : `pending-${Date.now()}`,
185
+ friendId: pending.friendId,
186
+ friendName: (0, shared_1.resolveFriendName)(friendsDir, pending.friendId),
187
+ channel: pending.channel,
188
+ key: pending.key,
189
+ bridgeId: delegatedFrom && typeof delegatedFrom.bridgeId === "string" ? delegatedFrom.bridgeId : null,
190
+ delegatedContent: typeof msg.content === "string" ? msg.content : "",
191
+ obligationId: typeof msg.obligationId === "string" ? msg.obligationId : null,
192
+ source: "pending",
193
+ timestamp: typeof msg.timestamp === "number" ? msg.timestamp : 0,
194
+ });
195
+ }
196
+ }
197
+ queueItems.sort((a, b) => a.timestamp - b.timestamp);
198
+ const returnObligations = (0, agent_machine_1.readObligationSummary)(agentRoot).items;
199
+ (0, runtime_1.emitNervesEvent)({
200
+ component: "heart",
201
+ event: "heart.mailbox_attention_read",
202
+ message: "reading mailbox attention queue",
203
+ meta: { agentName, queueLength: queueItems.length, pendingChannelCount: pendingChannels.length },
204
+ });
205
+ return {
206
+ queueLength: queueItems.length,
207
+ queueItems,
208
+ pendingChannels,
209
+ returnObligations,
210
+ };
211
+ }
212
+ /* v8 ignore start — defensive parsing */
213
+ function readBridgeInventory(agentRoot) {
214
+ const bridgesDir = path.join(agentRoot, "state", "bridges");
215
+ const items = [];
216
+ for (const file of (0, shared_1.safeReaddir)(bridgesDir)) {
217
+ if (!file.endsWith(".json"))
218
+ continue;
219
+ try {
220
+ const raw = fs.readFileSync(path.join(bridgesDir, file), "utf-8");
221
+ const bridge = JSON.parse(raw);
222
+ if (typeof bridge.id !== "string")
223
+ continue;
224
+ const attachedSessions = Array.isArray(bridge.attachedSessions)
225
+ ? bridge.attachedSessions
226
+ .filter((s) => typeof s.friendId === "string")
227
+ .map((s) => ({
228
+ friendId: s.friendId,
229
+ channel: typeof s.channel === "string" ? s.channel : "",
230
+ key: typeof s.key === "string" ? s.key : "",
231
+ sessionPath: typeof s.sessionPath === "string" ? s.sessionPath : "",
232
+ snapshot: typeof s.snapshot === "string" ? s.snapshot : null,
233
+ }))
234
+ : [];
235
+ const taskLink = bridge.task;
236
+ const normalizedTask = taskLink && typeof taskLink === "object" && typeof taskLink.taskName === "string"
237
+ ? {
238
+ taskName: taskLink.taskName,
239
+ path: typeof taskLink.path === "string" ? taskLink.path : "",
240
+ mode: typeof taskLink.mode === "string" ? taskLink.mode : "bound",
241
+ boundAt: typeof taskLink.boundAt === "string" ? taskLink.boundAt : "",
242
+ }
243
+ : null;
244
+ items.push({
245
+ id: bridge.id,
246
+ objective: typeof bridge.objective === "string" ? bridge.objective : "",
247
+ summary: typeof bridge.summary === "string" ? bridge.summary : "",
248
+ lifecycle: typeof bridge.lifecycle === "string" ? bridge.lifecycle : "unknown",
249
+ runtime: typeof bridge.runtime === "string" ? bridge.runtime : "unknown",
250
+ createdAt: typeof bridge.createdAt === "string" ? bridge.createdAt : "",
251
+ updatedAt: typeof bridge.updatedAt === "string" ? bridge.updatedAt : "",
252
+ attachedSessions,
253
+ task: normalizedTask,
254
+ });
255
+ }
256
+ catch {
257
+ // skip unparseable bridge files
258
+ }
259
+ }
260
+ items.sort((a, b) => b.updatedAt.localeCompare(a.updatedAt));
261
+ const activeCount = items.filter((item) => item.lifecycle === "active").length;
262
+ return {
263
+ totalCount: items.length,
264
+ activeCount,
265
+ items,
266
+ };
267
+ }
268
+ /* v8 ignore stop */
269
+ /* v8 ignore start — defensive parsing */
270
+ function readDaemonHealthDeep(healthPath) {
271
+ const resolvedPath = healthPath ?? path.join(process.env.HOME ?? "", ".ouro-cli", "daemon-health.json");
272
+ try {
273
+ const raw = fs.readFileSync(resolvedPath, "utf-8");
274
+ const health = JSON.parse(raw);
275
+ return {
276
+ // Layer 1: tighten the parse so only post-Layer-1 vocabulary
277
+ // carries through. Stale cached files that still hold legacy
278
+ // string values like "ok" or "running" — written by an older
279
+ // daemon binary — fall back to "unknown" so downstream Mailbox
280
+ // consumers can detect the unparseable case explicitly.
281
+ status: (0, daemon_health_1.isDaemonStatus)(health.status) ? health.status : "unknown",
282
+ mode: typeof health.mode === "string" ? health.mode : "unknown",
283
+ pid: typeof health.pid === "number" ? health.pid : 0,
284
+ startedAt: typeof health.startedAt === "string" ? health.startedAt : "",
285
+ uptimeSeconds: typeof health.uptimeSeconds === "number" ? health.uptimeSeconds : 0,
286
+ safeMode: health.safeMode && typeof health.safeMode === "object"
287
+ ? {
288
+ active: health.safeMode.active === true,
289
+ reason: typeof health.safeMode.reason === "string" ? health.safeMode.reason : "",
290
+ enteredAt: typeof health.safeMode.enteredAt === "string" ? health.safeMode.enteredAt : "",
291
+ }
292
+ : null,
293
+ degradedComponents: Array.isArray(health.degraded)
294
+ ? health.degraded.map((c) => ({
295
+ component: typeof c.component === "string" ? c.component : "",
296
+ reason: typeof c.reason === "string" ? c.reason : "",
297
+ since: typeof c.since === "string" ? c.since : "",
298
+ }))
299
+ : [],
300
+ agentHealth: health.agents && typeof health.agents === "object"
301
+ ? Object.fromEntries(Object.entries(health.agents).map(([name, entry]) => [
302
+ name,
303
+ {
304
+ status: typeof entry.status === "string" ? entry.status : "unknown",
305
+ pid: typeof entry.pid === "number" ? entry.pid : null,
306
+ crashes: typeof entry.crashes === "number" ? entry.crashes : 0,
307
+ },
308
+ ]))
309
+ : {},
310
+ habitHealth: health.habits && typeof health.habits === "object"
311
+ ? Object.fromEntries(Object.entries(health.habits).map(([name, entry]) => [
312
+ name,
313
+ {
314
+ cronStatus: typeof entry.cronStatus === "string" ? entry.cronStatus : "unknown",
315
+ lastFired: typeof entry.lastFired === "string" ? entry.lastFired : null,
316
+ fallback: entry.fallback === true,
317
+ },
318
+ ]))
319
+ : {},
320
+ };
321
+ }
322
+ catch {
323
+ return null;
324
+ }
325
+ }
326
+ /* v8 ignore stop */
327
+ function readNotesView(agentRoot) {
328
+ const recordPaths = (0, record_paths_1.resolveDeskRecordPaths)(agentRoot);
329
+ const diaryEntries = [];
330
+ try {
331
+ const raw = fs.readFileSync(recordPaths.factsPath, "utf-8");
332
+ for (const line of raw.split("\n")) {
333
+ if (!line.trim())
334
+ continue;
335
+ try {
336
+ const entry = JSON.parse(line);
337
+ if (typeof entry.id === "string" && typeof entry.text === "string") {
338
+ diaryEntries.push({
339
+ id: entry.id,
340
+ text: entry.text,
341
+ source: typeof entry.source === "string" ? entry.source : "",
342
+ createdAt: typeof entry.createdAt === "string" ? entry.createdAt : "",
343
+ });
344
+ }
345
+ }
346
+ catch {
347
+ // skip unparseable lines
348
+ }
349
+ }
350
+ }
351
+ catch {
352
+ // no diary facts file
353
+ }
354
+ diaryEntries.sort((a, b) => b.createdAt.localeCompare(a.createdAt));
355
+ const canonicalNotes = readCanonicalNotes(recordPaths.notesRoot);
356
+ return {
357
+ diaryEntryCount: diaryEntries.length,
358
+ recentDiaryEntries: diaryEntries.slice(0, NOTES_VIEW_LIMIT),
359
+ canonicalNoteCount: canonicalNotes.length,
360
+ recentCanonicalNotes: canonicalNotes.slice(0, NOTES_VIEW_LIMIT),
361
+ };
362
+ }
363
+ function readCanonicalNotes(notesRoot) {
364
+ const notes = [];
365
+ for (const filename of (0, shared_1.safeReaddir)(notesRoot)) {
366
+ if (!filename.endsWith(".md"))
367
+ continue;
368
+ const filePath = path.join(notesRoot, filename);
369
+ if ((0, shared_1.safeIsDirectory)(filePath))
370
+ continue;
371
+ try {
372
+ const raw = fs.readFileSync(filePath, "utf-8");
373
+ const { frontmatter, body } = parseMarkdownFrontmatter(raw);
374
+ const writtenAt = safeCanonicalNoteTimestamp(frontmatter.created_at, filePath);
375
+ notes.push({
376
+ filename,
377
+ title: firstMarkdownHeading(body) ?? titleFromFilename(filename),
378
+ tags: Array.isArray(frontmatter.tags) ? frontmatter.tags : [],
379
+ preview: body.slice(0, 200),
380
+ writtenAt,
381
+ });
382
+ }
383
+ catch {
384
+ // skip unreadable notes
385
+ }
386
+ }
387
+ notes.sort((a, b) => b.writtenAt.localeCompare(a.writtenAt) || b.filename.localeCompare(a.filename));
388
+ return notes;
389
+ }
390
+ function safeCanonicalNoteTimestamp(value, filePath) {
391
+ const fallback = (0, shared_1.safeFileMtime)(filePath);
392
+ if (typeof value !== "string")
393
+ return fallback;
394
+ const trimmed = value.trim();
395
+ if (!trimmed)
396
+ return fallback;
397
+ return Number.isNaN(new Date(trimmed).getTime()) ? fallback : trimmed;
398
+ }
399
+ function parseMarkdownFrontmatter(raw) {
400
+ const lines = raw.split(/\r?\n/);
401
+ if (lines[0] !== "---")
402
+ return { frontmatter: {}, body: raw };
403
+ const endIndex = lines.findIndex((line, index) => index > 0 && line === "---");
404
+ if (endIndex === -1)
405
+ return { frontmatter: {}, body: raw };
406
+ return {
407
+ frontmatter: parseMinimalFrontmatter(lines.slice(1, endIndex)),
408
+ body: lines.slice(endIndex + 1).join("\n").replace(/^\n/, ""),
409
+ };
410
+ }
411
+ function parseMinimalFrontmatter(lines) {
412
+ const frontmatter = {};
413
+ for (let index = 0; index < lines.length; index += 1) {
414
+ const line = lines[index];
415
+ const match = line?.match(/^([A-Za-z0-9_-]+):\s*(.*)$/);
416
+ if (!match)
417
+ continue;
418
+ const key = match[1];
419
+ const rawValue = match[2];
420
+ if (key === "tags") {
421
+ frontmatter.tags = parseFrontmatterTags(rawValue, lines, index + 1);
422
+ continue;
423
+ }
424
+ frontmatter[key] = rawValue.trim();
425
+ }
426
+ return frontmatter;
427
+ }
428
+ function parseFrontmatterTags(rawValue, lines, startIndex) {
429
+ const trimmed = rawValue.trim();
430
+ if (trimmed.startsWith("[") && trimmed.endsWith("]")) {
431
+ try {
432
+ const parsed = JSON.parse(trimmed);
433
+ return parsed.filter((item) => typeof item === "string");
434
+ }
435
+ catch {
436
+ return trimmed.slice(1, -1).split(",").map((tag) => tag.trim().replace(/^['"]|['"]$/g, "")).filter(Boolean);
437
+ }
438
+ }
439
+ if (trimmed.length > 0)
440
+ return [trimmed];
441
+ const tags = [];
442
+ for (let index = startIndex; index < lines.length; index += 1) {
443
+ const itemMatch = lines[index]?.match(/^\s*-\s*(.+)$/);
444
+ if (!itemMatch)
445
+ break;
446
+ const tag = itemMatch[1].trim().replace(/^['"]|['"]$/g, "");
447
+ if (tag)
448
+ tags.push(tag);
449
+ }
450
+ return tags;
451
+ }
452
+ function firstMarkdownHeading(body) {
453
+ const heading = body
454
+ .split(/\r?\n/)
455
+ .map((line) => line.match(/^#\s+(.+)$/)?.[1]?.trim())
456
+ .find((title) => title && title.length > 0);
457
+ return heading ?? null;
458
+ }
459
+ function titleFromFilename(filename) {
460
+ const stem = filename.replace(/\.md$/i, "").replace(/^\d{4}-\d{2}-\d{2}-/, "");
461
+ return stem.replace(/[-_]+/g, " ").trim() || filename;
462
+ }
463
+ function readFriendView(agentName, options = {}) {
464
+ const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
465
+ const agentRoot = path.join(bundlesRoot, `${agentName}.ouro`);
466
+ const friendsDir = path.join(agentRoot, "friends");
467
+ const sessionsDir = path.join(agentRoot, "state", "sessions");
468
+ const friends = [];
469
+ for (const file of (0, shared_1.safeReaddir)(friendsDir)) {
470
+ if (!file.endsWith(".json"))
471
+ continue;
472
+ const friendId = file.slice(0, -5);
473
+ try {
474
+ const raw = fs.readFileSync(path.join(friendsDir, file), "utf-8");
475
+ const record = JSON.parse(raw);
476
+ const friendSessionsDir = path.join(sessionsDir, friendId);
477
+ const channels = new Set();
478
+ let sessionCount = 0;
479
+ let latestActivity = null;
480
+ for (const channel of (0, shared_1.safeReaddir)(friendSessionsDir)) {
481
+ const channelDir = path.join(friendSessionsDir, channel);
482
+ if (!(0, shared_1.safeIsDirectory)(channelDir))
483
+ continue;
484
+ for (const keyFile of (0, shared_1.safeReaddir)(channelDir)) {
485
+ if (!keyFile.endsWith(".json"))
486
+ continue;
487
+ channels.add(channel);
488
+ sessionCount++;
489
+ const mtime = (0, shared_1.safeFileMtime)(path.join(channelDir, keyFile));
490
+ if (mtime && (!latestActivity || mtime > latestActivity)) {
491
+ latestActivity = mtime;
492
+ }
493
+ }
494
+ }
495
+ friends.push({
496
+ friendId,
497
+ friendName: typeof record.name === "string" ? record.name : friendId,
498
+ totalTokens: typeof record.totalTokens === "number" ? record.totalTokens : 0,
499
+ sessionCount,
500
+ channels: [...channels].sort(),
501
+ lastActivityAt: latestActivity,
502
+ });
503
+ }
504
+ catch {
505
+ // skip unparseable friend records
506
+ }
507
+ }
508
+ friends.sort((a, b) => b.totalTokens - a.totalTokens);
509
+ return {
510
+ totalFriends: friends.length,
511
+ friends,
512
+ };
513
+ }
514
+ function readLogView(logPath, limit = 100) {
515
+ if (!logPath || !fs.existsSync(logPath)) {
516
+ return { logPath, totalLines: 0, entries: [] };
517
+ }
518
+ try {
519
+ const raw = fs.readFileSync(logPath, "utf-8");
520
+ const lines = raw.split("\n").filter((l) => l.trim().length > 0);
521
+ const totalLines = lines.length;
522
+ const recentLines = lines.slice(-limit);
523
+ const entries = [];
524
+ for (const line of recentLines) {
525
+ try {
526
+ const parsed = JSON.parse(line);
527
+ entries.push({
528
+ ts: typeof parsed.ts === "string" ? parsed.ts : "",
529
+ level: typeof parsed.level === "string" ? parsed.level : "info",
530
+ event: typeof parsed.event === "string" ? parsed.event : "",
531
+ component: typeof parsed.component === "string" ? parsed.component : "",
532
+ message: typeof parsed.message === "string" ? parsed.message : "",
533
+ trace_id: typeof parsed.trace_id === "string" ? parsed.trace_id : "",
534
+ meta: parsed.meta && typeof parsed.meta === "object" ? parsed.meta : {},
535
+ });
536
+ }
537
+ catch {
538
+ // skip unparseable log lines
539
+ }
540
+ }
541
+ return { logPath, totalLines, entries };
542
+ }
543
+ catch { /* v8 ignore next */
544
+ return { logPath, totalLines: 0, entries: [] };
545
+ }
546
+ }
547
+ function readHabitView(agentRoot, options = {}) {
548
+ const habitsDir = path.join(agentRoot, "habits");
549
+ const now = options.now?.() ?? new Date();
550
+ const items = [];
551
+ for (const file of (0, shared_1.safeReaddir)(habitsDir)) {
552
+ if (!file.endsWith(".md"))
553
+ continue;
554
+ try {
555
+ const filePath = path.join(habitsDir, file);
556
+ const raw = fs.readFileSync(filePath, "utf-8");
557
+ if (!/^---\r?\n[\s\S]*?\r?\n---/.test(raw))
558
+ continue;
559
+ const habit = (0, habit_runtime_state_1.applyHabitRuntimeState)(agentRoot, (0, habit_parser_1.parseHabitFile)(raw, filePath));
560
+ const cadenceMs = parseCadenceMs(habit.cadence);
561
+ let isOverdue = false;
562
+ let overdueMs = null;
563
+ if (habit.status === "active" && habit.lastRun && cadenceMs) {
564
+ const elapsed = now.getTime() - Date.parse(habit.lastRun);
565
+ if (elapsed > cadenceMs) {
566
+ isOverdue = true;
567
+ overdueMs = elapsed - cadenceMs;
568
+ }
569
+ }
570
+ items.push({
571
+ name: habit.name,
572
+ title: habit.title,
573
+ cadence: habit.cadence,
574
+ status: habit.status,
575
+ lastRun: habit.lastRun,
576
+ bodyExcerpt: (0, shared_1.truncateExcerpt)(habit.body, 120),
577
+ isDegraded: false,
578
+ degradedReason: null,
579
+ isOverdue,
580
+ overdueMs,
581
+ });
582
+ }
583
+ catch {
584
+ // skip unparseable habit files
585
+ }
586
+ }
587
+ items.sort((a, b) => {
588
+ if (a.isOverdue && !b.isOverdue)
589
+ return -1;
590
+ if (!a.isOverdue && b.isOverdue)
591
+ return 1;
592
+ return a.name.localeCompare(b.name);
593
+ });
594
+ return {
595
+ totalCount: items.length,
596
+ activeCount: items.filter((h) => h.status === "active").length,
597
+ pausedCount: items.filter((h) => h.status === "paused").length,
598
+ degradedCount: items.filter((h) => h.isDegraded).length,
599
+ overdueCount: items.filter((h) => h.isOverdue).length,
600
+ items,
601
+ };
602
+ }
603
+ function readNeedsMeView(agentName, options = {}) {
604
+ const bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
605
+ const now = options.now?.() ?? new Date();
606
+ const agentRoot = path.join(bundlesRoot, `${agentName}.ouro`);
607
+ const items = [];
608
+ const prefs = readDeskPrefs(agentRoot);
609
+ const dismissed = new Set(prefs.dismissedObligations);
610
+ const sessions = (0, sessions_1.readSessionInventory)(agentName, options);
611
+ for (const s of sessions.items) {
612
+ if (s.replyState === "needs-reply") {
613
+ items.push({
614
+ urgency: "owed-reply",
615
+ label: `${s.friendName} is waiting for a reply`,
616
+ detail: `via ${s.channel} · ${s.latestUserExcerpt ? (0, shared_1.truncateExcerpt)(s.latestUserExcerpt, 80) : ""}`,
617
+ ref: { tab: "sessions", focus: `${s.friendId}/${s.channel}/${s.key}` },
618
+ ageMs: now.getTime() - Date.parse(s.lastActivityAt),
619
+ });
620
+ }
621
+ }
622
+ const obligations = (0, agent_machine_1.readObligationSummary)(agentRoot);
623
+ for (const o of obligations.items) {
624
+ if (dismissed.has(o.id))
625
+ continue;
626
+ const ageMs = now.getTime() - Date.parse(o.updatedAt);
627
+ const isStale = ageMs > 24 * 60 * 60 * 1000;
628
+ const hasResult = o.currentSurface !== null;
629
+ const isOpen = o.status === "pending" || o.status === "investigating" || o.status === "waiting_for_merge" || o.status === "updating_runtime";
630
+ if (isOpen) {
631
+ items.push({
632
+ urgency: hasResult ? "return-ready" : isStale ? "stale-delegation" : "blocking-obligation",
633
+ label: (0, shared_1.truncateExcerpt)(o.content, 80) ?? o.id,
634
+ detail: hasResult ? `result ready — ${o.currentSurface.kind}: ${o.currentSurface.label}` : `${o.status}${o.nextAction ? ` · next: ${o.nextAction}` : ""}`,
635
+ ref: { tab: "work", focus: o.id },
636
+ ageMs,
637
+ });
638
+ }
639
+ }
640
+ const pendingChannels = scanPendingChannels(agentRoot);
641
+ for (const p of pendingChannels) {
642
+ const friendName = (0, shared_1.resolveFriendName)(path.join(agentRoot, "friends"), p.friendId);
643
+ items.push({
644
+ urgency: "stale-delegation",
645
+ label: `${p.messageCount} pending from ${friendName}`,
646
+ detail: `${p.channel}/${p.key}`,
647
+ ref: { tab: "connections" },
648
+ ageMs: null,
649
+ });
650
+ }
651
+ const habits = readHabitView(agentRoot, options);
652
+ for (const h of habits.items) {
653
+ if (h.isOverdue) {
654
+ items.push({
655
+ urgency: "overdue-habit",
656
+ label: `${h.title} is overdue`,
657
+ detail: `every ${h.cadence} · last ${h.lastRun}`,
658
+ ref: { tab: "inner" },
659
+ ageMs: h.overdueMs,
660
+ });
661
+ }
662
+ }
663
+ const urgencyOrder = {
664
+ "owed-reply": 0,
665
+ "blocking-obligation": 1,
666
+ "broken-return": 2,
667
+ "stale-delegation": 3,
668
+ "return-ready": 4,
669
+ "overdue-habit": 5,
670
+ };
671
+ items.sort((a, b) => urgencyOrder[a.urgency] - urgencyOrder[b.urgency]);
672
+ return { items };
673
+ }
674
+ /* v8 ignore start — defensive JSON parsing in desk prefs reader */
675
+ function readDeskPrefs(agentRoot) {
676
+ const prefsPath = path.join(agentRoot, "state", "mailbox-prefs.json");
677
+ const legacyPrefsPath = path.join(agentRoot, "state", "outlook-prefs.json");
678
+ const defaults = {
679
+ carrying: null,
680
+ statusLine: null,
681
+ tabOrder: null,
682
+ starredFriends: [],
683
+ pinnedConstellations: [],
684
+ dismissedObligations: [],
685
+ };
686
+ try {
687
+ const raw = fs.readFileSync(fs.existsSync(prefsPath) ? prefsPath : legacyPrefsPath, "utf-8");
688
+ const parsed = JSON.parse(raw);
689
+ return {
690
+ carrying: typeof parsed.carrying === "string" ? parsed.carrying : null,
691
+ statusLine: typeof parsed.statusLine === "string" ? parsed.statusLine : null,
692
+ tabOrder: Array.isArray(parsed.tabOrder) ? parsed.tabOrder.filter((t) => typeof t === "string") : null,
693
+ starredFriends: Array.isArray(parsed.starredFriends) ? parsed.starredFriends.filter((f) => typeof f === "string") : [],
694
+ pinnedConstellations: Array.isArray(parsed.pinnedConstellations)
695
+ ? parsed.pinnedConstellations.map((c) => ({
696
+ label: typeof c.label === "string" ? c.label : "",
697
+ friendIds: Array.isArray(c.friendIds) ? c.friendIds.filter((f) => typeof f === "string") : [],
698
+ taskRefs: Array.isArray(c.taskRefs) ? c.taskRefs.filter((t) => typeof t === "string") : [],
699
+ bridgeIds: Array.isArray(c.bridgeIds) ? c.bridgeIds.filter((b) => typeof b === "string") : [],
700
+ codingIds: Array.isArray(c.codingIds) ? c.codingIds.filter((c2) => typeof c2 === "string") : [],
701
+ }))
702
+ : [],
703
+ dismissedObligations: Array.isArray(parsed.dismissedObligations) ? parsed.dismissedObligations.filter((id) => typeof id === "string") : [],
704
+ };
705
+ }
706
+ catch {
707
+ return defaults;
708
+ }
709
+ }
710
+ /* v8 ignore stop */
711
+ function parseCadenceMs(cadence) {
712
+ if (!cadence)
713
+ return null;
714
+ const match = /^(\d+)\s*(m|min|h|hr|d|day)s?$/i.exec(cadence.trim());
715
+ if (!match)
716
+ return null;
717
+ const value = parseInt(match[1], 10);
718
+ const multipliers = {
719
+ m: 60 * 1000,
720
+ min: 60 * 1000,
721
+ h: 60 * 60 * 1000,
722
+ hr: 60 * 60 * 1000,
723
+ d: 24 * 60 * 60 * 1000,
724
+ day: 24 * 60 * 60 * 1000,
725
+ };
726
+ const unit = match[2].toLowerCase();
727
+ return value * multipliers[unit];
728
+ }