@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +226 -183
- package/SerpentGuide.ouro/agent.json +82 -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 +3418 -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 +237 -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 +58 -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 +426 -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 +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +193 -130
- package/dist/heart/core.js +1010 -261
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +490 -0
- package/dist/heart/daemon/agent-discovery.js +157 -0
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +216 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +640 -0
- package/dist/heart/daemon/cli-exec.js +7239 -0
- package/dist/heart/daemon/cli-help.js +493 -0
- package/dist/heart/daemon/cli-parse.js +1533 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +561 -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 -697
- package/dist/heart/daemon/daemon-entry.js +359 -8
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +813 -19
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +615 -0
- package/dist/heart/daemon/health-monitor.js +92 -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 +171 -0
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/os-cron-deps.js +134 -0
- 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 +215 -1
- 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 +431 -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 +307 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +264 -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 +54 -136
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- 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 +274 -61
- 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/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/outlook/outlook-http-hooks.js +66 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +244 -0
- package/dist/heart/outlook/outlook-http-static.js +103 -0
- package/dist/heart/outlook/outlook-http-transport.js +116 -0
- package/dist/heart/outlook/outlook-http.js +99 -0
- package/dist/heart/outlook/outlook-read.js +31 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +195 -0
- package/dist/heart/outlook/readers/agent-machine.js +382 -0
- package/dist/heart/outlook/readers/continuity-readers.js +336 -0
- package/dist/heart/outlook/readers/mail.js +362 -0
- package/dist/heart/outlook/readers/runtime-readers.js +644 -0
- package/dist/heart/outlook/readers/sessions.js +232 -0
- package/dist/heart/outlook/readers/shared.js +111 -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 +255 -0
- package/dist/heart/provider-credentials.js +424 -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-state.js +216 -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 +63 -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 +260 -0
- package/dist/heart/sense-truth.js +68 -0
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +1089 -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-transcript.js +167 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +129 -34
- package/dist/heart/sync.js +332 -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/tool-activity-callbacks.js +36 -0
- package/dist/heart/tool-description.js +135 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +372 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +425 -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 +115 -0
- package/dist/heart/versioning/update-hooks.js +142 -0
- package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +606 -0
- package/dist/mailroom/core.js +672 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +426 -0
- package/dist/mailroom/mbox-import.js +382 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +219 -0
- package/dist/mailroom/search-cache.js +182 -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 +173 -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 +73 -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 +1144 -117
- 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 +101 -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/runtime.js +5 -1
- package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/outlook-ui/index.html +15 -0
- 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 +774 -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 +111 -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 +255 -0
- package/dist/repertoire/mcp-manager.js +305 -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-base.js +49 -707
- package/dist/repertoire/tools-bluebubbles.js +94 -0
- package/dist/repertoire/tools-bridge.js +141 -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 +105 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1377 -0
- package/dist/repertoire/tools-notes.js +376 -0
- package/dist/repertoire/tools-session.js +749 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +64 -61
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +356 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +149 -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 +561 -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/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 +73 -0
- package/dist/senses/bluebubbles/inbound-log.js +126 -0
- package/dist/senses/bluebubbles/index.js +1881 -0
- package/dist/senses/bluebubbles/media.js +389 -0
- package/dist/senses/bluebubbles/model.js +282 -0
- package/dist/senses/bluebubbles/mutation-log.js +116 -0
- package/dist/senses/bluebubbles/processed-log.js +111 -0
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/bluebubbles/session-cleanup.js +72 -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 +605 -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 +83 -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 +768 -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 +199 -16
- package/dist/senses/inner-dialog.js +640 -91
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +665 -0
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +844 -197
- package/dist/senses/trust-gate.js +207 -2
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +47 -6
- 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 +101 -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/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
|
@@ -0,0 +1,685 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.createBlueBubblesClient = createBlueBubblesClient;
|
|
37
|
+
const node_crypto_1 = require("node:crypto");
|
|
38
|
+
const config_1 = require("../../heart/config");
|
|
39
|
+
const identity_1 = require("../../heart/identity");
|
|
40
|
+
const bluebubbles_health_diagnostics_1 = require("../../heart/daemon/bluebubbles-health-diagnostics");
|
|
41
|
+
const runtime_1 = require("../../nerves/runtime");
|
|
42
|
+
const minimax_1 = require("../../heart/providers/minimax");
|
|
43
|
+
const minimax_vlm_1 = require("../../heart/providers/minimax-vlm");
|
|
44
|
+
const model_1 = require("./model");
|
|
45
|
+
const media_1 = require("./media");
|
|
46
|
+
function buildBlueBubblesApiUrl(baseUrl, endpoint, password) {
|
|
47
|
+
const root = baseUrl.endsWith("/") ? baseUrl : `${baseUrl}/`;
|
|
48
|
+
const url = new URL(endpoint.replace(/^\//, ""), root);
|
|
49
|
+
url.searchParams.set("password", password);
|
|
50
|
+
return url.toString();
|
|
51
|
+
}
|
|
52
|
+
function asRecord(value) {
|
|
53
|
+
return value && typeof value === "object" && !Array.isArray(value)
|
|
54
|
+
? value
|
|
55
|
+
: null;
|
|
56
|
+
}
|
|
57
|
+
function readString(record, key) {
|
|
58
|
+
const value = record[key];
|
|
59
|
+
return typeof value === "string" ? value : undefined;
|
|
60
|
+
}
|
|
61
|
+
function extractMessageGuid(payload) {
|
|
62
|
+
if (!payload || typeof payload !== "object")
|
|
63
|
+
return undefined;
|
|
64
|
+
const record = payload;
|
|
65
|
+
const data = record.data && typeof record.data === "object" && !Array.isArray(record.data)
|
|
66
|
+
? record.data
|
|
67
|
+
: null;
|
|
68
|
+
const candidates = [
|
|
69
|
+
record.messageGuid,
|
|
70
|
+
record.messageId,
|
|
71
|
+
record.guid,
|
|
72
|
+
data?.messageGuid,
|
|
73
|
+
data?.messageId,
|
|
74
|
+
data?.guid,
|
|
75
|
+
];
|
|
76
|
+
for (const candidate of candidates) {
|
|
77
|
+
if (typeof candidate === "string" && candidate.trim()) {
|
|
78
|
+
return candidate.trim();
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return undefined;
|
|
82
|
+
}
|
|
83
|
+
async function parseJsonBody(response) {
|
|
84
|
+
const raw = await response.text();
|
|
85
|
+
if (!raw.trim())
|
|
86
|
+
return null;
|
|
87
|
+
try {
|
|
88
|
+
return JSON.parse(raw);
|
|
89
|
+
}
|
|
90
|
+
catch {
|
|
91
|
+
return null;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
function describeCaughtValue(error) {
|
|
95
|
+
return error instanceof Error ? error.message : String(error);
|
|
96
|
+
}
|
|
97
|
+
function buildRepairUrl(baseUrl, messageGuid, password) {
|
|
98
|
+
const url = buildBlueBubblesApiUrl(baseUrl, `/api/v1/message/${encodeURIComponent(messageGuid)}`, password);
|
|
99
|
+
const parsed = new URL(url);
|
|
100
|
+
parsed.searchParams.set("with", "attachments,payloadData,chats,messageSummaryInfo");
|
|
101
|
+
return parsed.toString();
|
|
102
|
+
}
|
|
103
|
+
function extractChatIdentifierFromGuid(chatGuid) {
|
|
104
|
+
const parts = chatGuid.split(";");
|
|
105
|
+
return parts.length >= 3 ? parts[2]?.trim() || undefined : undefined;
|
|
106
|
+
}
|
|
107
|
+
function extractChatGuid(value) {
|
|
108
|
+
const record = asRecord(value);
|
|
109
|
+
const candidates = [
|
|
110
|
+
record?.chatGuid,
|
|
111
|
+
record?.guid,
|
|
112
|
+
record?.chat_guid,
|
|
113
|
+
record?.identifier,
|
|
114
|
+
record?.chatIdentifier,
|
|
115
|
+
record?.chat_identifier,
|
|
116
|
+
];
|
|
117
|
+
for (const candidate of candidates) {
|
|
118
|
+
if (typeof candidate === "string" && candidate.trim()) {
|
|
119
|
+
return candidate.trim();
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
return undefined;
|
|
123
|
+
}
|
|
124
|
+
function extractQueriedChatIdentifier(chat, chatGuid) {
|
|
125
|
+
const explicitIdentifier = readString(chat, "chatIdentifier")
|
|
126
|
+
?? readString(chat, "identifier")
|
|
127
|
+
?? readString(chat, "chat_identifier");
|
|
128
|
+
if (explicitIdentifier) {
|
|
129
|
+
return explicitIdentifier;
|
|
130
|
+
}
|
|
131
|
+
return extractChatIdentifierFromGuid(chatGuid);
|
|
132
|
+
}
|
|
133
|
+
function extractChatQueryRows(payload) {
|
|
134
|
+
const record = asRecord(payload);
|
|
135
|
+
const data = Array.isArray(record?.data) ? record.data : payload;
|
|
136
|
+
if (!Array.isArray(data)) {
|
|
137
|
+
return [];
|
|
138
|
+
}
|
|
139
|
+
return data.map((entry) => asRecord(entry)).filter((entry) => entry !== null);
|
|
140
|
+
}
|
|
141
|
+
function extractMessageQueryRows(payload) {
|
|
142
|
+
const record = asRecord(payload);
|
|
143
|
+
const data = asRecord(record?.data);
|
|
144
|
+
const rows = Array.isArray(record?.data) ? record.data
|
|
145
|
+
: Array.isArray(data?.messages) ? data.messages
|
|
146
|
+
: Array.isArray(data?.results) ? data.results
|
|
147
|
+
: Array.isArray(record?.messages) ? record.messages
|
|
148
|
+
: Array.isArray(payload) ? payload
|
|
149
|
+
: [];
|
|
150
|
+
return rows.map((entry) => asRecord(entry)).filter((entry) => entry !== null);
|
|
151
|
+
}
|
|
152
|
+
async function resolveChatGuidForIdentifier(config, channelConfig, chatIdentifier) {
|
|
153
|
+
const trimmedIdentifier = chatIdentifier.trim();
|
|
154
|
+
if (!trimmedIdentifier)
|
|
155
|
+
return undefined;
|
|
156
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, "/api/v1/chat/query", config.password);
|
|
157
|
+
const response = await fetch(url, {
|
|
158
|
+
method: "POST",
|
|
159
|
+
headers: { "Content-Type": "application/json" },
|
|
160
|
+
body: JSON.stringify({
|
|
161
|
+
limit: 500,
|
|
162
|
+
offset: 0,
|
|
163
|
+
with: ["participants"],
|
|
164
|
+
}),
|
|
165
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
166
|
+
});
|
|
167
|
+
if (!response.ok) {
|
|
168
|
+
return undefined;
|
|
169
|
+
}
|
|
170
|
+
const payload = await parseJsonBody(response);
|
|
171
|
+
const rows = extractChatQueryRows(payload);
|
|
172
|
+
for (const row of rows) {
|
|
173
|
+
const guid = extractChatGuid(row);
|
|
174
|
+
if (!guid)
|
|
175
|
+
continue;
|
|
176
|
+
const identifier = extractQueriedChatIdentifier(row, guid);
|
|
177
|
+
if (identifier === trimmedIdentifier || guid === trimmedIdentifier) {
|
|
178
|
+
return guid;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
return undefined;
|
|
182
|
+
}
|
|
183
|
+
function collectPreviewStrings(value, out, depth = 0) {
|
|
184
|
+
if (depth > 4 || out.length >= 4)
|
|
185
|
+
return;
|
|
186
|
+
if (typeof value === "string") {
|
|
187
|
+
const trimmed = value.trim();
|
|
188
|
+
if (trimmed)
|
|
189
|
+
out.push(trimmed);
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
if (Array.isArray(value)) {
|
|
193
|
+
for (const entry of value)
|
|
194
|
+
collectPreviewStrings(entry, out, depth + 1);
|
|
195
|
+
return;
|
|
196
|
+
}
|
|
197
|
+
const record = asRecord(value);
|
|
198
|
+
if (!record)
|
|
199
|
+
return;
|
|
200
|
+
const preferredKeys = ["title", "summary", "subtitle", "previewText", "siteName", "host", "url"];
|
|
201
|
+
for (const key of preferredKeys) {
|
|
202
|
+
if (out.length >= 4)
|
|
203
|
+
break;
|
|
204
|
+
collectPreviewStrings(record[key], out, depth + 1);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
function extractLinkPreviewText(data) {
|
|
208
|
+
const values = [];
|
|
209
|
+
collectPreviewStrings(data.payloadData, values);
|
|
210
|
+
collectPreviewStrings(data.messageSummaryInfo, values);
|
|
211
|
+
const unique = [...new Set(values.map((value) => value.trim()).filter(Boolean))];
|
|
212
|
+
if (unique.length === 0)
|
|
213
|
+
return undefined;
|
|
214
|
+
return unique.slice(0, 2).join(" — ");
|
|
215
|
+
}
|
|
216
|
+
function applyRepairNotice(event, notice) {
|
|
217
|
+
return {
|
|
218
|
+
...event,
|
|
219
|
+
requiresRepair: false,
|
|
220
|
+
repairNotice: notice,
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
function hasRecoverableMessageContent(event) {
|
|
224
|
+
return event.kind === "message"
|
|
225
|
+
&& (event.textForAgent.trim().length > 0
|
|
226
|
+
|| event.attachments.length > 0
|
|
227
|
+
|| event.hasPayloadData);
|
|
228
|
+
}
|
|
229
|
+
function hydrateTextForAgent(event, rawData) {
|
|
230
|
+
if (event.kind !== "message") {
|
|
231
|
+
return { ...event, requiresRepair: false };
|
|
232
|
+
}
|
|
233
|
+
if (event.balloonBundleId !== "com.apple.messages.URLBalloonProvider") {
|
|
234
|
+
return { ...event, requiresRepair: false };
|
|
235
|
+
}
|
|
236
|
+
const previewText = extractLinkPreviewText(rawData);
|
|
237
|
+
if (!previewText) {
|
|
238
|
+
return { ...event, requiresRepair: false };
|
|
239
|
+
}
|
|
240
|
+
const base = event.text.trim();
|
|
241
|
+
const textForAgent = base
|
|
242
|
+
? `${base}\n[link preview: ${previewText}]`
|
|
243
|
+
: `[link preview: ${previewText}]`;
|
|
244
|
+
return {
|
|
245
|
+
...event,
|
|
246
|
+
textForAgent,
|
|
247
|
+
requiresRepair: false,
|
|
248
|
+
};
|
|
249
|
+
}
|
|
250
|
+
function extractRepairData(payload) {
|
|
251
|
+
const record = asRecord(payload);
|
|
252
|
+
return asRecord(record?.data) ?? record;
|
|
253
|
+
}
|
|
254
|
+
function providerSupportsAudioInput(provider) {
|
|
255
|
+
void provider;
|
|
256
|
+
return false;
|
|
257
|
+
}
|
|
258
|
+
async function resolveChatGuid(chat, config, channelConfig) {
|
|
259
|
+
return chat.chatGuid
|
|
260
|
+
?? await resolveChatGuidForIdentifier(config, channelConfig, chat.chatIdentifier ?? "");
|
|
261
|
+
}
|
|
262
|
+
function createBlueBubblesClient(config = (0, config_1.getBlueBubblesConfig)(), channelConfig = (0, config_1.getBlueBubblesChannelConfig)()) {
|
|
263
|
+
return {
|
|
264
|
+
async sendText(params) {
|
|
265
|
+
const trimmedText = params.text.trim();
|
|
266
|
+
if (!trimmedText) {
|
|
267
|
+
throw new Error("BlueBubbles send requires non-empty text.");
|
|
268
|
+
}
|
|
269
|
+
const resolvedChatGuid = await resolveChatGuid(params.chat, config, channelConfig);
|
|
270
|
+
if (!resolvedChatGuid) {
|
|
271
|
+
throw new Error("BlueBubbles send currently requires chat.chatGuid from the inbound event.");
|
|
272
|
+
}
|
|
273
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, "/api/v1/message/text", config.password);
|
|
274
|
+
const body = {
|
|
275
|
+
chatGuid: resolvedChatGuid,
|
|
276
|
+
tempGuid: (0, node_crypto_1.randomUUID)(),
|
|
277
|
+
message: trimmedText,
|
|
278
|
+
};
|
|
279
|
+
if (params.replyToMessageGuid?.trim()) {
|
|
280
|
+
body.method = "private-api";
|
|
281
|
+
body.selectedMessageGuid = params.replyToMessageGuid.trim();
|
|
282
|
+
body.partIndex = 0;
|
|
283
|
+
}
|
|
284
|
+
(0, runtime_1.emitNervesEvent)({
|
|
285
|
+
component: "senses",
|
|
286
|
+
event: "senses.bluebubbles_send_start",
|
|
287
|
+
message: "sending bluebubbles message",
|
|
288
|
+
meta: {
|
|
289
|
+
chatGuid: resolvedChatGuid,
|
|
290
|
+
hasReplyTarget: Boolean(params.replyToMessageGuid?.trim()),
|
|
291
|
+
},
|
|
292
|
+
});
|
|
293
|
+
const response = await fetch(url, {
|
|
294
|
+
method: "POST",
|
|
295
|
+
headers: { "Content-Type": "application/json" },
|
|
296
|
+
body: JSON.stringify(body),
|
|
297
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
298
|
+
});
|
|
299
|
+
if (!response.ok) {
|
|
300
|
+
const errorText = await response.text().catch(() => "");
|
|
301
|
+
(0, runtime_1.emitNervesEvent)({
|
|
302
|
+
level: "error",
|
|
303
|
+
component: "senses",
|
|
304
|
+
event: "senses.bluebubbles_send_error",
|
|
305
|
+
message: "bluebubbles send failed",
|
|
306
|
+
meta: {
|
|
307
|
+
status: response.status,
|
|
308
|
+
reason: errorText || "unknown",
|
|
309
|
+
},
|
|
310
|
+
});
|
|
311
|
+
throw new Error(`BlueBubbles send failed (${response.status}): ${errorText || "unknown"}`);
|
|
312
|
+
}
|
|
313
|
+
const payload = await parseJsonBody(response);
|
|
314
|
+
const messageGuid = extractMessageGuid(payload);
|
|
315
|
+
(0, runtime_1.emitNervesEvent)({
|
|
316
|
+
component: "senses",
|
|
317
|
+
event: "senses.bluebubbles_send_end",
|
|
318
|
+
message: "bluebubbles message sent",
|
|
319
|
+
meta: {
|
|
320
|
+
chatGuid: resolvedChatGuid,
|
|
321
|
+
messageGuid: messageGuid ?? null,
|
|
322
|
+
},
|
|
323
|
+
});
|
|
324
|
+
return { messageGuid };
|
|
325
|
+
},
|
|
326
|
+
async editMessage(params) {
|
|
327
|
+
const messageGuid = params.messageGuid.trim();
|
|
328
|
+
const text = params.text.trim();
|
|
329
|
+
if (!messageGuid) {
|
|
330
|
+
throw new Error("BlueBubbles edit requires messageGuid.");
|
|
331
|
+
}
|
|
332
|
+
if (!text) {
|
|
333
|
+
throw new Error("BlueBubbles edit requires non-empty text.");
|
|
334
|
+
}
|
|
335
|
+
const editTimeoutMs = Math.max(channelConfig.requestTimeoutMs, 120000);
|
|
336
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, `/api/v1/message/${encodeURIComponent(messageGuid)}/edit`, config.password);
|
|
337
|
+
const response = await fetch(url, {
|
|
338
|
+
method: "POST",
|
|
339
|
+
headers: { "Content-Type": "application/json" },
|
|
340
|
+
body: JSON.stringify({
|
|
341
|
+
editedMessage: text,
|
|
342
|
+
backwardsCompatibilityMessage: params.backwardsCompatibilityMessage ?? `Edited to: ${text}`,
|
|
343
|
+
partIndex: typeof params.partIndex === "number" ? params.partIndex : 0,
|
|
344
|
+
}),
|
|
345
|
+
signal: AbortSignal.timeout(editTimeoutMs),
|
|
346
|
+
});
|
|
347
|
+
if (!response.ok) {
|
|
348
|
+
const errorText = await response.text().catch(() => "");
|
|
349
|
+
throw new Error(`BlueBubbles edit failed (${response.status}): ${errorText || "unknown"}`);
|
|
350
|
+
}
|
|
351
|
+
},
|
|
352
|
+
async setTyping(chat, typing) {
|
|
353
|
+
const resolvedChatGuid = await resolveChatGuid(chat, config, channelConfig);
|
|
354
|
+
if (!resolvedChatGuid) {
|
|
355
|
+
return;
|
|
356
|
+
}
|
|
357
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, `/api/v1/chat/${encodeURIComponent(resolvedChatGuid)}/typing`, config.password);
|
|
358
|
+
const response = await fetch(url, {
|
|
359
|
+
method: typing ? "POST" : "DELETE",
|
|
360
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
361
|
+
});
|
|
362
|
+
if (!response.ok) {
|
|
363
|
+
const errorText = await response.text().catch(() => "");
|
|
364
|
+
throw new Error(`BlueBubbles typing failed (${response.status}): ${errorText || "unknown"}`);
|
|
365
|
+
}
|
|
366
|
+
},
|
|
367
|
+
async markChatRead(chat) {
|
|
368
|
+
const resolvedChatGuid = await resolveChatGuid(chat, config, channelConfig);
|
|
369
|
+
if (!resolvedChatGuid) {
|
|
370
|
+
return;
|
|
371
|
+
}
|
|
372
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, `/api/v1/chat/${encodeURIComponent(resolvedChatGuid)}/read`, config.password);
|
|
373
|
+
const response = await fetch(url, {
|
|
374
|
+
method: "POST",
|
|
375
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
376
|
+
});
|
|
377
|
+
if (!response.ok) {
|
|
378
|
+
const errorText = await response.text().catch(() => "");
|
|
379
|
+
throw new Error(`BlueBubbles read failed (${response.status}): ${errorText || "unknown"}`);
|
|
380
|
+
}
|
|
381
|
+
},
|
|
382
|
+
async checkHealth() {
|
|
383
|
+
(0, runtime_1.emitNervesEvent)({
|
|
384
|
+
component: "senses",
|
|
385
|
+
event: "senses.bluebubbles_healthcheck_start",
|
|
386
|
+
message: "probing bluebubbles upstream health",
|
|
387
|
+
meta: { serverUrl: config.serverUrl },
|
|
388
|
+
});
|
|
389
|
+
const result = await (0, bluebubbles_health_diagnostics_1.probeBlueBubblesHealth)({
|
|
390
|
+
serverUrl: config.serverUrl,
|
|
391
|
+
password: config.password,
|
|
392
|
+
requestTimeoutMs: channelConfig.requestTimeoutMs,
|
|
393
|
+
fetchImpl: fetch,
|
|
394
|
+
});
|
|
395
|
+
if (!result.ok) {
|
|
396
|
+
(0, runtime_1.emitNervesEvent)({
|
|
397
|
+
level: "warn",
|
|
398
|
+
component: "senses",
|
|
399
|
+
event: "senses.bluebubbles_healthcheck_error",
|
|
400
|
+
message: "bluebubbles upstream health probe failed",
|
|
401
|
+
meta: {
|
|
402
|
+
serverUrl: config.serverUrl,
|
|
403
|
+
status: result.status,
|
|
404
|
+
reason: result.reason,
|
|
405
|
+
classification: result.classification,
|
|
406
|
+
detail: (0, bluebubbles_health_diagnostics_1.redactBlueBubblesHealthDetailForNerves)(result.detail),
|
|
407
|
+
},
|
|
408
|
+
});
|
|
409
|
+
throw new Error(result.detail);
|
|
410
|
+
}
|
|
411
|
+
(0, runtime_1.emitNervesEvent)({
|
|
412
|
+
component: "senses",
|
|
413
|
+
event: "senses.bluebubbles_healthcheck_end",
|
|
414
|
+
message: "bluebubbles upstream health probe succeeded",
|
|
415
|
+
meta: { serverUrl: config.serverUrl },
|
|
416
|
+
});
|
|
417
|
+
},
|
|
418
|
+
async listRecentMessages(params = {}) {
|
|
419
|
+
const limit = Math.max(1, Math.min(100, Math.floor(params.limit ?? 50)));
|
|
420
|
+
const offset = Math.max(0, Math.floor(params.offset ?? 0));
|
|
421
|
+
const url = buildBlueBubblesApiUrl(config.serverUrl, "/api/v1/message/query", config.password);
|
|
422
|
+
(0, runtime_1.emitNervesEvent)({
|
|
423
|
+
component: "senses",
|
|
424
|
+
event: "senses.bluebubbles_query_recent_start",
|
|
425
|
+
message: "querying recent bluebubbles messages",
|
|
426
|
+
meta: { limit, offset },
|
|
427
|
+
});
|
|
428
|
+
const response = await fetch(url, {
|
|
429
|
+
method: "POST",
|
|
430
|
+
headers: { "Content-Type": "application/json" },
|
|
431
|
+
body: JSON.stringify({
|
|
432
|
+
limit,
|
|
433
|
+
offset,
|
|
434
|
+
sort: "DESC",
|
|
435
|
+
with: ["chats", "attachments", "payloadData", "messageSummaryInfo"],
|
|
436
|
+
}),
|
|
437
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
438
|
+
});
|
|
439
|
+
if (!response.ok) {
|
|
440
|
+
const errorText = await response.text().catch(() => "");
|
|
441
|
+
(0, runtime_1.emitNervesEvent)({
|
|
442
|
+
level: "warn",
|
|
443
|
+
component: "senses",
|
|
444
|
+
event: "senses.bluebubbles_query_recent_error",
|
|
445
|
+
message: "bluebubbles recent message query failed",
|
|
446
|
+
meta: {
|
|
447
|
+
status: response.status,
|
|
448
|
+
reason: errorText || "unknown",
|
|
449
|
+
},
|
|
450
|
+
});
|
|
451
|
+
throw new Error(`BlueBubbles recent message query failed (${response.status}): ${errorText || "unknown"}`);
|
|
452
|
+
}
|
|
453
|
+
const payload = await parseJsonBody(response);
|
|
454
|
+
const rows = extractMessageQueryRows(payload);
|
|
455
|
+
const messages = [];
|
|
456
|
+
for (const row of rows) {
|
|
457
|
+
try {
|
|
458
|
+
messages.push((0, model_1.normalizeBlueBubblesEvent)({ type: "new-message", data: row }));
|
|
459
|
+
}
|
|
460
|
+
catch (error) {
|
|
461
|
+
(0, runtime_1.emitNervesEvent)({
|
|
462
|
+
level: "warn",
|
|
463
|
+
component: "senses",
|
|
464
|
+
event: "senses.bluebubbles_query_recent_skip",
|
|
465
|
+
message: "skipped unusable bluebubbles recent message row",
|
|
466
|
+
meta: {
|
|
467
|
+
reason: describeCaughtValue(error),
|
|
468
|
+
},
|
|
469
|
+
});
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
(0, runtime_1.emitNervesEvent)({
|
|
473
|
+
component: "senses",
|
|
474
|
+
event: "senses.bluebubbles_query_recent_end",
|
|
475
|
+
message: "queried recent bluebubbles messages",
|
|
476
|
+
meta: {
|
|
477
|
+
rows: rows.length,
|
|
478
|
+
normalized: messages.length,
|
|
479
|
+
},
|
|
480
|
+
});
|
|
481
|
+
return messages;
|
|
482
|
+
},
|
|
483
|
+
async repairEvent(event) {
|
|
484
|
+
if (!event.requiresRepair) {
|
|
485
|
+
(0, runtime_1.emitNervesEvent)({
|
|
486
|
+
component: "senses",
|
|
487
|
+
event: "senses.bluebubbles_repair_skipped",
|
|
488
|
+
message: "bluebubbles event repair skipped",
|
|
489
|
+
meta: {
|
|
490
|
+
kind: event.kind,
|
|
491
|
+
messageGuid: event.messageGuid,
|
|
492
|
+
},
|
|
493
|
+
});
|
|
494
|
+
return event;
|
|
495
|
+
}
|
|
496
|
+
(0, runtime_1.emitNervesEvent)({
|
|
497
|
+
level: "warn",
|
|
498
|
+
component: "senses",
|
|
499
|
+
event: "senses.bluebubbles_repair_start",
|
|
500
|
+
message: "repairing bluebubbles event by guid",
|
|
501
|
+
meta: {
|
|
502
|
+
kind: event.kind,
|
|
503
|
+
messageGuid: event.messageGuid,
|
|
504
|
+
eventType: event.eventType,
|
|
505
|
+
},
|
|
506
|
+
});
|
|
507
|
+
const url = buildRepairUrl(config.serverUrl, event.messageGuid, config.password);
|
|
508
|
+
try {
|
|
509
|
+
const response = await fetch(url, {
|
|
510
|
+
method: "GET",
|
|
511
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
512
|
+
});
|
|
513
|
+
if (!response.ok) {
|
|
514
|
+
const errorText = await response.text().catch(() => "");
|
|
515
|
+
const repaired = applyRepairNotice(event, `BlueBubbles repair failed: ${errorText || `HTTP ${response.status}`}`);
|
|
516
|
+
(0, runtime_1.emitNervesEvent)({
|
|
517
|
+
level: "warn",
|
|
518
|
+
component: "senses",
|
|
519
|
+
event: "senses.bluebubbles_repair_error",
|
|
520
|
+
message: "bluebubbles repair request failed",
|
|
521
|
+
meta: {
|
|
522
|
+
messageGuid: event.messageGuid,
|
|
523
|
+
status: response.status,
|
|
524
|
+
reason: errorText || "unknown",
|
|
525
|
+
},
|
|
526
|
+
});
|
|
527
|
+
return repaired;
|
|
528
|
+
}
|
|
529
|
+
const payload = await parseJsonBody(response);
|
|
530
|
+
const data = extractRepairData(payload);
|
|
531
|
+
if (!data || typeof data.guid !== "string") {
|
|
532
|
+
const repaired = applyRepairNotice(event, "BlueBubbles repair failed: invalid message payload");
|
|
533
|
+
(0, runtime_1.emitNervesEvent)({
|
|
534
|
+
level: "warn",
|
|
535
|
+
component: "senses",
|
|
536
|
+
event: "senses.bluebubbles_repair_error",
|
|
537
|
+
message: "bluebubbles repair returned unusable payload",
|
|
538
|
+
meta: {
|
|
539
|
+
messageGuid: event.messageGuid,
|
|
540
|
+
},
|
|
541
|
+
});
|
|
542
|
+
return repaired;
|
|
543
|
+
}
|
|
544
|
+
const normalized = (0, model_1.normalizeBlueBubblesEvent)({
|
|
545
|
+
type: event.eventType,
|
|
546
|
+
data,
|
|
547
|
+
});
|
|
548
|
+
const recoveredMessage = event.kind === "mutation"
|
|
549
|
+
&& !event.shouldNotifyAgent
|
|
550
|
+
? (0, model_1.normalizeBlueBubblesEvent)({
|
|
551
|
+
type: "new-message",
|
|
552
|
+
data,
|
|
553
|
+
})
|
|
554
|
+
: null;
|
|
555
|
+
let hydrated = recoveredMessage && hasRecoverableMessageContent(recoveredMessage)
|
|
556
|
+
? hydrateTextForAgent(recoveredMessage, data)
|
|
557
|
+
: hydrateTextForAgent(normalized, data);
|
|
558
|
+
if (hydrated.kind === "message" &&
|
|
559
|
+
hydrated.balloonBundleId !== "com.apple.messages.URLBalloonProvider" &&
|
|
560
|
+
hydrated.attachments.length > 0) {
|
|
561
|
+
const agentConfig = (0, identity_1.loadAgentConfig)();
|
|
562
|
+
const chatModel = agentConfig.humanFacing.model;
|
|
563
|
+
const chatProvider = agentConfig.humanFacing.provider;
|
|
564
|
+
const vlmDescribe = async (params) => {
|
|
565
|
+
if (chatProvider !== "minimax") {
|
|
566
|
+
throw new Error("VLM fallback requires a minimax credential for this agent — " +
|
|
567
|
+
"configure one or switch to a vision-capable chat model");
|
|
568
|
+
}
|
|
569
|
+
const { readProviderCredentialRecord } = await Promise.resolve().then(() => __importStar(require("../../heart/provider-credentials")));
|
|
570
|
+
const credential = await readProviderCredentialRecord((0, identity_1.getAgentName)(), "minimax");
|
|
571
|
+
const apiKey = credential.ok ? credential.record.credentials.apiKey : undefined;
|
|
572
|
+
if (!apiKey) {
|
|
573
|
+
throw new Error("VLM fallback: minimax API key not found in the agent vault — " +
|
|
574
|
+
"run `ouro auth --agent <agent> --provider minimax`");
|
|
575
|
+
}
|
|
576
|
+
return (0, minimax_vlm_1.minimaxVlmDescribe)({
|
|
577
|
+
apiKey: String(apiKey),
|
|
578
|
+
prompt: params.prompt,
|
|
579
|
+
imageDataUrl: params.imageDataUrl,
|
|
580
|
+
baseURL: minimax_1.MINIMAX_PROVIDER_BASE_URL,
|
|
581
|
+
attachmentGuid: params.attachmentGuid,
|
|
582
|
+
mimeType: params.mimeType,
|
|
583
|
+
chatModel: params.chatModel,
|
|
584
|
+
});
|
|
585
|
+
};
|
|
586
|
+
// VLM prompt context wants the user's raw inbound text — NOT
|
|
587
|
+
// the agent-facing rendering, which now carries the attachment
|
|
588
|
+
// marker after the B2 fix. Pull it from the source payload.
|
|
589
|
+
const rawUserText = typeof data.text === "string"
|
|
590
|
+
? (data.text).trim()
|
|
591
|
+
: "";
|
|
592
|
+
const media = await (0, media_1.hydrateBlueBubblesAttachments)(hydrated.attachments, config, channelConfig, {
|
|
593
|
+
preferAudioInput: providerSupportsAudioInput(chatProvider),
|
|
594
|
+
chatModel,
|
|
595
|
+
vlmDescribe,
|
|
596
|
+
userText: rawUserText,
|
|
597
|
+
});
|
|
598
|
+
const transcriptSuffix = media.transcriptAdditions.map((entry) => `[${entry}]`).join("\n");
|
|
599
|
+
const noticeSuffix = media.notices.map((entry) => `[${entry}]`).join("\n");
|
|
600
|
+
const combinedSuffix = [transcriptSuffix, noticeSuffix].filter(Boolean).join("\n");
|
|
601
|
+
hydrated = {
|
|
602
|
+
...hydrated,
|
|
603
|
+
inputPartsForAgent: media.inputParts.length > 0 ? media.inputParts : undefined,
|
|
604
|
+
textForAgent: [hydrated.textForAgent, combinedSuffix].filter(Boolean).join("\n"),
|
|
605
|
+
};
|
|
606
|
+
}
|
|
607
|
+
(0, runtime_1.emitNervesEvent)({
|
|
608
|
+
level: "warn",
|
|
609
|
+
component: "senses",
|
|
610
|
+
event: "senses.bluebubbles_repair_end",
|
|
611
|
+
message: "bluebubbles event repaired",
|
|
612
|
+
meta: {
|
|
613
|
+
kind: hydrated.kind,
|
|
614
|
+
messageGuid: hydrated.messageGuid,
|
|
615
|
+
repairedFrom: event.kind,
|
|
616
|
+
promotedFromMutation: event.kind === "mutation" && hydrated.kind === "message",
|
|
617
|
+
},
|
|
618
|
+
});
|
|
619
|
+
return hydrated;
|
|
620
|
+
}
|
|
621
|
+
catch (error) {
|
|
622
|
+
const reason = error instanceof Error ? error.message : String(error);
|
|
623
|
+
(0, runtime_1.emitNervesEvent)({
|
|
624
|
+
level: "warn",
|
|
625
|
+
component: "senses",
|
|
626
|
+
event: "senses.bluebubbles_repair_error",
|
|
627
|
+
message: "bluebubbles repair threw",
|
|
628
|
+
meta: {
|
|
629
|
+
messageGuid: event.messageGuid,
|
|
630
|
+
reason,
|
|
631
|
+
},
|
|
632
|
+
});
|
|
633
|
+
return applyRepairNotice(event, `BlueBubbles repair failed: ${reason}`);
|
|
634
|
+
}
|
|
635
|
+
},
|
|
636
|
+
async getMessageText(messageGuid) {
|
|
637
|
+
const url = buildRepairUrl(config.serverUrl, messageGuid, config.password);
|
|
638
|
+
try {
|
|
639
|
+
const response = await fetch(url, {
|
|
640
|
+
method: "GET",
|
|
641
|
+
signal: AbortSignal.timeout(channelConfig.requestTimeoutMs),
|
|
642
|
+
});
|
|
643
|
+
if (!response.ok) {
|
|
644
|
+
(0, runtime_1.emitNervesEvent)({
|
|
645
|
+
level: "warn",
|
|
646
|
+
component: "senses",
|
|
647
|
+
event: "senses.bluebubbles_get_message_text_error",
|
|
648
|
+
message: "failed to fetch message text",
|
|
649
|
+
meta: { messageGuid, status: response.status },
|
|
650
|
+
});
|
|
651
|
+
return null;
|
|
652
|
+
}
|
|
653
|
+
const payload = await parseJsonBody(response);
|
|
654
|
+
const data = extractRepairData(payload);
|
|
655
|
+
if (!data || typeof data.text !== "string") {
|
|
656
|
+
(0, runtime_1.emitNervesEvent)({
|
|
657
|
+
level: "warn",
|
|
658
|
+
component: "senses",
|
|
659
|
+
event: "senses.bluebubbles_get_message_text_error",
|
|
660
|
+
message: "message payload missing text field",
|
|
661
|
+
meta: { messageGuid, hasData: !!data, textType: data ? typeof data.text : "n/a" },
|
|
662
|
+
});
|
|
663
|
+
return null;
|
|
664
|
+
}
|
|
665
|
+
(0, runtime_1.emitNervesEvent)({
|
|
666
|
+
component: "senses",
|
|
667
|
+
event: "senses.bluebubbles_get_message_text",
|
|
668
|
+
message: "fetched message text by guid",
|
|
669
|
+
meta: { messageGuid },
|
|
670
|
+
});
|
|
671
|
+
return data.text.trim() || null;
|
|
672
|
+
}
|
|
673
|
+
catch (error) {
|
|
674
|
+
(0, runtime_1.emitNervesEvent)({
|
|
675
|
+
level: "warn",
|
|
676
|
+
component: "senses",
|
|
677
|
+
event: "senses.bluebubbles_get_message_text_error",
|
|
678
|
+
message: "exception fetching message text",
|
|
679
|
+
meta: { messageGuid, reason: error instanceof Error ? error.message : String(error) },
|
|
680
|
+
});
|
|
681
|
+
return null;
|
|
682
|
+
}
|
|
683
|
+
},
|
|
684
|
+
};
|
|
685
|
+
}
|