@ouro.bot/cli 0.1.0-alpha.35 → 0.1.0-alpha.350

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 (324) 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/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +2088 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +832 -0
  16. package/dist/heart/agent-entry.js +37 -2
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +463 -0
  28. package/dist/heart/bridges/manager.js +358 -0
  29. package/dist/heart/bridges/state-machine.js +135 -0
  30. package/dist/heart/bridges/store.js +123 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +53 -21
  35. package/dist/heart/core.js +744 -252
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +561 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +170 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +591 -0
  44. package/dist/heart/daemon/cli-exec.js +2633 -0
  45. package/dist/heart/daemon/cli-help.js +306 -0
  46. package/dist/heart/daemon/cli-parse.js +913 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +512 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/daemon-cli.js +30 -1171
  51. package/dist/heart/daemon/daemon-entry.js +358 -3
  52. package/dist/heart/daemon/daemon-health.js +141 -0
  53. package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
  54. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  55. package/dist/heart/daemon/daemon.js +751 -58
  56. package/dist/heart/daemon/doctor-types.js +8 -0
  57. package/dist/heart/daemon/doctor.js +465 -0
  58. package/dist/heart/daemon/health-monitor.js +79 -1
  59. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  60. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  61. package/dist/heart/daemon/http-health-probe.js +80 -0
  62. package/dist/heart/daemon/inner-status.js +89 -0
  63. package/dist/heart/daemon/interactive-repair.js +91 -0
  64. package/dist/heart/daemon/launchd.js +46 -9
  65. package/dist/heart/daemon/log-tailer.js +82 -12
  66. package/dist/heart/daemon/logs-prune.js +105 -0
  67. package/dist/heart/daemon/message-router.js +17 -8
  68. package/dist/heart/daemon/os-cron-deps.js +134 -0
  69. package/dist/heart/daemon/ouro-bot-entry.js +1 -1
  70. package/dist/heart/daemon/process-manager.js +201 -0
  71. package/dist/heart/daemon/provider-discovery.js +105 -0
  72. package/dist/heart/daemon/pulse.js +463 -0
  73. package/dist/heart/daemon/run-hooks.js +2 -0
  74. package/dist/heart/daemon/runtime-logging.js +67 -16
  75. package/dist/heart/daemon/runtime-metadata.js +101 -0
  76. package/dist/heart/daemon/runtime-mode.js +67 -0
  77. package/dist/heart/daemon/safe-mode.js +161 -0
  78. package/dist/heart/daemon/sense-manager.js +72 -3
  79. package/dist/heart/daemon/session-id-resolver.js +131 -0
  80. package/dist/heart/daemon/skill-management-installer.js +94 -0
  81. package/dist/heart/daemon/socket-client.js +307 -0
  82. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  83. package/dist/heart/daemon/startup-tui.js +237 -0
  84. package/dist/heart/daemon/task-scheduler.js +3 -25
  85. package/dist/heart/daemon/thoughts.js +510 -0
  86. package/dist/heart/daemon/up-progress.js +135 -0
  87. package/dist/heart/delegation.js +62 -0
  88. package/dist/heart/habits/habit-migration.js +181 -0
  89. package/dist/heart/habits/habit-parser.js +140 -0
  90. package/dist/heart/habits/habit-scheduler.js +371 -0
  91. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -120
  92. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  93. package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
  94. package/dist/heart/{daemon → hatch}/specialist-tools.js +56 -10
  95. package/dist/heart/identity.js +154 -59
  96. package/dist/heart/kept-notes.js +357 -0
  97. package/dist/heart/kicks.js +2 -20
  98. package/dist/heart/machine-identity.js +161 -0
  99. package/dist/heart/mcp/mcp-server.js +653 -0
  100. package/dist/heart/migrate-config.js +127 -0
  101. package/dist/heart/model-capabilities.js +59 -0
  102. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  103. package/dist/heart/outlook/outlook-http-response.js +7 -0
  104. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  105. package/dist/heart/outlook/outlook-http-static.js +99 -0
  106. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  107. package/dist/heart/outlook/outlook-http.js +99 -0
  108. package/dist/heart/outlook/outlook-read.js +28 -0
  109. package/dist/heart/outlook/outlook-types.js +27 -0
  110. package/dist/heart/outlook/outlook-view.js +194 -0
  111. package/dist/heart/outlook/readers/agent-machine.js +355 -0
  112. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  113. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  114. package/dist/heart/outlook/readers/sessions.js +231 -0
  115. package/dist/heart/outlook/readers/shared.js +111 -0
  116. package/dist/heart/progress-story.js +42 -0
  117. package/dist/heart/provider-attempt.js +133 -0
  118. package/dist/heart/provider-binding-resolver.js +240 -0
  119. package/dist/heart/provider-credential-pool.js +395 -0
  120. package/dist/heart/provider-failover.js +135 -0
  121. package/dist/heart/provider-models.js +81 -0
  122. package/dist/heart/provider-ping.js +258 -0
  123. package/dist/heart/provider-state.js +208 -0
  124. package/dist/heart/providers/anthropic-token.js +163 -0
  125. package/dist/heart/providers/anthropic.js +171 -50
  126. package/dist/heart/providers/azure.js +97 -11
  127. package/dist/heart/providers/error-classification.js +63 -0
  128. package/dist/heart/providers/github-copilot.js +135 -0
  129. package/dist/heart/providers/minimax-vlm.js +189 -0
  130. package/dist/heart/providers/minimax.js +23 -6
  131. package/dist/heart/providers/openai-codex.js +33 -23
  132. package/dist/heart/session-activity.js +190 -0
  133. package/dist/heart/session-events.js +727 -0
  134. package/dist/heart/session-transcript.js +162 -0
  135. package/dist/heart/start-of-turn-packet.js +341 -0
  136. package/dist/heart/streaming.js +36 -27
  137. package/dist/heart/sync.js +332 -0
  138. package/dist/heart/target-resolution.js +127 -0
  139. package/dist/heart/tempo.js +93 -0
  140. package/dist/heart/temporal-view.js +41 -0
  141. package/dist/heart/tool-activity-callbacks.js +36 -0
  142. package/dist/heart/tool-description.js +135 -0
  143. package/dist/heart/tool-friction.js +55 -0
  144. package/dist/heart/tool-loop.js +200 -0
  145. package/dist/heart/turn-context.js +358 -0
  146. package/dist/heart/turn-coordinator.js +28 -0
  147. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  148. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  149. package/dist/heart/versioning/ouro-path-installer.js +296 -0
  150. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  151. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  152. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  153. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  154. package/dist/mind/bundle-manifest.js +7 -1
  155. package/dist/mind/context.js +140 -94
  156. package/dist/mind/diary-integrity.js +60 -0
  157. package/dist/mind/{memory.js → diary.js} +84 -96
  158. package/dist/mind/embedding-provider.js +60 -0
  159. package/dist/mind/file-state.js +179 -0
  160. package/dist/mind/first-impressions.js +14 -1
  161. package/dist/mind/friends/channel.js +56 -0
  162. package/dist/mind/friends/group-context.js +144 -0
  163. package/dist/mind/friends/resolver.js +38 -1
  164. package/dist/mind/friends/store-file.js +58 -3
  165. package/dist/mind/friends/trust-explanation.js +74 -0
  166. package/dist/mind/friends/types.js +9 -1
  167. package/dist/mind/journal-index.js +161 -0
  168. package/dist/mind/note-search.js +268 -0
  169. package/dist/mind/obligation-steering.js +221 -0
  170. package/dist/mind/pending.js +74 -7
  171. package/dist/mind/prompt.js +1000 -112
  172. package/dist/mind/provenance-trust.js +26 -0
  173. package/dist/mind/scrutiny.js +173 -0
  174. package/dist/mind/token-estimate.js +8 -12
  175. package/dist/nerves/cli-logging.js +7 -1
  176. package/dist/nerves/coverage/audit-rules.js +15 -6
  177. package/dist/nerves/coverage/audit.js +28 -2
  178. package/dist/nerves/coverage/cli.js +1 -1
  179. package/dist/nerves/coverage/file-completeness.js +83 -5
  180. package/dist/nerves/coverage/run-artifacts.js +1 -1
  181. package/dist/nerves/event-buffer.js +111 -0
  182. package/dist/nerves/index.js +224 -4
  183. package/dist/nerves/observation.js +20 -0
  184. package/dist/nerves/redact.js +79 -0
  185. package/dist/nerves/runtime.js +5 -1
  186. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  187. package/dist/outlook-ui/assets/index-xTdv64BV.js +61 -0
  188. package/dist/outlook-ui/index.html +15 -0
  189. package/dist/repertoire/ado-client.js +15 -56
  190. package/dist/repertoire/ado-semantic.js +11 -10
  191. package/dist/repertoire/api-client.js +97 -0
  192. package/dist/repertoire/bitwarden-store.js +319 -0
  193. package/dist/repertoire/bundle-templates.js +72 -0
  194. package/dist/repertoire/bw-installer.js +79 -0
  195. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  196. package/dist/repertoire/coding/context-pack.js +330 -0
  197. package/dist/repertoire/coding/feedback.js +197 -30
  198. package/dist/repertoire/coding/manager.js +158 -9
  199. package/dist/repertoire/coding/spawner.js +55 -9
  200. package/dist/repertoire/coding/tools.js +170 -7
  201. package/dist/repertoire/commerce-errors.js +109 -0
  202. package/dist/repertoire/commerce-self-test.js +156 -0
  203. package/dist/repertoire/credential-access.js +527 -0
  204. package/dist/repertoire/duffel-client.js +185 -0
  205. package/dist/repertoire/github-client.js +14 -55
  206. package/dist/repertoire/graph-client.js +11 -52
  207. package/dist/repertoire/guardrails.js +375 -0
  208. package/dist/repertoire/mcp-client.js +255 -0
  209. package/dist/repertoire/mcp-manager.js +305 -0
  210. package/dist/repertoire/mcp-tools.js +63 -0
  211. package/dist/repertoire/shell-sessions.js +133 -0
  212. package/dist/repertoire/skills.js +15 -24
  213. package/dist/repertoire/stripe-client.js +131 -0
  214. package/dist/repertoire/tasks/board.js +43 -5
  215. package/dist/repertoire/tasks/fix.js +182 -0
  216. package/dist/repertoire/tasks/index.js +28 -10
  217. package/dist/repertoire/tasks/lifecycle.js +2 -2
  218. package/dist/repertoire/tasks/parser.js +3 -2
  219. package/dist/repertoire/tasks/scanner.js +194 -37
  220. package/dist/repertoire/tasks/transitions.js +16 -79
  221. package/dist/repertoire/tool-results.js +29 -0
  222. package/dist/repertoire/tools-attachments.js +316 -0
  223. package/dist/repertoire/tools-base.js +45 -771
  224. package/dist/repertoire/tools-bluebubbles.js +1 -0
  225. package/dist/repertoire/tools-bridge.js +141 -0
  226. package/dist/repertoire/tools-bundle.js +984 -0
  227. package/dist/repertoire/tools-config.js +185 -0
  228. package/dist/repertoire/tools-continuity.js +248 -0
  229. package/dist/repertoire/tools-credential.js +182 -0
  230. package/dist/repertoire/tools-files.js +342 -0
  231. package/dist/repertoire/tools-flight.js +224 -0
  232. package/dist/repertoire/tools-flow.js +105 -0
  233. package/dist/repertoire/tools-github.js +1 -7
  234. package/dist/repertoire/tools-notes.js +376 -0
  235. package/dist/repertoire/tools-session.js +739 -0
  236. package/dist/repertoire/tools-shell.js +120 -0
  237. package/dist/repertoire/tools-stripe.js +180 -0
  238. package/dist/repertoire/tools-surface.js +243 -0
  239. package/dist/repertoire/tools-teams.js +12 -62
  240. package/dist/repertoire/tools-travel.js +125 -0
  241. package/dist/repertoire/tools-user-profile.js +144 -0
  242. package/dist/repertoire/tools-vault.js +110 -0
  243. package/dist/repertoire/tools.js +144 -138
  244. package/dist/repertoire/travel-api-client.js +360 -0
  245. package/dist/repertoire/user-profile.js +118 -0
  246. package/dist/repertoire/vault-setup.js +241 -0
  247. package/dist/scripts/claude-code-hook.js +41 -0
  248. package/dist/scripts/claude-code-stop-hook.js +47 -0
  249. package/dist/senses/attention-queue.js +116 -0
  250. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  251. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  252. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +225 -9
  253. package/dist/senses/bluebubbles/entry.js +13 -0
  254. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  255. package/dist/senses/bluebubbles/index.js +1590 -0
  256. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  257. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  258. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  259. package/dist/senses/bluebubbles/replay.js +129 -0
  260. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  261. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  262. package/dist/senses/cli/bracketed-paste.js +82 -0
  263. package/dist/senses/cli/image-paste.js +287 -0
  264. package/dist/senses/cli/image-ref-navigation.js +75 -0
  265. package/dist/senses/cli/ink-app.js +156 -0
  266. package/dist/senses/cli/inline-diff.js +64 -0
  267. package/dist/senses/cli/input-keys.js +174 -0
  268. package/dist/senses/cli/kill-ring.js +86 -0
  269. package/dist/senses/cli/message-list.js +51 -0
  270. package/dist/senses/cli/ouro-tui.js +605 -0
  271. package/dist/senses/cli/spinner-imperative.js +135 -0
  272. package/dist/senses/cli/spinner.js +101 -0
  273. package/dist/senses/cli/status-line.js +60 -0
  274. package/dist/senses/cli/streaming-markdown.js +526 -0
  275. package/dist/senses/cli/tool-display.js +83 -0
  276. package/dist/senses/cli/tool-render.js +85 -0
  277. package/dist/senses/cli/tui-store.js +240 -0
  278. package/dist/senses/cli/virtual-list.js +35 -0
  279. package/dist/senses/cli-entry.js +1 -1
  280. package/dist/senses/cli-layout.js +187 -0
  281. package/dist/senses/cli.js +603 -246
  282. package/dist/senses/commands.js +65 -1
  283. package/dist/senses/continuity.js +94 -0
  284. package/dist/senses/habit-turn-message.js +108 -0
  285. package/dist/senses/inner-dialog-worker.js +112 -19
  286. package/dist/senses/inner-dialog.js +633 -86
  287. package/dist/senses/pipeline.js +567 -0
  288. package/dist/senses/shared-turn.js +199 -0
  289. package/dist/senses/surface-tool.js +68 -0
  290. package/dist/senses/teams.js +665 -160
  291. package/dist/senses/trust-gate.js +112 -2
  292. package/package.json +29 -7
  293. package/skills/agent-commerce.md +106 -0
  294. package/skills/browser-navigation.md +110 -0
  295. package/skills/commerce-setup-guide.md +116 -0
  296. package/skills/commerce-setup.md +84 -0
  297. package/skills/configure-dev-tools.md +81 -0
  298. package/skills/travel-planning.md +138 -0
  299. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  300. package/dist/heart/daemon/subagent-installer.js +0 -134
  301. package/dist/mind/associative-recall.js +0 -197
  302. package/dist/senses/bluebubbles-entry.js +0 -11
  303. package/dist/senses/bluebubbles.js +0 -548
  304. package/dist/senses/debug-activity.js +0 -127
  305. package/subagents/README.md +0 -73
  306. package/subagents/work-doer.md +0 -235
  307. package/subagents/work-merger.md +0 -618
  308. package/subagents/work-planner.md +0 -382
  309. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  310. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  311. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  312. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  313. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  314. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  315. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  316. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  317. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  318. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  319. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  320. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  321. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  322. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  323. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  324. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -1,97 +1,122 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.hasToolIntent = exports.buildSystem = exports.toResponsesTools = exports.toResponsesInput = exports.streamResponsesApi = exports.streamChatCompletion = exports.getToolsForChannel = exports.summarizeArgs = exports.execTool = exports.tools = void 0;
4
3
  exports.createProviderRegistry = createProviderRegistry;
5
4
  exports.resetProviderRuntime = resetProviderRuntime;
6
5
  exports.getModel = getModel;
7
6
  exports.getProvider = getProvider;
8
7
  exports.createSummarize = createSummarize;
9
8
  exports.getProviderDisplayLabel = getProviderDisplayLabel;
9
+ exports.isExternalStateQuery = isExternalStateQuery;
10
+ exports.getSettleRetryError = getSettleRetryError;
10
11
  exports.stripLastToolCalls = stripLastToolCalls;
11
12
  exports.repairOrphanedToolCalls = repairOrphanedToolCalls;
12
- exports.isTransientError = isTransientError;
13
- exports.classifyTransientError = classifyTransientError;
14
13
  exports.runAgent = runAgent;
15
14
  const config_1 = require("./config");
16
15
  const identity_1 = require("./identity");
17
16
  const tools_1 = require("../repertoire/tools");
18
17
  const channel_1 = require("../mind/friends/channel");
19
- // Kick detection preserved but disabled — see comment in agent loop below.
20
- // import { detectKick } from "./kicks";
21
- // import type { KickReason } from "./kicks";
18
+ const tools_2 = require("../repertoire/tools");
22
19
  const runtime_1 = require("../nerves/runtime");
23
20
  const context_1 = require("../mind/context");
24
21
  const prompt_1 = require("../mind/prompt");
25
- const associative_recall_1 = require("../mind/associative-recall");
22
+ const kept_notes_1 = require("./kept-notes");
26
23
  const anthropic_1 = require("./providers/anthropic");
27
24
  const azure_1 = require("./providers/azure");
28
25
  const minimax_1 = require("./providers/minimax");
29
26
  const openai_codex_1 = require("./providers/openai-codex");
30
- let _providerRuntime = null;
27
+ const github_copilot_1 = require("./providers/github-copilot");
28
+ const identity_2 = require("./identity");
29
+ const socket_client_1 = require("./daemon/socket-client");
30
+ const obligations_1 = require("../arc/obligations");
31
+ const tool_loop_1 = require("./tool-loop");
32
+ const packets_1 = require("../arc/packets");
33
+ const tool_friction_1 = require("./tool-friction");
34
+ const provider_models_1 = require("./provider-models");
35
+ const provider_attempt_1 = require("./provider-attempt");
36
+ const _providerRuntimes = {
37
+ human: null,
38
+ agent: null,
39
+ };
40
+ function getProviderRuntimeFingerprint(facing) {
41
+ const config = (0, identity_1.loadAgentConfig)();
42
+ const facingConfig = facing === "human" ? config.humanFacing : config.agentFacing;
43
+ const provider = facingConfig.provider;
44
+ const model = facingConfig.model;
45
+ const providerConfig = (0, config_1.getProviderConfig)(provider);
46
+ return JSON.stringify({ provider, model, ...providerConfig });
47
+ }
31
48
  function createProviderRegistry() {
32
49
  const factories = {
33
50
  azure: azure_1.createAzureProviderRuntime,
34
51
  anthropic: anthropic_1.createAnthropicProviderRuntime,
35
52
  minimax: minimax_1.createMinimaxProviderRuntime,
36
53
  "openai-codex": openai_codex_1.createOpenAICodexProviderRuntime,
54
+ "github-copilot": github_copilot_1.createGithubCopilotProviderRuntime,
37
55
  };
38
56
  return {
39
- resolve() {
40
- const provider = (0, identity_1.loadAgentConfig)().provider;
41
- return factories[provider]();
57
+ resolve(provider, model) {
58
+ const resolvedProvider = provider ?? (0, identity_1.loadAgentConfig)().humanFacing.provider;
59
+ const resolvedModel = model ?? (0, identity_1.loadAgentConfig)().humanFacing.model;
60
+ return factories[resolvedProvider](resolvedModel);
42
61
  },
43
62
  };
44
63
  }
45
- function getProviderRuntime() {
46
- if (!_providerRuntime) {
47
- try {
48
- _providerRuntime = createProviderRegistry().resolve();
49
- }
50
- catch (error) {
51
- const msg = error instanceof Error ? error.message : String(error);
52
- (0, runtime_1.emitNervesEvent)({
53
- level: "error",
54
- event: "engine.provider_init_error",
55
- component: "engine",
56
- message: msg,
57
- meta: {},
58
- });
59
- // eslint-disable-next-line no-console -- pre-boot guard: provider init failure
60
- console.error(`\n[fatal] ${msg}\n`);
61
- process.exit(1);
62
- throw new Error("unreachable");
63
- }
64
- if (!_providerRuntime) {
65
- (0, runtime_1.emitNervesEvent)({
66
- level: "error",
67
- event: "engine.provider_init_error",
68
- component: "engine",
69
- message: "provider runtime could not be initialized.",
70
- meta: {},
71
- });
72
- process.exit(1);
73
- throw new Error("unreachable");
64
+ function getProviderRuntime(facing = "human") {
65
+ try {
66
+ const fingerprint = getProviderRuntimeFingerprint(facing);
67
+ const cached = _providerRuntimes[facing];
68
+ if (!cached || cached.fingerprint !== fingerprint) {
69
+ const config = (0, identity_1.loadAgentConfig)();
70
+ const facingConfig = facing === "human" ? config.humanFacing : config.agentFacing;
71
+ const runtime = createProviderRegistry().resolve(facingConfig.provider, facingConfig.model);
72
+ _providerRuntimes[facing] = runtime ? { fingerprint, runtime } : null;
74
73
  }
75
74
  }
76
- return _providerRuntime;
75
+ catch (error) {
76
+ const msg = error instanceof Error ? error.message : String(error);
77
+ (0, runtime_1.emitNervesEvent)({
78
+ level: "error",
79
+ event: "engine.provider_init_error",
80
+ component: "engine",
81
+ message: msg,
82
+ meta: {},
83
+ });
84
+ // eslint-disable-next-line no-console -- pre-boot guard: provider init failure
85
+ console.error(`\n[fatal] ${msg}\n`);
86
+ process.exit(1);
87
+ throw new Error("unreachable");
88
+ }
89
+ if (!_providerRuntimes[facing]) {
90
+ (0, runtime_1.emitNervesEvent)({
91
+ level: "error",
92
+ event: "engine.provider_init_error",
93
+ component: "engine",
94
+ message: "provider runtime could not be initialized.",
95
+ meta: {},
96
+ });
97
+ process.exit(1);
98
+ throw new Error("unreachable");
99
+ }
100
+ return _providerRuntimes[facing].runtime;
77
101
  }
78
102
  /**
79
- * Clear the cached provider runtime so the next call to getProviderRuntime()
80
- * re-creates it from current config. Used by the adoption specialist to
81
- * switch provider context without restarting the process.
103
+ * Clear the cached provider runtime so the next access re-creates it from
104
+ * current config. Runtime access also auto-refreshes when the selected
105
+ * provider fingerprint changes on disk.
82
106
  */
83
107
  function resetProviderRuntime() {
84
- _providerRuntime = null;
108
+ _providerRuntimes.human = null;
109
+ _providerRuntimes.agent = null;
85
110
  }
86
- function getModel() {
87
- return getProviderRuntime().model;
111
+ function getModel(facing = "human") {
112
+ return getProviderRuntime(facing).model;
88
113
  }
89
- function getProvider() {
90
- return getProviderRuntime().id;
114
+ function getProvider(facing = "human") {
115
+ return getProviderRuntime(facing).id;
91
116
  }
92
- function createSummarize() {
117
+ function createSummarize(facing = "human") {
93
118
  return async (transcript, instruction) => {
94
- const runtime = getProviderRuntime();
119
+ const runtime = getProviderRuntime(facing);
95
120
  const client = runtime.client;
96
121
  const response = await client.chat.completions.create({
97
122
  model: runtime.model,
@@ -104,34 +129,138 @@ function createSummarize() {
104
129
  return response.choices?.[0]?.message?.content ?? transcript;
105
130
  };
106
131
  }
107
- function getProviderDisplayLabel() {
108
- const model = getModel();
132
+ function getProviderDisplayLabel(facing = "human") {
133
+ const config = (0, identity_1.loadAgentConfig)();
134
+ const facingConfig = facing === "human" ? config.humanFacing : config.agentFacing;
135
+ const provider = facingConfig.provider;
136
+ const model = facingConfig.model || "unknown";
109
137
  const providerLabelBuilders = {
110
- azure: () => `azure openai (${(0, config_1.getAzureConfig)().deployment || "default"}, model: ${model})`,
138
+ azure: () => {
139
+ const azureCfg = (0, config_1.getAzureConfig)();
140
+ return `azure openai (${azureCfg.deployment || "default"}, model: ${model})`;
141
+ },
111
142
  anthropic: () => `anthropic (${model})`,
112
143
  minimax: () => `minimax (${model})`,
113
144
  "openai-codex": () => `openai codex (${model})`,
145
+ /* v8 ignore next -- branch: tested via display label unit test @preserve */
146
+ "github-copilot": () => `github copilot (${model})`,
147
+ };
148
+ return providerLabelBuilders[provider]();
149
+ }
150
+ // Sole-call tools must be the only tool call in a turn. When they appear
151
+ // alongside other tools, the sole-call tool is rejected with this message.
152
+ const SOLE_CALL_REJECTION = {
153
+ settle: "rejected: settle must be the only tool call. finish your work first, then call settle alone.",
154
+ observe: "rejected: observe must be the only tool call. call observe alone when you want to stay silent.",
155
+ rest: "rejected: rest must be the only tool call. finish your work first, then call rest alone.",
156
+ };
157
+ function parseSettlePayload(argumentsText) {
158
+ try {
159
+ const parsed = JSON.parse(argumentsText);
160
+ if (typeof parsed === "string") {
161
+ return { answer: parsed };
162
+ }
163
+ if (!parsed || typeof parsed !== "object") {
164
+ return {};
165
+ }
166
+ const answer = typeof parsed.answer === "string" ? parsed.answer : undefined;
167
+ const rawIntent = parsed.intent;
168
+ const intent = rawIntent === "complete" || rawIntent === "blocked" || rawIntent === "direct_reply"
169
+ ? rawIntent
170
+ : undefined;
171
+ return { answer, intent };
172
+ }
173
+ catch {
174
+ return {};
175
+ }
176
+ }
177
+ function parsePonderPayload(argumentsText) {
178
+ try {
179
+ const parsed = JSON.parse(argumentsText);
180
+ return parsed && typeof parsed === "object" ? parsed : {};
181
+ }
182
+ catch {
183
+ return {};
184
+ }
185
+ }
186
+ function parseSuccessCriteria(raw) {
187
+ if (typeof raw !== "string")
188
+ return null;
189
+ const criteria = raw
190
+ .split("\n")
191
+ .map((line) => line.replace(/^\s*[-*]\s*/, "").trim())
192
+ .filter((line) => line.length > 0);
193
+ return criteria.length > 0 ? criteria : null;
194
+ }
195
+ function parsePacketPayload(raw) {
196
+ if (typeof raw !== "string")
197
+ return null;
198
+ try {
199
+ const parsed = JSON.parse(raw);
200
+ return parsed && typeof parsed === "object" && !Array.isArray(parsed)
201
+ ? parsed
202
+ : null;
203
+ }
204
+ catch {
205
+ return null;
206
+ }
207
+ }
208
+ function normalizeLegacyPonderArgs(parsed) {
209
+ if (typeof parsed.thought !== "string" || parsed.thought.trim().length === 0) {
210
+ return parsed;
211
+ }
212
+ return {
213
+ action: "create",
214
+ kind: "reflection",
215
+ objective: parsed.thought.trim(),
216
+ summary: typeof parsed.say === "string" ? parsed.say.trim() : "",
217
+ success_criteria: "- preserve the thread for later work",
218
+ payload_json: "{}",
114
219
  };
115
- return providerLabelBuilders[getProvider()]();
116
220
  }
117
- // Re-export tools, execTool, summarizeArgs from ./tools for backward compat
118
- var tools_2 = require("../repertoire/tools");
119
- Object.defineProperty(exports, "tools", { enumerable: true, get: function () { return tools_2.tools; } });
120
- Object.defineProperty(exports, "execTool", { enumerable: true, get: function () { return tools_2.execTool; } });
121
- Object.defineProperty(exports, "summarizeArgs", { enumerable: true, get: function () { return tools_2.summarizeArgs; } });
122
- Object.defineProperty(exports, "getToolsForChannel", { enumerable: true, get: function () { return tools_2.getToolsForChannel; } });
123
- // Re-export streaming functions for backward compat
124
- var streaming_1 = require("./streaming");
125
- Object.defineProperty(exports, "streamChatCompletion", { enumerable: true, get: function () { return streaming_1.streamChatCompletion; } });
126
- Object.defineProperty(exports, "streamResponsesApi", { enumerable: true, get: function () { return streaming_1.streamResponsesApi; } });
127
- Object.defineProperty(exports, "toResponsesInput", { enumerable: true, get: function () { return streaming_1.toResponsesInput; } });
128
- Object.defineProperty(exports, "toResponsesTools", { enumerable: true, get: function () { return streaming_1.toResponsesTools; } });
129
- // Re-export prompt functions for backward compat
130
- var prompt_2 = require("../mind/prompt");
131
- Object.defineProperty(exports, "buildSystem", { enumerable: true, get: function () { return prompt_2.buildSystem; } });
132
- // Re-export kick utilities for backward compat
133
- var kicks_1 = require("./kicks");
134
- Object.defineProperty(exports, "hasToolIntent", { enumerable: true, get: function () { return kicks_1.hasToolIntent; } });
221
+ function buildPonderResult(packet, action, returnObligationId) {
222
+ return JSON.stringify({
223
+ ok: true,
224
+ packet_id: packet.id,
225
+ action,
226
+ status: packet.status,
227
+ return_obligation_id: returnObligationId,
228
+ }, null, 2);
229
+ }
230
+ /** Returns true when a tool call queries external state (GitHub, npm registry). */
231
+ function isExternalStateQuery(toolName, args) {
232
+ if (toolName !== "shell")
233
+ return false;
234
+ const cmd = String(args.command ?? "");
235
+ return /\bgh\s+(pr|run|api|issue)\b/.test(cmd) || /\bnpm\s+(view|info|show)\b/.test(cmd);
236
+ }
237
+ function getSettleRetryError(mustResolveBeforeHandoff, intent, sawSteeringFollowUp, _delegationDecision, sawSendMessageSelf, sawPonder, _sawQuerySession, currentObligation, innerJob, sawExternalStateQuery) {
238
+ // Delegation adherence removed: the delegation decision is surfaced in the
239
+ // system prompt as a suggestion. Hard-gating settle caused infinite
240
+ // rejection loops where the agent couldn't respond to the user at all.
241
+ // The agent is free to follow or ignore the delegation hint.
242
+ // 2. Pending obligation not addressed
243
+ if (innerJob?.obligationStatus === "pending" && !sawSendMessageSelf && !sawPonder) {
244
+ return "you're still holding something from an earlier conversation -- someone is waiting for your answer. finish the thought first, or ponder to keep working on it privately.";
245
+ }
246
+ // 3. mustResolveBeforeHandoff + missing intent
247
+ if (mustResolveBeforeHandoff && !intent) {
248
+ return "your settle is missing required intent. when you must keep going until done or blocked, call settle again with answer plus intent=complete, blocked, or direct_reply.";
249
+ }
250
+ // 4. mustResolveBeforeHandoff + direct_reply without follow-up
251
+ if (mustResolveBeforeHandoff && intent === "direct_reply" && !sawSteeringFollowUp) {
252
+ return "your settle used intent=direct_reply without a newer steering follow-up. continue the unresolved work, or call settle again with intent=complete or blocked when appropriate.";
253
+ }
254
+ // 5. mustResolveBeforeHandoff + complete while a live return loop is still active
255
+ if (mustResolveBeforeHandoff && intent === "complete" && currentObligation && !sawSteeringFollowUp) {
256
+ return "you still owe the live session a visible return on this work. don't end the turn yet — continue until you've brought back the external-state update, or use intent=blocked with the concrete blocker.";
257
+ }
258
+ // 6. External-state grounding: obligation + complete requires fresh external verification
259
+ if (intent === "complete" && currentObligation && !sawExternalStateQuery && !sawSteeringFollowUp) {
260
+ return "you're claiming this work is complete, but the external state hasn't been verified this turn. ground your claim with a fresh check (gh pr view, npm view, gh run view, etc.) before calling settle.";
261
+ }
262
+ return null;
263
+ }
135
264
  function upsertSystemPrompt(messages, systemText) {
136
265
  const systemMessage = { role: "system", content: systemText };
137
266
  if (messages[0]?.role === "system") {
@@ -237,49 +366,67 @@ function isContextOverflow(err) {
237
366
  return true;
238
367
  return false;
239
368
  }
240
- // Detect transient network errors worth retrying
241
- function isTransientError(err) {
242
- if (!(err instanceof Error))
243
- return false;
244
- const msg = err.message || "";
245
- const code = err.code || "";
246
- // Node.js network error codes
247
- if (["ECONNRESET", "ECONNREFUSED", "ENOTFOUND", "ETIMEDOUT", "EPIPE",
248
- "EAI_AGAIN", "EHOSTUNREACH", "ENETUNREACH", "ECONNABORTED"].includes(code))
249
- return true;
250
- // OpenAI SDK / fetch errors
251
- if (msg.includes("fetch failed"))
252
- return true;
253
- if (msg.includes("network") && !msg.includes("context"))
254
- return true;
255
- if (msg.includes("ECONNRESET") || msg.includes("ETIMEDOUT"))
256
- return true;
257
- if (msg.includes("socket hang up"))
258
- return true;
259
- if (msg.includes("getaddrinfo"))
260
- return true;
261
- // HTTP 429 / 500 / 502 / 503 / 504
262
- const status = err.status;
263
- if (status === 429 || status === 500 || status === 502 || status === 503 || status === 504)
264
- return true;
265
- return false;
369
+ const RETRY_LABELS = {
370
+ "auth-failure": "auth error",
371
+ "usage-limit": "usage limit",
372
+ "rate-limit": "rate limited",
373
+ "server-error": "server error",
374
+ "network-error": "network error",
375
+ "unknown": "error",
376
+ };
377
+ function waitForProviderRetry(delayMs, signal) {
378
+ if (!signal) {
379
+ return new Promise((resolve) => {
380
+ setTimeout(resolve, delayMs);
381
+ });
382
+ }
383
+ return new Promise((resolve, reject) => {
384
+ let timer;
385
+ const onAbort = () => {
386
+ clearTimeout(timer);
387
+ reject(new provider_attempt_1.ProviderAttemptAbortError());
388
+ };
389
+ timer = setTimeout(() => {
390
+ signal.removeEventListener("abort", onAbort);
391
+ resolve();
392
+ }, delayMs);
393
+ if (signal.aborted) {
394
+ onAbort();
395
+ return;
396
+ }
397
+ signal.addEventListener("abort", onAbort, { once: true });
398
+ });
266
399
  }
267
- function classifyTransientError(err) {
268
- if (!(err instanceof Error))
269
- return "unknown error";
270
- const status = err.status;
271
- if (status === 429)
272
- return "rate limited";
273
- if (status === 401 || status === 403)
274
- return "auth error";
275
- if (status && status >= 500)
276
- return "server error";
277
- return "network error";
400
+ function buildAuthFailureGuidance(provider, model, agentName, detail) {
401
+ const mismatch = (0, provider_models_1.getProviderModelMismatchMessage)(provider, model);
402
+ const modelLabel = model
403
+ ? mismatch
404
+ ? `${provider} [configured model: ${model}]`
405
+ : `${provider} (${model})`
406
+ : provider;
407
+ const lines = [`${modelLabel} authentication failed.`];
408
+ const cleanDetail = detail.replace(/\s+/g, " ").trim();
409
+ if (cleanDetail)
410
+ lines.push(`provider detail: ${cleanDetail.length > 300 ? `${cleanDetail.slice(0, 297)}...` : cleanDetail}`);
411
+ lines.push("");
412
+ lines.push("To keep using this provider:");
413
+ lines.push(` 1. Run \`ouro auth --agent ${agentName} --provider ${provider}\``);
414
+ if (mismatch) {
415
+ const defaultModel = (0, provider_models_1.getDefaultModelForProvider)(provider);
416
+ lines.push("");
417
+ lines.push("Config warning:");
418
+ lines.push(` - ${mismatch}`);
419
+ lines.push(" - Repair the configured model with:");
420
+ lines.push(` \`ouro config model --agent ${agentName} --facing human ${defaultModel}\``);
421
+ lines.push(` \`ouro config model --agent ${agentName} --facing agent ${defaultModel}\``);
422
+ }
423
+ lines.push("");
424
+ lines.push(`To use another configured provider instead, run \`ouro auth switch --agent ${agentName} --provider <provider>\`.`);
425
+ return lines.join("\n");
278
426
  }
279
- const MAX_RETRIES = 3;
280
- const RETRY_BASE_MS = 2000;
281
427
  async function runAgent(messages, callbacks, channel, signal, options) {
282
- const providerRuntime = getProviderRuntime();
428
+ const facing = (0, channel_1.channelToFacing)(channel);
429
+ const providerRuntime = getProviderRuntime(facing);
283
430
  const provider = providerRuntime.id;
284
431
  const toolChoiceRequired = options?.toolChoiceRequired ?? true;
285
432
  const traceId = options?.traceId;
@@ -305,7 +452,12 @@ async function runAgent(messages, callbacks, channel, signal, options) {
305
452
  // so turn execution remains consistent and non-fatal.
306
453
  if (channel) {
307
454
  try {
308
- const refreshed = await (0, prompt_1.buildSystem)(channel, options, currentContext);
455
+ const buildSystemOptions = {
456
+ ...options,
457
+ providerCapabilities: providerRuntime.capabilities,
458
+ supportedReasoningEfforts: providerRuntime.supportedReasoningEfforts,
459
+ };
460
+ const refreshed = await (0, prompt_1.buildSystem)(channel, buildSystemOptions, currentContext);
309
461
  upsertSystemPrompt(messages, refreshed);
310
462
  }
311
463
  catch (error) {
@@ -327,32 +479,115 @@ async function runAgent(messages, callbacks, channel, signal, options) {
327
479
  });
328
480
  }
329
481
  }
330
- await (0, associative_recall_1.injectAssociativeRecall)(messages);
331
- // kickCount and lastKickReason preserved but unused while kick detection is disabled.
332
- // let kickCount = 0;
333
- // let lastKickReason: KickReason | null = null;
482
+ if (channel) {
483
+ await (0, kept_notes_1.injectKeptNotes)(messages, {
484
+ channel,
485
+ friend: currentContext?.friend,
486
+ judge: async (input) => (0, kept_notes_1.createKeptNotesJudge)(getProviderRuntime("agent"), signal)(input),
487
+ signal,
488
+ traceId,
489
+ });
490
+ }
334
491
  let done = false;
335
492
  let lastUsage;
336
493
  let overflowRetried = false;
337
- let retryCount = 0;
494
+ let outcome = "settled";
495
+ let completion;
496
+ let terminalError;
497
+ let terminalErrorClassification;
498
+ let sawSteeringFollowUp = false;
499
+ let mustResolveBeforeHandoffActive = options?.mustResolveBeforeHandoff === true;
500
+ let currentReasoningEffort = "medium";
501
+ let sawSendMessageSelf = false;
502
+ let sawPonder = false;
503
+ let sawQuerySession = false;
504
+ let sawBridgeManage = false;
505
+ let sawExternalStateQuery = false;
506
+ const toolLoopState = (0, tool_loop_1.createToolLoopState)();
507
+ const toolFrictionLedger = (0, tool_friction_1.createToolFrictionLedger)();
508
+ const finishTerminalProviderError = (error, classification) => {
509
+ terminalError = error;
510
+ terminalErrorClassification = classification;
511
+ /* v8 ignore start — auth-failure guidance: tested via provider error classification tests @preserve */
512
+ if (terminalErrorClassification === "auth-failure") {
513
+ const agentName = (0, identity_2.getAgentName)();
514
+ const currentProvider = providerRuntime.id;
515
+ callbacks.onError(new Error(buildAuthFailureGuidance(currentProvider, providerRuntime.model, agentName, terminalError.message)), "terminal");
516
+ }
517
+ else {
518
+ callbacks.onError(terminalError, "terminal");
519
+ }
520
+ /* v8 ignore stop */
521
+ (0, runtime_1.emitNervesEvent)({
522
+ level: "error",
523
+ event: "engine.error",
524
+ trace_id: traceId,
525
+ component: "engine",
526
+ message: terminalError.message,
527
+ meta: {
528
+ provider: providerRuntime.id,
529
+ model: providerRuntime.model,
530
+ errorClassification: terminalErrorClassification,
531
+ },
532
+ });
533
+ stripLastToolCalls(messages);
534
+ outcome = "errored";
535
+ done = true;
536
+ };
338
537
  // Prevent MaxListenersExceeded warning — each iteration adds a listener
339
538
  try {
340
539
  require("events").setMaxListeners(50, signal);
341
540
  }
342
541
  catch { /* unsupported */ }
343
542
  const toolPreferences = currentContext?.friend?.toolPreferences;
344
- const baseTools = options?.tools ?? (0, tools_1.getToolsForChannel)(channel ? (0, channel_1.getChannelCapabilities)(channel) : undefined, toolPreferences && Object.keys(toolPreferences).length > 0 ? toolPreferences : undefined, currentContext);
543
+ const baseTools = options?.tools ?? (0, tools_1.getToolsForChannel)(channel ? (0, channel_1.getChannelCapabilities)(channel) : undefined, toolPreferences && Object.keys(toolPreferences).length > 0 ? toolPreferences : undefined, currentContext, providerRuntime.capabilities, options?.mcpManager, providerRuntime.model);
544
+ // Augment tool context with reasoning effort controls from provider
545
+ const augmentedToolContext = options?.toolContext
546
+ ? {
547
+ ...options.toolContext,
548
+ supportedReasoningEfforts: providerRuntime.supportedReasoningEfforts,
549
+ setReasoningEffort: (level) => { currentReasoningEffort = level; },
550
+ activeWorkFrame: options?.activeWorkFrame,
551
+ }
552
+ : undefined;
345
553
  // Rebase provider-owned turn state from canonical messages at user-turn start.
346
554
  // This prevents stale provider caches from replaying prior-turn context.
347
555
  providerRuntime.resetTurnState(messages);
348
556
  while (!done) {
349
- // When toolChoiceRequired is true (the default), include final_answer
350
- // so the model can signal completion. With tool_choice: required, the
351
- // model must call a tool every turn final_answer is how it exits.
352
- // Overridable via options.toolChoiceRequired = false (e.g. CLI).
353
- const activeTools = toolChoiceRequired ? [...baseTools, tools_1.finalAnswerTool] : baseTools;
557
+ // Channel-based tool filtering:
558
+ // - Inner dialog: exclude send_message (delivery via surface), observe (no one to observe)
559
+ // - 1:1 sessions: exclude observe (can't ignore someone talking directly to you)
560
+ // - Group chats: observe available
561
+ //
562
+ // ponder, settle/rest, surface, and observe are always assembled based on channel context.
563
+ // ponder is available in ALL channels (outer: think privately, inner: keep turning).
564
+ // Inner dialog gets restTool instead of settleTool (rest = end turn, gated by attention queue).
565
+ // toolChoiceRequired only controls whether tool_choice: "required" is set in the API call.
566
+ const isInnerDialog = channel === "inner";
567
+ const filteredBaseTools = isInnerDialog
568
+ ? baseTools.filter((t) => t.function.name !== "send_message")
569
+ : baseTools;
570
+ const activeTools = [
571
+ ...filteredBaseTools,
572
+ tools_1.ponderTool,
573
+ ...(isInnerDialog ? [tools_2.surfaceToolDef, tools_1.restTool] : []),
574
+ ...((currentContext?.isGroupChat || options?.isReactionSignal) && !isInnerDialog ? [tools_1.observeTool] : []),
575
+ ...(!isInnerDialog ? [tools_1.settleTool] : []),
576
+ ];
354
577
  const steeringFollowUps = options?.drainSteeringFollowUps?.() ?? [];
355
578
  if (steeringFollowUps.length > 0) {
579
+ const hasSupersedingFollowUp = steeringFollowUps.some((followUp) => followUp.effect === "clear_and_supersede");
580
+ if (hasSupersedingFollowUp) {
581
+ mustResolveBeforeHandoffActive = false;
582
+ options?.setMustResolveBeforeHandoff?.(false);
583
+ outcome = "superseded";
584
+ break;
585
+ }
586
+ if (steeringFollowUps.some((followUp) => followUp.effect === "set_no_handoff")) {
587
+ mustResolveBeforeHandoffActive = true;
588
+ options?.setMustResolveBeforeHandoff?.(true);
589
+ }
590
+ sawSteeringFollowUp = true;
356
591
  for (const followUp of steeringFollowUps) {
357
592
  messages.push({ role: "user", content: followUp.text });
358
593
  }
@@ -360,22 +595,76 @@ async function runAgent(messages, callbacks, channel, signal, options) {
360
595
  }
361
596
  // Yield so pending I/O (stdin Ctrl-C) can be processed between iterations
362
597
  await new Promise((r) => setImmediate(r));
363
- if (signal?.aborted)
598
+ if (signal?.aborted) {
599
+ outcome = "aborted";
364
600
  break;
601
+ }
365
602
  try {
366
- callbacks.onModelStart();
367
- const result = await providerRuntime.streamTurn({
368
- messages,
369
- activeTools,
370
- callbacks,
371
- signal,
372
- traceId,
373
- toolChoiceRequired,
603
+ const callProviderTurn = async () => {
604
+ callbacks.onModelStart();
605
+ try {
606
+ return await providerRuntime.streamTurn({
607
+ messages,
608
+ activeTools,
609
+ callbacks,
610
+ signal,
611
+ traceId,
612
+ toolChoiceRequired,
613
+ reasoningEffort: currentReasoningEffort,
614
+ eagerSettleStreaming: true,
615
+ });
616
+ }
617
+ catch (error) {
618
+ if (signal?.aborted)
619
+ throw new provider_attempt_1.ProviderAttemptAbortError();
620
+ throw error;
621
+ }
622
+ };
623
+ const callProviderTurnWithOverflowRecovery = async () => {
624
+ try {
625
+ return await callProviderTurn();
626
+ }
627
+ catch (error) {
628
+ if (error instanceof provider_attempt_1.ProviderAttemptAbortError)
629
+ throw error;
630
+ if (isContextOverflow(error) && !overflowRetried) {
631
+ overflowRetried = true;
632
+ stripLastToolCalls(messages);
633
+ const { maxTokens, contextMargin } = (0, config_1.getContextConfig)();
634
+ const trimmed = (0, context_1.trimMessages)(messages, maxTokens, contextMargin, maxTokens * 2);
635
+ messages.splice(0, messages.length, ...trimmed);
636
+ providerRuntime.resetTurnState(messages);
637
+ callbacks.onError(new Error("context trimmed, retrying..."), "transient");
638
+ return callProviderTurn();
639
+ }
640
+ throw error;
641
+ }
642
+ };
643
+ const attempt = await (0, provider_attempt_1.runProviderAttempt)({
644
+ operation: "turn",
645
+ provider: providerRuntime.id,
646
+ model: providerRuntime.model,
647
+ run: callProviderTurnWithOverflowRecovery,
648
+ classifyError: (error) => providerRuntime.classifyError(error),
649
+ onRetry: (record, maxAttempts) => {
650
+ const delayMs = record.delayMs;
651
+ const seconds = delayMs / 1000;
652
+ const cause = RETRY_LABELS[record.classification];
653
+ callbacks.onError(new Error(`${cause}, retrying in ${seconds}s (${record.attempt}/${maxAttempts})...`), "transient");
654
+ },
655
+ sleep: async (delayMs) => {
656
+ await waitForProviderRetry(delayMs, signal);
657
+ providerRuntime.resetTurnState(messages);
658
+ },
374
659
  });
660
+ if (!attempt.ok) {
661
+ finishTerminalProviderError(attempt.error, attempt.classification);
662
+ continue;
663
+ }
664
+ const result = attempt.value;
375
665
  // Track usage from the latest API call
376
666
  if (result.usage)
377
667
  lastUsage = result.usage;
378
- retryCount = 0; // reset on success
379
668
  // SHARED: build CC-format assistant message from TurnResult
380
669
  const msg = {
381
670
  role: "assistant",
@@ -394,52 +683,76 @@ async function runAgent(messages, callbacks, channel, signal, options) {
394
683
  if (reasoningItems.length > 0) {
395
684
  msg._reasoning_items = reasoningItems;
396
685
  }
686
+ // Store thinking blocks (Anthropic) on the assistant message for round-tripping
687
+ const thinkingItems = result.outputItems.filter((item) => "type" in item && (item.type === "thinking" || item.type === "redacted_thinking"));
688
+ if (thinkingItems.length > 0) {
689
+ msg._thinking_blocks = thinkingItems;
690
+ }
691
+ // Phase annotation for Codex provider
692
+ const hasPhaseAnnotation = providerRuntime.capabilities.has("phase-annotation");
693
+ const isSoleSettle = result.toolCalls.length === 1 && result.toolCalls[0].name === "settle";
694
+ if (hasPhaseAnnotation) {
695
+ msg.phase = isSoleSettle ? "settle" : "commentary";
696
+ }
397
697
  if (!result.toolCalls.length) {
398
- // Kick detection is disabled while tool_choice: required + final_answer
399
- // is the primary loop control mechanism. The model should never reach
400
- // this path (tool_choice: required forces a tool call), but if it does,
401
- // accept the response as-is rather than risk false-positive kicks.
402
- //
403
- // Preserved for future use — re-enable by uncommenting:
404
- // const kick = detectKick(result.content, options);
405
- // if (kick) {
406
- // kickCount++;
407
- // lastKickReason = kick.reason;
408
- // callbacks.onKick?.();
409
- // const kickContent = result.content
410
- // ? result.content + "\n\n" + kick.message
411
- // : kick.message;
412
- // messages.push({ role: "assistant", content: kickContent });
413
- // providerRuntime.resetTurnState(messages);
414
- // continue;
415
- // }
698
+ // No tool calls accept response as-is.
699
+ // (Kick detection disabled; tool_choice: required + settle
700
+ // is the primary loop control. See src/heart/kicks.ts to re-enable.)
416
701
  messages.push(msg);
417
702
  done = true;
418
703
  }
419
704
  else {
420
- // Check for final_answer sole call: intercept before tool execution
421
- const isSoleFinalAnswer = result.toolCalls.length === 1 && result.toolCalls[0].name === "final_answer";
422
- if (isSoleFinalAnswer) {
423
- // Extract answer from the tool call arguments.
424
- // Supports: {"answer":"text"}, "text" (JSON string), retry on failure.
425
- let answer;
426
- try {
427
- const parsed = JSON.parse(result.toolCalls[0].arguments);
428
- if (typeof parsed === "string") {
429
- answer = parsed;
430
- }
431
- else if (parsed.answer != null) {
432
- answer = parsed.answer;
433
- }
434
- // else: valid JSON but no answer field — answer stays undefined (retry)
705
+ // Check for settle sole call: intercept before tool execution
706
+ if (isSoleSettle) {
707
+ /* v8 ignore next -- defensive: JSON.parse catch for malformed settle args @preserve */
708
+ const settleArgs = (() => { try {
709
+ return JSON.parse(result.toolCalls[0].arguments);
435
710
  }
436
711
  catch {
437
- // JSON parsing failed (e.g. truncated output) — answer stays undefined (retry)
712
+ return {};
713
+ } })();
714
+ callbacks.onToolStart("settle", settleArgs);
715
+ // Inner dialog attention queue gate: reject settle if items remain
716
+ const attentionQueue = (augmentedToolContext ?? options?.toolContext)?.delegatedOrigins;
717
+ if (isInnerDialog && attentionQueue && attentionQueue.length > 0) {
718
+ callbacks.onToolEnd("settle", (0, tools_1.summarizeArgs)("settle", settleArgs), false);
719
+ callbacks.onClearText?.();
720
+ messages.push(msg);
721
+ const gateMessage = "you're holding thoughts someone is waiting for — surface them before you settle.";
722
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: gateMessage });
723
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, gateMessage);
724
+ continue;
438
725
  }
439
- if (answer != null) {
440
- if (result.finalAnswerStreamed) {
726
+ // Extract answer from the tool call arguments.
727
+ // Supports: {"answer":"text","intent":"..."} or "text" (JSON string).
728
+ const { answer, intent } = parseSettlePayload(result.toolCalls[0].arguments);
729
+ // Inner dialog settle: no CompletionMetadata, "(settled)" ack
730
+ if (isInnerDialog) {
731
+ callbacks.onToolEnd("settle", (0, tools_1.summarizeArgs)("settle", settleArgs), true);
732
+ messages.push(msg);
733
+ const settled = "(settled)";
734
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: settled });
735
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, settled);
736
+ outcome = "settled";
737
+ done = true;
738
+ continue;
739
+ }
740
+ const retryError = getSettleRetryError(mustResolveBeforeHandoffActive, intent, sawSteeringFollowUp, options?.delegationDecision, sawSendMessageSelf, sawPonder, sawQuerySession, options?.currentObligation ?? null, options?.activeWorkFrame?.inner?.job, sawExternalStateQuery);
741
+ const deliveredAnswer = answer;
742
+ const validDirectReply = mustResolveBeforeHandoffActive && intent === "direct_reply" && sawSteeringFollowUp;
743
+ const validTerminalIntent = intent === "complete" || intent === "blocked";
744
+ const validClosure = deliveredAnswer != null
745
+ && !retryError
746
+ && (!mustResolveBeforeHandoffActive || validDirectReply || validTerminalIntent);
747
+ if (validClosure) {
748
+ callbacks.onToolEnd("settle", (0, tools_1.summarizeArgs)("settle", settleArgs), true);
749
+ completion = {
750
+ answer: deliveredAnswer,
751
+ intent: validDirectReply ? "direct_reply" : intent === "blocked" ? "blocked" : "complete",
752
+ };
753
+ if (result.settleStreamed) {
441
754
  // The streaming layer already parsed and emitted the answer
442
- // progressively via FinalAnswerParser. Skip clearing and
755
+ // progressively via SettleParser. Skip clearing and
443
756
  // re-emitting to avoid double-delivery.
444
757
  }
445
758
  else {
@@ -447,37 +760,110 @@ async function runAgent(messages, callbacks, channel, signal, options) {
447
760
  callbacks.onClearText?.();
448
761
  // Emit the answer through the callback pipeline so channels receive it.
449
762
  // Never truncate -- channel adapters handle splitting long messages.
450
- callbacks.onTextChunk(answer);
763
+ callbacks.onTextChunk(deliveredAnswer);
451
764
  }
452
- // Keep the full assistant message (with tool_calls) for debuggability,
453
- // plus a synthetic tool response so the conversation stays valid on resume.
454
765
  messages.push(msg);
455
- messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: "(delivered)" });
456
- providerRuntime.appendToolOutput(result.toolCalls[0].id, "(delivered)");
457
- done = true;
766
+ if (validDirectReply) {
767
+ const resumeWork = "direct reply delivered. resume the unresolved obligation now and keep working until you can finish or clearly report that you are blocked.";
768
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: resumeWork });
769
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, resumeWork);
770
+ }
771
+ else {
772
+ const delivered = "(delivered)";
773
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: delivered });
774
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, delivered);
775
+ outcome = intent === "blocked" ? "blocked" : "settled";
776
+ done = true;
777
+ }
458
778
  }
459
779
  else {
460
- // Answer is undefined -- the model's final_answer was incomplete or
780
+ // Answer is undefined -- the model's settle was incomplete or
461
781
  // malformed. Clear any partial streamed text or noise, then push the
462
782
  // assistant msg + error tool result and let the model try again.
783
+ callbacks.onToolEnd("settle", (0, tools_1.summarizeArgs)("settle", settleArgs), false);
463
784
  callbacks.onClearText?.();
464
- const retryError = "your final_answer was incomplete or malformed. call final_answer again with your complete response.";
465
785
  messages.push(msg);
466
- messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: retryError });
467
- providerRuntime.appendToolOutput(result.toolCalls[0].id, retryError);
786
+ const toolRetryMessage = retryError
787
+ ?? "your settle was incomplete or malformed. call settle again with your complete response.";
788
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: toolRetryMessage });
789
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, toolRetryMessage);
468
790
  }
469
791
  continue;
470
792
  }
793
+ // Check for observe sole call: intercept before tool execution
794
+ const isSoleObserve = result.toolCalls.length === 1 && result.toolCalls[0].name === "observe";
795
+ if (isSoleObserve) {
796
+ /* v8 ignore next -- defensive: JSON.parse catch for malformed observe args @preserve */
797
+ const observeArgs = (() => { try {
798
+ return JSON.parse(result.toolCalls[0].arguments);
799
+ }
800
+ catch {
801
+ return {};
802
+ } })();
803
+ let reason;
804
+ if (typeof observeArgs?.reason === "string")
805
+ reason = observeArgs.reason;
806
+ callbacks.onToolStart("observe", observeArgs);
807
+ (0, runtime_1.emitNervesEvent)({
808
+ component: "engine",
809
+ event: "engine.observe",
810
+ message: "agent observed without responding",
811
+ meta: { ...(reason ? { reason } : {}) },
812
+ });
813
+ callbacks.onToolEnd("observe", (0, tools_1.summarizeArgs)("observe", observeArgs), true);
814
+ messages.push(msg);
815
+ const silenced = "(silenced)";
816
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: silenced });
817
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, silenced);
818
+ outcome = "observed";
819
+ done = true;
820
+ continue;
821
+ }
822
+ // Check for rest sole call: intercept before tool execution
823
+ const isSoleRest = result.toolCalls.length === 1 && result.toolCalls[0].name === "rest";
824
+ if (isSoleRest) {
825
+ const restArgs = (() => { try {
826
+ return JSON.parse(result.toolCalls[0].arguments);
827
+ }
828
+ catch {
829
+ return {};
830
+ } })();
831
+ callbacks.onToolStart("rest", restArgs);
832
+ // Attention queue gate: reject rest if items remain
833
+ const attentionQueue = (augmentedToolContext ?? options?.toolContext)?.delegatedOrigins;
834
+ if (attentionQueue && attentionQueue.length > 0) {
835
+ callbacks.onToolEnd("rest", (0, tools_1.summarizeArgs)("rest", restArgs), false);
836
+ messages.push(msg);
837
+ const gateMessage = "you're holding thoughts someone is waiting for — surface them before you rest.";
838
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: gateMessage });
839
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, gateMessage);
840
+ continue;
841
+ }
842
+ callbacks.onToolEnd("rest", (0, tools_1.summarizeArgs)("rest", restArgs), true);
843
+ messages.push(msg);
844
+ const ack = "(resting)";
845
+ messages.push({ role: "tool", tool_call_id: result.toolCalls[0].id, content: ack });
846
+ providerRuntime.appendToolOutput(result.toolCalls[0].id, ack);
847
+ (0, runtime_1.emitNervesEvent)({
848
+ component: "engine",
849
+ event: "engine.rested",
850
+ message: "resting until next heartbeat",
851
+ meta: { ...(typeof restArgs?.status === "string" ? { status: restArgs.status } : {}) },
852
+ });
853
+ outcome = "rested";
854
+ done = true;
855
+ continue;
856
+ }
471
857
  messages.push(msg);
472
- // SHARED: execute tools (final_answer in mixed calls is rejected inline)
858
+ // Execute tools (sole-call tools in mixed calls are rejected inline)
473
859
  for (const tc of result.toolCalls) {
474
860
  if (signal?.aborted)
475
861
  break;
476
- // Intercept final_answer in mixed call: reject it
477
- if (tc.name === "final_answer") {
478
- const rejection = "rejected: final_answer must be the only tool call. Finish your work first, then call final_answer alone.";
479
- messages.push({ role: "tool", tool_call_id: tc.id, content: rejection });
480
- providerRuntime.appendToolOutput(tc.id, rejection);
862
+ // Reject sole-call tools when mixed with other tool calls
863
+ const soleCallRejection = SOLE_CALL_REJECTION[tc.name];
864
+ if (soleCallRejection) {
865
+ messages.push({ role: "tool", tool_call_id: tc.id, content: soleCallRejection });
866
+ providerRuntime.appendToolOutput(tc.id, soleCallRejection);
481
867
  continue;
482
868
  }
483
869
  let args = {};
@@ -487,94 +873,195 @@ async function runAgent(messages, callbacks, channel, signal, options) {
487
873
  catch {
488
874
  /* ignore */
489
875
  }
490
- const argSummary = (0, tools_1.summarizeArgs)(tc.name, args);
491
- // Confirmation check for mutate tools
492
- if ((0, tools_1.isConfirmationRequired)(tc.name) && !options?.skipConfirmation) {
493
- let decision = "denied";
494
- if (callbacks.onConfirmAction) {
495
- decision = await callbacks.onConfirmAction(tc.name, args);
876
+ if (tc.name === "send_message" && args.friendId === "self") {
877
+ sawSendMessageSelf = true;
878
+ }
879
+ if (tc.name === "ponder") {
880
+ const parsedArgs = normalizeLegacyPonderArgs(parsePonderPayload(tc.arguments));
881
+ const argSummary = (0, tools_1.summarizeArgs)(tc.name, parsedArgs);
882
+ callbacks.onToolStart(tc.name, parsedArgs);
883
+ let toolResult;
884
+ let success = false;
885
+ try {
886
+ const action = parsedArgs.action ?? "create";
887
+ const currentSession = (augmentedToolContext ?? options?.toolContext)?.currentSession;
888
+ const currentOrigin = currentSession
889
+ ? { friendId: currentSession.friendId, channel: currentSession.channel, key: currentSession.key }
890
+ : undefined;
891
+ const isInnerChannel = currentOrigin?.friendId === "self" && currentOrigin?.channel === "inner";
892
+ const successCriteria = parseSuccessCriteria(parsedArgs.success_criteria);
893
+ const payload = parsePacketPayload(parsedArgs.payload_json);
894
+ let packet;
895
+ let returnObligationId = null;
896
+ let resultAction = "created";
897
+ if (action === "create") {
898
+ const kind = parsedArgs.kind;
899
+ const objective = typeof parsedArgs.objective === "string" ? parsedArgs.objective.trim() : "";
900
+ const summary = typeof parsedArgs.summary === "string" ? parsedArgs.summary.trim() : "";
901
+ if (!kind || !objective || !successCriteria || !payload) {
902
+ throw new Error("ponder create requires kind, objective, success_criteria, and valid payload_json.");
903
+ }
904
+ const agentRoot = (0, identity_2.getAgentRoot)();
905
+ let relatedObligationId;
906
+ if (currentOrigin && !isInnerChannel) {
907
+ try {
908
+ const obligation = (0, obligations_1.createObligation)(agentRoot, {
909
+ origin: currentOrigin,
910
+ content: objective,
911
+ });
912
+ relatedObligationId = obligation.id;
913
+ }
914
+ catch {
915
+ relatedObligationId = undefined;
916
+ }
917
+ }
918
+ const frictionSignature = kind === "harness_friction" && typeof payload.frictionSignature === "string"
919
+ ? payload.frictionSignature
920
+ : null;
921
+ const existing = frictionSignature && currentOrigin
922
+ ? (0, packets_1.findHarnessFrictionPacket)(agentRoot, currentOrigin, frictionSignature)
923
+ : null;
924
+ if (existing) {
925
+ resultAction = "revised";
926
+ returnObligationId = existing.relatedReturnObligationId ?? null;
927
+ packet = existing.status === "drafting"
928
+ ? (0, packets_1.revisePonderPacket)(agentRoot, existing.id, {
929
+ kind,
930
+ objective,
931
+ summary,
932
+ successCriteria,
933
+ payload,
934
+ })
935
+ : existing;
936
+ }
937
+ else {
938
+ returnObligationId = (0, obligations_1.generateObligationId)(Date.now());
939
+ packet = (0, packets_1.createPonderPacket)(agentRoot, {
940
+ kind,
941
+ objective,
942
+ summary,
943
+ successCriteria,
944
+ ...(currentOrigin ? { origin: currentOrigin } : {}),
945
+ ...(relatedObligationId ? { relatedObligationId } : {}),
946
+ relatedReturnObligationId: returnObligationId,
947
+ ...(parsedArgs.follows_packet_id ? { followsPacketId: parsedArgs.follows_packet_id } : {}),
948
+ payload,
949
+ });
950
+ (0, obligations_1.createReturnObligation)((0, identity_2.getAgentName)(), {
951
+ id: returnObligationId,
952
+ origin: currentOrigin ?? { friendId: "self", channel: "inner", key: "dialog" },
953
+ status: "queued",
954
+ delegatedContent: (summary || objective).length > 120 ? `${(summary || objective).slice(0, 117)}...` : (summary || objective),
955
+ packetId: packet.id,
956
+ createdAt: Date.now(),
957
+ });
958
+ }
959
+ }
960
+ else if (action === "revise") {
961
+ const packetId = typeof parsedArgs.packet_id === "string" ? parsedArgs.packet_id.trim() : "";
962
+ const kind = parsedArgs.kind;
963
+ const objective = typeof parsedArgs.objective === "string" ? parsedArgs.objective.trim() : "";
964
+ const summary = typeof parsedArgs.summary === "string" ? parsedArgs.summary.trim() : "";
965
+ if (!packetId || !kind || !objective || !successCriteria || !payload) {
966
+ throw new Error("ponder revise requires packet_id, kind, objective, success_criteria, and valid payload_json.");
967
+ }
968
+ packet = (0, packets_1.revisePonderPacket)((0, identity_2.getAgentRoot)(), packetId, {
969
+ kind,
970
+ objective,
971
+ summary,
972
+ successCriteria,
973
+ payload,
974
+ });
975
+ returnObligationId = packet.relatedReturnObligationId ?? null;
976
+ resultAction = "revised";
977
+ }
978
+ else {
979
+ throw new Error("ponder requires action=create or revise.");
980
+ }
981
+ try {
982
+ await (0, socket_client_1.requestInnerWake)((0, identity_2.getAgentName)());
983
+ }
984
+ catch { /* daemon may not be running */ }
985
+ sawPonder = true;
986
+ toolResult = buildPonderResult(packet, resultAction, returnObligationId);
987
+ success = true;
988
+ (0, runtime_1.emitNervesEvent)({
989
+ component: "engine",
990
+ event: "engine.ponder_packet",
991
+ message: "ponder packet touched",
992
+ meta: {
993
+ action: resultAction,
994
+ packetId: packet.id,
995
+ kind: packet.kind,
996
+ status: packet.status,
997
+ },
998
+ });
496
999
  }
497
- if (decision !== "confirmed") {
498
- const cancelled = "Action cancelled by user.";
499
- callbacks.onToolStart(tc.name, args);
500
- callbacks.onToolEnd(tc.name, argSummary, false);
501
- messages.push({ role: "tool", tool_call_id: tc.id, content: cancelled });
502
- providerRuntime.appendToolOutput(tc.id, cancelled);
503
- continue;
1000
+ catch (error) {
1001
+ toolResult = error instanceof Error ? error.message : String(error);
504
1002
  }
1003
+ callbacks.onToolEnd(tc.name, argSummary, success);
1004
+ messages.push({ role: "tool", tool_call_id: tc.id, content: toolResult });
1005
+ providerRuntime.appendToolOutput(tc.id, toolResult);
1006
+ continue;
1007
+ }
1008
+ /* v8 ignore next -- flag tested via truth-check integration tests @preserve */
1009
+ if (tc.name === "query_session")
1010
+ sawQuerySession = true;
1011
+ /* v8 ignore next -- flag tested via truth-check integration tests @preserve */
1012
+ if (tc.name === "bridge_manage")
1013
+ sawBridgeManage = true;
1014
+ /* v8 ignore next -- flag tested via truth-check integration tests @preserve */
1015
+ if (isExternalStateQuery(tc.name, args))
1016
+ sawExternalStateQuery = true;
1017
+ const argSummary = (0, tools_1.summarizeArgs)(tc.name, args);
1018
+ const toolLoop = (0, tool_loop_1.detectToolLoop)(toolLoopState, tc.name, args);
1019
+ if (toolLoop.stuck) {
1020
+ const rejection = `loop guard: ${toolLoop.message}`;
1021
+ callbacks.onToolStart(tc.name, args);
1022
+ callbacks.onToolEnd(tc.name, argSummary, false);
1023
+ messages.push({ role: "tool", tool_call_id: tc.id, content: rejection });
1024
+ providerRuntime.appendToolOutput(tc.id, rejection);
1025
+ continue;
505
1026
  }
506
1027
  callbacks.onToolStart(tc.name, args);
507
1028
  let toolResult;
508
1029
  let success;
509
1030
  try {
510
1031
  const execToolFn = options?.execTool ?? tools_1.execTool;
511
- toolResult = await execToolFn(tc.name, args, options?.toolContext);
1032
+ toolResult = await execToolFn(tc.name, args, augmentedToolContext ?? options?.toolContext);
512
1033
  success = true;
513
1034
  }
514
1035
  catch (e) {
515
1036
  toolResult = `error: ${e}`;
516
1037
  success = false;
517
1038
  }
518
- callbacks.onToolEnd(tc.name, argSummary, success);
1039
+ toolResult = (0, tool_friction_1.rewriteToolResultForModel)(tc.name, toolResult, toolFrictionLedger);
1040
+ (0, tool_loop_1.recordToolOutcome)(toolLoopState, tc.name, args, toolResult, success);
1041
+ callbacks.onToolEnd(tc.name, (0, tools_1.buildToolResultSummary)(tc.name, args, toolResult, success), success);
519
1042
  messages.push({ role: "tool", tool_call_id: tc.id, content: toolResult });
520
1043
  providerRuntime.appendToolOutput(tc.id, toolResult);
1044
+ callbacks.onToolResult?.(messages);
521
1045
  }
522
1046
  }
523
1047
  }
524
1048
  catch (e) {
525
1049
  // Abort is not an error — just stop cleanly
526
- if (signal?.aborted) {
1050
+ if (e instanceof provider_attempt_1.ProviderAttemptAbortError || signal?.aborted) {
527
1051
  stripLastToolCalls(messages);
1052
+ outcome = "aborted";
528
1053
  break;
529
1054
  }
530
- // Context overflow: trim aggressively and retry once
531
- if (isContextOverflow(e) && !overflowRetried) {
532
- overflowRetried = true;
533
- stripLastToolCalls(messages);
534
- const { maxTokens, contextMargin } = (0, config_1.getContextConfig)();
535
- const trimmed = (0, context_1.trimMessages)(messages, maxTokens, contextMargin, maxTokens * 2);
536
- messages.splice(0, messages.length, ...trimmed);
537
- providerRuntime.resetTurnState(messages);
538
- callbacks.onError(new Error("context trimmed, retrying..."), "transient");
539
- continue;
1055
+ const errorForClassification = e instanceof Error ? e : /* v8 ignore next -- defensive @preserve */ new Error(String(e));
1056
+ let providerClassification;
1057
+ try {
1058
+ providerClassification = providerRuntime.classifyError(errorForClassification);
540
1059
  }
541
- // Transient errors: retry with exponential backoff
542
- if (isTransientError(e) && retryCount < MAX_RETRIES) {
543
- retryCount++;
544
- const delay = RETRY_BASE_MS * Math.pow(2, retryCount - 1);
545
- const cause = classifyTransientError(e);
546
- callbacks.onError(new Error(`${cause}, retrying in ${delay / 1000}s (${retryCount}/${MAX_RETRIES})...`), "transient");
547
- // Wait with abort support
548
- const aborted = await new Promise((resolve) => {
549
- const timer = setTimeout(() => resolve(false), delay);
550
- if (signal) {
551
- const onAbort = () => { clearTimeout(timer); resolve(true); };
552
- if (signal.aborted) {
553
- clearTimeout(timer);
554
- resolve(true);
555
- return;
556
- }
557
- signal.addEventListener("abort", onAbort, { once: true });
558
- }
559
- });
560
- if (aborted) {
561
- stripLastToolCalls(messages);
562
- break;
563
- }
564
- providerRuntime.resetTurnState(messages);
565
- continue;
1060
+ catch {
1061
+ /* v8 ignore next -- defensive: classifyError should not throw @preserve */
1062
+ providerClassification = "unknown";
566
1063
  }
567
- callbacks.onError(e instanceof Error ? e : new Error(String(e)), "terminal");
568
- (0, runtime_1.emitNervesEvent)({
569
- level: "error",
570
- event: "engine.error",
571
- trace_id: traceId,
572
- component: "engine",
573
- message: e instanceof Error ? e.message : String(e),
574
- meta: {},
575
- });
576
- stripLastToolCalls(messages);
577
- done = true;
1064
+ finishTerminalProviderError(errorForClassification, providerClassification);
578
1065
  }
579
1066
  }
580
1067
  (0, runtime_1.emitNervesEvent)({
@@ -582,7 +1069,12 @@ async function runAgent(messages, callbacks, channel, signal, options) {
582
1069
  trace_id: traceId,
583
1070
  component: "engine",
584
1071
  message: "runAgent turn completed",
585
- meta: { done },
1072
+ meta: { done, sawPonder, sawQuerySession, sawBridgeManage },
586
1073
  });
587
- return { usage: lastUsage };
1074
+ return {
1075
+ usage: lastUsage,
1076
+ outcome,
1077
+ completion,
1078
+ ...(terminalError ? { error: terminalError, errorClassification: terminalErrorClassification } : {}),
1079
+ };
588
1080
  }