@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500

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 (380) hide show
  1. package/README.md +226 -183
  2. package/SerpentGuide.ouro/agent.json +82 -0
  3. package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
  5. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  6. package/assets/ouroboros.png +0 -0
  7. package/changelog.json +3418 -0
  8. package/dist/arc/attention-types.js +8 -0
  9. package/dist/arc/cares.js +140 -0
  10. package/dist/arc/episodes.js +117 -0
  11. package/dist/arc/intentions.js +133 -0
  12. package/dist/arc/json-store.js +117 -0
  13. package/dist/arc/obligations.js +237 -0
  14. package/dist/arc/packets.js +193 -0
  15. package/dist/arc/presence.js +185 -0
  16. package/dist/arc/task-lifecycle.js +65 -0
  17. package/dist/heart/active-work.js +989 -0
  18. package/dist/heart/agent-entry.js +58 -3
  19. package/dist/heart/attachments/image-normalize.js +194 -0
  20. package/dist/heart/attachments/materialize.js +97 -0
  21. package/dist/heart/attachments/originals.js +88 -0
  22. package/dist/heart/attachments/render.js +29 -0
  23. package/dist/heart/attachments/sources/adapter.js +2 -0
  24. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  25. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  26. package/dist/heart/attachments/sources/index.js +16 -0
  27. package/dist/heart/attachments/store.js +103 -0
  28. package/dist/heart/attachments/types.js +93 -0
  29. package/dist/heart/auth/auth-flow.js +426 -0
  30. package/dist/heart/background-operations.js +281 -0
  31. package/dist/heart/bridges/manager.js +358 -0
  32. package/dist/heart/bridges/state-machine.js +135 -0
  33. package/dist/heart/bridges/store.js +123 -0
  34. package/dist/heart/bundle-state.js +168 -0
  35. package/dist/heart/commitments.js +111 -0
  36. package/dist/heart/config-registry.js +304 -0
  37. package/dist/heart/config.js +193 -130
  38. package/dist/heart/core.js +1010 -261
  39. package/dist/heart/cross-chat-delivery.js +131 -0
  40. package/dist/heart/daemon/agent-config-check.js +490 -0
  41. package/dist/heart/daemon/agent-discovery.js +157 -0
  42. package/dist/heart/daemon/agent-service.js +360 -0
  43. package/dist/heart/daemon/agentic-repair.js +216 -0
  44. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  45. package/dist/heart/daemon/cadence.js +70 -0
  46. package/dist/heart/daemon/cli-defaults.js +640 -0
  47. package/dist/heart/daemon/cli-exec.js +7239 -0
  48. package/dist/heart/daemon/cli-help.js +493 -0
  49. package/dist/heart/daemon/cli-parse.js +1533 -0
  50. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  51. package/dist/heart/daemon/cli-render.js +561 -0
  52. package/dist/heart/daemon/cli-types.js +8 -0
  53. package/dist/heart/daemon/connect-bay.js +323 -0
  54. package/dist/heart/daemon/daemon-cli.js +30 -697
  55. package/dist/heart/daemon/daemon-entry.js +359 -8
  56. package/dist/heart/daemon/daemon-health.js +141 -0
  57. package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
  58. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  59. package/dist/heart/daemon/daemon.js +813 -19
  60. package/dist/heart/daemon/dns-workflow.js +394 -0
  61. package/dist/heart/daemon/doctor-types.js +8 -0
  62. package/dist/heart/daemon/doctor.js +615 -0
  63. package/dist/heart/daemon/health-monitor.js +92 -1
  64. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  65. package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
  66. package/dist/heart/daemon/http-health-probe.js +80 -0
  67. package/dist/heart/daemon/human-command-screens.js +234 -0
  68. package/dist/heart/daemon/human-readiness.js +114 -0
  69. package/dist/heart/daemon/inner-status.js +89 -0
  70. package/dist/heart/daemon/interactive-repair.js +394 -0
  71. package/dist/heart/daemon/launchd.js +171 -0
  72. package/dist/heart/daemon/log-tailer.js +82 -12
  73. package/dist/heart/daemon/logs-prune.js +110 -0
  74. package/dist/heart/daemon/message-router.js +17 -8
  75. package/dist/heart/daemon/os-cron-deps.js +134 -0
  76. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  77. package/dist/heart/daemon/ouro-entry.js +3 -1
  78. package/dist/heart/daemon/process-manager.js +215 -1
  79. package/dist/heart/daemon/provider-discovery.js +137 -0
  80. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  81. package/dist/heart/daemon/pulse.js +475 -0
  82. package/dist/heart/daemon/readiness-repair.js +365 -0
  83. package/dist/heart/daemon/run-hooks.js +39 -0
  84. package/dist/heart/daemon/runtime-logging.js +67 -16
  85. package/dist/heart/daemon/runtime-metadata.js +191 -0
  86. package/dist/heart/daemon/runtime-mode.js +67 -0
  87. package/dist/heart/daemon/safe-mode.js +161 -0
  88. package/dist/heart/daemon/sense-manager.js +431 -0
  89. package/dist/heart/daemon/session-id-resolver.js +131 -0
  90. package/dist/heart/daemon/skill-management-installer.js +94 -0
  91. package/dist/heart/daemon/socket-client.js +307 -0
  92. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  93. package/dist/heart/daemon/startup-tui.js +264 -0
  94. package/dist/heart/daemon/task-scheduler.js +3 -25
  95. package/dist/heart/daemon/terminal-ui.js +499 -0
  96. package/dist/heart/daemon/thoughts.js +524 -0
  97. package/dist/heart/daemon/up-progress.js +366 -0
  98. package/dist/heart/daemon/vault-items.js +56 -0
  99. package/dist/heart/delegation.js +62 -0
  100. package/dist/heart/habits/habit-migration.js +189 -0
  101. package/dist/heart/habits/habit-parser.js +140 -0
  102. package/dist/heart/habits/habit-runtime-state.js +100 -0
  103. package/dist/heart/habits/habit-scheduler.js +372 -0
  104. package/dist/heart/{daemon → hatch}/hatch-animation.js +10 -3
  105. package/dist/heart/{daemon → hatch}/hatch-flow.js +54 -136
  106. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  107. package/dist/heart/hatch/specialist-orchestrator.js +129 -0
  108. package/dist/heart/hatch/specialist-prompt.js +102 -0
  109. package/dist/heart/hatch/specialist-tools.js +306 -0
  110. package/dist/heart/identity.js +274 -61
  111. package/dist/heart/kept-notes.js +357 -0
  112. package/dist/heart/kicks.js +2 -20
  113. package/dist/heart/machine-identity.js +161 -0
  114. package/dist/heart/mail-import-discovery.js +353 -0
  115. package/dist/heart/mcp/mcp-server.js +653 -0
  116. package/dist/heart/migrate-config.js +100 -0
  117. package/dist/heart/model-capabilities.js +59 -0
  118. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  119. package/dist/heart/outlook/outlook-http-response.js +7 -0
  120. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  121. package/dist/heart/outlook/outlook-http-static.js +103 -0
  122. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  123. package/dist/heart/outlook/outlook-http.js +99 -0
  124. package/dist/heart/outlook/outlook-read.js +31 -0
  125. package/dist/heart/outlook/outlook-types.js +27 -0
  126. package/dist/heart/outlook/outlook-view.js +195 -0
  127. package/dist/heart/outlook/readers/agent-machine.js +382 -0
  128. package/dist/heart/outlook/readers/continuity-readers.js +336 -0
  129. package/dist/heart/outlook/readers/mail.js +362 -0
  130. package/dist/heart/outlook/readers/runtime-readers.js +644 -0
  131. package/dist/heart/outlook/readers/sessions.js +232 -0
  132. package/dist/heart/outlook/readers/shared.js +111 -0
  133. package/dist/heart/platform.js +81 -0
  134. package/dist/heart/progress-story.js +42 -0
  135. package/dist/heart/provider-attempt.js +134 -0
  136. package/dist/heart/provider-binding-resolver.js +255 -0
  137. package/dist/heart/provider-credentials.js +424 -0
  138. package/dist/heart/provider-failover.js +301 -0
  139. package/dist/heart/provider-models.js +81 -0
  140. package/dist/heart/provider-ping.js +262 -0
  141. package/dist/heart/provider-state.js +216 -0
  142. package/dist/heart/provider-visibility.js +188 -0
  143. package/dist/heart/providers/anthropic-token.js +131 -0
  144. package/dist/heart/providers/anthropic.js +202 -50
  145. package/dist/heart/providers/azure.js +104 -13
  146. package/dist/heart/providers/error-classification.js +63 -0
  147. package/dist/heart/providers/github-copilot.js +145 -0
  148. package/dist/heart/providers/minimax-vlm.js +189 -0
  149. package/dist/heart/providers/minimax.js +29 -7
  150. package/dist/heart/providers/openai-codex.js +63 -39
  151. package/dist/heart/runtime-capability-check.js +170 -0
  152. package/dist/heart/runtime-credentials.js +260 -0
  153. package/dist/heart/sense-truth.js +68 -0
  154. package/dist/heart/session-activity.js +190 -0
  155. package/dist/heart/session-events.js +1089 -0
  156. package/dist/heart/session-playback-cli-main.js +5 -0
  157. package/dist/heart/session-playback-cli.js +36 -0
  158. package/dist/heart/session-playback.js +231 -0
  159. package/dist/heart/session-transcript.js +167 -0
  160. package/dist/heart/start-of-turn-packet.js +345 -0
  161. package/dist/heart/streaming.js +129 -34
  162. package/dist/heart/sync.js +332 -0
  163. package/dist/heart/target-resolution.js +127 -0
  164. package/dist/heart/tempo.js +93 -0
  165. package/dist/heart/temporal-view.js +41 -0
  166. package/dist/heart/tool-activity-callbacks.js +36 -0
  167. package/dist/heart/tool-description.js +135 -0
  168. package/dist/heart/tool-friction.js +55 -0
  169. package/dist/heart/tool-loop.js +200 -0
  170. package/dist/heart/turn-context.js +372 -0
  171. package/dist/heart/turn-coordinator.js +28 -0
  172. package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
  173. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  174. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  175. package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
  176. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  177. package/dist/heart/versioning/staged-restart.js +146 -0
  178. package/dist/heart/versioning/update-checker.js +115 -0
  179. package/dist/heart/versioning/update-hooks.js +142 -0
  180. package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
  181. package/dist/mailroom/attention.js +167 -0
  182. package/dist/mailroom/autonomy.js +209 -0
  183. package/dist/mailroom/blob-store.js +606 -0
  184. package/dist/mailroom/core.js +672 -0
  185. package/dist/mailroom/entry.js +160 -0
  186. package/dist/mailroom/file-store.js +426 -0
  187. package/dist/mailroom/mbox-import.js +382 -0
  188. package/dist/mailroom/outbound.js +380 -0
  189. package/dist/mailroom/policy.js +263 -0
  190. package/dist/mailroom/reader.js +219 -0
  191. package/dist/mailroom/search-cache.js +182 -0
  192. package/dist/mailroom/search-relevance.js +319 -0
  193. package/dist/mailroom/smtp-ingress.js +176 -0
  194. package/dist/mailroom/source-state.js +176 -0
  195. package/dist/mailroom/thread.js +109 -0
  196. package/dist/mailroom/travel-extract.js +89 -0
  197. package/dist/mind/bundle-manifest.js +77 -1
  198. package/dist/mind/context.js +173 -94
  199. package/dist/mind/diary-integrity.js +60 -0
  200. package/dist/mind/{memory.js → diary.js} +84 -96
  201. package/dist/mind/embedding-provider.js +60 -0
  202. package/dist/mind/file-state.js +179 -0
  203. package/dist/mind/first-impressions.js +16 -2
  204. package/dist/mind/friends/channel.js +73 -0
  205. package/dist/mind/friends/group-context.js +144 -0
  206. package/dist/mind/friends/resolver.js +54 -2
  207. package/dist/mind/friends/store-file.js +58 -3
  208. package/dist/mind/friends/trust-explanation.js +74 -0
  209. package/dist/mind/friends/types.js +10 -2
  210. package/dist/mind/journal-index.js +161 -0
  211. package/dist/mind/note-search.js +268 -0
  212. package/dist/mind/obligation-steering.js +221 -0
  213. package/dist/mind/pending.js +76 -9
  214. package/dist/mind/phrases.js +1 -0
  215. package/dist/mind/prompt-refresh.js +3 -2
  216. package/dist/mind/prompt.js +1144 -117
  217. package/dist/mind/provenance-trust.js +26 -0
  218. package/dist/mind/scrutiny.js +173 -0
  219. package/dist/mind/token-estimate.js +8 -12
  220. package/dist/nerves/cli-logging.js +22 -3
  221. package/dist/nerves/coverage/audit-rules.js +15 -6
  222. package/dist/nerves/coverage/audit.js +28 -2
  223. package/dist/nerves/coverage/cli.js +1 -1
  224. package/dist/nerves/coverage/contract.js +5 -5
  225. package/dist/nerves/coverage/file-completeness.js +101 -5
  226. package/dist/nerves/coverage/run-artifacts.js +1 -1
  227. package/dist/nerves/event-buffer.js +111 -0
  228. package/dist/nerves/index.js +224 -4
  229. package/dist/nerves/observation.js +20 -0
  230. package/dist/nerves/redact.js +79 -0
  231. package/dist/nerves/runtime.js +5 -1
  232. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  233. package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
  234. package/dist/outlook-ui/index.html +15 -0
  235. package/dist/repertoire/ado-client.js +17 -56
  236. package/dist/repertoire/ado-semantic.js +11 -10
  237. package/dist/repertoire/api-client.js +97 -0
  238. package/dist/repertoire/bitwarden-store.js +774 -0
  239. package/dist/repertoire/bundle-templates.js +72 -0
  240. package/dist/repertoire/bw-installer.js +180 -0
  241. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  242. package/dist/repertoire/coding/context-pack.js +330 -0
  243. package/dist/repertoire/coding/feedback.js +301 -0
  244. package/dist/repertoire/coding/index.js +4 -1
  245. package/dist/repertoire/coding/manager.js +220 -13
  246. package/dist/repertoire/coding/spawner.js +58 -12
  247. package/dist/repertoire/coding/tools.js +209 -7
  248. package/dist/repertoire/commerce-errors.js +109 -0
  249. package/dist/repertoire/commerce-self-test.js +156 -0
  250. package/dist/repertoire/credential-access.js +111 -0
  251. package/dist/repertoire/data/ado-endpoints.json +188 -0
  252. package/dist/repertoire/duffel-client.js +185 -0
  253. package/dist/repertoire/github-client.js +14 -55
  254. package/dist/repertoire/graph-client.js +11 -52
  255. package/dist/repertoire/guardrails.js +396 -0
  256. package/dist/repertoire/mcp-client.js +255 -0
  257. package/dist/repertoire/mcp-manager.js +305 -0
  258. package/dist/repertoire/mcp-tools.js +63 -0
  259. package/dist/repertoire/shell-sessions.js +133 -0
  260. package/dist/repertoire/skills.js +15 -24
  261. package/dist/repertoire/stripe-client.js +131 -0
  262. package/dist/repertoire/tasks/board.js +43 -5
  263. package/dist/repertoire/tasks/fix.js +182 -0
  264. package/dist/repertoire/tasks/index.js +39 -13
  265. package/dist/repertoire/tasks/lifecycle.js +2 -2
  266. package/dist/repertoire/tasks/parser.js +3 -2
  267. package/dist/repertoire/tasks/scanner.js +194 -37
  268. package/dist/repertoire/tasks/transitions.js +16 -79
  269. package/dist/repertoire/tool-results.js +29 -0
  270. package/dist/repertoire/tools-attachments.js +317 -0
  271. package/dist/repertoire/tools-base.js +49 -707
  272. package/dist/repertoire/tools-bluebubbles.js +94 -0
  273. package/dist/repertoire/tools-bridge.js +141 -0
  274. package/dist/repertoire/tools-bundle.js +984 -0
  275. package/dist/repertoire/tools-config.js +185 -0
  276. package/dist/repertoire/tools-continuity.js +248 -0
  277. package/dist/repertoire/tools-credential.js +381 -0
  278. package/dist/repertoire/tools-files.js +342 -0
  279. package/dist/repertoire/tools-flight.js +224 -0
  280. package/dist/repertoire/tools-flow.js +105 -0
  281. package/dist/repertoire/tools-github.js +1 -7
  282. package/dist/repertoire/tools-mail.js +1377 -0
  283. package/dist/repertoire/tools-notes.js +376 -0
  284. package/dist/repertoire/tools-session.js +749 -0
  285. package/dist/repertoire/tools-shell.js +120 -0
  286. package/dist/repertoire/tools-stripe.js +180 -0
  287. package/dist/repertoire/tools-surface.js +243 -0
  288. package/dist/repertoire/tools-teams.js +64 -61
  289. package/dist/repertoire/tools-travel.js +125 -0
  290. package/dist/repertoire/tools-trip.js +356 -0
  291. package/dist/repertoire/tools-user-profile.js +144 -0
  292. package/dist/repertoire/tools-vault.js +40 -0
  293. package/dist/repertoire/tools.js +149 -98
  294. package/dist/repertoire/travel-api-client.js +360 -0
  295. package/dist/repertoire/user-profile.js +131 -0
  296. package/dist/repertoire/vault-setup.js +246 -0
  297. package/dist/repertoire/vault-unlock.js +561 -0
  298. package/dist/scripts/claude-code-hook.js +41 -0
  299. package/dist/scripts/claude-code-stop-hook.js +47 -0
  300. package/dist/senses/attention-queue.js +116 -0
  301. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  302. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  303. package/dist/senses/bluebubbles/client.js +685 -0
  304. package/dist/senses/bluebubbles/entry.js +73 -0
  305. package/dist/senses/bluebubbles/inbound-log.js +126 -0
  306. package/dist/senses/bluebubbles/index.js +1881 -0
  307. package/dist/senses/bluebubbles/media.js +389 -0
  308. package/dist/senses/bluebubbles/model.js +282 -0
  309. package/dist/senses/bluebubbles/mutation-log.js +116 -0
  310. package/dist/senses/bluebubbles/processed-log.js +111 -0
  311. package/dist/senses/bluebubbles/replay.js +129 -0
  312. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  313. package/dist/senses/bluebubbles/session-cleanup.js +72 -0
  314. package/dist/senses/cli/bracketed-paste.js +82 -0
  315. package/dist/senses/cli/image-paste.js +287 -0
  316. package/dist/senses/cli/image-ref-navigation.js +75 -0
  317. package/dist/senses/cli/ink-app.js +156 -0
  318. package/dist/senses/cli/inline-diff.js +64 -0
  319. package/dist/senses/cli/input-keys.js +174 -0
  320. package/dist/senses/cli/kill-ring.js +86 -0
  321. package/dist/senses/cli/message-list.js +51 -0
  322. package/dist/senses/cli/ouro-tui.js +605 -0
  323. package/dist/senses/cli/spinner-imperative.js +135 -0
  324. package/dist/senses/cli/spinner.js +101 -0
  325. package/dist/senses/cli/status-line.js +60 -0
  326. package/dist/senses/cli/streaming-markdown.js +526 -0
  327. package/dist/senses/cli/tool-display.js +83 -0
  328. package/dist/senses/cli/tool-render.js +85 -0
  329. package/dist/senses/cli/tui-store.js +240 -0
  330. package/dist/senses/cli/virtual-list.js +35 -0
  331. package/dist/senses/cli-entry.js +60 -8
  332. package/dist/senses/cli-layout.js +187 -0
  333. package/dist/senses/cli.js +768 -264
  334. package/dist/senses/commands.js +66 -3
  335. package/dist/senses/continuity.js +94 -0
  336. package/dist/senses/habit-turn-message.js +108 -0
  337. package/dist/senses/inner-dialog-worker.js +199 -16
  338. package/dist/senses/inner-dialog.js +640 -91
  339. package/dist/senses/mail-entry.js +66 -0
  340. package/dist/senses/mail.js +379 -0
  341. package/dist/senses/pipeline.js +665 -0
  342. package/dist/senses/proactive-content-guard.js +51 -0
  343. package/dist/senses/shared-turn.js +248 -0
  344. package/dist/senses/surface-tool.js +68 -0
  345. package/dist/senses/teams-entry.js +60 -8
  346. package/dist/senses/teams.js +844 -197
  347. package/dist/senses/trust-gate.js +207 -2
  348. package/dist/trips/core.js +138 -0
  349. package/dist/trips/store.js +146 -0
  350. package/package.json +47 -6
  351. package/skills/agent-commerce.md +106 -0
  352. package/skills/browser-navigation.md +117 -0
  353. package/skills/commerce-setup-guide.md +116 -0
  354. package/skills/commerce-setup.md +84 -0
  355. package/skills/configure-dev-tools.md +101 -0
  356. package/skills/travel-planning.md +138 -0
  357. package/AdoptionSpecialist.ouro/agent.json +0 -20
  358. package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
  359. package/dist/heart/daemon/specialist-orchestrator.js +0 -160
  360. package/dist/heart/daemon/specialist-prompt.js +0 -40
  361. package/dist/heart/daemon/specialist-session.js +0 -142
  362. package/dist/heart/daemon/specialist-tools.js +0 -128
  363. package/dist/heart/daemon/subagent-installer.js +0 -125
  364. package/dist/inner-worker-entry.js +0 -4
  365. package/dist/mind/associative-recall.js +0 -197
  366. package/subagents/README.md +0 -73
  367. package/subagents/work-doer.md +0 -233
  368. package/subagents/work-merger.md +0 -624
  369. package/subagents/work-planner.md +0 -373
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
@@ -34,16 +34,24 @@ var __importStar = (this && this.__importStar) || (function () {
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.DEFAULT_FLUSH_INTERVAL_MS = void 0;
37
+ exports.aiLabelEntities = aiLabelEntities;
37
38
  exports.stripMentions = stripMentions;
38
39
  exports.splitMessage = splitMessage;
40
+ exports.sanitizeFeedbackComment = sanitizeFeedbackComment;
41
+ exports.buildFeedbackSyntheticText = buildFeedbackSyntheticText;
42
+ exports.buildWelcomeCard = buildWelcomeCard;
39
43
  exports.createTeamsCallbacks = createTeamsCallbacks;
40
- exports.resolvePendingConfirmation = resolvePendingConfirmation;
41
44
  exports.withConversationLock = withConversationLock;
42
45
  exports.handleTeamsMessage = handleTeamsMessage;
46
+ exports.sendProactiveTeamsMessageToSession = sendProactiveTeamsMessageToSession;
47
+ exports.drainAndSendPendingTeams = drainAndSendPendingTeams;
43
48
  exports.startTeamsApp = startTeamsApp;
49
+ const fs = __importStar(require("fs"));
44
50
  const teams_apps_1 = require("@microsoft/teams.apps");
45
51
  const teams_dev_1 = require("@microsoft/teams.dev");
46
52
  const core_1 = require("../heart/core");
53
+ const tools_1 = require("../repertoire/tools");
54
+ const channel_1 = require("../mind/friends/channel");
47
55
  const config_1 = require("../heart/config");
48
56
  const prompt_1 = require("../mind/prompt");
49
57
  const phrases_1 = require("../mind/phrases");
@@ -53,13 +61,34 @@ const context_1 = require("../mind/context");
53
61
  const commands_1 = require("./commands");
54
62
  const nerves_1 = require("../nerves");
55
63
  const runtime_1 = require("../nerves/runtime");
64
+ const proactive_content_guard_1 = require("./proactive-content-guard");
56
65
  const store_file_1 = require("../mind/friends/store-file");
66
+ const types_1 = require("../mind/friends/types");
57
67
  const resolver_1 = require("../mind/friends/resolver");
58
68
  const tokens_1 = require("../mind/friends/tokens");
59
69
  const turn_coordinator_1 = require("../heart/turn-coordinator");
60
70
  const identity_1 = require("../heart/identity");
71
+ const mcp_manager_1 = require("../repertoire/mcp-manager");
72
+ const progress_story_1 = require("../heart/progress-story");
73
+ const tool_activity_callbacks_1 = require("../heart/tool-activity-callbacks");
74
+ const commands_2 = require("./commands");
75
+ const http = __importStar(require("http"));
61
76
  const path = __importStar(require("path"));
62
77
  const trust_gate_1 = require("./trust-gate");
78
+ const pipeline_1 = require("./pipeline");
79
+ const teamsFailoverStates = new Map();
80
+ const pending_1 = require("../mind/pending");
81
+ const continuity_1 = require("./continuity");
82
+ // AIGeneratedContent entity and feedbackLoopEnabled channelData for all outbound
83
+ // Teams messages. Required by Teams AI UX best practices.
84
+ function aiLabelEntities() {
85
+ return [{
86
+ type: "https://schema.org/Message",
87
+ "@type": "Message",
88
+ "@context": "https://schema.org",
89
+ additionalType: ["AIGeneratedContent"],
90
+ }];
91
+ }
63
92
  // Strip @mention markup from incoming messages.
64
93
  // Removes <at>...</at> tags and trims extra whitespace.
65
94
  // Fallback safety net -- the SDK's activity.mentions.stripText should handle
@@ -114,6 +143,46 @@ function splitMessage(text, maxLen) {
114
143
  }
115
144
  return chunks;
116
145
  }
146
+ // Sanitize user-provided feedback comments: truncate, strip control chars and newlines.
147
+ function sanitizeFeedbackComment(comment) {
148
+ const cleaned = comment.replace(/[\x00-\x1f\n\r]/g, "");
149
+ return cleaned.length > 200 ? cleaned.slice(0, 200) : cleaned;
150
+ }
151
+ // Build synthetic message text from a Teams feedback reaction.
152
+ function buildFeedbackSyntheticText(reaction, comment) {
153
+ const emoji = reaction === "like" ? "thumbs-up" : "thumbs-down";
154
+ if (comment) {
155
+ const sanitized = sanitizeFeedbackComment(comment);
156
+ return `[reacted with ${emoji} to your message: "${sanitized}"]`;
157
+ }
158
+ return `[reacted with ${emoji} to your message]`;
159
+ }
160
+ // Build a welcome Adaptive Card with prompt starters for new bot installs.
161
+ function buildWelcomeCard() {
162
+ const promptStarters = [
163
+ "What can you help me with?",
164
+ "Tell me about yourself",
165
+ "What's on my calendar today?",
166
+ "Summarize my recent emails",
167
+ ];
168
+ return {
169
+ type: "AdaptiveCard",
170
+ version: "1.5",
171
+ body: [
172
+ {
173
+ type: "TextBlock",
174
+ text: "Hey! I'm here and ready to help. Try one of these to get started, or just ask me anything.",
175
+ wrap: true,
176
+ size: "Medium",
177
+ },
178
+ ],
179
+ actions: promptStarters.map((prompt) => ({
180
+ type: "Action.Submit",
181
+ title: prompt,
182
+ data: { msteams: { type: "messageBack", text: prompt, displayText: prompt } },
183
+ })),
184
+ };
185
+ }
117
186
  // Create Teams-specific callbacks for the agent loop.
118
187
  // The SDK handles cumulative text, debouncing (500ms), and the streaming
119
188
  // protocol (streamSequence, streamId, informative/streaming/final types).
@@ -126,12 +195,16 @@ function splitMessage(text, maxLen) {
126
195
  // (transient status) or safeSend (terminal errors). Reasoning is accumulated
127
196
  // and periodically pushed via safeUpdate on the same flush timer tick.
128
197
  function createTeamsCallbacks(stream, controller, sendMessage, options) {
198
+ const MIN_INITIAL_CHARS = 20;
129
199
  let stopped = false; // set when stream signals cancellation (403)
130
200
  let hadToolRun = false;
131
201
  let hadRealOutput = false; // true once reasoning/tool output shown; suppresses phrases
132
202
  let reasoningBuf = ""; // accumulated reasoning text for status display
203
+ let totalEmitted = 0; // cumulative chars emitted via safeEmit (for >4000 finalization)
204
+ let streamFinalized = false; // true after stream.close() — subsequent flushes go to safeSend
133
205
  let textBuffer = ""; // accumulated text output for chunked streaming
134
206
  let streamHasContent = false; // tracks whether primary output has received content
207
+ let firstContentEmitted = false; // true after first content push — disables MIN_INITIAL_CHARS threshold
135
208
  let phraseTimer = null;
136
209
  let lastPhrase = "";
137
210
  let flushTimer = null;
@@ -181,15 +254,16 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
181
254
  result.catch(() => markStopped());
182
255
  }
183
256
  }
184
- // Safely emit a text delta to the stream.
257
+ // Safely emit a text delta to the stream with AI labels.
185
258
  // On error (e.g. 403 from Teams stop button), abort the controller.
186
259
  function safeEmit(text) {
187
260
  /* v8 ignore next -- defensive guard: stopped set by prior 403; tested via flush abort path @preserve */
188
261
  if (stopped)
189
262
  return;
190
263
  try {
191
- catchAsync(stream.emit(text));
264
+ catchAsync(stream.emit({ text, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } }));
192
265
  streamHasContent = true;
266
+ totalEmitted += text.length;
193
267
  }
194
268
  catch {
195
269
  markStopped();
@@ -204,7 +278,7 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
204
278
  try {
205
279
  // stream.emit() is typed as void but the Teams SDK returns a Promise
206
280
  // internally (async HTTP). Cast to capture the result for awaiting.
207
- const result = stream.emit(text);
281
+ const result = stream.emit({ text, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } });
208
282
  streamHasContent = true;
209
283
  if (result && typeof result.then === "function") {
210
284
  await result;
@@ -260,11 +334,49 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
260
334
  // emitted text into a single streaming message (cumulative), so every
261
335
  // periodic flush appends to the same response — not separate messages.
262
336
  // No preemptive splitting — sends full text. Error recovery happens in flush().
337
+ // Hybrid MIN_INITIAL_CHARS: hold back until >= MIN_INITIAL_CHARS accumulated
338
+ // before the first content emit, so phrase rotation shows while real content
339
+ // buffers. After first emit, flush normally (no threshold).
263
340
  function flushTextBuffer() {
264
341
  if (!textBuffer)
265
342
  return;
343
+ if (!firstContentEmitted && textBuffer.length < MIN_INITIAL_CHARS)
344
+ return;
345
+ // Proactive >4000 finalization: if cumulative emitted + buffer >= RECOVERY_CHUNK_SIZE,
346
+ // finalize the stream and send overflow via safeSend (follow-up message).
347
+ if (!streamFinalized && totalEmitted + textBuffer.length >= RECOVERY_CHUNK_SIZE) {
348
+ const remaining = RECOVERY_CHUNK_SIZE - totalEmitted;
349
+ /* v8 ignore next 2 -- defensive: remaining always > 0 because finalization runs once @preserve */
350
+ if (remaining > 0)
351
+ safeEmit(textBuffer.slice(0, remaining));
352
+ try {
353
+ stream.close();
354
+ }
355
+ catch { /* stream may already be dead */ }
356
+ streamFinalized = true;
357
+ /* v8 ignore next -- defensive ternary: remaining always > 0 at first finalization @preserve */
358
+ const overflow = textBuffer.slice(remaining > 0 ? remaining : 0);
359
+ textBuffer = "";
360
+ if (overflow)
361
+ safeSend(overflow);
362
+ if (!firstContentEmitted) {
363
+ firstContentEmitted = true;
364
+ stopPhraseRotation();
365
+ }
366
+ return;
367
+ }
368
+ if (streamFinalized) {
369
+ // After finalization, all content goes to safeSend (follow-up messages)
370
+ safeSend(textBuffer);
371
+ textBuffer = "";
372
+ return;
373
+ }
266
374
  safeEmit(textBuffer);
267
375
  textBuffer = "";
376
+ if (!firstContentEmitted) {
377
+ firstContentEmitted = true;
378
+ stopPhraseRotation();
379
+ }
268
380
  }
269
381
  function startPhraseRotation(pool) {
270
382
  stopPhraseRotation();
@@ -307,25 +419,45 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
307
419
  onTextChunk: (text) => {
308
420
  if (stopped)
309
421
  return;
310
- stopPhraseRotation();
422
+ // Don't stop phrase rotation here — let it continue until first content
423
+ // emit (handled in flushTextBuffer when MIN_INITIAL_CHARS threshold met).
311
424
  textBuffer += text;
312
425
  startFlushTimer();
313
426
  },
314
427
  onClearText: () => {
315
428
  textBuffer = "";
316
429
  },
317
- onToolStart: (name, args) => {
318
- stopPhraseRotation();
319
- flushTextBuffer();
320
- const argSummary = Object.values(args).join(", ");
321
- safeUpdate(`running ${name} (${argSummary})...`);
322
- hadToolRun = true;
323
- },
324
- onToolEnd: (name, summary, success) => {
325
- stopPhraseRotation();
326
- const msg = (0, format_1.formatToolResult)(name, summary, success);
327
- safeUpdate(msg);
328
- },
430
+ ...(() => {
431
+ const toolCbs = (0, tool_activity_callbacks_1.createToolActivityCallbacks)({
432
+ onDescription: (text) => safeUpdate(text),
433
+ /* v8 ignore next -- onResult only called in debug mode; tested via tool-activity-callbacks.test.ts @preserve */
434
+ onResult: (text) => safeUpdate(text),
435
+ /* v8 ignore next -- onFailure tested via onToolEnd failure test @preserve */
436
+ onFailure: (text) => safeUpdate(text),
437
+ isDebug: commands_2.getDebugMode,
438
+ });
439
+ return {
440
+ onToolStart: (name, args) => {
441
+ stopPhraseRotation();
442
+ // Force-flush any accumulated text, bypassing MIN_INITIAL_CHARS threshold
443
+ firstContentEmitted = true;
444
+ flushTextBuffer();
445
+ // Emit a placeholder to satisfy the 15s Copilot timeout for initial
446
+ // stream.emit(). Without this, long tool chains (e.g. ADO batch ops)
447
+ // never emit before the timeout and the user sees "this response was
448
+ // stopped". The placeholder is replaced by actual content on next emit.
449
+ // https://learn.microsoft.com/en-us/answers/questions/2288017/m365-custom-engine-agents-timeout-message-after-15
450
+ if (!streamHasContent)
451
+ safeEmit("\u23f3");
452
+ toolCbs.onToolStart(name, args);
453
+ hadToolRun = true;
454
+ },
455
+ onToolEnd: (name, summary, success) => {
456
+ stopPhraseRotation();
457
+ toolCbs.onToolEnd(name, summary, success);
458
+ },
459
+ };
460
+ })(),
329
461
  onKick: () => {
330
462
  stopPhraseRotation();
331
463
  const msg = (0, format_1.formatKick)();
@@ -335,7 +467,11 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
335
467
  stopPhraseRotation();
336
468
  if (stopped)
337
469
  return;
338
- const msg = (0, format_1.formatError)(error);
470
+ const msg = (0, progress_story_1.renderProgressStory)((0, progress_story_1.buildProgressStory)({
471
+ scope: "shared-work",
472
+ phase: "errored",
473
+ outcomeText: (0, format_1.formatError)(error),
474
+ }));
339
475
  if (severity === "transient") {
340
476
  safeUpdate(msg);
341
477
  }
@@ -343,38 +479,27 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
343
479
  safeSend(msg);
344
480
  }
345
481
  },
346
- onConfirmAction: options?.conversationId
347
- ? async (name, args) => {
348
- const convId = options.conversationId;
349
- const argsDesc = Object.entries(args).map(([k, v]) => `${k}: ${v}`).join(", ");
350
- safeUpdate(`Confirm action: ${name} (${argsDesc}) -- reply "yes" to confirm or "no" to cancel`);
351
- return new Promise((resolve) => {
352
- _pendingConfirmations.set(convId, resolve);
353
- // Auto-deny after 2 minutes to prevent indefinite blocking
354
- // (e.g. when the stream dies and the user never sees the prompt).
355
- setTimeout(() => {
356
- if (_pendingConfirmations.has(convId)) {
357
- _pendingConfirmations.delete(convId);
358
- resolve("denied");
359
- }
360
- }, 120_000);
361
- });
362
- }
363
- : undefined,
364
482
  flush: async () => {
365
483
  stopFlushTimer();
484
+ stopPhraseRotation();
366
485
  if (textBuffer) {
486
+ // Bypass MIN_INITIAL_CHARS threshold — flush delivers all remaining content
487
+ firstContentEmitted = true;
367
488
  const text = textBuffer;
368
489
  textBuffer = "";
369
- if (!stopped) {
490
+ if (streamFinalized && sendMessage) {
491
+ // Stream already finalized (>4000 path) — send remaining content as follow-up
492
+ safeSend(text);
493
+ }
494
+ else if (!stopped) {
370
495
  // Stream is alive — await the emit so we can catch async 413/failure
371
496
  // and fall through to sendMessage recovery.
372
497
  const ok = await tryEmit(text);
373
498
  if (!ok)
374
499
  markStopped();
375
500
  }
376
- if (stopped && sendMessage) {
377
- // Stream is dead — fall back to sendMessage; split on failure as recovery.
501
+ if (stopped && !streamFinalized && sendMessage) {
502
+ // Stream is dead (not from finalization) — fall back to sendMessage; split on failure as recovery.
378
503
  try {
379
504
  await sendMessage(text);
380
505
  }
@@ -385,32 +510,12 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
385
510
  }
386
511
  }
387
512
  }
388
- else if (!streamHasContent) {
389
- safeEmit("(completed with tool calls only \u2014 no text response)");
513
+ else if (!streamHasContent && !options?.suppressEmptyStreamMessage) {
514
+ safeEmit("(completed with tool calls only no text response)");
390
515
  }
391
516
  },
392
517
  };
393
518
  }
394
- // Per-conversation pending confirmation resolvers.
395
- // When a mutate tool needs confirmation, the resolver is stored here.
396
- // The next message from the same conversation resolves it.
397
- const _pendingConfirmations = new Map();
398
- // Confirmation response words (case-insensitive)
399
- const CONFIRM_WORDS = new Set(["yes", "confirm", "go", "y", "ok", "approve", "proceed"]);
400
- function resolvePendingConfirmation(convId, text) {
401
- const resolver = _pendingConfirmations.get(convId);
402
- if (!resolver)
403
- return false;
404
- _pendingConfirmations.delete(convId);
405
- const word = text.trim().toLowerCase();
406
- if (CONFIRM_WORDS.has(word)) {
407
- resolver("confirmed");
408
- }
409
- else {
410
- resolver("denied");
411
- }
412
- return true;
413
- }
414
519
  const _turnCoordinator = (0, turn_coordinator_1.createTurnCoordinator)();
415
520
  function teamsTurnKey(conversationId) {
416
521
  return `teams:${conversationId}`;
@@ -424,158 +529,313 @@ function getFriendStore() {
424
529
  const friendsPath = path.join((0, identity_1.getAgentRoot)(), "friends");
425
530
  return new store_file_1.FileFriendStore(friendsPath);
426
531
  }
532
+ function createTeamsCommandRegistry() {
533
+ const registry = (0, commands_1.createCommandRegistry)();
534
+ (0, commands_1.registerDefaultCommands)(registry);
535
+ return registry;
536
+ }
537
+ /* v8 ignore start -- superseding follow-up slash command handler; tested via startTeamsApp integration tests @preserve */
538
+ function handleTeamsSlashCommand(text, registry, friendId, conversationId, stream, emitResponse = true) {
539
+ const parsed = (0, commands_1.parseSlashCommand)(text);
540
+ if (!parsed)
541
+ return null;
542
+ const dispatchResult = registry.dispatch(parsed.command, { channel: "teams" });
543
+ if (!dispatchResult.handled || !dispatchResult.result) {
544
+ return null;
545
+ }
546
+ if (dispatchResult.result.action === "new") {
547
+ (0, context_1.deleteSession)((0, config_2.sessionPath)(friendId, "teams", conversationId));
548
+ if (emitResponse) {
549
+ stream.emit("session cleared");
550
+ }
551
+ return "new";
552
+ }
553
+ if (dispatchResult.result.action === "response") {
554
+ if (emitResponse) {
555
+ stream.emit(dispatchResult.result.message || "");
556
+ }
557
+ return "response";
558
+ }
559
+ return null;
560
+ }
561
+ /* v8 ignore stop */
427
562
  // Handle an incoming Teams message
428
- async function handleTeamsMessage(text, stream, conversationId, teamsContext, sendMessage) {
563
+ async function handleTeamsMessage(text, stream, conversationId, teamsContext, sendMessage, reactionOverrides) {
429
564
  const turnKey = teamsTurnKey(conversationId);
430
565
  // NOTE: Confirmation resolution is handled in the app.on("message") handler
431
566
  // BEFORE the conversation lock. By the time we get here, any pending
432
567
  // confirmation has already been resolved and the reply consumed.
433
568
  // Send first thinking phrase immediately so the user sees feedback
434
569
  // before sync I/O (session load, trim) blocks the event loop.
435
- stream.update((0, phrases_1.pickPhrase)((0, phrases_1.getPhrases)().thinking) + "...");
570
+ // Skip for reaction signals — they should be processed quietly.
571
+ if (!reactionOverrides) {
572
+ stream.update((0, phrases_1.pickPhrase)((0, phrases_1.getPhrases)().thinking) + "...");
573
+ }
436
574
  await new Promise(r => setImmediate(r));
437
- // Resolve context kernel (identity + channel) early so we can use the friend UUID for session path
575
+ // Resolve identity provider early for friend resolution + slash command session path
438
576
  const store = getFriendStore();
439
577
  const provider = teamsContext?.aadObjectId ? "aad" : "teams-conversation";
440
578
  const externalId = teamsContext?.aadObjectId || conversationId;
441
- const toolContext = teamsContext ? {
579
+ // Build FriendResolver for the pipeline
580
+ const resolver = new resolver_1.FriendResolver(store, {
581
+ provider,
582
+ externalId,
583
+ tenantId: teamsContext?.tenantId,
584
+ displayName: teamsContext?.displayName || "Unknown",
585
+ channel: "teams",
586
+ });
587
+ // Pre-resolve friend for session path + slash commands (pipeline will re-use the cached result)
588
+ const resolvedContext = await resolver.resolve();
589
+ const friendId = resolvedContext.friend.id;
590
+ // ── Teams adapter concerns: controller, callbacks, session path ──────────
591
+ const controller = new AbortController();
592
+ const channelConfig = (0, config_2.getTeamsChannelConfig)();
593
+ const callbacks = createTeamsCallbacks(stream, controller, sendMessage, { conversationId, flushIntervalMs: channelConfig.flushIntervalMs, ...(reactionOverrides?.suppressEmptyStreamMessage ? { suppressEmptyStreamMessage: true } : {}) });
594
+ const traceId = (0, nerves_1.createTraceId)();
595
+ const sessPath = (0, config_2.sessionPath)(friendId, "teams", conversationId);
596
+ const teamsCapabilities = (0, channel_1.getChannelCapabilities)("teams");
597
+ const pendingDir = (0, pending_1.getPendingDir)((0, identity_1.getAgentName)(), friendId, "teams", conversationId);
598
+ // Build Teams-specific toolContext fields for injection into the pipeline
599
+ const teamsToolContext = teamsContext ? {
442
600
  graphToken: teamsContext.graphToken,
443
601
  adoToken: teamsContext.adoToken,
444
602
  githubToken: teamsContext.githubToken,
445
603
  signin: teamsContext.signin,
446
- friendStore: store,
447
- summarize: (0, core_1.createSummarize)(),
448
- } : undefined;
449
- if (toolContext) {
450
- const resolver = new resolver_1.FriendResolver(store, {
451
- provider,
452
- externalId,
453
- tenantId: teamsContext?.tenantId,
454
- displayName: teamsContext?.displayName || "Unknown",
604
+ summarize: (0, core_1.createSummarize)("human"),
605
+ tenantId: teamsContext.tenantId,
606
+ botApi: teamsContext.botApi,
607
+ } : {};
608
+ let currentText = text;
609
+ const mcpManager = await (0, mcp_manager_1.getSharedMcpManager)() ?? undefined;
610
+ while (true) {
611
+ let drainedSteeringFollowUps = [];
612
+ // Build runAgentOptions with Teams-specific fields
613
+ const agentOptions = {
614
+ traceId,
615
+ toolContext: teamsToolContext,
616
+ mcpManager,
617
+ drainSteeringFollowUps: () => {
618
+ drainedSteeringFollowUps = _turnCoordinator.drainFollowUps(turnKey)
619
+ .map(({ text: followUpText, effect }) => ({ text: followUpText, effect }));
620
+ return drainedSteeringFollowUps;
621
+ },
622
+ ...(reactionOverrides?.isReactionSignal ? { isReactionSignal: true } : {}),
623
+ };
624
+ // ── Call shared pipeline ──────────────────────────────────────────
625
+ // Capture terminal errors — failover message replaces the error card if it triggers
626
+ let capturedTerminalError = null;
627
+ const teamsFailoverState = (() => {
628
+ if (!teamsFailoverStates.has(conversationId)) {
629
+ teamsFailoverStates.set(conversationId, { pending: null });
630
+ }
631
+ return teamsFailoverStates.get(conversationId);
632
+ })();
633
+ /* v8 ignore start -- failover-aware callback wrapper: tested via pipeline integration @preserve */
634
+ const failoverAwareCallbacks = {
635
+ ...callbacks,
636
+ onError: (error, severity) => {
637
+ if (severity === "terminal" && teamsFailoverState) {
638
+ capturedTerminalError = error;
639
+ return;
640
+ }
641
+ callbacks.onError(error, severity);
642
+ },
643
+ };
644
+ /* v8 ignore stop */
645
+ const result = await (0, pipeline_1.handleInboundTurn)({
455
646
  channel: "teams",
456
- });
457
- toolContext.context = await resolver.resolve();
458
- }
459
- const friendId = toolContext?.context?.friend?.id || "default";
460
- if (toolContext?.context?.friend) {
461
- const trustGate = (0, trust_gate_1.enforceTrustGate)({
462
- friend: toolContext.context.friend,
647
+ sessionKey: conversationId,
648
+ capabilities: teamsCapabilities,
649
+ messages: [{ role: "user", content: currentText }],
650
+ continuityIngressTexts: [currentText],
651
+ callbacks: failoverAwareCallbacks,
652
+ friendResolver: { resolve: () => Promise.resolve(resolvedContext) },
653
+ sessionLoader: {
654
+ loadOrCreate: async () => {
655
+ const existing = (0, context_1.loadSession)(sessPath);
656
+ const messages = existing?.messages && existing.messages.length > 0
657
+ ? existing.messages
658
+ : [{ role: "system", content: (0, prompt_1.flattenSystemPrompt)(await (0, prompt_1.buildSystem)("teams", {}, resolvedContext)) }];
659
+ (0, core_1.repairOrphanedToolCalls)(messages);
660
+ return {
661
+ messages,
662
+ sessionPath: sessPath,
663
+ state: existing?.state,
664
+ events: existing?.events,
665
+ };
666
+ },
667
+ },
668
+ pendingDir,
669
+ friendStore: store,
463
670
  provider,
464
671
  externalId,
465
672
  tenantId: teamsContext?.tenantId,
466
- channel: "teams",
673
+ isGroupChat: false,
674
+ groupHasFamilyMember: false,
675
+ hasExistingGroupWithFamily: false,
676
+ enforceTrustGate: trust_gate_1.enforceTrustGate,
677
+ drainPending: pending_1.drainPending,
678
+ drainDeferredReturns: (deferredFriendId) => (0, pending_1.drainDeferredReturns)((0, identity_1.getAgentName)(), deferredFriendId),
679
+ runAgent: (msgs, cb, channel, sig, opts) => (0, core_1.runAgent)(msgs, cb, channel, sig, {
680
+ ...opts,
681
+ toolContext: {
682
+ /* v8 ignore next -- default no-op signin; pipeline provides the real one @preserve */
683
+ signin: async () => undefined,
684
+ ...opts?.toolContext,
685
+ summarize: teamsToolContext.summarize,
686
+ },
687
+ }),
688
+ postTurn: (turnMessages, sessionPathArg, usage, hooks, state) => {
689
+ const prepared = (0, context_1.postTurnTrim)(turnMessages, usage, hooks);
690
+ (0, context_1.deferPostTurnPersist)(sessionPathArg, prepared, usage, state);
691
+ },
692
+ accumulateFriendTokens: tokens_1.accumulateFriendTokens,
693
+ signal: controller.signal,
694
+ runAgentOptions: agentOptions,
695
+ failoverState: teamsFailoverState,
467
696
  });
468
- if (!trustGate.allowed) {
469
- if (trustGate.reason === "stranger_first_reply") {
470
- stream.emit(trustGate.autoReply);
471
- }
472
- return;
473
- }
474
- }
475
- const registry = (0, commands_1.createCommandRegistry)();
476
- (0, commands_1.registerDefaultCommands)(registry);
477
- // Check for slash commands
478
- const parsed = (0, commands_1.parseSlashCommand)(text);
479
- if (parsed) {
480
- const dispatchResult = registry.dispatch(parsed.command, { channel: "teams" });
481
- if (dispatchResult.handled && dispatchResult.result) {
482
- if (dispatchResult.result.action === "new") {
483
- const sessPath = (0, config_2.sessionPath)(friendId, "teams", conversationId);
697
+ // ── Handle pipeline-intercepted commands ────────────────────────
698
+ if (result.turnOutcome === "command") {
699
+ if (result.commandAction === "new") {
484
700
  (0, context_1.deleteSession)(sessPath);
485
701
  stream.emit("session cleared");
486
- return;
487
702
  }
488
- else if (dispatchResult.result.action === "response") {
489
- stream.emit(dispatchResult.result.message || "");
490
- return;
703
+ // For "response" commands: pipeline already emitted the response via onTextChunk
704
+ await callbacks.flush();
705
+ return;
706
+ }
707
+ /* v8 ignore start -- failover display: tested via pipeline integration tests @preserve */
708
+ if (result.failoverMessage) {
709
+ stream.emit(result.failoverMessage);
710
+ }
711
+ else if (capturedTerminalError) {
712
+ callbacks.onError(capturedTerminalError, "terminal");
713
+ }
714
+ /* v8 ignore stop */
715
+ // ── Handle gate result ────────────────────────────────────────
716
+ if (!result.gateResult.allowed) {
717
+ if ("autoReply" in result.gateResult && result.gateResult.autoReply) {
718
+ stream.emit(result.gateResult.autoReply);
491
719
  }
720
+ return;
492
721
  }
722
+ // Flush any remaining accumulated text at end of turn
723
+ await callbacks.flush();
724
+ // After the agent loop, check if any tool returned AUTH_REQUIRED and trigger signin.
725
+ // This must happen after the stream is done so the OAuth card renders properly.
726
+ if (teamsContext && result.messages) {
727
+ const allContent = result.messages.map(m => typeof m.content === "string" ? m.content : "").join("\n");
728
+ if (allContent.includes("AUTH_REQUIRED:graph") && teamsContext.graphConnectionName)
729
+ await teamsContext.signin(teamsContext.graphConnectionName);
730
+ if (allContent.includes("AUTH_REQUIRED:ado") && teamsContext.adoConnectionName)
731
+ await teamsContext.signin(teamsContext.adoConnectionName);
732
+ if (allContent.includes("AUTH_REQUIRED:github") && teamsContext.githubConnectionName)
733
+ await teamsContext.signin(teamsContext.githubConnectionName);
734
+ }
735
+ if (result.turnOutcome !== "superseded") {
736
+ return;
737
+ }
738
+ const supersedingIndex = drainedSteeringFollowUps
739
+ .map((followUp) => followUp.effect)
740
+ .lastIndexOf("clear_and_supersede");
741
+ if (supersedingIndex < 0) {
742
+ return;
743
+ }
744
+ const supersedingFollowUp = drainedSteeringFollowUps[supersedingIndex];
745
+ const replayTail = drainedSteeringFollowUps
746
+ .slice(supersedingIndex + 1)
747
+ .map((followUp) => followUp.text.trim())
748
+ .filter((followUpText) => followUpText.length > 0)
749
+ .join("\n");
750
+ if (replayTail) {
751
+ currentText = replayTail;
752
+ continue;
753
+ }
754
+ if (handleTeamsSlashCommand(supersedingFollowUp.text, createTeamsCommandRegistry(), friendId, conversationId, stream, false)) {
755
+ return;
756
+ }
757
+ currentText = supersedingFollowUp.text;
493
758
  }
494
- // Load or create session
495
- const sessPath = (0, config_2.sessionPath)(friendId, "teams", conversationId);
496
- const existing = (0, context_1.loadSession)(sessPath);
497
- const messages = existing?.messages && existing.messages.length > 0
498
- ? existing.messages
499
- : [{ role: "system", content: await (0, prompt_1.buildSystem)("teams", undefined, toolContext?.context) }];
500
- // Push user message
501
- messages.push({ role: "user", content: text });
502
- // Run agent
503
- const controller = new AbortController();
504
- const channelConfig = (0, config_2.getTeamsChannelConfig)();
505
- const callbacks = createTeamsCallbacks(stream, controller, sendMessage, { conversationId, flushIntervalMs: channelConfig.flushIntervalMs });
506
- const traceId = (0, nerves_1.createTraceId)();
507
- const agentOptions = {};
508
- agentOptions.traceId = traceId;
509
- if (toolContext)
510
- agentOptions.toolContext = toolContext;
511
- if (channelConfig.skipConfirmation)
512
- agentOptions.skipConfirmation = true;
513
- agentOptions.drainSteeringFollowUps = () => _turnCoordinator.drainFollowUps(turnKey).map((m) => ({ text: m.text }));
514
- const result = await (0, core_1.runAgent)(messages, callbacks, "teams", controller.signal, agentOptions);
515
- // Flush any remaining accumulated text at end of turn
516
- await callbacks.flush();
517
- // After the agent loop, check if any tool returned AUTH_REQUIRED and trigger signin.
518
- // This must happen after the stream is done so the OAuth card renders properly.
519
- if (teamsContext) {
520
- const allContent = messages.map(m => typeof m.content === "string" ? m.content : "").join("\n");
521
- if (allContent.includes("AUTH_REQUIRED:graph"))
522
- await teamsContext.signin("graph");
523
- if (allContent.includes("AUTH_REQUIRED:ado"))
524
- await teamsContext.signin("ado");
525
- if (allContent.includes("AUTH_REQUIRED:github"))
526
- await teamsContext.signin("github");
527
- }
528
- // Trim context and save session
529
- (0, context_1.postTurn)(messages, sessPath, result.usage);
530
- // Accumulate token usage on friend record
531
- if (toolContext?.context?.friend?.id) {
532
- await (0, tokens_1.accumulateFriendTokens)(store, toolContext.context.friend.id, result.usage);
533
- }
534
- // SDK auto-closes the stream after our handler returns (app.process.js)
535
759
  }
536
- // Start the Teams app in DevtoolsPlugin mode (local dev) or Bot Service mode (real Teams).
537
- // Mode is determined by getTeamsConfig().clientId.
538
- // Text is always accumulated in textBuffer and flushed periodically (chunked streaming).
539
- function startTeamsApp() {
760
+ // Internal port for the secondary bot App (not exposed externally).
761
+ // The primary app proxies /api/messages-secondary → localhost:SECONDARY_PORT/api/messages.
762
+ const SECONDARY_INTERNAL_PORT = 3979;
763
+ // Collect all unique OAuth connection names across top-level config and tenant overrides.
764
+ /* v8 ignore start -- runtime Teams SDK config; no unit-testable surface @preserve */
765
+ function allOAuthConnectionNames() {
766
+ const oauthConfig = (0, config_1.getOAuthConfig)();
767
+ const names = new Set();
768
+ if (oauthConfig.graphConnectionName)
769
+ names.add(oauthConfig.graphConnectionName);
770
+ if (oauthConfig.adoConnectionName)
771
+ names.add(oauthConfig.adoConnectionName);
772
+ if (oauthConfig.githubConnectionName)
773
+ names.add(oauthConfig.githubConnectionName);
774
+ if (oauthConfig.tenantOverrides) {
775
+ for (const ov of Object.values(oauthConfig.tenantOverrides)) {
776
+ if (ov.graphConnectionName)
777
+ names.add(ov.graphConnectionName);
778
+ if (ov.adoConnectionName)
779
+ names.add(ov.adoConnectionName);
780
+ if (ov.githubConnectionName)
781
+ names.add(ov.githubConnectionName);
782
+ }
783
+ }
784
+ return [...names];
785
+ }
786
+ // Create an App instance from a TeamsConfig. Returns { app, mode }.
787
+ function createBotApp(teamsConfig) {
540
788
  const mentionStripping = { activity: { mentions: { stripText: true } } };
541
- const teamsConfig = (0, config_2.getTeamsConfig)();
542
- let app;
543
- let mode;
544
789
  const oauthConfig = (0, config_1.getOAuthConfig)();
545
- if (teamsConfig.clientId) {
546
- // Bot Service mode -- real Teams connection with SingleTenant credentials
547
- app = new teams_apps_1.App({
548
- clientId: teamsConfig.clientId,
549
- clientSecret: teamsConfig.clientSecret,
550
- tenantId: teamsConfig.tenantId,
551
- oauth: { defaultConnectionName: oauthConfig.graphConnectionName },
552
- ...mentionStripping,
553
- });
554
- mode = "Bot Service";
790
+ if (teamsConfig.clientId && teamsConfig.clientSecret) {
791
+ return {
792
+ app: new teams_apps_1.App({
793
+ clientId: teamsConfig.clientId,
794
+ clientSecret: teamsConfig.clientSecret,
795
+ tenantId: teamsConfig.tenantId,
796
+ oauth: { defaultConnectionName: oauthConfig.graphConnectionName },
797
+ ...mentionStripping,
798
+ }),
799
+ mode: "Bot Service (client secret)",
800
+ };
801
+ }
802
+ else if (teamsConfig.clientId) {
803
+ return {
804
+ app: new teams_apps_1.App({
805
+ clientId: teamsConfig.clientId,
806
+ tenantId: teamsConfig.tenantId,
807
+ ...(teamsConfig.managedIdentityClientId ? { managedIdentityClientId: teamsConfig.managedIdentityClientId } : {}),
808
+ oauth: { defaultConnectionName: oauthConfig.graphConnectionName },
809
+ ...mentionStripping,
810
+ }),
811
+ mode: "Bot Service (managed identity)",
812
+ };
555
813
  }
556
814
  else {
557
- // DevtoolsPlugin mode -- local development with Teams DevtoolsPlugin UI
558
- app = new teams_apps_1.App({
559
- plugins: [new teams_dev_1.DevtoolsPlugin()],
560
- ...mentionStripping,
561
- });
562
- mode = "DevtoolsPlugin";
815
+ return {
816
+ app: new teams_apps_1.App({
817
+ plugins: [new teams_dev_1.DevtoolsPlugin()],
818
+ ...mentionStripping,
819
+ }),
820
+ mode: "DevtoolsPlugin",
821
+ };
563
822
  }
823
+ }
824
+ /* v8 ignore stop */
825
+ // Register message, verify-state, and error handlers on an App instance.
826
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
827
+ function registerBotHandlers(app, label) {
828
+ const connectionNames = allOAuthConnectionNames();
564
829
  // Override default OAuth verify-state handler. The SDK's built-in handler
565
830
  // uses a single defaultConnectionName, which breaks multi-connection setups
566
831
  // (graph + ado + github). The verifyState activity only carries a `state`
567
832
  // code with no connectionName, so we try each configured connection until
568
833
  // one succeeds.
569
- const allConnectionNames = [
570
- oauthConfig.graphConnectionName,
571
- oauthConfig.adoConnectionName,
572
- oauthConfig.githubConnectionName,
573
- ].filter(Boolean);
574
- app.on("signin.verify-state", async (ctx) => {
834
+ app.on("signin.verify-state", (async (ctx) => {
575
835
  const { api, activity } = ctx;
576
836
  if (!activity.value?.state)
577
837
  return { status: 404 };
578
- for (const cn of allConnectionNames) {
838
+ for (const cn of connectionNames) {
579
839
  try {
580
840
  await api.users.token.get({
581
841
  channelId: activity.channelId,
@@ -583,14 +843,80 @@ function startTeamsApp() {
583
843
  connectionName: cn,
584
844
  code: activity.value.state,
585
845
  });
586
- (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.verify_state", component: "channels", message: `verify-state succeeded for connection "${cn}"`, meta: { connectionName: cn } });
846
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.verify_state", component: "channels", message: `[${label}] verify-state succeeded for connection "${cn}"`, meta: { connectionName: cn } });
587
847
  return { status: 200 };
588
848
  }
589
849
  catch { /* try next */ }
590
850
  }
591
- (0, runtime_1.emitNervesEvent)({ level: "warn", event: "channel.verify_state", component: "channels", message: "verify-state failed for all connections", meta: {} });
851
+ (0, runtime_1.emitNervesEvent)({ level: "warn", event: "channel.verify_state", component: "channels", message: `[${label}] verify-state failed for all connections`, meta: {} });
592
852
  return { status: 412 };
853
+ }));
854
+ // Handle Teams feedback reactions (thumbs up/down on AI-generated messages).
855
+ // SDK routes message/submitAction with actionName "feedback" to this event.
856
+ /* v8 ignore start -- Teams SDK invoke handler; requires live SDK context @preserve */
857
+ app.on("message.submit.feedback", async (ctx) => {
858
+ const { stream, activity } = ctx;
859
+ const reaction = activity.value?.actionValue?.reaction;
860
+ const comment = activity.value?.actionValue?.feedback;
861
+ const convId = activity.conversation?.id || "unknown";
862
+ const turnKey = teamsTurnKey(convId);
863
+ // Validate payload — graceful no-op for malformed invocations
864
+ if (activity.value?.actionName !== "feedback" || !reaction) {
865
+ return;
866
+ }
867
+ const syntheticText = buildFeedbackSyntheticText(reaction, comment);
868
+ // Turn coordination: if a turn is active, enqueue as steering follow-up
869
+ if (!_turnCoordinator.tryBeginTurn(turnKey)) {
870
+ _turnCoordinator.enqueueFollowUp(turnKey, {
871
+ conversationId: convId,
872
+ text: syntheticText,
873
+ receivedAt: Date.now(),
874
+ effect: (0, continuity_1.classifySteeringFollowUpEffect)(syntheticText),
875
+ });
876
+ return;
877
+ }
878
+ try {
879
+ const teamsContext = {
880
+ signin: async () => undefined,
881
+ aadObjectId: activity.from?.aadObjectId,
882
+ tenantId: activity.conversation?.tenantId,
883
+ displayName: activity.from?.name,
884
+ };
885
+ const ctxSend = async (t) => {
886
+ await ctx.send({ type: "message", text: t, replyToId: activity.replyToId, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } });
887
+ };
888
+ await handleTeamsMessage(syntheticText, stream, convId, teamsContext, ctxSend, { isReactionSignal: true, suppressEmptyStreamMessage: true });
889
+ }
890
+ catch (err) {
891
+ const msg = err instanceof Error ? err.message : String(err);
892
+ (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.feedback_handler_error", component: "channels", message: msg.slice(0, 200), meta: {} });
893
+ }
894
+ finally {
895
+ _turnCoordinator.endTurn(turnKey);
896
+ }
897
+ });
898
+ /* v8 ignore stop */
899
+ // Handle bot install — send welcome Adaptive Card with prompt starters.
900
+ /* v8 ignore start -- Teams SDK install handler; requires live SDK context @preserve */
901
+ app.on("install.add", async (ctx) => {
902
+ try {
903
+ const card = buildWelcomeCard();
904
+ await ctx.send({
905
+ type: "message",
906
+ attachments: [{
907
+ contentType: "application/vnd.microsoft.card.adaptive",
908
+ content: card,
909
+ }],
910
+ entities: aiLabelEntities(),
911
+ channelData: { feedbackLoopEnabled: true },
912
+ });
913
+ }
914
+ catch (err) {
915
+ const msg = err instanceof Error ? err.message : String(err);
916
+ (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.welcome_handler_error", component: "channels", message: msg.slice(0, 200), meta: {} });
917
+ }
593
918
  });
919
+ /* v8 ignore stop */
594
920
  app.on("message", async (ctx) => {
595
921
  const { stream, activity, api, signin } = ctx;
596
922
  const text = activity.text || "";
@@ -598,17 +924,41 @@ function startTeamsApp() {
598
924
  const turnKey = teamsTurnKey(convId);
599
925
  const userId = activity.from?.id || "";
600
926
  const channelId = activity.channelId || "msteams";
601
- (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.message_received", component: "channels", message: "incoming teams message", meta: { userId: userId.slice(0, 12), conversationId: convId.slice(0, 20) } });
602
- // Resolve pending confirmations IMMEDIATELY — before token fetches or
603
- // the conversation lock. The original message holds the lock while
604
- // awaiting confirmation, so acquiring it here would deadlock. Token
605
- // fetches are also unnecessary (and slow) for a simple yes/no reply.
606
- if (resolvePendingConfirmation(convId, text)) {
607
- // Don't emit on this stream — the original message's stream is still
608
- // active. Opening a second streaming response in the same conversation
609
- // can corrupt the first. The original stream will show tool progress
610
- // once the confirmation Promise resolves.
611
- return;
927
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.message_received", component: "channels", message: `[${label}] incoming teams message`, meta: { userId: userId.slice(0, 12), conversationId: convId.slice(0, 20) } });
928
+ const commandRegistry = createTeamsCommandRegistry();
929
+ const parsedSlashCommand = (0, commands_1.parseSlashCommand)(text);
930
+ if (parsedSlashCommand) {
931
+ const dispatchResult = commandRegistry.dispatch(parsedSlashCommand.command, { channel: "teams" });
932
+ if (dispatchResult.handled && dispatchResult.result) {
933
+ if (dispatchResult.result.action === "response") {
934
+ stream.emit(dispatchResult.result.message || "");
935
+ return;
936
+ }
937
+ if (dispatchResult.result.action === "new") {
938
+ const commandStore = getFriendStore();
939
+ const commandProvider = activity.from?.aadObjectId ? "aad" : "teams-conversation";
940
+ const commandExternalId = activity.from?.aadObjectId || convId;
941
+ const commandResolver = new resolver_1.FriendResolver(commandStore, {
942
+ provider: commandProvider,
943
+ externalId: commandExternalId,
944
+ tenantId: activity.conversation?.tenantId,
945
+ displayName: activity.from?.name || "Unknown",
946
+ channel: "teams",
947
+ });
948
+ const commandContext = await commandResolver.resolve();
949
+ (0, context_1.deleteSession)((0, config_2.sessionPath)(commandContext.friend.id, "teams", convId));
950
+ stream.emit("session cleared");
951
+ if (_turnCoordinator.isTurnActive(turnKey)) {
952
+ _turnCoordinator.enqueueFollowUp(turnKey, {
953
+ conversationId: convId,
954
+ text,
955
+ receivedAt: Date.now(),
956
+ effect: "clear_and_supersede",
957
+ });
958
+ }
959
+ return;
960
+ }
961
+ }
612
962
  }
613
963
  // If this conversation already has an active turn, steer follow-up input
614
964
  // into that turn and avoid starting a second concurrent turn.
@@ -617,31 +967,35 @@ function startTeamsApp() {
617
967
  conversationId: convId,
618
968
  text,
619
969
  receivedAt: Date.now(),
970
+ effect: (0, continuity_1.classifySteeringFollowUpEffect)(text),
620
971
  });
621
972
  return;
622
973
  }
623
974
  try {
975
+ // Resolve OAuth connection names for this user's tenant (supports per-tenant overrides).
976
+ const tenantId = activity.conversation?.tenantId;
977
+ const tenantOAuth = (0, config_1.resolveOAuthForTenant)(tenantId);
624
978
  // Fetch tokens for both OAuth connections independently.
625
979
  // Failures are silently caught -- the tool handler will request signin if needed.
626
980
  let graphToken;
627
981
  let adoToken;
628
982
  let githubToken;
629
983
  try {
630
- const graphRes = await api.users.token.get({ userId, connectionName: oauthConfig.graphConnectionName, channelId });
984
+ const graphRes = await api.users.token.get({ userId, connectionName: tenantOAuth.graphConnectionName, channelId });
631
985
  graphToken = graphRes?.token;
632
986
  }
633
987
  catch { /* no token yet — tool handler will trigger signin */ }
634
988
  try {
635
- const adoRes = await api.users.token.get({ userId, connectionName: oauthConfig.adoConnectionName, channelId });
989
+ const adoRes = await api.users.token.get({ userId, connectionName: tenantOAuth.adoConnectionName, channelId });
636
990
  adoToken = adoRes?.token;
637
991
  }
638
992
  catch { /* no token yet — tool handler will trigger signin */ }
639
993
  try {
640
- const githubRes = await api.users.token.get({ userId, connectionName: oauthConfig.githubConnectionName, channelId });
994
+ const githubRes = await api.users.token.get({ userId, connectionName: tenantOAuth.githubConnectionName, channelId });
641
995
  githubToken = githubRes?.token;
642
996
  }
643
997
  catch { /* no token yet — tool handler will trigger signin */ }
644
- (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.token_status", component: "channels", message: "oauth token availability", meta: { graph: !!graphToken, ado: !!adoToken, github: !!githubToken } });
998
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.token_status", component: "channels", message: "oauth token availability", meta: { graph: !!graphToken, ado: !!adoToken, github: !!githubToken, tenantId } });
645
999
  const teamsContext = {
646
1000
  graphToken,
647
1001
  adoToken,
@@ -661,12 +1015,17 @@ function startTeamsApp() {
661
1015
  aadObjectId: activity.from?.aadObjectId,
662
1016
  tenantId: activity.conversation?.tenantId,
663
1017
  displayName: activity.from?.name,
1018
+ graphConnectionName: tenantOAuth.graphConnectionName,
1019
+ adoConnectionName: tenantOAuth.adoConnectionName,
1020
+ githubConnectionName: tenantOAuth.githubConnectionName,
1021
+ /* v8 ignore next -- bot API availability branch; requires live SDK context @preserve */
1022
+ botApi: app.id && api ? { id: app.id, conversations: api.conversations } : undefined,
664
1023
  };
665
1024
  /* v8 ignore next 5 -- bot-framework integration callback; tested via handleTeamsMessage sendMessage path @preserve */
666
1025
  const ctxSend = async (t) => {
667
1026
  // Use send with replyToId (not reply, which adds a blockquote).
668
1027
  // replyToId anchors the message after the user's message in Copilot Chat.
669
- await ctx.send({ type: "message", text: t, replyToId: activity.id });
1028
+ await ctx.send({ type: "message", text: t, replyToId: activity.id, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } });
670
1029
  };
671
1030
  await handleTeamsMessage(text, stream, convId, teamsContext, ctxSend);
672
1031
  }
@@ -678,6 +1037,251 @@ function startTeamsApp() {
678
1037
  _turnCoordinator.endTurn(turnKey);
679
1038
  }
680
1039
  });
1040
+ app.event("error", ({ error }) => {
1041
+ const msg = error instanceof Error ? error.message : String(error);
1042
+ (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.app_error", component: "channels", message: `[${label}] ${msg}`, meta: {} });
1043
+ });
1044
+ }
1045
+ function findAadObjectId(friend) {
1046
+ for (const ext of friend.externalIds) {
1047
+ if (ext.provider === "aad" && !ext.externalId.startsWith("group:")) {
1048
+ return { aadObjectId: ext.externalId, tenantId: ext.tenantId };
1049
+ }
1050
+ }
1051
+ return undefined;
1052
+ }
1053
+ function resolveTeamsFriendStore(deps) {
1054
+ return deps.store
1055
+ ?? deps.createFriendStore?.()
1056
+ ?? new store_file_1.FileFriendStore(path.join((0, identity_1.getAgentRoot)(), "friends"));
1057
+ }
1058
+ function getTeamsConversations(botApi) {
1059
+ return botApi.conversations;
1060
+ }
1061
+ function hasExplicitCrossChatAuthorization(params) {
1062
+ return params.intent === "explicit_cross_chat"
1063
+ && types_1.TRUSTED_LEVELS.has(params.authorizingSession?.trustLevel ?? "stranger");
1064
+ }
1065
+ async function sendProactiveTeamsMessageToSession(params, deps) {
1066
+ const store = resolveTeamsFriendStore(deps);
1067
+ const conversations = getTeamsConversations(deps.botApi);
1068
+ let friend;
1069
+ try {
1070
+ friend = await store.get(params.friendId);
1071
+ }
1072
+ catch {
1073
+ friend = null;
1074
+ }
1075
+ if (!friend) {
1076
+ (0, runtime_1.emitNervesEvent)({
1077
+ level: "warn",
1078
+ component: "senses",
1079
+ event: "senses.teams_proactive_no_friend",
1080
+ message: "proactive send skipped: friend not found",
1081
+ meta: { friendId: params.friendId, sessionKey: params.sessionKey },
1082
+ });
1083
+ return { delivered: false, reason: "friend_not_found" };
1084
+ }
1085
+ if (!hasExplicitCrossChatAuthorization(params) && !types_1.TRUSTED_LEVELS.has(friend.trustLevel ?? "stranger")) {
1086
+ (0, runtime_1.emitNervesEvent)({
1087
+ component: "senses",
1088
+ event: "senses.teams_proactive_trust_skip",
1089
+ message: "proactive send skipped: trust level not allowed",
1090
+ meta: {
1091
+ friendId: params.friendId,
1092
+ trustLevel: friend.trustLevel ?? "unknown",
1093
+ intent: params.intent ?? "generic_outreach",
1094
+ authorizingTrustLevel: params.authorizingSession?.trustLevel ?? null,
1095
+ },
1096
+ });
1097
+ return { delivered: false, reason: "trust_skip" };
1098
+ }
1099
+ const aadInfo = findAadObjectId(friend);
1100
+ if (!aadInfo) {
1101
+ (0, runtime_1.emitNervesEvent)({
1102
+ level: "warn",
1103
+ component: "senses",
1104
+ event: "senses.teams_proactive_no_aad_id",
1105
+ message: "proactive send skipped: no AAD object ID found",
1106
+ meta: { friendId: params.friendId, sessionKey: params.sessionKey },
1107
+ });
1108
+ return { delivered: false, reason: "missing_target" };
1109
+ }
1110
+ const internalContentBlockReason = (0, proactive_content_guard_1.getProactiveInternalContentBlockReason)(params.text);
1111
+ if (internalContentBlockReason) {
1112
+ (0, proactive_content_guard_1.emitProactiveInternalContentBlocked)({
1113
+ friendId: params.friendId,
1114
+ sessionKey: params.sessionKey,
1115
+ reason: internalContentBlockReason,
1116
+ source: "session_send",
1117
+ });
1118
+ return { delivered: false, reason: "internal_content_blocked" };
1119
+ }
1120
+ try {
1121
+ const conversation = await conversations.create({
1122
+ bot: { id: deps.botApi.id },
1123
+ members: [{ id: aadInfo.aadObjectId, role: "user", name: friend.name || aadInfo.aadObjectId }],
1124
+ tenantId: aadInfo.tenantId,
1125
+ isGroup: false,
1126
+ });
1127
+ await conversations.activities(conversation.id).create({
1128
+ type: "message",
1129
+ text: params.text,
1130
+ });
1131
+ (0, runtime_1.emitNervesEvent)({
1132
+ component: "senses",
1133
+ event: "senses.teams_proactive_sent",
1134
+ message: "proactive teams message sent",
1135
+ meta: { friendId: params.friendId, aadObjectId: aadInfo.aadObjectId, sessionKey: params.sessionKey },
1136
+ });
1137
+ return { delivered: true };
1138
+ }
1139
+ catch (error) {
1140
+ (0, runtime_1.emitNervesEvent)({
1141
+ level: "error",
1142
+ component: "senses",
1143
+ event: "senses.teams_proactive_send_error",
1144
+ message: "proactive teams send failed",
1145
+ meta: {
1146
+ friendId: params.friendId,
1147
+ aadObjectId: aadInfo.aadObjectId,
1148
+ sessionKey: params.sessionKey,
1149
+ reason: error instanceof Error ? error.message : String(error),
1150
+ },
1151
+ });
1152
+ return { delivered: false, reason: "send_error" };
1153
+ }
1154
+ }
1155
+ function scanPendingTeamsFiles(pendingRoot) {
1156
+ const results = [];
1157
+ let friendIds;
1158
+ try {
1159
+ friendIds = fs.readdirSync(pendingRoot);
1160
+ }
1161
+ catch {
1162
+ return results;
1163
+ }
1164
+ for (const friendId of friendIds) {
1165
+ const teamsDir = path.join(pendingRoot, friendId, "teams");
1166
+ let keys;
1167
+ try {
1168
+ keys = fs.readdirSync(teamsDir);
1169
+ }
1170
+ catch {
1171
+ continue;
1172
+ }
1173
+ for (const key of keys) {
1174
+ const keyDir = path.join(teamsDir, key);
1175
+ let files;
1176
+ try {
1177
+ files = fs.readdirSync(keyDir);
1178
+ }
1179
+ catch {
1180
+ continue;
1181
+ }
1182
+ for (const file of files.filter((f) => f.endsWith(".json")).sort()) {
1183
+ const filePath = path.join(keyDir, file);
1184
+ try {
1185
+ const content = fs.readFileSync(filePath, "utf-8");
1186
+ results.push({ friendId, key, filePath, content });
1187
+ }
1188
+ catch {
1189
+ // skip unreadable files
1190
+ }
1191
+ }
1192
+ }
1193
+ }
1194
+ return results;
1195
+ }
1196
+ async function drainAndSendPendingTeams(store, botApi, pendingRoot) {
1197
+ const root = pendingRoot ?? path.join((0, identity_1.getAgentRoot)(), "state", "pending");
1198
+ const pendingFiles = scanPendingTeamsFiles(root);
1199
+ const result = { sent: 0, skipped: 0, failed: 0 };
1200
+ for (const { friendId, key, filePath, content } of pendingFiles) {
1201
+ let parsed;
1202
+ try {
1203
+ parsed = JSON.parse(content);
1204
+ }
1205
+ catch {
1206
+ result.failed++;
1207
+ try {
1208
+ fs.unlinkSync(filePath);
1209
+ }
1210
+ catch { /* ignore */ }
1211
+ continue;
1212
+ }
1213
+ const messageText = typeof parsed.content === "string" ? parsed.content : "";
1214
+ if (!messageText.trim()) {
1215
+ result.skipped++;
1216
+ try {
1217
+ fs.unlinkSync(filePath);
1218
+ }
1219
+ catch { /* ignore */ }
1220
+ continue;
1221
+ }
1222
+ const internalBlockReason = (0, proactive_content_guard_1.getProactiveInternalContentBlockReason)(messageText);
1223
+ if (internalBlockReason) {
1224
+ result.skipped++;
1225
+ try {
1226
+ fs.unlinkSync(filePath);
1227
+ }
1228
+ catch { /* ignore */ }
1229
+ (0, proactive_content_guard_1.emitProactiveInternalContentBlocked)({
1230
+ friendId,
1231
+ reason: internalBlockReason,
1232
+ source: "pending_drain",
1233
+ });
1234
+ continue;
1235
+ }
1236
+ const sendResult = await sendProactiveTeamsMessageToSession({
1237
+ friendId,
1238
+ sessionKey: key,
1239
+ text: messageText,
1240
+ intent: "generic_outreach",
1241
+ }, {
1242
+ botApi,
1243
+ store,
1244
+ });
1245
+ if (sendResult.delivered) {
1246
+ result.sent++;
1247
+ try {
1248
+ fs.unlinkSync(filePath);
1249
+ }
1250
+ catch { /* ignore */ }
1251
+ continue;
1252
+ }
1253
+ if (sendResult.reason === "friend_not_found" || sendResult.reason === "trust_skip" || sendResult.reason === "missing_target") {
1254
+ result.skipped++;
1255
+ try {
1256
+ fs.unlinkSync(filePath);
1257
+ }
1258
+ catch { /* ignore */ }
1259
+ continue;
1260
+ }
1261
+ result.failed++;
1262
+ }
1263
+ if (result.sent > 0 || result.skipped > 0 || result.failed > 0) {
1264
+ (0, runtime_1.emitNervesEvent)({
1265
+ component: "senses",
1266
+ event: "senses.teams_proactive_drain_complete",
1267
+ message: "teams proactive drain complete",
1268
+ meta: { sent: result.sent, skipped: result.skipped, failed: result.failed },
1269
+ });
1270
+ }
1271
+ return result;
1272
+ }
1273
+ // Start the Teams app in DevtoolsPlugin mode (local dev) or Bot Service mode (real Teams).
1274
+ // Mode is determined by getTeamsConfig().clientId.
1275
+ // Text is always accumulated in textBuffer and flushed periodically (chunked streaming).
1276
+ //
1277
+ // Dual-bot support: if teamsSecondary is configured with a clientId, a second App
1278
+ // instance starts on an internal port and the primary app proxies requests from
1279
+ // /api/messages-secondary to it. This lets a single App Service serve two bot
1280
+ // registrations (e.g. one per tenant) without SDK modifications.
1281
+ function startTeamsApp() {
1282
+ const teamsConfig = (0, config_2.getTeamsConfig)();
1283
+ const { app, mode } = createBotApp(teamsConfig);
1284
+ registerBotHandlers(app, "primary");
681
1285
  if (!process.listeners("unhandledRejection").some((l) => l.__agentHandler)) {
682
1286
  const handler = (err) => {
683
1287
  const msg = err instanceof Error ? err.message : String(err);
@@ -686,11 +1290,54 @@ function startTeamsApp() {
686
1290
  handler.__agentHandler = true;
687
1291
  process.on("unhandledRejection", handler);
688
1292
  }
689
- app.event("error", ({ error }) => {
690
- const msg = error instanceof Error ? error.message : String(error);
691
- (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.app_error", component: "channels", message: msg, meta: {} });
692
- });
693
- const port = (0, config_2.getTeamsChannelConfig)().port;
1293
+ /* v8 ignore next -- PORT env branch; runtime-only @preserve */
1294
+ const port = process.env.PORT ? Number(process.env.PORT) : (0, config_2.getTeamsChannelConfig)().port;
694
1295
  app.start(port);
695
- (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.app_started", component: "channels", message: `Teams bot started on port ${port} with ${mode} (chunked streaming)`, meta: { port, mode } });
1296
+ // Diagnostic: log tool count at startup to verify deploy
1297
+ const startupTools = (0, tools_1.getToolsForChannel)((0, channel_1.getChannelCapabilities)("teams"));
1298
+ const toolNames = startupTools.map((t) => t.function.name);
1299
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.app_started", component: "channels", message: `Teams bot started on port ${port} with ${mode} (chunked streaming)`, meta: { port, mode, toolCount: toolNames.length, hasProactive: toolNames.includes("teams_send_message") } });
1300
+ // --- Secondary bot (dual-bot support) ---
1301
+ // If teamsSecondary has a clientId, start a second App on an internal port
1302
+ // and proxy /api/messages-secondary on the primary app to it.
1303
+ /* v8 ignore start -- dual-bot proxy wiring; requires live Teams SDK + HTTP @preserve */
1304
+ const secondaryConfig = (0, config_1.getTeamsSecondaryConfig)();
1305
+ if (secondaryConfig.clientId) {
1306
+ const { app: secondaryApp, mode: secondaryMode } = createBotApp(secondaryConfig);
1307
+ registerBotHandlers(secondaryApp, "secondary");
1308
+ secondaryApp.start(SECONDARY_INTERNAL_PORT);
1309
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.app_started", component: "channels", message: `Secondary bot started on internal port ${SECONDARY_INTERNAL_PORT} with ${secondaryMode}`, meta: { port: SECONDARY_INTERNAL_PORT, mode: secondaryMode } });
1310
+ // Proxy: forward /api/messages-secondary on the primary app's Express
1311
+ // to localhost:SECONDARY_INTERNAL_PORT/api/messages.
1312
+ // The SDK's HttpPlugin exposes .post() bound to its Express instance.
1313
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
1314
+ const httpPlugin = app.http;
1315
+ httpPlugin.post("/api/messages-secondary", (req, res) => {
1316
+ const body = JSON.stringify(req.body);
1317
+ const proxyReq = http.request({
1318
+ hostname: "127.0.0.1",
1319
+ port: SECONDARY_INTERNAL_PORT,
1320
+ path: "/api/messages",
1321
+ method: "POST",
1322
+ headers: {
1323
+ ...req.headers,
1324
+ "content-length": Buffer.byteLength(body).toString(),
1325
+ },
1326
+ }, (proxyRes) => {
1327
+ res.writeHead(proxyRes.statusCode ?? 200, proxyRes.headers);
1328
+ proxyRes.pipe(res);
1329
+ });
1330
+ proxyReq.on("error", (err) => {
1331
+ (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.proxy_error", component: "channels", message: `secondary proxy error: ${err.message}`, meta: {} });
1332
+ if (!res.headersSent) {
1333
+ res.writeHead(502);
1334
+ res.end("Bad Gateway");
1335
+ }
1336
+ });
1337
+ proxyReq.write(body);
1338
+ proxyReq.end();
1339
+ });
1340
+ (0, runtime_1.emitNervesEvent)({ level: "info", event: "channel.proxy_ready", component: "channels", message: "proxy /api/messages-secondary → secondary bot ready", meta: {} });
1341
+ }
1342
+ /* v8 ignore stop */
696
1343
  }