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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (440) hide show
  1. package/README.md +229 -183
  2. package/RepairGuide.ouro/agent.json +5 -0
  3. package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
  4. package/RepairGuide.ouro/psyche/SOUL.md +55 -0
  5. package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
  6. package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
  7. package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
  8. package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
  9. package/SerpentGuide.ouro/agent.json +83 -0
  10. package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
  12. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  13. package/assets/ouroboros.png +0 -0
  14. package/changelog.json +4182 -0
  15. package/dist/arc/attention-types.js +8 -0
  16. package/dist/arc/cares.js +140 -0
  17. package/dist/arc/episodes.js +117 -0
  18. package/dist/arc/intentions.js +133 -0
  19. package/dist/arc/json-store.js +117 -0
  20. package/dist/arc/obligations.js +254 -0
  21. package/dist/arc/packets.js +193 -0
  22. package/dist/arc/presence.js +185 -0
  23. package/dist/arc/task-lifecycle.js +65 -0
  24. package/dist/heart/active-work.js +989 -0
  25. package/dist/heart/agent-entry.js +69 -3
  26. package/dist/heart/attachments/image-normalize.js +194 -0
  27. package/dist/heart/attachments/materialize.js +97 -0
  28. package/dist/heart/attachments/originals.js +88 -0
  29. package/dist/heart/attachments/render.js +29 -0
  30. package/dist/heart/attachments/sources/adapter.js +2 -0
  31. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  32. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  33. package/dist/heart/attachments/sources/index.js +16 -0
  34. package/dist/heart/attachments/store.js +103 -0
  35. package/dist/heart/attachments/types.js +93 -0
  36. package/dist/heart/auth/auth-flow.js +479 -0
  37. package/dist/heart/awaiting/await-alert.js +146 -0
  38. package/dist/heart/awaiting/await-expiry.js +108 -0
  39. package/dist/heart/awaiting/await-loader.js +91 -0
  40. package/dist/heart/awaiting/await-parser.js +141 -0
  41. package/dist/heart/awaiting/await-runtime-state.js +97 -0
  42. package/dist/heart/awaiting/await-scheduler.js +377 -0
  43. package/dist/heart/background-operations.js +281 -0
  44. package/dist/heart/bridges/manager.js +358 -0
  45. package/dist/heart/bridges/state-machine.js +135 -0
  46. package/dist/heart/bridges/store.js +123 -0
  47. package/dist/heart/bundle-state.js +168 -0
  48. package/dist/heart/commitments.js +142 -0
  49. package/dist/heart/config-registry.js +322 -0
  50. package/dist/heart/config.js +164 -135
  51. package/dist/heart/core.js +1069 -260
  52. package/dist/heart/cross-chat-delivery.js +131 -0
  53. package/dist/heart/daemon/agent-config-check.js +419 -0
  54. package/dist/heart/daemon/agent-discovery.js +180 -0
  55. package/dist/heart/daemon/agent-service.js +522 -0
  56. package/dist/heart/daemon/agentic-repair.js +547 -0
  57. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  58. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  59. package/dist/heart/daemon/cadence.js +70 -0
  60. package/dist/heart/daemon/cli-defaults.js +776 -0
  61. package/dist/heart/daemon/cli-exec.js +7571 -0
  62. package/dist/heart/daemon/cli-help.js +498 -0
  63. package/dist/heart/daemon/cli-parse.js +1599 -0
  64. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  65. package/dist/heart/daemon/cli-render.js +763 -0
  66. package/dist/heart/daemon/cli-types.js +8 -0
  67. package/dist/heart/daemon/connect-bay.js +323 -0
  68. package/dist/heart/daemon/daemon-cli.js +30 -758
  69. package/dist/heart/daemon/daemon-entry.js +540 -8
  70. package/dist/heart/daemon/daemon-health.js +176 -0
  71. package/dist/heart/daemon/daemon-rollup.js +57 -0
  72. package/dist/heart/daemon/daemon-runtime-sync.js +287 -0
  73. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  74. package/dist/heart/daemon/daemon.js +972 -20
  75. package/dist/heart/daemon/dns-workflow.js +394 -0
  76. package/dist/heart/daemon/doctor-types.js +8 -0
  77. package/dist/heart/daemon/doctor.js +873 -0
  78. package/dist/heart/daemon/health-monitor.js +122 -1
  79. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  80. package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
  81. package/dist/heart/daemon/http-health-probe.js +80 -0
  82. package/dist/heart/daemon/human-command-screens.js +234 -0
  83. package/dist/heart/daemon/human-readiness.js +114 -0
  84. package/dist/heart/daemon/inner-status.js +89 -0
  85. package/dist/heart/daemon/interactive-repair.js +394 -0
  86. package/dist/heart/daemon/launchd.js +188 -0
  87. package/dist/heart/daemon/log-tailer.js +82 -12
  88. package/dist/heart/daemon/logs-prune.js +110 -0
  89. package/dist/heart/daemon/mcp-canary.js +297 -0
  90. package/dist/heart/daemon/message-router.js +17 -8
  91. package/dist/heart/daemon/os-cron-deps.js +135 -0
  92. package/dist/heart/daemon/os-cron.js +14 -12
  93. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  94. package/dist/heart/daemon/ouro-entry.js +3 -1
  95. package/dist/heart/daemon/process-manager.js +381 -26
  96. package/dist/heart/daemon/provider-discovery.js +137 -0
  97. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  98. package/dist/heart/daemon/pulse.js +475 -0
  99. package/dist/heart/daemon/readiness-repair.js +365 -0
  100. package/dist/heart/daemon/run-hooks.js +39 -0
  101. package/dist/heart/daemon/runtime-logging.js +67 -16
  102. package/dist/heart/daemon/runtime-metadata.js +191 -0
  103. package/dist/heart/daemon/runtime-mode.js +67 -0
  104. package/dist/heart/daemon/safe-mode.js +161 -0
  105. package/dist/heart/daemon/sense-manager.js +731 -0
  106. package/dist/heart/daemon/session-id-resolver.js +131 -0
  107. package/dist/heart/daemon/skill-management-installer.js +94 -0
  108. package/dist/heart/daemon/socket-client.js +349 -0
  109. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  110. package/dist/heart/daemon/startup-tui.js +330 -0
  111. package/dist/heart/daemon/task-scheduler.js +3 -25
  112. package/dist/heart/daemon/terminal-ui.js +499 -0
  113. package/dist/heart/daemon/thoughts.js +524 -0
  114. package/dist/heart/daemon/up-progress.js +366 -0
  115. package/dist/heart/daemon/vault-items.js +56 -0
  116. package/dist/heart/delegation.js +62 -0
  117. package/dist/heart/habits/habit-migration.js +189 -0
  118. package/dist/heart/habits/habit-parser.js +140 -0
  119. package/dist/heart/habits/habit-runtime-state.js +100 -0
  120. package/dist/heart/habits/habit-scheduler.js +372 -0
  121. package/dist/heart/{daemon → hatch}/hatch-animation.js +10 -3
  122. package/dist/heart/{daemon → hatch}/hatch-flow.js +34 -136
  123. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  124. package/dist/heart/hatch/specialist-orchestrator.js +129 -0
  125. package/dist/heart/hatch/specialist-prompt.js +102 -0
  126. package/dist/heart/hatch/specialist-tools.js +306 -0
  127. package/dist/heart/identity.js +281 -67
  128. package/dist/heart/kept-notes.js +357 -0
  129. package/dist/heart/kicks.js +2 -20
  130. package/dist/heart/machine-identity.js +161 -0
  131. package/dist/heart/mail-import-discovery.js +353 -0
  132. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  133. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  134. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  135. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  136. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  137. package/dist/heart/mailbox/mailbox-http.js +99 -0
  138. package/dist/heart/mailbox/mailbox-read.js +31 -0
  139. package/dist/heart/mailbox/mailbox-types.js +27 -0
  140. package/dist/heart/mailbox/mailbox-view.js +195 -0
  141. package/dist/heart/mailbox/readers/agent-machine.js +382 -0
  142. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  143. package/dist/heart/mailbox/readers/mail.js +367 -0
  144. package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
  145. package/dist/heart/mailbox/readers/sessions.js +232 -0
  146. package/dist/heart/mailbox/readers/shared.js +111 -0
  147. package/dist/heart/mcp/mcp-server.js +656 -0
  148. package/dist/heart/migrate-config.js +100 -0
  149. package/dist/heart/model-capabilities.js +59 -0
  150. package/dist/heart/platform.js +81 -0
  151. package/dist/heart/progress-story.js +42 -0
  152. package/dist/heart/provider-attempt.js +134 -0
  153. package/dist/heart/provider-binding-resolver.js +267 -0
  154. package/dist/heart/provider-credentials.js +425 -0
  155. package/dist/heart/provider-failover.js +301 -0
  156. package/dist/heart/provider-models.js +81 -0
  157. package/dist/heart/provider-ping.js +262 -0
  158. package/dist/heart/provider-readiness-cache.js +40 -0
  159. package/dist/heart/provider-visibility.js +188 -0
  160. package/dist/heart/providers/anthropic-token.js +131 -0
  161. package/dist/heart/providers/anthropic.js +202 -50
  162. package/dist/heart/providers/azure.js +104 -13
  163. package/dist/heart/providers/error-classification.js +127 -0
  164. package/dist/heart/providers/github-copilot.js +145 -0
  165. package/dist/heart/providers/minimax-vlm.js +189 -0
  166. package/dist/heart/providers/minimax.js +29 -7
  167. package/dist/heart/providers/openai-codex.js +63 -39
  168. package/dist/heart/runtime-capability-check.js +170 -0
  169. package/dist/heart/runtime-credentials.js +367 -0
  170. package/dist/heart/runtime-cwd.js +87 -0
  171. package/dist/heart/sense-truth.js +70 -0
  172. package/dist/heart/session-activity.js +190 -0
  173. package/dist/heart/session-events.js +1149 -0
  174. package/dist/heart/session-playback-cli-main.js +5 -0
  175. package/dist/heart/session-playback-cli.js +36 -0
  176. package/dist/heart/session-playback.js +231 -0
  177. package/dist/heart/session-stats-cli-main.js +5 -0
  178. package/dist/heart/session-stats.js +182 -0
  179. package/dist/heart/session-transcript.js +243 -0
  180. package/dist/heart/start-of-turn-packet.js +345 -0
  181. package/dist/heart/streaming.js +129 -34
  182. package/dist/heart/sync-classification.js +176 -0
  183. package/dist/heart/sync.js +449 -0
  184. package/dist/heart/target-resolution.js +127 -0
  185. package/dist/heart/tempo.js +93 -0
  186. package/dist/heart/temporal-view.js +41 -0
  187. package/dist/heart/timeouts.js +101 -0
  188. package/dist/heart/tool-activity-callbacks.js +59 -0
  189. package/dist/heart/tool-description.js +143 -0
  190. package/dist/heart/tool-friction.js +55 -0
  191. package/dist/heart/tool-loop.js +200 -0
  192. package/dist/heart/turn-context.js +421 -0
  193. package/dist/heart/turn-coordinator.js +28 -0
  194. package/dist/heart/versioning/ouro-bot-global-installer.js +129 -0
  195. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  196. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  197. package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
  198. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  199. package/dist/heart/versioning/staged-restart.js +146 -0
  200. package/dist/heart/versioning/update-checker.js +116 -0
  201. package/dist/heart/versioning/update-hooks.js +142 -0
  202. package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
  203. package/dist/mailbox-ui/assets/index-B-461hes.js +61 -0
  204. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  205. package/dist/mailbox-ui/index.html +15 -0
  206. package/dist/mailroom/attention.js +167 -0
  207. package/dist/mailroom/autonomy.js +209 -0
  208. package/dist/mailroom/blob-store.js +700 -0
  209. package/dist/mailroom/body-cache.js +61 -0
  210. package/dist/mailroom/core.js +788 -0
  211. package/dist/mailroom/entry.js +160 -0
  212. package/dist/mailroom/file-store.js +457 -0
  213. package/dist/mailroom/mbox-import.js +393 -0
  214. package/dist/mailroom/migration.js +164 -0
  215. package/dist/mailroom/outbound.js +380 -0
  216. package/dist/mailroom/policy.js +263 -0
  217. package/dist/mailroom/reader.js +233 -0
  218. package/dist/mailroom/search-cache.js +268 -0
  219. package/dist/mailroom/search-relevance.js +319 -0
  220. package/dist/mailroom/smtp-ingress.js +176 -0
  221. package/dist/mailroom/source-state.js +176 -0
  222. package/dist/mailroom/thread.js +109 -0
  223. package/dist/mailroom/travel-extract.js +89 -0
  224. package/dist/mind/bundle-manifest.js +77 -1
  225. package/dist/mind/context.js +174 -94
  226. package/dist/mind/diary-integrity.js +60 -0
  227. package/dist/mind/{memory.js → diary.js} +84 -96
  228. package/dist/mind/embedding-provider.js +60 -0
  229. package/dist/mind/file-state.js +179 -0
  230. package/dist/mind/first-impressions.js +16 -2
  231. package/dist/mind/friends/channel.js +74 -0
  232. package/dist/mind/friends/group-context.js +144 -0
  233. package/dist/mind/friends/resolver.js +54 -2
  234. package/dist/mind/friends/store-file.js +58 -3
  235. package/dist/mind/friends/trust-explanation.js +74 -0
  236. package/dist/mind/friends/types.js +10 -2
  237. package/dist/mind/journal-index.js +161 -0
  238. package/dist/mind/note-search.js +268 -0
  239. package/dist/mind/obligation-steering.js +221 -0
  240. package/dist/mind/pending.js +76 -9
  241. package/dist/mind/phrases.js +1 -0
  242. package/dist/mind/prompt-refresh.js +3 -2
  243. package/dist/mind/prompt.js +1267 -130
  244. package/dist/mind/provenance-trust.js +26 -0
  245. package/dist/mind/scrutiny.js +173 -0
  246. package/dist/mind/token-estimate.js +8 -12
  247. package/dist/nerves/cli-logging.js +22 -3
  248. package/dist/nerves/coverage/audit-rules.js +15 -6
  249. package/dist/nerves/coverage/audit.js +28 -2
  250. package/dist/nerves/coverage/cli.js +1 -1
  251. package/dist/nerves/coverage/contract.js +5 -5
  252. package/dist/nerves/coverage/file-completeness.js +129 -5
  253. package/dist/nerves/coverage/run-artifacts.js +1 -1
  254. package/dist/nerves/event-buffer.js +111 -0
  255. package/dist/nerves/index.js +224 -4
  256. package/dist/nerves/observation.js +20 -0
  257. package/dist/nerves/redact.js +79 -0
  258. package/dist/nerves/review/cli-main.js +5 -0
  259. package/dist/nerves/review/cli.js +156 -0
  260. package/dist/nerves/review/core.js +152 -0
  261. package/dist/nerves/runtime.js +5 -1
  262. package/dist/repertoire/ado-client.js +17 -56
  263. package/dist/repertoire/ado-semantic.js +11 -10
  264. package/dist/repertoire/api-client.js +97 -0
  265. package/dist/repertoire/bitwarden-store.js +997 -0
  266. package/dist/repertoire/bundle-templates.js +72 -0
  267. package/dist/repertoire/bw-installer.js +180 -0
  268. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  269. package/dist/repertoire/coding/context-pack.js +330 -0
  270. package/dist/repertoire/coding/feedback.js +301 -0
  271. package/dist/repertoire/coding/index.js +4 -1
  272. package/dist/repertoire/coding/manager.js +220 -13
  273. package/dist/repertoire/coding/spawner.js +58 -12
  274. package/dist/repertoire/coding/tools.js +209 -7
  275. package/dist/repertoire/commerce-errors.js +109 -0
  276. package/dist/repertoire/commerce-self-test.js +156 -0
  277. package/dist/repertoire/credential-access.js +178 -0
  278. package/dist/repertoire/data/ado-endpoints.json +188 -0
  279. package/dist/repertoire/duffel-client.js +185 -0
  280. package/dist/repertoire/github-client.js +14 -55
  281. package/dist/repertoire/graph-client.js +11 -52
  282. package/dist/repertoire/guardrails.js +396 -0
  283. package/dist/repertoire/mcp-client.js +295 -0
  284. package/dist/repertoire/mcp-manager.js +362 -0
  285. package/dist/repertoire/mcp-tools.js +63 -0
  286. package/dist/repertoire/shell-sessions.js +133 -0
  287. package/dist/repertoire/skills.js +15 -24
  288. package/dist/repertoire/stripe-client.js +131 -0
  289. package/dist/repertoire/tasks/board.js +43 -5
  290. package/dist/repertoire/tasks/fix.js +182 -0
  291. package/dist/repertoire/tasks/index.js +39 -13
  292. package/dist/repertoire/tasks/lifecycle.js +2 -2
  293. package/dist/repertoire/tasks/parser.js +3 -2
  294. package/dist/repertoire/tasks/scanner.js +194 -37
  295. package/dist/repertoire/tasks/transitions.js +16 -79
  296. package/dist/repertoire/tool-results.js +29 -0
  297. package/dist/repertoire/tools-attachments.js +317 -0
  298. package/dist/repertoire/tools-awaiting.js +360 -0
  299. package/dist/repertoire/tools-base.js +56 -707
  300. package/dist/repertoire/tools-bluebubbles.js +94 -0
  301. package/dist/repertoire/tools-bridge.js +142 -0
  302. package/dist/repertoire/tools-bundle.js +984 -0
  303. package/dist/repertoire/tools-config.js +185 -0
  304. package/dist/repertoire/tools-continuity.js +248 -0
  305. package/dist/repertoire/tools-credential.js +381 -0
  306. package/dist/repertoire/tools-files.js +342 -0
  307. package/dist/repertoire/tools-flight.js +224 -0
  308. package/dist/repertoire/tools-flow.js +119 -0
  309. package/dist/repertoire/tools-github.js +1 -7
  310. package/dist/repertoire/tools-mail.js +1916 -0
  311. package/dist/repertoire/tools-notes.js +421 -0
  312. package/dist/repertoire/tools-obligations.js +142 -0
  313. package/dist/repertoire/tools-runtime.js +61 -0
  314. package/dist/repertoire/tools-session.js +809 -0
  315. package/dist/repertoire/tools-shell.js +120 -0
  316. package/dist/repertoire/tools-stripe.js +180 -0
  317. package/dist/repertoire/tools-surface.js +345 -0
  318. package/dist/repertoire/tools-teams.js +64 -61
  319. package/dist/repertoire/tools-travel.js +125 -0
  320. package/dist/repertoire/tools-trip.js +604 -0
  321. package/dist/repertoire/tools-user-profile.js +144 -0
  322. package/dist/repertoire/tools-vault.js +40 -0
  323. package/dist/repertoire/tools-voice.js +144 -0
  324. package/dist/repertoire/tools.js +154 -98
  325. package/dist/repertoire/travel-api-client.js +360 -0
  326. package/dist/repertoire/user-profile.js +131 -0
  327. package/dist/repertoire/vault-setup.js +246 -0
  328. package/dist/repertoire/vault-unlock.js +594 -0
  329. package/dist/scripts/claude-code-hook.js +41 -0
  330. package/dist/scripts/claude-code-stop-hook.js +47 -0
  331. package/dist/senses/attention-queue.js +116 -0
  332. package/dist/senses/await-turn-message.js +58 -0
  333. package/dist/senses/bluebubbles/active-turns.js +216 -0
  334. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  335. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  336. package/dist/senses/bluebubbles/client.js +685 -0
  337. package/dist/senses/bluebubbles/entry.js +77 -0
  338. package/dist/senses/bluebubbles/inbound-log.js +126 -0
  339. package/dist/senses/bluebubbles/index.js +2548 -0
  340. package/dist/senses/bluebubbles/media.js +389 -0
  341. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +45 -16
  342. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  343. package/dist/senses/bluebubbles/processed-log.js +133 -0
  344. package/dist/senses/bluebubbles/replay.js +137 -0
  345. package/dist/senses/bluebubbles/runtime-state.js +137 -0
  346. package/dist/senses/bluebubbles/session-cleanup.js +72 -0
  347. package/dist/senses/bluebubbles-meta-guard.js +40 -0
  348. package/dist/senses/cli/bracketed-paste.js +82 -0
  349. package/dist/senses/cli/image-paste.js +287 -0
  350. package/dist/senses/cli/image-ref-navigation.js +75 -0
  351. package/dist/senses/cli/ink-app.js +156 -0
  352. package/dist/senses/cli/inline-diff.js +64 -0
  353. package/dist/senses/cli/input-keys.js +174 -0
  354. package/dist/senses/cli/kill-ring.js +86 -0
  355. package/dist/senses/cli/message-list.js +51 -0
  356. package/dist/senses/cli/ouro-tui.js +607 -0
  357. package/dist/senses/cli/spinner-imperative.js +135 -0
  358. package/dist/senses/cli/spinner.js +101 -0
  359. package/dist/senses/cli/status-line.js +60 -0
  360. package/dist/senses/cli/streaming-markdown.js +526 -0
  361. package/dist/senses/cli/tool-display.js +85 -0
  362. package/dist/senses/cli/tool-render.js +85 -0
  363. package/dist/senses/cli/tui-store.js +240 -0
  364. package/dist/senses/cli/virtual-list.js +35 -0
  365. package/dist/senses/cli-entry.js +60 -8
  366. package/dist/senses/cli-layout.js +187 -0
  367. package/dist/senses/cli.js +777 -264
  368. package/dist/senses/commands.js +66 -3
  369. package/dist/senses/continuity.js +94 -0
  370. package/dist/senses/habit-turn-message.js +108 -0
  371. package/dist/senses/inner-dialog-worker.js +209 -16
  372. package/dist/senses/inner-dialog.js +682 -91
  373. package/dist/senses/mail-entry.js +66 -0
  374. package/dist/senses/mail.js +379 -0
  375. package/dist/senses/pipeline.js +751 -0
  376. package/dist/senses/proactive-content-guard.js +51 -0
  377. package/dist/senses/shared-turn.js +392 -0
  378. package/dist/senses/surface-tool.js +70 -0
  379. package/dist/senses/teams-entry.js +60 -8
  380. package/dist/senses/teams.js +925 -195
  381. package/dist/senses/trust-gate.js +207 -2
  382. package/dist/senses/voice/audio-playback.js +237 -0
  383. package/dist/senses/voice/audio-routing.js +119 -0
  384. package/dist/senses/voice/elevenlabs.js +202 -0
  385. package/dist/senses/voice/floor-control.js +431 -0
  386. package/dist/senses/voice/floor-controller.js +115 -0
  387. package/dist/senses/voice/golden-path.js +116 -0
  388. package/dist/senses/voice/index.js +29 -0
  389. package/dist/senses/voice/meeting.js +113 -0
  390. package/dist/senses/voice/outbound.js +190 -0
  391. package/dist/senses/voice/phone.js +33 -0
  392. package/dist/senses/voice/playback.js +139 -0
  393. package/dist/senses/voice/realtime-eval.js +496 -0
  394. package/dist/senses/voice/realtime-trace.js +531 -0
  395. package/dist/senses/voice/transcript.js +70 -0
  396. package/dist/senses/voice/turn.js +191 -0
  397. package/dist/senses/voice/twilio-phone-runtime.js +807 -0
  398. package/dist/senses/voice/twilio-phone.js +5077 -0
  399. package/dist/senses/voice/types.js +2 -0
  400. package/dist/senses/voice/whisper.js +161 -0
  401. package/dist/senses/voice-entry.js +81 -0
  402. package/dist/senses/voice-realtime-eval-command.js +99 -0
  403. package/dist/senses/voice-realtime-eval-entry.js +21 -0
  404. package/dist/senses/voice-twilio-entry.js +87 -0
  405. package/dist/trips/core.js +138 -0
  406. package/dist/trips/store.js +265 -0
  407. package/package.json +52 -7
  408. package/skills/agent-commerce.md +106 -0
  409. package/skills/browser-navigation.md +117 -0
  410. package/skills/commerce-setup-guide.md +116 -0
  411. package/skills/commerce-setup.md +84 -0
  412. package/skills/configure-dev-tools.md +99 -0
  413. package/skills/travel-planning.md +138 -0
  414. package/AdoptionSpecialist.ouro/agent.json +0 -20
  415. package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
  416. package/dist/heart/daemon/specialist-orchestrator.js +0 -160
  417. package/dist/heart/daemon/specialist-prompt.js +0 -40
  418. package/dist/heart/daemon/specialist-session.js +0 -142
  419. package/dist/heart/daemon/specialist-tools.js +0 -128
  420. package/dist/heart/daemon/subagent-installer.js +0 -125
  421. package/dist/inner-worker-entry.js +0 -4
  422. package/dist/mind/associative-recall.js +0 -197
  423. package/dist/senses/bluebubbles-client.js +0 -279
  424. package/dist/senses/bluebubbles-entry.js +0 -11
  425. package/dist/senses/bluebubbles.js +0 -332
  426. package/subagents/README.md +0 -73
  427. package/subagents/work-doer.md +0 -233
  428. package/subagents/work-merger.md +0 -624
  429. package/subagents/work-planner.md +0 -373
  430. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  431. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  432. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  433. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  434. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  435. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  436. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  437. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  438. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  439. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  440. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
@@ -0,0 +1,144 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.userProfileToolDefinitions = void 0;
4
+ const user_profile_1 = require("./user-profile");
5
+ const credential_access_1 = require("./credential-access");
6
+ const runtime_1 = require("../nerves/runtime");
7
+ function requireFamilyContext(ctx) {
8
+ if (!ctx?.context?.friend?.id) {
9
+ return "no friend context — cannot access user profile.";
10
+ }
11
+ if (ctx.context.friend.trustLevel !== "family") {
12
+ return "user profile access requires family trust level.";
13
+ }
14
+ return { friendId: ctx.context.friend.id };
15
+ }
16
+ exports.userProfileToolDefinitions = [
17
+ {
18
+ tool: {
19
+ type: "function",
20
+ function: {
21
+ name: "user_profile_store",
22
+ description: "Store or update user profile fields (legal name, DOB, passport, etc.) in the agent's vault. Fields are merged with any existing profile. Requires family trust level.",
23
+ parameters: {
24
+ type: "object",
25
+ properties: {
26
+ fields: {
27
+ type: "string",
28
+ description: 'JSON object with profile fields to store/update. Keys: legalName, dateOfBirth, gender, nationality, passport, driverLicense, email, phone, addresses, loyaltyPrograms, preferences, emergencyContact.',
29
+ },
30
+ },
31
+ required: ["fields"],
32
+ },
33
+ },
34
+ },
35
+ handler: async (args, ctx) => {
36
+ (0, runtime_1.emitNervesEvent)({
37
+ component: "repertoire",
38
+ event: "repertoire.tool_user_profile_store",
39
+ message: "user_profile_store invoked",
40
+ meta: { tool: "user_profile_store" },
41
+ });
42
+ const guard = requireFamilyContext(ctx);
43
+ if (typeof guard === "string")
44
+ return guard;
45
+ let fields;
46
+ try {
47
+ fields = JSON.parse(args.fields);
48
+ }
49
+ catch {
50
+ return "invalid JSON in fields parameter.";
51
+ }
52
+ try {
53
+ const store = (0, credential_access_1.getCredentialStore)();
54
+ await (0, user_profile_1.updateUserProfileFields)(guard.friendId, fields, store);
55
+ return `profile fields stored for ${guard.friendId}.`;
56
+ }
57
+ catch (err) {
58
+ /* v8 ignore next -- defensive: updateUserProfileFields errors are always Error instances @preserve */
59
+ return `failed to store profile: ${err instanceof Error ? err.message : String(err)}`;
60
+ }
61
+ },
62
+ summaryKeys: ["fields"],
63
+ },
64
+ {
65
+ tool: {
66
+ type: "function",
67
+ function: {
68
+ name: "user_profile_get",
69
+ description: "Retrieve a specific field from a user's profile. Only returns the requested field, never the full profile. Requires family trust level.",
70
+ parameters: {
71
+ type: "object",
72
+ properties: {
73
+ field: {
74
+ type: "string",
75
+ description: "The profile field to retrieve: legalName, dateOfBirth, gender, nationality, passport, driverLicense, email, phone, addresses, loyaltyPrograms, preferences, emergencyContact.",
76
+ },
77
+ },
78
+ required: ["field"],
79
+ },
80
+ },
81
+ },
82
+ handler: async (args, ctx) => {
83
+ (0, runtime_1.emitNervesEvent)({
84
+ component: "repertoire",
85
+ event: "repertoire.tool_user_profile_get",
86
+ message: "user_profile_get invoked",
87
+ meta: { tool: "user_profile_get", field: args.field },
88
+ });
89
+ const guard = requireFamilyContext(ctx);
90
+ if (typeof guard === "string")
91
+ return guard;
92
+ try {
93
+ const store = (0, credential_access_1.getCredentialStore)();
94
+ const value = await (0, user_profile_1.getUserProfileField)(guard.friendId, args.field, store);
95
+ if (value === undefined) {
96
+ return `field "${args.field}" is not set on the profile.`;
97
+ }
98
+ /* v8 ignore next -- platform-dependent v8 branch counting on ternary @preserve */
99
+ return typeof value === "string" ? value : JSON.stringify(value, null, 2);
100
+ }
101
+ catch (err) {
102
+ /* v8 ignore next -- defensive: getUserProfileField errors are always Error instances @preserve */
103
+ return `failed to get profile field: ${err instanceof Error ? err.message : String(err)}`;
104
+ }
105
+ },
106
+ summaryKeys: ["field"],
107
+ },
108
+ {
109
+ tool: {
110
+ type: "function",
111
+ function: {
112
+ name: "user_profile_delete",
113
+ description: "Delete a user's entire profile from the vault. This is irreversible. Requires family trust level.",
114
+ parameters: {
115
+ type: "object",
116
+ properties: {},
117
+ },
118
+ },
119
+ },
120
+ handler: async (_args, ctx) => {
121
+ (0, runtime_1.emitNervesEvent)({
122
+ component: "repertoire",
123
+ event: "repertoire.tool_user_profile_delete",
124
+ message: "user_profile_delete invoked",
125
+ meta: { tool: "user_profile_delete" },
126
+ });
127
+ const guard = requireFamilyContext(ctx);
128
+ if (typeof guard === "string")
129
+ return guard;
130
+ try {
131
+ const store = (0, credential_access_1.getCredentialStore)();
132
+ const deleted = await (0, user_profile_1.deleteUserProfile)(guard.friendId, store);
133
+ return deleted
134
+ ? `profile deleted for ${guard.friendId}.`
135
+ : `no profile found for ${guard.friendId}.`;
136
+ }
137
+ catch (err) {
138
+ /* v8 ignore next -- defensive: deleteUserProfile errors are always Error instances @preserve */
139
+ return `failed to delete profile: ${err instanceof Error ? err.message : String(err)}`;
140
+ }
141
+ },
142
+ summaryKeys: [],
143
+ },
144
+ ];
@@ -0,0 +1,40 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.vaultToolDefinitions = void 0;
4
+ const identity_1 = require("../heart/identity");
5
+ const runtime_1 = require("../nerves/runtime");
6
+ exports.vaultToolDefinitions = [
7
+ {
8
+ tool: {
9
+ type: "function",
10
+ function: {
11
+ name: "vault_setup",
12
+ description: "Set up the agent's credential vault. Creates a Bitwarden account on the configured Vaultwarden server. One-time operation — idempotent if vault already exists.",
13
+ parameters: {
14
+ type: "object",
15
+ properties: {},
16
+ },
17
+ },
18
+ },
19
+ handler: async () => {
20
+ const agentName = (0, identity_1.getAgentName)();
21
+ (0, runtime_1.emitNervesEvent)({
22
+ component: "repertoire",
23
+ event: "repertoire.vault_tool_call",
24
+ message: "vault_setup invoked",
25
+ meta: { tool: "vault_setup", agentName },
26
+ });
27
+ return [
28
+ "Vault setup is human-required.",
29
+ "",
30
+ "Why I cannot do it here:",
31
+ " Creating or unlocking a vault requires secret entry that must stay out of agent context.",
32
+ "",
33
+ "Do this in a terminal:",
34
+ ` ouro vault create --agent ${agentName}`,
35
+ ` ouro vault unlock --agent ${agentName}`,
36
+ ].join("\n");
37
+ },
38
+ summaryKeys: [],
39
+ },
40
+ ];
@@ -0,0 +1,144 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.voiceToolDefinitions = void 0;
4
+ const runtime_1 = require("../nerves/runtime");
5
+ exports.voiceToolDefinitions = [{
6
+ tool: {
7
+ type: "function",
8
+ function: {
9
+ name: "voice_end_call",
10
+ description: [
11
+ "request that the current live voice call end.",
12
+ "Use this only in a voice session when the conversation is actually finished or the caller asked to hang up.",
13
+ "After calling it, settle with a brief goodbye if you have not already spoken one; the transport will try to let already-sent speech finish before hanging up.",
14
+ ].join(" "),
15
+ parameters: {
16
+ type: "object",
17
+ properties: {
18
+ reason: {
19
+ type: "string",
20
+ description: "short reason for ending the call, such as caller said goodbye",
21
+ },
22
+ },
23
+ },
24
+ },
25
+ },
26
+ handler: async (args, ctx) => {
27
+ const reason = typeof args.reason === "string" && args.reason.trim().length > 0
28
+ ? args.reason.trim()
29
+ : undefined;
30
+ (0, runtime_1.emitNervesEvent)({
31
+ component: "tools",
32
+ event: "tool.voice_end_call_start",
33
+ message: "voice end-call tool requested",
34
+ meta: { hasActiveCall: String(Boolean(ctx?.voiceCall)), ...(reason ? { reasonLength: reason.length } : {}) },
35
+ });
36
+ if (!ctx?.voiceCall) {
37
+ (0, runtime_1.emitNervesEvent)({
38
+ component: "tools",
39
+ event: "tool.voice_end_call_end",
40
+ message: "voice end-call tool had no active call",
41
+ meta: { hasActiveCall: "false" },
42
+ });
43
+ return "no active voice call to end";
44
+ }
45
+ await ctx.voiceCall.requestEnd(reason);
46
+ (0, runtime_1.emitNervesEvent)({
47
+ component: "tools",
48
+ event: "tool.voice_end_call_end",
49
+ message: "voice end-call request accepted",
50
+ meta: { hasActiveCall: "true" },
51
+ });
52
+ return "(voice call ending)";
53
+ },
54
+ }, {
55
+ tool: {
56
+ type: "function",
57
+ function: {
58
+ name: "voice_play_audio",
59
+ description: [
60
+ "play a short non-speech audio clip into the current live voice call.",
61
+ "Use this when the caller asks to hear a tone, sample, clip, or other audio over the phone.",
62
+ "For a simple test use source=tone. For clips, use source=url or source=file with a short audio asset; the transport may cap duration.",
63
+ "Do not use it for your normal spoken response.",
64
+ ].join(" "),
65
+ parameters: {
66
+ type: "object",
67
+ properties: {
68
+ source: {
69
+ type: "string",
70
+ enum: ["tone", "url", "file"],
71
+ description: "audio source type; defaults to tone",
72
+ },
73
+ url: {
74
+ type: "string",
75
+ description: "http(s) URL of a short audio clip when source=url",
76
+ },
77
+ path: {
78
+ type: "string",
79
+ description: "path to a short local audio clip when source=file",
80
+ },
81
+ label: {
82
+ type: "string",
83
+ description: "short human-readable label for the audio",
84
+ },
85
+ toneHz: {
86
+ type: "number",
87
+ description: "tone frequency in Hz when source=tone",
88
+ },
89
+ durationMs: {
90
+ type: "number",
91
+ description: "requested duration in milliseconds; short clips only",
92
+ },
93
+ },
94
+ },
95
+ },
96
+ },
97
+ handler: async (args, ctx) => {
98
+ const source = args.source === "url" || args.source === "file" || args.source === "tone"
99
+ ? args.source
100
+ : "tone";
101
+ const durationMs = typeof args.durationMs === "string" && args.durationMs.trim()
102
+ ? Number(args.durationMs)
103
+ : undefined;
104
+ const toneHz = typeof args.toneHz === "string" && args.toneHz.trim()
105
+ ? Number(args.toneHz)
106
+ : undefined;
107
+ const request = {
108
+ source,
109
+ /* v8 ignore next -- sparse playback argument permutations are covered in the transport-level voice_play_audio tests @preserve */
110
+ ...(typeof args.url === "string" ? { url: args.url } : {}),
111
+ /* v8 ignore next -- sparse playback argument permutations are covered in the transport-level voice_play_audio tests @preserve */
112
+ ...(typeof args.path === "string" ? { path: args.path } : {}),
113
+ /* v8 ignore next -- sparse playback argument permutations are covered in the transport-level voice_play_audio tests @preserve */
114
+ ...(typeof args.label === "string" ? { label: args.label } : {}),
115
+ ...(Number.isFinite(toneHz) ? { toneHz } : {}),
116
+ ...(Number.isFinite(durationMs) ? { durationMs } : {}),
117
+ };
118
+ (0, runtime_1.emitNervesEvent)({
119
+ component: "tools",
120
+ event: "tool.voice_play_audio_start",
121
+ message: "voice play-audio tool requested",
122
+ meta: { hasActiveCallAudio: String(Boolean(ctx?.voiceCall?.playAudio)), source },
123
+ });
124
+ if (!ctx?.voiceCall?.playAudio) {
125
+ (0, runtime_1.emitNervesEvent)({
126
+ component: "tools",
127
+ event: "tool.voice_play_audio_end",
128
+ message: "voice play-audio tool had no active audio-capable call",
129
+ meta: { hasActiveCallAudio: "false", source },
130
+ });
131
+ return "no active voice call audio path";
132
+ }
133
+ const result = await ctx.voiceCall.playAudio(request);
134
+ (0, runtime_1.emitNervesEvent)({
135
+ component: "tools",
136
+ event: "tool.voice_play_audio_end",
137
+ message: "voice play-audio request accepted",
138
+ meta: { hasActiveCallAudio: "true", source, durationMs: String(result.durationMs) },
139
+ });
140
+ if (result.toolResult?.trim())
141
+ return result.toolResult.trim();
142
+ return `(played audio: ${result.label}, ${Math.round(result.durationMs)}ms)`;
143
+ },
144
+ }];
@@ -1,29 +1,54 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.teamsTools = exports.finalAnswerTool = exports.tools = void 0;
3
+ exports.surfaceToolDef = exports.speakTool = exports.restTool = exports.ponderTool = exports.observeTool = exports.settleTool = exports.tools = void 0;
4
+ exports.resetMcpDefinitions = resetMcpDefinitions;
4
5
  exports.getToolsForChannel = getToolsForChannel;
5
- exports.isConfirmationRequired = isConfirmationRequired;
6
6
  exports.execTool = execTool;
7
7
  exports.summarizeArgs = summarizeArgs;
8
+ exports.buildToolResultSummary = buildToolResultSummary;
8
9
  const tools_base_1 = require("./tools-base");
9
10
  const tools_teams_1 = require("./tools-teams");
11
+ const tools_bluebubbles_1 = require("./tools-bluebubbles");
10
12
  const ado_semantic_1 = require("./ado-semantic");
11
13
  const tools_github_1 = require("./tools-github");
14
+ const tools_bundle_1 = require("./tools-bundle");
12
15
  const runtime_1 = require("../nerves/runtime");
16
+ const guardrails_1 = require("./guardrails");
17
+ const identity_1 = require("../heart/identity");
18
+ const tools_surface_1 = require("./tools-surface");
19
+ const mcp_tools_1 = require("./mcp-tools");
20
+ const tools_voice_1 = require("./tools-voice");
21
+ function safeGetAgentRoot() {
22
+ try {
23
+ return (0, identity_1.getAgentRoot)();
24
+ }
25
+ catch {
26
+ return undefined;
27
+ }
28
+ }
13
29
  // Re-export types and constants used by the rest of the codebase
14
30
  var tools_base_2 = require("./tools-base");
15
31
  Object.defineProperty(exports, "tools", { enumerable: true, get: function () { return tools_base_2.tools; } });
16
- Object.defineProperty(exports, "finalAnswerTool", { enumerable: true, get: function () { return tools_base_2.finalAnswerTool; } });
17
- var tools_teams_2 = require("./tools-teams");
18
- Object.defineProperty(exports, "teamsTools", { enumerable: true, get: function () { return tools_teams_2.teamsTools; } });
32
+ Object.defineProperty(exports, "settleTool", { enumerable: true, get: function () { return tools_base_2.settleTool; } });
33
+ Object.defineProperty(exports, "observeTool", { enumerable: true, get: function () { return tools_base_2.observeTool; } });
34
+ Object.defineProperty(exports, "ponderTool", { enumerable: true, get: function () { return tools_base_2.ponderTool; } });
35
+ Object.defineProperty(exports, "restTool", { enumerable: true, get: function () { return tools_base_2.restTool; } });
36
+ Object.defineProperty(exports, "speakTool", { enumerable: true, get: function () { return tools_base_2.speakTool; } });
37
+ // Re-export surface tool schema for consumers (e.g. heart/core.ts)
38
+ var tools_surface_2 = require("./tools-surface");
39
+ Object.defineProperty(exports, "surfaceToolDef", { enumerable: true, get: function () { return tools_surface_2.surfaceToolDef; } });
19
40
  // All tool definitions in a single registry
20
- const allDefinitions = [...tools_base_1.baseToolDefinitions, ...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions];
21
- const REMOTE_BLOCKED_LOCAL_TOOLS = new Set(["shell", "read_file", "write_file", "git_commit", "gh_cli"]);
22
- function baseToolsForCapabilities(capabilities) {
23
- const isRemoteChannel = capabilities?.channel === "teams" || capabilities?.channel === "bluebubbles";
24
- if (!isRemoteChannel)
25
- return tools_base_1.tools;
26
- return tools_base_1.tools.filter((tool) => !REMOTE_BLOCKED_LOCAL_TOOLS.has(tool.function.name));
41
+ const allDefinitions = [...tools_base_1.baseToolDefinitions, ...tools_bluebubbles_1.bluebubblesToolDefinitions, ...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions, ...tools_bundle_1.bundleToolDefinitions, ...tools_voice_1.voiceToolDefinitions, tools_surface_1.surfaceToolDefinition];
42
+ // MCP tool definitions — populated each time getToolsForChannel() is called with an mcpManager.
43
+ // Kept separate from allDefinitions so execTool can find them.
44
+ let mcpDefinitions = [];
45
+ /** Exported for testing — reset the MCP definitions cache. */
46
+ function resetMcpDefinitions() {
47
+ mcpDefinitions = [];
48
+ }
49
+ function baseToolsForCapabilities() {
50
+ // Use baseToolDefinitions at call time so dynamically-added tools are included
51
+ return tools_base_1.baseToolDefinitions.map((d) => d.tool);
27
52
  }
28
53
  // Apply a single tool preference to a tool schema, returning a new object.
29
54
  function applyPreference(tool, pref) {
@@ -35,48 +60,82 @@ function applyPreference(tool, pref) {
35
60
  },
36
61
  };
37
62
  }
63
+ // Filter out tools whose requiredCapability is not in the provider's capability set.
64
+ // Uses baseToolDefinitions at call time so dynamically-added tools are included.
65
+ // Only base tools can have requiredCapability (integration tools do not).
66
+ function filterByCapability(toolList, providerCapabilities) {
67
+ return toolList.filter((tool) => {
68
+ const def = tools_base_1.baseToolDefinitions.find((d) => d.tool.function.name === tool.function.name);
69
+ if (!def?.requiredCapability)
70
+ return true;
71
+ return providerCapabilities?.has(def.requiredCapability) === true;
72
+ });
73
+ }
38
74
  // Return the appropriate tools list based on channel capabilities.
39
75
  // Base tools (no integration) are always included.
40
76
  // Teams/integration tools are included only if their integration is in availableIntegrations.
41
77
  // When toolPreferences is provided, matching preferences are appended to tool descriptions.
42
- function getToolsForChannel(capabilities, toolPreferences) {
43
- const baseTools = baseToolsForCapabilities(capabilities);
78
+ // When providerCapabilities is provided, tools with requiredCapability are filtered.
79
+ function getToolsForChannel(capabilities, toolPreferences, _context, providerCapabilities, mcpManager, _chatModel) {
80
+ const baseTools = baseToolsForCapabilities();
81
+ const bluebubblesTools = capabilities?.channel === "bluebubbles"
82
+ ? tools_bluebubbles_1.bluebubblesToolDefinitions.map((d) => d.tool)
83
+ : [];
84
+ const voiceTools = capabilities?.channel === "voice"
85
+ ? tools_voice_1.voiceToolDefinitions.map((d) => d.tool)
86
+ : [];
87
+ let result;
44
88
  if (!capabilities || capabilities.availableIntegrations.length === 0) {
45
- return baseTools;
89
+ result = [...baseTools, ...bluebubblesTools, ...voiceTools];
46
90
  }
47
- const available = new Set(capabilities.availableIntegrations);
48
- const integrationDefs = [...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions].filter((d) => d.integration && available.has(d.integration));
49
- if (!toolPreferences || Object.keys(toolPreferences).length === 0) {
50
- return [...baseTools, ...integrationDefs.map((d) => d.tool)];
91
+ else {
92
+ const available = new Set(capabilities.availableIntegrations);
93
+ const channelDefs = [...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions];
94
+ // Include tools whose integration is available, plus channel tools with no integration gate (e.g. teams_send_message)
95
+ const integrationDefs = channelDefs.filter((d) => d.integration ? available.has(d.integration) : capabilities.channel === "teams");
96
+ if (!toolPreferences || Object.keys(toolPreferences).length === 0) {
97
+ result = [...baseTools, ...bluebubblesTools, ...voiceTools, ...integrationDefs.map((d) => d.tool)];
98
+ }
99
+ else {
100
+ // Build a map of integration -> preference text for fast lookup
101
+ const prefMap = new Map();
102
+ for (const [key, value] of Object.entries(toolPreferences)) {
103
+ prefMap.set(key, value);
104
+ }
105
+ // Apply preferences to matching integration tools (new objects, no mutation)
106
+ // d.integration is guaranteed truthy -- integrationDefs are pre-filtered above
107
+ const enrichedIntegrationTools = integrationDefs.map((d) => {
108
+ const pref = prefMap.get(d.integration);
109
+ return pref ? applyPreference(d.tool, pref) : d.tool;
110
+ });
111
+ result = [...baseTools, ...bluebubblesTools, ...voiceTools, ...enrichedIntegrationTools];
112
+ }
51
113
  }
52
- // Build a map of integration -> preference text for fast lookup
53
- const prefMap = new Map();
54
- for (const [key, value] of Object.entries(toolPreferences)) {
55
- prefMap.set(key, value);
114
+ // Append first-class MCP tools when mcpManager is provided
115
+ if (mcpManager) {
116
+ mcpDefinitions = (0, mcp_tools_1.mcpToolsAsDefinitions)(mcpManager);
117
+ const mcpSchemas = mcpDefinitions.map((d) => d.tool);
118
+ result = [...result, ...mcpSchemas];
56
119
  }
57
- // Apply preferences to matching integration tools (new objects, no mutation)
58
- // d.integration is guaranteed truthy -- integrationDefs are pre-filtered above
59
- const enrichedIntegrationTools = integrationDefs.map((d) => {
60
- const pref = prefMap.get(d.integration);
61
- return pref ? applyPreference(d.tool, pref) : d.tool;
62
- });
63
- return [...baseTools, ...enrichedIntegrationTools];
120
+ return filterByCapability(result, providerCapabilities);
121
+ }
122
+ // Look up a tool definition from the combined registry (native + MCP).
123
+ function findDefinition(toolName) {
124
+ return allDefinitions.find((d) => d.tool.function.name === toolName)
125
+ ?? mcpDefinitions.find((d) => d.tool.function.name === toolName);
64
126
  }
65
- // Check whether a tool requires user confirmation before execution.
66
- // Reads from ToolDefinition.confirmationRequired instead of a separate Set.
67
- function isConfirmationRequired(toolName) {
68
- const def = allDefinitions.find((d) => d.tool.function.name === toolName);
69
- return def?.confirmationRequired === true;
127
+ function normalizeGuardArgs(_name, args) {
128
+ return args;
70
129
  }
71
130
  async function execTool(name, args, ctx) {
72
131
  (0, runtime_1.emitNervesEvent)({
73
132
  event: "tool.start",
74
133
  component: "tools",
75
134
  message: "tool execution started",
76
- meta: { name },
135
+ meta: { name, ...(name === "shell" && args.command ? { command: args.command } : {}) },
77
136
  });
78
- // Look up from combined registry
79
- const def = allDefinitions.find((d) => d.tool.function.name === name);
137
+ // Look up from combined registry (native + MCP)
138
+ const def = findDefinition(name);
80
139
  if (!def) {
81
140
  (0, runtime_1.emitNervesEvent)({
82
141
  level: "error",
@@ -87,17 +146,26 @@ async function execTool(name, args, ctx) {
87
146
  });
88
147
  return `unknown: ${name}`;
89
148
  }
90
- const isRemoteChannel = ctx?.context?.channel?.channel === "teams" || ctx?.context?.channel?.channel === "bluebubbles";
91
- if (isRemoteChannel && REMOTE_BLOCKED_LOCAL_TOOLS.has(name)) {
92
- const message = "I can't do that from here because I'm talking to multiple people in a shared remote channel, and local shell/file/git/gh operations could let conversations interfere with each other. Ask me for a remote-safe alternative (Graph/ADO/web), or run that operation from CLI.";
149
+ // Guardrail check: structural + trust-level
150
+ const mcpDef = mcpDefinitions.find((d) => d.tool.function.name === name);
151
+ const guardContext = {
152
+ readPaths: tools_base_1.editFileReadTracker,
153
+ trustLevel: ctx?.context?.friend?.trustLevel,
154
+ agentRoot: safeGetAgentRoot(),
155
+ ...(mcpDef?.mcpServer ? { mcpServerName: mcpDef.mcpServer } : {}),
156
+ ...(ctx?.context?.isGroupChat !== undefined ? { isGroupChat: (ctx?.context).isGroupChat } : {}),
157
+ };
158
+ const guardArgs = normalizeGuardArgs(name, args);
159
+ const guardResult = (0, guardrails_1.guardInvocation)(name, guardArgs, guardContext);
160
+ if (!guardResult.allowed) {
93
161
  (0, runtime_1.emitNervesEvent)({
94
162
  level: "warn",
95
- event: "tool.error",
163
+ event: "tool.guardrail_block",
96
164
  component: "tools",
97
- message: "blocked local tool in remote channel",
98
- meta: { name, channel: ctx?.context?.channel?.channel },
165
+ message: "guardrail blocked tool execution",
166
+ meta: { name, reason: guardResult.reason },
99
167
  });
100
- return message;
168
+ return guardResult.reason;
101
169
  }
102
170
  try {
103
171
  const result = await def.handler(args, ctx);
@@ -141,59 +209,47 @@ function summarizeUnknownArgs(args) {
141
209
  return summarizeKeyValues(args, keys);
142
210
  }
143
211
  function summarizeArgs(name, args) {
144
- // Check teams tools first
145
- const teamsSummary = (0, tools_teams_1.summarizeTeamsArgs)(name, args);
146
- if (teamsSummary !== undefined)
147
- return teamsSummary;
148
- // Check github tools
149
- const githubSummary = (0, tools_github_1.summarizeGithubArgs)(name, args);
150
- if (githubSummary !== undefined)
151
- return githubSummary;
152
- // Base tools
153
- if (name === "read_file" || name === "write_file")
154
- return summarizeKeyValues(args, ["path"]);
155
- if (name === "shell")
156
- return summarizeKeyValues(args, ["command"]);
157
- if (name === "list_directory")
158
- return summarizeKeyValues(args, ["path"]);
159
- if (name === "git_commit")
160
- return summarizeKeyValues(args, ["message"]);
161
- if (name === "gh_cli")
162
- return summarizeKeyValues(args, ["command"]);
163
- if (name === "load_skill")
164
- return summarizeKeyValues(args, ["name"]);
165
- if (name === "task_create")
166
- return summarizeKeyValues(args, ["title", "type", "category"]);
167
- if (name === "task_update_status")
168
- return summarizeKeyValues(args, ["name", "status"]);
169
- if (name === "task_board_status")
170
- return summarizeKeyValues(args, ["status"]);
171
- if (name === "task_board_action")
172
- return summarizeKeyValues(args, ["scope"]);
173
- if (name === "task_board" || name === "task_board_deps" || name === "task_board_sessions")
174
- return "";
175
- if (name === "coding_spawn")
176
- return summarizeKeyValues(args, ["runner", "workdir", "taskRef"]);
177
- if (name === "coding_status")
178
- return summarizeKeyValues(args, ["sessionId"]);
179
- if (name === "coding_send_input")
180
- return summarizeKeyValues(args, ["sessionId", "input"]);
181
- if (name === "coding_kill")
182
- return summarizeKeyValues(args, ["sessionId"]);
183
- if (name === "claude")
184
- return summarizeKeyValues(args, ["prompt"]);
185
- if (name === "web_search")
186
- return summarizeKeyValues(args, ["query"]);
187
- if (name === "memory_search")
188
- return summarizeKeyValues(args, ["query"]);
189
- if (name === "memory_save")
190
- return summarizeKeyValues(args, ["text", "about"]);
191
- if (name === "get_friend_note")
192
- return summarizeKeyValues(args, ["friendId"]);
193
- if (name === "save_friend_note") {
194
- return summarizeKeyValues(args, ["type", "key", "content"]);
212
+ const def = findDefinition(name);
213
+ if (def && def.summaryKeys !== undefined) {
214
+ return summarizeKeyValues(args, def.summaryKeys);
195
215
  }
196
- if (name === "ado_backlog_list")
197
- return summarizeKeyValues(args, ["organization", "project"]);
198
216
  return summarizeUnknownArgs(args);
199
217
  }
218
+ /**
219
+ * Build an enriched tool result summary for display on remote channels.
220
+ * For recognized tools, includes result-derived info (diff stats, exit codes, etc.).
221
+ * Falls back to arg-based summary for unrecognized tools.
222
+ */
223
+ function buildToolResultSummary(name, args, _result, success) {
224
+ /* v8 ignore start -- branches: ?? fallbacks and ternaries in tool summary formatting @preserve */
225
+ switch (name) {
226
+ case "edit_file": {
227
+ if (!success)
228
+ return summarizeArgs(name, args);
229
+ const addedLines = (args.new_string ?? "").split("\n").length;
230
+ const removedLines = (args.old_string ?? "").split("\n").length;
231
+ return `+${addedLines} -${removedLines} lines in ${args.path ?? "unknown"}`;
232
+ }
233
+ case "shell": {
234
+ const cmd = args.command ?? "?";
235
+ const exitCode = success ? 0 : 1;
236
+ return `$ ${cmd} (exit ${exitCode})`;
237
+ }
238
+ case "read_file":
239
+ return `path=${args.path ?? "unknown"}`;
240
+ case "write_file":
241
+ return `path=${args.path ?? "unknown"}`;
242
+ case "glob":
243
+ return `pattern=${args.pattern ?? "?"} ${args.cwd ? `cwd=${args.cwd}` : ""}`.trim();
244
+ case "grep":
245
+ return `pattern=${args.pattern ?? "?"} ${args.path ? `path=${args.path}` : ""}`.trim();
246
+ case "coding_spawn": {
247
+ const taskRef = args.taskRef ?? "unknown";
248
+ const status = success ? "spawned" : "failed";
249
+ return `${taskRef} -> ${status}`;
250
+ }
251
+ default:
252
+ return summarizeArgs(name, args);
253
+ }
254
+ /* v8 ignore stop */
255
+ }