@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.
Files changed (314) hide show
  1. package/README.md +188 -187
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
  4. package/changelog.json +2031 -0
  5. package/dist/arc/attention-types.js +8 -0
  6. package/dist/arc/cares.js +140 -0
  7. package/dist/arc/episodes.js +117 -0
  8. package/dist/arc/intentions.js +133 -0
  9. package/dist/arc/json-store.js +117 -0
  10. package/dist/arc/obligations.js +237 -0
  11. package/dist/arc/packets.js +193 -0
  12. package/dist/arc/presence.js +185 -0
  13. package/dist/arc/task-lifecycle.js +65 -0
  14. package/dist/heart/active-work.js +832 -0
  15. package/dist/heart/agent-entry.js +37 -2
  16. package/dist/heart/attachments/image-normalize.js +194 -0
  17. package/dist/heart/attachments/materialize.js +97 -0
  18. package/dist/heart/attachments/originals.js +88 -0
  19. package/dist/heart/attachments/render.js +29 -0
  20. package/dist/heart/attachments/sources/adapter.js +2 -0
  21. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  22. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  23. package/dist/heart/attachments/sources/index.js +16 -0
  24. package/dist/heart/attachments/store.js +103 -0
  25. package/dist/heart/attachments/types.js +93 -0
  26. package/dist/heart/auth/auth-flow.js +463 -0
  27. package/dist/heart/bridges/manager.js +358 -0
  28. package/dist/heart/bridges/state-machine.js +135 -0
  29. package/dist/heart/bridges/store.js +123 -0
  30. package/dist/heart/bundle-state.js +168 -0
  31. package/dist/heart/commitments.js +111 -0
  32. package/dist/heart/config-registry.js +304 -0
  33. package/dist/heart/config.js +53 -21
  34. package/dist/heart/core.js +695 -195
  35. package/dist/heart/cross-chat-delivery.js +131 -0
  36. package/dist/heart/daemon/agent-config-check.js +292 -0
  37. package/dist/heart/daemon/agent-discovery.js +79 -3
  38. package/dist/heart/daemon/agent-service.js +360 -0
  39. package/dist/heart/daemon/agentic-repair.js +170 -0
  40. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  41. package/dist/heart/daemon/cadence.js +70 -0
  42. package/dist/heart/daemon/cli-defaults.js +591 -0
  43. package/dist/heart/daemon/cli-exec.js +2297 -0
  44. package/dist/heart/daemon/cli-help.js +306 -0
  45. package/dist/heart/daemon/cli-parse.js +824 -0
  46. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  47. package/dist/heart/daemon/cli-render.js +512 -0
  48. package/dist/heart/daemon/cli-types.js +8 -0
  49. package/dist/heart/daemon/daemon-cli.js +30 -1171
  50. package/dist/heart/daemon/daemon-entry.js +358 -3
  51. package/dist/heart/daemon/daemon-health.js +141 -0
  52. package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
  53. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  54. package/dist/heart/daemon/daemon.js +751 -58
  55. package/dist/heart/daemon/doctor-types.js +8 -0
  56. package/dist/heart/daemon/doctor.js +401 -0
  57. package/dist/heart/daemon/health-monitor.js +79 -1
  58. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  59. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  60. package/dist/heart/daemon/http-health-probe.js +80 -0
  61. package/dist/heart/daemon/inner-status.js +89 -0
  62. package/dist/heart/daemon/interactive-repair.js +91 -0
  63. package/dist/heart/daemon/launchd.js +46 -9
  64. package/dist/heart/daemon/log-tailer.js +82 -12
  65. package/dist/heart/daemon/logs-prune.js +105 -0
  66. package/dist/heart/daemon/message-router.js +17 -8
  67. package/dist/heart/daemon/os-cron-deps.js +134 -0
  68. package/dist/heart/daemon/ouro-bot-entry.js +1 -1
  69. package/dist/heart/daemon/process-manager.js +201 -0
  70. package/dist/heart/daemon/provider-discovery.js +105 -0
  71. package/dist/heart/daemon/pulse.js +463 -0
  72. package/dist/heart/daemon/run-hooks.js +2 -0
  73. package/dist/heart/daemon/runtime-logging.js +67 -16
  74. package/dist/heart/daemon/runtime-metadata.js +101 -0
  75. package/dist/heart/daemon/runtime-mode.js +67 -0
  76. package/dist/heart/daemon/safe-mode.js +161 -0
  77. package/dist/heart/daemon/sense-manager.js +72 -3
  78. package/dist/heart/daemon/session-id-resolver.js +131 -0
  79. package/dist/heart/daemon/skill-management-installer.js +94 -0
  80. package/dist/heart/daemon/socket-client.js +307 -0
  81. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  82. package/dist/heart/daemon/startup-tui.js +237 -0
  83. package/dist/heart/daemon/task-scheduler.js +3 -25
  84. package/dist/heart/daemon/thoughts.js +510 -0
  85. package/dist/heart/daemon/up-progress.js +135 -0
  86. package/dist/heart/delegation.js +62 -0
  87. package/dist/heart/habits/habit-migration.js +181 -0
  88. package/dist/heart/habits/habit-parser.js +140 -0
  89. package/dist/heart/habits/habit-scheduler.js +371 -0
  90. package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -120
  91. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  92. package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
  93. package/dist/heart/{daemon → hatch}/specialist-tools.js +49 -3
  94. package/dist/heart/identity.js +154 -59
  95. package/dist/heart/kicks.js +2 -20
  96. package/dist/heart/mcp/mcp-server.js +653 -0
  97. package/dist/heart/migrate-config.js +127 -0
  98. package/dist/heart/model-capabilities.js +59 -0
  99. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  100. package/dist/heart/outlook/outlook-http-response.js +7 -0
  101. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  102. package/dist/heart/outlook/outlook-http-static.js +99 -0
  103. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  104. package/dist/heart/outlook/outlook-http.js +99 -0
  105. package/dist/heart/outlook/outlook-read.js +28 -0
  106. package/dist/heart/outlook/outlook-types.js +27 -0
  107. package/dist/heart/outlook/outlook-view.js +194 -0
  108. package/dist/heart/outlook/readers/agent-machine.js +355 -0
  109. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  110. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  111. package/dist/heart/outlook/readers/sessions.js +231 -0
  112. package/dist/heart/outlook/readers/shared.js +111 -0
  113. package/dist/heart/progress-story.js +42 -0
  114. package/dist/heart/provider-failover.js +135 -0
  115. package/dist/heart/provider-models.js +81 -0
  116. package/dist/heart/provider-ping.js +234 -0
  117. package/dist/heart/providers/anthropic-token.js +163 -0
  118. package/dist/heart/providers/anthropic.js +171 -50
  119. package/dist/heart/providers/azure.js +97 -11
  120. package/dist/heart/providers/error-classification.js +63 -0
  121. package/dist/heart/providers/github-copilot.js +135 -0
  122. package/dist/heart/providers/minimax-vlm.js +189 -0
  123. package/dist/heart/providers/minimax.js +23 -6
  124. package/dist/heart/providers/openai-codex.js +33 -23
  125. package/dist/heart/session-activity.js +190 -0
  126. package/dist/heart/session-events.js +726 -0
  127. package/dist/heart/session-recall.js +162 -0
  128. package/dist/heart/start-of-turn-packet.js +341 -0
  129. package/dist/heart/streaming.js +36 -27
  130. package/dist/heart/sync.js +332 -0
  131. package/dist/heart/target-resolution.js +127 -0
  132. package/dist/heart/tempo.js +93 -0
  133. package/dist/heart/temporal-view.js +41 -0
  134. package/dist/heart/tool-activity-callbacks.js +36 -0
  135. package/dist/heart/tool-description.js +135 -0
  136. package/dist/heart/tool-friction.js +55 -0
  137. package/dist/heart/tool-loop.js +200 -0
  138. package/dist/heart/turn-context.js +358 -0
  139. package/dist/heart/turn-coordinator.js +28 -0
  140. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  141. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  142. package/dist/heart/{daemon → versioning}/ouro-path-installer.js +78 -35
  143. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  144. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  145. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  146. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  147. package/dist/mind/associative-recall.js +137 -66
  148. package/dist/mind/bundle-manifest.js +7 -1
  149. package/dist/mind/context.js +89 -93
  150. package/dist/mind/diary-integrity.js +60 -0
  151. package/dist/mind/{memory.js → diary.js} +84 -96
  152. package/dist/mind/embedding-provider.js +60 -0
  153. package/dist/mind/file-state.js +179 -0
  154. package/dist/mind/first-impressions.js +14 -1
  155. package/dist/mind/friends/channel.js +56 -0
  156. package/dist/mind/friends/group-context.js +144 -0
  157. package/dist/mind/friends/resolver.js +37 -0
  158. package/dist/mind/friends/store-file.js +58 -3
  159. package/dist/mind/friends/trust-explanation.js +74 -0
  160. package/dist/mind/friends/types.js +8 -0
  161. package/dist/mind/journal-index.js +161 -0
  162. package/dist/mind/obligation-steering.js +221 -0
  163. package/dist/mind/pending.js +74 -7
  164. package/dist/mind/prompt.js +999 -111
  165. package/dist/mind/provenance-trust.js +26 -0
  166. package/dist/mind/scrutiny.js +173 -0
  167. package/dist/mind/token-estimate.js +8 -12
  168. package/dist/nerves/cli-logging.js +7 -1
  169. package/dist/nerves/coverage/audit.js +1 -1
  170. package/dist/nerves/coverage/file-completeness.js +83 -5
  171. package/dist/nerves/coverage/run-artifacts.js +1 -1
  172. package/dist/nerves/event-buffer.js +111 -0
  173. package/dist/nerves/index.js +224 -4
  174. package/dist/nerves/observation.js +20 -0
  175. package/dist/nerves/redact.js +79 -0
  176. package/dist/nerves/runtime.js +5 -1
  177. package/dist/outlook-ui/assets/index-DC7sZefn.js +61 -0
  178. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  179. package/dist/outlook-ui/index.html +15 -0
  180. package/dist/repertoire/ado-client.js +15 -56
  181. package/dist/repertoire/ado-semantic.js +11 -10
  182. package/dist/repertoire/api-client.js +97 -0
  183. package/dist/repertoire/bitwarden-store.js +319 -0
  184. package/dist/repertoire/bundle-templates.js +72 -0
  185. package/dist/repertoire/bw-installer.js +79 -0
  186. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  187. package/dist/repertoire/coding/context-pack.js +330 -0
  188. package/dist/repertoire/coding/feedback.js +197 -30
  189. package/dist/repertoire/coding/manager.js +158 -9
  190. package/dist/repertoire/coding/spawner.js +55 -9
  191. package/dist/repertoire/coding/tools.js +170 -7
  192. package/dist/repertoire/commerce-errors.js +109 -0
  193. package/dist/repertoire/commerce-self-test.js +156 -0
  194. package/dist/repertoire/credential-access.js +527 -0
  195. package/dist/repertoire/duffel-client.js +185 -0
  196. package/dist/repertoire/github-client.js +14 -55
  197. package/dist/repertoire/graph-client.js +11 -52
  198. package/dist/repertoire/guardrails.js +375 -0
  199. package/dist/repertoire/mcp-client.js +255 -0
  200. package/dist/repertoire/mcp-manager.js +305 -0
  201. package/dist/repertoire/mcp-tools.js +63 -0
  202. package/dist/repertoire/shell-sessions.js +133 -0
  203. package/dist/repertoire/skills.js +14 -23
  204. package/dist/repertoire/stripe-client.js +131 -0
  205. package/dist/repertoire/tasks/board.js +43 -5
  206. package/dist/repertoire/tasks/fix.js +182 -0
  207. package/dist/repertoire/tasks/index.js +28 -10
  208. package/dist/repertoire/tasks/lifecycle.js +2 -2
  209. package/dist/repertoire/tasks/parser.js +3 -2
  210. package/dist/repertoire/tasks/scanner.js +194 -37
  211. package/dist/repertoire/tasks/transitions.js +16 -79
  212. package/dist/repertoire/tool-results.js +29 -0
  213. package/dist/repertoire/tools-attachments.js +316 -0
  214. package/dist/repertoire/tools-base.js +45 -771
  215. package/dist/repertoire/tools-bluebubbles.js +1 -0
  216. package/dist/repertoire/tools-bridge.js +141 -0
  217. package/dist/repertoire/tools-bundle.js +984 -0
  218. package/dist/repertoire/tools-config.js +185 -0
  219. package/dist/repertoire/tools-continuity.js +248 -0
  220. package/dist/repertoire/tools-credential.js +182 -0
  221. package/dist/repertoire/tools-files.js +342 -0
  222. package/dist/repertoire/tools-flight.js +224 -0
  223. package/dist/repertoire/tools-flow.js +105 -0
  224. package/dist/repertoire/tools-github.js +1 -7
  225. package/dist/repertoire/tools-memory.js +376 -0
  226. package/dist/repertoire/tools-session.js +739 -0
  227. package/dist/repertoire/tools-shell.js +120 -0
  228. package/dist/repertoire/tools-stripe.js +180 -0
  229. package/dist/repertoire/tools-surface.js +243 -0
  230. package/dist/repertoire/tools-teams.js +12 -62
  231. package/dist/repertoire/tools-travel.js +125 -0
  232. package/dist/repertoire/tools-user-profile.js +144 -0
  233. package/dist/repertoire/tools-vault.js +110 -0
  234. package/dist/repertoire/tools.js +144 -138
  235. package/dist/repertoire/travel-api-client.js +360 -0
  236. package/dist/repertoire/user-profile.js +118 -0
  237. package/dist/repertoire/vault-setup.js +241 -0
  238. package/dist/scripts/claude-code-hook.js +41 -0
  239. package/dist/scripts/claude-code-stop-hook.js +47 -0
  240. package/dist/senses/attention-queue.js +116 -0
  241. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  242. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  243. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +225 -9
  244. package/dist/senses/bluebubbles/entry.js +13 -0
  245. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  246. package/dist/senses/bluebubbles/index.js +1590 -0
  247. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  248. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  249. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  250. package/dist/senses/bluebubbles/replay.js +129 -0
  251. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  252. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  253. package/dist/senses/cli/bracketed-paste.js +82 -0
  254. package/dist/senses/cli/image-paste.js +287 -0
  255. package/dist/senses/cli/image-ref-navigation.js +75 -0
  256. package/dist/senses/cli/ink-app.js +156 -0
  257. package/dist/senses/cli/inline-diff.js +64 -0
  258. package/dist/senses/cli/input-keys.js +174 -0
  259. package/dist/senses/cli/kill-ring.js +86 -0
  260. package/dist/senses/cli/message-list.js +51 -0
  261. package/dist/senses/cli/ouro-tui.js +605 -0
  262. package/dist/senses/cli/spinner-imperative.js +135 -0
  263. package/dist/senses/cli/spinner.js +101 -0
  264. package/dist/senses/cli/status-line.js +60 -0
  265. package/dist/senses/cli/streaming-markdown.js +526 -0
  266. package/dist/senses/cli/tool-display.js +83 -0
  267. package/dist/senses/cli/tool-render.js +85 -0
  268. package/dist/senses/cli/tui-store.js +240 -0
  269. package/dist/senses/cli/virtual-list.js +35 -0
  270. package/dist/senses/cli-entry.js +1 -1
  271. package/dist/senses/cli-layout.js +187 -0
  272. package/dist/senses/cli.js +595 -246
  273. package/dist/senses/commands.js +65 -1
  274. package/dist/senses/continuity.js +94 -0
  275. package/dist/senses/habit-turn-message.js +108 -0
  276. package/dist/senses/inner-dialog-worker.js +112 -19
  277. package/dist/senses/inner-dialog.js +633 -86
  278. package/dist/senses/pipeline.js +567 -0
  279. package/dist/senses/shared-turn.js +199 -0
  280. package/dist/senses/surface-tool.js +68 -0
  281. package/dist/senses/teams.js +665 -160
  282. package/dist/senses/trust-gate.js +112 -2
  283. package/package.json +29 -7
  284. package/skills/agent-commerce.md +106 -0
  285. package/skills/browser-navigation.md +110 -0
  286. package/skills/commerce-setup-guide.md +116 -0
  287. package/skills/commerce-setup.md +84 -0
  288. package/skills/configure-dev-tools.md +81 -0
  289. package/skills/travel-planning.md +138 -0
  290. package/dist/heart/daemon/subagent-installer.js +0 -134
  291. package/dist/senses/bluebubbles-entry.js +0 -11
  292. package/dist/senses/bluebubbles.js +0 -547
  293. package/dist/senses/debug-activity.js +0 -124
  294. package/subagents/README.md +0 -73
  295. package/subagents/work-doer.md +0 -235
  296. package/subagents/work-merger.md +0 -618
  297. package/subagents/work-planner.md +0 -382
  298. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  299. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  300. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  301. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  302. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  303. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  304. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  305. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  306. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  307. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  308. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  309. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +0 -0
  310. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  311. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  312. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  313. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  314. /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
+ }
@@ -1,16 +1,16 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.FinalAnswerStreamer = exports.FinalAnswerParser = void 0;
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
- // `final_answer` tool call JSON arguments as they stream in.
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 FinalAnswerParser {
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 FinalAnswerParser.PREFIXES) {
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.FinalAnswerParser = FinalAnswerParser;
80
- // Shared helper: wraps FinalAnswerParser with onClearText + onTextChunk wiring.
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 FinalAnswerStreamer {
84
- parser = new FinalAnswerParser();
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
- constructor(callbacks) {
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 final_answer as detected. Calls onClearText on the callbacks. */
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.FinalAnswerStreamer = FinalAnswerStreamer;
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
- input.push({ role: "assistant", content: typeof a.content === "string" ? a.content : "" });
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 FinalAnswerStreamer(callbacks);
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 final_answer tool call on first name delta.
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 === "final_answer" && !answerStreamer.detected
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 final_answer argument deltas to the parser for progressive
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 === "final_answer"
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
- finalAnswerStreamed: answerStreamer.streamed,
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 FinalAnswerStreamer(callbacks);
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 final_answer function call -- clear any streamed noise.
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) === "final_answer" && functionCallCount === 1) {
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 final_answer argument deltas to the parser for progressive
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 === "final_answer"
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
- finalAnswerStreamed: answerStreamer.streamed,
497
+ settleStreamed: answerStreamer.streamed,
489
498
  };
490
499
  }