@ouro.bot/cli 0.1.0-alpha.34 → 0.1.0-alpha.340
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 +188 -187
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
- package/changelog.json +2031 -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 +832 -0
- package/dist/heart/agent-entry.js +37 -2
- 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 +463 -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 +53 -21
- package/dist/heart/core.js +695 -195
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +292 -0
- package/dist/heart/daemon/agent-discovery.js +79 -3
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +170 -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 +591 -0
- package/dist/heart/daemon/cli-exec.js +2297 -0
- package/dist/heart/daemon/cli-help.js +306 -0
- package/dist/heart/daemon/cli-parse.js +824 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +512 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/daemon-cli.js +30 -1171
- package/dist/heart/daemon/daemon-entry.js +358 -3
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +751 -58
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +401 -0
- package/dist/heart/daemon/health-monitor.js +79 -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/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +91 -0
- package/dist/heart/daemon/launchd.js +46 -9
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +105 -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 +1 -1
- package/dist/heart/daemon/process-manager.js +201 -0
- package/dist/heart/daemon/provider-discovery.js +105 -0
- package/dist/heart/daemon/pulse.js +463 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +101 -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 +72 -3
- 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 +237 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/thoughts.js +510 -0
- package/dist/heart/daemon/up-progress.js +135 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/habits/habit-migration.js +181 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-scheduler.js +371 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -120
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
- package/dist/heart/{daemon → hatch}/specialist-tools.js +49 -3
- package/dist/heart/identity.js +154 -59
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +127 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/outlook/outlook-http-hooks.js +64 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +232 -0
- package/dist/heart/outlook/outlook-http-static.js +99 -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 +28 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +194 -0
- package/dist/heart/outlook/readers/agent-machine.js +355 -0
- package/dist/heart/outlook/readers/continuity-readers.js +332 -0
- package/dist/heart/outlook/readers/runtime-readers.js +660 -0
- package/dist/heart/outlook/readers/sessions.js +231 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-failover.js +135 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +234 -0
- package/dist/heart/providers/anthropic-token.js +163 -0
- package/dist/heart/providers/anthropic.js +171 -50
- package/dist/heart/providers/azure.js +97 -11
- package/dist/heart/providers/error-classification.js +63 -0
- package/dist/heart/providers/github-copilot.js +135 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +23 -6
- package/dist/heart/providers/openai-codex.js +33 -23
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +726 -0
- package/dist/heart/session-recall.js +162 -0
- package/dist/heart/start-of-turn-packet.js +341 -0
- package/dist/heart/streaming.js +36 -27
- 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 +358 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-path-installer.js +78 -35
- 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 +12 -2
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mind/associative-recall.js +137 -66
- package/dist/mind/bundle-manifest.js +7 -1
- package/dist/mind/context.js +89 -93
- 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 +14 -1
- package/dist/mind/friends/channel.js +56 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +37 -0
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +8 -0
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +74 -7
- package/dist/mind/prompt.js +999 -111
- 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 +7 -1
- package/dist/nerves/coverage/audit.js +1 -1
- package/dist/nerves/coverage/file-completeness.js +83 -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-DC7sZefn.js +61 -0
- package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
- package/dist/outlook-ui/index.html +15 -0
- 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 +319 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +79 -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 +197 -30
- package/dist/repertoire/coding/manager.js +158 -9
- 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 +527 -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 +375 -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 +14 -23
- 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 +28 -10
- 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 +316 -0
- package/dist/repertoire/tools-base.js +45 -771
- 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 +182 -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-memory.js +376 -0
- package/dist/repertoire/tools-session.js +739 -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 +12 -62
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +110 -0
- package/dist/repertoire/tools.js +144 -138
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +118 -0
- package/dist/repertoire/vault-setup.js +241 -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 → bluebubbles/client.js} +225 -9
- package/dist/senses/bluebubbles/entry.js +13 -0
- package/dist/senses/bluebubbles/inbound-log.js +113 -0
- package/dist/senses/bluebubbles/index.js +1590 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- 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 +1 -1
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +595 -246
- package/dist/senses/commands.js +65 -1
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +112 -19
- package/dist/senses/inner-dialog.js +633 -86
- package/dist/senses/pipeline.js +567 -0
- package/dist/senses/shared-turn.js +199 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams.js +665 -160
- package/dist/senses/trust-gate.js +112 -2
- package/package.json +29 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +110 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +81 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/subagent-installer.js +0 -134
- package/dist/senses/bluebubbles-entry.js +0 -11
- package/dist/senses/bluebubbles.js +0 -547
- package/dist/senses/debug-activity.js +0 -124
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -235
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -382
- /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-serpent.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,162 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.recallSession = recallSession;
|
|
4
|
+
exports.searchSessionTranscript = searchSessionTranscript;
|
|
5
|
+
const runtime_1 = require("../nerves/runtime");
|
|
6
|
+
const session_events_1 = require("./session-events");
|
|
7
|
+
function normalizeSessionMessages(events) {
|
|
8
|
+
return events
|
|
9
|
+
.map((event) => ({
|
|
10
|
+
id: event.id,
|
|
11
|
+
role: event.role,
|
|
12
|
+
content: (0, session_events_1.extractEventText)(event),
|
|
13
|
+
timestamp: (0, session_events_1.formatSessionEventTimestamp)(event),
|
|
14
|
+
}))
|
|
15
|
+
.filter((message) => message.role !== "system" && message.content.length > 0);
|
|
16
|
+
}
|
|
17
|
+
function buildSummaryInstruction(friendId, channel, trustLevel) {
|
|
18
|
+
if (friendId === "self" && channel === "inner") {
|
|
19
|
+
return "summarize this session transcript fully and transparently. this is my own inner dialog — include all details, decisions, and reasoning.";
|
|
20
|
+
}
|
|
21
|
+
return `summarize this session transcript. the person asking has trust level: ${trustLevel}. family=full transparency, friend=share work and general topics but protect other people's identities, acquaintance=very guarded minimal disclosure.`;
|
|
22
|
+
}
|
|
23
|
+
function clip(text, limit = 160) {
|
|
24
|
+
const compact = text.replace(/\s+/g, " ").trim();
|
|
25
|
+
return compact.length > limit ? compact.slice(0, limit - 1) + "…" : compact;
|
|
26
|
+
}
|
|
27
|
+
function buildSnapshot(summary, tailMessages) {
|
|
28
|
+
const lines = [`recent focus: ${clip(summary, 240)}`];
|
|
29
|
+
const latestUser = [...tailMessages].reverse().find((message) => message.role === "user")?.content;
|
|
30
|
+
const latestAssistant = [...tailMessages].reverse().find((message) => message.role === "assistant")?.content;
|
|
31
|
+
if (latestUser) {
|
|
32
|
+
lines.push(`latest user: ${clip(latestUser)}`);
|
|
33
|
+
}
|
|
34
|
+
if (latestAssistant) {
|
|
35
|
+
lines.push(`latest assistant: ${clip(latestAssistant)}`);
|
|
36
|
+
}
|
|
37
|
+
return lines.join("\n");
|
|
38
|
+
}
|
|
39
|
+
function buildSearchSnapshot(query, messages, includeLatestTurn = true) {
|
|
40
|
+
const lines = [`history query: "${clip(query, 120)}"`];
|
|
41
|
+
if (!includeLatestTurn) {
|
|
42
|
+
return lines.join("\n");
|
|
43
|
+
}
|
|
44
|
+
const latestUser = [...messages].reverse().find((message) => message.role === "user")?.content;
|
|
45
|
+
const latestAssistant = [...messages].reverse().find((message) => message.role === "assistant")?.content;
|
|
46
|
+
if (latestUser) {
|
|
47
|
+
lines.push(`latest user: ${clip(latestUser)}`);
|
|
48
|
+
}
|
|
49
|
+
if (latestAssistant) {
|
|
50
|
+
lines.push(`latest assistant: ${clip(latestAssistant)}`);
|
|
51
|
+
}
|
|
52
|
+
return lines.join("\n");
|
|
53
|
+
}
|
|
54
|
+
function buildSearchExcerpts(messages, query, maxMatches) {
|
|
55
|
+
const normalizedQuery = query.trim().toLowerCase();
|
|
56
|
+
if (!normalizedQuery)
|
|
57
|
+
return [];
|
|
58
|
+
const candidates = [];
|
|
59
|
+
let lastMatchIndex = -2;
|
|
60
|
+
for (let i = 0; i < messages.length; i++) {
|
|
61
|
+
if (!messages[i].content.toLowerCase().includes(normalizedQuery))
|
|
62
|
+
continue;
|
|
63
|
+
if (i <= lastMatchIndex + 1)
|
|
64
|
+
continue;
|
|
65
|
+
lastMatchIndex = i;
|
|
66
|
+
const start = Math.max(0, i - 1);
|
|
67
|
+
const end = Math.min(messages.length, i + 2);
|
|
68
|
+
const excerpt = messages
|
|
69
|
+
.slice(start, end)
|
|
70
|
+
.map((message) => `[${message.timestamp} | ${message.role} | ${message.id}] ${clip(message.content, 200)}`)
|
|
71
|
+
.join("\n");
|
|
72
|
+
const signature = messages
|
|
73
|
+
.slice(start, end)
|
|
74
|
+
.map((message) => `[${message.role}] ${clip(message.content, 200)}`)
|
|
75
|
+
.join("\n");
|
|
76
|
+
const score = messages
|
|
77
|
+
.slice(start, end)
|
|
78
|
+
.filter((message) => message.content.toLowerCase().includes(normalizedQuery))
|
|
79
|
+
.length;
|
|
80
|
+
candidates.push({ excerpt, signature, score, index: i });
|
|
81
|
+
}
|
|
82
|
+
const seen = new Set();
|
|
83
|
+
return candidates
|
|
84
|
+
.sort((a, b) => b.score - a.score || a.index - b.index)
|
|
85
|
+
.filter((candidate) => {
|
|
86
|
+
if (seen.has(candidate.signature))
|
|
87
|
+
return false;
|
|
88
|
+
seen.add(candidate.signature);
|
|
89
|
+
return true;
|
|
90
|
+
})
|
|
91
|
+
.slice(0, maxMatches)
|
|
92
|
+
.map((candidate) => candidate.excerpt);
|
|
93
|
+
}
|
|
94
|
+
async function recallSession(options) {
|
|
95
|
+
(0, runtime_1.emitNervesEvent)({
|
|
96
|
+
component: "daemon",
|
|
97
|
+
event: "daemon.session_recall",
|
|
98
|
+
message: "recalling session transcript tail",
|
|
99
|
+
meta: {
|
|
100
|
+
friendId: options.friendId,
|
|
101
|
+
channel: options.channel,
|
|
102
|
+
key: options.key,
|
|
103
|
+
messageCount: options.messageCount,
|
|
104
|
+
},
|
|
105
|
+
});
|
|
106
|
+
const envelope = (0, session_events_1.loadSessionEnvelopeFile)(options.sessionPath);
|
|
107
|
+
if (!envelope)
|
|
108
|
+
return { kind: "missing" };
|
|
109
|
+
const tailMessages = normalizeSessionMessages(envelope.events).slice(-options.messageCount);
|
|
110
|
+
if (tailMessages.length === 0) {
|
|
111
|
+
return { kind: "empty" };
|
|
112
|
+
}
|
|
113
|
+
const transcript = tailMessages
|
|
114
|
+
.map((message) => `[${message.timestamp} | ${message.role} | ${message.id}] ${message.content}`)
|
|
115
|
+
.join("\n");
|
|
116
|
+
const summary = options.summarize
|
|
117
|
+
? await options.summarize(transcript, buildSummaryInstruction(options.friendId, options.channel, options.trustLevel ?? "family"))
|
|
118
|
+
: transcript;
|
|
119
|
+
return {
|
|
120
|
+
kind: "ok",
|
|
121
|
+
transcript,
|
|
122
|
+
summary,
|
|
123
|
+
snapshot: buildSnapshot(summary, tailMessages),
|
|
124
|
+
tailMessages,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
async function searchSessionTranscript(options) {
|
|
128
|
+
(0, runtime_1.emitNervesEvent)({
|
|
129
|
+
component: "daemon",
|
|
130
|
+
event: "daemon.session_search",
|
|
131
|
+
message: "searching session transcript",
|
|
132
|
+
meta: {
|
|
133
|
+
friendId: options.friendId,
|
|
134
|
+
channel: options.channel,
|
|
135
|
+
key: options.key,
|
|
136
|
+
query: options.query,
|
|
137
|
+
maxMatches: options.maxMatches ?? 5,
|
|
138
|
+
},
|
|
139
|
+
});
|
|
140
|
+
const envelope = (0, session_events_1.loadSessionEnvelopeFile)(options.sessionPath);
|
|
141
|
+
if (!envelope)
|
|
142
|
+
return { kind: "missing" };
|
|
143
|
+
const messages = normalizeSessionMessages(envelope.events);
|
|
144
|
+
if (messages.length === 0) {
|
|
145
|
+
return { kind: "empty" };
|
|
146
|
+
}
|
|
147
|
+
const query = options.query.trim();
|
|
148
|
+
const matches = buildSearchExcerpts(messages, query, options.maxMatches ?? 5);
|
|
149
|
+
if (matches.length === 0) {
|
|
150
|
+
return {
|
|
151
|
+
kind: "no_match",
|
|
152
|
+
query,
|
|
153
|
+
snapshot: buildSearchSnapshot(query, messages),
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
return {
|
|
157
|
+
kind: "ok",
|
|
158
|
+
query,
|
|
159
|
+
snapshot: buildSearchSnapshot(query, messages, false),
|
|
160
|
+
matches,
|
|
161
|
+
};
|
|
162
|
+
}
|
|
@@ -0,0 +1,341 @@
|
|
|
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.buildCapabilitiesSection = buildCapabilitiesSection;
|
|
37
|
+
exports.buildStartOfTurnPacket = buildStartOfTurnPacket;
|
|
38
|
+
exports.renderStartOfTurnPacket = renderStartOfTurnPacket;
|
|
39
|
+
exports.renderCompactStartOfTurnPacket = renderCompactStartOfTurnPacket;
|
|
40
|
+
const fs = __importStar(require("fs"));
|
|
41
|
+
const path = __importStar(require("path"));
|
|
42
|
+
const runtime_1 = require("../nerves/runtime");
|
|
43
|
+
const bundle_state_1 = require("./bundle-state");
|
|
44
|
+
const tempo_1 = require("./tempo");
|
|
45
|
+
function estimateTokens(text) {
|
|
46
|
+
return Math.ceil(text.length / 4);
|
|
47
|
+
}
|
|
48
|
+
// --- Section builders (pure derivation from authored data, no embellishment) ---
|
|
49
|
+
function buildPlotLine(episodes, tempo) {
|
|
50
|
+
if (episodes.length === 0)
|
|
51
|
+
return "";
|
|
52
|
+
const limit = tempo === "crisis" ? 3 : tempo === "brief" ? 2 : 5;
|
|
53
|
+
const selected = episodes.slice(0, limit);
|
|
54
|
+
const lines = selected.map((ep) => {
|
|
55
|
+
const parts = [`- ${ep.summary}`];
|
|
56
|
+
if (ep.whyItMattered && tempo !== "brief") {
|
|
57
|
+
parts[0] += ` (${ep.whyItMattered})`;
|
|
58
|
+
}
|
|
59
|
+
return parts[0];
|
|
60
|
+
});
|
|
61
|
+
return lines.join("\n");
|
|
62
|
+
}
|
|
63
|
+
function buildObligationsSection(obligations) {
|
|
64
|
+
if (obligations.length === 0)
|
|
65
|
+
return "";
|
|
66
|
+
return obligations
|
|
67
|
+
.map((ob) => {
|
|
68
|
+
const parts = [`- ${ob.content}`];
|
|
69
|
+
if (ob.meaning?.resumeHint) {
|
|
70
|
+
parts.push(` [hint: ${ob.meaning.resumeHint}]`);
|
|
71
|
+
}
|
|
72
|
+
if (ob.meaning?.stalenessClass && ob.meaning.stalenessClass !== "fresh") {
|
|
73
|
+
parts.push(` (${ob.meaning.stalenessClass})`);
|
|
74
|
+
}
|
|
75
|
+
if (ob.meaning?.waitingOn) {
|
|
76
|
+
parts.push(` waiting on ${ob.meaning.waitingOn.kind}: ${ob.meaning.waitingOn.target}`);
|
|
77
|
+
}
|
|
78
|
+
return parts.join("");
|
|
79
|
+
})
|
|
80
|
+
.join("\n");
|
|
81
|
+
}
|
|
82
|
+
function buildCaresSection(cares) {
|
|
83
|
+
if (cares.length === 0)
|
|
84
|
+
return "";
|
|
85
|
+
return cares
|
|
86
|
+
.map((c) => {
|
|
87
|
+
const parts = [`- ${c.label}`];
|
|
88
|
+
if (c.salience !== "low") {
|
|
89
|
+
parts.push(` [${c.salience}]`);
|
|
90
|
+
}
|
|
91
|
+
if (c.currentRisk) {
|
|
92
|
+
parts.push(` risk: ${c.currentRisk}`);
|
|
93
|
+
}
|
|
94
|
+
return parts.join("");
|
|
95
|
+
})
|
|
96
|
+
.join("\n");
|
|
97
|
+
}
|
|
98
|
+
function buildPresenceSection(peers) {
|
|
99
|
+
if (peers.length === 0)
|
|
100
|
+
return "";
|
|
101
|
+
return peers
|
|
102
|
+
.map((p) => `- ${p.agentName}: ${p.availability}/${p.lane}`)
|
|
103
|
+
.join("\n");
|
|
104
|
+
}
|
|
105
|
+
function buildResumeHint(view, obligations) {
|
|
106
|
+
// Compose from authored obligation resumeHints and top intentions
|
|
107
|
+
const hints = [];
|
|
108
|
+
const effectiveObligations = obligations ?? view.activeObligations;
|
|
109
|
+
for (const ob of effectiveObligations) {
|
|
110
|
+
if (ob.meaning?.resumeHint) {
|
|
111
|
+
hints.push(ob.meaning.resumeHint);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
for (const intent of view.openIntentions.slice(0, 3)) {
|
|
115
|
+
hints.push(intent.content);
|
|
116
|
+
}
|
|
117
|
+
if (hints.length === 0) {
|
|
118
|
+
// Fall back to top obligation content
|
|
119
|
+
if (effectiveObligations.length > 0) {
|
|
120
|
+
hints.push(effectiveObligations[0].content);
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
if (hints.length === 0)
|
|
124
|
+
return "";
|
|
125
|
+
return hints.join("; ");
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Build a compact summary of capability changes between runtime versions.
|
|
129
|
+
* Reads bundle-meta.json for version info and changelog.json for change descriptions.
|
|
130
|
+
* Returns undefined if no version change detected or files are missing/malformed.
|
|
131
|
+
*/
|
|
132
|
+
function buildCapabilitiesSection(bundleRoot) {
|
|
133
|
+
let meta;
|
|
134
|
+
try {
|
|
135
|
+
const raw = fs.readFileSync(path.join(bundleRoot, "bundle-meta.json"), "utf-8");
|
|
136
|
+
meta = JSON.parse(raw);
|
|
137
|
+
}
|
|
138
|
+
catch {
|
|
139
|
+
(0, runtime_1.emitNervesEvent)({
|
|
140
|
+
component: "heart",
|
|
141
|
+
event: "heart.capabilities_check",
|
|
142
|
+
message: "bundle-meta.json missing or malformed",
|
|
143
|
+
meta: { bundleRoot },
|
|
144
|
+
});
|
|
145
|
+
return undefined;
|
|
146
|
+
}
|
|
147
|
+
const currentVersion = meta.runtimeVersion;
|
|
148
|
+
const previousVersion = meta.previousRuntimeVersion;
|
|
149
|
+
if (typeof currentVersion !== "string" || typeof previousVersion !== "string") {
|
|
150
|
+
return undefined;
|
|
151
|
+
}
|
|
152
|
+
if (currentVersion === previousVersion) {
|
|
153
|
+
return undefined;
|
|
154
|
+
}
|
|
155
|
+
(0, runtime_1.emitNervesEvent)({
|
|
156
|
+
component: "heart",
|
|
157
|
+
event: "heart.capabilities_version_change",
|
|
158
|
+
message: `runtime version changed from ${previousVersion} to ${currentVersion}`,
|
|
159
|
+
meta: { previousVersion, currentVersion },
|
|
160
|
+
});
|
|
161
|
+
let changelog;
|
|
162
|
+
try {
|
|
163
|
+
const raw = fs.readFileSync(path.join(bundleRoot, "changelog.json"), "utf-8");
|
|
164
|
+
changelog = JSON.parse(raw);
|
|
165
|
+
}
|
|
166
|
+
catch {
|
|
167
|
+
// changelog missing -- still report the version change
|
|
168
|
+
}
|
|
169
|
+
const parts = [`Updated from ${previousVersion} to ${currentVersion}`];
|
|
170
|
+
if (changelog && Array.isArray(changelog)) {
|
|
171
|
+
const relevantChanges = [];
|
|
172
|
+
for (const entry of changelog) {
|
|
173
|
+
if (entry.version === previousVersion)
|
|
174
|
+
break;
|
|
175
|
+
if (Array.isArray(entry.changes)) {
|
|
176
|
+
relevantChanges.push(...entry.changes);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
if (relevantChanges.length > 0) {
|
|
180
|
+
parts.push(`: ${relevantChanges.join("; ")}`);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
return parts.join("");
|
|
184
|
+
}
|
|
185
|
+
function buildStartOfTurnPacket(view, opts) {
|
|
186
|
+
const tempo = view.tempo;
|
|
187
|
+
const tokenBudget = tempo_1.TEMPO_BUDGETS[tempo];
|
|
188
|
+
const effectiveObligations = opts?.canonicalObligations ? opts.canonicalObligations.all : view.activeObligations;
|
|
189
|
+
const packet = {
|
|
190
|
+
plotLine: buildPlotLine(view.recentEpisodes, tempo),
|
|
191
|
+
obligations: buildObligationsSection(effectiveObligations),
|
|
192
|
+
cares: buildCaresSection(view.activeCares),
|
|
193
|
+
presence: buildPresenceSection(view.peerPresence),
|
|
194
|
+
resumeHint: buildResumeHint(view, opts?.canonicalObligations ? effectiveObligations : undefined),
|
|
195
|
+
currentSessionTiming: opts?.currentSessionTiming,
|
|
196
|
+
tempo,
|
|
197
|
+
tokenBudget,
|
|
198
|
+
assembledAt: new Date().toISOString(),
|
|
199
|
+
};
|
|
200
|
+
(0, runtime_1.emitNervesEvent)({
|
|
201
|
+
component: "heart",
|
|
202
|
+
event: "heart.start_of_turn_packet_built",
|
|
203
|
+
message: `start-of-turn packet built: tempo=${tempo}`,
|
|
204
|
+
meta: {
|
|
205
|
+
tempo,
|
|
206
|
+
plotLineTokens: estimateTokens(packet.plotLine),
|
|
207
|
+
obligationsTokens: estimateTokens(packet.obligations),
|
|
208
|
+
caresTokens: estimateTokens(packet.cares),
|
|
209
|
+
presenceTokens: estimateTokens(packet.presence),
|
|
210
|
+
resumeHintTokens: estimateTokens(packet.resumeHint),
|
|
211
|
+
},
|
|
212
|
+
});
|
|
213
|
+
return packet;
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* Renders a start-of-turn packet to prompt text, respecting token budget.
|
|
217
|
+
* Truncation priority (last truncated first):
|
|
218
|
+
* resumeHint (PROTECTED) > obligations > cares > plotLine > presence
|
|
219
|
+
* So presence is truncated first, then plotLine, then cares, then obligations.
|
|
220
|
+
* resumeHint is never truncated.
|
|
221
|
+
*/
|
|
222
|
+
function renderStartOfTurnPacket(packet) {
|
|
223
|
+
const budget = packet.tokenBudget;
|
|
224
|
+
// Assemble sections in priority order (highest priority first for budget allocation)
|
|
225
|
+
// Each section is { label, content, priority } where lower priority number = truncated first
|
|
226
|
+
const sections = [
|
|
227
|
+
// bundleState and syncFailure share the top remediation tier — both
|
|
228
|
+
// are actionable "fix your git" signals. bundleState is preferred
|
|
229
|
+
// because it's structured (array of enum values) while syncFailure is
|
|
230
|
+
// a legacy free-form string; both render when populated.
|
|
231
|
+
{ label: "bundleState", content: (0, bundle_state_1.renderBundleStateHint)(packet.bundleState ?? []), priority: 7 },
|
|
232
|
+
{ label: "syncFailure", content: packet.syncFailure ?? "", priority: 7 },
|
|
233
|
+
{ label: "resume", content: packet.resumeHint, priority: 6 },
|
|
234
|
+
{ label: "sessionTiming", content: packet.currentSessionTiming ?? "", priority: 5 },
|
|
235
|
+
{ label: "obligations", content: packet.obligations, priority: 5 },
|
|
236
|
+
{ label: "cares", content: packet.cares, priority: 4 },
|
|
237
|
+
{ label: "plot", content: packet.plotLine, priority: 3 },
|
|
238
|
+
{ label: "capabilities", content: packet.capabilities ?? "", priority: 2 },
|
|
239
|
+
{ label: "presence", content: packet.presence, priority: 1 },
|
|
240
|
+
].filter((s) => s.content.length > 0);
|
|
241
|
+
if (sections.length === 0) {
|
|
242
|
+
(0, runtime_1.emitNervesEvent)({
|
|
243
|
+
component: "heart",
|
|
244
|
+
event: "heart.start_of_turn_packet_rendered",
|
|
245
|
+
message: "start-of-turn packet rendered: empty",
|
|
246
|
+
meta: { tokens: 0, tempo: packet.tempo },
|
|
247
|
+
});
|
|
248
|
+
return "";
|
|
249
|
+
}
|
|
250
|
+
// Build the rendered output, truncating lower-priority sections first
|
|
251
|
+
let rendered = formatSections(sections);
|
|
252
|
+
let tokens = estimateTokens(rendered);
|
|
253
|
+
// Truncate sections from lowest priority until we fit budget
|
|
254
|
+
const sortedByPriority = [...sections].sort((a, b) => a.priority - b.priority);
|
|
255
|
+
for (const section of sortedByPriority) {
|
|
256
|
+
if (tokens <= budget.max)
|
|
257
|
+
break;
|
|
258
|
+
// Skip resumeHint — it's PROTECTED
|
|
259
|
+
if (section.label === "resume")
|
|
260
|
+
continue;
|
|
261
|
+
// Remove this section entirely
|
|
262
|
+
const idx = sections.findIndex((s) => s.label === section.label);
|
|
263
|
+
sections.splice(idx, 1);
|
|
264
|
+
rendered = formatSections(sections);
|
|
265
|
+
tokens = estimateTokens(rendered);
|
|
266
|
+
}
|
|
267
|
+
// If still over budget after removing all non-protected sections, trim what's left
|
|
268
|
+
if (tokens > budget.max) {
|
|
269
|
+
const maxChars = budget.max * 4;
|
|
270
|
+
rendered = rendered.slice(0, maxChars);
|
|
271
|
+
}
|
|
272
|
+
(0, runtime_1.emitNervesEvent)({
|
|
273
|
+
component: "heart",
|
|
274
|
+
event: "heart.start_of_turn_packet_rendered",
|
|
275
|
+
message: `start-of-turn packet rendered: ${tokens} tokens`,
|
|
276
|
+
meta: { tokens, tempo: packet.tempo, sectionCount: sections.length },
|
|
277
|
+
});
|
|
278
|
+
return rendered;
|
|
279
|
+
}
|
|
280
|
+
function formatSections(sections) {
|
|
281
|
+
const parts = [];
|
|
282
|
+
for (const section of sections) {
|
|
283
|
+
switch (section.label) {
|
|
284
|
+
case "resume":
|
|
285
|
+
parts.push(`**Next:** ${section.content}`);
|
|
286
|
+
break;
|
|
287
|
+
case "obligations":
|
|
288
|
+
parts.push(`**Owed:**\n${section.content}`);
|
|
289
|
+
break;
|
|
290
|
+
case "cares":
|
|
291
|
+
parts.push(`**Cares:**\n${section.content}`);
|
|
292
|
+
break;
|
|
293
|
+
case "plot":
|
|
294
|
+
parts.push(`**Recent:**\n${section.content}`);
|
|
295
|
+
break;
|
|
296
|
+
case "presence":
|
|
297
|
+
parts.push(`**Peers:**\n${section.content}`);
|
|
298
|
+
break;
|
|
299
|
+
case "capabilities":
|
|
300
|
+
parts.push(`**Capabilities:** ${section.content}`);
|
|
301
|
+
break;
|
|
302
|
+
case "syncFailure":
|
|
303
|
+
parts.push(`**Sync warning:** ${section.content}`);
|
|
304
|
+
break;
|
|
305
|
+
case "bundleState":
|
|
306
|
+
parts.push(`**Bundle:** ${section.content}`);
|
|
307
|
+
break;
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
return parts.join("\n\n");
|
|
311
|
+
}
|
|
312
|
+
/**
|
|
313
|
+
* Ultra-compact version for coding context (max 200 tokens).
|
|
314
|
+
* Just resumeHint + top obligation + top care, single-line bullets.
|
|
315
|
+
*/
|
|
316
|
+
function renderCompactStartOfTurnPacket(packet) {
|
|
317
|
+
const parts = [];
|
|
318
|
+
if (packet.resumeHint) {
|
|
319
|
+
parts.push(`next: ${packet.resumeHint}`);
|
|
320
|
+
}
|
|
321
|
+
if (packet.obligations) {
|
|
322
|
+
// Just first line of obligations
|
|
323
|
+
const firstOb = packet.obligations.split("\n")[0];
|
|
324
|
+
parts.push(`owed: ${firstOb.replace(/^- /, "")}`);
|
|
325
|
+
}
|
|
326
|
+
if (packet.cares) {
|
|
327
|
+
const firstCare = packet.cares.split("\n")[0];
|
|
328
|
+
parts.push(`care: ${firstCare.replace(/^- /, "")}`);
|
|
329
|
+
}
|
|
330
|
+
const compact = parts.join(" | ");
|
|
331
|
+
// Hard cap at 200 tokens (800 chars)
|
|
332
|
+
const maxChars = 200 * 4;
|
|
333
|
+
const result = compact.length > maxChars ? compact.slice(0, maxChars) : compact;
|
|
334
|
+
(0, runtime_1.emitNervesEvent)({
|
|
335
|
+
component: "heart",
|
|
336
|
+
event: "heart.start_of_turn_packet_compact_rendered",
|
|
337
|
+
message: `compact start-of-turn packet: ${estimateTokens(result)} tokens`,
|
|
338
|
+
meta: { tokens: estimateTokens(result) },
|
|
339
|
+
});
|
|
340
|
+
return result;
|
|
341
|
+
}
|
package/dist/heart/streaming.js
CHANGED
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
3
|
+
exports.SettleStreamer = exports.SettleParser = void 0;
|
|
4
4
|
exports.toResponsesInput = toResponsesInput;
|
|
5
5
|
exports.toResponsesTools = toResponsesTools;
|
|
6
6
|
exports.streamChatCompletion = streamChatCompletion;
|
|
7
7
|
exports.streamResponsesApi = streamResponsesApi;
|
|
8
8
|
const runtime_1 = require("../nerves/runtime");
|
|
9
9
|
// Character-level state machine that extracts the answer value from
|
|
10
|
-
// `
|
|
10
|
+
// `settle` tool call JSON arguments as they stream in.
|
|
11
11
|
// Scans for prefix `"answer":"` or `"answer": "` in the character stream,
|
|
12
12
|
// then emits text handling JSON escapes, stopping at unescaped closing `"`.
|
|
13
|
-
class
|
|
13
|
+
class SettleParser {
|
|
14
14
|
// Possible prefixes to match (with and without space after colon)
|
|
15
15
|
static PREFIXES = ['"answer":"', '"answer": "'];
|
|
16
16
|
// Buffer of characters seen so far (pre-activation only)
|
|
@@ -29,7 +29,7 @@ class FinalAnswerParser {
|
|
|
29
29
|
if (!this._active) {
|
|
30
30
|
this.buf += ch;
|
|
31
31
|
// Check if any prefix has been fully matched in the buffer
|
|
32
|
-
for (const prefix of
|
|
32
|
+
for (const prefix of SettleParser.PREFIXES) {
|
|
33
33
|
if (this.buf.endsWith(prefix)) {
|
|
34
34
|
this._active = true;
|
|
35
35
|
break;
|
|
@@ -76,21 +76,25 @@ class FinalAnswerParser {
|
|
|
76
76
|
return out;
|
|
77
77
|
}
|
|
78
78
|
}
|
|
79
|
-
exports.
|
|
80
|
-
// Shared helper: wraps
|
|
79
|
+
exports.SettleParser = SettleParser;
|
|
80
|
+
// Shared helper: wraps SettleParser with onClearText + onTextChunk wiring.
|
|
81
81
|
// Used by all streaming providers (Chat Completions, Responses API, Anthropic)
|
|
82
|
-
// so the eager-match streaming pattern lives in one place.
|
|
83
|
-
class
|
|
84
|
-
parser = new
|
|
82
|
+
// so the eager-match settle streaming pattern lives in one place.
|
|
83
|
+
class SettleStreamer {
|
|
84
|
+
parser = new SettleParser();
|
|
85
85
|
_detected = false;
|
|
86
86
|
callbacks;
|
|
87
|
-
|
|
87
|
+
enabled;
|
|
88
|
+
constructor(callbacks, enabled = true) {
|
|
88
89
|
this.callbacks = callbacks;
|
|
90
|
+
this.enabled = enabled;
|
|
89
91
|
}
|
|
90
92
|
get detected() { return this._detected; }
|
|
91
93
|
get streamed() { return this.parser.active; }
|
|
92
|
-
/** Mark
|
|
94
|
+
/** Mark settle as detected. Calls onClearText on the callbacks. */
|
|
93
95
|
activate() {
|
|
96
|
+
if (!this.enabled)
|
|
97
|
+
return;
|
|
94
98
|
if (this._detected)
|
|
95
99
|
return;
|
|
96
100
|
this._detected = true;
|
|
@@ -98,6 +102,8 @@ class FinalAnswerStreamer {
|
|
|
98
102
|
}
|
|
99
103
|
/** Feed an argument delta through the parser. Emits text via onTextChunk. */
|
|
100
104
|
processDelta(delta) {
|
|
105
|
+
if (!this.enabled)
|
|
106
|
+
return;
|
|
101
107
|
if (!this._detected)
|
|
102
108
|
return;
|
|
103
109
|
const text = this.parser.process(delta);
|
|
@@ -105,7 +111,7 @@ class FinalAnswerStreamer {
|
|
|
105
111
|
this.callbacks.onTextChunk(text);
|
|
106
112
|
}
|
|
107
113
|
}
|
|
108
|
-
exports.
|
|
114
|
+
exports.SettleStreamer = SettleStreamer;
|
|
109
115
|
function toResponsesUserContent(content) {
|
|
110
116
|
if (typeof content === "string") {
|
|
111
117
|
return content;
|
|
@@ -185,7 +191,10 @@ function toResponsesInput(messages) {
|
|
|
185
191
|
}
|
|
186
192
|
}
|
|
187
193
|
if (a.content) {
|
|
188
|
-
|
|
194
|
+
const assistantItem = { role: "assistant", content: typeof a.content === "string" ? a.content : "" };
|
|
195
|
+
if (a.phase)
|
|
196
|
+
assistantItem.phase = a.phase;
|
|
197
|
+
input.push(assistantItem);
|
|
189
198
|
}
|
|
190
199
|
if (a.tool_calls) {
|
|
191
200
|
for (const tc of a.tool_calls) {
|
|
@@ -224,7 +233,7 @@ function toResponsesTools(ccTools) {
|
|
|
224
233
|
strict: false,
|
|
225
234
|
}));
|
|
226
235
|
}
|
|
227
|
-
async function streamChatCompletion(client, createParams, callbacks, signal) {
|
|
236
|
+
async function streamChatCompletion(client, createParams, callbacks, signal, eagerSettleStreaming = true) {
|
|
228
237
|
(0, runtime_1.emitNervesEvent)({
|
|
229
238
|
component: "engine",
|
|
230
239
|
event: "engine.stream_start",
|
|
@@ -238,7 +247,7 @@ async function streamChatCompletion(client, createParams, callbacks, signal) {
|
|
|
238
247
|
let toolCalls = {};
|
|
239
248
|
let streamStarted = false;
|
|
240
249
|
let usage;
|
|
241
|
-
const answerStreamer = new
|
|
250
|
+
const answerStreamer = new SettleStreamer(callbacks, eagerSettleStreaming);
|
|
242
251
|
// State machine for parsing inline <think> tags (MiniMax pattern)
|
|
243
252
|
let contentBuf = "";
|
|
244
253
|
let inThinkTag = false;
|
|
@@ -353,19 +362,19 @@ async function streamChatCompletion(client, createParams, callbacks, signal) {
|
|
|
353
362
|
toolCalls[tc.index].id = tc.id;
|
|
354
363
|
if (tc.function?.name) {
|
|
355
364
|
toolCalls[tc.index].name = tc.function.name;
|
|
356
|
-
// Detect
|
|
365
|
+
// Detect settle tool call on first name delta.
|
|
357
366
|
// Only activate streaming if this is the sole tool call (index 0
|
|
358
367
|
// and no other indices seen). Mixed calls are rejected by core.ts.
|
|
359
|
-
if (tc.function.name === "
|
|
368
|
+
if (tc.function.name === "settle" && !answerStreamer.detected
|
|
360
369
|
&& tc.index === 0 && Object.keys(toolCalls).length === 1) {
|
|
361
370
|
answerStreamer.activate();
|
|
362
371
|
}
|
|
363
372
|
}
|
|
364
373
|
if (tc.function?.arguments) {
|
|
365
374
|
toolCalls[tc.index].arguments += tc.function.arguments;
|
|
366
|
-
// Feed
|
|
375
|
+
// Feed settle argument deltas to the parser for progressive
|
|
367
376
|
// streaming, but only when it appears to be the sole tool call.
|
|
368
|
-
if (answerStreamer.detected && toolCalls[tc.index].name === "
|
|
377
|
+
if (answerStreamer.detected && toolCalls[tc.index].name === "settle"
|
|
369
378
|
&& Object.keys(toolCalls).length === 1) {
|
|
370
379
|
answerStreamer.processDelta(tc.function.arguments);
|
|
371
380
|
}
|
|
@@ -381,10 +390,10 @@ async function streamChatCompletion(client, createParams, callbacks, signal) {
|
|
|
381
390
|
toolCalls: Object.values(toolCalls),
|
|
382
391
|
outputItems: [],
|
|
383
392
|
usage,
|
|
384
|
-
|
|
393
|
+
settleStreamed: answerStreamer.streamed,
|
|
385
394
|
};
|
|
386
395
|
}
|
|
387
|
-
async function streamResponsesApi(client, createParams, callbacks, signal) {
|
|
396
|
+
async function streamResponsesApi(client, createParams, callbacks, signal, eagerSettleStreaming = true) {
|
|
388
397
|
(0, runtime_1.emitNervesEvent)({
|
|
389
398
|
component: "engine",
|
|
390
399
|
event: "engine.stream_start",
|
|
@@ -399,7 +408,7 @@ async function streamResponsesApi(client, createParams, callbacks, signal) {
|
|
|
399
408
|
const outputItems = [];
|
|
400
409
|
let currentToolCall = null;
|
|
401
410
|
let usage;
|
|
402
|
-
const answerStreamer = new
|
|
411
|
+
const answerStreamer = new SettleStreamer(callbacks, eagerSettleStreaming);
|
|
403
412
|
let functionCallCount = 0;
|
|
404
413
|
for await (const event of response) {
|
|
405
414
|
if (signal?.aborted)
|
|
@@ -429,10 +438,10 @@ async function streamResponsesApi(client, createParams, callbacks, signal) {
|
|
|
429
438
|
name: String(event.item.name),
|
|
430
439
|
arguments: "",
|
|
431
440
|
};
|
|
432
|
-
// Detect
|
|
441
|
+
// Detect settle function call -- clear any streamed noise.
|
|
433
442
|
// Only activate when this is the first (and so far only) function call.
|
|
434
443
|
// Mixed calls are rejected by core.ts; no need to stream their args.
|
|
435
|
-
if (String(event.item.name) === "
|
|
444
|
+
if (String(event.item.name) === "settle" && functionCallCount === 1) {
|
|
436
445
|
answerStreamer.activate();
|
|
437
446
|
}
|
|
438
447
|
}
|
|
@@ -441,9 +450,9 @@ async function streamResponsesApi(client, createParams, callbacks, signal) {
|
|
|
441
450
|
case "response.function_call_arguments.delta": {
|
|
442
451
|
if (currentToolCall) {
|
|
443
452
|
currentToolCall.arguments += event.delta;
|
|
444
|
-
// Feed
|
|
453
|
+
// Feed settle argument deltas to the parser for progressive
|
|
445
454
|
// streaming, but only when it appears to be the sole function call.
|
|
446
|
-
if (answerStreamer.detected && currentToolCall.name === "
|
|
455
|
+
if (answerStreamer.detected && currentToolCall.name === "settle"
|
|
447
456
|
&& functionCallCount === 1) {
|
|
448
457
|
answerStreamer.processDelta(String(event.delta));
|
|
449
458
|
}
|
|
@@ -485,6 +494,6 @@ async function streamResponsesApi(client, createParams, callbacks, signal) {
|
|
|
485
494
|
toolCalls,
|
|
486
495
|
outputItems,
|
|
487
496
|
usage,
|
|
488
|
-
|
|
497
|
+
settleStreamed: answerStreamer.streamed,
|
|
489
498
|
};
|
|
490
499
|
}
|