@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.
- package/.eslintrc.js +4 -4
- package/CHANGELOG.md +117 -106
- package/dist/_virtual/_rolldown/runtime.mjs +25 -1
- package/dist/agent/index.cjs +643 -0
- package/dist/agent/index.cjs.map +1 -0
- package/dist/agent/index.d.cts +263 -0
- package/dist/agent/index.d.cts.map +1 -0
- package/dist/agent/index.d.mts +263 -0
- package/dist/agent/index.d.mts.map +1 -0
- package/dist/agent/index.mjs +635 -0
- package/dist/agent/index.mjs.map +1 -0
- package/dist/graphql/message-conversion/agui-to-gql.cjs.map +1 -1
- package/dist/graphql/message-conversion/agui-to-gql.mjs.map +1 -1
- package/dist/lib/integrations/nextjs/app-router.cjs +2 -2
- package/dist/lib/integrations/nextjs/app-router.cjs.map +1 -1
- package/dist/lib/integrations/nextjs/app-router.mjs +1 -1
- package/dist/lib/integrations/nextjs/app-router.mjs.map +1 -1
- package/dist/lib/integrations/node-http/index.cjs +2 -3
- package/dist/lib/integrations/node-http/index.cjs.map +1 -1
- package/dist/lib/integrations/node-http/index.mjs +1 -1
- package/dist/lib/integrations/node-http/index.mjs.map +1 -1
- package/dist/lib/runtime/agent-integrations/langgraph/agent.cjs +1 -1
- package/dist/lib/runtime/agent-integrations/langgraph/agent.d.cts +2 -2
- package/dist/lib/runtime/agent-integrations/langgraph/agent.d.cts.map +1 -1
- package/dist/lib/runtime/agent-integrations/langgraph/agent.d.mts +3 -3
- package/dist/lib/runtime/agent-integrations/langgraph/agent.d.mts.map +1 -1
- package/dist/lib/runtime/agent-integrations/langgraph/agent.mjs +1 -1
- package/dist/lib/runtime/copilot-runtime.cjs +7 -5
- package/dist/lib/runtime/copilot-runtime.cjs.map +1 -1
- package/dist/lib/runtime/copilot-runtime.d.cts +10 -8
- package/dist/lib/runtime/copilot-runtime.d.cts.map +1 -1
- package/dist/lib/runtime/copilot-runtime.d.mts +10 -8
- package/dist/lib/runtime/copilot-runtime.d.mts.map +1 -1
- package/dist/lib/runtime/copilot-runtime.mjs +7 -5
- package/dist/lib/runtime/copilot-runtime.mjs.map +1 -1
- package/dist/lib/runtime/telemetry-agent-runner.cjs +2 -2
- package/dist/lib/runtime/telemetry-agent-runner.cjs.map +1 -1
- package/dist/lib/runtime/telemetry-agent-runner.d.cts +2 -1
- package/dist/lib/runtime/telemetry-agent-runner.d.cts.map +1 -1
- package/dist/lib/runtime/telemetry-agent-runner.d.mts +2 -1
- package/dist/lib/runtime/telemetry-agent-runner.d.mts.map +1 -1
- package/dist/lib/runtime/telemetry-agent-runner.mjs +1 -1
- package/dist/lib/runtime/telemetry-agent-runner.mjs.map +1 -1
- package/dist/lib/telemetry-client.cjs +1 -1
- package/dist/lib/telemetry-client.mjs +1 -1
- package/dist/package.cjs +21 -4
- package/dist/package.mjs +21 -4
- package/dist/service-adapters/anthropic/anthropic-adapter.d.mts +1 -1
- package/dist/v2/index.cjs +41 -15
- package/dist/v2/index.d.cts +14 -2
- package/dist/v2/index.d.mts +14 -2
- package/dist/v2/index.mjs +13 -4
- package/dist/v2/runtime/endpoints/express-single.cjs +190 -0
- package/dist/v2/runtime/endpoints/express-single.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/express-single.d.cts +16 -0
- package/dist/v2/runtime/endpoints/express-single.d.cts.map +1 -0
- package/dist/v2/runtime/endpoints/express-single.d.mts +16 -0
- package/dist/v2/runtime/endpoints/express-single.d.mts.map +1 -0
- package/dist/v2/runtime/endpoints/express-single.mjs +187 -0
- package/dist/v2/runtime/endpoints/express-single.mjs.map +1 -0
- package/dist/v2/runtime/endpoints/express-utils.cjs +119 -0
- package/dist/v2/runtime/endpoints/express-utils.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/express-utils.mjs +117 -0
- package/dist/v2/runtime/endpoints/express-utils.mjs.map +1 -0
- package/dist/v2/runtime/endpoints/express.cjs +217 -0
- package/dist/v2/runtime/endpoints/express.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/express.d.cts +16 -0
- package/dist/v2/runtime/endpoints/express.d.cts.map +1 -0
- package/dist/v2/runtime/endpoints/express.d.mts +16 -0
- package/dist/v2/runtime/endpoints/express.d.mts.map +1 -0
- package/dist/v2/runtime/endpoints/express.mjs +214 -0
- package/dist/v2/runtime/endpoints/express.mjs.map +1 -0
- package/dist/v2/runtime/endpoints/hono-single.cjs +141 -0
- package/dist/v2/runtime/endpoints/hono-single.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/hono-single.d.cts +41 -0
- package/dist/v2/runtime/endpoints/hono-single.d.cts.map +1 -0
- package/dist/v2/runtime/endpoints/hono-single.d.mts +41 -0
- package/dist/v2/runtime/endpoints/hono-single.d.mts.map +1 -0
- package/dist/v2/runtime/endpoints/hono-single.mjs +140 -0
- package/dist/v2/runtime/endpoints/hono-single.mjs.map +1 -0
- package/dist/v2/runtime/endpoints/hono.cjs +248 -0
- package/dist/v2/runtime/endpoints/hono.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/hono.d.cts +164 -0
- package/dist/v2/runtime/endpoints/hono.d.cts.map +1 -0
- package/dist/v2/runtime/endpoints/hono.d.mts +164 -0
- package/dist/v2/runtime/endpoints/hono.d.mts.map +1 -0
- package/dist/v2/runtime/endpoints/hono.mjs +247 -0
- package/dist/v2/runtime/endpoints/hono.mjs.map +1 -0
- package/dist/v2/runtime/endpoints/index.d.cts +5 -0
- package/dist/v2/runtime/endpoints/index.d.mts +5 -0
- package/dist/v2/runtime/endpoints/single-route-helpers.cjs +68 -0
- package/dist/v2/runtime/endpoints/single-route-helpers.cjs.map +1 -0
- package/dist/v2/runtime/endpoints/single-route-helpers.mjs +65 -0
- package/dist/v2/runtime/endpoints/single-route-helpers.mjs.map +1 -0
- package/dist/v2/runtime/handlers/get-runtime-info.cjs +51 -0
- package/dist/v2/runtime/handlers/get-runtime-info.cjs.map +1 -0
- package/dist/v2/runtime/handlers/get-runtime-info.mjs +51 -0
- package/dist/v2/runtime/handlers/get-runtime-info.mjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-connect.cjs +49 -0
- package/dist/v2/runtime/handlers/handle-connect.cjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-connect.mjs +49 -0
- package/dist/v2/runtime/handlers/handle-connect.mjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-run.cjs +61 -0
- package/dist/v2/runtime/handlers/handle-run.cjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-run.mjs +61 -0
- package/dist/v2/runtime/handlers/handle-run.mjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-stop.cjs +47 -0
- package/dist/v2/runtime/handlers/handle-stop.cjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-stop.mjs +46 -0
- package/dist/v2/runtime/handlers/handle-stop.mjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-transcribe.cjs +112 -0
- package/dist/v2/runtime/handlers/handle-transcribe.cjs.map +1 -0
- package/dist/v2/runtime/handlers/handle-transcribe.mjs +111 -0
- package/dist/v2/runtime/handlers/handle-transcribe.mjs.map +1 -0
- package/dist/v2/runtime/handlers/header-utils.cjs +26 -0
- package/dist/v2/runtime/handlers/header-utils.cjs.map +1 -0
- package/dist/v2/runtime/handlers/header-utils.mjs +25 -0
- package/dist/v2/runtime/handlers/header-utils.mjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/connect.cjs +37 -0
- package/dist/v2/runtime/handlers/intelligence/connect.cjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/connect.mjs +37 -0
- package/dist/v2/runtime/handlers/intelligence/connect.mjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/run.cjs +89 -0
- package/dist/v2/runtime/handlers/intelligence/run.cjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/run.mjs +88 -0
- package/dist/v2/runtime/handlers/intelligence/run.mjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/thread-names.cjs +146 -0
- package/dist/v2/runtime/handlers/intelligence/thread-names.cjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/thread-names.mjs +145 -0
- package/dist/v2/runtime/handlers/intelligence/thread-names.mjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/threads.cjs +159 -0
- package/dist/v2/runtime/handlers/intelligence/threads.cjs.map +1 -0
- package/dist/v2/runtime/handlers/intelligence/threads.mjs +154 -0
- package/dist/v2/runtime/handlers/intelligence/threads.mjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/agent-utils.cjs +74 -0
- package/dist/v2/runtime/handlers/shared/agent-utils.cjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/agent-utils.mjs +70 -0
- package/dist/v2/runtime/handlers/shared/agent-utils.mjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/intelligence-utils.cjs +21 -0
- package/dist/v2/runtime/handlers/shared/intelligence-utils.cjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/intelligence-utils.mjs +20 -0
- package/dist/v2/runtime/handlers/shared/intelligence-utils.mjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/json-response.cjs +12 -0
- package/dist/v2/runtime/handlers/shared/json-response.cjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/json-response.mjs +10 -0
- package/dist/v2/runtime/handlers/shared/json-response.mjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.cjs +20 -0
- package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.cjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.mjs +20 -0
- package/dist/v2/runtime/handlers/shared/resolve-intelligence-user.mjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/sse-response.cjs +69 -0
- package/dist/v2/runtime/handlers/shared/sse-response.cjs.map +1 -0
- package/dist/v2/runtime/handlers/shared/sse-response.mjs +68 -0
- package/dist/v2/runtime/handlers/shared/sse-response.mjs.map +1 -0
- package/dist/v2/runtime/handlers/sse/connect.cjs +18 -0
- package/dist/v2/runtime/handlers/sse/connect.cjs.map +1 -0
- package/dist/v2/runtime/handlers/sse/connect.mjs +18 -0
- package/dist/v2/runtime/handlers/sse/connect.mjs.map +1 -0
- package/dist/v2/runtime/handlers/sse/run.cjs +18 -0
- package/dist/v2/runtime/handlers/sse/run.cjs.map +1 -0
- package/dist/v2/runtime/handlers/sse/run.mjs +18 -0
- package/dist/v2/runtime/handlers/sse/run.mjs.map +1 -0
- package/dist/v2/runtime/index.d.cts +13 -0
- package/dist/v2/runtime/index.d.mts +14 -0
- package/dist/v2/runtime/intelligence-platform/client.cjs +333 -0
- package/dist/v2/runtime/intelligence-platform/client.cjs.map +1 -0
- package/dist/v2/runtime/intelligence-platform/client.d.cts +336 -0
- package/dist/v2/runtime/intelligence-platform/client.d.cts.map +1 -0
- package/dist/v2/runtime/intelligence-platform/client.d.mts +336 -0
- package/dist/v2/runtime/intelligence-platform/client.d.mts.map +1 -0
- package/dist/v2/runtime/intelligence-platform/client.mjs +331 -0
- package/dist/v2/runtime/intelligence-platform/client.mjs.map +1 -0
- package/dist/v2/runtime/intelligence-platform/index.d.mts +2 -0
- package/dist/v2/runtime/middleware-sse-parser.cjs +138 -0
- package/dist/v2/runtime/middleware-sse-parser.cjs.map +1 -0
- package/dist/v2/runtime/middleware-sse-parser.d.cts +22 -0
- package/dist/v2/runtime/middleware-sse-parser.d.cts.map +1 -0
- package/dist/v2/runtime/middleware-sse-parser.d.mts +22 -0
- package/dist/v2/runtime/middleware-sse-parser.d.mts.map +1 -0
- package/dist/v2/runtime/middleware-sse-parser.mjs +137 -0
- package/dist/v2/runtime/middleware-sse-parser.mjs.map +1 -0
- package/dist/v2/runtime/middleware.cjs +35 -0
- package/dist/v2/runtime/middleware.cjs.map +1 -0
- package/dist/v2/runtime/middleware.d.cts +32 -0
- package/dist/v2/runtime/middleware.d.cts.map +1 -0
- package/dist/v2/runtime/middleware.d.mts +32 -0
- package/dist/v2/runtime/middleware.d.mts.map +1 -0
- package/dist/v2/runtime/middleware.mjs +33 -0
- package/dist/v2/runtime/middleware.mjs.map +1 -0
- package/dist/v2/runtime/runner/agent-runner.cjs +8 -0
- package/dist/v2/runtime/runner/agent-runner.cjs.map +1 -0
- package/dist/v2/runtime/runner/agent-runner.d.cts +32 -0
- package/dist/v2/runtime/runner/agent-runner.d.cts.map +1 -0
- package/dist/v2/runtime/runner/agent-runner.d.mts +32 -0
- package/dist/v2/runtime/runner/agent-runner.d.mts.map +1 -0
- package/dist/v2/runtime/runner/agent-runner.mjs +7 -0
- package/dist/v2/runtime/runner/agent-runner.mjs.map +1 -0
- package/dist/v2/runtime/runner/in-memory.cjs +223 -0
- package/dist/v2/runtime/runner/in-memory.cjs.map +1 -0
- package/dist/v2/runtime/runner/in-memory.d.cts +15 -0
- package/dist/v2/runtime/runner/in-memory.d.cts.map +1 -0
- package/dist/v2/runtime/runner/in-memory.d.mts +15 -0
- package/dist/v2/runtime/runner/in-memory.d.mts.map +1 -0
- package/dist/v2/runtime/runner/in-memory.mjs +222 -0
- package/dist/v2/runtime/runner/in-memory.mjs.map +1 -0
- package/dist/v2/runtime/runner/index.d.cts +6 -0
- package/dist/v2/runtime/runner/index.d.mts +6 -0
- package/dist/v2/runtime/runner/index.mjs +7 -0
- package/dist/v2/runtime/runner/intelligence.cjs +246 -0
- package/dist/v2/runtime/runner/intelligence.cjs.map +1 -0
- package/dist/v2/runtime/runner/intelligence.d.cts +57 -0
- package/dist/v2/runtime/runner/intelligence.d.cts.map +1 -0
- package/dist/v2/runtime/runner/intelligence.d.mts +57 -0
- package/dist/v2/runtime/runner/intelligence.d.mts.map +1 -0
- package/dist/v2/runtime/runner/intelligence.mjs +245 -0
- package/dist/v2/runtime/runner/intelligence.mjs.map +1 -0
- package/dist/v2/runtime/runtime.cjs +101 -0
- package/dist/v2/runtime/runtime.cjs.map +1 -0
- package/dist/v2/runtime/runtime.d.cts +132 -0
- package/dist/v2/runtime/runtime.d.cts.map +1 -0
- package/dist/v2/runtime/runtime.d.mts +133 -0
- package/dist/v2/runtime/runtime.d.mts.map +1 -0
- package/dist/v2/runtime/runtime.mjs +97 -0
- package/dist/v2/runtime/runtime.mjs.map +1 -0
- package/dist/v2/runtime/telemetry/scarf-client.cjs +32 -0
- package/dist/v2/runtime/telemetry/scarf-client.cjs.map +1 -0
- package/dist/v2/runtime/telemetry/scarf-client.mjs +32 -0
- package/dist/v2/runtime/telemetry/scarf-client.mjs.map +1 -0
- package/dist/v2/runtime/telemetry/telemetry-client.cjs +35 -0
- package/dist/v2/runtime/telemetry/telemetry-client.cjs.map +1 -0
- package/dist/v2/runtime/telemetry/telemetry-client.mjs +35 -0
- package/dist/v2/runtime/telemetry/telemetry-client.mjs.map +1 -0
- package/dist/v2/runtime/transcription-service/transcription-service.cjs +8 -0
- package/dist/v2/runtime/transcription-service/transcription-service.cjs.map +1 -0
- package/dist/v2/runtime/transcription-service/transcription-service.d.cts +15 -0
- package/dist/v2/runtime/transcription-service/transcription-service.d.cts.map +1 -0
- package/dist/v2/runtime/transcription-service/transcription-service.d.mts +15 -0
- package/dist/v2/runtime/transcription-service/transcription-service.d.mts.map +1 -0
- package/dist/v2/runtime/transcription-service/transcription-service.mjs +7 -0
- package/dist/v2/runtime/transcription-service/transcription-service.mjs.map +1 -0
- package/package.json +22 -5
- package/src/agent/__tests__/ai-sdk-v6-compat.test.ts +116 -0
- package/src/agent/__tests__/basic-agent.test.ts +1248 -0
- package/src/agent/__tests__/config-tools-execution.test.ts +516 -0
- package/src/agent/__tests__/mcp-clients.test.ts +260 -0
- package/src/agent/__tests__/property-overrides.test.ts +598 -0
- package/src/agent/__tests__/standard-schema-tools.test.ts +313 -0
- package/src/agent/__tests__/standard-schema-types.test.ts +158 -0
- package/src/agent/__tests__/state-tools.test.ts +436 -0
- package/src/agent/__tests__/test-helpers.ts +178 -0
- package/src/agent/__tests__/utils.test.ts +536 -0
- package/src/agent/__tests__/zod-regression.test.ts +350 -0
- package/src/agent/index.ts +1305 -0
- package/src/graphql/message-conversion/agui-to-gql.test.ts +1 -1
- package/src/graphql/message-conversion/agui-to-gql.ts +1 -1
- package/src/graphql/message-conversion/gql-to-agui.ts +1 -1
- package/src/graphql/message-conversion/roundtrip-conversion.test.ts +1 -1
- package/src/lib/integrations/nextjs/app-router.ts +2 -2
- package/src/lib/integrations/node-http/index.ts +2 -2
- package/src/lib/runtime/copilot-runtime.ts +3 -5
- package/src/lib/runtime/telemetry-agent-runner.ts +1 -1
- package/src/service-adapters/conversion.test.ts +1 -1
- package/src/service-adapters/conversion.ts +1 -28
- package/src/v2/index.ts +5 -2
- package/src/v2/runtime/__tests__/cors-credentials.test.ts +320 -0
- package/src/v2/runtime/__tests__/express-abort-signal.test.ts +25 -0
- package/src/v2/runtime/__tests__/express-body-order.test.ts +76 -0
- package/src/v2/runtime/__tests__/express-single-sse.test.ts +122 -0
- package/src/v2/runtime/__tests__/get-runtime-info.test.ts +141 -0
- package/src/v2/runtime/__tests__/handle-connect.test.ts +423 -0
- package/src/v2/runtime/__tests__/handle-run.test.ts +910 -0
- package/src/v2/runtime/__tests__/handle-threads.test.ts +388 -0
- package/src/v2/runtime/__tests__/handle-transcribe.test.ts +301 -0
- package/src/v2/runtime/__tests__/header-utils.test.ts +88 -0
- package/src/v2/runtime/__tests__/in-process-agent-runner-messages.test.ts +230 -0
- package/src/v2/runtime/__tests__/in-process-agent-runner.test.ts +1030 -0
- package/src/v2/runtime/__tests__/middleware-express.test.ts +206 -0
- package/src/v2/runtime/__tests__/middleware-single-express.test.ts +211 -0
- package/src/v2/runtime/__tests__/middleware-single.test.ts +225 -0
- package/src/v2/runtime/__tests__/middleware-sse-parser.test.ts +187 -0
- package/src/v2/runtime/__tests__/middleware.test.ts +251 -0
- package/src/v2/runtime/__tests__/routing-express.test.ts +174 -0
- package/src/v2/runtime/__tests__/routing-single-express.test.ts +168 -0
- package/src/v2/runtime/__tests__/routing-single.test.ts +193 -0
- package/src/v2/runtime/__tests__/routing.test.ts +257 -0
- package/src/v2/runtime/__tests__/runtime.test.ts +123 -0
- package/src/v2/runtime/__tests__/telemetry.test.ts +167 -0
- package/src/v2/runtime/__tests__/thread-names.test.ts +188 -0
- package/src/v2/runtime/endpoints/express-single.ts +231 -0
- package/src/v2/runtime/endpoints/express-utils.ts +182 -0
- package/src/v2/runtime/endpoints/express.ts +275 -0
- package/src/v2/runtime/endpoints/hono-single.ts +212 -0
- package/src/v2/runtime/endpoints/hono.ts +314 -0
- package/src/v2/runtime/endpoints/index.ts +4 -0
- package/src/v2/runtime/endpoints/single-route-helpers.ts +125 -0
- package/src/v2/runtime/express.ts +2 -0
- package/src/v2/runtime/handler.ts +3 -0
- package/src/v2/runtime/handlers/get-runtime-info.ts +79 -0
- package/src/v2/runtime/handlers/handle-connect.ts +76 -0
- package/src/v2/runtime/handlers/handle-run.ts +89 -0
- package/src/v2/runtime/handlers/handle-stop.ts +76 -0
- package/src/v2/runtime/handlers/handle-threads.ts +7 -0
- package/src/v2/runtime/handlers/handle-transcribe.ts +256 -0
- package/src/v2/runtime/handlers/header-utils.ts +24 -0
- package/src/v2/runtime/handlers/intelligence/connect.ts +65 -0
- package/src/v2/runtime/handlers/intelligence/run.ts +152 -0
- package/src/v2/runtime/handlers/intelligence/thread-names.ts +246 -0
- package/src/v2/runtime/handlers/intelligence/threads.ts +233 -0
- package/src/v2/runtime/handlers/shared/agent-utils.ts +136 -0
- package/src/v2/runtime/handlers/shared/intelligence-utils.ts +21 -0
- package/src/v2/runtime/handlers/shared/json-response.ts +6 -0
- package/src/v2/runtime/handlers/shared/resolve-intelligence-user.ts +25 -0
- package/src/v2/runtime/handlers/shared/sse-response.ts +100 -0
- package/src/v2/runtime/handlers/sse/connect.ts +24 -0
- package/src/v2/runtime/handlers/sse/run.ts +27 -0
- package/src/v2/runtime/index.ts +20 -0
- package/src/v2/runtime/intelligence-platform/__tests__/client.test.ts +605 -0
- package/src/v2/runtime/intelligence-platform/client.ts +659 -0
- package/src/v2/runtime/intelligence-platform/index.ts +10 -0
- package/src/v2/runtime/middleware-sse-parser.ts +200 -0
- package/src/v2/runtime/middleware.ts +115 -0
- package/src/v2/runtime/runner/__tests__/finalize-events.test.ts +109 -0
- package/src/v2/runtime/runner/__tests__/in-memory-runner.e2e.test.ts +775 -0
- package/src/v2/runtime/runner/__tests__/in-memory-runner.test.ts +363 -0
- package/src/v2/runtime/runner/__tests__/intelligence-runner.test.ts +981 -0
- package/src/v2/runtime/runner/agent-runner.ts +36 -0
- package/src/v2/runtime/runner/in-memory.ts +381 -0
- package/src/v2/runtime/runner/index.ts +4 -0
- package/src/v2/runtime/runner/intelligence.ts +429 -0
- package/src/v2/runtime/runtime.ts +260 -0
- package/src/v2/runtime/telemetry/events.ts +35 -0
- package/src/v2/runtime/telemetry/index.ts +7 -0
- package/src/v2/runtime/telemetry/scarf-client.ts +39 -0
- package/src/v2/runtime/telemetry/telemetry-client.ts +70 -0
- package/src/v2/runtime/transcription-service/transcription-service.ts +11 -0
- package/tsconfig.json +9 -2
- package/tsdown.config.ts +1 -0
|
@@ -0,0 +1,1030 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach } from "vitest";
|
|
2
|
+
import { InMemoryAgentRunner } from "../runner/in-memory";
|
|
3
|
+
import {
|
|
4
|
+
AbstractAgent,
|
|
5
|
+
BaseEvent,
|
|
6
|
+
EventType,
|
|
7
|
+
RunAgentInput,
|
|
8
|
+
} from "@ag-ui/client";
|
|
9
|
+
import { firstValueFrom } from "rxjs";
|
|
10
|
+
import { toArray } from "rxjs/operators";
|
|
11
|
+
|
|
12
|
+
const stripTerminalEvents = (events: BaseEvent[]) =>
|
|
13
|
+
events.filter(
|
|
14
|
+
(event) =>
|
|
15
|
+
event.type !== EventType.RUN_FINISHED &&
|
|
16
|
+
event.type !== EventType.RUN_ERROR,
|
|
17
|
+
);
|
|
18
|
+
|
|
19
|
+
// Mock agent implementations for testing
|
|
20
|
+
class MockAgent extends AbstractAgent {
|
|
21
|
+
private events: BaseEvent[];
|
|
22
|
+
private delay: number;
|
|
23
|
+
|
|
24
|
+
constructor(events: BaseEvent[] = [], delay: number = 0) {
|
|
25
|
+
super();
|
|
26
|
+
this.events = events;
|
|
27
|
+
this.delay = delay;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async runAgent(
|
|
31
|
+
input: RunAgentInput,
|
|
32
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
33
|
+
): Promise<void> {
|
|
34
|
+
for (const event of this.events) {
|
|
35
|
+
if (this.delay > 0) {
|
|
36
|
+
await new Promise((resolve) => setTimeout(resolve, this.delay));
|
|
37
|
+
}
|
|
38
|
+
options.onEvent({ event });
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
clone(): AbstractAgent {
|
|
43
|
+
return new MockAgent(this.events, this.delay);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
class DelayedEventAgent extends AbstractAgent {
|
|
48
|
+
private eventCount: number;
|
|
49
|
+
private eventDelay: number;
|
|
50
|
+
private prefix: string;
|
|
51
|
+
|
|
52
|
+
constructor(
|
|
53
|
+
eventCount: number = 5,
|
|
54
|
+
eventDelay: number = 10,
|
|
55
|
+
prefix: string = "delayed",
|
|
56
|
+
) {
|
|
57
|
+
super();
|
|
58
|
+
this.eventCount = eventCount;
|
|
59
|
+
this.eventDelay = eventDelay;
|
|
60
|
+
this.prefix = prefix;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
async runAgent(
|
|
64
|
+
input: RunAgentInput,
|
|
65
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
66
|
+
): Promise<void> {
|
|
67
|
+
for (let i = 0; i < this.eventCount; i++) {
|
|
68
|
+
await new Promise((resolve) => setTimeout(resolve, this.eventDelay));
|
|
69
|
+
options.onEvent({
|
|
70
|
+
event: {
|
|
71
|
+
type: "message",
|
|
72
|
+
id: `${this.prefix}-${i}`,
|
|
73
|
+
timestamp: new Date().toISOString(),
|
|
74
|
+
data: { index: i, prefix: this.prefix },
|
|
75
|
+
} as BaseEvent,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
clone(): AbstractAgent {
|
|
81
|
+
return new DelayedEventAgent(this.eventCount, this.eventDelay, this.prefix);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
class ErrorThrowingAgent extends AbstractAgent {
|
|
86
|
+
private throwAfterEvents: number;
|
|
87
|
+
private errorMessage: string;
|
|
88
|
+
|
|
89
|
+
constructor(
|
|
90
|
+
throwAfterEvents: number = 2,
|
|
91
|
+
errorMessage: string = "Test error",
|
|
92
|
+
) {
|
|
93
|
+
super();
|
|
94
|
+
this.throwAfterEvents = throwAfterEvents;
|
|
95
|
+
this.errorMessage = errorMessage;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
async runAgent(
|
|
99
|
+
input: RunAgentInput,
|
|
100
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
101
|
+
): Promise<void> {
|
|
102
|
+
for (let i = 0; i < this.throwAfterEvents; i++) {
|
|
103
|
+
options.onEvent({
|
|
104
|
+
event: {
|
|
105
|
+
type: "message",
|
|
106
|
+
id: `error-agent-${i}`,
|
|
107
|
+
timestamp: new Date().toISOString(),
|
|
108
|
+
data: { index: i },
|
|
109
|
+
} as BaseEvent,
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
throw new Error(this.errorMessage);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
clone(): AbstractAgent {
|
|
116
|
+
return new ErrorThrowingAgent(this.throwAfterEvents, this.errorMessage);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
class StoppableAgent extends AbstractAgent {
|
|
121
|
+
private shouldStop = false;
|
|
122
|
+
private eventDelay: number;
|
|
123
|
+
|
|
124
|
+
constructor(eventDelay = 5) {
|
|
125
|
+
super();
|
|
126
|
+
this.eventDelay = eventDelay;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
async runAgent(
|
|
130
|
+
input: RunAgentInput,
|
|
131
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
132
|
+
): Promise<void> {
|
|
133
|
+
this.shouldStop = false;
|
|
134
|
+
let counter = 0;
|
|
135
|
+
|
|
136
|
+
while (!this.shouldStop && counter < 10_000) {
|
|
137
|
+
await new Promise((resolve) => setTimeout(resolve, this.eventDelay));
|
|
138
|
+
const event: BaseEvent = {
|
|
139
|
+
type: "message",
|
|
140
|
+
id: `stoppable-${counter}`,
|
|
141
|
+
timestamp: new Date().toISOString(),
|
|
142
|
+
data: { counter },
|
|
143
|
+
} as BaseEvent;
|
|
144
|
+
options.onEvent({ event });
|
|
145
|
+
counter += 1;
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
abortRun(): void {
|
|
150
|
+
this.shouldStop = true;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
clone(): AbstractAgent {
|
|
154
|
+
return new StoppableAgent(this.eventDelay);
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
class OpenEventsAgent extends AbstractAgent {
|
|
159
|
+
private shouldStop = false;
|
|
160
|
+
|
|
161
|
+
async runAgent(
|
|
162
|
+
input: RunAgentInput,
|
|
163
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
164
|
+
): Promise<void> {
|
|
165
|
+
this.shouldStop = false;
|
|
166
|
+
const messageId = "open-message";
|
|
167
|
+
const toolCallId = "open-tool";
|
|
168
|
+
|
|
169
|
+
options.onEvent({
|
|
170
|
+
event: {
|
|
171
|
+
type: EventType.TEXT_MESSAGE_START,
|
|
172
|
+
messageId,
|
|
173
|
+
role: "assistant",
|
|
174
|
+
} as BaseEvent,
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
options.onEvent({
|
|
178
|
+
event: {
|
|
179
|
+
type: EventType.TEXT_MESSAGE_CONTENT,
|
|
180
|
+
messageId,
|
|
181
|
+
delta: "Partial content",
|
|
182
|
+
} as BaseEvent,
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
options.onEvent({
|
|
186
|
+
event: {
|
|
187
|
+
type: EventType.TOOL_CALL_START,
|
|
188
|
+
toolCallId,
|
|
189
|
+
toolCallName: "testTool",
|
|
190
|
+
parentMessageId: messageId,
|
|
191
|
+
} as BaseEvent,
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
while (!this.shouldStop) {
|
|
195
|
+
await new Promise((resolve) => setTimeout(resolve, 5));
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
abortRun(): void {
|
|
200
|
+
this.shouldStop = true;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
clone(): AbstractAgent {
|
|
204
|
+
return new OpenEventsAgent();
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
class MultiEventAgent extends AbstractAgent {
|
|
209
|
+
private runId: string;
|
|
210
|
+
|
|
211
|
+
constructor(runId: string) {
|
|
212
|
+
super();
|
|
213
|
+
this.runId = runId;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
async runAgent(
|
|
217
|
+
input: RunAgentInput,
|
|
218
|
+
options: { onEvent: (event: { event: BaseEvent }) => void },
|
|
219
|
+
): Promise<void> {
|
|
220
|
+
// Emit different types of events
|
|
221
|
+
const eventTypes = ["start", "message", "tool_call", "tool_result", "end"];
|
|
222
|
+
|
|
223
|
+
for (const eventType of eventTypes) {
|
|
224
|
+
options.onEvent({
|
|
225
|
+
event: {
|
|
226
|
+
type: eventType,
|
|
227
|
+
id: `${this.runId}-${eventType}`,
|
|
228
|
+
timestamp: new Date().toISOString(),
|
|
229
|
+
data: {
|
|
230
|
+
runId: this.runId,
|
|
231
|
+
eventType,
|
|
232
|
+
metadata: { source: "MultiEventAgent" },
|
|
233
|
+
},
|
|
234
|
+
} as BaseEvent,
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
clone(): AbstractAgent {
|
|
240
|
+
return new MultiEventAgent(this.runId);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
describe("InMemoryAgentRunner", () => {
|
|
245
|
+
let runner: InMemoryAgentRunner;
|
|
246
|
+
|
|
247
|
+
beforeEach(() => {
|
|
248
|
+
runner = new InMemoryAgentRunner();
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
describe("Basic Functionality", () => {
|
|
252
|
+
it("should run a single agent and collect all events", async () => {
|
|
253
|
+
const threadId = "test-thread-1";
|
|
254
|
+
const events: BaseEvent[] = [
|
|
255
|
+
{
|
|
256
|
+
type: "start",
|
|
257
|
+
id: "1",
|
|
258
|
+
timestamp: new Date().toISOString(),
|
|
259
|
+
data: {},
|
|
260
|
+
} as BaseEvent,
|
|
261
|
+
{
|
|
262
|
+
type: "message",
|
|
263
|
+
id: "2",
|
|
264
|
+
timestamp: new Date().toISOString(),
|
|
265
|
+
data: { text: "Hello" },
|
|
266
|
+
} as BaseEvent,
|
|
267
|
+
{
|
|
268
|
+
type: "end",
|
|
269
|
+
id: "3",
|
|
270
|
+
timestamp: new Date().toISOString(),
|
|
271
|
+
data: {},
|
|
272
|
+
} as BaseEvent,
|
|
273
|
+
];
|
|
274
|
+
|
|
275
|
+
const agent = new MockAgent(events);
|
|
276
|
+
const input: RunAgentInput = {
|
|
277
|
+
messages: [],
|
|
278
|
+
state: {},
|
|
279
|
+
threadId,
|
|
280
|
+
runId: "run-1",
|
|
281
|
+
};
|
|
282
|
+
|
|
283
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
284
|
+
const collectedEvents = await firstValueFrom(
|
|
285
|
+
runObservable.pipe(toArray()),
|
|
286
|
+
);
|
|
287
|
+
|
|
288
|
+
const agentEvents = stripTerminalEvents(collectedEvents);
|
|
289
|
+
expect(agentEvents).toEqual(events);
|
|
290
|
+
});
|
|
291
|
+
|
|
292
|
+
it("should allow connecting after run completes and receive all past events", async () => {
|
|
293
|
+
const threadId = "test-thread-3";
|
|
294
|
+
const events: BaseEvent[] = [
|
|
295
|
+
{
|
|
296
|
+
type: "message",
|
|
297
|
+
id: "past-1",
|
|
298
|
+
timestamp: new Date().toISOString(),
|
|
299
|
+
data: {},
|
|
300
|
+
} as BaseEvent,
|
|
301
|
+
{
|
|
302
|
+
type: "message",
|
|
303
|
+
id: "past-2",
|
|
304
|
+
timestamp: new Date().toISOString(),
|
|
305
|
+
data: {},
|
|
306
|
+
} as BaseEvent,
|
|
307
|
+
];
|
|
308
|
+
|
|
309
|
+
const agent = new MockAgent(events);
|
|
310
|
+
const input: RunAgentInput = {
|
|
311
|
+
messages: [],
|
|
312
|
+
state: {},
|
|
313
|
+
threadId,
|
|
314
|
+
runId: "run-3",
|
|
315
|
+
};
|
|
316
|
+
|
|
317
|
+
// Run and wait for completion
|
|
318
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
319
|
+
await firstValueFrom(runObservable.pipe(toArray()));
|
|
320
|
+
|
|
321
|
+
// Connect after completion
|
|
322
|
+
const connectObservable = runner.connect({ threadId });
|
|
323
|
+
const collectedEvents = await firstValueFrom(
|
|
324
|
+
connectObservable.pipe(toArray()),
|
|
325
|
+
);
|
|
326
|
+
|
|
327
|
+
const storedAgentEvents = stripTerminalEvents(collectedEvents);
|
|
328
|
+
expect(storedAgentEvents).toEqual(events);
|
|
329
|
+
});
|
|
330
|
+
});
|
|
331
|
+
|
|
332
|
+
describe("Multiple Runs", () => {
|
|
333
|
+
it("should accumulate events from multiple sequential runs on same thread", async () => {
|
|
334
|
+
const threadId = "test-thread-multi-1";
|
|
335
|
+
|
|
336
|
+
// First run
|
|
337
|
+
const agent1 = new MultiEventAgent("run-1");
|
|
338
|
+
const input1: RunAgentInput = {
|
|
339
|
+
messages: [],
|
|
340
|
+
state: {},
|
|
341
|
+
threadId,
|
|
342
|
+
runId: "run-1",
|
|
343
|
+
};
|
|
344
|
+
|
|
345
|
+
const run1 = runner.run({ threadId, agent: agent1, input: input1 });
|
|
346
|
+
await firstValueFrom(run1.pipe(toArray()));
|
|
347
|
+
|
|
348
|
+
// Second run
|
|
349
|
+
const agent2 = new MultiEventAgent("run-2");
|
|
350
|
+
const input2: RunAgentInput = {
|
|
351
|
+
messages: [],
|
|
352
|
+
state: {},
|
|
353
|
+
threadId,
|
|
354
|
+
runId: "run-2",
|
|
355
|
+
};
|
|
356
|
+
|
|
357
|
+
const run2 = runner.run({ threadId, agent: agent2, input: input2 });
|
|
358
|
+
await firstValueFrom(run2.pipe(toArray()));
|
|
359
|
+
|
|
360
|
+
// Third run
|
|
361
|
+
const agent3 = new MultiEventAgent("run-3");
|
|
362
|
+
const input3: RunAgentInput = {
|
|
363
|
+
messages: [],
|
|
364
|
+
state: {},
|
|
365
|
+
threadId,
|
|
366
|
+
runId: "run-3",
|
|
367
|
+
};
|
|
368
|
+
|
|
369
|
+
const run3 = runner.run({ threadId, agent: agent3, input: input3 });
|
|
370
|
+
await firstValueFrom(run3.pipe(toArray()));
|
|
371
|
+
|
|
372
|
+
// Connect and verify all events
|
|
373
|
+
const connectObservable = runner.connect({ threadId });
|
|
374
|
+
const allEvents = await firstValueFrom(connectObservable.pipe(toArray()));
|
|
375
|
+
|
|
376
|
+
const agentEvents = stripTerminalEvents(allEvents);
|
|
377
|
+
expect(agentEvents).toHaveLength(15); // 5 events per run × 3 runs
|
|
378
|
+
|
|
379
|
+
// Verify events from all runs are present
|
|
380
|
+
const run1Events = agentEvents.filter((e) => e.id?.startsWith("run-1"));
|
|
381
|
+
const run2Events = agentEvents.filter((e) => e.id?.startsWith("run-2"));
|
|
382
|
+
const run3Events = agentEvents.filter((e) => e.id?.startsWith("run-3"));
|
|
383
|
+
|
|
384
|
+
expect(run1Events).toHaveLength(5);
|
|
385
|
+
expect(run2Events).toHaveLength(5);
|
|
386
|
+
expect(run3Events).toHaveLength(5);
|
|
387
|
+
|
|
388
|
+
// Verify order preservation
|
|
389
|
+
const runOrder = agentEvents.map(
|
|
390
|
+
(e) => e.id?.split("-")[0] + "-" + e.id?.split("-")[1],
|
|
391
|
+
);
|
|
392
|
+
expect(runOrder.slice(0, 5).every((id) => id?.startsWith("run-1"))).toBe(
|
|
393
|
+
true,
|
|
394
|
+
);
|
|
395
|
+
expect(runOrder.slice(5, 10).every((id) => id?.startsWith("run-2"))).toBe(
|
|
396
|
+
true,
|
|
397
|
+
);
|
|
398
|
+
expect(
|
|
399
|
+
runOrder.slice(10, 15).every((id) => id?.startsWith("run-3")),
|
|
400
|
+
).toBe(true);
|
|
401
|
+
});
|
|
402
|
+
|
|
403
|
+
it("should handle connect during multiple runs", async () => {
|
|
404
|
+
const threadId = "test-thread-multi-2";
|
|
405
|
+
|
|
406
|
+
// Start first run
|
|
407
|
+
const agent1 = new DelayedEventAgent(5, 20, "first");
|
|
408
|
+
const input1: RunAgentInput = {
|
|
409
|
+
messages: [],
|
|
410
|
+
state: {},
|
|
411
|
+
threadId,
|
|
412
|
+
runId: "run-1",
|
|
413
|
+
};
|
|
414
|
+
|
|
415
|
+
const run1Observable = runner.run({
|
|
416
|
+
threadId,
|
|
417
|
+
agent: agent1,
|
|
418
|
+
input: input1,
|
|
419
|
+
});
|
|
420
|
+
|
|
421
|
+
// Wait a bit to ensure first run is in progress
|
|
422
|
+
await new Promise((resolve) => setTimeout(resolve, 50));
|
|
423
|
+
|
|
424
|
+
// Connect during first run
|
|
425
|
+
const connectObservable = runner.connect({ threadId });
|
|
426
|
+
const eventCollector = firstValueFrom(connectObservable.pipe(toArray()));
|
|
427
|
+
|
|
428
|
+
// Wait for first run to complete
|
|
429
|
+
await firstValueFrom(run1Observable.pipe(toArray()));
|
|
430
|
+
|
|
431
|
+
// Collect all events from the connect during first run
|
|
432
|
+
const allEvents = await eventCollector;
|
|
433
|
+
|
|
434
|
+
// Connect only receives events from the first run since it completes
|
|
435
|
+
const firstRunAgentEvents = stripTerminalEvents(allEvents);
|
|
436
|
+
expect(firstRunAgentEvents).toHaveLength(5);
|
|
437
|
+
const firstRunEvents = firstRunAgentEvents.filter((e) =>
|
|
438
|
+
e.id?.startsWith("first"),
|
|
439
|
+
);
|
|
440
|
+
|
|
441
|
+
expect(firstRunEvents).toHaveLength(5);
|
|
442
|
+
|
|
443
|
+
// Start second run
|
|
444
|
+
const agent2 = new DelayedEventAgent(3, 10, "second");
|
|
445
|
+
const input2: RunAgentInput = {
|
|
446
|
+
messages: [],
|
|
447
|
+
state: {},
|
|
448
|
+
threadId,
|
|
449
|
+
runId: "run-2",
|
|
450
|
+
};
|
|
451
|
+
|
|
452
|
+
const run2Observable = runner.run({
|
|
453
|
+
threadId,
|
|
454
|
+
agent: agent2,
|
|
455
|
+
input: input2,
|
|
456
|
+
});
|
|
457
|
+
await firstValueFrom(run2Observable.pipe(toArray()));
|
|
458
|
+
|
|
459
|
+
// Connect after both runs to verify all events are accumulated
|
|
460
|
+
const allEventsAfter = await firstValueFrom(
|
|
461
|
+
runner.connect({ threadId }).pipe(toArray()),
|
|
462
|
+
);
|
|
463
|
+
expect(stripTerminalEvents(allEventsAfter)).toHaveLength(8); // 5 from first + 3 from second
|
|
464
|
+
});
|
|
465
|
+
|
|
466
|
+
it("should preserve event order across different agent types", async () => {
|
|
467
|
+
const threadId = "test-thread-multi-3";
|
|
468
|
+
|
|
469
|
+
// Run different types of agents
|
|
470
|
+
const agents = [
|
|
471
|
+
new MockAgent([
|
|
472
|
+
{
|
|
473
|
+
type: "mock",
|
|
474
|
+
id: "mock-1",
|
|
475
|
+
timestamp: new Date().toISOString(),
|
|
476
|
+
data: {},
|
|
477
|
+
} as BaseEvent,
|
|
478
|
+
{
|
|
479
|
+
type: "mock",
|
|
480
|
+
id: "mock-2",
|
|
481
|
+
timestamp: new Date().toISOString(),
|
|
482
|
+
data: {},
|
|
483
|
+
} as BaseEvent,
|
|
484
|
+
]),
|
|
485
|
+
new MultiEventAgent("multi"),
|
|
486
|
+
new DelayedEventAgent(2, 0, "delayed"),
|
|
487
|
+
];
|
|
488
|
+
|
|
489
|
+
for (let i = 0; i < agents.length; i++) {
|
|
490
|
+
const input: RunAgentInput = {
|
|
491
|
+
messages: [],
|
|
492
|
+
state: {},
|
|
493
|
+
threadId,
|
|
494
|
+
runId: `run-${i}`,
|
|
495
|
+
};
|
|
496
|
+
|
|
497
|
+
const run = runner.run({ threadId, agent: agents[i], input });
|
|
498
|
+
await firstValueFrom(run.pipe(toArray()));
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
// Verify all events are preserved
|
|
502
|
+
const connectObservable = runner.connect({ threadId });
|
|
503
|
+
const allEvents = await firstValueFrom(connectObservable.pipe(toArray()));
|
|
504
|
+
|
|
505
|
+
const agentEvents = stripTerminalEvents(allEvents);
|
|
506
|
+
expect(agentEvents).toHaveLength(9); // 2 + 5 + 2
|
|
507
|
+
|
|
508
|
+
// Verify event groups are in order
|
|
509
|
+
expect(agentEvents[0].id).toBe("mock-1");
|
|
510
|
+
expect(agentEvents[1].id).toBe("mock-2");
|
|
511
|
+
expect(agentEvents[2].id).toContain("multi");
|
|
512
|
+
expect(agentEvents[7].id).toContain("delayed");
|
|
513
|
+
});
|
|
514
|
+
});
|
|
515
|
+
|
|
516
|
+
describe("Concurrent Subscribers", () => {
|
|
517
|
+
it("should provide same events to multiple concurrent connect calls", async () => {
|
|
518
|
+
const threadId = "test-thread-concurrent-1";
|
|
519
|
+
const agent = new MultiEventAgent("concurrent");
|
|
520
|
+
const input: RunAgentInput = {
|
|
521
|
+
messages: [],
|
|
522
|
+
state: {},
|
|
523
|
+
threadId,
|
|
524
|
+
runId: "run-concurrent",
|
|
525
|
+
};
|
|
526
|
+
|
|
527
|
+
// Run agent
|
|
528
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
529
|
+
await firstValueFrom(runObservable.pipe(toArray()));
|
|
530
|
+
|
|
531
|
+
// Multiple concurrent connects
|
|
532
|
+
const connect1 = runner.connect({ threadId });
|
|
533
|
+
const connect2 = runner.connect({ threadId });
|
|
534
|
+
const connect3 = runner.connect({ threadId });
|
|
535
|
+
|
|
536
|
+
const [events1, events2, events3] = await Promise.all([
|
|
537
|
+
firstValueFrom(connect1.pipe(toArray())),
|
|
538
|
+
firstValueFrom(connect2.pipe(toArray())),
|
|
539
|
+
firstValueFrom(connect3.pipe(toArray())),
|
|
540
|
+
]);
|
|
541
|
+
|
|
542
|
+
// All should receive same events including RUN_FINISHED
|
|
543
|
+
const agentEvents1 = stripTerminalEvents(events1);
|
|
544
|
+
const agentEvents2 = stripTerminalEvents(events2);
|
|
545
|
+
const agentEvents3 = stripTerminalEvents(events3);
|
|
546
|
+
expect(agentEvents1).toHaveLength(5);
|
|
547
|
+
expect(agentEvents2).toHaveLength(5);
|
|
548
|
+
expect(agentEvents3).toHaveLength(5);
|
|
549
|
+
expect(events1).toEqual(events2);
|
|
550
|
+
expect(events2).toEqual(events3);
|
|
551
|
+
});
|
|
552
|
+
|
|
553
|
+
it("should handle late subscribers during active run", async () => {
|
|
554
|
+
const threadId = "test-thread-concurrent-2";
|
|
555
|
+
const agent = new DelayedEventAgent(10, 20, "late-sub");
|
|
556
|
+
const input: RunAgentInput = {
|
|
557
|
+
messages: [],
|
|
558
|
+
state: {},
|
|
559
|
+
threadId,
|
|
560
|
+
runId: "run-late",
|
|
561
|
+
};
|
|
562
|
+
|
|
563
|
+
// Start run
|
|
564
|
+
runner.run({ threadId, agent, input });
|
|
565
|
+
|
|
566
|
+
// Connect at different times during the run
|
|
567
|
+
await new Promise((resolve) => setTimeout(resolve, 50)); // After ~2 events
|
|
568
|
+
const connect1 = runner.connect({ threadId });
|
|
569
|
+
|
|
570
|
+
await new Promise((resolve) => setTimeout(resolve, 60)); // After ~5 events
|
|
571
|
+
const connect2 = runner.connect({ threadId });
|
|
572
|
+
|
|
573
|
+
await new Promise((resolve) => setTimeout(resolve, 80)); // After ~9 events
|
|
574
|
+
const connect3 = runner.connect({ threadId });
|
|
575
|
+
|
|
576
|
+
const [events1, events2, events3] = await Promise.all([
|
|
577
|
+
firstValueFrom(connect1.pipe(toArray())),
|
|
578
|
+
firstValueFrom(connect2.pipe(toArray())),
|
|
579
|
+
firstValueFrom(connect3.pipe(toArray())),
|
|
580
|
+
]);
|
|
581
|
+
|
|
582
|
+
// All subscribers should eventually receive all events plus RUN_FINISHED
|
|
583
|
+
const agentEvents1 = stripTerminalEvents(events1);
|
|
584
|
+
const agentEvents2 = stripTerminalEvents(events2);
|
|
585
|
+
const agentEvents3 = stripTerminalEvents(events3);
|
|
586
|
+
expect(agentEvents1).toHaveLength(10);
|
|
587
|
+
expect(agentEvents2).toHaveLength(10);
|
|
588
|
+
expect(agentEvents3).toHaveLength(10);
|
|
589
|
+
|
|
590
|
+
// Verify they all have the same events
|
|
591
|
+
expect(events1.map((e) => e.id)).toEqual(events2.map((e) => e.id));
|
|
592
|
+
expect(events2.map((e) => e.id)).toEqual(events3.map((e) => e.id));
|
|
593
|
+
});
|
|
594
|
+
});
|
|
595
|
+
|
|
596
|
+
describe("Error Handling", () => {
|
|
597
|
+
it("should throw error when thread is already running", async () => {
|
|
598
|
+
const threadId = "test-thread-error-1";
|
|
599
|
+
const agent = new DelayedEventAgent(5, 50, "blocking");
|
|
600
|
+
const input: RunAgentInput = {
|
|
601
|
+
messages: [],
|
|
602
|
+
state: {},
|
|
603
|
+
threadId,
|
|
604
|
+
runId: "run-1",
|
|
605
|
+
};
|
|
606
|
+
|
|
607
|
+
// Start first run
|
|
608
|
+
runner.run({ threadId, agent, input });
|
|
609
|
+
|
|
610
|
+
// Try to start another run on same thread immediately
|
|
611
|
+
expect(() => {
|
|
612
|
+
runner.run({ threadId, agent, input });
|
|
613
|
+
}).toThrow("Thread already running");
|
|
614
|
+
});
|
|
615
|
+
|
|
616
|
+
it("should handle agent errors gracefully", async () => {
|
|
617
|
+
const threadId = "test-thread-error-2";
|
|
618
|
+
const agent = new ErrorThrowingAgent(3, "Agent crashed!");
|
|
619
|
+
const input: RunAgentInput = {
|
|
620
|
+
messages: [],
|
|
621
|
+
state: {},
|
|
622
|
+
threadId,
|
|
623
|
+
runId: "run-error-1",
|
|
624
|
+
};
|
|
625
|
+
|
|
626
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
627
|
+
const events = await firstValueFrom(runObservable.pipe(toArray()));
|
|
628
|
+
|
|
629
|
+
// Should still receive events emitted before error
|
|
630
|
+
expect(events.at(-1)?.type).toBe(EventType.RUN_ERROR);
|
|
631
|
+
const preErrorEvents = events.slice(0, -1);
|
|
632
|
+
expect(preErrorEvents).toHaveLength(3);
|
|
633
|
+
expect(preErrorEvents.every((e) => e.id?.startsWith("error-agent"))).toBe(
|
|
634
|
+
true,
|
|
635
|
+
);
|
|
636
|
+
|
|
637
|
+
// Should be able to run again after error
|
|
638
|
+
const agent2 = new MockAgent([
|
|
639
|
+
{
|
|
640
|
+
type: "recovery",
|
|
641
|
+
id: "recovery-1",
|
|
642
|
+
timestamp: new Date().toISOString(),
|
|
643
|
+
data: {},
|
|
644
|
+
} as BaseEvent,
|
|
645
|
+
]);
|
|
646
|
+
|
|
647
|
+
const input2: RunAgentInput = {
|
|
648
|
+
messages: [],
|
|
649
|
+
state: {},
|
|
650
|
+
threadId,
|
|
651
|
+
runId: "run-error-2",
|
|
652
|
+
};
|
|
653
|
+
|
|
654
|
+
const run2 = runner.run({ threadId, agent: agent2, input: input2 });
|
|
655
|
+
const events2 = await firstValueFrom(run2.pipe(toArray()));
|
|
656
|
+
|
|
657
|
+
const recoveryEvents = stripTerminalEvents(events2);
|
|
658
|
+
expect(recoveryEvents).toHaveLength(1); // Only events from current run
|
|
659
|
+
expect(recoveryEvents[0].id).toBe("recovery-1");
|
|
660
|
+
|
|
661
|
+
// Connect should have all events including from errored run
|
|
662
|
+
const allEvents = await firstValueFrom(
|
|
663
|
+
runner.connect({ threadId }).pipe(toArray()),
|
|
664
|
+
);
|
|
665
|
+
expect(
|
|
666
|
+
allEvents.filter((event) => event.type === EventType.RUN_ERROR).length,
|
|
667
|
+
).toBeGreaterThanOrEqual(1);
|
|
668
|
+
const storedAgentEvents = stripTerminalEvents(allEvents);
|
|
669
|
+
expect(storedAgentEvents).toHaveLength(4); // 3 from error run + 1 from recovery
|
|
670
|
+
});
|
|
671
|
+
|
|
672
|
+
it("should properly set isRunning to false after agent error", async () => {
|
|
673
|
+
const threadId = "test-thread-error-3";
|
|
674
|
+
const agent = new ErrorThrowingAgent(1, "Quick fail");
|
|
675
|
+
const input: RunAgentInput = {
|
|
676
|
+
messages: [],
|
|
677
|
+
state: {},
|
|
678
|
+
threadId,
|
|
679
|
+
runId: "run-fail",
|
|
680
|
+
};
|
|
681
|
+
|
|
682
|
+
// Run and wait for completion (even with error)
|
|
683
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
684
|
+
await firstValueFrom(runObservable.pipe(toArray()));
|
|
685
|
+
|
|
686
|
+
// Verify thread is not running
|
|
687
|
+
const isRunning = await runner.isRunning({ threadId });
|
|
688
|
+
expect(isRunning).toBe(false);
|
|
689
|
+
|
|
690
|
+
// Should be able to run again
|
|
691
|
+
const agent2 = new MockAgent([
|
|
692
|
+
{
|
|
693
|
+
type: "test",
|
|
694
|
+
id: "after-error",
|
|
695
|
+
timestamp: new Date().toISOString(),
|
|
696
|
+
data: {},
|
|
697
|
+
} as BaseEvent,
|
|
698
|
+
]);
|
|
699
|
+
|
|
700
|
+
const input2: RunAgentInput = {
|
|
701
|
+
messages: [],
|
|
702
|
+
state: {},
|
|
703
|
+
threadId,
|
|
704
|
+
runId: "run-fail-2",
|
|
705
|
+
};
|
|
706
|
+
|
|
707
|
+
expect(() => {
|
|
708
|
+
runner.run({ threadId, agent: agent2, input: input2 });
|
|
709
|
+
}).not.toThrow();
|
|
710
|
+
});
|
|
711
|
+
});
|
|
712
|
+
|
|
713
|
+
describe("Edge Cases", () => {
|
|
714
|
+
it("should return EMPTY observable when connecting to non-existent thread", async () => {
|
|
715
|
+
const connectObservable = runner.connect({
|
|
716
|
+
threadId: "non-existent-thread",
|
|
717
|
+
});
|
|
718
|
+
|
|
719
|
+
// EMPTY completes immediately with no values
|
|
720
|
+
let completed = false;
|
|
721
|
+
let eventCount = 0;
|
|
722
|
+
|
|
723
|
+
await new Promise<void>((resolve) => {
|
|
724
|
+
connectObservable.subscribe({
|
|
725
|
+
next: () => eventCount++,
|
|
726
|
+
complete: () => {
|
|
727
|
+
completed = true;
|
|
728
|
+
resolve();
|
|
729
|
+
},
|
|
730
|
+
});
|
|
731
|
+
});
|
|
732
|
+
|
|
733
|
+
expect(completed).toBe(true);
|
|
734
|
+
expect(eventCount).toBe(0);
|
|
735
|
+
});
|
|
736
|
+
|
|
737
|
+
it("should handle very large number of events", async () => {
|
|
738
|
+
const threadId = "test-thread-large";
|
|
739
|
+
const eventCount = 1000;
|
|
740
|
+
const events: BaseEvent[] = [];
|
|
741
|
+
|
|
742
|
+
for (let i = 0; i < eventCount; i++) {
|
|
743
|
+
events.push({
|
|
744
|
+
type: "bulk",
|
|
745
|
+
id: `bulk-${i}`,
|
|
746
|
+
timestamp: new Date().toISOString(),
|
|
747
|
+
data: { index: i, payload: "x".repeat(100) },
|
|
748
|
+
} as BaseEvent);
|
|
749
|
+
}
|
|
750
|
+
|
|
751
|
+
const agent = new MockAgent(events);
|
|
752
|
+
const input: RunAgentInput = {
|
|
753
|
+
messages: [],
|
|
754
|
+
state: {},
|
|
755
|
+
threadId,
|
|
756
|
+
runId: "run-large",
|
|
757
|
+
};
|
|
758
|
+
|
|
759
|
+
// Run with large event set
|
|
760
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
761
|
+
await firstValueFrom(runObservable.pipe(toArray()));
|
|
762
|
+
|
|
763
|
+
// Connect and verify all events are preserved
|
|
764
|
+
const connectObservable = runner.connect({ threadId });
|
|
765
|
+
const collectedEvents = await firstValueFrom(
|
|
766
|
+
connectObservable.pipe(toArray()),
|
|
767
|
+
);
|
|
768
|
+
|
|
769
|
+
const bulkEvents = stripTerminalEvents(collectedEvents);
|
|
770
|
+
expect(bulkEvents).toHaveLength(eventCount);
|
|
771
|
+
expect(bulkEvents[0].id).toBe("bulk-0");
|
|
772
|
+
expect(bulkEvents[eventCount - 1].id).toBe(`bulk-${eventCount - 1}`);
|
|
773
|
+
});
|
|
774
|
+
|
|
775
|
+
it("should return false for isRunning on non-existent thread", async () => {
|
|
776
|
+
const isRunning = await runner.isRunning({ threadId: "non-existent" });
|
|
777
|
+
expect(isRunning).toBe(false);
|
|
778
|
+
});
|
|
779
|
+
|
|
780
|
+
it("should properly track isRunning state", async () => {
|
|
781
|
+
const threadId = "test-thread-running";
|
|
782
|
+
const agent = new DelayedEventAgent(5, 20, "running");
|
|
783
|
+
const input: RunAgentInput = {
|
|
784
|
+
messages: [],
|
|
785
|
+
state: {},
|
|
786
|
+
threadId,
|
|
787
|
+
runId: "run-running",
|
|
788
|
+
};
|
|
789
|
+
|
|
790
|
+
// Check before run
|
|
791
|
+
expect(await runner.isRunning({ threadId })).toBe(false);
|
|
792
|
+
|
|
793
|
+
// Start run
|
|
794
|
+
const runObservable = runner.run({ threadId, agent, input });
|
|
795
|
+
|
|
796
|
+
// Check during run
|
|
797
|
+
expect(await runner.isRunning({ threadId })).toBe(true);
|
|
798
|
+
|
|
799
|
+
// Wait for completion
|
|
800
|
+
await firstValueFrom(runObservable.pipe(toArray()));
|
|
801
|
+
|
|
802
|
+
// Check after run
|
|
803
|
+
expect(await runner.isRunning({ threadId })).toBe(false);
|
|
804
|
+
});
|
|
805
|
+
|
|
806
|
+
it("should return false when stopping a non-existent thread", async () => {
|
|
807
|
+
await expect(runner.stop({ threadId: "missing-thread" })).resolves.toBe(
|
|
808
|
+
false,
|
|
809
|
+
);
|
|
810
|
+
});
|
|
811
|
+
|
|
812
|
+
it("should stop an active run and complete streams", async () => {
|
|
813
|
+
const threadId = "test-thread-stop";
|
|
814
|
+
const agent = new StoppableAgent(2);
|
|
815
|
+
const input: RunAgentInput = {
|
|
816
|
+
messages: [],
|
|
817
|
+
state: {},
|
|
818
|
+
threadId,
|
|
819
|
+
runId: "run-stop",
|
|
820
|
+
};
|
|
821
|
+
|
|
822
|
+
const run$ = runner.run({ threadId, agent, input });
|
|
823
|
+
const collected = firstValueFrom(run$.pipe(toArray()));
|
|
824
|
+
|
|
825
|
+
// Allow the run loop to start and emit a couple of events
|
|
826
|
+
await new Promise((resolve) => setTimeout(resolve, 20));
|
|
827
|
+
expect(await runner.isRunning({ threadId })).toBe(true);
|
|
828
|
+
|
|
829
|
+
const stopped = await runner.stop({ threadId });
|
|
830
|
+
expect(stopped).toBe(true);
|
|
831
|
+
|
|
832
|
+
const events = await collected;
|
|
833
|
+
expect(events.length).toBeGreaterThan(0);
|
|
834
|
+
expect(events[events.length - 1].type).toBe(EventType.RUN_FINISHED);
|
|
835
|
+
expect(await runner.isRunning({ threadId })).toBe(false);
|
|
836
|
+
});
|
|
837
|
+
|
|
838
|
+
it("should close open text and tool events when stopping", async () => {
|
|
839
|
+
const threadId = "test-thread-open-events";
|
|
840
|
+
const agent = new OpenEventsAgent();
|
|
841
|
+
const input: RunAgentInput = {
|
|
842
|
+
messages: [],
|
|
843
|
+
state: {},
|
|
844
|
+
threadId,
|
|
845
|
+
runId: "run-open",
|
|
846
|
+
};
|
|
847
|
+
|
|
848
|
+
const run$ = runner.run({ threadId, agent, input });
|
|
849
|
+
const collected = firstValueFrom(run$.pipe(toArray()));
|
|
850
|
+
|
|
851
|
+
await new Promise((resolve) => setTimeout(resolve, 20));
|
|
852
|
+
await runner.stop({ threadId });
|
|
853
|
+
|
|
854
|
+
const events = await collected;
|
|
855
|
+
const endingTypes = events.slice(-4).map((event) => event.type);
|
|
856
|
+
expect(endingTypes).toEqual([
|
|
857
|
+
EventType.TEXT_MESSAGE_END,
|
|
858
|
+
EventType.TOOL_CALL_END,
|
|
859
|
+
EventType.TOOL_CALL_RESULT,
|
|
860
|
+
EventType.RUN_FINISHED,
|
|
861
|
+
]);
|
|
862
|
+
});
|
|
863
|
+
|
|
864
|
+
it("should handle thread isolation correctly", async () => {
|
|
865
|
+
const thread1 = "test-thread-iso-1";
|
|
866
|
+
const thread2 = "test-thread-iso-2";
|
|
867
|
+
|
|
868
|
+
const agent1 = new MockAgent([
|
|
869
|
+
{
|
|
870
|
+
type: "thread1",
|
|
871
|
+
id: "t1-event",
|
|
872
|
+
timestamp: new Date().toISOString(),
|
|
873
|
+
data: {},
|
|
874
|
+
} as BaseEvent,
|
|
875
|
+
]);
|
|
876
|
+
const agent2 = new MockAgent([
|
|
877
|
+
{
|
|
878
|
+
type: "thread2",
|
|
879
|
+
id: "t2-event",
|
|
880
|
+
timestamp: new Date().toISOString(),
|
|
881
|
+
data: {},
|
|
882
|
+
} as BaseEvent,
|
|
883
|
+
]);
|
|
884
|
+
|
|
885
|
+
// Run on different threads
|
|
886
|
+
const run1 = runner.run({
|
|
887
|
+
threadId: thread1,
|
|
888
|
+
agent: agent1,
|
|
889
|
+
input: { messages: [], state: {}, threadId: thread1, runId: "run-t1" },
|
|
890
|
+
});
|
|
891
|
+
const run2 = runner.run({
|
|
892
|
+
threadId: thread2,
|
|
893
|
+
agent: agent2,
|
|
894
|
+
input: { messages: [], state: {}, threadId: thread2, runId: "run-t2" },
|
|
895
|
+
});
|
|
896
|
+
|
|
897
|
+
await Promise.all([
|
|
898
|
+
firstValueFrom(run1.pipe(toArray())),
|
|
899
|
+
firstValueFrom(run2.pipe(toArray())),
|
|
900
|
+
]);
|
|
901
|
+
|
|
902
|
+
// Connect to each thread
|
|
903
|
+
const events1 = await firstValueFrom(
|
|
904
|
+
runner.connect({ threadId: thread1 }).pipe(toArray()),
|
|
905
|
+
);
|
|
906
|
+
const events2 = await firstValueFrom(
|
|
907
|
+
runner.connect({ threadId: thread2 }).pipe(toArray()),
|
|
908
|
+
);
|
|
909
|
+
|
|
910
|
+
// Verify isolation
|
|
911
|
+
const thread1Events = stripTerminalEvents(events1);
|
|
912
|
+
const thread2Events = stripTerminalEvents(events2);
|
|
913
|
+
expect(thread1Events).toHaveLength(1);
|
|
914
|
+
expect(thread1Events[0].id).toBe("t1-event");
|
|
915
|
+
|
|
916
|
+
expect(thread2Events).toHaveLength(1);
|
|
917
|
+
expect(thread2Events[0].id).toBe("t2-event");
|
|
918
|
+
});
|
|
919
|
+
});
|
|
920
|
+
|
|
921
|
+
describe("Complex Scenarios", () => {
|
|
922
|
+
it("should handle rapid sequential runs with mixed event patterns", async () => {
|
|
923
|
+
const threadId = "test-thread-complex-1";
|
|
924
|
+
const runs = [
|
|
925
|
+
{
|
|
926
|
+
agent: new MockAgent([
|
|
927
|
+
{
|
|
928
|
+
type: "instant",
|
|
929
|
+
id: "instant-1",
|
|
930
|
+
timestamp: new Date().toISOString(),
|
|
931
|
+
data: {},
|
|
932
|
+
} as BaseEvent,
|
|
933
|
+
]),
|
|
934
|
+
runId: "run-1",
|
|
935
|
+
},
|
|
936
|
+
{ agent: new DelayedEventAgent(3, 5, "delayed"), runId: "run-2" },
|
|
937
|
+
{
|
|
938
|
+
agent: new MockAgent([
|
|
939
|
+
{
|
|
940
|
+
type: "instant",
|
|
941
|
+
id: "instant-2",
|
|
942
|
+
timestamp: new Date().toISOString(),
|
|
943
|
+
data: {},
|
|
944
|
+
} as BaseEvent,
|
|
945
|
+
]),
|
|
946
|
+
runId: "run-3",
|
|
947
|
+
},
|
|
948
|
+
{ agent: new MultiEventAgent("multi"), runId: "run-4" },
|
|
949
|
+
{ agent: new DelayedEventAgent(2, 10, "slow"), runId: "run-5" },
|
|
950
|
+
];
|
|
951
|
+
|
|
952
|
+
for (const { agent, runId } of runs) {
|
|
953
|
+
const input: RunAgentInput = {
|
|
954
|
+
messages: [],
|
|
955
|
+
state: {},
|
|
956
|
+
threadId,
|
|
957
|
+
runId,
|
|
958
|
+
};
|
|
959
|
+
|
|
960
|
+
const run = runner.run({ threadId, agent, input });
|
|
961
|
+
await firstValueFrom(run.pipe(toArray()));
|
|
962
|
+
}
|
|
963
|
+
|
|
964
|
+
const allEvents = await firstValueFrom(
|
|
965
|
+
runner.connect({ threadId }).pipe(toArray()),
|
|
966
|
+
);
|
|
967
|
+
|
|
968
|
+
const agentEvents = stripTerminalEvents(allEvents);
|
|
969
|
+
expect(agentEvents).toHaveLength(12); // 1 + 3 + 1 + 5 + 2
|
|
970
|
+
|
|
971
|
+
// Verify event ordering
|
|
972
|
+
expect(agentEvents[0].id).toBe("instant-1");
|
|
973
|
+
expect(agentEvents[1].id).toContain("delayed-0");
|
|
974
|
+
expect(agentEvents[4].id).toBe("instant-2");
|
|
975
|
+
expect(agentEvents[5].id).toContain("multi-start");
|
|
976
|
+
expect(agentEvents[10].id).toContain("slow-0");
|
|
977
|
+
});
|
|
978
|
+
|
|
979
|
+
it("should handle subscriber that connects between runs", async () => {
|
|
980
|
+
const threadId = "test-thread-complex-2";
|
|
981
|
+
|
|
982
|
+
// First run
|
|
983
|
+
const agent1 = new MultiEventAgent("first");
|
|
984
|
+
const run1 = runner.run({
|
|
985
|
+
threadId,
|
|
986
|
+
agent: agent1,
|
|
987
|
+
input: { messages: [], state: {}, threadId, runId: "run-1" },
|
|
988
|
+
});
|
|
989
|
+
await firstValueFrom(run1.pipe(toArray()));
|
|
990
|
+
|
|
991
|
+
// Connect after first run - should only get first run events
|
|
992
|
+
const midConnectObservable = runner.connect({ threadId });
|
|
993
|
+
const midEvents = await firstValueFrom(
|
|
994
|
+
midConnectObservable.pipe(toArray()),
|
|
995
|
+
);
|
|
996
|
+
|
|
997
|
+
const midAgentEvents = stripTerminalEvents(midEvents);
|
|
998
|
+
expect(midAgentEvents).toHaveLength(5); // Only events from first run
|
|
999
|
+
const firstRunEvents = midAgentEvents.filter((e) =>
|
|
1000
|
+
e.id?.includes("first"),
|
|
1001
|
+
);
|
|
1002
|
+
expect(firstRunEvents).toHaveLength(5);
|
|
1003
|
+
|
|
1004
|
+
// Second run
|
|
1005
|
+
const agent2 = new MultiEventAgent("second");
|
|
1006
|
+
const run2 = runner.run({
|
|
1007
|
+
threadId,
|
|
1008
|
+
agent: agent2,
|
|
1009
|
+
input: { messages: [], state: {}, threadId, runId: "run-2" },
|
|
1010
|
+
});
|
|
1011
|
+
await firstValueFrom(run2.pipe(toArray()));
|
|
1012
|
+
|
|
1013
|
+
// Connect after both runs to verify all events
|
|
1014
|
+
const allEvents = await firstValueFrom(
|
|
1015
|
+
runner.connect({ threadId }).pipe(toArray()),
|
|
1016
|
+
);
|
|
1017
|
+
const allAgentEvents = stripTerminalEvents(allEvents);
|
|
1018
|
+
expect(allAgentEvents).toHaveLength(10); // Events from both runs
|
|
1019
|
+
|
|
1020
|
+
const allFirstRunEvents = allAgentEvents.filter((e) =>
|
|
1021
|
+
e.id?.includes("first"),
|
|
1022
|
+
);
|
|
1023
|
+
const allSecondRunEvents = allAgentEvents.filter((e) =>
|
|
1024
|
+
e.id?.includes("second"),
|
|
1025
|
+
);
|
|
1026
|
+
expect(allFirstRunEvents).toHaveLength(5);
|
|
1027
|
+
expect(allSecondRunEvents).toHaveLength(5);
|
|
1028
|
+
});
|
|
1029
|
+
});
|
|
1030
|
+
});
|