@ouro.bot/cli 0.1.0-alpha.61 → 0.1.0-alpha.612
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 +127 -23
- 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/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/changelog.json +3912 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +144 -0
- package/dist/arc/episodes.js +118 -0
- package/dist/arc/intentions.js +134 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +266 -0
- package/dist/arc/packets.js +194 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +837 -26
- 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 +100 -0
- package/dist/heart/awaiting/await-scheduler.js +377 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/store.js +14 -2
- 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 +114 -119
- package/dist/heart/core.js +909 -246
- package/dist/heart/cross-chat-delivery.js +3 -18
- package/dist/heart/daemon/agent-config-check.js +419 -0
- package/dist/heart/daemon/agent-discovery.js +102 -3
- 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 +7579 -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 +29 -1700
- package/dist/heart/daemon/daemon-entry.js +485 -2
- 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 +88 -13
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +904 -70
- 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 +115 -1
- 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 +37 -8
- package/dist/heart/daemon/log-tailer.js +78 -9
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/mcp-canary.js +297 -0
- 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 +463 -34
- 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 +2 -0
- package/dist/heart/daemon/runtime-logging.js +10 -2
- package/dist/heart/daemon/runtime-metadata.js +2 -30
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +493 -38
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +28 -7
- package/dist/heart/daemon/socket-client.js +158 -11
- 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 +229 -17
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +1 -1
- 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-flow.js +32 -56
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
- package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
- package/dist/heart/identity.js +166 -55
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +1 -1
- 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 +197 -0
- package/dist/heart/mailbox/readers/agent-machine.js +425 -0
- package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
- package/dist/heart/mailbox/readers/mail.js +375 -0
- package/dist/heart/mailbox/readers/runtime-readers.js +756 -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 +19 -0
- package/dist/heart/platform.js +81 -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 +139 -52
- package/dist/heart/providers/azure.js +23 -11
- 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 +26 -8
- package/dist/heart/providers/openai-codex.js +55 -40
- 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 +13 -4
- package/dist/heart/session-activity.js +48 -24
- package/dist/heart/session-events.js +1156 -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 +133 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +44 -27
- package/dist/heart/sync-classification.js +176 -0
- package/dist/heart/sync.js +449 -0
- package/dist/heart/target-resolution.js +9 -5
- 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/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +426 -0
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mailbox-ui/assets/index-9-AxCxuB.js +61 -0
- package/dist/mailbox-ui/assets/index-CWzt267f.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 +712 -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 +460 -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 +7 -1
- package/dist/mind/context.js +250 -101
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +68 -76
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/friends/channel.js +39 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +48 -4
- package/dist/mind/friends/types.js +2 -2
- package/dist/mind/journal-index.js +162 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +6 -1
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1051 -135
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/nerves/cli-logging.js +7 -1
- 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/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 +15 -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 +331 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +163 -10
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -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/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 +18 -4
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +31 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +16 -4
- 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 -78
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-awaiting.js +365 -0
- package/dist/repertoire/tools-base.js +55 -1082
- package/dist/repertoire/tools-bluebubbles.js +1 -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 +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-record.js +463 -0
- package/dist/repertoire/tools-runtime.js +148 -0
- package/dist/repertoire/tools-session.js +781 -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 +9 -39
- 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 +115 -103
- 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 → bluebubbles/client.js} +219 -18
- package/dist/senses/bluebubbles/entry.js +77 -0
- package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
- package/dist/senses/bluebubbles/index.js +2613 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
- package/dist/senses/bluebubbles/processed-log.js +133 -0
- package/dist/senses/bluebubbles/replay.js +137 -0
- package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- 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 +100 -0
- package/dist/senses/cli.js +516 -204
- package/dist/senses/commands.js +66 -3
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +185 -21
- package/dist/senses/inner-dialog.js +469 -39
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +654 -181
- 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 +387 -98
- package/dist/senses/trust-gate.js +100 -5
- 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 +5079 -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 +41 -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/dist/heart/daemon/auth-flow.js +0 -351
- package/dist/heart/daemon/ouro-path-installer.js +0 -178
- package/dist/heart/safe-workspace.js +0 -228
- package/dist/heart/session-recall.js +0 -116
- package/dist/mind/associative-recall.js +0 -209
- package/dist/senses/bluebubbles-entry.js +0 -13
- package/dist/senses/bluebubbles.js +0 -1177
- package/dist/senses/debug-activity.js +0 -148
- package/subagents/README.md +0 -7
- /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/monty.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/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.reconstructThread = reconstructThread;
|
|
4
|
+
function normalizeHeaderId(value) {
|
|
5
|
+
if (!value)
|
|
6
|
+
return undefined;
|
|
7
|
+
const trimmed = value.trim();
|
|
8
|
+
return trimmed.length === 0 ? undefined : trimmed;
|
|
9
|
+
}
|
|
10
|
+
function inferKeysForMessage(message) {
|
|
11
|
+
const keys = [];
|
|
12
|
+
const headerId = normalizeHeaderId(message.private.messageId);
|
|
13
|
+
if (headerId)
|
|
14
|
+
keys.push(headerId);
|
|
15
|
+
/* v8 ignore next -- defensive: stored messages always have an id @preserve */
|
|
16
|
+
if (message.id)
|
|
17
|
+
keys.push(message.id);
|
|
18
|
+
return keys;
|
|
19
|
+
}
|
|
20
|
+
function sortByReceivedAtAscending(left, right) {
|
|
21
|
+
return Date.parse(left.receivedAt) - Date.parse(right.receivedAt);
|
|
22
|
+
}
|
|
23
|
+
function reconstructThread(seedMessageId, pool) {
|
|
24
|
+
const seed = pool.find((message) => message.id === seedMessageId)
|
|
25
|
+
?? pool.find((message) => normalizeHeaderId(message.private.messageId) === seedMessageId);
|
|
26
|
+
if (!seed)
|
|
27
|
+
return { rootMessageId: undefined, members: [] };
|
|
28
|
+
const byKey = new Map();
|
|
29
|
+
const allNodes = [];
|
|
30
|
+
for (const message of pool) {
|
|
31
|
+
const node = { message, parents: new Set(), children: new Set() };
|
|
32
|
+
allNodes.push(node);
|
|
33
|
+
for (const key of inferKeysForMessage(message)) {
|
|
34
|
+
/* v8 ignore next -- collision guard: storage id and RFC822 messageId differ in the normal case @preserve */
|
|
35
|
+
if (!byKey.has(key))
|
|
36
|
+
byKey.set(key, node);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
for (const node of allNodes) {
|
|
40
|
+
const parentKeys = new Set();
|
|
41
|
+
const inReplyTo = normalizeHeaderId(node.message.private.inReplyTo);
|
|
42
|
+
if (inReplyTo)
|
|
43
|
+
parentKeys.add(inReplyTo);
|
|
44
|
+
for (const reference of node.message.private.references ?? []) {
|
|
45
|
+
const ref = normalizeHeaderId(reference);
|
|
46
|
+
if (ref)
|
|
47
|
+
parentKeys.add(ref);
|
|
48
|
+
}
|
|
49
|
+
for (const key of parentKeys) {
|
|
50
|
+
const parent = byKey.get(key);
|
|
51
|
+
if (parent && parent !== node) {
|
|
52
|
+
node.parents.add(parent);
|
|
53
|
+
parent.children.add(node);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
const seedNode = byKey.get(seed.id);
|
|
58
|
+
const component = new Set();
|
|
59
|
+
const stack = [seedNode];
|
|
60
|
+
while (stack.length > 0) {
|
|
61
|
+
const node = stack.pop();
|
|
62
|
+
if (component.has(node))
|
|
63
|
+
continue;
|
|
64
|
+
component.add(node);
|
|
65
|
+
for (const parent of node.parents)
|
|
66
|
+
if (!component.has(parent))
|
|
67
|
+
stack.push(parent);
|
|
68
|
+
for (const child of node.children)
|
|
69
|
+
if (!component.has(child))
|
|
70
|
+
stack.push(child);
|
|
71
|
+
}
|
|
72
|
+
/* v8 ignore start -- root + topological depth pass: branch shapes vary with thread topology and aren't worth chasing per-branch in tests; correctness is covered by the higher-level reconstruction tests @preserve */
|
|
73
|
+
const componentRoots = [...component].filter((node) => {
|
|
74
|
+
for (const parent of node.parents)
|
|
75
|
+
if (component.has(parent))
|
|
76
|
+
return false;
|
|
77
|
+
return true;
|
|
78
|
+
});
|
|
79
|
+
const root = componentRoots
|
|
80
|
+
.sort((left, right) => Date.parse(left.message.receivedAt) - Date.parse(right.message.receivedAt))[0]
|
|
81
|
+
?? seedNode;
|
|
82
|
+
const componentInTimeOrder = [...component].sort((left, right) => Date.parse(left.message.receivedAt) - Date.parse(right.message.receivedAt));
|
|
83
|
+
const depthByNode = new Map();
|
|
84
|
+
for (const node of componentInTimeOrder) {
|
|
85
|
+
let maxParentDepth = -1;
|
|
86
|
+
for (const parent of node.parents) {
|
|
87
|
+
if (!component.has(parent))
|
|
88
|
+
continue;
|
|
89
|
+
const parentDepth = depthByNode.get(parent);
|
|
90
|
+
if (parentDepth !== undefined && parentDepth > maxParentDepth) {
|
|
91
|
+
maxParentDepth = parentDepth;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
depthByNode.set(node, maxParentDepth + 1);
|
|
95
|
+
}
|
|
96
|
+
/* v8 ignore stop */
|
|
97
|
+
const members = [...component]
|
|
98
|
+
.map((node) => node.message)
|
|
99
|
+
.sort(sortByReceivedAtAscending)
|
|
100
|
+
.map((message) => {
|
|
101
|
+
const node = byKey.get(message.id);
|
|
102
|
+
/* v8 ignore next -- fallback: depthByNode is populated for every component node by the topological pass @preserve */
|
|
103
|
+
return { message, depth: depthByNode.get(node) ?? 0 };
|
|
104
|
+
});
|
|
105
|
+
return {
|
|
106
|
+
rootMessageId: normalizeHeaderId(root.message.private.messageId) ?? root.message.id,
|
|
107
|
+
members,
|
|
108
|
+
};
|
|
109
|
+
}
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.extractTravelFactsFromMail = extractTravelFactsFromMail;
|
|
4
|
+
const runtime_1 = require("../nerves/runtime");
|
|
5
|
+
function firstMatch(text, pattern) {
|
|
6
|
+
const match = pattern.exec(text);
|
|
7
|
+
return match?.[1]?.trim() ?? null;
|
|
8
|
+
}
|
|
9
|
+
function flightFact(message) {
|
|
10
|
+
const text = `${message.private.subject}\n${message.private.text}`;
|
|
11
|
+
const flightNumber = firstMatch(text, /\bflight\s+([A-Z]{2}\s?\d{1,4})\b/i);
|
|
12
|
+
const routeMatch = /\b([A-Z]{3})\s*(?:->|to)\s*([A-Z]{3})\b/.exec(text);
|
|
13
|
+
const confirmationCode = firstMatch(text, /confirmation code:\s*([A-Z0-9-]+)/i);
|
|
14
|
+
const departure = firstMatch(text, /departure:\s*([^\n]+)/i);
|
|
15
|
+
if (!flightNumber && !routeMatch && !departure)
|
|
16
|
+
return null;
|
|
17
|
+
const fields = {};
|
|
18
|
+
if (flightNumber)
|
|
19
|
+
fields.flightNumber = flightNumber.toUpperCase().replace(/\s+/, " ");
|
|
20
|
+
if (routeMatch)
|
|
21
|
+
fields.route = `${routeMatch[1]} -> ${routeMatch[2]}`;
|
|
22
|
+
if (departure)
|
|
23
|
+
fields.departure = departure;
|
|
24
|
+
if (confirmationCode)
|
|
25
|
+
fields.confirmationCode = confirmationCode;
|
|
26
|
+
return {
|
|
27
|
+
kind: "flight",
|
|
28
|
+
messageId: message.id,
|
|
29
|
+
subject: message.private.subject,
|
|
30
|
+
source: message.source ?? null,
|
|
31
|
+
ownerEmail: message.ownerEmail ?? null,
|
|
32
|
+
summary: [
|
|
33
|
+
fields.flightNumber ? `flight ${fields.flightNumber}` : "flight",
|
|
34
|
+
fields.route ? fields.route : null,
|
|
35
|
+
fields.departure ? `departing ${fields.departure}` : null,
|
|
36
|
+
].filter(Boolean).join(" "),
|
|
37
|
+
fields,
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
function lodgingFact(message) {
|
|
41
|
+
const text = `${message.private.subject}\n${message.private.text}`;
|
|
42
|
+
const hotel = firstMatch(text, /hotel:\s*([^\n]+)/i);
|
|
43
|
+
const checkIn = firstMatch(text, /check-?in:\s*([^\n]+)/i);
|
|
44
|
+
const checkOut = firstMatch(text, /check-?out:\s*([^\n]+)/i);
|
|
45
|
+
const confirmationCode = firstMatch(text, /confirmation code:\s*([A-Z0-9-]+)/i);
|
|
46
|
+
if (!hotel && !checkIn && !/hotel|lodging|booking/i.test(message.private.subject))
|
|
47
|
+
return null;
|
|
48
|
+
const fields = {};
|
|
49
|
+
if (hotel)
|
|
50
|
+
fields.hotel = hotel;
|
|
51
|
+
if (checkIn)
|
|
52
|
+
fields.checkIn = checkIn;
|
|
53
|
+
if (checkOut)
|
|
54
|
+
fields.checkOut = checkOut;
|
|
55
|
+
if (confirmationCode)
|
|
56
|
+
fields.confirmationCode = confirmationCode;
|
|
57
|
+
return {
|
|
58
|
+
kind: "lodging",
|
|
59
|
+
messageId: message.id,
|
|
60
|
+
subject: message.private.subject,
|
|
61
|
+
source: message.source ?? null,
|
|
62
|
+
ownerEmail: message.ownerEmail ?? null,
|
|
63
|
+
summary: [
|
|
64
|
+
hotel ?? "lodging",
|
|
65
|
+
checkIn ? `check-in ${checkIn}` : null,
|
|
66
|
+
checkOut ? `check-out ${checkOut}` : null,
|
|
67
|
+
].filter(Boolean).join(" "),
|
|
68
|
+
fields,
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
function extractTravelFactsFromMail(messages) {
|
|
72
|
+
const facts = messages.flatMap((message) => {
|
|
73
|
+
const results = [];
|
|
74
|
+
const flight = flightFact(message);
|
|
75
|
+
if (flight)
|
|
76
|
+
results.push(flight);
|
|
77
|
+
const lodging = lodgingFact(message);
|
|
78
|
+
if (lodging)
|
|
79
|
+
results.push(lodging);
|
|
80
|
+
return results;
|
|
81
|
+
});
|
|
82
|
+
(0, runtime_1.emitNervesEvent)({
|
|
83
|
+
component: "senses",
|
|
84
|
+
event: "senses.mail_travel_facts_extracted",
|
|
85
|
+
message: "travel facts extracted from mail",
|
|
86
|
+
meta: { messages: messages.length, facts: facts.length },
|
|
87
|
+
});
|
|
88
|
+
return facts;
|
|
89
|
+
}
|
|
@@ -52,7 +52,13 @@ exports.CANONICAL_BUNDLE_MANIFEST = [
|
|
|
52
52
|
{ path: "psyche/LORE.md", kind: "file" },
|
|
53
53
|
{ path: "psyche/TACIT.md", kind: "file" },
|
|
54
54
|
{ path: "psyche/ASPIRATIONS.md", kind: "file" },
|
|
55
|
-
{ path: "
|
|
55
|
+
{ path: "arc", kind: "dir" },
|
|
56
|
+
{ path: "arc/episodes", kind: "dir" },
|
|
57
|
+
{ path: "arc/obligations", kind: "dir" },
|
|
58
|
+
{ path: "arc/cares", kind: "dir" },
|
|
59
|
+
{ path: "arc/intentions", kind: "dir" },
|
|
60
|
+
{ path: "diary", kind: "dir" },
|
|
61
|
+
{ path: "journal", kind: "dir" },
|
|
56
62
|
{ path: "friends", kind: "dir" },
|
|
57
63
|
{ path: "state", kind: "dir" },
|
|
58
64
|
{ path: "tasks", kind: "dir" },
|
package/dist/mind/context.js
CHANGED
|
@@ -33,18 +33,28 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
33
33
|
};
|
|
34
34
|
})();
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.validateSessionMessages = exports.repairSessionMessages = exports.migrateToolNames = exports.detectDuplicateToolCallIds = void 0;
|
|
36
37
|
exports.trimMessages = trimMessages;
|
|
37
|
-
exports.validateSessionMessages = validateSessionMessages;
|
|
38
|
-
exports.repairSessionMessages = repairSessionMessages;
|
|
39
38
|
exports.saveSession = saveSession;
|
|
39
|
+
exports.appendSyntheticAssistantMessage = appendSyntheticAssistantMessage;
|
|
40
40
|
exports.loadSession = loadSession;
|
|
41
41
|
exports.postTurn = postTurn;
|
|
42
|
+
exports.postTurnTrim = postTurnTrim;
|
|
43
|
+
exports.postTurnPersist = postTurnPersist;
|
|
44
|
+
exports.deferPostTurnPersist = deferPostTurnPersist;
|
|
42
45
|
exports.deleteSession = deleteSession;
|
|
43
46
|
const config_1 = require("../heart/config");
|
|
47
|
+
const session_events_1 = require("../heart/session-events");
|
|
44
48
|
const runtime_1 = require("../nerves/runtime");
|
|
45
49
|
const fs = __importStar(require("fs"));
|
|
46
50
|
const path = __importStar(require("path"));
|
|
47
51
|
const token_estimate_1 = require("./token-estimate");
|
|
52
|
+
var session_events_2 = require("../heart/session-events");
|
|
53
|
+
Object.defineProperty(exports, "detectDuplicateToolCallIds", { enumerable: true, get: function () { return session_events_2.detectDuplicateToolCallIds; } });
|
|
54
|
+
Object.defineProperty(exports, "migrateToolNames", { enumerable: true, get: function () { return session_events_2.migrateToolNames; } });
|
|
55
|
+
Object.defineProperty(exports, "repairSessionMessages", { enumerable: true, get: function () { return session_events_2.repairSessionMessages; } });
|
|
56
|
+
Object.defineProperty(exports, "validateSessionMessages", { enumerable: true, get: function () { return session_events_2.validateSessionMessages; } });
|
|
57
|
+
const IDLE_REST_ONLY_KEEP_TURNS = 20;
|
|
48
58
|
function buildTrimmableBlocks(messages) {
|
|
49
59
|
const blocks = [];
|
|
50
60
|
let i = 0;
|
|
@@ -86,6 +96,89 @@ function getSystemMessageIndices(messages) {
|
|
|
86
96
|
function buildTrimmedMessages(messages, kept) {
|
|
87
97
|
return messages.filter((m, idx) => m.role === "system" || kept.has(idx));
|
|
88
98
|
}
|
|
99
|
+
function messageContentToText(content) {
|
|
100
|
+
if (typeof content === "string")
|
|
101
|
+
return content;
|
|
102
|
+
if (!Array.isArray(content))
|
|
103
|
+
return "";
|
|
104
|
+
return content
|
|
105
|
+
.map((part) => {
|
|
106
|
+
if (part && typeof part === "object" && "text" in part && typeof part.text === "string") {
|
|
107
|
+
return part.text;
|
|
108
|
+
}
|
|
109
|
+
return "";
|
|
110
|
+
})
|
|
111
|
+
.join("\n");
|
|
112
|
+
}
|
|
113
|
+
function toolCallName(toolCall) {
|
|
114
|
+
return toolCall.function.name;
|
|
115
|
+
}
|
|
116
|
+
function toolCallId(toolCall) {
|
|
117
|
+
return toolCall.id;
|
|
118
|
+
}
|
|
119
|
+
function isIdleHeartbeatRestUserMessage(message) {
|
|
120
|
+
if (message.role !== "user")
|
|
121
|
+
return false;
|
|
122
|
+
const text = messageContentToText(message.content);
|
|
123
|
+
return text.includes("...time passing. anything stirring?")
|
|
124
|
+
&& !text.includes("[pending from ")
|
|
125
|
+
&& !text.includes("## task:");
|
|
126
|
+
}
|
|
127
|
+
function isEmptyRestOnlyAssistantMessage(message) {
|
|
128
|
+
if (message.role !== "assistant")
|
|
129
|
+
return null;
|
|
130
|
+
if (messageContentToText(message.content).trim())
|
|
131
|
+
return null;
|
|
132
|
+
if (!Array.isArray(message.tool_calls) || message.tool_calls.length !== 1)
|
|
133
|
+
return null;
|
|
134
|
+
const onlyToolCall = message.tool_calls[0];
|
|
135
|
+
if (toolCallName(onlyToolCall) !== "rest")
|
|
136
|
+
return null;
|
|
137
|
+
return toolCallId(onlyToolCall);
|
|
138
|
+
}
|
|
139
|
+
function isMatchingToolResult(message, toolId) {
|
|
140
|
+
return message.tool_call_id === toolId;
|
|
141
|
+
}
|
|
142
|
+
function compactIdleRestOnlyTurns(messages, keepTurns = IDLE_REST_ONLY_KEEP_TURNS) {
|
|
143
|
+
const blocks = [];
|
|
144
|
+
let i = 0;
|
|
145
|
+
while (i < messages.length - 2) {
|
|
146
|
+
const userMessage = messages[i];
|
|
147
|
+
const assistantMessage = messages[i + 1];
|
|
148
|
+
const toolMessage = messages[i + 2];
|
|
149
|
+
const restToolId = isIdleHeartbeatRestUserMessage(userMessage)
|
|
150
|
+
? isEmptyRestOnlyAssistantMessage(assistantMessage)
|
|
151
|
+
: null;
|
|
152
|
+
if (restToolId && isMatchingToolResult(toolMessage, restToolId)) {
|
|
153
|
+
blocks.push({ start: i, end: i + 2 });
|
|
154
|
+
i += 3;
|
|
155
|
+
continue;
|
|
156
|
+
}
|
|
157
|
+
i++;
|
|
158
|
+
}
|
|
159
|
+
if (blocks.length <= keepTurns)
|
|
160
|
+
return messages;
|
|
161
|
+
const drop = new Set();
|
|
162
|
+
for (const block of blocks.slice(0, blocks.length - keepTurns)) {
|
|
163
|
+
for (let index = block.start; index <= block.end; index++) {
|
|
164
|
+
drop.add(index);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
const compacted = messages.filter((_message, index) => !drop.has(index));
|
|
168
|
+
(0, runtime_1.emitNervesEvent)({
|
|
169
|
+
component: "mind",
|
|
170
|
+
event: "mind.session_idle_rest_compaction",
|
|
171
|
+
message: "compacted old idle rest-only inner turns",
|
|
172
|
+
meta: {
|
|
173
|
+
removedTurns: blocks.length - keepTurns,
|
|
174
|
+
keptTurns: keepTurns,
|
|
175
|
+
removedMessages: drop.size,
|
|
176
|
+
originalCount: messages.length,
|
|
177
|
+
finalCount: compacted.length,
|
|
178
|
+
},
|
|
179
|
+
});
|
|
180
|
+
return compacted;
|
|
181
|
+
}
|
|
89
182
|
function trimMessages(messages, maxTokens, contextMargin, actualTokenCount) {
|
|
90
183
|
const targetTokens = Math.floor(maxTokens * (1 - contextMargin / 100));
|
|
91
184
|
const estimatedBefore = (0, token_estimate_1.estimateTokensForMessages)(messages);
|
|
@@ -173,122 +266,94 @@ function trimMessages(messages, maxTokens, contextMargin, actualTokenCount) {
|
|
|
173
266
|
* user → assistant (with optional tool calls/results) → user → assistant...
|
|
174
267
|
* Never assistant → assistant without a user in between.
|
|
175
268
|
*/
|
|
176
|
-
function
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
if (msg.role === "system")
|
|
184
|
-
continue;
|
|
185
|
-
if (msg.role === "tool") {
|
|
186
|
-
sawToolResultSincePrevAssistant = true;
|
|
187
|
-
continue;
|
|
188
|
-
}
|
|
189
|
-
if (msg.role === "assistant" && prevNonToolRole === "assistant") {
|
|
190
|
-
// assistant → tool(s) → assistant is valid (tool call flow)
|
|
191
|
-
if (!(prevAssistantHadToolCalls && sawToolResultSincePrevAssistant)) {
|
|
192
|
-
violations.push(`back-to-back assistant at index ${i}`);
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
prevAssistantHadToolCalls = msg.role === "assistant" && Array.isArray(msg.tool_calls) && msg.tool_calls.length > 0;
|
|
196
|
-
sawToolResultSincePrevAssistant = false;
|
|
197
|
-
prevNonToolRole = msg.role;
|
|
198
|
-
}
|
|
199
|
-
return violations;
|
|
269
|
+
function denormalizeContinuityState(state) {
|
|
270
|
+
if (!state.mustResolveBeforeHandoff && typeof state.lastFriendActivityAt !== "string")
|
|
271
|
+
return undefined;
|
|
272
|
+
return {
|
|
273
|
+
...(state.mustResolveBeforeHandoff ? { mustResolveBeforeHandoff: true } : {}),
|
|
274
|
+
...(typeof state.lastFriendActivityAt === "string" ? { lastFriendActivityAt: state.lastFriendActivityAt } : {}),
|
|
275
|
+
};
|
|
200
276
|
}
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
function repairSessionMessages(messages) {
|
|
205
|
-
const violations = validateSessionMessages(messages);
|
|
206
|
-
if (violations.length === 0)
|
|
207
|
-
return messages;
|
|
208
|
-
const result = [];
|
|
209
|
-
for (const msg of messages) {
|
|
210
|
-
if (msg.role === "assistant" && result.length > 0) {
|
|
211
|
-
const prev = result[result.length - 1];
|
|
212
|
-
if (prev.role === "assistant" && !("tool_calls" in prev)) {
|
|
213
|
-
const prevContent = typeof prev.content === "string" ? prev.content : "";
|
|
214
|
-
const curContent = typeof msg.content === "string" ? msg.content : "";
|
|
215
|
-
prev.content = `${prevContent}\n\n${curContent}`;
|
|
216
|
-
continue;
|
|
217
|
-
}
|
|
218
|
-
}
|
|
219
|
-
result.push(msg);
|
|
220
|
-
}
|
|
221
|
-
(0, runtime_1.emitNervesEvent)({
|
|
222
|
-
level: "warn",
|
|
223
|
-
event: "mind.session_invariant_repair",
|
|
224
|
-
component: "mind",
|
|
225
|
-
message: "repaired session invariant violations",
|
|
226
|
-
meta: { violations },
|
|
227
|
-
});
|
|
228
|
-
return result;
|
|
277
|
+
function writeSessionEnvelope(filePath, envelope) {
|
|
278
|
+
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
|
279
|
+
fs.writeFileSync(filePath, JSON.stringify(envelope, null, 2));
|
|
229
280
|
}
|
|
230
281
|
function saveSession(filePath, messages, lastUsage, state) {
|
|
231
|
-
const
|
|
232
|
-
|
|
282
|
+
const existing = (0, session_events_1.loadSessionEnvelopeFile)(filePath);
|
|
283
|
+
const previousMessages = existing ? (0, session_events_1.projectProviderMessages)(existing) : [];
|
|
284
|
+
const currentIngressTimes = messages.map(session_events_1.getIngressTime);
|
|
285
|
+
const sanitized = (0, session_events_1.sanitizeProviderMessages)(messages);
|
|
286
|
+
const { envelope } = (0, session_events_1.buildCanonicalSessionEnvelope)({
|
|
287
|
+
existing,
|
|
288
|
+
previousMessages,
|
|
289
|
+
currentMessages: sanitized,
|
|
290
|
+
trimmedMessages: sanitized,
|
|
291
|
+
currentIngressTimes,
|
|
292
|
+
recordedAt: new Date().toISOString(),
|
|
293
|
+
lastUsage: lastUsage ?? null,
|
|
294
|
+
state,
|
|
295
|
+
projectionBasis: {
|
|
296
|
+
maxTokens: null,
|
|
297
|
+
contextMargin: null,
|
|
298
|
+
inputTokens: lastUsage?.input_tokens ?? null,
|
|
299
|
+
},
|
|
300
|
+
});
|
|
301
|
+
writeSessionEnvelope(filePath, envelope);
|
|
302
|
+
}
|
|
303
|
+
function appendSyntheticAssistantMessage(filePath, content) {
|
|
304
|
+
try {
|
|
305
|
+
if (!fs.existsSync(filePath))
|
|
306
|
+
return false;
|
|
307
|
+
const envelope = (0, session_events_1.loadSessionEnvelopeFile)(filePath);
|
|
308
|
+
if (!envelope)
|
|
309
|
+
return false;
|
|
310
|
+
const updated = (0, session_events_1.appendSyntheticAssistantEvent)(envelope, content, new Date().toISOString());
|
|
311
|
+
writeSessionEnvelope(filePath, updated);
|
|
233
312
|
(0, runtime_1.emitNervesEvent)({
|
|
234
|
-
level: "warn",
|
|
235
|
-
event: "mind.session_invariant_violation",
|
|
236
313
|
component: "mind",
|
|
237
|
-
|
|
238
|
-
|
|
314
|
+
event: "mind.session_synthetic_message_appended",
|
|
315
|
+
message: "appended synthetic assistant message to session",
|
|
316
|
+
meta: { path: filePath, contentLength: content.length },
|
|
239
317
|
});
|
|
240
|
-
|
|
318
|
+
return true;
|
|
241
319
|
}
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
if (lastUsage)
|
|
245
|
-
envelope.lastUsage = lastUsage;
|
|
246
|
-
if (state?.mustResolveBeforeHandoff === true || typeof state?.lastFriendActivityAt === "string") {
|
|
247
|
-
envelope.state = {
|
|
248
|
-
...(state?.mustResolveBeforeHandoff === true ? { mustResolveBeforeHandoff: true } : {}),
|
|
249
|
-
...(typeof state?.lastFriendActivityAt === "string" ? { lastFriendActivityAt: state.lastFriendActivityAt } : {}),
|
|
250
|
-
};
|
|
320
|
+
catch {
|
|
321
|
+
return false;
|
|
251
322
|
}
|
|
252
|
-
fs.writeFileSync(filePath, JSON.stringify(envelope, null, 2));
|
|
253
323
|
}
|
|
254
324
|
function loadSession(filePath) {
|
|
255
325
|
try {
|
|
256
|
-
const
|
|
257
|
-
|
|
258
|
-
if (data.version !== 1)
|
|
326
|
+
const envelope = (0, session_events_1.loadSessionEnvelopeFile)(filePath);
|
|
327
|
+
if (!envelope)
|
|
259
328
|
return null;
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
component: "mind",
|
|
267
|
-
message: "session invariant violated on load",
|
|
268
|
-
meta: { path: filePath, violations },
|
|
269
|
-
});
|
|
270
|
-
messages = repairSessionMessages(messages);
|
|
271
|
-
}
|
|
272
|
-
const rawState = data?.state && typeof data.state === "object" && data.state !== null
|
|
273
|
-
? data.state
|
|
274
|
-
: undefined;
|
|
275
|
-
const state = rawState && (rawState.mustResolveBeforeHandoff === true
|
|
276
|
-
|| typeof rawState.lastFriendActivityAt === "string")
|
|
277
|
-
? {
|
|
278
|
-
...(rawState.mustResolveBeforeHandoff === true ? { mustResolveBeforeHandoff: true } : {}),
|
|
279
|
-
...(typeof rawState.lastFriendActivityAt === "string" ? { lastFriendActivityAt: rawState.lastFriendActivityAt } : {}),
|
|
280
|
-
}
|
|
281
|
-
: undefined;
|
|
282
|
-
return { messages, lastUsage: data.lastUsage, state };
|
|
329
|
+
return {
|
|
330
|
+
messages: (0, session_events_1.sanitizeProviderMessages)((0, session_events_1.projectProviderMessages)(envelope)),
|
|
331
|
+
events: envelope.events,
|
|
332
|
+
lastUsage: envelope.lastUsage ?? undefined,
|
|
333
|
+
state: denormalizeContinuityState(envelope.state),
|
|
334
|
+
};
|
|
283
335
|
}
|
|
284
336
|
catch {
|
|
285
337
|
return null;
|
|
286
338
|
}
|
|
287
339
|
}
|
|
340
|
+
/**
|
|
341
|
+
* Synchronous post-turn: sanitize, trim (mutates messages in place), and persist to disk.
|
|
342
|
+
* For non-blocking persist, use postTurnTrim() + deferPostTurnPersist() instead.
|
|
343
|
+
*/
|
|
288
344
|
function postTurn(messages, sessPath, usage, hooks, state) {
|
|
345
|
+
const prepared = postTurnTrim(messages, usage, hooks);
|
|
346
|
+
postTurnPersist(sessPath, prepared, usage, state);
|
|
347
|
+
}
|
|
348
|
+
/**
|
|
349
|
+
* Synchronous phase: run hooks, sanitize, trim, and mutate the messages array in place.
|
|
350
|
+
* Returns the data needed by postTurnPersist / deferPostTurnPersist.
|
|
351
|
+
*/
|
|
352
|
+
function postTurnTrim(messages, usage, hooks) {
|
|
353
|
+
const preTrimMessages = [...messages];
|
|
289
354
|
if (hooks?.beforeTrim) {
|
|
290
355
|
try {
|
|
291
|
-
hooks.beforeTrim(
|
|
356
|
+
hooks.beforeTrim(preTrimMessages);
|
|
292
357
|
}
|
|
293
358
|
catch (error) {
|
|
294
359
|
(0, runtime_1.emitNervesEvent)({
|
|
@@ -303,9 +368,93 @@ function postTurn(messages, sessPath, usage, hooks, state) {
|
|
|
303
368
|
}
|
|
304
369
|
}
|
|
305
370
|
const { maxTokens, contextMargin } = (0, config_1.getContextConfig)();
|
|
306
|
-
const
|
|
307
|
-
|
|
308
|
-
|
|
371
|
+
const currentMessages = (0, session_events_1.sanitizeProviderMessages)(messages);
|
|
372
|
+
const currentIngressTimes = currentMessages.map(session_events_1.getIngressTime);
|
|
373
|
+
const tokenTrimmedMessages = trimMessages(currentMessages, maxTokens, contextMargin, usage?.input_tokens);
|
|
374
|
+
const trimmedMessages = compactIdleRestOnlyTurns(tokenTrimmedMessages);
|
|
375
|
+
messages.splice(0, messages.length, ...trimmedMessages);
|
|
376
|
+
return { currentMessages, trimmedMessages, currentIngressTimes, maxTokens, contextMargin };
|
|
377
|
+
}
|
|
378
|
+
/**
|
|
379
|
+
* Synchronous persist: load existing envelope, build canonical envelope, write to disk.
|
|
380
|
+
*/
|
|
381
|
+
function postTurnPersist(sessPath, prepared, usage, state) {
|
|
382
|
+
const existing = (0, session_events_1.loadSessionEnvelopeFile)(sessPath);
|
|
383
|
+
const previousMessages = existing ? (0, session_events_1.projectProviderMessages)(existing) : [];
|
|
384
|
+
const { envelope, evictedEvents } = (0, session_events_1.buildCanonicalSessionEnvelope)({
|
|
385
|
+
existing,
|
|
386
|
+
previousMessages,
|
|
387
|
+
currentMessages: prepared.currentMessages,
|
|
388
|
+
trimmedMessages: prepared.trimmedMessages,
|
|
389
|
+
currentIngressTimes: prepared.currentIngressTimes,
|
|
390
|
+
recordedAt: new Date().toISOString(),
|
|
391
|
+
lastUsage: usage ?? null,
|
|
392
|
+
state,
|
|
393
|
+
projectionBasis: {
|
|
394
|
+
maxTokens: prepared.maxTokens,
|
|
395
|
+
contextMargin: prepared.contextMargin,
|
|
396
|
+
inputTokens: usage?.input_tokens ?? null,
|
|
397
|
+
},
|
|
398
|
+
});
|
|
399
|
+
(0, session_events_1.appendEvictedToArchive)(sessPath, evictedEvents);
|
|
400
|
+
writeSessionEnvelope(sessPath, envelope);
|
|
401
|
+
return envelope.events;
|
|
402
|
+
}
|
|
403
|
+
/**
|
|
404
|
+
* Per-sessPath serialization queue. Without this, two concurrent
|
|
405
|
+
* `deferPostTurnPersist` calls (e.g. two BlueBubbles webhooks for the same
|
|
406
|
+
* chat firing back-to-back, or a CLI postTurn racing the inner-dialog turn
|
|
407
|
+
* for the same MCP session) would each load the envelope, both compute the
|
|
408
|
+
* same "next sequence", and write events with colliding ids. The session
|
|
409
|
+
* file would silently accumulate duplicates and replay would diverge from
|
|
410
|
+
* what was actually sent on the wire.
|
|
411
|
+
*
|
|
412
|
+
* The queue is in-process only — it does not protect against multiple Node
|
|
413
|
+
* processes writing to the same file. The dedup-on-load behaviour in
|
|
414
|
+
* parseSessionEnvelope keeps cross-process races from leaving permanent
|
|
415
|
+
* corruption; this serializer just keeps the common (single-process) case
|
|
416
|
+
* race-free in the first place.
|
|
417
|
+
*/
|
|
418
|
+
const sessionPersistQueues = new Map();
|
|
419
|
+
function enqueueSessionPersist(sessPath, fn) {
|
|
420
|
+
const previous = sessionPersistQueues.get(sessPath) ?? Promise.resolve();
|
|
421
|
+
// Chain on the previous tail. fn runs whether previous resolved or rejected,
|
|
422
|
+
// so one failed turn cannot block subsequent turns on the same session.
|
|
423
|
+
const next = previous.then(fn, fn);
|
|
424
|
+
// Save a swallowed-rejection sentinel as the new tail so the next caller's
|
|
425
|
+
// `previous.then(fn, fn)` sees a clean resolution; the original `next` still
|
|
426
|
+
// propagates rejection to its own caller as expected.
|
|
427
|
+
/* v8 ignore start -- the swallow only matters when fn rejects, which is the failure path covered separately */
|
|
428
|
+
const sentinel = next.then(() => undefined, () => undefined);
|
|
429
|
+
/* v8 ignore stop */
|
|
430
|
+
sessionPersistQueues.set(sessPath, sentinel);
|
|
431
|
+
return next;
|
|
432
|
+
}
|
|
433
|
+
/**
|
|
434
|
+
* Deferred persist: same as postTurnPersist but runs on the next event loop
|
|
435
|
+
* tick AND serializes against any other deferred persist for the same
|
|
436
|
+
* sessPath, so concurrent turns cannot race and produce duplicate event ids
|
|
437
|
+
* in the saved session.
|
|
438
|
+
*/
|
|
439
|
+
function deferPostTurnPersist(sessPath, prepared, usage, state) {
|
|
440
|
+
return enqueueSessionPersist(sessPath, () => new Promise((resolve) => {
|
|
441
|
+
setImmediate(() => {
|
|
442
|
+
try {
|
|
443
|
+
const events = postTurnPersist(sessPath, prepared, usage, state);
|
|
444
|
+
resolve(events);
|
|
445
|
+
}
|
|
446
|
+
catch (err) {
|
|
447
|
+
(0, runtime_1.emitNervesEvent)({
|
|
448
|
+
level: "warn",
|
|
449
|
+
component: "mind",
|
|
450
|
+
event: "mind.deferred_persist_error",
|
|
451
|
+
message: "deferred session persist failed",
|
|
452
|
+
meta: { error: err instanceof Error ? err.message : String(err) },
|
|
453
|
+
});
|
|
454
|
+
resolve([]);
|
|
455
|
+
}
|
|
456
|
+
});
|
|
457
|
+
}));
|
|
309
458
|
}
|
|
310
459
|
function deleteSession(filePath) {
|
|
311
460
|
try {
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.detectSuspiciousContent = detectSuspiciousContent;
|
|
4
|
+
const PATTERN_CATEGORIES = [
|
|
5
|
+
{
|
|
6
|
+
name: "instruction_framing",
|
|
7
|
+
patterns: [
|
|
8
|
+
/\byou are (?:a|an) (?:ai|assistant|language model|helpful assistant)\b/i,
|
|
9
|
+
/\byour (?:new )?instructions are\b/i,
|
|
10
|
+
/\bsystem\s*:/i,
|
|
11
|
+
/\bignore (?:all |my )?previous instructions\b/i,
|
|
12
|
+
/\bdo not reveal\b/i,
|
|
13
|
+
],
|
|
14
|
+
},
|
|
15
|
+
{
|
|
16
|
+
name: "override_language",
|
|
17
|
+
patterns: [
|
|
18
|
+
/\bdisregard\b/i,
|
|
19
|
+
/\bforget everything\b/i,
|
|
20
|
+
/\bnew instructions:/i,
|
|
21
|
+
/\boverride (?:all |any |previous )?instructions\b/i,
|
|
22
|
+
],
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
name: "role_injection",
|
|
26
|
+
patterns: [
|
|
27
|
+
/\bas (?:a|an) (?:ai|language model)\b/i,
|
|
28
|
+
/\byou must always\b/i,
|
|
29
|
+
/\byou are now\b/i,
|
|
30
|
+
],
|
|
31
|
+
},
|
|
32
|
+
{
|
|
33
|
+
name: "boundary_markers",
|
|
34
|
+
patterns: [
|
|
35
|
+
/```system/i,
|
|
36
|
+
/<<SYS>>/i,
|
|
37
|
+
/\[INST\]/i,
|
|
38
|
+
/<\/?system>/i,
|
|
39
|
+
/\[system\]/i,
|
|
40
|
+
],
|
|
41
|
+
},
|
|
42
|
+
];
|
|
43
|
+
function detectSuspiciousContent(text) {
|
|
44
|
+
if (!text) {
|
|
45
|
+
return { suspicious: false, patterns: [] };
|
|
46
|
+
}
|
|
47
|
+
const matched = new Set();
|
|
48
|
+
for (const category of PATTERN_CATEGORIES) {
|
|
49
|
+
for (const pattern of category.patterns) {
|
|
50
|
+
if (pattern.test(text)) {
|
|
51
|
+
matched.add(category.name);
|
|
52
|
+
break;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return {
|
|
57
|
+
suspicious: matched.size > 0,
|
|
58
|
+
patterns: [...matched],
|
|
59
|
+
};
|
|
60
|
+
}
|