@xopcai/xopc 0.0.87 → 0.0.88

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 (385) hide show
  1. package/dist/browser-ext/manifest.json +1 -1
  2. package/dist/extensions/feishu/src/outbound/media-load.js +1 -1
  3. package/dist/extensions/feishu/src/workflow-progress.js +1 -1
  4. package/dist/extensions/telegram/src/plugin.js +1 -1
  5. package/dist/extensions/telegram/src/routing-integration.js +2 -2
  6. package/dist/extensions/telegram/src/workflow-progress.js +1 -1
  7. package/dist/extensions/telegram/xopc.extension.json +1 -1
  8. package/dist/extensions/weixin/src/api/api.js +2 -2
  9. package/dist/extensions/weixin/src/auth/accounts.js +1 -1
  10. package/dist/extensions/weixin/src/cdn/upload.js +1 -1
  11. package/dist/extensions/weixin/src/media/data-url.js +1 -1
  12. package/dist/extensions/weixin/src/messaging/debug-mode.js +1 -1
  13. package/dist/extensions/weixin/src/messaging/inbound.js +1 -1
  14. package/dist/extensions/weixin/src/messaging/process-message.js +1 -1
  15. package/dist/extensions/weixin/src/plugin.js +1 -1
  16. package/dist/extensions/weixin/src/storage/sync-buf.js +1 -1
  17. package/dist/extensions/weixin/src/workflow-progress.js +1 -1
  18. package/dist/gateway/static/root/assets/agents-CRxETUZx.js +222 -0
  19. package/dist/gateway/static/root/assets/{apps-page-Dg8R-Szf.js → apps-page-wKWf3l57.js} +1 -1
  20. package/dist/gateway/static/root/assets/channels-settings-DDbqVNkx.js +1 -0
  21. package/dist/gateway/static/root/assets/{channels-status-swr-BSHqqCF1.js → channels-status-swr-DIsl75Y3.js} +1 -1
  22. package/dist/gateway/static/root/assets/copy-SxMW6Xpc.js +1 -0
  23. package/dist/gateway/static/root/assets/{cron-api-0h_QT8U3.js → cron-api-N9hvuRrn.js} +1 -1
  24. package/dist/gateway/static/root/assets/{cron-page-BkfKFfFk.js → cron-page-tlNGNxhP.js} +1 -1
  25. package/dist/gateway/static/root/assets/{dist-Cmjp2APP.js → dist-CJwfHYvT.js} +1 -1
  26. package/dist/gateway/static/root/assets/{extension-debug-page-CFa9z_1N.js → extension-debug-page-BVJohZoZ.js} +1 -1
  27. package/dist/gateway/static/root/assets/{extension-page-BI8eaTPq.js → extension-page-BT2tmElC.js} +1 -1
  28. package/dist/gateway/static/root/assets/extension-settings-page-BSS47c2j.js +1 -0
  29. package/dist/gateway/static/root/assets/{fetch-DRqwef_Q.js → fetch-BaFNUtkE.js} +1 -1
  30. package/dist/gateway/static/root/assets/{field-primitives-BiNHBo2Y.js → field-primitives-QwYEq6Hz.js} +1 -1
  31. package/dist/gateway/static/root/assets/{heartbeat-config-api-ZRb8qhuz.js → heartbeat-config-api-BVSidEDJ.js} +1 -1
  32. package/dist/gateway/static/root/assets/index-CqZzHNEg.css +1 -0
  33. package/dist/gateway/static/root/assets/{index-Cu7bKuUi.js → index-qNrVJp-y.js} +97 -97
  34. package/dist/gateway/static/root/assets/{logs-page-BFZ8GgCv.js → logs-page-DDonPVLn.js} +1 -1
  35. package/dist/gateway/static/root/assets/sessions-page-DKt-Wmib.js +1 -0
  36. package/dist/gateway/static/root/assets/{settings-form-section-DiqqVs6m.js → settings-form-section-B8N3A3Zo.js} +1 -1
  37. package/dist/gateway/static/root/assets/settings-page-DcJjvvw4.js +3 -0
  38. package/dist/gateway/static/root/assets/{share-preview-page-n1Gprylk.js → share-preview-page-Q7KqkO-u.js} +1 -1
  39. package/dist/gateway/static/root/assets/skills-page-DuJ4BTO3.js +2 -0
  40. package/dist/gateway/static/root/assets/{theme-store-CZOh1nT3.js → theme-store-BbRc5ugR.js} +1 -1
  41. package/dist/gateway/static/root/assets/url-D6jvVYIA.js +7 -0
  42. package/dist/gateway/static/root/assets/{utils-CkWBfxs4.js → utils-CxDGduqK.js} +1 -1
  43. package/dist/gateway/static/root/assets/voice-api-key-field-CTyHz7L_.js +1 -0
  44. package/dist/gateway/static/root/assets/workflows-page-GacJ41Fv.js +27 -0
  45. package/dist/gateway/static/root/index.html +6 -5
  46. package/dist/package.js +1 -1
  47. package/dist/src/agent/agent-manager.js +7 -7
  48. package/dist/src/agent/agent-scope.js +1 -1
  49. package/dist/src/agent/bootstrap/load-bootstrap-files.js +1 -1
  50. package/dist/src/agent/child-agent-factory.d.ts +15 -0
  51. package/dist/src/agent/child-agent-factory.js +35 -2
  52. package/dist/src/agent/child-agent-factory.js.map +1 -1
  53. package/dist/src/agent/client-error-format.d.ts +20 -0
  54. package/dist/src/agent/client-error-format.js +97 -0
  55. package/dist/src/agent/client-error-format.js.map +1 -0
  56. package/dist/src/agent/context/workspace-seed.js +2 -2
  57. package/dist/src/agent/embedded/run-turn.js +23 -4
  58. package/dist/src/agent/embedded/run-turn.js.map +1 -1
  59. package/dist/src/agent/goals/goal-locale.d.ts +1 -1
  60. package/dist/src/agent/goals/goal-run-store.js +4 -4
  61. package/dist/src/agent/goals/persistent-goal-service.js +1 -1
  62. package/dist/src/agent/goals/post-turn.js +2 -2
  63. package/dist/src/agent/image/load-image-media.js +2 -2
  64. package/dist/src/agent/inbound/turn-dispatcher.js +1 -1
  65. package/dist/src/agent/inbound/turn-dispatcher.js.map +1 -1
  66. package/dist/src/agent/ipc/bus.js +1 -1
  67. package/dist/src/agent/ipc/inbox.js +2 -2
  68. package/dist/src/agent/ipc/socket.js +1 -1
  69. package/dist/src/agent/mcp/bundle-mcp-materialize.js +1 -1
  70. package/dist/src/agent/mcp/bundle-mcp-runtime.js +1 -1
  71. package/dist/src/agent/mcp/mcp-transport-config.js +1 -1
  72. package/dist/src/agent/mcp/mcp-transport.js +1 -1
  73. package/dist/src/agent/memory/builtin-memory-store.js +1 -1
  74. package/dist/src/agent/memory/dreaming/deep-promotion.js +1 -1
  75. package/dist/src/agent/memory/dreaming/events.js +1 -1
  76. package/dist/src/agent/memory/dreaming/last-run.js +1 -1
  77. package/dist/src/agent/memory/dreaming/light-sweep.js +1 -1
  78. package/dist/src/agent/memory/dreaming/preview.js +1 -1
  79. package/dist/src/agent/memory/dreaming/rem-patterns.js +1 -1
  80. package/dist/src/agent/memory/dreaming/short-term-store.js +1 -1
  81. package/dist/src/agent/memory/dreaming/utils.js +1 -1
  82. package/dist/src/agent/memory/plugin-discovery.js +1 -1
  83. package/dist/src/agent/models/manager.js +1 -1
  84. package/dist/src/agent/orchestration/llm-turn-retry.d.ts +2 -0
  85. package/dist/src/agent/orchestration/llm-turn-retry.js +9 -1
  86. package/dist/src/agent/orchestration/llm-turn-retry.js.map +1 -1
  87. package/dist/src/agent/prompt/service-prompt-builder.js +2 -2
  88. package/dist/src/agent/reply/post-compaction-context.js +1 -1
  89. package/dist/src/agent/reply/workspace-boundary-read.js +1 -1
  90. package/dist/src/agent/sandbox/path-policy.js +2 -2
  91. package/dist/src/agent/service/build-direct-message-content.js +1 -1
  92. package/dist/src/agent/service/process-direct-streaming.js +19 -3
  93. package/dist/src/agent/service/process-direct-streaming.js.map +1 -1
  94. package/dist/src/agent/service/webchat-tts.d.ts +1 -2
  95. package/dist/src/agent/service/webchat-tts.js +1 -1
  96. package/dist/src/agent/service/webchat-tts.js.map +1 -1
  97. package/dist/src/agent/service.js +4 -4
  98. package/dist/src/agent/session/session-inspector.js +1 -1
  99. package/dist/src/agent/skills/config.js +1 -1
  100. package/dist/src/agent/skills/hub-hash.js +2 -2
  101. package/dist/src/agent/skills/hub-lock.js +1 -1
  102. package/dist/src/agent/skills/hub-pull.js +2 -2
  103. package/dist/src/agent/skills/index.js +1 -1
  104. package/dist/src/agent/skills/managed-store.js +1 -1
  105. package/dist/src/agent/skills/scanner.js +1 -1
  106. package/dist/src/agent/skills/skill-manage-ops.js +1 -1
  107. package/dist/src/agent/skills/skill-manager.js +1 -1
  108. package/dist/src/agent/tools/dreaming-tool.js +1 -1
  109. package/dist/src/agent/tools/factory.js +1 -1
  110. package/dist/src/agent/tools/image-generate-tool.js +1 -1
  111. package/dist/src/agent/tools/send-media.js +1 -1
  112. package/dist/src/agent/tools/skill-manage-tool.js +1 -1
  113. package/dist/src/agent/tools/workflow-tool.js +64 -16
  114. package/dist/src/agent/tools/workflow-tool.js.map +1 -1
  115. package/dist/src/agent/tools/write.js +1 -1
  116. package/dist/src/agent/workflow/agent-progress.d.ts +5 -0
  117. package/dist/src/agent/workflow/agent-progress.js +65 -0
  118. package/dist/src/agent/workflow/agent-progress.js.map +1 -0
  119. package/dist/src/agent/workflow/builtins/audit-repo.d.ts +1 -1
  120. package/dist/src/agent/workflow/builtins/audit-repo.js +14 -0
  121. package/dist/src/agent/workflow/builtins/audit-repo.js.map +1 -1
  122. package/dist/src/agent/workflow/builtins/debug-incident.d.ts +1 -1
  123. package/dist/src/agent/workflow/builtins/debug-incident.js +14 -0
  124. package/dist/src/agent/workflow/builtins/debug-incident.js.map +1 -1
  125. package/dist/src/agent/workflow/builtins/implementation-plan.d.ts +12 -0
  126. package/dist/src/agent/workflow/builtins/implementation-plan.js +175 -0
  127. package/dist/src/agent/workflow/builtins/implementation-plan.js.map +1 -0
  128. package/dist/src/agent/workflow/builtins/index.d.ts +3 -1
  129. package/dist/src/agent/workflow/builtins/index.js +11 -1
  130. package/dist/src/agent/workflow/builtins/index.js.map +1 -1
  131. package/dist/src/agent/workflow/builtins/multi-perspective-review.d.ts +1 -1
  132. package/dist/src/agent/workflow/builtins/multi-perspective-review.js +14 -0
  133. package/dist/src/agent/workflow/builtins/multi-perspective-review.js.map +1 -1
  134. package/dist/src/agent/workflow/builtins/pr-review.d.ts +1 -1
  135. package/dist/src/agent/workflow/builtins/pr-review.js +14 -0
  136. package/dist/src/agent/workflow/builtins/pr-review.js.map +1 -1
  137. package/dist/src/agent/workflow/builtins/release-check.d.ts +11 -0
  138. package/dist/src/agent/workflow/builtins/release-check.js +165 -0
  139. package/dist/src/agent/workflow/builtins/release-check.js.map +1 -0
  140. package/dist/src/agent/workflow/builtins/research.d.ts +1 -1
  141. package/dist/src/agent/workflow/builtins/research.js +14 -0
  142. package/dist/src/agent/workflow/builtins/research.js.map +1 -1
  143. package/dist/src/agent/workflow/catalog.js +1 -1
  144. package/dist/src/agent/workflow/index.d.ts +2 -1
  145. package/dist/src/agent/workflow/index.js +3 -2
  146. package/dist/src/agent/workflow/meta-locale.d.ts +12 -0
  147. package/dist/src/agent/workflow/meta-locale.js +62 -0
  148. package/dist/src/agent/workflow/meta-locale.js.map +1 -0
  149. package/dist/src/agent/workflow/parser.js +3 -0
  150. package/dist/src/agent/workflow/parser.js.map +1 -1
  151. package/dist/src/agent/workflow/runtime.d.ts +2 -2
  152. package/dist/src/agent/workflow/runtime.js +21 -14
  153. package/dist/src/agent/workflow/runtime.js.map +1 -1
  154. package/dist/src/agent/workflow/snapshot.js +2 -12
  155. package/dist/src/agent/workflow/snapshot.js.map +1 -1
  156. package/dist/src/agent/workflow/step-labels.d.ts +8 -0
  157. package/dist/src/agent/workflow/step-labels.js +48 -0
  158. package/dist/src/agent/workflow/step-labels.js.map +1 -0
  159. package/dist/src/agent/workflow/subagent-runner.js +46 -1
  160. package/dist/src/agent/workflow/subagent-runner.js.map +1 -1
  161. package/dist/src/agent/workflow/types.d.ts +74 -1
  162. package/dist/src/auth/credentials.d.ts +5 -0
  163. package/dist/src/auth/credentials.js +12 -3
  164. package/dist/src/auth/credentials.js.map +1 -1
  165. package/dist/src/auth/profiles/store.js +1 -1
  166. package/dist/src/auth/sync-provider-auth.js +1 -1
  167. package/dist/src/browser/cache-dir-policy.js +1 -1
  168. package/dist/src/browser/cdp-local-launcher.js +2 -2
  169. package/dist/src/browser/providers/browser-ext-install.js +3 -3
  170. package/dist/src/browser/providers/cloakbrowser.js +4 -4
  171. package/dist/src/browser/providers/playwright-doctor.js +1 -1
  172. package/dist/src/browser/stealth.js +1 -1
  173. package/dist/src/channels/attachments/inbound-persist.js +1 -1
  174. package/dist/src/channels/attachments/outbound-tts-persist.js +1 -1
  175. package/dist/src/channels/outbound/persist-store.js +1 -1
  176. package/dist/src/channels/pairing/allow-from-file.js +1 -1
  177. package/dist/src/channels/pairing/pairing-store.js +2 -2
  178. package/dist/src/chat-commands/builtins/config.js +2 -2
  179. package/dist/src/chat-commands/context.js +1 -1
  180. package/dist/src/cli/commands/config.js +1 -1
  181. package/dist/src/cli/commands/doctor/checks/config-health.js +1 -1
  182. package/dist/src/cli/commands/doctor/checks/provider-auth.js +1 -1
  183. package/dist/src/cli/commands/doctor/checks/session-integrity.js +1 -1
  184. package/dist/src/cli/commands/doctor/checks/state-integrity.js +1 -1
  185. package/dist/src/cli/commands/doctor/checks/workspace-status.js +1 -1
  186. package/dist/src/cli/commands/extension-dev.js +1 -1
  187. package/dist/src/cli/commands/extension-marketplace.js +1 -1
  188. package/dist/src/cli/commands/extension-pack.js +1 -1
  189. package/dist/src/cli/commands/gateway/logs.js +1 -1
  190. package/dist/src/cli/commands/image.js +1 -1
  191. package/dist/src/cli/commands/init.js +4 -4
  192. package/dist/src/cli/commands/onboard.js +1 -1
  193. package/dist/src/cli/utils/init-workspace-core.js +2 -2
  194. package/dist/src/config/agent-profile.js +1 -1
  195. package/dist/src/config/agent-typed-models.d.ts +18 -0
  196. package/dist/src/config/agent-typed-models.js +53 -0
  197. package/dist/src/config/agent-typed-models.js.map +1 -0
  198. package/dist/src/config/gateway-bind.js +1 -1
  199. package/dist/src/config/index.js +6 -6
  200. package/dist/src/config/loader.js +2 -2
  201. package/dist/src/config/models-json.js +2 -2
  202. package/dist/src/config/paths-state.js +1 -1
  203. package/dist/src/config/profile.js +2 -2
  204. package/dist/src/config/schema.d.ts +52 -0
  205. package/dist/src/config/schema.js +39 -3
  206. package/dist/src/config/schema.js.map +1 -1
  207. package/dist/src/config/voice.d.ts +3 -28
  208. package/dist/src/config/voice.js +27 -261
  209. package/dist/src/config/voice.js.map +1 -1
  210. package/dist/src/config/workspace-path.js +1 -1
  211. package/dist/src/cron/executor.js +2 -2
  212. package/dist/src/cron/persistence.js +1 -1
  213. package/dist/src/cron/run-log-store.js +1 -1
  214. package/dist/src/daemon/constants.js +1 -1
  215. package/dist/src/daemon/install-plan.js +2 -2
  216. package/dist/src/daemon/launchd.js +2 -2
  217. package/dist/src/daemon/schtasks.js +2 -2
  218. package/dist/src/daemon/systemd.js +2 -2
  219. package/dist/src/extensions/bundle-mcp.js +1 -1
  220. package/dist/src/extensions/discover-extensions.js +1 -1
  221. package/dist/src/extensions/health.js +1 -1
  222. package/dist/src/extensions/loader.js +1 -1
  223. package/dist/src/extensions/lockfile.js +2 -2
  224. package/dist/src/gateway/agents-admin.d.ts +9 -0
  225. package/dist/src/gateway/agents-admin.js +18 -2
  226. package/dist/src/gateway/agents-admin.js.map +1 -1
  227. package/dist/src/gateway/config-tools-web.js +3 -2
  228. package/dist/src/gateway/config-tools-web.js.map +1 -1
  229. package/dist/src/gateway/file-path-classifier.js +2 -2
  230. package/dist/src/gateway/hono/lib/agent-model.d.ts +7 -0
  231. package/dist/src/gateway/hono/lib/agent-model.js +36 -1
  232. package/dist/src/gateway/hono/lib/agent-model.js.map +1 -1
  233. package/dist/src/gateway/hono/lib/config-payload.js +28 -5
  234. package/dist/src/gateway/hono/lib/config-payload.js.map +1 -1
  235. package/dist/src/gateway/hono/lib/extension-store.js +2 -2
  236. package/dist/src/gateway/hono/lib/mask-secret-length.d.ts +6 -0
  237. package/dist/src/gateway/hono/lib/mask-secret-length.js +16 -0
  238. package/dist/src/gateway/hono/lib/mask-secret-length.js.map +1 -0
  239. package/dist/src/gateway/hono/lib/safe-providers-config.d.ts +1 -1
  240. package/dist/src/gateway/hono/lib/safe-providers-config.js +2 -1
  241. package/dist/src/gateway/hono/lib/safe-providers-config.js.map +1 -1
  242. package/dist/src/gateway/hono/lib/safe-voice-config.js +2 -1
  243. package/dist/src/gateway/hono/lib/safe-voice-config.js.map +1 -1
  244. package/dist/src/gateway/hono/lib/static-ui.js +2 -2
  245. package/dist/src/gateway/hono/oauth.js +1 -1
  246. package/dist/src/gateway/hono/routes/agents.js +2 -2
  247. package/dist/src/gateway/hono/routes/auth-registry-extensions.js +1 -1
  248. package/dist/src/gateway/hono/routes/config-patch/agents.js +8 -2
  249. package/dist/src/gateway/hono/routes/config-patch/agents.js.map +1 -1
  250. package/dist/src/gateway/hono/routes/config-patch/gateway.js +3 -2
  251. package/dist/src/gateway/hono/routes/config-patch/gateway.js.map +1 -1
  252. package/dist/src/gateway/hono/routes/config-patch/misc.js +8 -3
  253. package/dist/src/gateway/hono/routes/config-patch/misc.js.map +1 -1
  254. package/dist/src/gateway/hono/routes/config.js +59 -0
  255. package/dist/src/gateway/hono/routes/config.js.map +1 -1
  256. package/dist/src/gateway/hono/routes/dreaming.js +1 -1
  257. package/dist/src/gateway/hono/routes/host-fs.js +2 -2
  258. package/dist/src/gateway/hono/routes/lazy-bundles.js +8 -0
  259. package/dist/src/gateway/hono/routes/lazy-bundles.js.map +1 -1
  260. package/dist/src/gateway/hono/routes/models.js +75 -12
  261. package/dist/src/gateway/hono/routes/models.js.map +1 -1
  262. package/dist/src/gateway/hono/routes/shares.js +1 -1
  263. package/dist/src/gateway/hono/routes/voice.js +75 -0
  264. package/dist/src/gateway/hono/routes/voice.js.map +1 -1
  265. package/dist/src/gateway/hono/routes/workflows.d.ts +3 -0
  266. package/dist/src/gateway/hono/routes/workflows.js +347 -0
  267. package/dist/src/gateway/hono/routes/workflows.js.map +1 -0
  268. package/dist/src/gateway/hono/routes/workspace.js +4 -4
  269. package/dist/src/gateway/lock.js +3 -3
  270. package/dist/src/gateway/ports.js +1 -1
  271. package/dist/src/gateway/service/agent-runner.js +2 -2
  272. package/dist/src/gateway/service/marketplace-service.js +2 -2
  273. package/dist/src/gateway/service/run-gateway-agent.js +2 -20
  274. package/dist/src/gateway/service/run-gateway-agent.js.map +1 -1
  275. package/dist/src/gateway/service.d.ts +3 -0
  276. package/dist/src/gateway/service.js +7 -1
  277. package/dist/src/gateway/service.js.map +1 -1
  278. package/dist/src/gateway/workspace-fs-file-list.js +1 -1
  279. package/dist/src/infra/restart.js +2 -2
  280. package/dist/src/infra/update-check.js +1 -1
  281. package/dist/src/infra/update-global.js +1 -1
  282. package/dist/src/infra/update-lock.js +3 -3
  283. package/dist/src/infra/update-runner.js +1 -1
  284. package/dist/src/infra/update-startup.js +2 -2
  285. package/dist/src/infra/write-file-atomic.js +2 -2
  286. package/dist/src/mcp/channel-bridge.js +1 -1
  287. package/dist/src/providers/auth-runtime/auth-profile-store.js +1 -1
  288. package/dist/src/providers/index.js +2 -2
  289. package/dist/src/providers/model-registry.js +1 -1
  290. package/dist/src/session/config-store.js +2 -2
  291. package/dist/src/session/init-session-turn.js +2 -2
  292. package/dist/src/session/parity/jsonl-transcript-io.js +2 -2
  293. package/dist/src/session/parity/sessions-json-file.js +1 -1
  294. package/dist/src/session/parity/transcript-file-lock.js +2 -2
  295. package/dist/src/session/parity/transcript-paths.js +1 -1
  296. package/dist/src/session/resolve-session.js +4 -4
  297. package/dist/src/session/search-index-cache.js +1 -1
  298. package/dist/src/session/search-index.js +1 -1
  299. package/dist/src/session/session-title.js +2 -2
  300. package/dist/src/session/store.js +5 -5
  301. package/dist/src/share/share-auto.js +2 -2
  302. package/dist/src/share/share-store.js +3 -3
  303. package/dist/src/share/share-thumbnail.js +2 -2
  304. package/dist/src/share/share-zip.js +1 -1
  305. package/dist/src/share/site-share-store.js +3 -3
  306. package/dist/src/share/site-static-serve.js +1 -1
  307. package/dist/src/tui/clipboard-image.js +3 -3
  308. package/dist/src/tui/theme-manager.js +1 -1
  309. package/dist/src/tui/tui-agent-events.js +2 -1
  310. package/dist/src/tui/tui-agent-events.js.map +1 -1
  311. package/dist/src/tui/tui-keybindings-file.js +1 -1
  312. package/dist/src/tui/tui-scoped-models.js +2 -2
  313. package/dist/src/tui/tui-settings.js +1 -1
  314. package/dist/src/tui/tui.js +3 -3
  315. package/dist/src/tunnel/frpc-binary.js +3 -3
  316. package/dist/src/tunnel/frpc-config.js +1 -1
  317. package/dist/src/tunnel/frpc-extract.js +1 -1
  318. package/dist/src/tunnel/tunnel-state.js +1 -1
  319. package/dist/src/utils/logger/audit.js +1 -1
  320. package/dist/src/utils/logger/log-store.js +1 -1
  321. package/dist/src/utils/logger/rotation.js +1 -1
  322. package/dist/src/voice/metadata/builtin.d.ts +2 -0
  323. package/dist/src/voice/metadata/builtin.js +420 -0
  324. package/dist/src/voice/metadata/builtin.js.map +1 -0
  325. package/dist/src/voice/metadata/index.d.ts +4 -0
  326. package/dist/src/voice/metadata/index.js +3 -0
  327. package/dist/src/voice/metadata/registry.d.ts +5 -0
  328. package/dist/src/voice/metadata/registry.js +34 -0
  329. package/dist/src/voice/metadata/registry.js.map +1 -0
  330. package/dist/src/voice/metadata/types.d.ts +41 -0
  331. package/dist/src/voice/metadata/types.js +1 -0
  332. package/dist/src/voice/stt/list-providers.d.ts +3 -3
  333. package/dist/src/voice/stt/list-providers.js +41 -6
  334. package/dist/src/voice/stt/list-providers.js.map +1 -1
  335. package/dist/src/voice/tts/audio.js +1 -1
  336. package/dist/src/voice/tts/list-providers.d.ts +3 -3
  337. package/dist/src/voice/tts/list-providers.js +41 -6
  338. package/dist/src/voice/tts/list-providers.js.map +1 -1
  339. package/dist/src/voice/tts/providers/edge-speech.js +2 -2
  340. package/dist/src/workflows/domain/command.d.ts +18 -0
  341. package/dist/src/workflows/domain/command.js +1 -0
  342. package/dist/src/workflows/domain/definition.d.ts +62 -0
  343. package/dist/src/workflows/domain/definition.js +1 -0
  344. package/dist/src/workflows/domain/event.d.ts +67 -0
  345. package/dist/src/workflows/domain/event.js +1 -0
  346. package/dist/src/workflows/domain/index.d.ts +5 -0
  347. package/dist/src/workflows/domain/index.js +2 -0
  348. package/dist/src/workflows/domain/result.d.ts +65 -0
  349. package/dist/src/workflows/domain/result.js +1 -0
  350. package/dist/src/workflows/domain/run.d.ts +120 -0
  351. package/dist/src/workflows/domain/run.js +14 -0
  352. package/dist/src/workflows/domain/run.js.map +1 -0
  353. package/dist/src/workflows/engine/index.d.ts +2 -0
  354. package/dist/src/workflows/engine/index.js +3 -0
  355. package/dist/src/workflows/engine/projector.d.ts +3 -0
  356. package/dist/src/workflows/engine/projector.js +205 -0
  357. package/dist/src/workflows/engine/projector.js.map +1 -0
  358. package/dist/src/workflows/engine/workflow-engine.d.ts +31 -0
  359. package/dist/src/workflows/engine/workflow-engine.js +188 -0
  360. package/dist/src/workflows/engine/workflow-engine.js.map +1 -0
  361. package/dist/src/workflows/index.d.ts +6 -0
  362. package/dist/src/workflows/index.js +11 -0
  363. package/dist/src/workflows/runtime/index.d.ts +1 -0
  364. package/dist/src/workflows/runtime/index.js +4 -0
  365. package/dist/src/workflows/runtime/script-runtime.d.ts +3 -0
  366. package/dist/src/workflows/runtime/script-runtime.js +3 -0
  367. package/dist/src/workflows/store/event-store.d.ts +17 -0
  368. package/dist/src/workflows/store/event-store.js +83 -0
  369. package/dist/src/workflows/store/event-store.js.map +1 -0
  370. package/dist/src/workflows/store/paths.d.ts +7 -0
  371. package/dist/src/workflows/store/paths.js +26 -0
  372. package/dist/src/workflows/store/paths.js.map +1 -0
  373. package/dist/src/workflows/store/run-store.d.ts +13 -0
  374. package/dist/src/workflows/store/run-store.js +68 -0
  375. package/dist/src/workflows/store/run-store.js.map +1 -0
  376. package/package.json +5 -5
  377. package/dist/gateway/static/root/assets/agents-BEAbXpuP.js +0 -222
  378. package/dist/gateway/static/root/assets/channels-settings-yohw9YSu.js +0 -1
  379. package/dist/gateway/static/root/assets/extension-settings-page-x4BB7q1X.js +0 -1
  380. package/dist/gateway/static/root/assets/index-a5gWIdZQ.css +0 -1
  381. package/dist/gateway/static/root/assets/sessions-page-CD7AfB-2.js +0 -1
  382. package/dist/gateway/static/root/assets/settings-page-BBOjEQW3.js +0 -3
  383. package/dist/gateway/static/root/assets/skills-page-CcN_gj--.js +0 -2
  384. package/dist/gateway/static/root/assets/url-Dd8Q7kZZ.js +0 -3
  385. package/dist/gateway/static/root/assets/voice-api-key-field-O6awz9hi.js +0 -1
@@ -1,4 +1,5 @@
1
1
  import { appendPiTranscriptMessage } from "../../session/parity/jsonl-transcript-io.js";
2
+ import { formatAgentRunErrorForClient } from "../client-error-format.js";
2
3
  import { resolveEffectiveReasoningLevel } from "../../session/thinking-resolve.js";
3
4
  import { initSessionTurn } from "../../session/init-session-turn.js";
4
5
  import "../../session/index.js";
@@ -19,6 +20,18 @@ async function* runProcessDirectStreaming(deps, input) {
19
20
  const visible = applyReasoningVisibilityToSseEvent(event, reasoningLevel);
20
21
  if (visible !== null) queue.push(visible);
21
22
  };
23
+ const formatStreamError = (raw) => {
24
+ let provider;
25
+ let modelRef;
26
+ try {
27
+ provider = deps.modelManager.getResolvedModelForSession(sessionKey).provider;
28
+ modelRef = deps.modelManager.getModelForSession(sessionKey);
29
+ } catch {}
30
+ return formatAgentRunErrorForClient(raw, {
31
+ provider,
32
+ modelRef
33
+ });
34
+ };
22
35
  if (channel === "webchat") deps.registerWebchatSsePublisher(sessionKey, pushVisible);
23
36
  const signal = input.signal;
24
37
  let userAborted = false;
@@ -149,18 +162,21 @@ async function* runProcessDirectStreaming(deps, input) {
149
162
  abortSignal: signal,
150
163
  onEvent: (embeddedEvent) => {
151
164
  const mapped = mapEmbeddedEventToGatewaySse(embeddedEvent);
152
- if (mapped) pushVisible(mapped);
165
+ if (mapped) {
166
+ if (mapped.type === "error" && typeof mapped.content === "string") mapped.content = formatStreamError(mapped.content);
167
+ pushVisible(mapped);
168
+ }
153
169
  }
154
170
  });
155
171
  if (result.lastAssistantText) deps.onTurnComplete?.(sessionKey, result.lastAssistantText);
156
172
  if (!result.ok && result.errorMessage && !abortHandled) pushVisible({
157
173
  type: "error",
158
- content: result.errorMessage
174
+ content: formatStreamError(result.errorMessage)
159
175
  });
160
176
  } catch (err) {
161
177
  if (!abortHandled) pushVisible({
162
178
  type: "error",
163
- content: err instanceof Error ? err.message : String(err)
179
+ content: formatStreamError(err instanceof Error ? err.message : String(err))
164
180
  });
165
181
  } finally {
166
182
  queue.close();
@@ -1 +1 @@
1
- {"version":3,"file":"process-direct-streaming.js","names":[],"sources":["../../../../src/agent/service/process-direct-streaming.ts"],"sourcesContent":["import type { AgentMessage } from '@earendil-works/pi-agent-core';\n\nimport type { Config } from '../../config/schema.js';\nimport type { InternalAttachmentRoots } from '../../channels/attachments/inbound-persist.js';\nimport {\n isVoiceLikeAttachment,\n mergeVoiceTranscriptsIntoUserText,\n mergeSttConfigFromAppConfig,\n} from '../../channels/attachments/voice-stt-webchat.js';\nimport {\n resolveEffectiveReasoningLevel,\n initSessionTurn,\n type SessionConfigStore,\n type SessionStore,\n} from '../../session/index.js';\nimport { appendPiTranscriptMessage } from '../../session/parity/jsonl-transcript-io.js';\nimport type { SessionContext } from '../session/index.js';\nimport { applyReasoningVisibilityToSseEvent } from '../streaming/reasoning-visibility-sse.js';\nimport type { ReasoningLevel } from '../transcript/thinking-types.js';\nimport { abortEmbeddedRun } from '../embedded/runs.js';\nimport { mapEmbeddedEventToGatewaySse } from '../embedded/map-stream-events.js';\nimport type { AgentInstanceGateway } from '../agent-instance-gateway.js';\nimport type { CommandHandler } from '../messaging/command-handler.js';\nimport type { ModelManager } from '../models/index.js';\n\nimport { AsyncQueue } from './async-queue.js';\nimport {\n hydratePerTurnState,\n runDirectAgentTurn,\n tryRunSlashCommand,\n} from './direct-turn-helpers.js';\n\nexport type DirectStreamInboundAttachment = {\n type: string;\n mimeType?: string;\n data?: string;\n name?: string;\n size?: number;\n workspaceRelativePath?: string;\n};\n\nexport type ProcessDirectStreamLog = {\n info: (obj: Record<string, unknown>, msg: string) => void;\n warn: (obj: Record<string, unknown>, msg: string) => void;\n debug?: (obj: Record<string, unknown>, msg: string) => void;\n};\n\nexport interface ProcessDirectStreamingDeps {\n log: ProcessDirectStreamLog;\n parseSessionKey: (sessionKey: string) => { channel: string; chatId: string };\n initDirectStreamingSession: (\n sessionKey: string,\n channel: string,\n chatId: string,\n ) => SessionContext;\n registerWebchatSsePublisher: (\n sessionKey: string,\n publisher: (event: { type: string; [key: string]: unknown }) => void,\n ) => void;\n unregisterWebchatSsePublisher: (sessionKey: string) => void;\n agentManager: AgentInstanceGateway;\n hydrateSessionWorkspaceFromStore: (sessionKey: string) => Promise<void>;\n hydrateSessionModelFromStore: (sessionKey: string) => Promise<void>;\n sessionStore: SessionStore;\n modelManager: ModelManager;\n applyResolvedThinkingLevel: (sessionKey: string, thinking?: string | null) => Promise<void>;\n getConfig: () => Config | undefined;\n sessionConfigStore: SessionConfigStore;\n attachmentRootsForSession: (sessionKey: string) => InternalAttachmentRoots;\n commandHandler: Pick<CommandHandler, 'executeCommandAndAggregateReply'>;\n prepareInboundAttachments: (\n sessionKey: string,\n attachments?: DirectStreamInboundAttachment[],\n ) => Promise<DirectStreamInboundAttachment[] | undefined>;\n buildMessageContent: (\n content: string,\n attachments: DirectStreamInboundAttachment[] | undefined,\n sessionKey: string,\n ) => Promise<Array<{ type: 'text'; text: string } | { type: 'image'; data: string; mimeType: string }>>;\n recordPersistentGoalStreamOutcome?: (\n sessionKey: string,\n outcome: { skipPersistentGoalPostTurn: boolean },\n ) => void;\n onTurnComplete?: (sessionKey: string, lastAssistantText?: string) => void;\n /** Disk-only transcript sync (slash receipt already streamed as tokens). */\n reloadWebchatTranscript?: (sessionKey: string) => void;\n maybeEmitWebchatTts: (\n sessionKey: string,\n hadInboundVoice: boolean,\n ) => Promise<{ type: 'tts_audio'; workspaceRelativePath: string; mimeType: string; name: string } | null>;\n endDirectRequestContext: () => void;\n resetSession: (sessionKey: string) => Promise<{ sessionId: string; previousSessionId: string } | null>;\n}\n\nexport interface ProcessDirectStreamingInput {\n content: string;\n sessionKey?: string;\n attachments?: DirectStreamInboundAttachment[];\n thinking?: string;\n signal?: AbortSignal;\n}\n\nexport type ProcessDirectStreamingSseEvent = { type: string; [key: string]: unknown };\n\nexport async function* runProcessDirectStreaming(\n deps: ProcessDirectStreamingDeps,\n input: ProcessDirectStreamingInput,\n): AsyncGenerator<ProcessDirectStreamingSseEvent, void, unknown> {\n const sessionKey = input.sessionKey ?? 'agent:main:main';\n const { channel, chatId } = deps.parseSessionKey(sessionKey);\n const context = deps.initDirectStreamingSession(sessionKey, channel, chatId);\n\n const queue = new AsyncQueue<ProcessDirectStreamingSseEvent>();\n let reasoningLevel: ReasoningLevel = 'stream';\n\n const pushVisible = (event: ProcessDirectStreamingSseEvent) => {\n const visible = applyReasoningVisibilityToSseEvent(event, reasoningLevel);\n if (visible !== null) {\n queue.push(visible);\n }\n };\n\n if (channel === 'webchat') {\n deps.registerWebchatSsePublisher(sessionKey, pushVisible);\n }\n\n const signal = input.signal;\n let userAborted = false;\n let abortHandled = false;\n let inboundVoice = false;\n let ranSlashCommand = false;\n let mergedUserText = input.content;\n let webchatSlashReceipt: string | undefined;\n\n // Kick off the agent task in the background; events stream into `queue` as they happen\n // and the generator below drains `queue` until the task closes it.\n const taskPromise = (async () => {\n try {\n const cfg = deps.getConfig();\n let turnBody = input.content;\n let resetTriggeredAtInit = false;\n if (cfg) {\n const turn = await initSessionTurn({\n cfg,\n sessionKey,\n body: input.content,\n resetSession: deps.resetSession,\n });\n resetTriggeredAtInit = turn.resetTriggered;\n if (turn.bareReset && turn.ackMessage) {\n ranSlashCommand = true;\n webchatSlashReceipt = turn.ackMessage;\n pushVisible({ type: 'token', content: turn.ackMessage });\n return;\n }\n turnBody = turn.bodyStripped;\n if (turn.isNewSession) {\n deps.log.debug(\n {\n sessionKey,\n sessionId: turn.sessionId,\n previousSessionId: turn.previousSessionId,\n resetTriggered: turn.resetTriggered,\n staleRollover: turn.staleRollover,\n },\n 'Session reset boundary at direct turn start',\n );\n }\n }\n\n await hydratePerTurnState(deps, sessionKey, input.thinking);\n {\n const defReason = (deps.getConfig()?.agents?.defaults?.reasoningDefault ?? 'stream') as ReasoningLevel;\n reasoningLevel = await resolveEffectiveReasoningLevel(deps.sessionConfigStore, sessionKey, defReason);\n }\n\n const prepared = await deps.prepareInboundAttachments(sessionKey, input.attachments);\n\n const sttCfg = mergeSttConfigFromAppConfig(deps.getConfig()?.tools?.media?.audio, deps.getConfig()?.tools?.media);\n const voiceMerge = await mergeVoiceTranscriptsIntoUserText(\n deps.attachmentRootsForSession(sessionKey),\n prepared,\n turnBody,\n sttCfg,\n );\n mergedUserText = voiceMerge.text;\n inboundVoice = voiceMerge.inboundVoice;\n\n if (inboundVoice) {\n const transcriptParts = [\n voiceMerge.voiceTranscripts.filter(Boolean).join('\\n'),\n turnBody.trim(),\n ].filter(Boolean);\n const voiceAttachments = (prepared ?? []).filter(isVoiceLikeAttachment).map((att) => ({\n workspaceRelativePath: att.workspaceRelativePath,\n mimeType: att.mimeType,\n name: att.name,\n }));\n pushVisible({\n type: 'user_transcript',\n text: transcriptParts.join('\\n\\n'),\n attachments: voiceAttachments,\n });\n }\n\n const armAbort = () => {\n if (abortHandled) {\n return;\n }\n abortHandled = true;\n userAborted = true;\n void abortEmbeddedRun(sessionKey);\n queue.close();\n };\n if (signal) {\n if (signal.aborted) {\n armAbort();\n return;\n }\n signal.addEventListener('abort', armAbort, { once: true });\n }\n\n const slash = await tryRunSlashCommand(\n deps,\n { sessionKey, channel, chatId, senderId: context.senderId, isGroup: context.isGroup },\n mergedUserText,\n { skipResetCommands: resetTriggeredAtInit },\n );\n if (slash.matched) {\n ranSlashCommand = true;\n const text = slash.aggregatedText.trim();\n if (text) {\n webchatSlashReceipt = text;\n pushVisible({ type: 'token', content: text });\n } else if (channel === 'webchat') {\n webchatSlashReceipt =\n 'Command finished with no assistant text. If you used `/goal`, a follow-up turn may still be scheduled automatically.';\n pushVisible({ type: 'token', content: webchatSlashReceipt });\n }\n return;\n }\n\n const textForAgent = mergedUserText.trimStart().startsWith('/skill:')\n ? deps.agentManager.expandSkillUserText(mergedUserText)\n : mergedUserText;\n const messageContent = await deps.buildMessageContent(textForAgent, prepared, sessionKey);\n\n const userMessage = {\n role: 'user' as const,\n content: messageContent,\n timestamp: Date.now(),\n };\n if (channel === 'webchat') {\n pushVisible({\n type: 'user_message',\n timestamp: userMessage.timestamp,\n content: userMessage.content,\n attachments: prepared?.map((att) => ({\n type: att.type,\n mimeType: att.mimeType,\n name: att.name,\n size: att.size,\n workspaceRelativePath: att.workspaceRelativePath,\n })),\n });\n }\n\n const result = await runDirectAgentTurn(\n {\n sessionStore: deps.sessionStore,\n agentManager: deps.agentManager,\n modelManager: deps.modelManager,\n config: deps.getConfig(),\n },\n {\n sessionKey,\n userMessage,\n abortSignal: signal,\n onEvent: (embeddedEvent) => {\n const mapped = mapEmbeddedEventToGatewaySse(embeddedEvent);\n if (mapped) {\n pushVisible(mapped);\n }\n },\n },\n );\n\n if (result.lastAssistantText) {\n deps.onTurnComplete?.(sessionKey, result.lastAssistantText);\n }\n if (!result.ok && result.errorMessage && !abortHandled) {\n pushVisible({ type: 'error', content: result.errorMessage });\n }\n } catch (err) {\n if (!abortHandled) {\n pushVisible({ type: 'error', content: err instanceof Error ? err.message : String(err) });\n }\n } finally {\n queue.close();\n }\n })();\n\n try {\n for await (const event of queue) {\n yield event;\n }\n await taskPromise; // surface unexpected throws\n\n if (channel === 'webchat' && ranSlashCommand) {\n try {\n const { absPath } = await deps.sessionStore.resolveTranscriptPath(sessionKey);\n const workspaceDir = deps.agentManager.getResolvedWorkspaceForSession(sessionKey);\n const userMsg = {\n role: 'user' as const,\n content: [{ type: 'text' as const, text: mergedUserText }],\n timestamp: Date.now(),\n } as AgentMessage;\n await appendPiTranscriptMessage({\n absPath,\n cwd: workspaceDir,\n message: userMsg,\n sessionKey,\n });\n if (webchatSlashReceipt?.trim()) {\n const assistantMsg = {\n role: 'assistant' as const,\n content: [{ type: 'text' as const, text: webchatSlashReceipt.trim() }],\n timestamp: Date.now(),\n } as AgentMessage;\n await appendPiTranscriptMessage({\n absPath,\n cwd: workspaceDir,\n message: assistantMsg,\n sessionKey,\n });\n }\n deps.reloadWebchatTranscript?.(sessionKey);\n } catch (err) {\n deps.log.warn({ err, sessionKey }, 'Failed to persist webchat slash command receipt');\n }\n }\n\n if (!userAborted) {\n const ttsAudioEvent = await deps.maybeEmitWebchatTts(sessionKey, inboundVoice);\n if (ttsAudioEvent) {\n yield ttsAudioEvent;\n }\n }\n\n deps.recordPersistentGoalStreamOutcome?.(sessionKey, { skipPersistentGoalPostTurn: ranSlashCommand });\n } finally {\n if (channel === 'webchat') {\n deps.unregisterWebchatSsePublisher(sessionKey);\n }\n deps.endDirectRequestContext();\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"process-direct-streaming.js","names":[],"sources":["../../../../src/agent/service/process-direct-streaming.ts"],"sourcesContent":["import type { AgentMessage } from '@earendil-works/pi-agent-core';\n\nimport type { Config } from '../../config/schema.js';\nimport type { InternalAttachmentRoots } from '../../channels/attachments/inbound-persist.js';\nimport {\n isVoiceLikeAttachment,\n mergeVoiceTranscriptsIntoUserText,\n mergeSttConfigFromAppConfig,\n} from '../../channels/attachments/voice-stt-webchat.js';\nimport {\n resolveEffectiveReasoningLevel,\n initSessionTurn,\n type SessionConfigStore,\n type SessionStore,\n} from '../../session/index.js';\nimport { appendPiTranscriptMessage } from '../../session/parity/jsonl-transcript-io.js';\nimport type { SessionContext } from '../session/index.js';\nimport { applyReasoningVisibilityToSseEvent } from '../streaming/reasoning-visibility-sse.js';\nimport type { ReasoningLevel } from '../transcript/thinking-types.js';\nimport { formatAgentRunErrorForClient } from '../client-error-format.js';\nimport { abortEmbeddedRun } from '../embedded/runs.js';\nimport { mapEmbeddedEventToGatewaySse } from '../embedded/map-stream-events.js';\nimport type { AgentInstanceGateway } from '../agent-instance-gateway.js';\nimport type { CommandHandler } from '../messaging/command-handler.js';\nimport type { ModelManager } from '../models/index.js';\n\nimport { AsyncQueue } from './async-queue.js';\nimport {\n hydratePerTurnState,\n runDirectAgentTurn,\n tryRunSlashCommand,\n} from './direct-turn-helpers.js';\n\nexport type DirectStreamInboundAttachment = {\n type: string;\n mimeType?: string;\n data?: string;\n name?: string;\n size?: number;\n workspaceRelativePath?: string;\n};\n\nexport type ProcessDirectStreamLog = {\n info: (obj: Record<string, unknown>, msg: string) => void;\n warn: (obj: Record<string, unknown>, msg: string) => void;\n debug?: (obj: Record<string, unknown>, msg: string) => void;\n};\n\nexport interface ProcessDirectStreamingDeps {\n log: ProcessDirectStreamLog;\n parseSessionKey: (sessionKey: string) => { channel: string; chatId: string };\n initDirectStreamingSession: (\n sessionKey: string,\n channel: string,\n chatId: string,\n ) => SessionContext;\n registerWebchatSsePublisher: (\n sessionKey: string,\n publisher: (event: { type: string; [key: string]: unknown }) => void,\n ) => void;\n unregisterWebchatSsePublisher: (sessionKey: string) => void;\n agentManager: AgentInstanceGateway;\n hydrateSessionWorkspaceFromStore: (sessionKey: string) => Promise<void>;\n hydrateSessionModelFromStore: (sessionKey: string) => Promise<void>;\n sessionStore: SessionStore;\n modelManager: ModelManager;\n applyResolvedThinkingLevel: (sessionKey: string, thinking?: string | null) => Promise<void>;\n getConfig: () => Config | undefined;\n sessionConfigStore: SessionConfigStore;\n attachmentRootsForSession: (sessionKey: string) => InternalAttachmentRoots;\n commandHandler: Pick<CommandHandler, 'executeCommandAndAggregateReply'>;\n prepareInboundAttachments: (\n sessionKey: string,\n attachments?: DirectStreamInboundAttachment[],\n ) => Promise<DirectStreamInboundAttachment[] | undefined>;\n buildMessageContent: (\n content: string,\n attachments: DirectStreamInboundAttachment[] | undefined,\n sessionKey: string,\n ) => Promise<Array<{ type: 'text'; text: string } | { type: 'image'; data: string; mimeType: string }>>;\n recordPersistentGoalStreamOutcome?: (\n sessionKey: string,\n outcome: { skipPersistentGoalPostTurn: boolean },\n ) => void;\n onTurnComplete?: (sessionKey: string, lastAssistantText?: string) => void;\n /** Disk-only transcript sync (slash receipt already streamed as tokens). */\n reloadWebchatTranscript?: (sessionKey: string) => void;\n maybeEmitWebchatTts: (\n sessionKey: string,\n hadInboundVoice: boolean,\n ) => Promise<{ type: 'tts_audio'; workspaceRelativePath: string; mimeType: string; name: string } | null>;\n endDirectRequestContext: () => void;\n resetSession: (sessionKey: string) => Promise<{ sessionId: string; previousSessionId: string } | null>;\n}\n\nexport interface ProcessDirectStreamingInput {\n content: string;\n sessionKey?: string;\n attachments?: DirectStreamInboundAttachment[];\n thinking?: string;\n signal?: AbortSignal;\n}\n\nexport type ProcessDirectStreamingSseEvent = { type: string; [key: string]: unknown };\n\nexport async function* runProcessDirectStreaming(\n deps: ProcessDirectStreamingDeps,\n input: ProcessDirectStreamingInput,\n): AsyncGenerator<ProcessDirectStreamingSseEvent, void, unknown> {\n const sessionKey = input.sessionKey ?? 'agent:main:main';\n const { channel, chatId } = deps.parseSessionKey(sessionKey);\n const context = deps.initDirectStreamingSession(sessionKey, channel, chatId);\n\n const queue = new AsyncQueue<ProcessDirectStreamingSseEvent>();\n let reasoningLevel: ReasoningLevel = 'stream';\n\n const pushVisible = (event: ProcessDirectStreamingSseEvent) => {\n const visible = applyReasoningVisibilityToSseEvent(event, reasoningLevel);\n if (visible !== null) {\n queue.push(visible);\n }\n };\n\n const formatStreamError = (raw: string): string => {\n let provider: string | undefined;\n let modelRef: string | undefined;\n try {\n const resolved = deps.modelManager.getResolvedModelForSession(sessionKey);\n provider = resolved.provider;\n modelRef = deps.modelManager.getModelForSession(sessionKey);\n } catch {\n /* ignore — format without provider context */\n }\n return formatAgentRunErrorForClient(raw, { provider, modelRef });\n };\n\n if (channel === 'webchat') {\n deps.registerWebchatSsePublisher(sessionKey, pushVisible);\n }\n\n const signal = input.signal;\n let userAborted = false;\n let abortHandled = false;\n let inboundVoice = false;\n let ranSlashCommand = false;\n let mergedUserText = input.content;\n let webchatSlashReceipt: string | undefined;\n\n // Kick off the agent task in the background; events stream into `queue` as they happen\n // and the generator below drains `queue` until the task closes it.\n const taskPromise = (async () => {\n try {\n const cfg = deps.getConfig();\n let turnBody = input.content;\n let resetTriggeredAtInit = false;\n if (cfg) {\n const turn = await initSessionTurn({\n cfg,\n sessionKey,\n body: input.content,\n resetSession: deps.resetSession,\n });\n resetTriggeredAtInit = turn.resetTriggered;\n if (turn.bareReset && turn.ackMessage) {\n ranSlashCommand = true;\n webchatSlashReceipt = turn.ackMessage;\n pushVisible({ type: 'token', content: turn.ackMessage });\n return;\n }\n turnBody = turn.bodyStripped;\n if (turn.isNewSession) {\n deps.log.debug(\n {\n sessionKey,\n sessionId: turn.sessionId,\n previousSessionId: turn.previousSessionId,\n resetTriggered: turn.resetTriggered,\n staleRollover: turn.staleRollover,\n },\n 'Session reset boundary at direct turn start',\n );\n }\n }\n\n await hydratePerTurnState(deps, sessionKey, input.thinking);\n {\n const defReason = (deps.getConfig()?.agents?.defaults?.reasoningDefault ?? 'stream') as ReasoningLevel;\n reasoningLevel = await resolveEffectiveReasoningLevel(deps.sessionConfigStore, sessionKey, defReason);\n }\n\n const prepared = await deps.prepareInboundAttachments(sessionKey, input.attachments);\n\n const sttCfg = mergeSttConfigFromAppConfig(deps.getConfig()?.tools?.media?.audio, deps.getConfig()?.tools?.media);\n const voiceMerge = await mergeVoiceTranscriptsIntoUserText(\n deps.attachmentRootsForSession(sessionKey),\n prepared,\n turnBody,\n sttCfg,\n );\n mergedUserText = voiceMerge.text;\n inboundVoice = voiceMerge.inboundVoice;\n\n if (inboundVoice) {\n const transcriptParts = [\n voiceMerge.voiceTranscripts.filter(Boolean).join('\\n'),\n turnBody.trim(),\n ].filter(Boolean);\n const voiceAttachments = (prepared ?? []).filter(isVoiceLikeAttachment).map((att) => ({\n workspaceRelativePath: att.workspaceRelativePath,\n mimeType: att.mimeType,\n name: att.name,\n }));\n pushVisible({\n type: 'user_transcript',\n text: transcriptParts.join('\\n\\n'),\n attachments: voiceAttachments,\n });\n }\n\n const armAbort = () => {\n if (abortHandled) {\n return;\n }\n abortHandled = true;\n userAborted = true;\n void abortEmbeddedRun(sessionKey);\n queue.close();\n };\n if (signal) {\n if (signal.aborted) {\n armAbort();\n return;\n }\n signal.addEventListener('abort', armAbort, { once: true });\n }\n\n const slash = await tryRunSlashCommand(\n deps,\n { sessionKey, channel, chatId, senderId: context.senderId, isGroup: context.isGroup },\n mergedUserText,\n { skipResetCommands: resetTriggeredAtInit },\n );\n if (slash.matched) {\n ranSlashCommand = true;\n const text = slash.aggregatedText.trim();\n if (text) {\n webchatSlashReceipt = text;\n pushVisible({ type: 'token', content: text });\n } else if (channel === 'webchat') {\n webchatSlashReceipt =\n 'Command finished with no assistant text. If you used `/goal`, a follow-up turn may still be scheduled automatically.';\n pushVisible({ type: 'token', content: webchatSlashReceipt });\n }\n return;\n }\n\n const textForAgent = mergedUserText.trimStart().startsWith('/skill:')\n ? deps.agentManager.expandSkillUserText(mergedUserText)\n : mergedUserText;\n const messageContent = await deps.buildMessageContent(textForAgent, prepared, sessionKey);\n\n const userMessage = {\n role: 'user' as const,\n content: messageContent,\n timestamp: Date.now(),\n };\n if (channel === 'webchat') {\n pushVisible({\n type: 'user_message',\n timestamp: userMessage.timestamp,\n content: userMessage.content,\n attachments: prepared?.map((att) => ({\n type: att.type,\n mimeType: att.mimeType,\n name: att.name,\n size: att.size,\n workspaceRelativePath: att.workspaceRelativePath,\n })),\n });\n }\n\n const result = await runDirectAgentTurn(\n {\n sessionStore: deps.sessionStore,\n agentManager: deps.agentManager,\n modelManager: deps.modelManager,\n config: deps.getConfig(),\n },\n {\n sessionKey,\n userMessage,\n abortSignal: signal,\n onEvent: (embeddedEvent) => {\n const mapped = mapEmbeddedEventToGatewaySse(embeddedEvent);\n if (mapped) {\n if (mapped.type === 'error' && typeof mapped.content === 'string') {\n mapped.content = formatStreamError(mapped.content);\n }\n pushVisible(mapped);\n }\n },\n },\n );\n\n if (result.lastAssistantText) {\n deps.onTurnComplete?.(sessionKey, result.lastAssistantText);\n }\n if (!result.ok && result.errorMessage && !abortHandled) {\n pushVisible({ type: 'error', content: formatStreamError(result.errorMessage) });\n }\n } catch (err) {\n if (!abortHandled) {\n const em = err instanceof Error ? err.message : String(err);\n pushVisible({ type: 'error', content: formatStreamError(em) });\n }\n } finally {\n queue.close();\n }\n })();\n\n try {\n for await (const event of queue) {\n yield event;\n }\n await taskPromise; // surface unexpected throws\n\n if (channel === 'webchat' && ranSlashCommand) {\n try {\n const { absPath } = await deps.sessionStore.resolveTranscriptPath(sessionKey);\n const workspaceDir = deps.agentManager.getResolvedWorkspaceForSession(sessionKey);\n const userMsg = {\n role: 'user' as const,\n content: [{ type: 'text' as const, text: mergedUserText }],\n timestamp: Date.now(),\n } as AgentMessage;\n await appendPiTranscriptMessage({\n absPath,\n cwd: workspaceDir,\n message: userMsg,\n sessionKey,\n });\n if (webchatSlashReceipt?.trim()) {\n const assistantMsg = {\n role: 'assistant' as const,\n content: [{ type: 'text' as const, text: webchatSlashReceipt.trim() }],\n timestamp: Date.now(),\n } as AgentMessage;\n await appendPiTranscriptMessage({\n absPath,\n cwd: workspaceDir,\n message: assistantMsg,\n sessionKey,\n });\n }\n deps.reloadWebchatTranscript?.(sessionKey);\n } catch (err) {\n deps.log.warn({ err, sessionKey }, 'Failed to persist webchat slash command receipt');\n }\n }\n\n if (!userAborted) {\n const ttsAudioEvent = await deps.maybeEmitWebchatTts(sessionKey, inboundVoice);\n if (ttsAudioEvent) {\n yield ttsAudioEvent;\n }\n }\n\n deps.recordPersistentGoalStreamOutcome?.(sessionKey, { skipPersistentGoalPostTurn: ranSlashCommand });\n } finally {\n if (channel === 'webchat') {\n deps.unregisterWebchatSsePublisher(sessionKey);\n }\n deps.endDirectRequestContext();\n }\n}\n"],"mappings":"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"}
@@ -1,5 +1,4 @@
1
1
  import type { Config } from '../../config/schema.js';
2
- import type { AgentInstanceGateway } from '../agent-instance-gateway.js';
3
2
  import type { SessionStore } from '../../session/index.js';
4
3
  export type WebchatTtsResult = {
5
4
  type: 'tts_audio';
@@ -9,8 +8,8 @@ export type WebchatTtsResult = {
9
8
  };
10
9
  export type WebchatTtsDeps = {
11
10
  config: Config | undefined;
12
- agentManager: AgentInstanceGateway;
13
11
  sessionStore: SessionStore;
12
+ getLastAssistantPlainText: (sessionKey: string) => string;
14
13
  log: {
15
14
  warn: (obj: Record<string, unknown>, msg: string) => void;
16
15
  };
@@ -16,7 +16,7 @@ async function maybeEmitWebchatTts(deps, sessionKey, hadInboundVoice) {
16
16
  const ttsConfig = mergeTtsConfigFromAppConfig(deps.config?.messages?.tts);
17
17
  if (!isTTSAvailable(ttsConfig)) return null;
18
18
  if (!shouldUseTTS(ttsConfig, hadInboundVoice).useTTS) return null;
19
- const text = deps.agentManager.getLastAssistantContent(sessionKey)?.trim();
19
+ const text = deps.getLastAssistantPlainText(sessionKey).trim();
20
20
  if (!text) return null;
21
21
  try {
22
22
  const webOut = getChannelOutputFormat("webchat");
@@ -1 +1 @@
1
- {"version":3,"file":"webchat-tts.js","names":[],"sources":["../../../../src/agent/service/webchat-tts.ts"],"sourcesContent":["import type { Config } from '../../config/schema.js';\nimport { persistOutboundTtsAudio } from '../../channels/attachments/outbound-tts-persist.js';\nimport { compressAudio } from '../../voice/tts/audio.js';\nimport { speak } from '../../voice/tts/index.js';\nimport { mergeTtsConfigFromAppConfig } from '../../voice/tts/merge-config.js';\nimport { shouldUseTTS, getChannelOutputFormat } from '../../voice/tts/service.js';\nimport { isTTSAvailable } from '../../voice/tts/factory.js';\nimport { resolveAgentHomeDir } from '../agent-scope.js';\nimport { extractProfileAgentId } from '../../config/agent-profile.js';\nimport type { AgentInstanceGateway } from '../agent-instance-gateway.js';\nimport type { SessionStore } from '../../session/index.js';\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\n\nexport type WebchatTtsResult = {\n type: 'tts_audio';\n workspaceRelativePath: string;\n mimeType: string;\n name: string;\n};\n\nexport type WebchatTtsDeps = {\n config: Config | undefined;\n agentManager: AgentInstanceGateway;\n sessionStore: SessionStore;\n log: { warn: (obj: Record<string, unknown>, msg: string) => void };\n};\n\n/**\n * Generate TTS for webchat when config allows; persist under agent home `tts/`.\n */\nexport async function maybeEmitWebchatTts(\n deps: WebchatTtsDeps,\n sessionKey: string,\n hadInboundVoice: boolean,\n): Promise<WebchatTtsResult | null> {\n const ttsConfig = mergeTtsConfigFromAppConfig(deps.config?.messages?.tts);\n if (!isTTSAvailable(ttsConfig)) {\n return null;\n }\n const decision = shouldUseTTS(ttsConfig, hadInboundVoice);\n if (!decision.useTTS) {\n return null;\n }\n const text = deps.agentManager.getLastAssistantContent(sessionKey)?.trim();\n if (!text) {\n return null;\n }\n try {\n const webOut = getChannelOutputFormat('webchat');\n const fmt = webOut.format as 'opus' | 'mp3' | 'wav';\n const ttsResult = await speak(text, ttsConfig, {\n appConfig: deps.config,\n tts: { format: fmt },\n });\n const { buffer, format } = await compressAudio(\n Buffer.from(ttsResult.audio),\n ttsResult.format,\n webOut.format === 'mp3' ? 'mp3' : 'opus',\n );\n const normalizedMime =\n format === 'opus' || format === 'ogg'\n ? 'audio/ogg'\n : format === 'mp3' || format === 'mpeg'\n ? 'audio/mpeg'\n : format === 'wav'\n ? 'audio/wav'\n : `audio/${format}`;\n const cfg = deps.config!;\n const persisted = await persistOutboundTtsAudio(\n resolveAgentHomeDir(cfg, extractProfileAgentId(sessionKey, cfg)),\n sessionKey,\n buffer,\n format,\n );\n await appendAttachmentToLastAssistant(deps.sessionStore, sessionKey, {\n type: 'audio',\n mimeType: normalizedMime,\n name: persisted.name,\n size: persisted.size,\n workspaceRelativePath: persisted.workspaceRelativePath,\n });\n return {\n type: 'tts_audio',\n workspaceRelativePath: persisted.workspaceRelativePath,\n mimeType: normalizedMime,\n name: persisted.name,\n };\n } catch (err) {\n deps.log.warn({ err, sessionKey }, 'Webchat TTS failed');\n return null;\n }\n}\n\nexport async function appendAttachmentToLastAssistant(\n sessionStore: SessionStore,\n sessionKey: string,\n att: {\n type: string;\n mimeType: string;\n name: string;\n size: number;\n workspaceRelativePath: string;\n },\n): Promise<void> {\n const loaded = await sessionStore.load(sessionKey);\n for (let i = loaded.length - 1; i >= 0; i--) {\n const m = loaded[i] as { role?: string; attachments?: unknown[] };\n if (m.role === 'assistant') {\n const prev = (m.attachments ?? []) as Array<{ workspaceRelativePath?: string }>;\n if (prev.some((x) => x.workspaceRelativePath === att.workspaceRelativePath)) {\n return;\n }\n const next = [...prev, att];\n loaded[i] = { ...m, attachments: next } as unknown as AgentMessage;\n await sessionStore.saveMessages(sessionKey, loaded);\n return;\n }\n }\n}\n"],"mappings":";;;;;;;;;;kBAOwD;;;;AAuBxD,eAAsB,oBACpB,MACA,YACA,iBACkC;CAClC,MAAM,YAAY,4BAA4B,KAAK,QAAQ,UAAU,IAAI;AACzE,KAAI,CAAC,eAAe,UAAU,CAC5B,QAAO;AAGT,KAAI,CADa,aAAa,WAAW,gBAC5B,CAAC,OACZ,QAAO;CAET,MAAM,OAAO,KAAK,aAAa,wBAAwB,WAAW,EAAE,MAAM;AAC1E,KAAI,CAAC,KACH,QAAO;AAET,KAAI;EACF,MAAM,SAAS,uBAAuB,UAAU;EAChD,MAAM,MAAM,OAAO;EACnB,MAAM,YAAY,MAAM,MAAM,MAAM,WAAW;GAC7C,WAAW,KAAK;GAChB,KAAK,EAAE,QAAQ,KAAK;GACrB,CAAC;EACF,MAAM,EAAE,QAAQ,WAAW,MAAM,cAC/B,OAAO,KAAK,UAAU,MAAM,EAC5B,UAAU,QACV,OAAO,WAAW,QAAQ,QAAQ,OACnC;EACD,MAAM,iBACJ,WAAW,UAAU,WAAW,QAC5B,cACA,WAAW,SAAS,WAAW,SAC7B,eACA,WAAW,QACT,cACA,SAAS;EACnB,MAAM,MAAM,KAAK;EACjB,MAAM,YAAY,MAAM,wBACtB,oBAAoB,KAAK,sBAAsB,YAAY,IAAI,CAAC,EAChE,YACA,QACA,OACD;AACD,QAAM,gCAAgC,KAAK,cAAc,YAAY;GACnE,MAAM;GACN,UAAU;GACV,MAAM,UAAU;GAChB,MAAM,UAAU;GAChB,uBAAuB,UAAU;GAClC,CAAC;AACF,SAAO;GACL,MAAM;GACN,uBAAuB,UAAU;GACjC,UAAU;GACV,MAAM,UAAU;GACjB;UACM,KAAK;AACZ,OAAK,IAAI,KAAK;GAAE;GAAK;GAAY,EAAE,qBAAqB;AACxD,SAAO;;;AAIX,eAAsB,gCACpB,cACA,YACA,KAOe;CACf,MAAM,SAAS,MAAM,aAAa,KAAK,WAAW;AAClD,MAAK,IAAI,IAAI,OAAO,SAAS,GAAG,KAAK,GAAG,KAAK;EAC3C,MAAM,IAAI,OAAO;AACjB,MAAI,EAAE,SAAS,aAAa;GAC1B,MAAM,OAAQ,EAAE,eAAe,EAAE;AACjC,OAAI,KAAK,MAAM,MAAM,EAAE,0BAA0B,IAAI,sBAAsB,CACzE;GAEF,MAAM,OAAO,CAAC,GAAG,MAAM,IAAI;AAC3B,UAAO,KAAK;IAAE,GAAG;IAAG,aAAa;IAAM;AACvC,SAAM,aAAa,aAAa,YAAY,OAAO;AACnD"}
1
+ {"version":3,"file":"webchat-tts.js","names":[],"sources":["../../../../src/agent/service/webchat-tts.ts"],"sourcesContent":["import type { Config } from '../../config/schema.js';\nimport { persistOutboundTtsAudio } from '../../channels/attachments/outbound-tts-persist.js';\nimport { compressAudio } from '../../voice/tts/audio.js';\nimport { speak } from '../../voice/tts/index.js';\nimport { mergeTtsConfigFromAppConfig } from '../../voice/tts/merge-config.js';\nimport { shouldUseTTS, getChannelOutputFormat } from '../../voice/tts/service.js';\nimport { isTTSAvailable } from '../../voice/tts/factory.js';\nimport { resolveAgentHomeDir } from '../agent-scope.js';\nimport { extractProfileAgentId } from '../../config/agent-profile.js';\nimport type { SessionStore } from '../../session/index.js';\nimport type { AgentMessage } from '@earendil-works/pi-agent-core';\n\nexport type WebchatTtsResult = {\n type: 'tts_audio';\n workspaceRelativePath: string;\n mimeType: string;\n name: string;\n};\n\nexport type WebchatTtsDeps = {\n config: Config | undefined;\n sessionStore: SessionStore;\n getLastAssistantPlainText: (sessionKey: string) => string;\n log: { warn: (obj: Record<string, unknown>, msg: string) => void };\n};\n\n/**\n * Generate TTS for webchat when config allows; persist under agent home `tts/`.\n */\nexport async function maybeEmitWebchatTts(\n deps: WebchatTtsDeps,\n sessionKey: string,\n hadInboundVoice: boolean,\n): Promise<WebchatTtsResult | null> {\n const ttsConfig = mergeTtsConfigFromAppConfig(deps.config?.messages?.tts);\n if (!isTTSAvailable(ttsConfig)) {\n return null;\n }\n const decision = shouldUseTTS(ttsConfig, hadInboundVoice);\n if (!decision.useTTS) {\n return null;\n }\n const text = deps.getLastAssistantPlainText(sessionKey).trim();\n if (!text) {\n return null;\n }\n try {\n const webOut = getChannelOutputFormat('webchat');\n const fmt = webOut.format as 'opus' | 'mp3' | 'wav';\n const ttsResult = await speak(text, ttsConfig, {\n appConfig: deps.config,\n tts: { format: fmt },\n });\n const { buffer, format } = await compressAudio(\n Buffer.from(ttsResult.audio),\n ttsResult.format,\n webOut.format === 'mp3' ? 'mp3' : 'opus',\n );\n const normalizedMime =\n format === 'opus' || format === 'ogg'\n ? 'audio/ogg'\n : format === 'mp3' || format === 'mpeg'\n ? 'audio/mpeg'\n : format === 'wav'\n ? 'audio/wav'\n : `audio/${format}`;\n const cfg = deps.config!;\n const persisted = await persistOutboundTtsAudio(\n resolveAgentHomeDir(cfg, extractProfileAgentId(sessionKey, cfg)),\n sessionKey,\n buffer,\n format,\n );\n await appendAttachmentToLastAssistant(deps.sessionStore, sessionKey, {\n type: 'audio',\n mimeType: normalizedMime,\n name: persisted.name,\n size: persisted.size,\n workspaceRelativePath: persisted.workspaceRelativePath,\n });\n return {\n type: 'tts_audio',\n workspaceRelativePath: persisted.workspaceRelativePath,\n mimeType: normalizedMime,\n name: persisted.name,\n };\n } catch (err) {\n deps.log.warn({ err, sessionKey }, 'Webchat TTS failed');\n return null;\n }\n}\n\nexport async function appendAttachmentToLastAssistant(\n sessionStore: SessionStore,\n sessionKey: string,\n att: {\n type: string;\n mimeType: string;\n name: string;\n size: number;\n workspaceRelativePath: string;\n },\n): Promise<void> {\n const loaded = await sessionStore.load(sessionKey);\n for (let i = loaded.length - 1; i >= 0; i--) {\n const m = loaded[i] as { role?: string; attachments?: unknown[] };\n if (m.role === 'assistant') {\n const prev = (m.attachments ?? []) as Array<{ workspaceRelativePath?: string }>;\n if (prev.some((x) => x.workspaceRelativePath === att.workspaceRelativePath)) {\n return;\n }\n const next = [...prev, att];\n loaded[i] = { ...m, attachments: next } as unknown as AgentMessage;\n await sessionStore.saveMessages(sessionKey, loaded);\n return;\n }\n }\n}\n"],"mappings":";;;;;;;;;;kBAOwD;;;;AAsBxD,eAAsB,oBACpB,MACA,YACA,iBACkC;CAClC,MAAM,YAAY,4BAA4B,KAAK,QAAQ,UAAU,IAAI;AACzE,KAAI,CAAC,eAAe,UAAU,CAC5B,QAAO;AAGT,KAAI,CADa,aAAa,WAAW,gBAC5B,CAAC,OACZ,QAAO;CAET,MAAM,OAAO,KAAK,0BAA0B,WAAW,CAAC,MAAM;AAC9D,KAAI,CAAC,KACH,QAAO;AAET,KAAI;EACF,MAAM,SAAS,uBAAuB,UAAU;EAChD,MAAM,MAAM,OAAO;EACnB,MAAM,YAAY,MAAM,MAAM,MAAM,WAAW;GAC7C,WAAW,KAAK;GAChB,KAAK,EAAE,QAAQ,KAAK;GACrB,CAAC;EACF,MAAM,EAAE,QAAQ,WAAW,MAAM,cAC/B,OAAO,KAAK,UAAU,MAAM,EAC5B,UAAU,QACV,OAAO,WAAW,QAAQ,QAAQ,OACnC;EACD,MAAM,iBACJ,WAAW,UAAU,WAAW,QAC5B,cACA,WAAW,SAAS,WAAW,SAC7B,eACA,WAAW,QACT,cACA,SAAS;EACnB,MAAM,MAAM,KAAK;EACjB,MAAM,YAAY,MAAM,wBACtB,oBAAoB,KAAK,sBAAsB,YAAY,IAAI,CAAC,EAChE,YACA,QACA,OACD;AACD,QAAM,gCAAgC,KAAK,cAAc,YAAY;GACnE,MAAM;GACN,UAAU;GACV,MAAM,UAAU;GAChB,MAAM,UAAU;GAChB,uBAAuB,UAAU;GAClC,CAAC;AACF,SAAO;GACL,MAAM;GACN,uBAAuB,UAAU;GACjC,UAAU;GACV,MAAM,UAAU;GACjB;UACM,KAAK;AACZ,OAAK,IAAI,KAAK;GAAE;GAAK;GAAY,EAAE,qBAAqB;AACxD,SAAO;;;AAIX,eAAsB,gCACpB,cACA,YACA,KAOe;CACf,MAAM,SAAS,MAAM,aAAa,KAAK,WAAW;AAClD,MAAK,IAAI,IAAI,OAAO,SAAS,GAAG,KAAK,GAAG,KAAK;EAC3C,MAAM,IAAI,OAAO;AACjB,MAAI,EAAE,SAAS,aAAa;GAC1B,MAAM,OAAQ,EAAE,eAAe,EAAE;AACjC,OAAI,KAAK,MAAM,MAAM,EAAE,0BAA0B,IAAI,sBAAsB,CACzE;GAEF,MAAM,OAAO,CAAC,GAAG,MAAM,IAAI;AAC3B,UAAO,KAAK;IAAE,GAAG;IAAG,aAAa;IAAM;AACvC,SAAM,aAAa,aAAa,YAAY,OAAO;AACnD"}
@@ -1,9 +1,9 @@
1
+ import { createLogger } from "../utils/logger/index.js";
2
+ import { init_logger } from "../utils/logger.js";
3
+ import { init_agent_scope, resolveAgentHomeDir, resolveAgentProfileDir, resolveDefaultAgentId } from "./agent-scope.js";
1
4
  import { getAgentDefaultModelRef, init_schema } from "../config/schema.js";
2
5
  import { applyConfigOverrides } from "../config/runtime-overrides.js";
3
- import { init_agent_scope, resolveAgentHomeDir, resolveAgentProfileDir, resolveDefaultAgentId } from "./agent-scope.js";
4
6
  import { extractProfileAgentId, resolveEffectiveAgentProfileForSession } from "../config/agent-profile.js";
5
- import { createLogger } from "../utils/logger/index.js";
6
- import { init_logger } from "../utils/logger.js";
7
7
  import { extractTextContent } from "./context/workspace.js";
8
8
  import { onSessionTranscriptUpdate } from "../session/transcript-events.js";
9
9
  import { getWorkflowProgressBroker } from "./workflow/progress-broker.js";
@@ -59,8 +59,8 @@ import { reconcileManagedDreamingCronJobs } from "./service/reconcile-dreaming-c
59
59
  import { parseOutboundSessionKey } from "./service/parse-outbound-session-key.js";
60
60
  import { cleanTrailingErrors } from "./memory/message-sanitizer.js";
61
61
  import { tryApplySessionTranscriptHygiene } from "./transcript/transcript-hygiene.js";
62
- import { join } from "node:path";
63
62
  import { existsSync, readFileSync } from "node:fs";
63
+ import { join } from "node:path";
64
64
  //#region src/agent/service.ts
65
65
  init_schema();
66
66
  init_logger();
@@ -1,6 +1,6 @@
1
- import { resolveEffectiveAgentProfileForSession } from "../../config/agent-profile.js";
2
1
  import { createLogger } from "../../utils/logger/index.js";
3
2
  import { init_logger } from "../../utils/logger.js";
3
+ import { resolveEffectiveAgentProfileForSession } from "../../config/agent-profile.js";
4
4
  import { resolveEffectiveReasoningLevel, resolveEffectiveThinkingLevel } from "../../session/thinking-resolve.js";
5
5
  import { effectiveWorkspacePathForSession } from "../../session/session-workspace.js";
6
6
  import "../../session/index.js";
@@ -1,7 +1,7 @@
1
1
  import { __toCommonJS } from "../../../_virtual/_rolldown/runtime.js";
2
+ import { init_write_file_atomic, writeTextAtomicSync } from "../../infra/write-file-atomic.js";
2
3
  import { createLogger } from "../../utils/logger/index.js";
3
4
  import { init_logger } from "../../utils/logger.js";
4
- import { init_write_file_atomic, writeTextAtomicSync } from "../../infra/write-file-atomic.js";
5
5
  import { init_installer, installer_exports } from "./installer.js";
6
6
  import { join } from "path";
7
7
  import { existsSync, readFileSync } from "fs";
@@ -1,6 +1,6 @@
1
- import { join, relative } from "node:path";
2
- import { createReadStream, existsSync, readFileSync, readdirSync, statSync } from "node:fs";
3
1
  import { createHash } from "node:crypto";
2
+ import { createReadStream, existsSync, readFileSync, readdirSync, statSync } from "node:fs";
3
+ import { join, relative } from "node:path";
4
4
  //#region src/agent/skills/hub-hash.ts
5
5
  /**
6
6
  * Deterministic content hash for a skill directory (for hub lock / drift detection).
@@ -1,5 +1,5 @@
1
- import { init_paths, resolveSkillsLockPath } from "../../config/paths.js";
2
1
  import { init_write_file_atomic, writeTextAtomicSync } from "../../infra/write-file-atomic.js";
2
+ import { init_paths, resolveSkillsLockPath } from "../../config/paths.js";
3
3
  import { existsSync, readFileSync } from "node:fs";
4
4
  //#region src/agent/skills/hub-lock.ts
5
5
  /**
@@ -4,9 +4,9 @@ import { formatScanSummary, scanSkillDirectory } from "./scanner.js";
4
4
  import { getSkillsLockEntry, recordSkillsHubInstall } from "./hub-lock.js";
5
5
  import { installSkillFromZip, isValidSkillId } from "./managed-store.js";
6
6
  import { computeSkillTreeHashSync } from "./hub-hash.js";
7
- import { tmpdir } from "node:os";
8
- import { basename, join, normalize, resolve, sep } from "node:path";
9
7
  import { cpSync, existsSync, mkdirSync, mkdtempSync, readFileSync, readdirSync, rmSync, statSync, writeFileSync } from "node:fs";
8
+ import { basename, join, normalize, resolve, sep } from "node:path";
9
+ import { tmpdir } from "node:os";
10
10
  import { execFileSync } from "node:child_process";
11
11
  import { fileURLToPath } from "node:url";
12
12
  import { fetch } from "undici";
@@ -1,6 +1,6 @@
1
- import { resolveStateDir } from "../../config/paths-state.js";
2
1
  import { createLogger } from "../../utils/logger/index.js";
3
2
  import { init_logger } from "../../utils/logger.js";
3
+ import { resolveStateDir } from "../../config/paths-state.js";
4
4
  import { init_paths } from "../../config/paths.js";
5
5
  import { parseFrontmatter } from "../../markdown/frontmatter.js";
6
6
  import { findInstallSpec, getDefaultInstallerPreferences, hasBinary, init_installer, installSkill } from "./installer.js";
@@ -1,8 +1,8 @@
1
1
  import { init_paths, resolveSkillsDir } from "../../config/paths.js";
2
2
  import { parseFrontmatter } from "../../markdown/frontmatter.js";
3
3
  import { loadSkillsLock } from "./hub-lock.js";
4
- import { dirname, join, resolve, sep } from "node:path";
5
4
  import { existsSync, mkdirSync, readFileSync, readdirSync, rmSync, writeFileSync } from "node:fs";
5
+ import { dirname, join, resolve, sep } from "node:path";
6
6
  import AdmZip from "adm-zip";
7
7
  //#region src/agent/skills/managed-store.ts
8
8
  /**
@@ -1,7 +1,7 @@
1
1
  import { createLogger } from "../../utils/logger/index.js";
2
2
  import { init_logger } from "../../utils/logger.js";
3
- import { join } from "path";
4
3
  import { readFile, readdir } from "fs/promises";
4
+ import { join } from "path";
5
5
  //#region src/agent/skills/scanner.ts
6
6
  /**
7
7
  * Skill security scanner — scans skill directories for potentially dangerous code patterns.
@@ -2,9 +2,9 @@ import { init_paths, resolveSkillsDir } from "../../config/paths.js";
2
2
  import { parseFrontmatter } from "../../markdown/frontmatter.js";
3
3
  import { fuzzyFindText, normalizeForFuzzyMatch, normalizeToLF, restoreLineEndings } from "../tools/edit-diff.js";
4
4
  import { formatScanSummary, scanSkillDirectory } from "./scanner.js";
5
+ import { mkdir, writeFile } from "fs/promises";
5
6
  import { join, resolve, sep } from "path";
6
7
  import { existsSync, realpathSync } from "fs";
7
- import { mkdir, writeFile } from "fs/promises";
8
8
  //#region src/agent/skills/skill-manage-ops.ts
9
9
  /**
10
10
  * Shared logic for the skill_manage agent tool (create / edit / patch / delete / write_file / remove_file).
@@ -1,6 +1,6 @@
1
- import { resolveStateDir } from "../../config/paths-state.js";
2
1
  import { createLogger } from "../../utils/logger/index.js";
3
2
  import { init_logger } from "../../utils/logger.js";
3
+ import { resolveStateDir } from "../../config/paths-state.js";
4
4
  import { init_paths, resolveBundledSkillsDir } from "../../config/paths.js";
5
5
  import { createSkillConfigManager, isSkillEnabled } from "./config.js";
6
6
  import { formatSkillsForPrompt, selectSkillsVisibleInPrompt } from "./format-skills-prompt.js";
@@ -3,8 +3,8 @@ import { init_logger } from "../../utils/logger.js";
3
3
  import { SHORT_TERM_PROMOTION_LOCK_RELATIVE, SHORT_TERM_RECALL_STORE_RELATIVE } from "../memory/dreaming/constants.js";
4
4
  import { loadDreamingStore, saveDreamingStore } from "../memory/dreaming/short-term-store.js";
5
5
  import { resolveDreamingConfig } from "../memory/dreaming/config.js";
6
- import path from "node:path";
7
6
  import fs from "node:fs/promises";
7
+ import path from "node:path";
8
8
  import { Type } from "@sinclair/typebox";
9
9
  //#region src/agent/tools/dreaming-tool.ts
10
10
  init_logger();
@@ -1,6 +1,6 @@
1
- import { init_session_key, parseSessionKey } from "../../routing/session-key.js";
2
1
  import { createLogger } from "../../utils/logger/index.js";
3
2
  import { init_logger } from "../../utils/logger.js";
3
+ import { init_session_key, parseSessionKey } from "../../routing/session-key.js";
4
4
  import { createReadFileTool } from "./read.js";
5
5
  import { createWriteFileTool } from "./write.js";
6
6
  import { createEditFileTool } from "./edit.js";
@@ -4,9 +4,9 @@ import { isFailoverError } from "../failover-error.js";
4
4
  import { imageAssetFromDataUrl, imageFileExtensionForMimeType, mimeTypeFromFileName, parseImageDataUrl, sniffImageMimeType } from "../image/generation/image-assets.js";
5
5
  import { getImageGenerationProvider } from "../image/generation/provider-registry.js";
6
6
  import { generateImage, listImageGenerationProvidersSummary } from "../image/generation/runtime.js";
7
- import path from "node:path";
8
7
  import { randomBytes } from "node:crypto";
9
8
  import { mkdir, readFile, writeFile } from "node:fs/promises";
9
+ import path from "node:path";
10
10
  import { Type } from "@sinclair/typebox";
11
11
  //#region src/agent/tools/image-generate-tool.ts
12
12
  const DEFAULT_COUNT = 1;
@@ -1,7 +1,7 @@
1
1
  import { checkFileSafety } from "../prompt/safety.js";
2
2
  import { resolvePathUnderWorkspace } from "./tool-paths.js";
3
- import { basename } from "node:path";
4
3
  import { readFile } from "fs/promises";
4
+ import { basename } from "node:path";
5
5
  import { Type } from "@sinclair/typebox";
6
6
  //#region src/agent/tools/send-media.ts
7
7
  const SendMediaSchema = Type.Object({
@@ -2,9 +2,9 @@ import { resolveStateDir } from "../../config/paths-state.js";
2
2
  import { init_paths } from "../../config/paths.js";
3
3
  import { createSkillConfigManager } from "../skills/config.js";
4
4
  import { applyPatchToContent, atomicWriteUtf8, effectiveAgentWritePolicy, ensureCategorySegment, isPathInsideDir, maxSkillMdChars, maxSupportFileBytes, mutatableSkillOrNull, resolveCreateSkillDir, scanSkillDirOrError, validateSkillMdContent, validateSkillNameSegment, validateSupportingRelativePath } from "../skills/skill-manage-ops.js";
5
+ import { readFile, rm } from "fs/promises";
5
6
  import { join } from "path";
6
7
  import { existsSync, rmSync } from "fs";
7
- import { readFile, rm } from "fs/promises";
8
8
  import { Type } from "@sinclair/typebox";
9
9
  //#region src/agent/tools/skill-manage-tool.ts
10
10
  init_paths();
@@ -1,12 +1,15 @@
1
1
  import { createLogger } from "../../utils/logger/index.js";
2
2
  import { init_logger } from "../../utils/logger.js";
3
+ import { extractProfileAgentId } from "../../config/agent-profile.js";
3
4
  import { init_providers, resolveModel } from "../../providers/index.js";
5
+ import { applySubagentProgress } from "../workflow/agent-progress.js";
4
6
  import { parseWorkflowScript } from "../workflow/parser.js";
5
7
  import { getLastWorkflowMemory } from "../workflow/last-run-memory.js";
6
- import { previewValue, recomputeCounts, renderWorkflowText } from "../workflow/snapshot.js";
7
8
  import { runWorkflow } from "../workflow/runtime.js";
9
+ import { previewValue, recomputeCounts, renderWorkflowText } from "../workflow/snapshot.js";
8
10
  import { DelegateSubagentRunner } from "../workflow/subagent-runner.js";
9
11
  import "../workflow/index.js";
12
+ import { resolveModelRef } from "../../config/agent-typed-models.js";
10
13
  import { Type } from "@sinclair/typebox";
11
14
  //#region src/agent/tools/workflow-tool.ts
12
15
  /**
@@ -30,6 +33,7 @@ const DEFAULT_TIMEOUT_SEC = 1800;
30
33
  const MAX_TIMEOUT_SEC = 14400;
31
34
  const DEFAULT_MAX_CONCURRENCY = 16;
32
35
  const DEFAULT_MAX_SUBAGENTS = 1e3;
36
+ const PUSH_UPDATE_THROTTLE_MS = 300;
33
37
  const WorkflowToolSchema = Type.Object({
34
38
  name: Type.Optional(Type.String({ description: "Name of a saved workflow to run. Either `name` or `script` is required. Use `name` whenever the user references a known workflow (built-in or in ~/.xopc/workflows/)." })),
35
39
  script: Type.Optional(Type.String({ description: [
@@ -100,7 +104,7 @@ function createWorkflowTool(deps) {
100
104
  const snapshot = {
101
105
  name: meta.name,
102
106
  description: meta.description,
103
- phases: [],
107
+ phases: meta.phases?.map((p) => p.title) ?? [],
104
108
  logs: [],
105
109
  agents: [],
106
110
  agentCount: 0,
@@ -109,16 +113,33 @@ function createWorkflowTool(deps) {
109
113
  errorCount: 0,
110
114
  skippedCount: 0
111
115
  };
112
- const pushUpdate = (completed = false) => {
116
+ let lastUpdatePushedAtMs = Number.NEGATIVE_INFINITY;
117
+ let liveUpdatesDisabled = false;
118
+ const pushUpdate = (completed = false, immediate = false) => {
119
+ if (liveUpdatesDisabled) return;
113
120
  recomputeCounts(snapshot);
114
- onUpdate?.({
115
- content: [{
116
- type: "text",
117
- text: renderWorkflowText(snapshot, completed, { showResultPreviews: false })
118
- }],
119
- details: snapshot
120
- });
121
+ const nowMs = Date.now();
122
+ if (!(completed || immediate || nowMs - lastUpdatePushedAtMs >= PUSH_UPDATE_THROTTLE_MS)) return;
123
+ lastUpdatePushedAtMs = nowMs;
124
+ try {
125
+ onUpdate?.({
126
+ content: [{
127
+ type: "text",
128
+ text: renderWorkflowText(snapshot, completed, { showResultPreviews: false })
129
+ }],
130
+ details: snapshot
131
+ });
132
+ } catch (e) {
133
+ liveUpdatesDisabled = true;
134
+ const message = e instanceof Error ? e.message : String(e);
135
+ log.warn({
136
+ err: e,
137
+ errorMessage: message,
138
+ workflow: meta.name
139
+ }, `workflow live progress disabled: ${message}`);
140
+ }
121
141
  };
142
+ const subagentStream = wfCfg?.subagentStream ?? "steps";
122
143
  const runner = new DelegateSubagentRunner({
123
144
  workspace: deps.workspace,
124
145
  bus: deps.bus,
@@ -127,7 +148,12 @@ function createWorkflowTool(deps) {
127
148
  toolExecutorConfig: deps.toolExecutorConfig,
128
149
  buildChildTools: deps.buildChildTools
129
150
  });
130
- const resolveModelId = (modelId) => resolveModel(modelId);
151
+ const resolveModelId = (modelRef) => {
152
+ const config = deps.getConfig();
153
+ if (!config) throw new Error("workflow model resolution requires config");
154
+ const sessionKey = deps.getCurrentSessionKey?.();
155
+ return resolveModel(resolveModelRef(config, extractProfileAgentId(sessionKey, config), modelRef));
156
+ };
131
157
  const controller = new AbortController();
132
158
  const onParentAbort = () => controller.abort();
133
159
  signal?.addEventListener("abort", onParentAbort, { once: true });
@@ -152,24 +178,46 @@ function createWorkflowTool(deps) {
152
178
  if (!snapshot.phases.includes(title)) snapshot.phases.push(title);
153
179
  pushUpdate();
154
180
  },
155
- onAgentStart: (event) => {
181
+ onAgentQueued: (event) => {
156
182
  snapshot.agents.push({
157
183
  id: event.id,
158
184
  label: event.label,
159
185
  phase: event.phase,
160
186
  prompt: event.prompt,
161
- status: "running"
187
+ status: "queued"
162
188
  });
163
- pushUpdate();
189
+ pushUpdate(false, true);
190
+ },
191
+ onAgentStart: (event) => {
192
+ const agent = findAgentById(snapshot.agents, event.id);
193
+ if (agent) {
194
+ agent.status = "running";
195
+ agent.startedAtMs = Date.now();
196
+ } else snapshot.agents.push({
197
+ id: event.id,
198
+ label: event.label,
199
+ phase: event.phase,
200
+ prompt: event.prompt,
201
+ status: "running",
202
+ startedAtMs: Date.now()
203
+ });
204
+ pushUpdate(false, true);
164
205
  },
165
206
  onAgentEnd: (event) => {
166
207
  const agent = findAgentById(snapshot.agents, event.id);
167
208
  if (agent) {
168
209
  agent.status = event.status;
169
210
  agent.resultPreview = previewValue(event.result);
211
+ if (agent.startedAtMs != null) agent.durationMs = Date.now() - agent.startedAtMs;
212
+ agent.currentStep = void 0;
170
213
  }
171
- pushUpdate();
172
- }
214
+ pushUpdate(false, true);
215
+ },
216
+ enhanceSubagentRun: subagentStream === "off" ? void 0 : ({ id }) => ({ onProgress: (event) => {
217
+ const agent = findAgentById(snapshot.agents, id);
218
+ if (!agent) return;
219
+ if (applySubagentProgress(agent, event)) pushUpdate();
220
+ } })
173
221
  });
174
222
  if (result.agentCount === 0) {
175
223
  const reason = "workflow scripts must call agent() at least once; this workflow declared phases but never ran a subagent.";