@ouro.bot/cli 0.1.0-alpha.37 → 0.1.0-alpha.370

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 (329) hide show
  1. package/README.md +106 -14
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +2222 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +832 -0
  16. package/dist/heart/agent-entry.js +37 -2
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +378 -0
  28. package/dist/heart/bridges/manager.js +358 -0
  29. package/dist/heart/bridges/state-machine.js +135 -0
  30. package/dist/heart/bridges/store.js +123 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +107 -61
  35. package/dist/heart/core.js +803 -259
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +382 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +205 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +538 -0
  44. package/dist/heart/daemon/cli-exec.js +3081 -0
  45. package/dist/heart/daemon/cli-help.js +312 -0
  46. package/dist/heart/daemon/cli-parse.js +1023 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +560 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/daemon-cli.js +29 -1171
  51. package/dist/heart/daemon/daemon-entry.js +356 -3
  52. package/dist/heart/daemon/daemon-health.js +141 -0
  53. package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
  54. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  55. package/dist/heart/daemon/daemon.js +757 -58
  56. package/dist/heart/daemon/doctor-types.js +8 -0
  57. package/dist/heart/daemon/doctor.js +445 -0
  58. package/dist/heart/daemon/health-monitor.js +79 -1
  59. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  60. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  61. package/dist/heart/daemon/http-health-probe.js +80 -0
  62. package/dist/heart/daemon/inner-status.js +89 -0
  63. package/dist/heart/daemon/interactive-repair.js +148 -0
  64. package/dist/heart/daemon/launchd.js +46 -9
  65. package/dist/heart/daemon/log-tailer.js +82 -12
  66. package/dist/heart/daemon/logs-prune.js +105 -0
  67. package/dist/heart/daemon/message-router.js +17 -8
  68. package/dist/heart/daemon/os-cron-deps.js +134 -0
  69. package/dist/heart/daemon/ouro-bot-entry.js +1 -1
  70. package/dist/heart/daemon/process-manager.js +201 -0
  71. package/dist/heart/daemon/provider-discovery.js +113 -0
  72. package/dist/heart/daemon/pulse.js +475 -0
  73. package/dist/heart/daemon/run-hooks.js +2 -0
  74. package/dist/heart/daemon/runtime-logging.js +67 -16
  75. package/dist/heart/daemon/runtime-metadata.js +101 -0
  76. package/dist/heart/daemon/runtime-mode.js +67 -0
  77. package/dist/heart/daemon/safe-mode.js +161 -0
  78. package/dist/heart/daemon/sense-manager.js +72 -3
  79. package/dist/heart/daemon/session-id-resolver.js +131 -0
  80. package/dist/heart/daemon/skill-management-installer.js +94 -0
  81. package/dist/heart/daemon/socket-client.js +307 -0
  82. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  83. package/dist/heart/daemon/startup-tui.js +237 -0
  84. package/dist/heart/daemon/task-scheduler.js +3 -25
  85. package/dist/heart/daemon/thoughts.js +510 -0
  86. package/dist/heart/daemon/up-progress.js +135 -0
  87. package/dist/heart/delegation.js +62 -0
  88. package/dist/heart/habits/habit-migration.js +181 -0
  89. package/dist/heart/habits/habit-parser.js +140 -0
  90. package/dist/heart/habits/habit-scheduler.js +371 -0
  91. package/dist/heart/{daemon → hatch}/hatch-flow.js +55 -126
  92. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  93. package/dist/heart/{daemon → hatch}/specialist-prompt.js +11 -8
  94. package/dist/heart/{daemon → hatch}/specialist-tools.js +77 -11
  95. package/dist/heart/identity.js +154 -59
  96. package/dist/heart/kept-notes.js +357 -0
  97. package/dist/heart/kicks.js +2 -20
  98. package/dist/heart/machine-identity.js +161 -0
  99. package/dist/heart/mcp/mcp-server.js +653 -0
  100. package/dist/heart/migrate-config.js +100 -0
  101. package/dist/heart/model-capabilities.js +59 -0
  102. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  103. package/dist/heart/outlook/outlook-http-response.js +7 -0
  104. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  105. package/dist/heart/outlook/outlook-http-static.js +99 -0
  106. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  107. package/dist/heart/outlook/outlook-http.js +99 -0
  108. package/dist/heart/outlook/outlook-read.js +28 -0
  109. package/dist/heart/outlook/outlook-types.js +27 -0
  110. package/dist/heart/outlook/outlook-view.js +195 -0
  111. package/dist/heart/outlook/readers/agent-machine.js +359 -0
  112. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  113. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  114. package/dist/heart/outlook/readers/sessions.js +232 -0
  115. package/dist/heart/outlook/readers/shared.js +111 -0
  116. package/dist/heart/platform.js +81 -0
  117. package/dist/heart/progress-story.js +42 -0
  118. package/dist/heart/provider-attempt.js +133 -0
  119. package/dist/heart/provider-binding-resolver.js +239 -0
  120. package/dist/heart/provider-credentials.js +379 -0
  121. package/dist/heart/provider-failover.js +266 -0
  122. package/dist/heart/provider-models.js +81 -0
  123. package/dist/heart/provider-ping.js +237 -0
  124. package/dist/heart/provider-state.js +216 -0
  125. package/dist/heart/provider-visibility.js +180 -0
  126. package/dist/heart/providers/anthropic-token.js +131 -0
  127. package/dist/heart/providers/anthropic.js +193 -55
  128. package/dist/heart/providers/azure.js +103 -12
  129. package/dist/heart/providers/error-classification.js +63 -0
  130. package/dist/heart/providers/github-copilot.js +145 -0
  131. package/dist/heart/providers/minimax-vlm.js +189 -0
  132. package/dist/heart/providers/minimax.js +29 -7
  133. package/dist/heart/providers/openai-codex.js +39 -29
  134. package/dist/heart/session-activity.js +190 -0
  135. package/dist/heart/session-events.js +855 -0
  136. package/dist/heart/session-transcript.js +167 -0
  137. package/dist/heart/start-of-turn-packet.js +345 -0
  138. package/dist/heart/streaming.js +36 -27
  139. package/dist/heart/sync.js +332 -0
  140. package/dist/heart/target-resolution.js +127 -0
  141. package/dist/heart/tempo.js +93 -0
  142. package/dist/heart/temporal-view.js +41 -0
  143. package/dist/heart/tool-activity-callbacks.js +36 -0
  144. package/dist/heart/tool-description.js +135 -0
  145. package/dist/heart/tool-friction.js +55 -0
  146. package/dist/heart/tool-loop.js +200 -0
  147. package/dist/heart/turn-context.js +362 -0
  148. package/dist/heart/turn-coordinator.js +28 -0
  149. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  150. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  151. package/dist/heart/versioning/ouro-path-installer.js +301 -0
  152. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  153. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  154. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  155. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  156. package/dist/mind/bundle-manifest.js +7 -1
  157. package/dist/mind/context.js +141 -94
  158. package/dist/mind/diary-integrity.js +60 -0
  159. package/dist/mind/{memory.js → diary.js} +84 -96
  160. package/dist/mind/embedding-provider.js +60 -0
  161. package/dist/mind/file-state.js +179 -0
  162. package/dist/mind/first-impressions.js +14 -1
  163. package/dist/mind/friends/channel.js +56 -0
  164. package/dist/mind/friends/group-context.js +144 -0
  165. package/dist/mind/friends/resolver.js +38 -1
  166. package/dist/mind/friends/store-file.js +58 -3
  167. package/dist/mind/friends/trust-explanation.js +74 -0
  168. package/dist/mind/friends/types.js +9 -1
  169. package/dist/mind/journal-index.js +161 -0
  170. package/dist/mind/note-search.js +268 -0
  171. package/dist/mind/obligation-steering.js +221 -0
  172. package/dist/mind/pending.js +74 -7
  173. package/dist/mind/prompt-refresh.js +3 -2
  174. package/dist/mind/prompt.js +1030 -118
  175. package/dist/mind/provenance-trust.js +26 -0
  176. package/dist/mind/scrutiny.js +173 -0
  177. package/dist/mind/token-estimate.js +8 -12
  178. package/dist/nerves/cli-logging.js +7 -1
  179. package/dist/nerves/coverage/audit-rules.js +15 -6
  180. package/dist/nerves/coverage/audit.js +28 -2
  181. package/dist/nerves/coverage/cli.js +1 -1
  182. package/dist/nerves/coverage/file-completeness.js +83 -5
  183. package/dist/nerves/coverage/run-artifacts.js +1 -1
  184. package/dist/nerves/event-buffer.js +111 -0
  185. package/dist/nerves/index.js +224 -4
  186. package/dist/nerves/observation.js +20 -0
  187. package/dist/nerves/redact.js +79 -0
  188. package/dist/nerves/runtime.js +5 -1
  189. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  190. package/dist/outlook-ui/assets/index-xTdv64BV.js +61 -0
  191. package/dist/outlook-ui/index.html +15 -0
  192. package/dist/repertoire/ado-client.js +15 -56
  193. package/dist/repertoire/ado-semantic.js +11 -10
  194. package/dist/repertoire/api-client.js +97 -0
  195. package/dist/repertoire/bitwarden-store.js +365 -0
  196. package/dist/repertoire/bundle-templates.js +72 -0
  197. package/dist/repertoire/bw-installer.js +79 -0
  198. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  199. package/dist/repertoire/coding/context-pack.js +330 -0
  200. package/dist/repertoire/coding/feedback.js +197 -30
  201. package/dist/repertoire/coding/manager.js +158 -9
  202. package/dist/repertoire/coding/spawner.js +55 -9
  203. package/dist/repertoire/coding/tools.js +170 -7
  204. package/dist/repertoire/commerce-errors.js +109 -0
  205. package/dist/repertoire/commerce-self-test.js +156 -0
  206. package/dist/repertoire/credential-access.js +107 -0
  207. package/dist/repertoire/duffel-client.js +185 -0
  208. package/dist/repertoire/github-client.js +14 -55
  209. package/dist/repertoire/graph-client.js +11 -52
  210. package/dist/repertoire/guardrails.js +375 -0
  211. package/dist/repertoire/mcp-client.js +255 -0
  212. package/dist/repertoire/mcp-manager.js +305 -0
  213. package/dist/repertoire/mcp-tools.js +63 -0
  214. package/dist/repertoire/shell-sessions.js +133 -0
  215. package/dist/repertoire/skills.js +15 -24
  216. package/dist/repertoire/stripe-client.js +131 -0
  217. package/dist/repertoire/tasks/board.js +43 -5
  218. package/dist/repertoire/tasks/fix.js +182 -0
  219. package/dist/repertoire/tasks/index.js +28 -10
  220. package/dist/repertoire/tasks/lifecycle.js +2 -2
  221. package/dist/repertoire/tasks/parser.js +3 -2
  222. package/dist/repertoire/tasks/scanner.js +194 -37
  223. package/dist/repertoire/tasks/transitions.js +16 -79
  224. package/dist/repertoire/tool-results.js +29 -0
  225. package/dist/repertoire/tools-attachments.js +317 -0
  226. package/dist/repertoire/tools-base.js +45 -771
  227. package/dist/repertoire/tools-bluebubbles.js +1 -0
  228. package/dist/repertoire/tools-bridge.js +141 -0
  229. package/dist/repertoire/tools-bundle.js +984 -0
  230. package/dist/repertoire/tools-config.js +185 -0
  231. package/dist/repertoire/tools-continuity.js +248 -0
  232. package/dist/repertoire/tools-credential.js +182 -0
  233. package/dist/repertoire/tools-files.js +342 -0
  234. package/dist/repertoire/tools-flight.js +224 -0
  235. package/dist/repertoire/tools-flow.js +105 -0
  236. package/dist/repertoire/tools-github.js +1 -7
  237. package/dist/repertoire/tools-notes.js +376 -0
  238. package/dist/repertoire/tools-session.js +739 -0
  239. package/dist/repertoire/tools-shell.js +120 -0
  240. package/dist/repertoire/tools-stripe.js +180 -0
  241. package/dist/repertoire/tools-surface.js +243 -0
  242. package/dist/repertoire/tools-teams.js +12 -62
  243. package/dist/repertoire/tools-travel.js +125 -0
  244. package/dist/repertoire/tools-user-profile.js +144 -0
  245. package/dist/repertoire/tools-vault.js +40 -0
  246. package/dist/repertoire/tools.js +144 -138
  247. package/dist/repertoire/travel-api-client.js +360 -0
  248. package/dist/repertoire/user-profile.js +118 -0
  249. package/dist/repertoire/vault-setup.js +241 -0
  250. package/dist/repertoire/vault-unlock.js +359 -0
  251. package/dist/scripts/claude-code-hook.js +41 -0
  252. package/dist/scripts/claude-code-stop-hook.js +47 -0
  253. package/dist/senses/attention-queue.js +116 -0
  254. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  255. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  256. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +260 -9
  257. package/dist/senses/bluebubbles/entry.js +13 -0
  258. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  259. package/dist/senses/bluebubbles/index.js +1620 -0
  260. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  261. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  262. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  263. package/dist/senses/bluebubbles/replay.js +129 -0
  264. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  265. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  266. package/dist/senses/cli/bracketed-paste.js +82 -0
  267. package/dist/senses/cli/image-paste.js +287 -0
  268. package/dist/senses/cli/image-ref-navigation.js +75 -0
  269. package/dist/senses/cli/ink-app.js +156 -0
  270. package/dist/senses/cli/inline-diff.js +64 -0
  271. package/dist/senses/cli/input-keys.js +174 -0
  272. package/dist/senses/cli/kill-ring.js +86 -0
  273. package/dist/senses/cli/message-list.js +51 -0
  274. package/dist/senses/cli/ouro-tui.js +605 -0
  275. package/dist/senses/cli/spinner-imperative.js +135 -0
  276. package/dist/senses/cli/spinner.js +101 -0
  277. package/dist/senses/cli/status-line.js +60 -0
  278. package/dist/senses/cli/streaming-markdown.js +526 -0
  279. package/dist/senses/cli/tool-display.js +83 -0
  280. package/dist/senses/cli/tool-render.js +85 -0
  281. package/dist/senses/cli/tui-store.js +240 -0
  282. package/dist/senses/cli/virtual-list.js +35 -0
  283. package/dist/senses/cli-entry.js +1 -1
  284. package/dist/senses/cli-layout.js +187 -0
  285. package/dist/senses/cli.js +588 -250
  286. package/dist/senses/commands.js +66 -3
  287. package/dist/senses/continuity.js +94 -0
  288. package/dist/senses/habit-turn-message.js +108 -0
  289. package/dist/senses/inner-dialog-worker.js +112 -19
  290. package/dist/senses/inner-dialog.js +636 -86
  291. package/dist/senses/pipeline.js +602 -0
  292. package/dist/senses/proactive-content-guard.js +51 -0
  293. package/dist/senses/shared-turn.js +205 -0
  294. package/dist/senses/surface-tool.js +68 -0
  295. package/dist/senses/teams.js +693 -160
  296. package/dist/senses/trust-gate.js +112 -2
  297. package/package.json +29 -7
  298. package/skills/agent-commerce.md +106 -0
  299. package/skills/browser-navigation.md +110 -0
  300. package/skills/commerce-setup-guide.md +116 -0
  301. package/skills/commerce-setup.md +84 -0
  302. package/skills/configure-dev-tools.md +101 -0
  303. package/skills/travel-planning.md +138 -0
  304. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  305. package/dist/heart/daemon/subagent-installer.js +0 -134
  306. package/dist/mind/associative-recall.js +0 -197
  307. package/dist/senses/bluebubbles-entry.js +0 -11
  308. package/dist/senses/bluebubbles.js +0 -558
  309. package/dist/senses/debug-activity.js +0 -127
  310. package/subagents/README.md +0 -60
  311. package/subagents/work-doer.md +0 -235
  312. package/subagents/work-merger.md +0 -618
  313. package/subagents/work-planner.md +0 -382
  314. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  315. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  316. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  317. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  318. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  319. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  320. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  321. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  322. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  323. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  324. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  325. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  326. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  327. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  328. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  329. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -34,13 +34,19 @@ 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");
@@ -55,14 +61,34 @@ const context_1 = require("../mind/context");
55
61
  const commands_1 = require("./commands");
56
62
  const nerves_1 = require("../nerves");
57
63
  const runtime_1 = require("../nerves/runtime");
64
+ const proactive_content_guard_1 = require("./proactive-content-guard");
58
65
  const store_file_1 = require("../mind/friends/store-file");
66
+ const types_1 = require("../mind/friends/types");
59
67
  const resolver_1 = require("../mind/friends/resolver");
60
68
  const tokens_1 = require("../mind/friends/tokens");
61
69
  const turn_coordinator_1 = require("../heart/turn-coordinator");
62
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");
63
75
  const http = __importStar(require("http"));
64
76
  const path = __importStar(require("path"));
65
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
+ }
66
92
  // Strip @mention markup from incoming messages.
67
93
  // Removes <at>...</at> tags and trims extra whitespace.
68
94
  // Fallback safety net -- the SDK's activity.mentions.stripText should handle
@@ -117,6 +143,46 @@ function splitMessage(text, maxLen) {
117
143
  }
118
144
  return chunks;
119
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
+ }
120
186
  // Create Teams-specific callbacks for the agent loop.
121
187
  // The SDK handles cumulative text, debouncing (500ms), and the streaming
122
188
  // protocol (streamSequence, streamId, informative/streaming/final types).
@@ -129,12 +195,16 @@ function splitMessage(text, maxLen) {
129
195
  // (transient status) or safeSend (terminal errors). Reasoning is accumulated
130
196
  // and periodically pushed via safeUpdate on the same flush timer tick.
131
197
  function createTeamsCallbacks(stream, controller, sendMessage, options) {
198
+ const MIN_INITIAL_CHARS = 20;
132
199
  let stopped = false; // set when stream signals cancellation (403)
133
200
  let hadToolRun = false;
134
201
  let hadRealOutput = false; // true once reasoning/tool output shown; suppresses phrases
135
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
136
205
  let textBuffer = ""; // accumulated text output for chunked streaming
137
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
138
208
  let phraseTimer = null;
139
209
  let lastPhrase = "";
140
210
  let flushTimer = null;
@@ -184,15 +254,16 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
184
254
  result.catch(() => markStopped());
185
255
  }
186
256
  }
187
- // Safely emit a text delta to the stream.
257
+ // Safely emit a text delta to the stream with AI labels.
188
258
  // On error (e.g. 403 from Teams stop button), abort the controller.
189
259
  function safeEmit(text) {
190
260
  /* v8 ignore next -- defensive guard: stopped set by prior 403; tested via flush abort path @preserve */
191
261
  if (stopped)
192
262
  return;
193
263
  try {
194
- catchAsync(stream.emit(text));
264
+ catchAsync(stream.emit({ text, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } }));
195
265
  streamHasContent = true;
266
+ totalEmitted += text.length;
196
267
  }
197
268
  catch {
198
269
  markStopped();
@@ -207,7 +278,7 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
207
278
  try {
208
279
  // stream.emit() is typed as void but the Teams SDK returns a Promise
209
280
  // internally (async HTTP). Cast to capture the result for awaiting.
210
- const result = stream.emit(text);
281
+ const result = stream.emit({ text, entities: aiLabelEntities(), channelData: { feedbackLoopEnabled: true } });
211
282
  streamHasContent = true;
212
283
  if (result && typeof result.then === "function") {
213
284
  await result;
@@ -263,11 +334,49 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
263
334
  // emitted text into a single streaming message (cumulative), so every
264
335
  // periodic flush appends to the same response — not separate messages.
265
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).
266
340
  function flushTextBuffer() {
267
341
  if (!textBuffer)
268
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
+ }
269
374
  safeEmit(textBuffer);
270
375
  textBuffer = "";
376
+ if (!firstContentEmitted) {
377
+ firstContentEmitted = true;
378
+ stopPhraseRotation();
379
+ }
271
380
  }
272
381
  function startPhraseRotation(pool) {
273
382
  stopPhraseRotation();
@@ -310,32 +419,45 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
310
419
  onTextChunk: (text) => {
311
420
  if (stopped)
312
421
  return;
313
- 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).
314
424
  textBuffer += text;
315
425
  startFlushTimer();
316
426
  },
317
427
  onClearText: () => {
318
428
  textBuffer = "";
319
429
  },
320
- onToolStart: (name, args) => {
321
- stopPhraseRotation();
322
- flushTextBuffer();
323
- // Emit a placeholder to satisfy the 15s Copilot timeout for initial
324
- // stream.emit(). Without this, long tool chains (e.g. ADO batch ops)
325
- // never emit before the timeout and the user sees "this response was
326
- // stopped". The placeholder is replaced by actual content on next emit.
327
- // https://learn.microsoft.com/en-us/answers/questions/2288017/m365-custom-engine-agents-timeout-message-after-15
328
- if (!streamHasContent)
329
- safeEmit("⏳");
330
- const argSummary = (0, tools_1.summarizeArgs)(name, args) || Object.keys(args).join(", ");
331
- safeUpdate(`running ${name} (${argSummary})...`);
332
- hadToolRun = true;
333
- },
334
- onToolEnd: (name, summary, success) => {
335
- stopPhraseRotation();
336
- const msg = (0, format_1.formatToolResult)(name, summary, success);
337
- safeUpdate(msg);
338
- },
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
+ })(),
339
461
  onKick: () => {
340
462
  stopPhraseRotation();
341
463
  const msg = (0, format_1.formatKick)();
@@ -345,7 +467,11 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
345
467
  stopPhraseRotation();
346
468
  if (stopped)
347
469
  return;
348
- 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
+ }));
349
475
  if (severity === "transient") {
350
476
  safeUpdate(msg);
351
477
  }
@@ -353,38 +479,27 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
353
479
  safeSend(msg);
354
480
  }
355
481
  },
356
- onConfirmAction: options?.conversationId
357
- ? async (name, args) => {
358
- const convId = options.conversationId;
359
- const argsDesc = Object.entries(args).map(([k, v]) => `${k}: ${v}`).join(", ");
360
- safeUpdate(`Confirm action: ${name} (${argsDesc}) -- reply "yes" to confirm or "no" to cancel`);
361
- return new Promise((resolve) => {
362
- _pendingConfirmations.set(convId, resolve);
363
- // Auto-deny after 2 minutes to prevent indefinite blocking
364
- // (e.g. when the stream dies and the user never sees the prompt).
365
- setTimeout(() => {
366
- if (_pendingConfirmations.has(convId)) {
367
- _pendingConfirmations.delete(convId);
368
- resolve("denied");
369
- }
370
- }, 120_000);
371
- });
372
- }
373
- : undefined,
374
482
  flush: async () => {
375
483
  stopFlushTimer();
484
+ stopPhraseRotation();
376
485
  if (textBuffer) {
486
+ // Bypass MIN_INITIAL_CHARS threshold — flush delivers all remaining content
487
+ firstContentEmitted = true;
377
488
  const text = textBuffer;
378
489
  textBuffer = "";
379
- 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) {
380
495
  // Stream is alive — await the emit so we can catch async 413/failure
381
496
  // and fall through to sendMessage recovery.
382
497
  const ok = await tryEmit(text);
383
498
  if (!ok)
384
499
  markStopped();
385
500
  }
386
- if (stopped && sendMessage) {
387
- // 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.
388
503
  try {
389
504
  await sendMessage(text);
390
505
  }
@@ -395,32 +510,12 @@ function createTeamsCallbacks(stream, controller, sendMessage, options) {
395
510
  }
396
511
  }
397
512
  }
398
- else if (!streamHasContent) {
399
- 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)");
400
515
  }
401
516
  },
402
517
  };
403
518
  }
404
- // Per-conversation pending confirmation resolvers.
405
- // When a mutate tool needs confirmation, the resolver is stored here.
406
- // The next message from the same conversation resolves it.
407
- const _pendingConfirmations = new Map();
408
- // Confirmation response words (case-insensitive)
409
- const CONFIRM_WORDS = new Set(["yes", "confirm", "go", "y", "ok", "approve", "proceed"]);
410
- function resolvePendingConfirmation(convId, text) {
411
- const resolver = _pendingConfirmations.get(convId);
412
- if (!resolver)
413
- return false;
414
- _pendingConfirmations.delete(convId);
415
- const word = text.trim().toLowerCase();
416
- if (CONFIRM_WORDS.has(word)) {
417
- resolver("confirmed");
418
- }
419
- else {
420
- resolver("denied");
421
- }
422
- return true;
423
- }
424
519
  const _turnCoordinator = (0, turn_coordinator_1.createTurnCoordinator)();
425
520
  function teamsTurnKey(conversationId) {
426
521
  return `teams:${conversationId}`;
@@ -434,118 +529,233 @@ function getFriendStore() {
434
529
  const friendsPath = path.join((0, identity_1.getAgentRoot)(), "friends");
435
530
  return new store_file_1.FileFriendStore(friendsPath);
436
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 */
437
562
  // Handle an incoming Teams message
438
- async function handleTeamsMessage(text, stream, conversationId, teamsContext, sendMessage) {
563
+ async function handleTeamsMessage(text, stream, conversationId, teamsContext, sendMessage, reactionOverrides) {
439
564
  const turnKey = teamsTurnKey(conversationId);
440
565
  // NOTE: Confirmation resolution is handled in the app.on("message") handler
441
566
  // BEFORE the conversation lock. By the time we get here, any pending
442
567
  // confirmation has already been resolved and the reply consumed.
443
568
  // Send first thinking phrase immediately so the user sees feedback
444
569
  // before sync I/O (session load, trim) blocks the event loop.
445
- 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
+ }
446
574
  await new Promise(r => setImmediate(r));
447
- // 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
448
576
  const store = getFriendStore();
449
577
  const provider = teamsContext?.aadObjectId ? "aad" : "teams-conversation";
450
578
  const externalId = teamsContext?.aadObjectId || conversationId;
451
- 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 ? {
452
600
  graphToken: teamsContext.graphToken,
453
601
  adoToken: teamsContext.adoToken,
454
602
  githubToken: teamsContext.githubToken,
455
603
  signin: teamsContext.signin,
456
- friendStore: store,
457
- summarize: (0, core_1.createSummarize)(),
604
+ summarize: (0, core_1.createSummarize)("human"),
458
605
  tenantId: teamsContext.tenantId,
459
606
  botApi: teamsContext.botApi,
460
- } : undefined;
461
- if (toolContext) {
462
- const resolver = new resolver_1.FriendResolver(store, {
463
- provider,
464
- externalId,
465
- tenantId: teamsContext?.tenantId,
466
- displayName: teamsContext?.displayName || "Unknown",
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)({
467
646
  channel: "teams",
468
- });
469
- toolContext.context = await resolver.resolve();
470
- }
471
- const friendId = toolContext?.context?.friend?.id || "default";
472
- if (toolContext?.context?.friend) {
473
- const trustGate = (0, trust_gate_1.enforceTrustGate)({
474
- 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,
475
670
  provider,
476
671
  externalId,
477
672
  tenantId: teamsContext?.tenantId,
478
- 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,
479
696
  });
480
- if (!trustGate.allowed) {
481
- if (trustGate.reason === "stranger_first_reply") {
482
- stream.emit(trustGate.autoReply);
483
- }
484
- return;
485
- }
486
- }
487
- const registry = (0, commands_1.createCommandRegistry)();
488
- (0, commands_1.registerDefaultCommands)(registry);
489
- // Check for slash commands
490
- const parsed = (0, commands_1.parseSlashCommand)(text);
491
- if (parsed) {
492
- const dispatchResult = registry.dispatch(parsed.command, { channel: "teams" });
493
- if (dispatchResult.handled && dispatchResult.result) {
494
- if (dispatchResult.result.action === "new") {
495
- const sessPath = (0, config_2.sessionPath)(friendId, "teams", conversationId);
697
+ // ── Handle pipeline-intercepted commands ────────────────────────
698
+ if (result.turnOutcome === "command") {
699
+ if (result.commandAction === "new") {
496
700
  (0, context_1.deleteSession)(sessPath);
497
701
  stream.emit("session cleared");
498
- return;
499
702
  }
500
- else if (dispatchResult.result.action === "response") {
501
- stream.emit(dispatchResult.result.message || "");
502
- 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);
503
719
  }
720
+ return;
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;
504
753
  }
754
+ if (handleTeamsSlashCommand(supersedingFollowUp.text, createTeamsCommandRegistry(), friendId, conversationId, stream, false)) {
755
+ return;
756
+ }
757
+ currentText = supersedingFollowUp.text;
505
758
  }
506
- // Load or create session
507
- const sessPath = (0, config_2.sessionPath)(friendId, "teams", conversationId);
508
- const existing = (0, context_1.loadSession)(sessPath);
509
- const messages = existing?.messages && existing.messages.length > 0
510
- ? existing.messages
511
- : [{ role: "system", content: await (0, prompt_1.buildSystem)("teams", undefined, toolContext?.context) }];
512
- // Repair any orphaned tool calls from a previous aborted turn
513
- (0, core_1.repairOrphanedToolCalls)(messages);
514
- // Push user message
515
- messages.push({ role: "user", content: text });
516
- // Run agent
517
- const controller = new AbortController();
518
- const channelConfig = (0, config_2.getTeamsChannelConfig)();
519
- const callbacks = createTeamsCallbacks(stream, controller, sendMessage, { conversationId, flushIntervalMs: channelConfig.flushIntervalMs });
520
- const traceId = (0, nerves_1.createTraceId)();
521
- const agentOptions = {};
522
- agentOptions.traceId = traceId;
523
- if (toolContext)
524
- agentOptions.toolContext = toolContext;
525
- if (channelConfig.skipConfirmation)
526
- agentOptions.skipConfirmation = true;
527
- agentOptions.drainSteeringFollowUps = () => _turnCoordinator.drainFollowUps(turnKey).map((m) => ({ text: m.text }));
528
- const result = await (0, core_1.runAgent)(messages, callbacks, "teams", controller.signal, agentOptions);
529
- // Flush any remaining accumulated text at end of turn
530
- await callbacks.flush();
531
- // After the agent loop, check if any tool returned AUTH_REQUIRED and trigger signin.
532
- // This must happen after the stream is done so the OAuth card renders properly.
533
- if (teamsContext) {
534
- const allContent = messages.map(m => typeof m.content === "string" ? m.content : "").join("\n");
535
- if (allContent.includes("AUTH_REQUIRED:graph") && teamsContext.graphConnectionName)
536
- await teamsContext.signin(teamsContext.graphConnectionName);
537
- if (allContent.includes("AUTH_REQUIRED:ado") && teamsContext.adoConnectionName)
538
- await teamsContext.signin(teamsContext.adoConnectionName);
539
- if (allContent.includes("AUTH_REQUIRED:github") && teamsContext.githubConnectionName)
540
- await teamsContext.signin(teamsContext.githubConnectionName);
541
- }
542
- // Trim context and save session
543
- (0, context_1.postTurn)(messages, sessPath, result.usage);
544
- // Accumulate token usage on friend record
545
- if (toolContext?.context?.friend?.id) {
546
- await (0, tokens_1.accumulateFriendTokens)(store, toolContext.context.friend.id, result.usage);
547
- }
548
- // SDK auto-closes the stream after our handler returns (app.process.js)
549
759
  }
550
760
  // Internal port for the secondary bot App (not exposed externally).
551
761
  // The primary app proxies /api/messages-secondary → localhost:SECONDARY_PORT/api/messages.
@@ -621,7 +831,7 @@ function registerBotHandlers(app, label) {
621
831
  // (graph + ado + github). The verifyState activity only carries a `state`
622
832
  // code with no connectionName, so we try each configured connection until
623
833
  // one succeeds.
624
- app.on("signin.verify-state", async (ctx) => {
834
+ app.on("signin.verify-state", (async (ctx) => {
625
835
  const { api, activity } = ctx;
626
836
  if (!activity.value?.state)
627
837
  return { status: 404 };
@@ -640,7 +850,73 @@ function registerBotHandlers(app, label) {
640
850
  }
641
851
  (0, runtime_1.emitNervesEvent)({ level: "warn", event: "channel.verify_state", component: "channels", message: `[${label}] verify-state failed for all connections`, meta: {} });
642
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
+ }
643
918
  });
919
+ /* v8 ignore stop */
644
920
  app.on("message", async (ctx) => {
645
921
  const { stream, activity, api, signin } = ctx;
646
922
  const text = activity.text || "";
@@ -649,12 +925,40 @@ function registerBotHandlers(app, label) {
649
925
  const userId = activity.from?.id || "";
650
926
  const channelId = activity.channelId || "msteams";
651
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) } });
652
- // Resolve pending confirmations IMMEDIATELY — before token fetches or
653
- // the conversation lock. The original message holds the lock while
654
- // awaiting confirmation, so acquiring it here would deadlock. Token
655
- // fetches are also unnecessary (and slow) for a simple yes/no reply.
656
- if (resolvePendingConfirmation(convId, text)) {
657
- return;
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
+ }
658
962
  }
659
963
  // If this conversation already has an active turn, steer follow-up input
660
964
  // into that turn and avoid starting a second concurrent turn.
@@ -663,6 +967,7 @@ function registerBotHandlers(app, label) {
663
967
  conversationId: convId,
664
968
  text,
665
969
  receivedAt: Date.now(),
970
+ effect: (0, continuity_1.classifySteeringFollowUpEffect)(text),
666
971
  });
667
972
  return;
668
973
  }
@@ -720,7 +1025,7 @@ function registerBotHandlers(app, label) {
720
1025
  const ctxSend = async (t) => {
721
1026
  // Use send with replyToId (not reply, which adds a blockquote).
722
1027
  // replyToId anchors the message after the user's message in Copilot Chat.
723
- 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 } });
724
1029
  };
725
1030
  await handleTeamsMessage(text, stream, convId, teamsContext, ctxSend);
726
1031
  }
@@ -737,6 +1042,234 @@ function registerBotHandlers(app, label) {
737
1042
  (0, runtime_1.emitNervesEvent)({ level: "error", event: "channel.app_error", component: "channels", message: `[${label}] ${msg}`, meta: {} });
738
1043
  });
739
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
+ }
740
1273
  // Start the Teams app in DevtoolsPlugin mode (local dev) or Bot Service mode (real Teams).
741
1274
  // Mode is determined by getTeamsConfig().clientId.
742
1275
  // Text is always accumulated in textBuffer and flushed periodically (chunked streaming).