@copilotkit/runtime 1.57.3 → 1.58.0-canary.thread-id-propagation

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 (346) hide show
  1. package/dist/lib/observability.d.cts +1 -1
  2. package/dist/lib/observability.d.cts.map +1 -1
  3. package/dist/lib/observability.d.mts +1 -1
  4. package/dist/lib/observability.d.mts.map +1 -1
  5. package/dist/lib/runtime/copilot-runtime.cjs +2 -0
  6. package/dist/lib/runtime/copilot-runtime.cjs.map +1 -1
  7. package/dist/lib/runtime/copilot-runtime.d.cts.map +1 -1
  8. package/dist/lib/runtime/copilot-runtime.d.mts.map +1 -1
  9. package/dist/lib/runtime/copilot-runtime.mjs +2 -0
  10. package/dist/lib/runtime/copilot-runtime.mjs.map +1 -1
  11. package/dist/package.cjs +5 -8
  12. package/dist/package.mjs +5 -8
  13. package/dist/v2/runtime/core/runtime.cjs +4 -1
  14. package/dist/v2/runtime/core/runtime.cjs.map +1 -1
  15. package/dist/v2/runtime/core/runtime.d.cts.map +1 -1
  16. package/dist/v2/runtime/core/runtime.d.mts.map +1 -1
  17. package/dist/v2/runtime/core/runtime.mjs +4 -1
  18. package/dist/v2/runtime/core/runtime.mjs.map +1 -1
  19. package/dist/v2/runtime/handlers/get-runtime-info.cjs +1 -1
  20. package/dist/v2/runtime/handlers/get-runtime-info.mjs +1 -1
  21. package/dist/v2/runtime/handlers/handle-connect.cjs +1 -1
  22. package/dist/v2/runtime/handlers/handle-connect.mjs +1 -1
  23. package/dist/v2/runtime/handlers/handle-run.cjs +1 -1
  24. package/dist/v2/runtime/handlers/handle-run.mjs +1 -1
  25. package/dist/v2/runtime/handlers/shared/agent-utils.cjs +1 -1
  26. package/dist/v2/runtime/handlers/shared/agent-utils.cjs.map +1 -1
  27. package/dist/v2/runtime/handlers/shared/agent-utils.mjs +1 -1
  28. package/dist/v2/runtime/handlers/shared/agent-utils.mjs.map +1 -1
  29. package/dist/v2/runtime/telemetry/telemetry-client.cjs +22 -6
  30. package/dist/v2/runtime/telemetry/telemetry-client.cjs.map +1 -1
  31. package/dist/v2/runtime/telemetry/telemetry-client.mjs +27 -11
  32. package/dist/v2/runtime/telemetry/telemetry-client.mjs.map +1 -1
  33. package/package.json +9 -19
  34. package/skills/runtime/SKILL.md +98 -0
  35. package/skills/runtime/references/agent-runners-custom.md +161 -0
  36. package/skills/runtime/references/agent-runners-in-memory.md +64 -0
  37. package/skills/runtime/references/agent-runners-sqlite.md +90 -0
  38. package/skills/runtime/references/agent-runners.md +304 -0
  39. package/skills/runtime/references/built-in-agent-factory-modes.md +232 -0
  40. package/skills/runtime/references/built-in-agent-helper-utilities.md +123 -0
  41. package/skills/runtime/references/built-in-agent-model-identifiers.md +59 -0
  42. package/skills/runtime/references/built-in-agent.md +523 -0
  43. package/skills/runtime/references/intelligence-mode.md +336 -0
  44. package/skills/runtime/references/middleware.md +376 -0
  45. package/skills/runtime/references/server-side-tools.md +414 -0
  46. package/skills/runtime/references/setup-endpoint.md +503 -0
  47. package/skills/runtime/references/transcription.md +287 -0
  48. package/skills/runtime/references/wiring-a2a.md +40 -0
  49. package/skills/runtime/references/wiring-adk.md +45 -0
  50. package/skills/runtime/references/wiring-ag2.md +41 -0
  51. package/skills/runtime/references/wiring-agno.md +39 -0
  52. package/skills/runtime/references/wiring-aws-strands.md +59 -0
  53. package/skills/runtime/references/wiring-crewai-crews.md +51 -0
  54. package/skills/runtime/references/wiring-crewai-flows.md +45 -0
  55. package/skills/runtime/references/wiring-external-agents.md +348 -0
  56. package/skills/runtime/references/wiring-langgraph.md +50 -0
  57. package/skills/runtime/references/wiring-llamaindex.md +39 -0
  58. package/skills/runtime/references/wiring-mastra.md +70 -0
  59. package/skills/runtime/references/wiring-mcp-apps-middleware.md +68 -0
  60. package/skills/runtime/references/wiring-ms-agent-framework.md +41 -0
  61. package/skills/runtime/references/wiring-pydantic-ai.md +45 -0
  62. package/CHANGELOG.md +0 -3624
  63. package/__snapshots__/schema/schema.graphql +0 -371
  64. package/dist/v2/runtime/telemetry/scarf-client.cjs +0 -32
  65. package/dist/v2/runtime/telemetry/scarf-client.cjs.map +0 -1
  66. package/dist/v2/runtime/telemetry/scarf-client.mjs +0 -32
  67. package/dist/v2/runtime/telemetry/scarf-client.mjs.map +0 -1
  68. package/scripts/generate-gql-schema.ts +0 -16
  69. package/src/agent/__tests__/agent-test-helpers.ts +0 -476
  70. package/src/agent/__tests__/agent.test.ts +0 -593
  71. package/src/agent/__tests__/ai-sdk-v6-compat.test.ts +0 -116
  72. package/src/agent/__tests__/basic-agent.test.ts +0 -1698
  73. package/src/agent/__tests__/capabilities.test.ts +0 -81
  74. package/src/agent/__tests__/config-tools-execution.test.ts +0 -516
  75. package/src/agent/__tests__/converter-aisdk.test.ts +0 -692
  76. package/src/agent/__tests__/converter-custom.test.ts +0 -319
  77. package/src/agent/__tests__/converter-tanstack-input.test.ts +0 -211
  78. package/src/agent/__tests__/converter-tanstack.test.ts +0 -594
  79. package/src/agent/__tests__/mcp-clients.test.ts +0 -246
  80. package/src/agent/__tests__/mcp-servers-integration.test.ts +0 -373
  81. package/src/agent/__tests__/multimodal-tanstack.test.ts +0 -284
  82. package/src/agent/__tests__/multimodal.test.ts +0 -176
  83. package/src/agent/__tests__/property-overrides.test.ts +0 -598
  84. package/src/agent/__tests__/provider-id-collision.test.ts +0 -195
  85. package/src/agent/__tests__/standard-schema-tools.test.ts +0 -313
  86. package/src/agent/__tests__/standard-schema-types.test.ts +0 -158
  87. package/src/agent/__tests__/state-tools.test.ts +0 -436
  88. package/src/agent/__tests__/test-helpers.ts +0 -197
  89. package/src/agent/__tests__/utils.test.ts +0 -536
  90. package/src/agent/__tests__/zod-regression.test.ts +0 -350
  91. package/src/agent/converters/aisdk.ts +0 -326
  92. package/src/agent/converters/index.ts +0 -7
  93. package/src/agent/converters/tanstack.ts +0 -451
  94. package/src/agent/index.ts +0 -1743
  95. package/src/agents/langgraph/__tests__/event-source.test.ts +0 -256
  96. package/src/agents/langgraph/event-source.ts +0 -365
  97. package/src/agents/langgraph/events.ts +0 -394
  98. package/src/graphql/inputs/action.input.ts +0 -16
  99. package/src/graphql/inputs/agent-session.input.ts +0 -13
  100. package/src/graphql/inputs/agent-state.input.ts +0 -13
  101. package/src/graphql/inputs/cloud-guardrails.input.ts +0 -16
  102. package/src/graphql/inputs/cloud.input.ts +0 -8
  103. package/src/graphql/inputs/context-property.input.ts +0 -10
  104. package/src/graphql/inputs/copilot-context.input.ts +0 -10
  105. package/src/graphql/inputs/custom-property.input.ts +0 -15
  106. package/src/graphql/inputs/extensions.input.ts +0 -21
  107. package/src/graphql/inputs/forwarded-parameters.input.ts +0 -22
  108. package/src/graphql/inputs/frontend.input.ts +0 -14
  109. package/src/graphql/inputs/generate-copilot-response.input.ts +0 -59
  110. package/src/graphql/inputs/load-agent-state.input.ts +0 -10
  111. package/src/graphql/inputs/message.input.ts +0 -110
  112. package/src/graphql/inputs/meta-event.input.ts +0 -18
  113. package/src/graphql/message-conversion/agui-to-gql.test.ts +0 -1384
  114. package/src/graphql/message-conversion/agui-to-gql.ts +0 -384
  115. package/src/graphql/message-conversion/gql-to-agui.test.ts +0 -1653
  116. package/src/graphql/message-conversion/gql-to-agui.ts +0 -297
  117. package/src/graphql/message-conversion/index.ts +0 -2
  118. package/src/graphql/message-conversion/roundtrip-conversion.test.ts +0 -561
  119. package/src/graphql/resolvers/__tests__/resolve-message-id.test.ts +0 -25
  120. package/src/graphql/resolvers/copilot.resolver.ts +0 -785
  121. package/src/graphql/resolvers/resolve-message-id.ts +0 -14
  122. package/src/graphql/resolvers/state.resolver.ts +0 -30
  123. package/src/graphql/types/agents-response.type.ts +0 -19
  124. package/src/graphql/types/base/index.ts +0 -10
  125. package/src/graphql/types/converted/index.ts +0 -183
  126. package/src/graphql/types/copilot-response.type.ts +0 -141
  127. package/src/graphql/types/enums.ts +0 -38
  128. package/src/graphql/types/extensions-response.type.ts +0 -23
  129. package/src/graphql/types/guardrails-result.type.ts +0 -20
  130. package/src/graphql/types/load-agent-state-response.type.ts +0 -17
  131. package/src/graphql/types/message-status.type.ts +0 -48
  132. package/src/graphql/types/meta-events.type.ts +0 -78
  133. package/src/graphql/types/response-status.type.ts +0 -77
  134. package/src/index.ts +0 -3
  135. package/src/langgraph.ts +0 -1
  136. package/src/lib/__tests__/telemetry-disclosure.test.ts +0 -55
  137. package/src/lib/cloud/index.ts +0 -4
  138. package/src/lib/error-messages.ts +0 -211
  139. package/src/lib/index.ts +0 -52
  140. package/src/lib/integrations/index.ts +0 -6
  141. package/src/lib/integrations/nest/index.ts +0 -21
  142. package/src/lib/integrations/nextjs/app-router.ts +0 -47
  143. package/src/lib/integrations/nextjs/pages-router.ts +0 -45
  144. package/src/lib/integrations/node-express/index.ts +0 -21
  145. package/src/lib/integrations/node-http/__tests__/request-duck-type.test.ts +0 -66
  146. package/src/lib/integrations/node-http/index.ts +0 -187
  147. package/src/lib/integrations/node-http/request-handler.ts +0 -128
  148. package/src/lib/integrations/shared.ts +0 -112
  149. package/src/lib/logger.ts +0 -31
  150. package/src/lib/observability.ts +0 -167
  151. package/src/lib/runtime/__tests__/copilot-runtime-error.test.ts +0 -183
  152. package/src/lib/runtime/__tests__/handle-service-adapter.test.ts +0 -108
  153. package/src/lib/runtime/__tests__/mcp-tools-utils.test.ts +0 -499
  154. package/src/lib/runtime/__tests__/on-after-request.test.ts +0 -122
  155. package/src/lib/runtime/__tests__/retry-utils.test.ts +0 -137
  156. package/src/lib/runtime/__tests__/v1-agent-factory.test.ts +0 -109
  157. package/src/lib/runtime/agent-integrations/langgraph/__tests__/dispatch-event-filtering.test.ts +0 -345
  158. package/src/lib/runtime/agent-integrations/langgraph/__tests__/run-message-filtering.test.ts +0 -156
  159. package/src/lib/runtime/agent-integrations/langgraph/agent.ts +0 -263
  160. package/src/lib/runtime/agent-integrations/langgraph/consts.ts +0 -37
  161. package/src/lib/runtime/agent-integrations/langgraph/index.ts +0 -2
  162. package/src/lib/runtime/copilot-runtime.ts +0 -863
  163. package/src/lib/runtime/mcp-tools-utils.ts +0 -313
  164. package/src/lib/runtime/retry-utils.ts +0 -141
  165. package/src/lib/runtime/telemetry-agent-runner.ts +0 -151
  166. package/src/lib/runtime/types.ts +0 -48
  167. package/src/lib/runtime/utils.ts +0 -93
  168. package/src/lib/streaming.ts +0 -220
  169. package/src/lib/telemetry-client.ts +0 -66
  170. package/src/lib/telemetry-disclosure.ts +0 -53
  171. package/src/service-adapters/anthropic/anthropic-adapter.ts +0 -532
  172. package/src/service-adapters/anthropic/utils.ts +0 -219
  173. package/src/service-adapters/bedrock/bedrock-adapter.ts +0 -73
  174. package/src/service-adapters/conversion.test.ts +0 -56
  175. package/src/service-adapters/conversion.ts +0 -69
  176. package/src/service-adapters/empty/empty-adapter.ts +0 -38
  177. package/src/service-adapters/events.ts +0 -337
  178. package/src/service-adapters/experimental/ollama/ollama-adapter.ts +0 -84
  179. package/src/service-adapters/google/google-genai-adapter.test.ts +0 -151
  180. package/src/service-adapters/google/google-genai-adapter.ts +0 -95
  181. package/src/service-adapters/groq/groq-adapter.ts +0 -229
  182. package/src/service-adapters/index.ts +0 -18
  183. package/src/service-adapters/langchain/langchain-adapter.ts +0 -113
  184. package/src/service-adapters/langchain/langserve.ts +0 -88
  185. package/src/service-adapters/langchain/types.ts +0 -20
  186. package/src/service-adapters/langchain/utils.ts +0 -330
  187. package/src/service-adapters/openai/__tests__/openai-v5-compat.test.ts +0 -177
  188. package/src/service-adapters/openai/openai-adapter.ts +0 -324
  189. package/src/service-adapters/openai/openai-assistant-adapter.ts +0 -385
  190. package/src/service-adapters/openai/utils.ts +0 -305
  191. package/src/service-adapters/service-adapter.ts +0 -50
  192. package/src/service-adapters/shared/error-utils.ts +0 -64
  193. package/src/service-adapters/shared/index.ts +0 -2
  194. package/src/service-adapters/shared/sdk-client-utils.ts +0 -19
  195. package/src/service-adapters/unify/unify-adapter.ts +0 -165
  196. package/src/utils/failed-response-status-reasons.ts +0 -70
  197. package/src/utils/index.ts +0 -1
  198. package/src/v2/express.ts +0 -1
  199. package/src/v2/hono.ts +0 -1
  200. package/src/v2/index.ts +0 -5
  201. package/src/v2/node.ts +0 -1
  202. package/src/v2/runtime/__tests__/agents-factory.test.ts +0 -136
  203. package/src/v2/runtime/__tests__/backward-compat.test.ts +0 -261
  204. package/src/v2/runtime/__tests__/code-review-fixes.test.ts +0 -500
  205. package/src/v2/runtime/__tests__/cors-credentials.test.ts +0 -320
  206. package/src/v2/runtime/__tests__/debug-sse-response.test.ts +0 -302
  207. package/src/v2/runtime/__tests__/express-adapter.test.ts +0 -188
  208. package/src/v2/runtime/__tests__/express-body-order.test.ts +0 -76
  209. package/src/v2/runtime/__tests__/express-fetch-bridge.test.ts +0 -344
  210. package/src/v2/runtime/__tests__/express-single-sse.test.ts +0 -122
  211. package/src/v2/runtime/__tests__/express-single-telemetry.integration.test.ts +0 -65
  212. package/src/v2/runtime/__tests__/express-telemetry.integration.test.ts +0 -101
  213. package/src/v2/runtime/__tests__/fetch-cors.test.ts +0 -205
  214. package/src/v2/runtime/__tests__/fetch-handler-validation.test.ts +0 -440
  215. package/src/v2/runtime/__tests__/fetch-handler.test.ts +0 -456
  216. package/src/v2/runtime/__tests__/fetch-router.test.ts +0 -276
  217. package/src/v2/runtime/__tests__/get-runtime-info.test.ts +0 -335
  218. package/src/v2/runtime/__tests__/handle-connect.test.ts +0 -585
  219. package/src/v2/runtime/__tests__/handle-run.test.ts +0 -1388
  220. package/src/v2/runtime/__tests__/handle-threads.test.ts +0 -930
  221. package/src/v2/runtime/__tests__/handle-transcribe.test.ts +0 -301
  222. package/src/v2/runtime/__tests__/header-utils.test.ts +0 -88
  223. package/src/v2/runtime/__tests__/hono-adapter.test.ts +0 -150
  224. package/src/v2/runtime/__tests__/hono-single-telemetry.integration.test.ts +0 -46
  225. package/src/v2/runtime/__tests__/hono-telemetry.integration.test.ts +0 -99
  226. package/src/v2/runtime/__tests__/hooks-edge-cases.test.ts +0 -457
  227. package/src/v2/runtime/__tests__/hooks.test.ts +0 -557
  228. package/src/v2/runtime/__tests__/in-process-agent-runner-messages.test.ts +0 -230
  229. package/src/v2/runtime/__tests__/in-process-agent-runner.test.ts +0 -1030
  230. package/src/v2/runtime/__tests__/integration/bun/bun-servers.integration.test.ts +0 -27
  231. package/src/v2/runtime/__tests__/integration/bun/elysia-multi.ts +0 -32
  232. package/src/v2/runtime/__tests__/integration/bun/elysia-single.ts +0 -33
  233. package/src/v2/runtime/__tests__/integration/bun/hono-bun-multi.ts +0 -25
  234. package/src/v2/runtime/__tests__/integration/bun/hono-bun-single.ts +0 -32
  235. package/src/v2/runtime/__tests__/integration/helpers/create-test-runtime.ts +0 -15
  236. package/src/v2/runtime/__tests__/integration/helpers/sse-reader.ts +0 -45
  237. package/src/v2/runtime/__tests__/integration/helpers/test-agent.ts +0 -58
  238. package/src/v2/runtime/__tests__/integration/node-servers.integration.test.ts +0 -58
  239. package/src/v2/runtime/__tests__/integration/servers/express-multi.ts +0 -35
  240. package/src/v2/runtime/__tests__/integration/servers/express-single.ts +0 -36
  241. package/src/v2/runtime/__tests__/integration/servers/fetch-direct.ts +0 -39
  242. package/src/v2/runtime/__tests__/integration/servers/hono-multi.ts +0 -30
  243. package/src/v2/runtime/__tests__/integration/servers/hono-single.ts +0 -37
  244. package/src/v2/runtime/__tests__/integration/servers/node-multi.ts +0 -45
  245. package/src/v2/runtime/__tests__/integration/servers/node-single.ts +0 -46
  246. package/src/v2/runtime/__tests__/integration/servers/types.ts +0 -18
  247. package/src/v2/runtime/__tests__/integration/suites/debug-events.suite.ts +0 -253
  248. package/src/v2/runtime/__tests__/integration/suites/multi-endpoint.suite.ts +0 -358
  249. package/src/v2/runtime/__tests__/integration/suites/single-endpoint.suite.ts +0 -363
  250. package/src/v2/runtime/__tests__/intelligence-run-telemetry.test.ts +0 -194
  251. package/src/v2/runtime/__tests__/mcp-apps-middleware-integration.test.ts +0 -275
  252. package/src/v2/runtime/__tests__/middleware-express.test.ts +0 -208
  253. package/src/v2/runtime/__tests__/middleware-single-express.test.ts +0 -213
  254. package/src/v2/runtime/__tests__/middleware-single.test.ts +0 -225
  255. package/src/v2/runtime/__tests__/middleware-sse-parser.test.ts +0 -237
  256. package/src/v2/runtime/__tests__/middleware.test.ts +0 -250
  257. package/src/v2/runtime/__tests__/node-fetch-handler.test.ts +0 -157
  258. package/src/v2/runtime/__tests__/open-generative-ui-middleware.e2e.test.ts +0 -728
  259. package/src/v2/runtime/__tests__/router-edge-cases.test.ts +0 -217
  260. package/src/v2/runtime/__tests__/routing-express.test.ts +0 -174
  261. package/src/v2/runtime/__tests__/routing-single-express.test.ts +0 -168
  262. package/src/v2/runtime/__tests__/routing-single.test.ts +0 -193
  263. package/src/v2/runtime/__tests__/routing.test.ts +0 -257
  264. package/src/v2/runtime/__tests__/runtime.test.ts +0 -234
  265. package/src/v2/runtime/__tests__/sse-response-telemetry.test.ts +0 -108
  266. package/src/v2/runtime/__tests__/telemetry.test.ts +0 -167
  267. package/src/v2/runtime/__tests__/thread-names.test.ts +0 -188
  268. package/src/v2/runtime/core/__tests__/debug-event-bus.test.ts +0 -156
  269. package/src/v2/runtime/core/debug-event-bus.ts +0 -45
  270. package/src/v2/runtime/core/fetch-cors.ts +0 -136
  271. package/src/v2/runtime/core/fetch-handler.ts +0 -492
  272. package/src/v2/runtime/core/fetch-router.ts +0 -203
  273. package/src/v2/runtime/core/hooks.ts +0 -160
  274. package/src/v2/runtime/core/middleware-sse-parser.ts +0 -210
  275. package/src/v2/runtime/core/middleware.ts +0 -115
  276. package/src/v2/runtime/core/runtime.ts +0 -432
  277. package/src/v2/runtime/endpoints/express-fetch-bridge.ts +0 -137
  278. package/src/v2/runtime/endpoints/express-single.ts +0 -54
  279. package/src/v2/runtime/endpoints/express.ts +0 -179
  280. package/src/v2/runtime/endpoints/hono-single.ts +0 -60
  281. package/src/v2/runtime/endpoints/hono.ts +0 -89
  282. package/src/v2/runtime/endpoints/index.ts +0 -4
  283. package/src/v2/runtime/endpoints/node-fetch-handler.ts +0 -48
  284. package/src/v2/runtime/endpoints/node.ts +0 -28
  285. package/src/v2/runtime/endpoints/single-route-helpers.ts +0 -125
  286. package/src/v2/runtime/express.ts +0 -2
  287. package/src/v2/runtime/handlers/__tests__/handle-debug-events.test.ts +0 -176
  288. package/src/v2/runtime/handlers/get-runtime-info.ts +0 -101
  289. package/src/v2/runtime/handlers/handle-connect.ts +0 -80
  290. package/src/v2/runtime/handlers/handle-debug-events.ts +0 -52
  291. package/src/v2/runtime/handlers/handle-run.ts +0 -111
  292. package/src/v2/runtime/handlers/handle-stop.ts +0 -77
  293. package/src/v2/runtime/handlers/handle-threads.ts +0 -11
  294. package/src/v2/runtime/handlers/handle-transcribe.ts +0 -269
  295. package/src/v2/runtime/handlers/header-utils.ts +0 -24
  296. package/src/v2/runtime/handlers/intelligence/connect.ts +0 -102
  297. package/src/v2/runtime/handlers/intelligence/run.ts +0 -351
  298. package/src/v2/runtime/handlers/intelligence/thread-names.ts +0 -246
  299. package/src/v2/runtime/handlers/intelligence/threads.ts +0 -420
  300. package/src/v2/runtime/handlers/shared/agent-utils.ts +0 -154
  301. package/src/v2/runtime/handlers/shared/intelligence-utils.ts +0 -41
  302. package/src/v2/runtime/handlers/shared/json-response.ts +0 -9
  303. package/src/v2/runtime/handlers/shared/resolve-intelligence-user.ts +0 -28
  304. package/src/v2/runtime/handlers/shared/sse-response.ts +0 -215
  305. package/src/v2/runtime/handlers/sse/__tests__/sse-connect-agent-id.test.ts +0 -71
  306. package/src/v2/runtime/handlers/sse/connect.ts +0 -30
  307. package/src/v2/runtime/handlers/sse/run.ts +0 -40
  308. package/src/v2/runtime/hono.ts +0 -2
  309. package/src/v2/runtime/index.ts +0 -51
  310. package/src/v2/runtime/intelligence-platform/__tests__/client.test.ts +0 -601
  311. package/src/v2/runtime/intelligence-platform/__tests__/intelligence-mcp-helper.test.ts +0 -246
  312. package/src/v2/runtime/intelligence-platform/client.ts +0 -818
  313. package/src/v2/runtime/intelligence-platform/index.ts +0 -10
  314. package/src/v2/runtime/node.ts +0 -6
  315. package/src/v2/runtime/open-generative-ui-middleware.ts +0 -373
  316. package/src/v2/runtime/runner/__tests__/finalize-events.test.ts +0 -109
  317. package/src/v2/runtime/runner/__tests__/in-memory-runner.e2e.test.ts +0 -775
  318. package/src/v2/runtime/runner/__tests__/in-memory-runner.test.ts +0 -777
  319. package/src/v2/runtime/runner/__tests__/intelligence-runner.test.ts +0 -1039
  320. package/src/v2/runtime/runner/agent-runner.ts +0 -35
  321. package/src/v2/runtime/runner/in-memory.ts +0 -467
  322. package/src/v2/runtime/runner/index.ts +0 -4
  323. package/src/v2/runtime/runner/intelligence.ts +0 -498
  324. package/src/v2/runtime/telemetry/__tests__/instance-created.test.ts +0 -96
  325. package/src/v2/runtime/telemetry/events.ts +0 -35
  326. package/src/v2/runtime/telemetry/index.ts +0 -7
  327. package/src/v2/runtime/telemetry/instance-created.ts +0 -44
  328. package/src/v2/runtime/telemetry/scarf-client.ts +0 -39
  329. package/src/v2/runtime/telemetry/telemetry-client.ts +0 -70
  330. package/src/v2/runtime/transcription-service/transcription-service.ts +0 -11
  331. package/tests/global.d.ts +0 -1
  332. package/tests/service-adapters/anthropic/allowlist-approach.test.ts +0 -259
  333. package/tests/service-adapters/anthropic/anthropic-adapter-language-model.test.ts +0 -101
  334. package/tests/service-adapters/anthropic/anthropic-adapter.test.ts +0 -645
  335. package/tests/service-adapters/anthropic/utils-token-trimming.test.ts +0 -301
  336. package/tests/service-adapters/groq/groq-adapter-language-model.test.ts +0 -102
  337. package/tests/service-adapters/openai/allowlist-approach.test.ts +0 -294
  338. package/tests/service-adapters/openai/openai-adapter-language-model.test.ts +0 -122
  339. package/tests/service-adapters/openai/openai-adapter.test.ts +0 -291
  340. package/tests/service-adapters/shared/sdk-client-utils.test.ts +0 -36
  341. package/tests/setup.vitest.ts +0 -8
  342. package/tests/tsconfig.json +0 -10
  343. package/tsconfig.json +0 -20
  344. package/tsdown.config.ts +0 -45
  345. package/typedoc.json +0 -4
  346. package/vitest.config.mjs +0 -13
@@ -1,1388 +0,0 @@
1
- import { Observable } from "rxjs";
2
- import { describe, it, expect, vi } from "vitest";
3
- import {
4
- AbstractAgent,
5
- BaseEvent,
6
- EventType,
7
- HttpAgent,
8
- RunAgentInput,
9
- } from "@ag-ui/client";
10
- import { A2UIMiddleware } from "@ag-ui/a2ui-middleware";
11
- import { handleRunAgent } from "../handlers/handle-run";
12
- import { CopilotRuntime } from "../core/runtime";
13
- import { IntelligenceAgentRunner } from "../runner/intelligence";
14
- import { InMemoryAgentRunner } from "../runner/in-memory";
15
-
16
- describe("handleRunAgent", () => {
17
- const createMockRuntime = (
18
- agents: Record<string, unknown> = {},
19
- ): CopilotRuntime => {
20
- return {
21
- agents: Promise.resolve(agents),
22
- transcriptionService: undefined,
23
- beforeRequestMiddleware: undefined,
24
- afterRequestMiddleware: undefined,
25
- } as CopilotRuntime;
26
- };
27
-
28
- const createMockRequest = (): Request => {
29
- return new Request("https://example.com/agent/test/run", {
30
- method: "POST",
31
- });
32
- };
33
-
34
- it("should return 404 when agent does not exist", async () => {
35
- const runtime = createMockRuntime({}); // Empty agents
36
- const request = createMockRequest();
37
- const agentId = "nonexistent-agent";
38
-
39
- const response = await handleRunAgent({
40
- runtime,
41
- request,
42
- agentId,
43
- });
44
-
45
- expect(response.status).toBe(404);
46
- expect(response.headers.get("Content-Type")).toBe("application/json");
47
-
48
- const body = await response.json();
49
- expect(body).toEqual({
50
- error: "Agent not found",
51
- message: "Agent 'nonexistent-agent' does not exist",
52
- });
53
- });
54
-
55
- it("should return 500 when runtime.agents throws an error", async () => {
56
- const runtime = {
57
- agents: Promise.reject(new Error("Database connection failed")),
58
- transcriptionService: undefined,
59
- beforeRequestMiddleware: undefined,
60
- afterRequestMiddleware: undefined,
61
- } as CopilotRuntime;
62
- const request = createMockRequest();
63
- const agentId = "test-agent";
64
-
65
- const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
66
-
67
- try {
68
- const response = await handleRunAgent({
69
- runtime,
70
- request,
71
- agentId,
72
- });
73
-
74
- expect(response.status).toBe(500);
75
- expect(response.headers.get("Content-Type")).toBe("application/json");
76
-
77
- const body = await response.json();
78
- expect(body).toEqual({
79
- error: "Failed to run agent",
80
- message: "Database connection failed",
81
- });
82
- } finally {
83
- errorSpy.mockRestore();
84
- }
85
- });
86
-
87
- it("forwards only authorization and custom x- headers to HttpAgent runs", async () => {
88
- class RecordingHttpAgent extends HttpAgent {
89
- constructor(initialHeaders: Record<string, string>) {
90
- super({ url: "https://runtime.example/agent" });
91
- this.headers = initialHeaders;
92
- }
93
-
94
- clone(): AbstractAgent {
95
- return new RecordingHttpAgent({});
96
- }
97
- }
98
-
99
- const baseHeaders = {
100
- Authorization: "Bearer base",
101
- };
102
-
103
- const registeredAgent = new RecordingHttpAgent(baseHeaders);
104
-
105
- const recordedHeaders: Array<Record<string, string>> = [];
106
- let resolveRun: (() => void) | undefined;
107
- const runInvoked = new Promise<void>((resolve) => {
108
- resolveRun = resolve;
109
- });
110
-
111
- const runtime = {
112
- agents: Promise.resolve({ "test-agent": registeredAgent }),
113
- transcriptionService: undefined,
114
- beforeRequestMiddleware: undefined,
115
- afterRequestMiddleware: undefined,
116
- runner: {
117
- run: ({ agent }: { agent: AbstractAgent }) =>
118
- new Observable<BaseEvent>((subscriber) => {
119
- recordedHeaders.push({ ...(agent as HttpAgent).headers });
120
- resolveRun?.();
121
- subscriber.complete();
122
- }),
123
- connect: () =>
124
- new Observable<BaseEvent>((subscriber) => {
125
- subscriber.complete();
126
- }),
127
- isRunning: async () => false,
128
- stop: async () => false,
129
- },
130
- } as CopilotRuntime;
131
-
132
- const requestBody = {
133
- threadId: "thread-1",
134
- runId: "run-1",
135
- state: {},
136
- messages: [],
137
- tools: [],
138
- context: [],
139
- forwardedProps: {},
140
- };
141
-
142
- const request = new Request("https://example.com/agent/test-agent/run", {
143
- method: "POST",
144
- headers: {
145
- "Content-Type": "application/json",
146
- "X-Custom": "custom-value",
147
- Authorization: "Bearer forwarded",
148
- Origin: "http://localhost:4200",
149
- },
150
- body: JSON.stringify(requestBody),
151
- });
152
-
153
- const response = await handleRunAgent({
154
- runtime,
155
- request,
156
- agentId: "test-agent",
157
- });
158
-
159
- expect(response.status).toBe(200);
160
- await runInvoked;
161
-
162
- expect(recordedHeaders).toHaveLength(1);
163
- expect(recordedHeaders[0]).toMatchObject({
164
- authorization: "Bearer forwarded",
165
- "x-custom": "custom-value",
166
- });
167
- expect(recordedHeaders[0]).not.toHaveProperty("origin");
168
- expect(recordedHeaders[0]).not.toHaveProperty("content-type");
169
- });
170
-
171
- const createMockAgentWithUse = () => {
172
- const useSpy = vi.fn();
173
- const agent = {
174
- clone: () => ({ ...agent, use: useSpy }),
175
- use: useSpy,
176
- } as unknown as AbstractAgent;
177
- return { agent, useSpy };
178
- };
179
-
180
- const createMockRunner = () => ({
181
- run: () =>
182
- new Observable<BaseEvent>((subscriber) => {
183
- subscriber.complete();
184
- }),
185
- connect: () =>
186
- new Observable<BaseEvent>((subscriber) => {
187
- subscriber.complete();
188
- }),
189
- isRunning: async () => false,
190
- stop: async () => false,
191
- });
192
-
193
- const createRunRequest = (headers?: Record<string, string>) =>
194
- new Request("https://example.com/agent/my-agent/run", {
195
- method: "POST",
196
- headers: { "Content-Type": "application/json", ...headers },
197
- body: JSON.stringify({
198
- threadId: "thread-1",
199
- runId: "run-1",
200
- state: {},
201
- messages: [],
202
- tools: [],
203
- context: [],
204
- forwardedProps: {},
205
- }),
206
- });
207
-
208
- it("applies A2UIMiddleware to all agents when a2ui.enabled is true and no agents filter", async () => {
209
- const { agent, useSpy } = createMockAgentWithUse();
210
-
211
- const runtime = {
212
- agents: Promise.resolve({ "my-agent": agent }),
213
- transcriptionService: undefined,
214
- beforeRequestMiddleware: undefined,
215
- afterRequestMiddleware: undefined,
216
- runner: createMockRunner(),
217
- a2ui: { enabled: true, injectA2UITool: true },
218
- } as unknown as CopilotRuntime;
219
-
220
- await handleRunAgent({
221
- runtime,
222
- request: createRunRequest(),
223
- agentId: "my-agent",
224
- });
225
-
226
- expect(useSpy).toHaveBeenCalledOnce();
227
- expect(useSpy.mock.calls[0][0]).toBeInstanceOf(A2UIMiddleware);
228
- });
229
-
230
- it("applies A2UIMiddleware only to matching agent when agents filter is set", async () => {
231
- const { agent: matchingAgent, useSpy: matchingSpy } =
232
- createMockAgentWithUse();
233
- const { agent: otherAgent, useSpy: otherSpy } = createMockAgentWithUse();
234
-
235
- const makeRuntime = (agentId: string, targetAgent: AbstractAgent) =>
236
- ({
237
- agents: Promise.resolve({ [agentId]: targetAgent }),
238
- transcriptionService: undefined,
239
- beforeRequestMiddleware: undefined,
240
- afterRequestMiddleware: undefined,
241
- runner: createMockRunner(),
242
- a2ui: { enabled: true, agents: ["my-agent"] },
243
- }) as unknown as CopilotRuntime;
244
-
245
- // Should apply for "my-agent"
246
- await handleRunAgent({
247
- runtime: makeRuntime("my-agent", matchingAgent),
248
- request: createRunRequest(),
249
- agentId: "my-agent",
250
- });
251
- expect(matchingSpy).toHaveBeenCalledOnce();
252
-
253
- // Should NOT apply for "other-agent"
254
- const otherRequest = new Request("https://example.com/agent/other/run", {
255
- method: "POST",
256
- headers: { "Content-Type": "application/json" },
257
- body: JSON.stringify({
258
- threadId: "thread-1",
259
- runId: "run-1",
260
- state: {},
261
- messages: [],
262
- tools: [],
263
- context: [],
264
- forwardedProps: {},
265
- }),
266
- });
267
- await handleRunAgent({
268
- runtime: makeRuntime("other", otherAgent),
269
- request: otherRequest,
270
- agentId: "other",
271
- });
272
- expect(otherSpy).not.toHaveBeenCalled();
273
- });
274
-
275
- it("does not apply A2UIMiddleware when a2ui is omitted", async () => {
276
- const { agent, useSpy } = createMockAgentWithUse();
277
-
278
- const runtime = {
279
- agents: Promise.resolve({ "my-agent": agent }),
280
- transcriptionService: undefined,
281
- beforeRequestMiddleware: undefined,
282
- afterRequestMiddleware: undefined,
283
- runner: createMockRunner(),
284
- } as unknown as CopilotRuntime;
285
-
286
- await handleRunAgent({
287
- runtime,
288
- request: createRunRequest(),
289
- agentId: "my-agent",
290
- });
291
-
292
- expect(useSpy).not.toHaveBeenCalled();
293
- });
294
-
295
- describe("IntelligenceAgentRunner realtime credentials path", () => {
296
- /** Loose mock type for CopilotKitIntelligence — avoids `as any` while the class has private fields. */
297
- interface MockIntelligencePlatform {
298
- [key: string]: ((...args: any[]) => any) | undefined;
299
- }
300
-
301
- const createIntelligenceRuntime = (
302
- agent: AbstractAgent,
303
- platform?: MockIntelligencePlatform,
304
- options?: {
305
- generateThreadNames?: boolean;
306
- lockHeartbeatIntervalSeconds?: number;
307
- lockTtlSeconds?: number;
308
- identifyUser?: (
309
- request: Request,
310
- ) =>
311
- | { id: string; name: string }
312
- | Promise<{ id: string; name: string }>;
313
- },
314
- ) => {
315
- const runner = Object.create(IntelligenceAgentRunner.prototype);
316
- runner.run = vi.fn(
317
- () =>
318
- new Observable<BaseEvent>((subscriber) => {
319
- subscriber.complete();
320
- }),
321
- );
322
- return {
323
- agents: Promise.resolve({ "my-agent": agent }),
324
- transcriptionService: undefined,
325
- beforeRequestMiddleware: undefined,
326
- afterRequestMiddleware: undefined,
327
- runner,
328
- mode: "intelligence",
329
- generateThreadNames: options?.generateThreadNames ?? false,
330
- lockHeartbeatIntervalSeconds:
331
- options?.lockHeartbeatIntervalSeconds ?? 15,
332
- lockTtlSeconds: options?.lockTtlSeconds ?? 20,
333
- intelligence: {
334
- ɵgetClientWsUrl: vi.fn(() => "wss://runtime.example/client"),
335
- ...platform,
336
- },
337
- identifyUser:
338
- options?.identifyUser ??
339
- vi.fn().mockResolvedValue({ id: "user-1", name: "User One" }),
340
- } as unknown as CopilotRuntime;
341
- };
342
-
343
- const createAgentForIntelligence = () => {
344
- const createClone = () =>
345
- ({
346
- clone: vi.fn(() => createClone()),
347
- setMessages: vi.fn(),
348
- setState: vi.fn(),
349
- abortRun: vi.fn(),
350
- threadId: undefined,
351
- headers: {},
352
- runAgent: vi.fn().mockResolvedValue(undefined),
353
- }) as unknown as AbstractAgent;
354
-
355
- const agent = {
356
- clone: vi.fn(() => createClone()),
357
- setMessages: vi.fn(),
358
- setState: vi.fn(),
359
- abortRun: vi.fn(),
360
- threadId: undefined,
361
- headers: {},
362
- runAgent: vi.fn().mockResolvedValue(undefined),
363
- } as unknown as AbstractAgent;
364
- return agent;
365
- };
366
-
367
- it("returns joinToken JSON when lock is acquired", async () => {
368
- const agent = createAgentForIntelligence();
369
- const platform = {
370
- getOrCreateThread: vi.fn().mockResolvedValue({
371
- thread: { id: "thread-1", name: null },
372
- created: false,
373
- }),
374
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
375
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
376
- threadId: "thread-1",
377
- runId: "run-1",
378
- joinToken: "jt-123",
379
- }),
380
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
381
- };
382
- const runtime = createIntelligenceRuntime(agent, platform);
383
-
384
- const response = await handleRunAgent({
385
- runtime,
386
- request: createRunRequest(),
387
- agentId: "my-agent",
388
- });
389
-
390
- expect(response.status).toBe(200);
391
- expect(response.headers.get("Content-Type")).toBe("application/json");
392
- const body = await response.json();
393
- expect(body).toEqual({
394
- threadId: "thread-1",
395
- runId: "run-1",
396
- joinToken: "jt-123",
397
- realtime: {
398
- clientUrl: "wss://runtime.example/client",
399
- topic: "thread:thread-1",
400
- },
401
- });
402
- expect(platform.getOrCreateThread).toHaveBeenCalledWith({
403
- threadId: "thread-1",
404
- userId: "user-1",
405
- agentId: "my-agent",
406
- });
407
- expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
408
- threadId: "thread-1",
409
- runId: "run-1",
410
- userId: "user-1",
411
- agentId: "my-agent",
412
- ttlSeconds: 20,
413
- });
414
- expect(platform.getThreadMessages).toHaveBeenCalledWith({
415
- threadId: "thread-1",
416
- });
417
- });
418
-
419
- it("uses identifyUser instead of a conflicting X-User-Id header", async () => {
420
- const agent = createAgentForIntelligence();
421
- const platform = {
422
- getOrCreateThread: vi.fn().mockResolvedValue({
423
- thread: { id: "thread-1", name: null },
424
- created: false,
425
- }),
426
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
427
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
428
- threadId: "thread-1",
429
- runId: "run-1",
430
- joinToken: "jt-123",
431
- }),
432
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
433
- };
434
- const identifyUser = vi
435
- .fn()
436
- .mockResolvedValue({ id: "resolved-user", name: "Resolved User" });
437
- const runtime = createIntelligenceRuntime(agent, platform, {
438
- identifyUser,
439
- });
440
- const request = createRunRequest({ "X-User-Id": "legacy-user" });
441
-
442
- const response = await handleRunAgent({
443
- runtime,
444
- request,
445
- agentId: "my-agent",
446
- });
447
-
448
- expect(response.status).toBe(200);
449
- expect(identifyUser).toHaveBeenCalledTimes(1);
450
- expect(identifyUser).toHaveBeenCalledWith(request);
451
- expect(platform.getOrCreateThread).toHaveBeenCalledWith({
452
- threadId: "thread-1",
453
- userId: "resolved-user",
454
- agentId: "my-agent",
455
- });
456
- expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
457
- threadId: "thread-1",
458
- runId: "run-1",
459
- userId: "resolved-user",
460
- agentId: "my-agent",
461
- ttlSeconds: 20,
462
- });
463
- });
464
-
465
- it("starts the runner with canonical threadId and runId from the lock response", async () => {
466
- const agent = createAgentForIntelligence();
467
- const platform = {
468
- getOrCreateThread: vi.fn().mockResolvedValue({
469
- thread: { id: "thread-1", name: null },
470
- created: false,
471
- }),
472
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
473
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
474
- threadId: "canonical-thread",
475
- runId: "canonical-run",
476
- joinToken: "jt-456",
477
- }),
478
- };
479
- const runtime = createIntelligenceRuntime(agent, platform);
480
-
481
- await handleRunAgent({
482
- runtime,
483
- request: createRunRequest(),
484
- agentId: "my-agent",
485
- });
486
-
487
- expect(runtime.runner.run).toHaveBeenCalledWith(
488
- expect.objectContaining({
489
- threadId: "canonical-thread",
490
- input: expect.objectContaining({
491
- threadId: "canonical-thread",
492
- runId: "canonical-run",
493
- }),
494
- }),
495
- );
496
- });
497
-
498
- it("cleans up the lock and returns 502 when joinToken is missing", async () => {
499
- const agent = createAgentForIntelligence();
500
- const platform = {
501
- getOrCreateThread: vi.fn().mockResolvedValue({
502
- thread: { id: "thread-1", name: null },
503
- created: false,
504
- }),
505
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
506
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
507
- threadId: "thread-1",
508
- runId: "run-1",
509
- }),
510
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
511
- };
512
- const runtime = createIntelligenceRuntime(agent, platform);
513
-
514
- const response = await handleRunAgent({
515
- runtime,
516
- request: createRunRequest(),
517
- agentId: "my-agent",
518
- });
519
-
520
- expect(response.status).toBe(502);
521
- const body = await response.json();
522
- expect(body.error).toBe("Run connection credentials not available");
523
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledWith({
524
- threadId: "thread-1",
525
- runId: "run-1",
526
- });
527
- expect(runtime.runner.run).not.toHaveBeenCalled();
528
- });
529
-
530
- it("uses the requested lock owner when malformed credentials omit canonical IDs", async () => {
531
- const agent = createAgentForIntelligence();
532
- const platform = {
533
- getOrCreateThread: vi.fn().mockResolvedValue({
534
- thread: { id: "thread-1", name: null },
535
- created: false,
536
- }),
537
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
538
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
539
- joinToken: "jt-123",
540
- }),
541
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
542
- };
543
- const runtime = createIntelligenceRuntime(agent, platform);
544
-
545
- const response = await handleRunAgent({
546
- runtime,
547
- request: createRunRequest(),
548
- agentId: "my-agent",
549
- });
550
-
551
- expect(response.status).toBe(502);
552
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledWith({
553
- threadId: "thread-1",
554
- runId: "run-1",
555
- });
556
- expect(runtime.runner.run).not.toHaveBeenCalled();
557
- });
558
-
559
- it("returns 409 when thread lock is denied", async () => {
560
- const agent = createAgentForIntelligence();
561
- const platform = {
562
- getOrCreateThread: vi.fn().mockResolvedValue({
563
- thread: { id: "thread-1", name: null },
564
- created: false,
565
- }),
566
- getThreadMessages: vi.fn(),
567
- ɵacquireThreadLock: vi
568
- .fn()
569
- .mockRejectedValue(new Error("Thread is locked by another runner")),
570
- };
571
- const runtime = createIntelligenceRuntime(agent, platform);
572
-
573
- const response = await handleRunAgent({
574
- runtime,
575
- request: createRunRequest(),
576
- agentId: "my-agent",
577
- });
578
-
579
- expect(response.status).toBe(409);
580
- const body = await response.json();
581
- expect(body.error).toBe("Thread lock denied");
582
- });
583
-
584
- it("cleans up the canonical lock and returns 502 when runner start fails immediately", async () => {
585
- const agent = createAgentForIntelligence();
586
- const platform = {
587
- getOrCreateThread: vi.fn().mockResolvedValue({
588
- thread: { id: "thread-1", name: null },
589
- created: false,
590
- }),
591
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
592
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
593
- threadId: "canonical-thread",
594
- runId: "canonical-run",
595
- joinToken: "jt-123",
596
- }),
597
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
598
- };
599
- const runtime = createIntelligenceRuntime(agent, platform);
600
- runtime.runner.run = vi.fn(
601
- () =>
602
- new Observable<BaseEvent>((subscriber) => {
603
- subscriber.next({
604
- type: EventType.RUN_ERROR,
605
- message: "join failed",
606
- } as BaseEvent);
607
- subscriber.complete();
608
- }),
609
- );
610
-
611
- const response = await handleRunAgent({
612
- runtime,
613
- request: createRunRequest(),
614
- agentId: "my-agent",
615
- });
616
-
617
- expect(response.status).toBe(502);
618
- const body = await response.json();
619
- expect(body).toEqual({
620
- error: "Failed to start runner",
621
- message: "join failed",
622
- });
623
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledWith({
624
- threadId: "canonical-thread",
625
- runId: "canonical-run",
626
- });
627
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledTimes(1);
628
- });
629
-
630
- it("delays the run success response until the runner startup boundary resolves", async () => {
631
- const agent = createAgentForIntelligence();
632
- let resolveStartup: (() => void) | undefined;
633
- const startup = new Promise<void>((resolve) => {
634
- resolveStartup = resolve;
635
- });
636
- const platform = {
637
- getOrCreateThread: vi.fn().mockResolvedValue({
638
- thread: { id: "thread-1", name: null },
639
- created: false,
640
- }),
641
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
642
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
643
- threadId: "canonical-thread",
644
- runId: "canonical-run",
645
- joinToken: "jt-123",
646
- }),
647
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
648
- };
649
- const runtime = createIntelligenceRuntime(agent, platform);
650
- runtime.runner.runWithStartupBoundary = vi.fn(() => ({
651
- events: new Observable<BaseEvent>(() => {}),
652
- startup,
653
- }));
654
- let settled = false;
655
-
656
- const responsePromise = handleRunAgent({
657
- runtime,
658
- request: createRunRequest(),
659
- agentId: "my-agent",
660
- }).then((response) => {
661
- settled = true;
662
- return response;
663
- });
664
-
665
- await Promise.resolve();
666
-
667
- expect(settled).toBe(false);
668
-
669
- resolveStartup?.();
670
- const response = await responsePromise;
671
-
672
- expect(response.status).toBe(200);
673
- expect(runtime.runner.runWithStartupBoundary).toHaveBeenCalledWith(
674
- expect.objectContaining({
675
- threadId: "canonical-thread",
676
- input: expect.objectContaining({
677
- threadId: "canonical-thread",
678
- runId: "canonical-run",
679
- }),
680
- }),
681
- );
682
- });
683
-
684
- it("cleans up the lock and returns 502 when the runner startup boundary rejects", async () => {
685
- const agent = createAgentForIntelligence();
686
- const platform = {
687
- getOrCreateThread: vi.fn().mockResolvedValue({
688
- thread: { id: "thread-1", name: null },
689
- created: false,
690
- }),
691
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
692
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
693
- threadId: "canonical-thread",
694
- runId: "canonical-run",
695
- joinToken: "jt-123",
696
- }),
697
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
698
- };
699
- const runtime = createIntelligenceRuntime(agent, platform);
700
- runtime.runner.runWithStartupBoundary = vi.fn(() => ({
701
- events: new Observable<BaseEvent>(() => {}),
702
- startup: Promise.reject(new Error("Failed to join channel: denied")),
703
- }));
704
-
705
- const response = await handleRunAgent({
706
- runtime,
707
- request: createRunRequest(),
708
- agentId: "my-agent",
709
- });
710
-
711
- expect(response.status).toBe(502);
712
- const body = await response.json();
713
- expect(body).toEqual({
714
- error: "Failed to start runner",
715
- message: "Failed to join channel: denied",
716
- });
717
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledWith({
718
- threadId: "canonical-thread",
719
- runId: "canonical-run",
720
- });
721
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledTimes(1);
722
- });
723
-
724
- it("aborts the agent when lock renewal fails", async () => {
725
- vi.useFakeTimers();
726
- const runningAgent = {
727
- clone: vi.fn(),
728
- setMessages: vi.fn(),
729
- setState: vi.fn(),
730
- abortRun: vi.fn(),
731
- threadId: undefined,
732
- headers: {},
733
- runAgent: vi.fn().mockResolvedValue(undefined),
734
- } as unknown as AbstractAgent;
735
- const baseAgent = {
736
- clone: vi.fn(() => runningAgent),
737
- setMessages: vi.fn(),
738
- setState: vi.fn(),
739
- abortRun: vi.fn(),
740
- threadId: undefined,
741
- headers: {},
742
- runAgent: vi.fn().mockResolvedValue(undefined),
743
- } as unknown as AbstractAgent;
744
- const platform = {
745
- getOrCreateThread: vi.fn().mockResolvedValue({
746
- thread: { id: "thread-1", name: null },
747
- created: false,
748
- }),
749
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
750
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
751
- threadId: "canonical-thread",
752
- runId: "canonical-run",
753
- joinToken: "jt-123",
754
- }),
755
- ɵrenewThreadLock: vi.fn().mockRejectedValue(new Error("lost lock")),
756
- };
757
- const runtime = createIntelligenceRuntime(baseAgent, platform, {
758
- lockHeartbeatIntervalSeconds: 1,
759
- lockTtlSeconds: 5,
760
- });
761
- runtime.runner.run = vi.fn(() => new Observable<BaseEvent>(() => {}));
762
-
763
- try {
764
- const response = await handleRunAgent({
765
- runtime,
766
- request: createRunRequest(),
767
- agentId: "my-agent",
768
- });
769
- expect(response.status).toBe(200);
770
-
771
- await vi.advanceTimersByTimeAsync(1_000);
772
-
773
- expect(platform.ɵrenewThreadLock).toHaveBeenCalledWith({
774
- threadId: "canonical-thread",
775
- runId: "canonical-run",
776
- ttlSeconds: 5,
777
- });
778
- expect(runningAgent.abortRun).toHaveBeenCalledTimes(1);
779
- } finally {
780
- vi.useRealTimers();
781
- }
782
- });
783
-
784
- it("passes only unseen input messages to the runner for durable persistence", async () => {
785
- const agent = createAgentForIntelligence();
786
- const platform = {
787
- getOrCreateThread: vi.fn().mockResolvedValue({
788
- thread: { id: "thread-1", name: null },
789
- created: false,
790
- }),
791
- getThreadMessages: vi.fn().mockResolvedValue({
792
- messages: [
793
- {
794
- id: "msg-existing",
795
- role: "user",
796
- content: "First turn",
797
- },
798
- ],
799
- }),
800
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
801
- threadId: "thread-1",
802
- runId: "run-1",
803
- joinToken: "jt-123",
804
- }),
805
- };
806
- const runtime = createIntelligenceRuntime(agent, platform);
807
- const response = await handleRunAgent({
808
- runtime,
809
- request: new Request("https://example.com/agent/my-agent/run", {
810
- method: "POST",
811
- headers: {
812
- "Content-Type": "application/json",
813
- },
814
- body: JSON.stringify({
815
- threadId: "thread-1",
816
- runId: "run-1",
817
- state: {},
818
- messages: [
819
- {
820
- id: "msg-existing",
821
- role: "user",
822
- content: "First turn",
823
- },
824
- {
825
- id: "msg-new",
826
- role: "user",
827
- content: "Second turn",
828
- },
829
- ],
830
- tools: [],
831
- context: [],
832
- forwardedProps: {},
833
- }),
834
- }),
835
- agentId: "my-agent",
836
- });
837
-
838
- expect(response.status).toBe(200);
839
- expect(runtime.runner.run).toHaveBeenCalledWith(
840
- expect.objectContaining({
841
- persistedInputMessages: [
842
- {
843
- id: "msg-new",
844
- role: "user",
845
- content: "Second turn",
846
- },
847
- ],
848
- }),
849
- );
850
- });
851
-
852
- it("returns 502 when durable thread history lookup fails", async () => {
853
- const agent = createAgentForIntelligence();
854
- const platform = {
855
- getOrCreateThread: vi.fn().mockResolvedValue({
856
- thread: { id: "thread-1", name: null },
857
- created: false,
858
- }),
859
- getThreadMessages: vi
860
- .fn()
861
- .mockRejectedValue(new Error("history unavailable")),
862
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
863
- threadId: "thread-1",
864
- runId: "run-1",
865
- joinToken: "jt-123",
866
- }),
867
- ɵcleanupThreadLock: vi.fn().mockResolvedValue(undefined),
868
- };
869
- const runtime = createIntelligenceRuntime(agent, platform);
870
-
871
- const response = await handleRunAgent({
872
- runtime,
873
- request: createRunRequest(),
874
- agentId: "my-agent",
875
- });
876
-
877
- expect(response.status).toBe(502);
878
- const body = await response.json();
879
- expect(body.error).toBe("Thread history lookup failed");
880
- expect(platform.ɵcleanupThreadLock).toHaveBeenCalledWith({
881
- threadId: "thread-1",
882
- runId: "run-1",
883
- });
884
- expect(runtime.runner.run).not.toHaveBeenCalled();
885
- });
886
-
887
- it("creates the thread before locking when run targets a fresh thread", async () => {
888
- const agent = createAgentForIntelligence();
889
- const platform = {
890
- getOrCreateThread: vi.fn().mockResolvedValue({
891
- thread: { id: "thread-1", name: null },
892
- created: true,
893
- }),
894
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
895
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
896
- threadId: "thread-1",
897
- runId: "run-1",
898
- joinToken: "jt-created",
899
- }),
900
- };
901
- const runtime = createIntelligenceRuntime(agent, platform);
902
-
903
- const response = await handleRunAgent({
904
- runtime,
905
- request: createRunRequest(),
906
- agentId: "my-agent",
907
- });
908
-
909
- expect(response.status).toBe(200);
910
- expect(platform.getOrCreateThread).toHaveBeenCalledWith({
911
- threadId: "thread-1",
912
- userId: "user-1",
913
- agentId: "my-agent",
914
- });
915
- expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
916
- threadId: "thread-1",
917
- runId: "run-1",
918
- userId: "user-1",
919
- agentId: "my-agent",
920
- ttlSeconds: 20,
921
- });
922
- });
923
-
924
- it("generates and persists a thread name in the background for new unnamed threads", async () => {
925
- const namingAgent = {
926
- clone: vi.fn(),
927
- setMessages: vi.fn(),
928
- setState: vi.fn(),
929
- threadId: undefined,
930
- headers: {},
931
- runAgent: vi.fn().mockResolvedValue({
932
- newMessages: [
933
- {
934
- id: "assistant-1",
935
- role: "assistant",
936
- content: '{"title":"**Order refund** status"}',
937
- },
938
- ],
939
- }),
940
- } as unknown as AbstractAgent;
941
- const baseAgent = {
942
- clone: vi
943
- .fn()
944
- .mockReturnValueOnce({
945
- clone: vi.fn(),
946
- setMessages: vi.fn(),
947
- setState: vi.fn(),
948
- threadId: undefined,
949
- headers: {},
950
- runAgent: vi.fn().mockResolvedValue(undefined),
951
- })
952
- .mockReturnValueOnce(namingAgent),
953
- setMessages: vi.fn(),
954
- setState: vi.fn(),
955
- threadId: undefined,
956
- headers: {},
957
- runAgent: vi.fn().mockResolvedValue(undefined),
958
- } as unknown as AbstractAgent;
959
- const platform = {
960
- getOrCreateThread: vi.fn().mockResolvedValue({
961
- thread: { id: "thread-1", name: null },
962
- created: true,
963
- }),
964
- updateThread: vi.fn().mockResolvedValue({
965
- id: "thread-1",
966
- name: "Order refund status",
967
- }),
968
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
969
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
970
- threadId: "thread-1",
971
- runId: "run-1",
972
- joinToken: "jt-created",
973
- }),
974
- };
975
- const runtime = createIntelligenceRuntime(baseAgent, platform, {
976
- generateThreadNames: true,
977
- });
978
- const response = await handleRunAgent({
979
- runtime,
980
- request: new Request("https://example.com/agent/my-agent/run", {
981
- method: "POST",
982
- headers: {
983
- "Content-Type": "application/json",
984
- },
985
- body: JSON.stringify({
986
- threadId: "thread-1",
987
- runId: "run-1",
988
- state: {},
989
- messages: [
990
- {
991
- id: "user-1",
992
- role: "user",
993
- content: "Can you help me with my refund request?",
994
- },
995
- ],
996
- tools: [],
997
- context: [],
998
- forwardedProps: {},
999
- }),
1000
- }),
1001
- agentId: "my-agent",
1002
- });
1003
-
1004
- expect(response.status).toBe(200);
1005
- await vi.waitFor(() =>
1006
- expect(platform.updateThread).toHaveBeenCalledWith({
1007
- threadId: "thread-1",
1008
- userId: "user-1",
1009
- agentId: "my-agent",
1010
- updates: { name: "Order refund status" },
1011
- }),
1012
- );
1013
- expect(runtime.runner.run).toHaveBeenCalledTimes(1);
1014
- });
1015
-
1016
- it("does not generate a thread name when generateThreadNames is false", async () => {
1017
- const agent = createAgentForIntelligence();
1018
- const platform = {
1019
- getOrCreateThread: vi.fn().mockResolvedValue({
1020
- thread: { id: "thread-1", name: null },
1021
- created: true,
1022
- }),
1023
- updateThread: vi.fn(),
1024
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
1025
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
1026
- threadId: "thread-1",
1027
- runId: "run-1",
1028
- joinToken: "jt-created",
1029
- }),
1030
- };
1031
- const runtime = createIntelligenceRuntime(agent, platform, {
1032
- generateThreadNames: false,
1033
- });
1034
-
1035
- const response = await handleRunAgent({
1036
- runtime,
1037
- request: createRunRequest(),
1038
- agentId: "my-agent",
1039
- });
1040
-
1041
- expect(response.status).toBe(200);
1042
- await Promise.resolve();
1043
- expect(platform.updateThread).not.toHaveBeenCalled();
1044
- });
1045
-
1046
- it("does not generate a thread name when the created thread already has a name", async () => {
1047
- const agent = createAgentForIntelligence();
1048
- const platform = {
1049
- getOrCreateThread: vi.fn().mockResolvedValue({
1050
- thread: { id: "thread-1", name: "Existing name" },
1051
- created: true,
1052
- }),
1053
- updateThread: vi.fn(),
1054
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
1055
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
1056
- threadId: "thread-1",
1057
- runId: "run-1",
1058
- joinToken: "jt-created",
1059
- }),
1060
- };
1061
- const runtime = createIntelligenceRuntime(agent, platform, {
1062
- generateThreadNames: true,
1063
- });
1064
-
1065
- const response = await handleRunAgent({
1066
- runtime,
1067
- request: createRunRequest(),
1068
- agentId: "my-agent",
1069
- });
1070
-
1071
- expect(response.status).toBe(200);
1072
- await Promise.resolve();
1073
- expect(platform.updateThread).not.toHaveBeenCalled();
1074
- });
1075
-
1076
- it("retries thread naming three times and falls back to Untitled", async () => {
1077
- const namingAgent = {
1078
- clone: vi.fn(),
1079
- setMessages: vi.fn(),
1080
- setState: vi.fn(),
1081
- threadId: undefined,
1082
- headers: {},
1083
- runAgent: vi.fn().mockRejectedValue(new Error("naming failed")),
1084
- } as unknown as AbstractAgent;
1085
- const baseAgent = {
1086
- clone: vi
1087
- .fn()
1088
- .mockReturnValueOnce({
1089
- clone: vi.fn(),
1090
- setMessages: vi.fn(),
1091
- setState: vi.fn(),
1092
- threadId: undefined,
1093
- headers: {},
1094
- runAgent: vi.fn().mockResolvedValue(undefined),
1095
- })
1096
- .mockReturnValueOnce(namingAgent)
1097
- .mockReturnValueOnce(namingAgent)
1098
- .mockReturnValueOnce(namingAgent),
1099
- setMessages: vi.fn(),
1100
- setState: vi.fn(),
1101
- threadId: undefined,
1102
- headers: {},
1103
- runAgent: vi.fn().mockResolvedValue(undefined),
1104
- } as unknown as AbstractAgent;
1105
- const platform = {
1106
- getOrCreateThread: vi.fn().mockResolvedValue({
1107
- thread: { id: "thread-1", name: null },
1108
- created: true,
1109
- }),
1110
- updateThread: vi.fn(),
1111
- getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
1112
- ɵacquireThreadLock: vi.fn().mockResolvedValue({
1113
- threadId: "thread-1",
1114
- runId: "run-1",
1115
- joinToken: "jt-created",
1116
- }),
1117
- };
1118
- const runtime = createIntelligenceRuntime(baseAgent, platform, {
1119
- generateThreadNames: true,
1120
- });
1121
- const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
1122
-
1123
- try {
1124
- const response = await handleRunAgent({
1125
- runtime,
1126
- request: new Request("https://example.com/agent/my-agent/run", {
1127
- method: "POST",
1128
- headers: {
1129
- "Content-Type": "application/json",
1130
- },
1131
- body: JSON.stringify({
1132
- threadId: "thread-1",
1133
- runId: "run-1",
1134
- state: {},
1135
- messages: [
1136
- {
1137
- id: "user-1",
1138
- role: "user",
1139
- content: "Please help me name this failed thread.",
1140
- },
1141
- ],
1142
- tools: [],
1143
- context: [],
1144
- forwardedProps: {},
1145
- }),
1146
- }),
1147
- agentId: "my-agent",
1148
- });
1149
-
1150
- expect(response.status).toBe(200);
1151
- await vi.waitFor(() =>
1152
- expect(platform.updateThread).toHaveBeenCalledWith({
1153
- threadId: "thread-1",
1154
- userId: "user-1",
1155
- agentId: "my-agent",
1156
- updates: { name: "Untitled" },
1157
- }),
1158
- );
1159
- expect(namingAgent.runAgent).toHaveBeenCalledTimes(3);
1160
- expect(runtime.runner.run).toHaveBeenCalledTimes(1);
1161
- } finally {
1162
- errorSpy.mockRestore();
1163
- }
1164
- });
1165
-
1166
- it("returns 400 when identifyUser returns an invalid id", async () => {
1167
- const agent = createAgentForIntelligence();
1168
- const platform = {
1169
- getOrCreateThread: vi.fn(),
1170
- getThreadMessages: vi.fn(),
1171
- ɵacquireThreadLock: vi.fn(),
1172
- };
1173
- const runtime = createIntelligenceRuntime(agent, platform, {
1174
- identifyUser: vi.fn().mockResolvedValue({ id: "", name: "User" }),
1175
- });
1176
-
1177
- const response = await handleRunAgent({
1178
- runtime,
1179
- request: createRunRequest(),
1180
- agentId: "my-agent",
1181
- });
1182
-
1183
- expect(response.status).toBe(400);
1184
- expect(platform.getOrCreateThread).not.toHaveBeenCalled();
1185
- expect(platform.ɵacquireThreadLock).not.toHaveBeenCalled();
1186
- });
1187
-
1188
- it("returns 400 when identifyUser returns an invalid name", async () => {
1189
- const agent = createAgentForIntelligence();
1190
- const platform = {
1191
- getOrCreateThread: vi.fn(),
1192
- getThreadMessages: vi.fn(),
1193
- ɵacquireThreadLock: vi.fn(),
1194
- };
1195
- const runtime = createIntelligenceRuntime(agent, platform, {
1196
- identifyUser: vi.fn().mockResolvedValue({ id: "user-1", name: "" }),
1197
- });
1198
-
1199
- const response = await handleRunAgent({
1200
- runtime,
1201
- request: createRunRequest(),
1202
- agentId: "my-agent",
1203
- });
1204
-
1205
- expect(response.status).toBe(400);
1206
- expect(platform.getOrCreateThread).not.toHaveBeenCalled();
1207
- expect(platform.ɵacquireThreadLock).not.toHaveBeenCalled();
1208
- });
1209
-
1210
- it("returns 500 when identifyUser throws", async () => {
1211
- const agent = createAgentForIntelligence();
1212
- const platform = {
1213
- getOrCreateThread: vi.fn(),
1214
- getThreadMessages: vi.fn(),
1215
- ɵacquireThreadLock: vi.fn(),
1216
- };
1217
- const runtime = createIntelligenceRuntime(agent, platform, {
1218
- identifyUser: vi.fn().mockRejectedValue(new Error("auth failed")),
1219
- });
1220
- const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
1221
-
1222
- try {
1223
- const response = await handleRunAgent({
1224
- runtime,
1225
- request: createRunRequest(),
1226
- agentId: "my-agent",
1227
- });
1228
-
1229
- expect(response.status).toBe(500);
1230
- expect(platform.getOrCreateThread).not.toHaveBeenCalled();
1231
- expect(platform.ɵacquireThreadLock).not.toHaveBeenCalled();
1232
- } finally {
1233
- errorSpy.mockRestore();
1234
- }
1235
- });
1236
- });
1237
-
1238
- describe("telemetry", () => {
1239
- it("captures oss.runtime.copilot_request_created on every invocation", async () => {
1240
- // Dynamic import so we spy on the module singleton the handler uses.
1241
- const { telemetry } = await import("../telemetry");
1242
- const captureSpy = vi
1243
- .spyOn(telemetry, "capture")
1244
- .mockResolvedValue(undefined);
1245
-
1246
- try {
1247
- const runtime = createMockRuntime({});
1248
- await handleRunAgent({
1249
- runtime,
1250
- request: createMockRequest(),
1251
- agentId: "nonexistent-agent",
1252
- });
1253
-
1254
- expect(captureSpy).toHaveBeenCalledWith(
1255
- "oss.runtime.copilot_request_created",
1256
- expect.objectContaining({
1257
- requestType: "run",
1258
- "cloud.api_key_provided": false,
1259
- }),
1260
- );
1261
- } finally {
1262
- captureSpy.mockRestore();
1263
- }
1264
- });
1265
-
1266
- it("includes cloud.public_api_key when x-copilotcloud-public-api-key header is set", async () => {
1267
- const { telemetry } = await import("../telemetry");
1268
- const captureSpy = vi
1269
- .spyOn(telemetry, "capture")
1270
- .mockResolvedValue(undefined);
1271
-
1272
- try {
1273
- const runtime = createMockRuntime({});
1274
- const request = new Request("https://example.com/agent/test/run", {
1275
- method: "POST",
1276
- headers: {
1277
- "x-copilotcloud-public-api-key": "ck_pub_run_test",
1278
- },
1279
- });
1280
-
1281
- await handleRunAgent({
1282
- runtime,
1283
- request,
1284
- agentId: "nonexistent-agent",
1285
- });
1286
-
1287
- expect(captureSpy).toHaveBeenCalledWith(
1288
- "oss.runtime.copilot_request_created",
1289
- expect.objectContaining({
1290
- "cloud.api_key_provided": true,
1291
- "cloud.public_api_key": "ck_pub_run_test",
1292
- }),
1293
- );
1294
- } finally {
1295
- captureSpy.mockRestore();
1296
- }
1297
- });
1298
- });
1299
-
1300
- describe("agentId tagging on cloned agents", () => {
1301
- /**
1302
- * Pins handle-run.ts:40 — `agent.agentId = agentId` is set on the clone
1303
- * BEFORE the agent reaches the runner. Without it, InMemoryAgentRunner
1304
- * falls back to "default" when stamping historic runs, and listThreads
1305
- * returns rows with the wrong agentId. This breaks the agentId filter
1306
- * in `GET /threads?agentId=...` for the local-dev fallback.
1307
- *
1308
- * This test runs the full flow through InMemoryAgentRunner with an
1309
- * AbstractAgent whose own `agentId` field is undefined (matches the
1310
- * shape after `clone()` returns a fresh instance), and asserts the
1311
- * runner records the registry key, NOT "default".
1312
- */
1313
- class TaggingTestAgent extends AbstractAgent {
1314
- async runAgent(
1315
- _input: RunAgentInput,
1316
- options: { onEvent: (event: { event: BaseEvent }) => void },
1317
- ): Promise<void> {
1318
- // Emit a single TEXT_MESSAGE_END event so the run produces at least
1319
- // one event and gets persisted to historicRuns. RUN_STARTED /
1320
- // RUN_FINISHED are appended by the runner itself.
1321
- options.onEvent({
1322
- event: {
1323
- type: EventType.TEXT_MESSAGE_END,
1324
- messageId: "msg-1",
1325
- } as BaseEvent,
1326
- });
1327
- }
1328
-
1329
- clone(): AbstractAgent {
1330
- // The fresh clone has NO agentId — the only way the runner can know
1331
- // the registry key is if handle-run.ts:40 stamps it before the run.
1332
- return new TaggingTestAgent();
1333
- }
1334
- }
1335
-
1336
- const createRunRequestForAgent = (agentId: string, threadId: string) =>
1337
- new Request(`https://example.com/agent/${agentId}/run`, {
1338
- method: "POST",
1339
- headers: { "Content-Type": "application/json" },
1340
- body: JSON.stringify({
1341
- threadId,
1342
- runId: `run-${threadId}`,
1343
- state: {},
1344
- messages: [],
1345
- tools: [],
1346
- context: [],
1347
- forwardedProps: {},
1348
- }),
1349
- });
1350
-
1351
- it("propagates the registry agentId onto historic runs (NOT 'default')", async () => {
1352
- const runner = new InMemoryAgentRunner();
1353
- const agent = new TaggingTestAgent();
1354
- const runtime = new CopilotRuntime({
1355
- agents: { tagged: agent },
1356
- runner,
1357
- });
1358
-
1359
- // Use a unique threadId so this test does not collide with other
1360
- // tests that share the InMemoryAgentRunner GLOBAL_STORE.
1361
- const threadId = `thread-tagged-${Date.now()}-${Math.random()}`;
1362
-
1363
- const response = await handleRunAgent({
1364
- runtime,
1365
- request: createRunRequestForAgent("tagged", threadId),
1366
- agentId: "tagged",
1367
- });
1368
- expect(response.status).toBe(200);
1369
-
1370
- // Drain the SSE stream so the underlying observable run completes —
1371
- // historicRuns is only populated AFTER the run finalizes.
1372
- const reader = response.body!.getReader();
1373
- while (true) {
1374
- const { done } = await reader.read();
1375
- if (done) break;
1376
- }
1377
-
1378
- const threads = runner.listThreads();
1379
- const thisThread = threads.find((t) => t.id === threadId);
1380
- expect(thisThread).toBeDefined();
1381
- expect(thisThread!.agentId).toBe("tagged");
1382
- // Negative assertion locks the regression: a future change that drops
1383
- // the `agent.agentId = agentId` line in handle-run will surface as
1384
- // "default" here, not as a missing thread.
1385
- expect(thisThread!.agentId).not.toBe("default");
1386
- });
1387
- });
1388
- });