@copilotkit/runtime 1.54.1-next.6 → 1.55.0-next.7

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 (337) hide show
  1. package/.eslintrc.js +4 -4
  2. package/CHANGELOG.md +117 -106
  3. package/dist/_virtual/_rolldown/runtime.mjs +25 -1
  4. package/dist/agent/index.cjs +643 -0
  5. package/dist/agent/index.cjs.map +1 -0
  6. package/dist/agent/index.d.cts +263 -0
  7. package/dist/agent/index.d.cts.map +1 -0
  8. package/dist/agent/index.d.mts +263 -0
  9. package/dist/agent/index.d.mts.map +1 -0
  10. package/dist/agent/index.mjs +635 -0
  11. package/dist/agent/index.mjs.map +1 -0
  12. package/dist/graphql/message-conversion/agui-to-gql.cjs.map +1 -1
  13. package/dist/graphql/message-conversion/agui-to-gql.mjs.map +1 -1
  14. package/dist/lib/integrations/nextjs/app-router.cjs +2 -2
  15. package/dist/lib/integrations/nextjs/app-router.cjs.map +1 -1
  16. package/dist/lib/integrations/nextjs/app-router.mjs +1 -1
  17. package/dist/lib/integrations/nextjs/app-router.mjs.map +1 -1
  18. package/dist/lib/integrations/node-http/index.cjs +2 -3
  19. package/dist/lib/integrations/node-http/index.cjs.map +1 -1
  20. package/dist/lib/integrations/node-http/index.mjs +1 -1
  21. package/dist/lib/integrations/node-http/index.mjs.map +1 -1
  22. package/dist/lib/runtime/agent-integrations/langgraph/agent.cjs +1 -1
  23. package/dist/lib/runtime/agent-integrations/langgraph/agent.d.cts +2 -2
  24. package/dist/lib/runtime/agent-integrations/langgraph/agent.d.cts.map +1 -1
  25. package/dist/lib/runtime/agent-integrations/langgraph/agent.d.mts +3 -3
  26. package/dist/lib/runtime/agent-integrations/langgraph/agent.d.mts.map +1 -1
  27. package/dist/lib/runtime/agent-integrations/langgraph/agent.mjs +1 -1
  28. package/dist/lib/runtime/copilot-runtime.cjs +7 -5
  29. package/dist/lib/runtime/copilot-runtime.cjs.map +1 -1
  30. package/dist/lib/runtime/copilot-runtime.d.cts +10 -8
  31. package/dist/lib/runtime/copilot-runtime.d.cts.map +1 -1
  32. package/dist/lib/runtime/copilot-runtime.d.mts +10 -8
  33. package/dist/lib/runtime/copilot-runtime.d.mts.map +1 -1
  34. package/dist/lib/runtime/copilot-runtime.mjs +7 -5
  35. package/dist/lib/runtime/copilot-runtime.mjs.map +1 -1
  36. package/dist/lib/runtime/telemetry-agent-runner.cjs +2 -2
  37. package/dist/lib/runtime/telemetry-agent-runner.cjs.map +1 -1
  38. package/dist/lib/runtime/telemetry-agent-runner.d.cts +2 -1
  39. package/dist/lib/runtime/telemetry-agent-runner.d.cts.map +1 -1
  40. package/dist/lib/runtime/telemetry-agent-runner.d.mts +2 -1
  41. package/dist/lib/runtime/telemetry-agent-runner.d.mts.map +1 -1
  42. package/dist/lib/runtime/telemetry-agent-runner.mjs +1 -1
  43. package/dist/lib/runtime/telemetry-agent-runner.mjs.map +1 -1
  44. package/dist/lib/telemetry-client.cjs +1 -1
  45. package/dist/lib/telemetry-client.mjs +1 -1
  46. package/dist/package.cjs +21 -4
  47. package/dist/package.mjs +21 -4
  48. package/dist/service-adapters/anthropic/anthropic-adapter.d.mts +1 -1
  49. package/dist/v2/index.cjs +41 -15
  50. package/dist/v2/index.d.cts +14 -2
  51. package/dist/v2/index.d.mts +14 -2
  52. package/dist/v2/index.mjs +13 -4
  53. package/dist/v2/runtime/endpoints/express-single.cjs +190 -0
  54. package/dist/v2/runtime/endpoints/express-single.cjs.map +1 -0
  55. package/dist/v2/runtime/endpoints/express-single.d.cts +16 -0
  56. package/dist/v2/runtime/endpoints/express-single.d.cts.map +1 -0
  57. package/dist/v2/runtime/endpoints/express-single.d.mts +16 -0
  58. package/dist/v2/runtime/endpoints/express-single.d.mts.map +1 -0
  59. package/dist/v2/runtime/endpoints/express-single.mjs +187 -0
  60. package/dist/v2/runtime/endpoints/express-single.mjs.map +1 -0
  61. package/dist/v2/runtime/endpoints/express-utils.cjs +119 -0
  62. package/dist/v2/runtime/endpoints/express-utils.cjs.map +1 -0
  63. package/dist/v2/runtime/endpoints/express-utils.mjs +117 -0
  64. package/dist/v2/runtime/endpoints/express-utils.mjs.map +1 -0
  65. package/dist/v2/runtime/endpoints/express.cjs +217 -0
  66. package/dist/v2/runtime/endpoints/express.cjs.map +1 -0
  67. package/dist/v2/runtime/endpoints/express.d.cts +16 -0
  68. package/dist/v2/runtime/endpoints/express.d.cts.map +1 -0
  69. package/dist/v2/runtime/endpoints/express.d.mts +16 -0
  70. package/dist/v2/runtime/endpoints/express.d.mts.map +1 -0
  71. package/dist/v2/runtime/endpoints/express.mjs +214 -0
  72. package/dist/v2/runtime/endpoints/express.mjs.map +1 -0
  73. package/dist/v2/runtime/endpoints/hono-single.cjs +141 -0
  74. package/dist/v2/runtime/endpoints/hono-single.cjs.map +1 -0
  75. package/dist/v2/runtime/endpoints/hono-single.d.cts +41 -0
  76. package/dist/v2/runtime/endpoints/hono-single.d.cts.map +1 -0
  77. package/dist/v2/runtime/endpoints/hono-single.d.mts +41 -0
  78. package/dist/v2/runtime/endpoints/hono-single.d.mts.map +1 -0
  79. package/dist/v2/runtime/endpoints/hono-single.mjs +140 -0
  80. package/dist/v2/runtime/endpoints/hono-single.mjs.map +1 -0
  81. package/dist/v2/runtime/endpoints/hono.cjs +248 -0
  82. package/dist/v2/runtime/endpoints/hono.cjs.map +1 -0
  83. package/dist/v2/runtime/endpoints/hono.d.cts +164 -0
  84. package/dist/v2/runtime/endpoints/hono.d.cts.map +1 -0
  85. package/dist/v2/runtime/endpoints/hono.d.mts +164 -0
  86. package/dist/v2/runtime/endpoints/hono.d.mts.map +1 -0
  87. package/dist/v2/runtime/endpoints/hono.mjs +247 -0
  88. package/dist/v2/runtime/endpoints/hono.mjs.map +1 -0
  89. package/dist/v2/runtime/endpoints/index.d.cts +5 -0
  90. package/dist/v2/runtime/endpoints/index.d.mts +5 -0
  91. package/dist/v2/runtime/endpoints/single-route-helpers.cjs +68 -0
  92. package/dist/v2/runtime/endpoints/single-route-helpers.cjs.map +1 -0
  93. package/dist/v2/runtime/endpoints/single-route-helpers.mjs +65 -0
  94. package/dist/v2/runtime/endpoints/single-route-helpers.mjs.map +1 -0
  95. package/dist/v2/runtime/handlers/get-runtime-info.cjs +51 -0
  96. package/dist/v2/runtime/handlers/get-runtime-info.cjs.map +1 -0
  97. package/dist/v2/runtime/handlers/get-runtime-info.mjs +51 -0
  98. package/dist/v2/runtime/handlers/get-runtime-info.mjs.map +1 -0
  99. package/dist/v2/runtime/handlers/handle-connect.cjs +49 -0
  100. package/dist/v2/runtime/handlers/handle-connect.cjs.map +1 -0
  101. package/dist/v2/runtime/handlers/handle-connect.mjs +49 -0
  102. package/dist/v2/runtime/handlers/handle-connect.mjs.map +1 -0
  103. package/dist/v2/runtime/handlers/handle-run.cjs +61 -0
  104. package/dist/v2/runtime/handlers/handle-run.cjs.map +1 -0
  105. package/dist/v2/runtime/handlers/handle-run.mjs +61 -0
  106. package/dist/v2/runtime/handlers/handle-run.mjs.map +1 -0
  107. package/dist/v2/runtime/handlers/handle-stop.cjs +47 -0
  108. package/dist/v2/runtime/handlers/handle-stop.cjs.map +1 -0
  109. package/dist/v2/runtime/handlers/handle-stop.mjs +46 -0
  110. package/dist/v2/runtime/handlers/handle-stop.mjs.map +1 -0
  111. package/dist/v2/runtime/handlers/handle-transcribe.cjs +112 -0
  112. package/dist/v2/runtime/handlers/handle-transcribe.cjs.map +1 -0
  113. package/dist/v2/runtime/handlers/handle-transcribe.mjs +111 -0
  114. package/dist/v2/runtime/handlers/handle-transcribe.mjs.map +1 -0
  115. package/dist/v2/runtime/handlers/header-utils.cjs +26 -0
  116. package/dist/v2/runtime/handlers/header-utils.cjs.map +1 -0
  117. package/dist/v2/runtime/handlers/header-utils.mjs +25 -0
  118. package/dist/v2/runtime/handlers/header-utils.mjs.map +1 -0
  119. package/dist/v2/runtime/handlers/intelligence/connect.cjs +37 -0
  120. package/dist/v2/runtime/handlers/intelligence/connect.cjs.map +1 -0
  121. package/dist/v2/runtime/handlers/intelligence/connect.mjs +37 -0
  122. package/dist/v2/runtime/handlers/intelligence/connect.mjs.map +1 -0
  123. package/dist/v2/runtime/handlers/intelligence/run.cjs +89 -0
  124. package/dist/v2/runtime/handlers/intelligence/run.cjs.map +1 -0
  125. package/dist/v2/runtime/handlers/intelligence/run.mjs +88 -0
  126. package/dist/v2/runtime/handlers/intelligence/run.mjs.map +1 -0
  127. package/dist/v2/runtime/handlers/intelligence/thread-names.cjs +146 -0
  128. package/dist/v2/runtime/handlers/intelligence/thread-names.cjs.map +1 -0
  129. package/dist/v2/runtime/handlers/intelligence/thread-names.mjs +145 -0
  130. package/dist/v2/runtime/handlers/intelligence/thread-names.mjs.map +1 -0
  131. package/dist/v2/runtime/handlers/intelligence/threads.cjs +159 -0
  132. package/dist/v2/runtime/handlers/intelligence/threads.cjs.map +1 -0
  133. package/dist/v2/runtime/handlers/intelligence/threads.mjs +154 -0
  134. package/dist/v2/runtime/handlers/intelligence/threads.mjs.map +1 -0
  135. package/dist/v2/runtime/handlers/shared/agent-utils.cjs +74 -0
  136. package/dist/v2/runtime/handlers/shared/agent-utils.cjs.map +1 -0
  137. package/dist/v2/runtime/handlers/shared/agent-utils.mjs +70 -0
  138. package/dist/v2/runtime/handlers/shared/agent-utils.mjs.map +1 -0
  139. package/dist/v2/runtime/handlers/shared/intelligence-utils.cjs +21 -0
  140. package/dist/v2/runtime/handlers/shared/intelligence-utils.cjs.map +1 -0
  141. package/dist/v2/runtime/handlers/shared/intelligence-utils.mjs +20 -0
  142. package/dist/v2/runtime/handlers/shared/intelligence-utils.mjs.map +1 -0
  143. package/dist/v2/runtime/handlers/shared/json-response.cjs +12 -0
  144. package/dist/v2/runtime/handlers/shared/json-response.cjs.map +1 -0
  145. package/dist/v2/runtime/handlers/shared/json-response.mjs +10 -0
  146. package/dist/v2/runtime/handlers/shared/json-response.mjs.map +1 -0
  147. package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.cjs +20 -0
  148. package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.cjs.map +1 -0
  149. package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.mjs +20 -0
  150. package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.mjs.map +1 -0
  151. package/dist/v2/runtime/handlers/shared/sse-response.cjs +69 -0
  152. package/dist/v2/runtime/handlers/shared/sse-response.cjs.map +1 -0
  153. package/dist/v2/runtime/handlers/shared/sse-response.mjs +68 -0
  154. package/dist/v2/runtime/handlers/shared/sse-response.mjs.map +1 -0
  155. package/dist/v2/runtime/handlers/sse/connect.cjs +18 -0
  156. package/dist/v2/runtime/handlers/sse/connect.cjs.map +1 -0
  157. package/dist/v2/runtime/handlers/sse/connect.mjs +18 -0
  158. package/dist/v2/runtime/handlers/sse/connect.mjs.map +1 -0
  159. package/dist/v2/runtime/handlers/sse/run.cjs +18 -0
  160. package/dist/v2/runtime/handlers/sse/run.cjs.map +1 -0
  161. package/dist/v2/runtime/handlers/sse/run.mjs +18 -0
  162. package/dist/v2/runtime/handlers/sse/run.mjs.map +1 -0
  163. package/dist/v2/runtime/index.d.cts +13 -0
  164. package/dist/v2/runtime/index.d.mts +14 -0
  165. package/dist/v2/runtime/intelligence-platform/client.cjs +333 -0
  166. package/dist/v2/runtime/intelligence-platform/client.cjs.map +1 -0
  167. package/dist/v2/runtime/intelligence-platform/client.d.cts +336 -0
  168. package/dist/v2/runtime/intelligence-platform/client.d.cts.map +1 -0
  169. package/dist/v2/runtime/intelligence-platform/client.d.mts +336 -0
  170. package/dist/v2/runtime/intelligence-platform/client.d.mts.map +1 -0
  171. package/dist/v2/runtime/intelligence-platform/client.mjs +331 -0
  172. package/dist/v2/runtime/intelligence-platform/client.mjs.map +1 -0
  173. package/dist/v2/runtime/intelligence-platform/index.d.mts +2 -0
  174. package/dist/v2/runtime/middleware-sse-parser.cjs +138 -0
  175. package/dist/v2/runtime/middleware-sse-parser.cjs.map +1 -0
  176. package/dist/v2/runtime/middleware-sse-parser.d.cts +22 -0
  177. package/dist/v2/runtime/middleware-sse-parser.d.cts.map +1 -0
  178. package/dist/v2/runtime/middleware-sse-parser.d.mts +22 -0
  179. package/dist/v2/runtime/middleware-sse-parser.d.mts.map +1 -0
  180. package/dist/v2/runtime/middleware-sse-parser.mjs +137 -0
  181. package/dist/v2/runtime/middleware-sse-parser.mjs.map +1 -0
  182. package/dist/v2/runtime/middleware.cjs +35 -0
  183. package/dist/v2/runtime/middleware.cjs.map +1 -0
  184. package/dist/v2/runtime/middleware.d.cts +32 -0
  185. package/dist/v2/runtime/middleware.d.cts.map +1 -0
  186. package/dist/v2/runtime/middleware.d.mts +32 -0
  187. package/dist/v2/runtime/middleware.d.mts.map +1 -0
  188. package/dist/v2/runtime/middleware.mjs +33 -0
  189. package/dist/v2/runtime/middleware.mjs.map +1 -0
  190. package/dist/v2/runtime/runner/agent-runner.cjs +8 -0
  191. package/dist/v2/runtime/runner/agent-runner.cjs.map +1 -0
  192. package/dist/v2/runtime/runner/agent-runner.d.cts +32 -0
  193. package/dist/v2/runtime/runner/agent-runner.d.cts.map +1 -0
  194. package/dist/v2/runtime/runner/agent-runner.d.mts +32 -0
  195. package/dist/v2/runtime/runner/agent-runner.d.mts.map +1 -0
  196. package/dist/v2/runtime/runner/agent-runner.mjs +7 -0
  197. package/dist/v2/runtime/runner/agent-runner.mjs.map +1 -0
  198. package/dist/v2/runtime/runner/in-memory.cjs +223 -0
  199. package/dist/v2/runtime/runner/in-memory.cjs.map +1 -0
  200. package/dist/v2/runtime/runner/in-memory.d.cts +15 -0
  201. package/dist/v2/runtime/runner/in-memory.d.cts.map +1 -0
  202. package/dist/v2/runtime/runner/in-memory.d.mts +15 -0
  203. package/dist/v2/runtime/runner/in-memory.d.mts.map +1 -0
  204. package/dist/v2/runtime/runner/in-memory.mjs +222 -0
  205. package/dist/v2/runtime/runner/in-memory.mjs.map +1 -0
  206. package/dist/v2/runtime/runner/index.d.cts +6 -0
  207. package/dist/v2/runtime/runner/index.d.mts +6 -0
  208. package/dist/v2/runtime/runner/index.mjs +7 -0
  209. package/dist/v2/runtime/runner/intelligence.cjs +246 -0
  210. package/dist/v2/runtime/runner/intelligence.cjs.map +1 -0
  211. package/dist/v2/runtime/runner/intelligence.d.cts +57 -0
  212. package/dist/v2/runtime/runner/intelligence.d.cts.map +1 -0
  213. package/dist/v2/runtime/runner/intelligence.d.mts +57 -0
  214. package/dist/v2/runtime/runner/intelligence.d.mts.map +1 -0
  215. package/dist/v2/runtime/runner/intelligence.mjs +245 -0
  216. package/dist/v2/runtime/runner/intelligence.mjs.map +1 -0
  217. package/dist/v2/runtime/runtime.cjs +101 -0
  218. package/dist/v2/runtime/runtime.cjs.map +1 -0
  219. package/dist/v2/runtime/runtime.d.cts +132 -0
  220. package/dist/v2/runtime/runtime.d.cts.map +1 -0
  221. package/dist/v2/runtime/runtime.d.mts +133 -0
  222. package/dist/v2/runtime/runtime.d.mts.map +1 -0
  223. package/dist/v2/runtime/runtime.mjs +97 -0
  224. package/dist/v2/runtime/runtime.mjs.map +1 -0
  225. package/dist/v2/runtime/telemetry/scarf-client.cjs +32 -0
  226. package/dist/v2/runtime/telemetry/scarf-client.cjs.map +1 -0
  227. package/dist/v2/runtime/telemetry/scarf-client.mjs +32 -0
  228. package/dist/v2/runtime/telemetry/scarf-client.mjs.map +1 -0
  229. package/dist/v2/runtime/telemetry/telemetry-client.cjs +35 -0
  230. package/dist/v2/runtime/telemetry/telemetry-client.cjs.map +1 -0
  231. package/dist/v2/runtime/telemetry/telemetry-client.mjs +35 -0
  232. package/dist/v2/runtime/telemetry/telemetry-client.mjs.map +1 -0
  233. package/dist/v2/runtime/transcription-service/transcription-service.cjs +8 -0
  234. package/dist/v2/runtime/transcription-service/transcription-service.cjs.map +1 -0
  235. package/dist/v2/runtime/transcription-service/transcription-service.d.cts +15 -0
  236. package/dist/v2/runtime/transcription-service/transcription-service.d.cts.map +1 -0
  237. package/dist/v2/runtime/transcription-service/transcription-service.d.mts +15 -0
  238. package/dist/v2/runtime/transcription-service/transcription-service.d.mts.map +1 -0
  239. package/dist/v2/runtime/transcription-service/transcription-service.mjs +7 -0
  240. package/dist/v2/runtime/transcription-service/transcription-service.mjs.map +1 -0
  241. package/package.json +22 -5
  242. package/src/agent/__tests__/ai-sdk-v6-compat.test.ts +116 -0
  243. package/src/agent/__tests__/basic-agent.test.ts +1248 -0
  244. package/src/agent/__tests__/config-tools-execution.test.ts +516 -0
  245. package/src/agent/__tests__/mcp-clients.test.ts +260 -0
  246. package/src/agent/__tests__/property-overrides.test.ts +598 -0
  247. package/src/agent/__tests__/standard-schema-tools.test.ts +313 -0
  248. package/src/agent/__tests__/standard-schema-types.test.ts +158 -0
  249. package/src/agent/__tests__/state-tools.test.ts +436 -0
  250. package/src/agent/__tests__/test-helpers.ts +178 -0
  251. package/src/agent/__tests__/utils.test.ts +536 -0
  252. package/src/agent/__tests__/zod-regression.test.ts +350 -0
  253. package/src/agent/index.ts +1305 -0
  254. package/src/graphql/message-conversion/agui-to-gql.test.ts +1 -1
  255. package/src/graphql/message-conversion/agui-to-gql.ts +1 -1
  256. package/src/graphql/message-conversion/gql-to-agui.ts +1 -1
  257. package/src/graphql/message-conversion/roundtrip-conversion.test.ts +1 -1
  258. package/src/lib/integrations/nextjs/app-router.ts +2 -2
  259. package/src/lib/integrations/node-http/index.ts +2 -2
  260. package/src/lib/runtime/copilot-runtime.ts +3 -5
  261. package/src/lib/runtime/telemetry-agent-runner.ts +1 -1
  262. package/src/service-adapters/conversion.test.ts +1 -1
  263. package/src/service-adapters/conversion.ts +1 -28
  264. package/src/v2/index.ts +5 -2
  265. package/src/v2/runtime/__tests__/cors-credentials.test.ts +320 -0
  266. package/src/v2/runtime/__tests__/express-abort-signal.test.ts +25 -0
  267. package/src/v2/runtime/__tests__/express-body-order.test.ts +76 -0
  268. package/src/v2/runtime/__tests__/express-single-sse.test.ts +122 -0
  269. package/src/v2/runtime/__tests__/get-runtime-info.test.ts +141 -0
  270. package/src/v2/runtime/__tests__/handle-connect.test.ts +423 -0
  271. package/src/v2/runtime/__tests__/handle-run.test.ts +910 -0
  272. package/src/v2/runtime/__tests__/handle-threads.test.ts +388 -0
  273. package/src/v2/runtime/__tests__/handle-transcribe.test.ts +301 -0
  274. package/src/v2/runtime/__tests__/header-utils.test.ts +88 -0
  275. package/src/v2/runtime/__tests__/in-process-agent-runner-messages.test.ts +230 -0
  276. package/src/v2/runtime/__tests__/in-process-agent-runner.test.ts +1030 -0
  277. package/src/v2/runtime/__tests__/middleware-express.test.ts +206 -0
  278. package/src/v2/runtime/__tests__/middleware-single-express.test.ts +211 -0
  279. package/src/v2/runtime/__tests__/middleware-single.test.ts +225 -0
  280. package/src/v2/runtime/__tests__/middleware-sse-parser.test.ts +187 -0
  281. package/src/v2/runtime/__tests__/middleware.test.ts +251 -0
  282. package/src/v2/runtime/__tests__/routing-express.test.ts +174 -0
  283. package/src/v2/runtime/__tests__/routing-single-express.test.ts +168 -0
  284. package/src/v2/runtime/__tests__/routing-single.test.ts +193 -0
  285. package/src/v2/runtime/__tests__/routing.test.ts +257 -0
  286. package/src/v2/runtime/__tests__/runtime.test.ts +123 -0
  287. package/src/v2/runtime/__tests__/telemetry.test.ts +167 -0
  288. package/src/v2/runtime/__tests__/thread-names.test.ts +188 -0
  289. package/src/v2/runtime/endpoints/express-single.ts +231 -0
  290. package/src/v2/runtime/endpoints/express-utils.ts +182 -0
  291. package/src/v2/runtime/endpoints/express.ts +275 -0
  292. package/src/v2/runtime/endpoints/hono-single.ts +212 -0
  293. package/src/v2/runtime/endpoints/hono.ts +314 -0
  294. package/src/v2/runtime/endpoints/index.ts +4 -0
  295. package/src/v2/runtime/endpoints/single-route-helpers.ts +125 -0
  296. package/src/v2/runtime/express.ts +2 -0
  297. package/src/v2/runtime/handler.ts +3 -0
  298. package/src/v2/runtime/handlers/get-runtime-info.ts +79 -0
  299. package/src/v2/runtime/handlers/handle-connect.ts +76 -0
  300. package/src/v2/runtime/handlers/handle-run.ts +89 -0
  301. package/src/v2/runtime/handlers/handle-stop.ts +76 -0
  302. package/src/v2/runtime/handlers/handle-threads.ts +7 -0
  303. package/src/v2/runtime/handlers/handle-transcribe.ts +256 -0
  304. package/src/v2/runtime/handlers/header-utils.ts +24 -0
  305. package/src/v2/runtime/handlers/intelligence/connect.ts +65 -0
  306. package/src/v2/runtime/handlers/intelligence/run.ts +152 -0
  307. package/src/v2/runtime/handlers/intelligence/thread-names.ts +246 -0
  308. package/src/v2/runtime/handlers/intelligence/threads.ts +233 -0
  309. package/src/v2/runtime/handlers/shared/agent-utils.ts +136 -0
  310. package/src/v2/runtime/handlers/shared/intelligence-utils.ts +21 -0
  311. package/src/v2/runtime/handlers/shared/json-response.ts +6 -0
  312. package/src/v2/runtime/handlers/shared/resolve-intelligence-user.ts +25 -0
  313. package/src/v2/runtime/handlers/shared/sse-response.ts +100 -0
  314. package/src/v2/runtime/handlers/sse/connect.ts +24 -0
  315. package/src/v2/runtime/handlers/sse/run.ts +27 -0
  316. package/src/v2/runtime/index.ts +20 -0
  317. package/src/v2/runtime/intelligence-platform/__tests__/client.test.ts +605 -0
  318. package/src/v2/runtime/intelligence-platform/client.ts +659 -0
  319. package/src/v2/runtime/intelligence-platform/index.ts +10 -0
  320. package/src/v2/runtime/middleware-sse-parser.ts +200 -0
  321. package/src/v2/runtime/middleware.ts +115 -0
  322. package/src/v2/runtime/runner/__tests__/finalize-events.test.ts +109 -0
  323. package/src/v2/runtime/runner/__tests__/in-memory-runner.e2e.test.ts +775 -0
  324. package/src/v2/runtime/runner/__tests__/in-memory-runner.test.ts +363 -0
  325. package/src/v2/runtime/runner/__tests__/intelligence-runner.test.ts +981 -0
  326. package/src/v2/runtime/runner/agent-runner.ts +36 -0
  327. package/src/v2/runtime/runner/in-memory.ts +381 -0
  328. package/src/v2/runtime/runner/index.ts +4 -0
  329. package/src/v2/runtime/runner/intelligence.ts +429 -0
  330. package/src/v2/runtime/runtime.ts +260 -0
  331. package/src/v2/runtime/telemetry/events.ts +35 -0
  332. package/src/v2/runtime/telemetry/index.ts +7 -0
  333. package/src/v2/runtime/telemetry/scarf-client.ts +39 -0
  334. package/src/v2/runtime/telemetry/telemetry-client.ts +70 -0
  335. package/src/v2/runtime/transcription-service/transcription-service.ts +11 -0
  336. package/tsconfig.json +9 -2
  337. package/tsdown.config.ts +1 -0
@@ -0,0 +1,910 @@
1
+ import { Observable } from "rxjs";
2
+ import { describe, it, expect, vi } from "vitest";
3
+ import { AbstractAgent, HttpAgent } from "@ag-ui/client";
4
+ import { A2UIMiddleware } from "@ag-ui/a2ui-middleware";
5
+ import { handleRunAgent } from "../handlers/handle-run";
6
+ import { CopilotRuntime } from "../runtime";
7
+ import { IntelligenceAgentRunner } from "../runner/intelligence";
8
+ import { CopilotKitIntelligence } from "../intelligence-platform/client";
9
+
10
+ describe("handleRunAgent", () => {
11
+ const createMockRuntime = (
12
+ agents: Record<string, unknown> = {},
13
+ ): CopilotRuntime => {
14
+ return {
15
+ agents: Promise.resolve(agents),
16
+ transcriptionService: undefined,
17
+ beforeRequestMiddleware: undefined,
18
+ afterRequestMiddleware: undefined,
19
+ } as CopilotRuntime;
20
+ };
21
+
22
+ const createMockRequest = (): Request => {
23
+ return new Request("https://example.com/agent/test/run", {
24
+ method: "POST",
25
+ });
26
+ };
27
+
28
+ it("should return 404 when agent does not exist", async () => {
29
+ const runtime = createMockRuntime({}); // Empty agents
30
+ const request = createMockRequest();
31
+ const agentId = "nonexistent-agent";
32
+
33
+ const response = await handleRunAgent({
34
+ runtime,
35
+ request,
36
+ agentId,
37
+ });
38
+
39
+ expect(response.status).toBe(404);
40
+ expect(response.headers.get("Content-Type")).toBe("application/json");
41
+
42
+ const body = await response.json();
43
+ expect(body).toEqual({
44
+ error: "Agent not found",
45
+ message: "Agent 'nonexistent-agent' does not exist",
46
+ });
47
+ });
48
+
49
+ it("should return 500 when runtime.agents throws an error", async () => {
50
+ const runtime = {
51
+ agents: Promise.reject(new Error("Database connection failed")),
52
+ transcriptionService: undefined,
53
+ beforeRequestMiddleware: undefined,
54
+ afterRequestMiddleware: undefined,
55
+ } as CopilotRuntime;
56
+ const request = createMockRequest();
57
+ const agentId = "test-agent";
58
+
59
+ const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
60
+
61
+ try {
62
+ const response = await handleRunAgent({
63
+ runtime,
64
+ request,
65
+ agentId,
66
+ });
67
+
68
+ expect(response.status).toBe(500);
69
+ expect(response.headers.get("Content-Type")).toBe("application/json");
70
+
71
+ const body = await response.json();
72
+ expect(body).toEqual({
73
+ error: "Failed to run agent",
74
+ message: "Database connection failed",
75
+ });
76
+ } finally {
77
+ errorSpy.mockRestore();
78
+ }
79
+ });
80
+
81
+ it("forwards only authorization and custom x- headers to HttpAgent runs", async () => {
82
+ class RecordingHttpAgent extends HttpAgent {
83
+ constructor(initialHeaders: Record<string, string>) {
84
+ super({ url: "https://runtime.example/agent" });
85
+ this.headers = initialHeaders;
86
+ }
87
+
88
+ clone(): AbstractAgent {
89
+ return new RecordingHttpAgent({});
90
+ }
91
+ }
92
+
93
+ const baseHeaders = {
94
+ Authorization: "Bearer base",
95
+ };
96
+
97
+ const registeredAgent = new RecordingHttpAgent(baseHeaders);
98
+
99
+ const recordedHeaders: Array<Record<string, string>> = [];
100
+ let resolveRun: (() => void) | undefined;
101
+ const runInvoked = new Promise<void>((resolve) => {
102
+ resolveRun = resolve;
103
+ });
104
+
105
+ const runtime = {
106
+ agents: Promise.resolve({ "test-agent": registeredAgent }),
107
+ transcriptionService: undefined,
108
+ beforeRequestMiddleware: undefined,
109
+ afterRequestMiddleware: undefined,
110
+ runner: {
111
+ run: ({ agent }: { agent: AbstractAgent }) =>
112
+ new Observable<BaseEvent>((subscriber) => {
113
+ recordedHeaders.push({ ...(agent as HttpAgent).headers });
114
+ resolveRun?.();
115
+ subscriber.complete();
116
+ }),
117
+ connect: () =>
118
+ new Observable<BaseEvent>((subscriber) => {
119
+ subscriber.complete();
120
+ }),
121
+ isRunning: async () => false,
122
+ stop: async () => false,
123
+ },
124
+ } as CopilotRuntime;
125
+
126
+ const requestBody = {
127
+ threadId: "thread-1",
128
+ runId: "run-1",
129
+ state: {},
130
+ messages: [],
131
+ tools: [],
132
+ context: [],
133
+ forwardedProps: {},
134
+ };
135
+
136
+ const request = new Request("https://example.com/agent/test-agent/run", {
137
+ method: "POST",
138
+ headers: {
139
+ "Content-Type": "application/json",
140
+ "X-Custom": "custom-value",
141
+ Authorization: "Bearer forwarded",
142
+ Origin: "http://localhost:4200",
143
+ },
144
+ body: JSON.stringify(requestBody),
145
+ });
146
+
147
+ const response = await handleRunAgent({
148
+ runtime,
149
+ request,
150
+ agentId: "test-agent",
151
+ });
152
+
153
+ expect(response.status).toBe(200);
154
+ await runInvoked;
155
+
156
+ expect(recordedHeaders).toHaveLength(1);
157
+ expect(recordedHeaders[0]).toMatchObject({
158
+ authorization: "Bearer forwarded",
159
+ "x-custom": "custom-value",
160
+ });
161
+ expect(recordedHeaders[0]).not.toHaveProperty("origin");
162
+ expect(recordedHeaders[0]).not.toHaveProperty("content-type");
163
+ });
164
+
165
+ const createMockAgentWithUse = () => {
166
+ const useSpy = vi.fn();
167
+ const agent = {
168
+ clone: () => ({ ...agent, use: useSpy }),
169
+ use: useSpy,
170
+ } as unknown as AbstractAgent;
171
+ return { agent, useSpy };
172
+ };
173
+
174
+ const createMockRunner = () => ({
175
+ run: () =>
176
+ new Observable<BaseEvent>((subscriber) => {
177
+ subscriber.complete();
178
+ }),
179
+ connect: () =>
180
+ new Observable<BaseEvent>((subscriber) => {
181
+ subscriber.complete();
182
+ }),
183
+ isRunning: async () => false,
184
+ stop: async () => false,
185
+ });
186
+
187
+ const createRunRequest = (headers?: Record<string, string>) =>
188
+ new Request("https://example.com/agent/my-agent/run", {
189
+ method: "POST",
190
+ headers: { "Content-Type": "application/json", ...headers },
191
+ body: JSON.stringify({
192
+ threadId: "thread-1",
193
+ runId: "run-1",
194
+ state: {},
195
+ messages: [],
196
+ tools: [],
197
+ context: [],
198
+ forwardedProps: {},
199
+ }),
200
+ });
201
+
202
+ it("applies A2UIMiddleware to all agents when a2ui.enabled is true and no agents filter", async () => {
203
+ const { agent, useSpy } = createMockAgentWithUse();
204
+
205
+ const runtime = {
206
+ agents: Promise.resolve({ "my-agent": agent }),
207
+ transcriptionService: undefined,
208
+ beforeRequestMiddleware: undefined,
209
+ afterRequestMiddleware: undefined,
210
+ runner: createMockRunner(),
211
+ a2ui: { enabled: true, injectA2UITool: true },
212
+ } as unknown as CopilotRuntime;
213
+
214
+ await handleRunAgent({
215
+ runtime,
216
+ request: createRunRequest(),
217
+ agentId: "my-agent",
218
+ });
219
+
220
+ expect(useSpy).toHaveBeenCalledOnce();
221
+ expect(useSpy.mock.calls[0][0]).toBeInstanceOf(A2UIMiddleware);
222
+ });
223
+
224
+ it("applies A2UIMiddleware only to matching agent when agents filter is set", async () => {
225
+ const { agent: matchingAgent, useSpy: matchingSpy } =
226
+ createMockAgentWithUse();
227
+ const { agent: otherAgent, useSpy: otherSpy } = createMockAgentWithUse();
228
+
229
+ const makeRuntime = (agentId: string, targetAgent: AbstractAgent) =>
230
+ ({
231
+ agents: Promise.resolve({ [agentId]: targetAgent }),
232
+ transcriptionService: undefined,
233
+ beforeRequestMiddleware: undefined,
234
+ afterRequestMiddleware: undefined,
235
+ runner: createMockRunner(),
236
+ a2ui: { enabled: true, agents: ["my-agent"] },
237
+ }) as unknown as CopilotRuntime;
238
+
239
+ // Should apply for "my-agent"
240
+ await handleRunAgent({
241
+ runtime: makeRuntime("my-agent", matchingAgent),
242
+ request: createRunRequest(),
243
+ agentId: "my-agent",
244
+ });
245
+ expect(matchingSpy).toHaveBeenCalledOnce();
246
+
247
+ // Should NOT apply for "other-agent"
248
+ const otherRequest = new Request("https://example.com/agent/other/run", {
249
+ method: "POST",
250
+ headers: { "Content-Type": "application/json" },
251
+ body: JSON.stringify({
252
+ threadId: "thread-1",
253
+ runId: "run-1",
254
+ state: {},
255
+ messages: [],
256
+ tools: [],
257
+ context: [],
258
+ forwardedProps: {},
259
+ }),
260
+ });
261
+ await handleRunAgent({
262
+ runtime: makeRuntime("other", otherAgent),
263
+ request: otherRequest,
264
+ agentId: "other",
265
+ });
266
+ expect(otherSpy).not.toHaveBeenCalled();
267
+ });
268
+
269
+ it("does not apply A2UIMiddleware when a2ui is omitted", async () => {
270
+ const { agent, useSpy } = createMockAgentWithUse();
271
+
272
+ const runtime = {
273
+ agents: Promise.resolve({ "my-agent": agent }),
274
+ transcriptionService: undefined,
275
+ beforeRequestMiddleware: undefined,
276
+ afterRequestMiddleware: undefined,
277
+ runner: createMockRunner(),
278
+ } as unknown as CopilotRuntime;
279
+
280
+ await handleRunAgent({
281
+ runtime,
282
+ request: createRunRequest(),
283
+ agentId: "my-agent",
284
+ });
285
+
286
+ expect(useSpy).not.toHaveBeenCalled();
287
+ });
288
+
289
+ describe("IntelligenceAgentRunner join code path", () => {
290
+ const createIntelligenceRuntime = (
291
+ agent: AbstractAgent,
292
+ platform?: Partial<CopilotKitIntelligence>,
293
+ options?: {
294
+ generateThreadNames?: boolean;
295
+ identifyUser?: (
296
+ request: Request,
297
+ ) => { id: string } | Promise<{ id: string }>;
298
+ },
299
+ ) => {
300
+ const runner = Object.create(IntelligenceAgentRunner.prototype);
301
+ runner.run = vi.fn(
302
+ () =>
303
+ new Observable<BaseEvent>((subscriber) => {
304
+ subscriber.complete();
305
+ }),
306
+ );
307
+ return {
308
+ agents: Promise.resolve({ "my-agent": agent }),
309
+ transcriptionService: undefined,
310
+ beforeRequestMiddleware: undefined,
311
+ afterRequestMiddleware: undefined,
312
+ runner,
313
+ mode: "intelligence",
314
+ generateThreadNames: options?.generateThreadNames ?? false,
315
+ intelligence: platform,
316
+ identifyUser:
317
+ options?.identifyUser ?? vi.fn().mockResolvedValue({ id: "user-1" }),
318
+ } as unknown as CopilotRuntime;
319
+ };
320
+
321
+ const createAgentForIntelligence = () => {
322
+ const createClone = () =>
323
+ ({
324
+ clone: vi.fn(() => createClone()),
325
+ setMessages: vi.fn(),
326
+ setState: vi.fn(),
327
+ threadId: undefined,
328
+ headers: {},
329
+ runAgent: vi.fn().mockResolvedValue(undefined),
330
+ }) as unknown as AbstractAgent;
331
+
332
+ const agent = {
333
+ clone: vi.fn(() => createClone()),
334
+ setMessages: vi.fn(),
335
+ setState: vi.fn(),
336
+ threadId: undefined,
337
+ headers: {},
338
+ runAgent: vi.fn().mockResolvedValue(undefined),
339
+ } as unknown as AbstractAgent;
340
+ return agent;
341
+ };
342
+
343
+ it("returns joinToken JSON when lock is acquired", async () => {
344
+ const agent = createAgentForIntelligence();
345
+ const platform = {
346
+ getOrCreateThread: vi.fn().mockResolvedValue({
347
+ thread: { id: "thread-1", name: null },
348
+ created: false,
349
+ }),
350
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
351
+ ɵacquireThreadLock: vi
352
+ .fn()
353
+ .mockResolvedValue({ joinToken: "jt-123", joinCode: "jc-123" }),
354
+ };
355
+ const runtime = createIntelligenceRuntime(agent, platform as any);
356
+
357
+ const response = await handleRunAgent({
358
+ runtime,
359
+ request: createRunRequest(),
360
+ agentId: "my-agent",
361
+ });
362
+
363
+ expect(response.status).toBe(200);
364
+ expect(response.headers.get("Content-Type")).toBe("application/json");
365
+ const body = await response.json();
366
+ expect(body).toEqual({ joinToken: "jt-123" });
367
+ expect(platform.getOrCreateThread).toHaveBeenCalledWith({
368
+ threadId: "thread-1",
369
+ userId: "user-1",
370
+ agentId: "my-agent",
371
+ });
372
+ expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
373
+ threadId: "thread-1",
374
+ runId: "run-1",
375
+ userId: "user-1",
376
+ });
377
+ expect(platform.getThreadMessages).toHaveBeenCalledWith({
378
+ threadId: "thread-1",
379
+ });
380
+ });
381
+
382
+ it("uses identifyUser instead of a conflicting X-User-Id header", async () => {
383
+ const agent = createAgentForIntelligence();
384
+ const platform = {
385
+ getOrCreateThread: vi.fn().mockResolvedValue({
386
+ thread: { id: "thread-1", name: null },
387
+ created: false,
388
+ }),
389
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
390
+ ɵacquireThreadLock: vi
391
+ .fn()
392
+ .mockResolvedValue({ joinToken: "jt-123", joinCode: "jc-123" }),
393
+ };
394
+ const identifyUser = vi.fn().mockResolvedValue({ id: "resolved-user" });
395
+ const runtime = createIntelligenceRuntime(agent, platform as any, {
396
+ identifyUser,
397
+ });
398
+ const request = createRunRequest({ "X-User-Id": "legacy-user" });
399
+
400
+ const response = await handleRunAgent({
401
+ runtime,
402
+ request,
403
+ agentId: "my-agent",
404
+ });
405
+
406
+ expect(response.status).toBe(200);
407
+ expect(identifyUser).toHaveBeenCalledTimes(1);
408
+ expect(identifyUser).toHaveBeenCalledWith(request);
409
+ expect(platform.getOrCreateThread).toHaveBeenCalledWith({
410
+ threadId: "thread-1",
411
+ userId: "resolved-user",
412
+ agentId: "my-agent",
413
+ });
414
+ expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
415
+ threadId: "thread-1",
416
+ runId: "run-1",
417
+ userId: "resolved-user",
418
+ });
419
+ });
420
+
421
+ it("passes joinCode to runner.run() when provided", async () => {
422
+ const agent = createAgentForIntelligence();
423
+ const platform = {
424
+ getOrCreateThread: vi.fn().mockResolvedValue({
425
+ thread: { id: "thread-1", name: null },
426
+ created: false,
427
+ }),
428
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
429
+ ɵacquireThreadLock: vi
430
+ .fn()
431
+ .mockResolvedValue({ joinToken: "jt-456", joinCode: "jc-456" }),
432
+ };
433
+ const runtime = createIntelligenceRuntime(agent, platform as any);
434
+
435
+ await handleRunAgent({
436
+ runtime,
437
+ request: createRunRequest(),
438
+ agentId: "my-agent",
439
+ });
440
+
441
+ expect(runtime.runner.run).toHaveBeenCalledWith(
442
+ expect.objectContaining({ joinCode: "jc-456" }),
443
+ );
444
+ });
445
+
446
+ it("returns 502 when joinToken is missing", async () => {
447
+ const agent = createAgentForIntelligence();
448
+ const platform = {
449
+ getOrCreateThread: vi.fn().mockResolvedValue({
450
+ thread: { id: "thread-1", name: null },
451
+ created: false,
452
+ }),
453
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
454
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({ joinCode: "jc-789" }),
455
+ };
456
+ const runtime = createIntelligenceRuntime(agent, platform as any);
457
+
458
+ const response = await handleRunAgent({
459
+ runtime,
460
+ request: createRunRequest(),
461
+ agentId: "my-agent",
462
+ });
463
+
464
+ expect(response.status).toBe(502);
465
+ const body = await response.json();
466
+ expect(body.error).toBe("Join token not available");
467
+ expect(runtime.runner.run).not.toHaveBeenCalled();
468
+ });
469
+
470
+ it("returns 409 when thread lock is denied", async () => {
471
+ const agent = createAgentForIntelligence();
472
+ const platform = {
473
+ getOrCreateThread: vi.fn().mockResolvedValue({
474
+ thread: { id: "thread-1", name: null },
475
+ created: false,
476
+ }),
477
+ getThreadMessages: vi.fn(),
478
+ ɵacquireThreadLock: vi
479
+ .fn()
480
+ .mockRejectedValue(new Error("Thread is locked by another runner")),
481
+ };
482
+ const runtime = createIntelligenceRuntime(agent, platform as any);
483
+
484
+ const response = await handleRunAgent({
485
+ runtime,
486
+ request: createRunRequest(),
487
+ agentId: "my-agent",
488
+ });
489
+
490
+ expect(response.status).toBe(409);
491
+ const body = await response.json();
492
+ expect(body.error).toBe("Thread lock denied");
493
+ });
494
+
495
+ it("passes only unseen input messages to the runner for durable persistence", async () => {
496
+ const agent = createAgentForIntelligence();
497
+ const platform = {
498
+ getOrCreateThread: vi.fn().mockResolvedValue({
499
+ thread: { id: "thread-1", name: null },
500
+ created: false,
501
+ }),
502
+ getThreadMessages: vi.fn().mockResolvedValue({
503
+ messages: [
504
+ {
505
+ id: "msg-existing",
506
+ role: "user",
507
+ content: "First turn",
508
+ },
509
+ ],
510
+ }),
511
+ ɵacquireThreadLock: vi
512
+ .fn()
513
+ .mockResolvedValue({ joinToken: "jt-123", joinCode: "jc-123" }),
514
+ };
515
+ const runtime = createIntelligenceRuntime(agent, platform as any);
516
+ const response = await handleRunAgent({
517
+ runtime,
518
+ request: new Request("https://example.com/agent/my-agent/run", {
519
+ method: "POST",
520
+ headers: {
521
+ "Content-Type": "application/json",
522
+ },
523
+ body: JSON.stringify({
524
+ threadId: "thread-1",
525
+ runId: "run-1",
526
+ state: {},
527
+ messages: [
528
+ {
529
+ id: "msg-existing",
530
+ role: "user",
531
+ content: "First turn",
532
+ },
533
+ {
534
+ id: "msg-new",
535
+ role: "user",
536
+ content: "Second turn",
537
+ },
538
+ ],
539
+ tools: [],
540
+ context: [],
541
+ forwardedProps: {},
542
+ }),
543
+ }),
544
+ agentId: "my-agent",
545
+ });
546
+
547
+ expect(response.status).toBe(200);
548
+ expect(runtime.runner.run).toHaveBeenCalledWith(
549
+ expect.objectContaining({
550
+ persistedInputMessages: [
551
+ {
552
+ id: "msg-new",
553
+ role: "user",
554
+ content: "Second turn",
555
+ },
556
+ ],
557
+ }),
558
+ );
559
+ });
560
+
561
+ it("returns 502 when durable thread history lookup fails", async () => {
562
+ const agent = createAgentForIntelligence();
563
+ const platform = {
564
+ getOrCreateThread: vi.fn().mockResolvedValue({
565
+ thread: { id: "thread-1", name: null },
566
+ created: false,
567
+ }),
568
+ getThreadMessages: vi
569
+ .fn()
570
+ .mockRejectedValue(new Error("history unavailable")),
571
+ ɵacquireThreadLock: vi
572
+ .fn()
573
+ .mockResolvedValue({ joinToken: "jt-123", joinCode: "jc-123" }),
574
+ };
575
+ const runtime = createIntelligenceRuntime(agent, platform as any);
576
+
577
+ const response = await handleRunAgent({
578
+ runtime,
579
+ request: createRunRequest(),
580
+ agentId: "my-agent",
581
+ });
582
+
583
+ expect(response.status).toBe(502);
584
+ const body = await response.json();
585
+ expect(body.error).toBe("Thread history lookup failed");
586
+ expect(runtime.runner.run).not.toHaveBeenCalled();
587
+ });
588
+
589
+ it("creates the thread before locking when run targets a fresh thread", async () => {
590
+ const agent = createAgentForIntelligence();
591
+ const platform = {
592
+ getOrCreateThread: vi.fn().mockResolvedValue({
593
+ thread: { id: "thread-1", name: null },
594
+ created: true,
595
+ }),
596
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
597
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({
598
+ joinToken: "jt-created",
599
+ joinCode: "jc-created",
600
+ }),
601
+ };
602
+ const runtime = createIntelligenceRuntime(agent, platform as any);
603
+
604
+ const response = await handleRunAgent({
605
+ runtime,
606
+ request: createRunRequest(),
607
+ agentId: "my-agent",
608
+ });
609
+
610
+ expect(response.status).toBe(200);
611
+ expect(platform.getOrCreateThread).toHaveBeenCalledWith({
612
+ threadId: "thread-1",
613
+ userId: "user-1",
614
+ agentId: "my-agent",
615
+ });
616
+ expect(platform.ɵacquireThreadLock).toHaveBeenCalledWith({
617
+ threadId: "thread-1",
618
+ runId: "run-1",
619
+ userId: "user-1",
620
+ });
621
+ });
622
+
623
+ it("generates and persists a thread name in the background for new unnamed threads", async () => {
624
+ const namingAgent = {
625
+ clone: vi.fn(),
626
+ setMessages: vi.fn(),
627
+ setState: vi.fn(),
628
+ threadId: undefined,
629
+ headers: {},
630
+ runAgent: vi.fn().mockResolvedValue({
631
+ newMessages: [
632
+ {
633
+ id: "assistant-1",
634
+ role: "assistant",
635
+ content: '{"title":"**Order refund** status"}',
636
+ },
637
+ ],
638
+ }),
639
+ } as unknown as AbstractAgent;
640
+ const baseAgent = {
641
+ clone: vi
642
+ .fn()
643
+ .mockReturnValueOnce({
644
+ clone: vi.fn(),
645
+ setMessages: vi.fn(),
646
+ setState: vi.fn(),
647
+ threadId: undefined,
648
+ headers: {},
649
+ runAgent: vi.fn().mockResolvedValue(undefined),
650
+ })
651
+ .mockReturnValueOnce(namingAgent),
652
+ setMessages: vi.fn(),
653
+ setState: vi.fn(),
654
+ threadId: undefined,
655
+ headers: {},
656
+ runAgent: vi.fn().mockResolvedValue(undefined),
657
+ } as unknown as AbstractAgent;
658
+ const platform = {
659
+ getOrCreateThread: vi.fn().mockResolvedValue({
660
+ thread: { id: "thread-1", name: null },
661
+ created: true,
662
+ }),
663
+ updateThread: vi.fn().mockResolvedValue({
664
+ id: "thread-1",
665
+ name: "Order refund status",
666
+ }),
667
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
668
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({
669
+ joinToken: "jt-created",
670
+ joinCode: "jc-created",
671
+ }),
672
+ };
673
+ const runtime = createIntelligenceRuntime(baseAgent, platform as any, {
674
+ generateThreadNames: true,
675
+ });
676
+ const response = await handleRunAgent({
677
+ runtime,
678
+ request: new Request("https://example.com/agent/my-agent/run", {
679
+ method: "POST",
680
+ headers: {
681
+ "Content-Type": "application/json",
682
+ },
683
+ body: JSON.stringify({
684
+ threadId: "thread-1",
685
+ runId: "run-1",
686
+ state: {},
687
+ messages: [
688
+ {
689
+ id: "user-1",
690
+ role: "user",
691
+ content: "Can you help me with my refund request?",
692
+ },
693
+ ],
694
+ tools: [],
695
+ context: [],
696
+ forwardedProps: {},
697
+ }),
698
+ }),
699
+ agentId: "my-agent",
700
+ });
701
+
702
+ expect(response.status).toBe(200);
703
+ await vi.waitFor(() =>
704
+ expect(platform.updateThread).toHaveBeenCalledWith({
705
+ threadId: "thread-1",
706
+ userId: "user-1",
707
+ agentId: "my-agent",
708
+ updates: { name: "Order refund status" },
709
+ }),
710
+ );
711
+ expect(runtime.runner.run).toHaveBeenCalledTimes(1);
712
+ });
713
+
714
+ it("does not generate a thread name when generateThreadNames is false", async () => {
715
+ const agent = createAgentForIntelligence();
716
+ const platform = {
717
+ getOrCreateThread: vi.fn().mockResolvedValue({
718
+ thread: { id: "thread-1", name: null },
719
+ created: true,
720
+ }),
721
+ updateThread: vi.fn(),
722
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
723
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({
724
+ joinToken: "jt-created",
725
+ joinCode: "jc-created",
726
+ }),
727
+ };
728
+ const runtime = createIntelligenceRuntime(agent, platform as any, {
729
+ generateThreadNames: false,
730
+ });
731
+
732
+ const response = await handleRunAgent({
733
+ runtime,
734
+ request: createRunRequest(),
735
+ agentId: "my-agent",
736
+ });
737
+
738
+ expect(response.status).toBe(200);
739
+ await Promise.resolve();
740
+ expect(platform.updateThread).not.toHaveBeenCalled();
741
+ });
742
+
743
+ it("does not generate a thread name when the created thread already has a name", async () => {
744
+ const agent = createAgentForIntelligence();
745
+ const platform = {
746
+ getOrCreateThread: vi.fn().mockResolvedValue({
747
+ thread: { id: "thread-1", name: "Existing name" },
748
+ created: true,
749
+ }),
750
+ updateThread: vi.fn(),
751
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
752
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({
753
+ joinToken: "jt-created",
754
+ joinCode: "jc-created",
755
+ }),
756
+ };
757
+ const runtime = createIntelligenceRuntime(agent, platform as any, {
758
+ generateThreadNames: true,
759
+ });
760
+
761
+ const response = await handleRunAgent({
762
+ runtime,
763
+ request: createRunRequest(),
764
+ agentId: "my-agent",
765
+ });
766
+
767
+ expect(response.status).toBe(200);
768
+ await Promise.resolve();
769
+ expect(platform.updateThread).not.toHaveBeenCalled();
770
+ });
771
+
772
+ it("retries thread naming three times and falls back to Untitled", async () => {
773
+ const namingAgent = {
774
+ clone: vi.fn(),
775
+ setMessages: vi.fn(),
776
+ setState: vi.fn(),
777
+ threadId: undefined,
778
+ headers: {},
779
+ runAgent: vi.fn().mockRejectedValue(new Error("naming failed")),
780
+ } as unknown as AbstractAgent;
781
+ const baseAgent = {
782
+ clone: vi
783
+ .fn()
784
+ .mockReturnValueOnce({
785
+ clone: vi.fn(),
786
+ setMessages: vi.fn(),
787
+ setState: vi.fn(),
788
+ threadId: undefined,
789
+ headers: {},
790
+ runAgent: vi.fn().mockResolvedValue(undefined),
791
+ })
792
+ .mockReturnValueOnce(namingAgent)
793
+ .mockReturnValueOnce(namingAgent)
794
+ .mockReturnValueOnce(namingAgent),
795
+ setMessages: vi.fn(),
796
+ setState: vi.fn(),
797
+ threadId: undefined,
798
+ headers: {},
799
+ runAgent: vi.fn().mockResolvedValue(undefined),
800
+ } as unknown as AbstractAgent;
801
+ const platform = {
802
+ getOrCreateThread: vi.fn().mockResolvedValue({
803
+ thread: { id: "thread-1", name: null },
804
+ created: true,
805
+ }),
806
+ updateThread: vi.fn(),
807
+ getThreadMessages: vi.fn().mockResolvedValue({ messages: [] }),
808
+ ɵacquireThreadLock: vi.fn().mockResolvedValue({
809
+ joinToken: "jt-created",
810
+ joinCode: "jc-created",
811
+ }),
812
+ };
813
+ const runtime = createIntelligenceRuntime(baseAgent, platform as any, {
814
+ generateThreadNames: true,
815
+ });
816
+ const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
817
+
818
+ try {
819
+ const response = await handleRunAgent({
820
+ runtime,
821
+ request: new Request("https://example.com/agent/my-agent/run", {
822
+ method: "POST",
823
+ headers: {
824
+ "Content-Type": "application/json",
825
+ },
826
+ body: JSON.stringify({
827
+ threadId: "thread-1",
828
+ runId: "run-1",
829
+ state: {},
830
+ messages: [
831
+ {
832
+ id: "user-1",
833
+ role: "user",
834
+ content: "Please help me name this failed thread.",
835
+ },
836
+ ],
837
+ tools: [],
838
+ context: [],
839
+ forwardedProps: {},
840
+ }),
841
+ }),
842
+ agentId: "my-agent",
843
+ });
844
+
845
+ expect(response.status).toBe(200);
846
+ await vi.waitFor(() =>
847
+ expect(platform.updateThread).toHaveBeenCalledWith({
848
+ threadId: "thread-1",
849
+ userId: "user-1",
850
+ agentId: "my-agent",
851
+ updates: { name: "Untitled" },
852
+ }),
853
+ );
854
+ expect(namingAgent.runAgent).toHaveBeenCalledTimes(3);
855
+ expect(runtime.runner.run).toHaveBeenCalledTimes(1);
856
+ } finally {
857
+ errorSpy.mockRestore();
858
+ }
859
+ });
860
+
861
+ it("returns 400 when identifyUser returns an invalid id", async () => {
862
+ const agent = createAgentForIntelligence();
863
+ const platform = {
864
+ getOrCreateThread: vi.fn(),
865
+ getThreadMessages: vi.fn(),
866
+ ɵacquireThreadLock: vi.fn(),
867
+ };
868
+ const runtime = createIntelligenceRuntime(agent, platform as any, {
869
+ identifyUser: vi.fn().mockResolvedValue({ id: "" }),
870
+ });
871
+
872
+ const response = await handleRunAgent({
873
+ runtime,
874
+ request: createRunRequest(),
875
+ agentId: "my-agent",
876
+ });
877
+
878
+ expect(response.status).toBe(400);
879
+ expect(platform.getOrCreateThread).not.toHaveBeenCalled();
880
+ expect(platform.ɵacquireThreadLock).not.toHaveBeenCalled();
881
+ });
882
+
883
+ it("returns 500 when identifyUser throws", async () => {
884
+ const agent = createAgentForIntelligence();
885
+ const platform = {
886
+ getOrCreateThread: vi.fn(),
887
+ getThreadMessages: vi.fn(),
888
+ ɵacquireThreadLock: vi.fn(),
889
+ };
890
+ const runtime = createIntelligenceRuntime(agent, platform as any, {
891
+ identifyUser: vi.fn().mockRejectedValue(new Error("auth failed")),
892
+ });
893
+ const errorSpy = vi.spyOn(console, "error").mockImplementation(() => {});
894
+
895
+ try {
896
+ const response = await handleRunAgent({
897
+ runtime,
898
+ request: createRunRequest(),
899
+ agentId: "my-agent",
900
+ });
901
+
902
+ expect(response.status).toBe(500);
903
+ expect(platform.getOrCreateThread).not.toHaveBeenCalled();
904
+ expect(platform.ɵacquireThreadLock).not.toHaveBeenCalled();
905
+ } finally {
906
+ errorSpy.mockRestore();
907
+ }
908
+ });
909
+ });
910
+ });