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