@cloudflare/tanstack-ai 0.1.7 → 0.1.8

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 (36) hide show
  1. package/README.md +29 -0
  2. package/dist/adapters/anthropic.cjs +2 -2
  3. package/dist/adapters/anthropic.mjs +1 -1
  4. package/dist/adapters/gemini.cjs +1 -1
  5. package/dist/adapters/grok.cjs +2 -2
  6. package/dist/adapters/grok.mjs +1 -1
  7. package/dist/adapters/openai.cjs +2 -2
  8. package/dist/adapters/openai.mjs +1 -1
  9. package/dist/adapters/openrouter.cjs +2 -2
  10. package/dist/adapters/openrouter.mjs +1 -1
  11. package/dist/adapters/workers-ai-image.cjs +2 -2
  12. package/dist/adapters/workers-ai-image.mjs +1 -1
  13. package/dist/adapters/workers-ai-summarize.cjs +2 -2
  14. package/dist/adapters/workers-ai-summarize.mjs +1 -1
  15. package/dist/adapters/workers-ai-transcription.cjs +2 -2
  16. package/dist/adapters/workers-ai-transcription.mjs +1 -1
  17. package/dist/adapters/workers-ai-tts.cjs +2 -2
  18. package/dist/adapters/workers-ai-tts.mjs +1 -1
  19. package/dist/adapters/workers-ai.cjs +1 -1
  20. package/dist/adapters/workers-ai.d.cts +31 -4
  21. package/dist/adapters/workers-ai.d.mts +31 -4
  22. package/dist/adapters/workers-ai.mjs +14 -3
  23. package/dist/adapters/workers-ai.mjs.map +1 -1
  24. package/dist/{create-fetcher-5iL34e6H.cjs → create-fetcher-By-hTiT9.cjs} +3 -1
  25. package/dist/create-fetcher-By-hTiT9.cjs.map +1 -0
  26. package/dist/{create-fetcher-DY7wfYYy.mjs → create-fetcher-CeUOJgrh.mjs} +3 -1
  27. package/dist/create-fetcher-CeUOJgrh.mjs.map +1 -0
  28. package/dist/index.cjs +1 -1
  29. package/dist/index.d.cts +2 -2
  30. package/dist/index.d.mts +2 -2
  31. package/dist/{workers-ai-Bm7Up4or.cjs → workers-ai-BOZ5iyhY.cjs} +15 -4
  32. package/dist/workers-ai-BOZ5iyhY.cjs.map +1 -0
  33. package/package.json +1 -1
  34. package/dist/create-fetcher-5iL34e6H.cjs.map +0 -1
  35. package/dist/create-fetcher-DY7wfYYy.mjs.map +0 -1
  36. package/dist/workers-ai-Bm7Up4or.cjs.map +0 -1
package/README.md CHANGED
@@ -68,6 +68,35 @@ const adapter = createWorkersAiChat("@cf/moonshotai/kimi-k2.5", {
68
68
  });
69
69
  ```
70
70
 
71
+ ### Reasoning Controls
72
+
73
+ Reasoning-capable Workers AI models (GLM-4.7-flash, Kimi K2.5/K2.6, GPT-OSS, QwQ) accept `reasoning_effort` and `chat_template_kwargs` on their inputs. Pass them per-call through `modelOptions`:
74
+
75
+ ```typescript
76
+ import { chat } from "@tanstack/ai";
77
+ import { createWorkersAiChat } from "@cloudflare/tanstack-ai";
78
+
79
+ const adapter = createWorkersAiChat("@cf/zai-org/glm-4.7-flash", {
80
+ binding: env.AI,
81
+ });
82
+
83
+ const response = chat({
84
+ adapter,
85
+ stream: true,
86
+ messages: [{ role: "user", content: "Summarize in one sentence." }],
87
+ modelOptions: {
88
+ // "low" | "medium" | "high" | null — null disables reasoning.
89
+ reasoning_effort: "low",
90
+ // Toggle thinking on models that expose template kwargs (GLM, Kimi).
91
+ chat_template_kwargs: { enable_thinking: false },
92
+ },
93
+ });
94
+ ```
95
+
96
+ This works in all four config modes (binding, REST, gateway binding, gateway REST). `modelOptions` keys with `undefined` values are stripped; `null` values are preserved.
97
+
98
+ See the [Workers AI docs](https://developers.cloudflare.com/workers-ai/) for per-model reasoning capabilities.
99
+
71
100
  ### Vision (Image Inputs)
72
101
 
73
102
  Send images to vision-capable chat models:
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  let _tanstack_ai_anthropic = require("@tanstack/ai-anthropic");
5
5
  //#region src/adapters/anthropic.ts
6
6
  function buildAnthropicConfig(config) {
@@ -1,4 +1,4 @@
1
- import { t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { ANTHROPIC_MODELS, AnthropicSummarizeAdapter, AnthropicTextAdapter } from "@tanstack/ai-anthropic";
3
3
  //#region src/adapters/anthropic.ts
4
4
  function buildAnthropicConfig(config) {
@@ -1,5 +1,5 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
3
  let _tanstack_ai_gemini = require("@tanstack/ai-gemini");
4
4
  //#region src/adapters/gemini.ts
5
5
  /**
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  let _tanstack_ai_grok = require("@tanstack/ai-grok");
5
5
  //#region src/adapters/grok.ts
6
6
  /**
@@ -1,4 +1,4 @@
1
- import { t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { GROK_CHAT_MODELS, GROK_IMAGE_MODELS, GrokImageAdapter, GrokSummarizeAdapter, GrokTextAdapter } from "@tanstack/ai-grok";
3
3
  //#region src/adapters/grok.ts
4
4
  /**
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  let _tanstack_ai_openai = require("@tanstack/ai-openai");
5
5
  //#region src/adapters/openai.ts
6
6
  /**
@@ -1,4 +1,4 @@
1
- import { t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { OPENAI_CHAT_MODELS, OPENAI_IMAGE_MODELS, OPENAI_TRANSCRIPTION_MODELS, OPENAI_TTS_MODELS, OPENAI_VIDEO_MODELS, OpenAIImageAdapter, OpenAISummarizeAdapter, OpenAITTSAdapter, OpenAITextAdapter, OpenAITranscriptionAdapter, OpenAIVideoAdapter } from "@tanstack/ai-openai";
3
3
  //#region src/adapters/openai.ts
4
4
  /**
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  let _tanstack_ai_openrouter = require("@tanstack/ai-openrouter");
5
5
  let _openrouter_sdk = require("@openrouter/sdk");
6
6
  //#region src/adapters/openrouter.ts
@@ -1,4 +1,4 @@
1
- import { t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { OpenRouterImageAdapter, OpenRouterSummarizeAdapter, OpenRouterTextAdapter } from "@tanstack/ai-openrouter";
3
3
  import { HTTPClient } from "@openrouter/sdk";
4
4
  //#region src/adapters/openrouter.ts
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  const require_defineProperty = require("../defineProperty-DQoAg20E.cjs");
5
5
  const require_workers_ai_rest = require("../workers-ai-rest-CkNCtBwv.cjs");
6
6
  const require_binary = require("../binary-C9FAYwZj.cjs");
@@ -1,4 +1,4 @@
1
- import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { t as _defineProperty } from "../defineProperty-CbyrzcbA.mjs";
3
3
  import { t as workersAiRestFetch } from "../workers-ai-rest-GKy2r7eG.mjs";
4
4
  import { n as uint8ArrayToBase64, t as binaryToBase64 } from "../binary-p4H_N_3M.mjs";
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  const require_defineProperty = require("../defineProperty-DQoAg20E.cjs");
5
5
  const require_workers_ai_rest = require("../workers-ai-rest-CkNCtBwv.cjs");
6
6
  let _tanstack_ai_adapters = require("@tanstack/ai/adapters");
@@ -1,4 +1,4 @@
1
- import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { t as _defineProperty } from "../defineProperty-CbyrzcbA.mjs";
3
3
  import { t as workersAiRestFetch } from "../workers-ai-rest-GKy2r7eG.mjs";
4
4
  import { BaseSummarizeAdapter } from "@tanstack/ai/adapters";
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  const require_defineProperty = require("../defineProperty-DQoAg20E.cjs");
5
5
  const require_workers_ai_rest = require("../workers-ai-rest-CkNCtBwv.cjs");
6
6
  const require_binary = require("../binary-C9FAYwZj.cjs");
@@ -1,4 +1,4 @@
1
- import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { t as _defineProperty } from "../defineProperty-CbyrzcbA.mjs";
3
3
  import { n as workersAiRestFetchBinary, t as workersAiRestFetch } from "../workers-ai-rest-GKy2r7eG.mjs";
4
4
  import { n as uint8ArrayToBase64 } from "../binary-p4H_N_3M.mjs";
@@ -1,6 +1,6 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- require("../workers-ai-Bm7Up4or.cjs");
3
- const require_create_fetcher = require("../create-fetcher-5iL34e6H.cjs");
2
+ require("../workers-ai-BOZ5iyhY.cjs");
3
+ const require_create_fetcher = require("../create-fetcher-By-hTiT9.cjs");
4
4
  const require_defineProperty = require("../defineProperty-DQoAg20E.cjs");
5
5
  const require_workers_ai_rest = require("../workers-ai-rest-CkNCtBwv.cjs");
6
6
  const require_binary = require("../binary-C9FAYwZj.cjs");
@@ -1,4 +1,4 @@
1
- import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { t as _defineProperty } from "../defineProperty-CbyrzcbA.mjs";
3
3
  import { t as workersAiRestFetch } from "../workers-ai-rest-GKy2r7eG.mjs";
4
4
  import { n as uint8ArrayToBase64, t as binaryToBase64 } from "../binary-p4H_N_3M.mjs";
@@ -1,4 +1,4 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- const require_workers_ai = require("../workers-ai-Bm7Up4or.cjs");
2
+ const require_workers_ai = require("../workers-ai-BOZ5iyhY.cjs");
3
3
  exports.WorkersAiTextAdapter = require_workers_ai.WorkersAiTextAdapter;
4
4
  exports.createWorkersAiChat = require_workers_ai.createWorkersAiChat;
@@ -5,14 +5,41 @@ import { AiModels, BaseAiTextGeneration } from "@cloudflare/workers-types";
5
5
 
6
6
  //#region src/adapters/workers-ai.d.ts
7
7
  type WorkersAiTextModel = { [K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never }[keyof AiModels] | (string & {});
8
- declare class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<TModel, any, any, any> {
8
+ interface WorkersAiTextModelOptions {
9
+ /**
10
+ * Controls the reasoning budget for reasoning-capable models
11
+ * (e.g. `@cf/zai-org/glm-4.7-flash`, `@cf/moonshotai/kimi-k2.5`,
12
+ * `@cf/openai/gpt-oss-120b`).
13
+ *
14
+ * `null` is a valid value and disables reasoning for models that support it.
15
+ */
16
+ reasoning_effort?: "low" | "medium" | "high" | null;
17
+ /**
18
+ * Chat-template overrides for reasoning-capable models that expose
19
+ * thinking toggles (e.g. GLM, Kimi).
20
+ */
21
+ chat_template_kwargs?: {
22
+ /** Whether to enable reasoning. Enabled by default on reasoning models. */enable_thinking?: boolean; /** If false, preserves reasoning context between turns. */
23
+ clear_thinking?: boolean;
24
+ };
25
+ /**
26
+ * Escape hatch for other Workers AI inputs-level parameters.
27
+ *
28
+ * Keys placed here are merged into the outbound request body and forwarded
29
+ * to the underlying transport (binding / REST / gateway). Only fields that
30
+ * the binding shim knows about are extracted for direct `env.AI` bindings;
31
+ * everything is passed through on REST and gateway paths.
32
+ */
33
+ [key: string]: unknown;
34
+ }
35
+ declare class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<TModel, WorkersAiTextModelOptions, any, any> {
9
36
  name: "workers-ai";
10
37
  private client;
11
38
  constructor(model: TModel, config: WorkersAiAdapterConfig);
12
- chatStream(options: TextOptions<any>): AsyncIterable<StreamChunk>;
13
- structuredOutput(options: StructuredOutputOptions<any>): Promise<StructuredOutputResult<unknown>>;
39
+ chatStream(options: TextOptions<WorkersAiTextModelOptions>): AsyncIterable<StreamChunk>;
40
+ structuredOutput(options: StructuredOutputOptions<WorkersAiTextModelOptions>): Promise<StructuredOutputResult<unknown>>;
14
41
  }
15
42
  declare function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig): WorkersAiTextAdapter<WorkersAiTextModel>;
16
43
  //#endregion
17
- export { WorkersAiTextAdapter, WorkersAiTextModel, createWorkersAiChat };
44
+ export { WorkersAiTextAdapter, WorkersAiTextModel, WorkersAiTextModelOptions, createWorkersAiChat };
18
45
  //# sourceMappingURL=workers-ai.d.cts.map
@@ -5,14 +5,41 @@ import { AiModels, BaseAiTextGeneration } from "@cloudflare/workers-types";
5
5
 
6
6
  //#region src/adapters/workers-ai.d.ts
7
7
  type WorkersAiTextModel = { [K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never }[keyof AiModels] | (string & {});
8
- declare class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<TModel, any, any, any> {
8
+ interface WorkersAiTextModelOptions {
9
+ /**
10
+ * Controls the reasoning budget for reasoning-capable models
11
+ * (e.g. `@cf/zai-org/glm-4.7-flash`, `@cf/moonshotai/kimi-k2.5`,
12
+ * `@cf/openai/gpt-oss-120b`).
13
+ *
14
+ * `null` is a valid value and disables reasoning for models that support it.
15
+ */
16
+ reasoning_effort?: "low" | "medium" | "high" | null;
17
+ /**
18
+ * Chat-template overrides for reasoning-capable models that expose
19
+ * thinking toggles (e.g. GLM, Kimi).
20
+ */
21
+ chat_template_kwargs?: {
22
+ /** Whether to enable reasoning. Enabled by default on reasoning models. */enable_thinking?: boolean; /** If false, preserves reasoning context between turns. */
23
+ clear_thinking?: boolean;
24
+ };
25
+ /**
26
+ * Escape hatch for other Workers AI inputs-level parameters.
27
+ *
28
+ * Keys placed here are merged into the outbound request body and forwarded
29
+ * to the underlying transport (binding / REST / gateway). Only fields that
30
+ * the binding shim knows about are extracted for direct `env.AI` bindings;
31
+ * everything is passed through on REST and gateway paths.
32
+ */
33
+ [key: string]: unknown;
34
+ }
35
+ declare class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<TModel, WorkersAiTextModelOptions, any, any> {
9
36
  name: "workers-ai";
10
37
  private client;
11
38
  constructor(model: TModel, config: WorkersAiAdapterConfig);
12
- chatStream(options: TextOptions<any>): AsyncIterable<StreamChunk>;
13
- structuredOutput(options: StructuredOutputOptions<any>): Promise<StructuredOutputResult<unknown>>;
39
+ chatStream(options: TextOptions<WorkersAiTextModelOptions>): AsyncIterable<StreamChunk>;
40
+ structuredOutput(options: StructuredOutputOptions<WorkersAiTextModelOptions>): Promise<StructuredOutputResult<unknown>>;
14
41
  }
15
42
  declare function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig): WorkersAiTextAdapter<WorkersAiTextModel>;
16
43
  //#endregion
17
- export { WorkersAiTextAdapter, WorkersAiTextModel, createWorkersAiChat };
44
+ export { WorkersAiTextAdapter, WorkersAiTextModel, WorkersAiTextModelOptions, createWorkersAiChat };
18
45
  //# sourceMappingURL=workers-ai.d.mts.map
@@ -1,4 +1,4 @@
1
- import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, n as createWorkersAiBindingFetch, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-DY7wfYYy.mjs";
1
+ import { a as validateWorkersAiConfig, i as isDirectCredentialsConfig, n as createWorkersAiBindingFetch, r as isDirectBindingConfig, t as createGatewayFetch } from "../create-fetcher-CeUOJgrh.mjs";
2
2
  import { t as _defineProperty } from "../defineProperty-CbyrzcbA.mjs";
3
3
  import { BaseTextAdapter } from "@tanstack/ai/adapters";
4
4
  import OpenAI from "openai";
@@ -128,6 +128,12 @@ function buildOpenAITools(tools) {
128
128
  function generateId(prefix = "chatcmpl") {
129
129
  return `${prefix}-${crypto.randomUUID().replace(/-/g, "").slice(0, 16)}`;
130
130
  }
131
+ function normalizeModelOptions(modelOptions) {
132
+ if (modelOptions === null || typeof modelOptions !== "object" || Array.isArray(modelOptions)) return {};
133
+ const out = {};
134
+ for (const [key, value] of Object.entries(modelOptions)) if (value !== void 0) out[key] = value;
135
+ return out;
136
+ }
131
137
  var WorkersAiTextAdapter = class extends BaseTextAdapter {
132
138
  constructor(model, config) {
133
139
  super({ apiKey: "unused" }, model);
@@ -136,7 +142,8 @@ var WorkersAiTextAdapter = class extends BaseTextAdapter {
136
142
  this.client = buildWorkersAiClient(config);
137
143
  }
138
144
  async *chatStream(options) {
139
- const { systemPrompts, messages, tools, temperature, maxTokens, model } = options;
145
+ const { systemPrompts, messages, tools, temperature, maxTokens, model, modelOptions } = options;
146
+ const extraBody = normalizeModelOptions(modelOptions);
140
147
  const openAIMessages = buildOpenAIMessages(systemPrompts, messages);
141
148
  const openAITools = buildOpenAITools(tools);
142
149
  const timestamp = Date.now();
@@ -154,6 +161,7 @@ var WorkersAiTextAdapter = class extends BaseTextAdapter {
154
161
  let stream;
155
162
  try {
156
163
  stream = await this.client.chat.completions.create({
164
+ ...extraBody,
157
165
  model: model ?? this.model,
158
166
  messages: openAIMessages,
159
167
  tools: openAITools,
@@ -165,6 +173,7 @@ var WorkersAiTextAdapter = class extends BaseTextAdapter {
165
173
  } catch (streamError) {
166
174
  console.warn("[tanstack-ai] Streaming failed, falling back to non-streaming:", streamError instanceof Error ? streamError.message : streamError);
167
175
  const nonStreamResult = await this.client.chat.completions.create({
176
+ ...extraBody,
168
177
  model: model ?? this.model,
169
178
  messages: openAIMessages,
170
179
  tools: openAITools,
@@ -423,9 +432,11 @@ var WorkersAiTextAdapter = class extends BaseTextAdapter {
423
432
  }
424
433
  async structuredOutput(options) {
425
434
  const { outputSchema, chatOptions } = options;
426
- const { systemPrompts, messages, temperature, model } = chatOptions;
435
+ const { systemPrompts, messages, temperature, model, modelOptions } = chatOptions;
436
+ const extraBody = normalizeModelOptions(modelOptions);
427
437
  const openAIMessages = buildOpenAIMessages(systemPrompts, messages, { includeToolMessages: false });
428
438
  const response = await this.client.chat.completions.create({
439
+ ...extraBody,
429
440
  model: model ?? this.model,
430
441
  messages: openAIMessages,
431
442
  temperature,
@@ -1 +1 @@
1
- {"version":3,"file":"workers-ai.mjs","names":[],"sources":["../../src/adapters/workers-ai.ts"],"sourcesContent":["import type { AiModels, BaseAiTextGeneration } from \"@cloudflare/workers-types\";\nimport type { ContentPart, ModelMessage, StreamChunk, TextOptions } from \"@tanstack/ai\";\nimport {\n\tBaseTextAdapter,\n\ttype StructuredOutputOptions,\n\ttype StructuredOutputResult,\n} from \"@tanstack/ai/adapters\";\nimport OpenAI from \"openai\";\nimport {\n\ttype WorkersAiAdapterConfig,\n\ttype AiGatewayAdapterConfig,\n\tcreateGatewayFetch,\n\tcreateWorkersAiBindingFetch,\n\tisDirectBindingConfig,\n\tisDirectCredentialsConfig,\n\tvalidateWorkersAiConfig,\n} from \"../utils/create-fetcher\";\n\n// ---------------------------------------------------------------------------\n// Model types derived from @cloudflare/workers-types\n// ---------------------------------------------------------------------------\n\nexport type WorkersAiTextModel =\n\t| {\n\t\t\t[K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never;\n\t }[keyof AiModels]\n\t| (string & {});\n\n// ---------------------------------------------------------------------------\n// Helpers: build the right OpenAI client depending on config mode\n// ---------------------------------------------------------------------------\n\nfunction buildWorkersAiClient(config: WorkersAiAdapterConfig): OpenAI {\n\tvalidateWorkersAiConfig(config);\n\n\tconst sessionHeaders: Record<string, string> | undefined = config.sessionAffinity\n\t\t? { \"x-session-affinity\": config.sessionAffinity }\n\t\t: undefined;\n\n\tif (isDirectBindingConfig(config)) {\n\t\t// Plain binding mode: shim translates OpenAI fetch calls to env.AI.run()\n\t\treturn new OpenAI({\n\t\t\tapiKey: \"unused\",\n\t\t\tfetch: createWorkersAiBindingFetch(\n\t\t\t\tconfig.binding,\n\t\t\t\tsessionHeaders ? { extraHeaders: sessionHeaders } : undefined,\n\t\t\t),\n\t\t});\n\t}\n\n\tif (isDirectCredentialsConfig(config)) {\n\t\t// Plain REST mode: point OpenAI SDK at Workers AI's OpenAI-compatible endpoint\n\t\treturn new OpenAI({\n\t\t\tbaseURL: `https://api.cloudflare.com/client/v4/accounts/${config.accountId}/ai/v1`,\n\t\t\tapiKey: config.apiKey,\n\t\t\tdefaultHeaders: sessionHeaders,\n\t\t});\n\t}\n\n\t// Gateway mode (existing): use createGatewayFetch\n\tconst gatewayConfig = config as AiGatewayAdapterConfig;\n\treturn new OpenAI({\n\t\tfetch: createGatewayFetch(\"workers-ai\", gatewayConfig, sessionHeaders),\n\t\tapiKey: gatewayConfig.apiKey ?? \"unused\",\n\t});\n}\n\n// ---------------------------------------------------------------------------\n// Shared message-building helpers\n// ---------------------------------------------------------------------------\n\nfunction extractTextContent(content: ModelMessage[\"content\"]): string {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n}\n\n/**\n * Convert a single TanStack AI {@link ContentPart} to the OpenAI Chat\n * Completions multi-modal format.\n *\n * TODO: handle other content types (audio, video, document)\n */\nfunction convertContentPart(part: ContentPart): OpenAI.Chat.ChatCompletionContentPart | undefined {\n\tswitch (part.type) {\n\t\tcase \"text\":\n\t\t\tif (part.content) {\n\t\t\t\treturn { type: \"text\", text: part.content };\n\t\t\t}\n\t\t\treturn undefined;\n\t\tcase \"image\": {\n\t\t\tlet url: string;\n\t\t\tif (part.source.type === \"data\") {\n\t\t\t\turl = part.source.value.startsWith(\"data:\")\n\t\t\t\t\t? part.source.value\n\t\t\t\t\t: `data:${part.source.mimeType};base64,${part.source.value}`;\n\t\t\t} else {\n\t\t\t\turl = part.source.value;\n\t\t\t}\n\t\t\treturn { type: \"image_url\", image_url: { url } };\n\t\t}\n\t\tdefault:\n\t\t\t// audio, video, document — not supported for now\n\t\t\tconsole.warn(\n\t\t\t\t`[@cloudflare/tanstack-ai] Unsupported content part type \"${part.type}\" — skipping`,\n\t\t\t);\n\t\t\treturn undefined;\n\t}\n}\n\n/**\n * Build OpenAI-compatible user message content from TanStack AI content.\n *\n * If the content has only text parts, returns a plain string.\n * If it includes image parts, returns an array of content parts in\n * OpenAI's multi-modal format (text + image_url).\n */\nfunction buildUserContent(\n\tcontent: ModelMessage[\"content\"],\n): string | OpenAI.Chat.ChatCompletionContentPart[] {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\n\tconst hasImages = content.some((p) => p.type === \"image\");\n\tif (!hasImages) {\n\t\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n\t}\n\n\tconst parts: OpenAI.Chat.ChatCompletionContentPart[] = [];\n\tfor (const part of content) {\n\t\tconst converted = convertContentPart(part);\n\t\tif (converted) {\n\t\t\tparts.push(converted);\n\t\t}\n\t}\n\treturn parts;\n}\n\nfunction buildOpenAIMessages(\n\tsystemPrompts: string[] | undefined,\n\tmessages: ModelMessage[],\n\toptions?: { includeToolMessages?: boolean },\n): OpenAI.Chat.ChatCompletionMessageParam[] {\n\tconst includeTools = options?.includeToolMessages ?? true;\n\tconst openAIMessages: OpenAI.Chat.ChatCompletionMessageParam[] = [];\n\n\tif (systemPrompts && systemPrompts.length > 0) {\n\t\topenAIMessages.push({\n\t\t\trole: \"system\",\n\t\t\tcontent: systemPrompts.join(\"\\n\"),\n\t\t});\n\t}\n\n\tfor (const message of messages) {\n\t\tif (message.role === \"user\") {\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"user\",\n\t\t\t\tcontent: buildUserContent(message.content),\n\t\t\t});\n\t\t} else if (message.role === \"assistant\") {\n\t\t\tconst assistantMessage: OpenAI.Chat.ChatCompletionAssistantMessageParam = {\n\t\t\t\trole: \"assistant\",\n\t\t\t\tcontent: extractTextContent(message.content),\n\t\t\t};\n\t\t\tif (includeTools && message.toolCalls && message.toolCalls.length > 0) {\n\t\t\t\tassistantMessage.tool_calls = message.toolCalls.map((tc) => ({\n\t\t\t\t\tid: tc.id,\n\t\t\t\t\ttype: \"function\" as const,\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function.name,\n\t\t\t\t\t\targuments: tc.function.arguments,\n\t\t\t\t\t},\n\t\t\t\t}));\n\t\t\t}\n\t\t\topenAIMessages.push(assistantMessage);\n\t\t} else if (includeTools && message.role === \"tool\") {\n\t\t\tlet toolContent: string;\n\t\t\tif (typeof message.content === \"string\") {\n\t\t\t\ttry {\n\t\t\t\t\tJSON.parse(message.content);\n\t\t\t\t\ttoolContent = message.content;\n\t\t\t\t} catch {\n\t\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t}\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"tool\",\n\t\t\t\ttool_call_id: message.toolCallId || `tool_${crypto.randomUUID().slice(0, 8)}`,\n\t\t\t\tcontent: toolContent,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn openAIMessages;\n}\n\nfunction buildOpenAITools(\n\ttools: Array<{ name: string; description: string; inputSchema?: unknown }> | undefined,\n): OpenAI.Chat.ChatCompletionTool[] | undefined {\n\tif (!tools) return undefined;\n\treturn tools.map((tool) => ({\n\t\ttype: \"function\" as const,\n\t\tfunction: {\n\t\t\tname: tool.name,\n\t\t\tdescription: tool.description,\n\t\t\tparameters: tool.inputSchema as Record<string, unknown>,\n\t\t},\n\t}));\n}\n\n// ---------------------------------------------------------------------------\n// ID generation\n// ---------------------------------------------------------------------------\n\nfunction generateId(prefix = \"chatcmpl\"): string {\n\treturn `${prefix}-${crypto.randomUUID().replace(/-/g, \"\").slice(0, 16)}`;\n}\n\n// ---------------------------------------------------------------------------\n// WorkersAiTextAdapter: chat / structured output via OpenAI Chat Completions\n// ---------------------------------------------------------------------------\n\n// TODO: Replace `any` generic params with proper types once BaseTextAdapter's\n// provider-options generics stabilize. Workers AI doesn't have provider-specific\n// options in the TanStack sense, so `any` is pragmatic for now.\nexport class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<\n\tTModel,\n\t// eslint-disable-next-line @typescript-eslint/no-explicit-any -- BaseTextAdapter generic params are opaque\n\tany,\n\tany,\n\tany\n> {\n\tname = \"workers-ai\" as const;\n\n\tprivate client: OpenAI;\n\n\tconstructor(model: TModel, config: WorkersAiAdapterConfig) {\n\t\tsuper({ apiKey: \"unused\" }, model);\n\t\tthis.client = buildWorkersAiClient(config);\n\t}\n\n\tasync *chatStream(options: TextOptions<any>): AsyncIterable<StreamChunk> {\n\t\tconst { systemPrompts, messages, tools, temperature, maxTokens, model } = options;\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages);\n\t\tconst openAITools = buildOpenAITools(tools);\n\n\t\tconst timestamp = Date.now();\n\t\tconst runId = generateId();\n\t\tconst messageId = generateId();\n\t\tlet hasEmittedRunStarted = false;\n\t\tlet hasEmittedTextMessageStart = false;\n\t\tlet accumulatedContent = \"\";\n\t\tlet hasEmittedStepStarted = false;\n\t\tlet accumulatedReasoning = \"\";\n\t\tconst stepId = generateId();\n\t\tlet hasReceivedFinishReason = false;\n\t\tconst toolCallsInProgress = new Map<\n\t\t\tnumber,\n\t\t\t{ id: string; name: string; arguments: string; started: boolean }\n\t\t>();\n\n\t\ttry {\n\t\t\tlet stream: AsyncIterable<OpenAI.Chat.Completions.ChatCompletionChunk>;\n\t\t\ttry {\n\t\t\t\tstream = await this.client.chat.completions.create({\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t\tstream: true,\n\t\t\t\t\tstream_options: { include_usage: true },\n\t\t\t\t});\n\t\t\t} catch (streamError: unknown) {\n\t\t\t\t// Some models (e.g. GPT-OSS) don't support streaming via the REST API.\n\t\t\t\t// Fall back to a non-streaming call and yield the result as a single chunk.\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Streaming failed, falling back to non-streaming:\",\n\t\t\t\t\tstreamError instanceof Error ? streamError.message : streamError,\n\t\t\t\t);\n\t\t\t\tconst nonStreamResult = await this.client.chat.completions.create({\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t});\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\n\t\t\t\tconst msg = nonStreamResult.choices[0]?.message;\n\t\t\t\tif (msg?.content) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: msg.content,\n\t\t\t\t\t\tcontent: msg.content,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tif (msg?.tool_calls) {\n\t\t\t\t\tfor (const tc of msg.tool_calls) {\n\t\t\t\t\t\tif (tc.type !== \"function\") continue;\n\t\t\t\t\t\tconst fn = tc.function;\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = fn.arguments ? JSON.parse(fn.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst finishReason = nonStreamResult.choices[0]?.finish_reason;\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tusage: nonStreamResult.usage\n\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\tpromptTokens: nonStreamResult.usage.prompt_tokens,\n\t\t\t\t\t\t\t\tcompletionTokens: nonStreamResult.usage.completion_tokens,\n\t\t\t\t\t\t\t\ttotalTokens: nonStreamResult.usage.total_tokens,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t: undefined,\n\t\t\t\t\tfinishReason:\n\t\t\t\t\t\tfinishReason === \"tool_calls\" || finishReason === \"function_call\"\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: ((finishReason as \"stop\" | \"length\" | \"content_filter\") ?? \"stop\"),\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor await (const chunk of stream) {\n\t\t\t\tif (!chunk.choices || chunk.choices.length === 0) continue;\n\t\t\t\tconst choice = chunk.choices[0];\n\t\t\t\tif (!choice) continue;\n\n\t\t\t\t// Emit RUN_STARTED on first chunk\n\t\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\t\thasEmittedRunStarted = true;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tconst delta = choice.delta;\n\n\t\t\t\t// Reasoning content (used by models like QwQ, DeepSeek R1, Kimi K2.5)\n\t\t\t\t// The OpenAI SDK doesn't type this field, but models send it as an extension.\n\t\t\t\tconst reasoningContent = ((delta as Record<string, unknown>).reasoning_content ??\n\t\t\t\t\t(delta as Record<string, unknown>).reasoning) as string | undefined;\n\t\t\t\tif (reasoningContent) {\n\t\t\t\t\t// RUN_STARTED is already guaranteed by the guard above\n\t\t\t\t\tif (!hasEmittedStepStarted) {\n\t\t\t\t\t\thasEmittedStepStarted = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"STEP_STARTED\",\n\t\t\t\t\t\t\tstepId,\n\t\t\t\t\t\t\tstepType: \"thinking\",\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t\taccumulatedReasoning += reasoningContent;\n\t\t\t\t\t// TODO: TanStack AI's StreamProcessor currently treats STEP_FINISHED as an\n\t\t\t\t\t// incremental reasoning event (with `delta` + accumulated `content`), so we\n\t\t\t\t\t// emit one per token. If TanStack AI adds a dedicated STEP_CONTENT event\n\t\t\t\t\t// type, this should be updated to emit STEP_CONTENT per token and a single\n\t\t\t\t\t// STEP_FINISHED when reasoning ends (i.e. when the first non-reasoning\n\t\t\t\t\t// content or finish_reason arrives).\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"STEP_FINISHED\",\n\t\t\t\t\t\tstepId,\n\t\t\t\t\t\tdelta: reasoningContent,\n\t\t\t\t\t\tcontent: accumulatedReasoning,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Text content\n\t\t\t\tif (delta.content) {\n\t\t\t\t\tif (!hasEmittedTextMessageStart) {\n\t\t\t\t\t\thasEmittedTextMessageStart = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\taccumulatedContent += delta.content;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: delta.content,\n\t\t\t\t\t\tcontent: accumulatedContent,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Tool calls\n\t\t\t\tif (delta.tool_calls) {\n\t\t\t\t\tfor (const toolCallDelta of delta.tool_calls) {\n\t\t\t\t\t\tconst index = toolCallDelta.index;\n\n\t\t\t\t\t\tif (!toolCallsInProgress.has(index)) {\n\t\t\t\t\t\t\t// Always generate a unique ID per tool call index.\n\t\t\t\t\t\t\t// The backend may send the same ID for multiple tool calls,\n\t\t\t\t\t\t\t// so we cannot trust toolCallDelta.id to be unique.\n\t\t\t\t\t\t\ttoolCallsInProgress.set(index, {\n\t\t\t\t\t\t\t\tid: generateId(\"chatcmpl-tool\"),\n\t\t\t\t\t\t\t\tname: toolCallDelta.function?.name || \"\",\n\t\t\t\t\t\t\t\targuments: \"\",\n\t\t\t\t\t\t\t\tstarted: false,\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst toolCall = toolCallsInProgress.get(index)!;\n\n\t\t\t\t\t\t// Only update name if provided (ID is already set at creation time\n\t\t\t\t\t\t// and should not be overwritten by subsequent chunks that may have\n\t\t\t\t\t\t// duplicate/shared IDs from the backend)\n\t\t\t\t\t\tif (toolCallDelta.function?.name) {\n\t\t\t\t\t\t\ttoolCall.name = toolCallDelta.function.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments) {\n\t\t\t\t\t\t\ttoolCall.arguments += toolCallDelta.function.arguments;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Emit TOOL_CALL_START once we have id and name\n\t\t\t\t\t\tif (toolCall.id && toolCall.name && !toolCall.started) {\n\t\t\t\t\t\t\ttoolCall.started = true;\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tindex,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Stream tool call arguments\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments && toolCall.started) {\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_ARGS\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tdelta: toolCallDelta.function.arguments,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Finish\n\t\t\t\tif (choice.finish_reason) {\n\t\t\t\t\thasReceivedFinishReason = true;\n\n\t\t\t\t\t// End tool calls\n\t\t\t\t\tif (choice.finish_reason === \"tool_calls\" || toolCallsInProgress.size > 0) {\n\t\t\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\t\tparsedInput = toolCall.arguments\n\t\t\t\t\t\t\t\t\t? JSON.parse(toolCall.arguments)\n\t\t\t\t\t\t\t\t\t: {};\n\t\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tconst computedFinishReason =\n\t\t\t\t\t\tchoice.finish_reason === \"tool_calls\" ||\n\t\t\t\t\t\tchoice.finish_reason === \"function_call\" ||\n\t\t\t\t\t\ttoolCallsInProgress.size > 0\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: (choice.finish_reason as \"stop\" | \"length\" | \"content_filter\");\n\n\t\t\t\t\t// End text message if started\n\t\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Emit RUN_FINISHED\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tusage: chunk.usage\n\t\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\t\tpromptTokens: chunk.usage.prompt_tokens,\n\t\t\t\t\t\t\t\t\tcompletionTokens: chunk.usage.completion_tokens,\n\t\t\t\t\t\t\t\t\ttotalTokens: chunk.usage.total_tokens,\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t: undefined,\n\t\t\t\t\t\tfinishReason: computedFinishReason,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Premature stream termination: the stream ended without a finish_reason.\n\t\t\t// This can happen when Workers AI truncates a response or the connection drops.\n\t\t\t// Emit proper closing events so the consumer doesn't hang.\n\t\t\tif (hasEmittedRunStarted && !hasReceivedFinishReason) {\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Stream ended without finish_reason — possible truncation or connection drop\",\n\t\t\t\t);\n\n\t\t\t\t// Close any open tool calls\n\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\tif (toolCall.started) {\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = toolCall.arguments ? JSON.parse(toolCall.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Close text message if open\n\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tfinishReason: \"stop\",\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\tconst code =\n\t\t\t\terror instanceof Error ? (error as Error & { code?: string }).code : undefined;\n\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t\tyield {\n\t\t\t\ttype: \"RUN_ERROR\",\n\t\t\t\trunId,\n\t\t\t\tmodel: model ?? this.model,\n\t\t\t\ttimestamp,\n\t\t\t\terror: {\n\t\t\t\t\tmessage: message || \"Unknown error\",\n\t\t\t\t\tcode,\n\t\t\t\t},\n\t\t\t} satisfies StreamChunk;\n\t\t}\n\t}\n\n\tasync structuredOutput(\n\t\toptions: StructuredOutputOptions<any>,\n\t): Promise<StructuredOutputResult<unknown>> {\n\t\tconst { outputSchema, chatOptions } = options;\n\t\tconst { systemPrompts, messages, temperature, model } = chatOptions;\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages, {\n\t\t\tincludeToolMessages: false,\n\t\t});\n\n\t\tconst response = await this.client.chat.completions.create({\n\t\t\tmodel: model ?? this.model,\n\t\t\tmessages: openAIMessages,\n\t\t\ttemperature,\n\t\t\tstream: false,\n\t\t\tresponse_format: {\n\t\t\t\ttype: \"json_schema\",\n\t\t\t\tjson_schema: {\n\t\t\t\t\tname: \"structured_output\",\n\t\t\t\t\tstrict: true,\n\t\t\t\t\tschema: outputSchema,\n\t\t\t\t},\n\t\t\t},\n\t\t});\n\n\t\tconst choice = response.choices?.[0];\n\n\t\tif (!choice) {\n\t\t\tthrow new Error(\n\t\t\t\t`Workers AI structured output returned no choices: ${JSON.stringify(response)}`,\n\t\t\t);\n\t\t}\n\n\t\tconst rawContent = choice.message?.content ?? \"\";\n\n\t\t// Workers AI REST endpoint may return `content` as an already-parsed object\n\t\t// when using json_schema response format, so normalise both cases.\n\t\tlet data: unknown;\n\t\tlet rawText: string;\n\n\t\tif (typeof rawContent === \"string\") {\n\t\t\trawText = rawContent;\n\t\t\ttry {\n\t\t\t\tdata = JSON.parse(rawText);\n\t\t\t} catch {\n\t\t\t\tdata = rawText;\n\t\t\t}\n\t\t} else {\n\t\t\t// Already an object — stringify for rawText, use directly for data\n\t\t\tdata = rawContent;\n\t\t\trawText = JSON.stringify(rawContent);\n\t\t}\n\n\t\treturn { data, rawText };\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// Factory functions\n// ---------------------------------------------------------------------------\n\nexport function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig) {\n\treturn new WorkersAiTextAdapter(model, config);\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"workers-ai.mjs","names":[],"sources":["../../src/adapters/workers-ai.ts"],"sourcesContent":["import type { AiModels, BaseAiTextGeneration } from \"@cloudflare/workers-types\";\nimport type { ContentPart, ModelMessage, StreamChunk, TextOptions } from \"@tanstack/ai\";\nimport {\n\tBaseTextAdapter,\n\ttype StructuredOutputOptions,\n\ttype StructuredOutputResult,\n} from \"@tanstack/ai/adapters\";\nimport OpenAI from \"openai\";\nimport {\n\ttype WorkersAiAdapterConfig,\n\ttype AiGatewayAdapterConfig,\n\tcreateGatewayFetch,\n\tcreateWorkersAiBindingFetch,\n\tisDirectBindingConfig,\n\tisDirectCredentialsConfig,\n\tvalidateWorkersAiConfig,\n} from \"../utils/create-fetcher\";\n\n// ---------------------------------------------------------------------------\n// Model types derived from @cloudflare/workers-types\n// ---------------------------------------------------------------------------\n\nexport type WorkersAiTextModel =\n\t| {\n\t\t\t[K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never;\n\t }[keyof AiModels]\n\t| (string & {});\n\n// ---------------------------------------------------------------------------\n// Provider-specific options forwarded to Workers AI's chat completions inputs.\n//\n// These correspond to fields on `ChatCompletionsCommonOptions` in\n// `@cloudflare/workers-types`. They are passed verbatim into the request body\n// sent to the Workers AI binding / REST endpoint / AI Gateway, and ultimately\n// land on the `inputs` object of `binding.run(model, inputs)`.\n//\n// Pass via `modelOptions` on a per-call basis:\n//\n// await adapter.chatStream({\n// model, messages,\n// modelOptions: {\n// reasoning_effort: \"low\",\n// chat_template_kwargs: { enable_thinking: false },\n// },\n// });\n// ---------------------------------------------------------------------------\n\nexport interface WorkersAiTextModelOptions {\n\t/**\n\t * Controls the reasoning budget for reasoning-capable models\n\t * (e.g. `@cf/zai-org/glm-4.7-flash`, `@cf/moonshotai/kimi-k2.5`,\n\t * `@cf/openai/gpt-oss-120b`).\n\t *\n\t * `null` is a valid value and disables reasoning for models that support it.\n\t */\n\treasoning_effort?: \"low\" | \"medium\" | \"high\" | null;\n\t/**\n\t * Chat-template overrides for reasoning-capable models that expose\n\t * thinking toggles (e.g. GLM, Kimi).\n\t */\n\tchat_template_kwargs?: {\n\t\t/** Whether to enable reasoning. Enabled by default on reasoning models. */\n\t\tenable_thinking?: boolean;\n\t\t/** If false, preserves reasoning context between turns. */\n\t\tclear_thinking?: boolean;\n\t};\n\t/**\n\t * Escape hatch for other Workers AI inputs-level parameters.\n\t *\n\t * Keys placed here are merged into the outbound request body and forwarded\n\t * to the underlying transport (binding / REST / gateway). Only fields that\n\t * the binding shim knows about are extracted for direct `env.AI` bindings;\n\t * everything is passed through on REST and gateway paths.\n\t */\n\t[key: string]: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Helpers: build the right OpenAI client depending on config mode\n// ---------------------------------------------------------------------------\n\nfunction buildWorkersAiClient(config: WorkersAiAdapterConfig): OpenAI {\n\tvalidateWorkersAiConfig(config);\n\n\tconst sessionHeaders: Record<string, string> | undefined = config.sessionAffinity\n\t\t? { \"x-session-affinity\": config.sessionAffinity }\n\t\t: undefined;\n\n\tif (isDirectBindingConfig(config)) {\n\t\t// Plain binding mode: shim translates OpenAI fetch calls to env.AI.run()\n\t\treturn new OpenAI({\n\t\t\tapiKey: \"unused\",\n\t\t\tfetch: createWorkersAiBindingFetch(\n\t\t\t\tconfig.binding,\n\t\t\t\tsessionHeaders ? { extraHeaders: sessionHeaders } : undefined,\n\t\t\t),\n\t\t});\n\t}\n\n\tif (isDirectCredentialsConfig(config)) {\n\t\t// Plain REST mode: point OpenAI SDK at Workers AI's OpenAI-compatible endpoint\n\t\treturn new OpenAI({\n\t\t\tbaseURL: `https://api.cloudflare.com/client/v4/accounts/${config.accountId}/ai/v1`,\n\t\t\tapiKey: config.apiKey,\n\t\t\tdefaultHeaders: sessionHeaders,\n\t\t});\n\t}\n\n\t// Gateway mode (existing): use createGatewayFetch\n\tconst gatewayConfig = config as AiGatewayAdapterConfig;\n\treturn new OpenAI({\n\t\tfetch: createGatewayFetch(\"workers-ai\", gatewayConfig, sessionHeaders),\n\t\tapiKey: gatewayConfig.apiKey ?? \"unused\",\n\t});\n}\n\n// ---------------------------------------------------------------------------\n// Shared message-building helpers\n// ---------------------------------------------------------------------------\n\nfunction extractTextContent(content: ModelMessage[\"content\"]): string {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n}\n\n/**\n * Convert a single TanStack AI {@link ContentPart} to the OpenAI Chat\n * Completions multi-modal format.\n *\n * TODO: handle other content types (audio, video, document)\n */\nfunction convertContentPart(part: ContentPart): OpenAI.Chat.ChatCompletionContentPart | undefined {\n\tswitch (part.type) {\n\t\tcase \"text\":\n\t\t\tif (part.content) {\n\t\t\t\treturn { type: \"text\", text: part.content };\n\t\t\t}\n\t\t\treturn undefined;\n\t\tcase \"image\": {\n\t\t\tlet url: string;\n\t\t\tif (part.source.type === \"data\") {\n\t\t\t\turl = part.source.value.startsWith(\"data:\")\n\t\t\t\t\t? part.source.value\n\t\t\t\t\t: `data:${part.source.mimeType};base64,${part.source.value}`;\n\t\t\t} else {\n\t\t\t\turl = part.source.value;\n\t\t\t}\n\t\t\treturn { type: \"image_url\", image_url: { url } };\n\t\t}\n\t\tdefault:\n\t\t\t// audio, video, document — not supported for now\n\t\t\tconsole.warn(\n\t\t\t\t`[@cloudflare/tanstack-ai] Unsupported content part type \"${part.type}\" — skipping`,\n\t\t\t);\n\t\t\treturn undefined;\n\t}\n}\n\n/**\n * Build OpenAI-compatible user message content from TanStack AI content.\n *\n * If the content has only text parts, returns a plain string.\n * If it includes image parts, returns an array of content parts in\n * OpenAI's multi-modal format (text + image_url).\n */\nfunction buildUserContent(\n\tcontent: ModelMessage[\"content\"],\n): string | OpenAI.Chat.ChatCompletionContentPart[] {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\n\tconst hasImages = content.some((p) => p.type === \"image\");\n\tif (!hasImages) {\n\t\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n\t}\n\n\tconst parts: OpenAI.Chat.ChatCompletionContentPart[] = [];\n\tfor (const part of content) {\n\t\tconst converted = convertContentPart(part);\n\t\tif (converted) {\n\t\t\tparts.push(converted);\n\t\t}\n\t}\n\treturn parts;\n}\n\nfunction buildOpenAIMessages(\n\tsystemPrompts: string[] | undefined,\n\tmessages: ModelMessage[],\n\toptions?: { includeToolMessages?: boolean },\n): OpenAI.Chat.ChatCompletionMessageParam[] {\n\tconst includeTools = options?.includeToolMessages ?? true;\n\tconst openAIMessages: OpenAI.Chat.ChatCompletionMessageParam[] = [];\n\n\tif (systemPrompts && systemPrompts.length > 0) {\n\t\topenAIMessages.push({\n\t\t\trole: \"system\",\n\t\t\tcontent: systemPrompts.join(\"\\n\"),\n\t\t});\n\t}\n\n\tfor (const message of messages) {\n\t\tif (message.role === \"user\") {\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"user\",\n\t\t\t\tcontent: buildUserContent(message.content),\n\t\t\t});\n\t\t} else if (message.role === \"assistant\") {\n\t\t\tconst assistantMessage: OpenAI.Chat.ChatCompletionAssistantMessageParam = {\n\t\t\t\trole: \"assistant\",\n\t\t\t\tcontent: extractTextContent(message.content),\n\t\t\t};\n\t\t\tif (includeTools && message.toolCalls && message.toolCalls.length > 0) {\n\t\t\t\tassistantMessage.tool_calls = message.toolCalls.map((tc) => ({\n\t\t\t\t\tid: tc.id,\n\t\t\t\t\ttype: \"function\" as const,\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function.name,\n\t\t\t\t\t\targuments: tc.function.arguments,\n\t\t\t\t\t},\n\t\t\t\t}));\n\t\t\t}\n\t\t\topenAIMessages.push(assistantMessage);\n\t\t} else if (includeTools && message.role === \"tool\") {\n\t\t\tlet toolContent: string;\n\t\t\tif (typeof message.content === \"string\") {\n\t\t\t\ttry {\n\t\t\t\t\tJSON.parse(message.content);\n\t\t\t\t\ttoolContent = message.content;\n\t\t\t\t} catch {\n\t\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t}\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"tool\",\n\t\t\t\ttool_call_id: message.toolCallId || `tool_${crypto.randomUUID().slice(0, 8)}`,\n\t\t\t\tcontent: toolContent,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn openAIMessages;\n}\n\nfunction buildOpenAITools(\n\ttools: Array<{ name: string; description: string; inputSchema?: unknown }> | undefined,\n): OpenAI.Chat.ChatCompletionTool[] | undefined {\n\tif (!tools) return undefined;\n\treturn tools.map((tool) => ({\n\t\ttype: \"function\" as const,\n\t\tfunction: {\n\t\t\tname: tool.name,\n\t\t\tdescription: tool.description,\n\t\t\tparameters: tool.inputSchema as Record<string, unknown>,\n\t\t},\n\t}));\n}\n\n// ---------------------------------------------------------------------------\n// ID generation\n// ---------------------------------------------------------------------------\n\nfunction generateId(prefix = \"chatcmpl\"): string {\n\treturn `${prefix}-${crypto.randomUUID().replace(/-/g, \"\").slice(0, 16)}`;\n}\n\n// ---------------------------------------------------------------------------\n// modelOptions normalization\n//\n// Users pass Workers AI-specific chat params (e.g. `reasoning_effort`,\n// `chat_template_kwargs`) via `modelOptions`. We merge these into the outbound\n// request body so they reach the binding / REST / gateway transports.\n//\n// Spread order matters: these go FIRST in the request body so that TanStack-AI\n// managed fields (`model`, `messages`, `temperature`, `max_tokens`, `stream`,\n// `tools`, `response_format`, ...) always win if a user accidentally sets\n// them both at the top level and inside `modelOptions`.\n//\n// `undefined` values are stripped so that JSON.stringify (and our binding shim\n// which does `!== undefined` checks) see them as absent. `null` values are\n// preserved — they're meaningful for fields like `reasoning_effort: null`\n// which explicitly disables reasoning on some models.\n// ---------------------------------------------------------------------------\nfunction normalizeModelOptions(\n\tmodelOptions: WorkersAiTextModelOptions | undefined,\n): Record<string, unknown> {\n\t// Guard against runtime misuse. TanStack AI types this as an object, but\n\t// users can always bypass with `as any`. `Object.entries` on a string\n\t// surprisingly returns per-character tuples (e.g. `Object.entries(\"ab\") →\n\t// [[\"0\",\"a\"],[\"1\",\"b\"]]`), which would leak spurious keys into the body.\n\t// Arrays similarly become index-keyed. Only accept plain objects.\n\tif (modelOptions === null || typeof modelOptions !== \"object\" || Array.isArray(modelOptions)) {\n\t\treturn {};\n\t}\n\tconst out: Record<string, unknown> = {};\n\tfor (const [key, value] of Object.entries(modelOptions)) {\n\t\tif (value !== undefined) out[key] = value;\n\t}\n\treturn out;\n}\n\n// ---------------------------------------------------------------------------\n// WorkersAiTextAdapter: chat / structured output via OpenAI Chat Completions\n// ---------------------------------------------------------------------------\n\nexport class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<\n\tTModel,\n\tWorkersAiTextModelOptions,\n\t// eslint-disable-next-line @typescript-eslint/no-explicit-any -- remaining BaseTextAdapter generics are opaque\n\tany,\n\tany\n> {\n\tname = \"workers-ai\" as const;\n\n\tprivate client: OpenAI;\n\n\tconstructor(model: TModel, config: WorkersAiAdapterConfig) {\n\t\tsuper({ apiKey: \"unused\" }, model);\n\t\tthis.client = buildWorkersAiClient(config);\n\t}\n\n\tasync *chatStream(options: TextOptions<WorkersAiTextModelOptions>): AsyncIterable<StreamChunk> {\n\t\tconst { systemPrompts, messages, tools, temperature, maxTokens, model, modelOptions } =\n\t\t\toptions;\n\t\tconst extraBody = normalizeModelOptions(modelOptions);\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages);\n\t\tconst openAITools = buildOpenAITools(tools);\n\n\t\tconst timestamp = Date.now();\n\t\tconst runId = generateId();\n\t\tconst messageId = generateId();\n\t\tlet hasEmittedRunStarted = false;\n\t\tlet hasEmittedTextMessageStart = false;\n\t\tlet accumulatedContent = \"\";\n\t\tlet hasEmittedStepStarted = false;\n\t\tlet accumulatedReasoning = \"\";\n\t\tconst stepId = generateId();\n\t\tlet hasReceivedFinishReason = false;\n\t\tconst toolCallsInProgress = new Map<\n\t\t\tnumber,\n\t\t\t{ id: string; name: string; arguments: string; started: boolean }\n\t\t>();\n\n\t\ttry {\n\t\t\tlet stream: AsyncIterable<OpenAI.Chat.Completions.ChatCompletionChunk>;\n\t\t\ttry {\n\t\t\t\tstream = await this.client.chat.completions.create({\n\t\t\t\t\t...extraBody,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t\tstream: true,\n\t\t\t\t\tstream_options: { include_usage: true },\n\t\t\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsStreaming);\n\t\t\t} catch (streamError: unknown) {\n\t\t\t\t// Some models (e.g. GPT-OSS) don't support streaming via the REST API.\n\t\t\t\t// Fall back to a non-streaming call and yield the result as a single chunk.\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Streaming failed, falling back to non-streaming:\",\n\t\t\t\t\tstreamError instanceof Error ? streamError.message : streamError,\n\t\t\t\t);\n\t\t\t\tconst nonStreamResult = await this.client.chat.completions.create({\n\t\t\t\t\t...extraBody,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsNonStreaming);\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\n\t\t\t\tconst msg = nonStreamResult.choices[0]?.message;\n\t\t\t\tif (msg?.content) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: msg.content,\n\t\t\t\t\t\tcontent: msg.content,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tif (msg?.tool_calls) {\n\t\t\t\t\tfor (const tc of msg.tool_calls) {\n\t\t\t\t\t\tif (tc.type !== \"function\") continue;\n\t\t\t\t\t\tconst fn = tc.function;\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = fn.arguments ? JSON.parse(fn.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst finishReason = nonStreamResult.choices[0]?.finish_reason;\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tusage: nonStreamResult.usage\n\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\tpromptTokens: nonStreamResult.usage.prompt_tokens,\n\t\t\t\t\t\t\t\tcompletionTokens: nonStreamResult.usage.completion_tokens,\n\t\t\t\t\t\t\t\ttotalTokens: nonStreamResult.usage.total_tokens,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t: undefined,\n\t\t\t\t\tfinishReason:\n\t\t\t\t\t\tfinishReason === \"tool_calls\" || finishReason === \"function_call\"\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: ((finishReason as \"stop\" | \"length\" | \"content_filter\") ?? \"stop\"),\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor await (const chunk of stream) {\n\t\t\t\tif (!chunk.choices || chunk.choices.length === 0) continue;\n\t\t\t\tconst choice = chunk.choices[0];\n\t\t\t\tif (!choice) continue;\n\n\t\t\t\t// Emit RUN_STARTED on first chunk\n\t\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\t\thasEmittedRunStarted = true;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tconst delta = choice.delta;\n\n\t\t\t\t// Reasoning content (used by models like QwQ, DeepSeek R1, Kimi K2.5)\n\t\t\t\t// The OpenAI SDK doesn't type this field, but models send it as an extension.\n\t\t\t\tconst reasoningContent = ((delta as Record<string, unknown>).reasoning_content ??\n\t\t\t\t\t(delta as Record<string, unknown>).reasoning) as string | undefined;\n\t\t\t\tif (reasoningContent) {\n\t\t\t\t\t// RUN_STARTED is already guaranteed by the guard above\n\t\t\t\t\tif (!hasEmittedStepStarted) {\n\t\t\t\t\t\thasEmittedStepStarted = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"STEP_STARTED\",\n\t\t\t\t\t\t\tstepId,\n\t\t\t\t\t\t\tstepType: \"thinking\",\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t\taccumulatedReasoning += reasoningContent;\n\t\t\t\t\t// TODO: TanStack AI's StreamProcessor currently treats STEP_FINISHED as an\n\t\t\t\t\t// incremental reasoning event (with `delta` + accumulated `content`), so we\n\t\t\t\t\t// emit one per token. If TanStack AI adds a dedicated STEP_CONTENT event\n\t\t\t\t\t// type, this should be updated to emit STEP_CONTENT per token and a single\n\t\t\t\t\t// STEP_FINISHED when reasoning ends (i.e. when the first non-reasoning\n\t\t\t\t\t// content or finish_reason arrives).\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"STEP_FINISHED\",\n\t\t\t\t\t\tstepId,\n\t\t\t\t\t\tdelta: reasoningContent,\n\t\t\t\t\t\tcontent: accumulatedReasoning,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Text content\n\t\t\t\tif (delta.content) {\n\t\t\t\t\tif (!hasEmittedTextMessageStart) {\n\t\t\t\t\t\thasEmittedTextMessageStart = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\taccumulatedContent += delta.content;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: delta.content,\n\t\t\t\t\t\tcontent: accumulatedContent,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Tool calls\n\t\t\t\tif (delta.tool_calls) {\n\t\t\t\t\tfor (const toolCallDelta of delta.tool_calls) {\n\t\t\t\t\t\tconst index = toolCallDelta.index;\n\n\t\t\t\t\t\tif (!toolCallsInProgress.has(index)) {\n\t\t\t\t\t\t\t// Always generate a unique ID per tool call index.\n\t\t\t\t\t\t\t// The backend may send the same ID for multiple tool calls,\n\t\t\t\t\t\t\t// so we cannot trust toolCallDelta.id to be unique.\n\t\t\t\t\t\t\ttoolCallsInProgress.set(index, {\n\t\t\t\t\t\t\t\tid: generateId(\"chatcmpl-tool\"),\n\t\t\t\t\t\t\t\tname: toolCallDelta.function?.name || \"\",\n\t\t\t\t\t\t\t\targuments: \"\",\n\t\t\t\t\t\t\t\tstarted: false,\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst toolCall = toolCallsInProgress.get(index)!;\n\n\t\t\t\t\t\t// Only update name if provided (ID is already set at creation time\n\t\t\t\t\t\t// and should not be overwritten by subsequent chunks that may have\n\t\t\t\t\t\t// duplicate/shared IDs from the backend)\n\t\t\t\t\t\tif (toolCallDelta.function?.name) {\n\t\t\t\t\t\t\ttoolCall.name = toolCallDelta.function.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments) {\n\t\t\t\t\t\t\ttoolCall.arguments += toolCallDelta.function.arguments;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Emit TOOL_CALL_START once we have id and name\n\t\t\t\t\t\tif (toolCall.id && toolCall.name && !toolCall.started) {\n\t\t\t\t\t\t\ttoolCall.started = true;\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tindex,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Stream tool call arguments\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments && toolCall.started) {\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_ARGS\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tdelta: toolCallDelta.function.arguments,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Finish\n\t\t\t\tif (choice.finish_reason) {\n\t\t\t\t\thasReceivedFinishReason = true;\n\n\t\t\t\t\t// End tool calls\n\t\t\t\t\tif (choice.finish_reason === \"tool_calls\" || toolCallsInProgress.size > 0) {\n\t\t\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\t\tparsedInput = toolCall.arguments\n\t\t\t\t\t\t\t\t\t? JSON.parse(toolCall.arguments)\n\t\t\t\t\t\t\t\t\t: {};\n\t\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tconst computedFinishReason =\n\t\t\t\t\t\tchoice.finish_reason === \"tool_calls\" ||\n\t\t\t\t\t\tchoice.finish_reason === \"function_call\" ||\n\t\t\t\t\t\ttoolCallsInProgress.size > 0\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: (choice.finish_reason as \"stop\" | \"length\" | \"content_filter\");\n\n\t\t\t\t\t// End text message if started\n\t\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Emit RUN_FINISHED\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tusage: chunk.usage\n\t\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\t\tpromptTokens: chunk.usage.prompt_tokens,\n\t\t\t\t\t\t\t\t\tcompletionTokens: chunk.usage.completion_tokens,\n\t\t\t\t\t\t\t\t\ttotalTokens: chunk.usage.total_tokens,\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t: undefined,\n\t\t\t\t\t\tfinishReason: computedFinishReason,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Premature stream termination: the stream ended without a finish_reason.\n\t\t\t// This can happen when Workers AI truncates a response or the connection drops.\n\t\t\t// Emit proper closing events so the consumer doesn't hang.\n\t\t\tif (hasEmittedRunStarted && !hasReceivedFinishReason) {\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Stream ended without finish_reason — possible truncation or connection drop\",\n\t\t\t\t);\n\n\t\t\t\t// Close any open tool calls\n\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\tif (toolCall.started) {\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = toolCall.arguments ? JSON.parse(toolCall.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Close text message if open\n\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tfinishReason: \"stop\",\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\tconst code =\n\t\t\t\terror instanceof Error ? (error as Error & { code?: string }).code : undefined;\n\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t\tyield {\n\t\t\t\ttype: \"RUN_ERROR\",\n\t\t\t\trunId,\n\t\t\t\tmodel: model ?? this.model,\n\t\t\t\ttimestamp,\n\t\t\t\terror: {\n\t\t\t\t\tmessage: message || \"Unknown error\",\n\t\t\t\t\tcode,\n\t\t\t\t},\n\t\t\t} satisfies StreamChunk;\n\t\t}\n\t}\n\n\tasync structuredOutput(\n\t\toptions: StructuredOutputOptions<WorkersAiTextModelOptions>,\n\t): Promise<StructuredOutputResult<unknown>> {\n\t\tconst { outputSchema, chatOptions } = options;\n\t\tconst { systemPrompts, messages, temperature, model, modelOptions } = chatOptions;\n\t\tconst extraBody = normalizeModelOptions(modelOptions);\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages, {\n\t\t\tincludeToolMessages: false,\n\t\t});\n\n\t\tconst response = await this.client.chat.completions.create({\n\t\t\t...extraBody,\n\t\t\tmodel: model ?? this.model,\n\t\t\tmessages: openAIMessages,\n\t\t\ttemperature,\n\t\t\tstream: false,\n\t\t\tresponse_format: {\n\t\t\t\ttype: \"json_schema\",\n\t\t\t\tjson_schema: {\n\t\t\t\t\tname: \"structured_output\",\n\t\t\t\t\tstrict: true,\n\t\t\t\t\tschema: outputSchema,\n\t\t\t\t},\n\t\t\t},\n\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsNonStreaming);\n\n\t\tconst choice = response.choices?.[0];\n\n\t\tif (!choice) {\n\t\t\tthrow new Error(\n\t\t\t\t`Workers AI structured output returned no choices: ${JSON.stringify(response)}`,\n\t\t\t);\n\t\t}\n\n\t\tconst rawContent = choice.message?.content ?? \"\";\n\n\t\t// Workers AI REST endpoint may return `content` as an already-parsed object\n\t\t// when using json_schema response format, so normalise both cases.\n\t\tlet data: unknown;\n\t\tlet rawText: string;\n\n\t\tif (typeof rawContent === \"string\") {\n\t\t\trawText = rawContent;\n\t\t\ttry {\n\t\t\t\tdata = JSON.parse(rawText);\n\t\t\t} catch {\n\t\t\t\tdata = rawText;\n\t\t\t}\n\t\t} else {\n\t\t\t// Already an object — stringify for rawText, use directly for data\n\t\t\tdata = rawContent;\n\t\t\trawText = JSON.stringify(rawContent);\n\t\t}\n\n\t\treturn { data, rawText };\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// Factory functions\n// ---------------------------------------------------------------------------\n\nexport function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig) {\n\treturn new WorkersAiTextAdapter(model, config);\n}\n"],"mappings":"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"}
@@ -109,6 +109,8 @@ function createWorkersAiBindingFetch(binding, options) {
109
109
  if (typeof body.max_tokens === "number") inputs.max_tokens = body.max_tokens;
110
110
  if (body.response_format) inputs.response_format = body.response_format;
111
111
  if (stream) inputs.stream = true;
112
+ if (body.reasoning_effort !== void 0) inputs.reasoning_effort = body.reasoning_effort;
113
+ if (body.chat_template_kwargs !== void 0) inputs.chat_template_kwargs = body.chat_template_kwargs;
112
114
  const runOptions = {};
113
115
  if (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;
114
116
  if (init?.signal) runOptions.signal = init.signal;
@@ -307,4 +309,4 @@ Object.defineProperty(exports, "validateWorkersAiConfig", {
307
309
  }
308
310
  });
309
311
 
310
- //# sourceMappingURL=create-fetcher-5iL34e6H.cjs.map
312
+ //# sourceMappingURL=create-fetcher-By-hTiT9.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-fetcher-By-hTiT9.cjs","names":[],"sources":["../src/utils/create-fetcher.ts"],"sourcesContent":["// ---------------------------------------------------------------------------\n// AI Gateway types (for third-party providers + Workers AI through gateway)\n// ---------------------------------------------------------------------------\n\nexport interface CloudflareAiGateway {\n\trun(request: unknown): Promise<Response>;\n}\n\nexport interface AiGatewayBindingConfig {\n\t/**\n\t * The AI Gateway binding\n\t * @example\n\t * env.AI.gateway('my-gateway-id')\n\t */\n\tbinding: CloudflareAiGateway;\n\t/**\n\t * The Provider API Key if you want to manually pass it, ignore if using Unified Billing or BYOK.\n\t */\n\tapiKey?: string;\n}\n\nexport type AiGatewayCredentialsConfig = {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The AI Gateway ID\n\t */\n\tgatewayId: string;\n} & (\n\t| {\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey: string;\n\t\t\tapiKey?: string;\n\t }\n\t| {\n\t\t\t/** Provider API Key */\n\t\t\tapiKey: string;\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey?: string;\n\t }\n);\n\nexport interface AiGatewayConfig {\n\tskipCache?: boolean;\n\tcacheTtl?: number;\n\tcustomCacheKey?: string;\n\tmetadata?: Record<string, unknown>;\n}\n\nexport type AiGatewayAdapterConfig = (AiGatewayBindingConfig | AiGatewayCredentialsConfig) &\n\tAiGatewayConfig;\n\n// ---------------------------------------------------------------------------\n// Plain Workers AI types (direct binding or REST, no gateway)\n// ---------------------------------------------------------------------------\n\n/**\n * The Workers AI binding interface (env.AI).\n * Accepts a model name and inputs, returns results directly.\n * Includes `gateway()` which is present on `env.AI` but not on `env.AI.gateway(id)`,\n * enabling structural discrimination from `CloudflareAiGateway`.\n */\nexport interface WorkersAiBinding {\n\trun(\n\t\tmodel: string,\n\t\tinputs: Record<string, unknown>,\n\t\toptions?: Record<string, unknown>,\n\t): Promise<unknown>;\n\tgateway(gatewayId: string): CloudflareAiGateway;\n}\n\nexport interface WorkersAiDirectBindingConfig {\n\t/**\n\t * The Workers AI binding (env.AI).\n\t * @example\n\t * { binding: env.AI }\n\t */\n\tbinding: WorkersAiBinding;\n}\n\nexport interface WorkersAiDirectCredentialsConfig {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The Cloudflare API key for Workers AI\n\t */\n\tapiKey: string;\n}\n\n/**\n * Config for Workers AI adapters. Supports four modes:\n * - Plain binding: `{ binding: env.AI }`\n * - Plain REST: `{ accountId, apiKey }`\n * - AI Gateway binding: `{ binding: env.AI.gateway(id) }`\n * - AI Gateway REST: `{ accountId, gatewayId, ... }`\n *\n * The third union member intersects `AiGatewayAdapterConfig` with `{ apiKey?: string }`.\n * For the gateway binding variant, `AiGatewayBindingConfig` already includes `apiKey?`,\n * so the intersection is redundant there. For the gateway credentials variant, this\n * `apiKey` represents the Workers AI token (used in the `Authorization` header to the\n * upstream provider), distinct from `cfApiKey` (used in the `cf-aig-authorization`\n * header for authenticated gateways).\n */\nexport type WorkersAiAdapterConfig = (\n\t| WorkersAiDirectBindingConfig\n\t| WorkersAiDirectCredentialsConfig\n\t| (AiGatewayAdapterConfig & { apiKey?: string })\n) & {\n\t/**\n\t * Session affinity key for prefix-cache optimization.\n\t * Routes requests with the same key to the same backend replica.\n\t */\n\tsessionAffinity?: string;\n};\n\n// ---------------------------------------------------------------------------\n// Config detection helpers\n// ---------------------------------------------------------------------------\n\n/** Returns true if this is a plain Workers AI binding config (`{ binding: env.AI }`) */\nexport function isDirectBindingConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectBindingConfig {\n\t// env.AI has a .gateway() method; env.AI.gateway(id) does not.\n\t// This distinguishes direct bindings from AI Gateway bindings.\n\treturn (\n\t\t\"binding\" in config &&\n\t\ttypeof (config.binding as unknown as Record<string, unknown>).gateway === \"function\"\n\t);\n}\n\n/** Returns true if this is a plain Workers AI REST config (accountId + apiKey, no gatewayId) */\nexport function isDirectCredentialsConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectCredentialsConfig {\n\treturn \"accountId\" in config && \"apiKey\" in config && !(\"gatewayId\" in config);\n}\n\n/** Returns true if this is an AI Gateway config (has gateway binding or `gatewayId`) */\nexport function isGatewayConfig(config: WorkersAiAdapterConfig): config is AiGatewayAdapterConfig {\n\tif (\"gatewayId\" in config) return true;\n\t// Has `binding` but NOT a direct Workers AI binding (no .gateway method)\n\treturn \"binding\" in config && !isDirectBindingConfig(config);\n}\n\n// ---------------------------------------------------------------------------\n// Config validation\n// ---------------------------------------------------------------------------\n\n/**\n * Validates that a WorkersAiAdapterConfig contains a valid configuration.\n * Throws an error if neither a binding, credentials (accountId + apiKey),\n * nor a gateway configuration is provided.\n */\nexport function validateWorkersAiConfig(config: WorkersAiAdapterConfig): void {\n\tif (\n\t\t!isDirectBindingConfig(config) &&\n\t\t!isDirectCredentialsConfig(config) &&\n\t\t!isGatewayConfig(config)\n\t) {\n\t\tthrow new Error(\n\t\t\t\"Invalid Workers AI configuration: you must provide either a binding (e.g. { binding: env.AI }), \" +\n\t\t\t\t\"credentials ({ accountId, apiKey }), or a gateway configuration ({ binding: env.AI.gateway(id) } \" +\n\t\t\t\t\"or { accountId, gatewayId }).\",\n\t\t);\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// createGatewayFetch -- for routing through AI Gateway\n// ---------------------------------------------------------------------------\n\nexport function createGatewayFetch(\n\tprovider: string,\n\tconfig: AiGatewayAdapterConfig,\n\theaders: Record<string, string> = {},\n): typeof fetch {\n\treturn (input, init) => {\n\t\tlet query: Record<string, unknown> = {};\n\n\t\tconst url =\n\t\t\ttypeof input === \"string\" ? input : input instanceof URL ? input.href : input.url;\n\t\tconst urlObj = new URL(url);\n\n\t\t// Extract endpoint path (remove /v1/ prefix if present)\n\t\tconst endpoint = urlObj.pathname.replace(/^\\/v1\\//, \"\").replace(/^\\//, \"\") + urlObj.search;\n\n\t\tif (init?.body) {\n\t\t\ttry {\n\t\t\t\tquery = JSON.parse(init.body as string);\n\t\t\t} catch {\n\t\t\t\tquery = { _raw: init.body };\n\t\t\t}\n\t\t}\n\n\t\tconst cacheHeaders: Record<string, string> = {};\n\n\t\tif (\"skipCache\" in config && config.skipCache) {\n\t\t\tcacheHeaders[\"cf-aig-skip-cache\"] = \"true\";\n\t\t}\n\n\t\tif (typeof config.cacheTtl === \"number\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-ttl\"] = String(config.cacheTtl);\n\t\t}\n\n\t\tif (typeof config.customCacheKey === \"string\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-key\"] = config.customCacheKey;\n\t\t}\n\n\t\tif (typeof config.metadata === \"object\") {\n\t\t\tcacheHeaders[\"cf-aig-metadata\"] = JSON.stringify(config.metadata);\n\t\t}\n\n\t\tconst request: {\n\t\t\tprovider: string;\n\t\t\tendpoint: string;\n\t\t\theaders: Record<string, string>;\n\t\t\tquery: Record<string, unknown>;\n\t\t} = {\n\t\t\tprovider,\n\t\t\tendpoint,\n\t\t\theaders: {\n\t\t\t\t...(init?.headers as Record<string, string> | undefined),\n\t\t\t\t...headers,\n\t\t\t\t...cacheHeaders,\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tquery,\n\t\t};\n\n\t\tif (provider === \"workers-ai\") {\n\t\t\tif (!request.endpoint.startsWith(\"run/\")) {\n\t\t\t\trequest.endpoint = `run/${query.model}`;\n\t\t\t}\n\t\t\tdelete query.model;\n\t\t\tdelete query.instructions;\n\t\t}\n\n\t\tif (config.apiKey) {\n\t\t\trequest.headers[\"authorization\"] = `Bearer ${config.apiKey}`;\n\t\t}\n\n\t\tif (\"binding\" in config) {\n\t\t\treturn (\n\t\t\t\tconfig.binding as {\n\t\t\t\t\trun(req: unknown, opts?: { signal?: AbortSignal }): Promise<Response>;\n\t\t\t\t}\n\t\t\t).run(request, { signal: init?.signal ?? undefined });\n\t\t}\n\n\t\treturn fetch(\n\t\t\t`https://gateway.ai.cloudflare.com/v1/${config.accountId}/${config.gatewayId}`,\n\t\t\t{\n\t\t\t\t...init,\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t\t\t...headers,\n\t\t\t\t\t...cacheHeaders,\n\t\t\t\t\t...(config.cfApiKey\n\t\t\t\t\t\t? { \"cf-aig-authorization\": `Bearer ${config.cfApiKey}` }\n\t\t\t\t\t\t: {}),\n\t\t\t\t},\n\t\t\t\tbody: JSON.stringify(request),\n\t\t\t},\n\t\t);\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// createWorkersAiBindingFetch -- shim that makes env.AI look like an OpenAI endpoint\n// ---------------------------------------------------------------------------\n\n/**\n * Normalize messages before passing to Workers AI binding.\n *\n * The binding has strict schema validation that may differ from the OpenAI API:\n * - `content` must not be null\n *\n * Content arrays (with image_url parts) are passed through as-is since the\n * Workers AI binding accepts them at runtime for vision-capable models.\n */\nfunction normalizeMessagesForBinding(\n\tmessages: Record<string, unknown>[],\n): Record<string, unknown>[] {\n\treturn messages.map((msg) => {\n\t\tconst normalized = { ...msg };\n\n\t\t// content: null → content: \"\"\n\t\tif (normalized.content === null || normalized.content === undefined) {\n\t\t\tnormalized.content = \"\";\n\t\t}\n\n\t\treturn normalized;\n\t});\n}\n\n/**\n * Creates a fetch function that intercepts OpenAI SDK requests and translates them\n * to Workers AI binding calls (env.AI.run). This allows the WorkersAiTextAdapter\n * to use the OpenAI SDK against a plain Workers AI binding.\n *\n * NOTE: The `input` URL parameter is intentionally ignored. The model name and all\n * request parameters are extracted from the JSON body, matching Workers AI's\n * `binding.run(model, inputs)` calling convention.\n */\nexport function createWorkersAiBindingFetch(\n\tbinding: WorkersAiBinding,\n\toptions?: { extraHeaders?: Record<string, string> },\n): typeof fetch {\n\treturn async (_input, init) => {\n\t\tif (!init?.body) {\n\t\t\treturn new Response(\"No body\", { status: 400 });\n\t\t}\n\n\t\tlet body: Record<string, unknown>;\n\t\ttry {\n\t\t\tbody = JSON.parse(init.body as string);\n\t\t} catch {\n\t\t\treturn new Response(\"Invalid JSON body\", { status: 400 });\n\t\t}\n\n\t\tconst model = body.model as string;\n\t\tconst stream = body.stream as boolean | undefined;\n\n\t\t// Build Workers AI inputs from OpenAI format\n\t\tconst inputs: Record<string, unknown> = {};\n\t\tif (body.messages) {\n\t\t\tinputs.messages = normalizeMessagesForBinding(\n\t\t\t\tbody.messages as Record<string, unknown>[],\n\t\t\t);\n\t\t}\n\t\tif (body.tools) inputs.tools = body.tools;\n\t\tif (typeof body.temperature === \"number\") inputs.temperature = body.temperature;\n\t\tif (typeof body.max_tokens === \"number\") inputs.max_tokens = body.max_tokens;\n\t\tif (body.response_format) inputs.response_format = body.response_format;\n\t\tif (stream) inputs.stream = true;\n\n\t\t// Workers AI-specific reasoning controls. These belong on the INPUTS object\n\t\t// passed to binding.run(model, inputs), not on the options (3rd) arg.\n\t\t// See https://github.com/cloudflare/ai/issues/503.\n\t\t//\n\t\t// `reasoning_effort: null` is a valid value (disables reasoning on models\n\t\t// that support it), so we check `!== undefined` rather than truthiness.\n\t\tif (body.reasoning_effort !== undefined) {\n\t\t\tinputs.reasoning_effort = body.reasoning_effort;\n\t\t}\n\t\tif (body.chat_template_kwargs !== undefined) {\n\t\t\tinputs.chat_template_kwargs = body.chat_template_kwargs;\n\t\t}\n\n\t\tconst runOptions: Record<string, unknown> = {};\n\t\tif (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;\n\t\tif (init?.signal) runOptions.signal = init.signal;\n\n\t\tconst result = await binding.run(\n\t\t\tmodel,\n\t\t\tinputs,\n\t\t\tObject.keys(runOptions).length > 0 ? runOptions : undefined,\n\t\t);\n\n\t\tif (stream && result instanceof ReadableStream) {\n\t\t\t// Workers AI returns an SSE stream with `data: {\"response\":\"chunk\"}` format.\n\t\t\t// Transform it to OpenAI-compatible SSE format.\n\t\t\tconst transformed = transformWorkersAiStream(\n\t\t\t\tresult as ReadableStream<Uint8Array>,\n\t\t\t\tmodel,\n\t\t\t);\n\t\t\treturn new Response(transformed, {\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"text/event-stream\",\n\t\t\t\t\t\"Cache-Control\": \"no-cache\",\n\t\t\t\t},\n\t\t\t});\n\t\t}\n\n\t\t// Graceful degradation: some models return a complete (non-streaming)\n\t\t// response even when `stream: true` is requested. Fall through to the\n\t\t// non-streaming wrapper which produces a valid OpenAI Chat Completion\n\t\t// response that the SDK can consume.\n\n\t\t// Non-streaming: Workers AI returns { response: \"text\", tool_calls?: [...] }\n\t\t// Wrap into OpenAI Chat Completion format.\n\t\tconst responseObj =\n\t\t\ttypeof result === \"object\" && result !== null\n\t\t\t\t? (result as Record<string, unknown>)\n\t\t\t\t: { response: String(result) };\n\n\t\tconst responseText =\n\t\t\ttypeof responseObj.response === \"string\"\n\t\t\t\t? responseObj.response\n\t\t\t\t: typeof responseObj.response === \"object\" && responseObj.response !== null\n\t\t\t\t\t? JSON.stringify(responseObj.response)\n\t\t\t\t\t: \"\";\n\n\t\tconst message: Record<string, unknown> = {\n\t\t\trole: \"assistant\",\n\t\t\tcontent: responseText,\n\t\t};\n\t\tlet finishReason = \"stop\";\n\n\t\t// Handle tool calls if present in Workers AI response\n\t\tif (Array.isArray(responseObj.tool_calls) && responseObj.tool_calls.length > 0) {\n\t\t\tfinishReason = \"tool_calls\";\n\t\t\tmessage.tool_calls = responseObj.tool_calls.map(\n\t\t\t\t(tc: {\n\t\t\t\t\tid?: string;\n\t\t\t\t\tname?: string;\n\t\t\t\t\targuments: unknown;\n\t\t\t\t\tfunction?: { name: string; arguments?: unknown };\n\t\t\t\t}) => ({\n\t\t\t\t\tid: tc.id || crypto.randomUUID(),\n\t\t\t\t\ttype: \"function\",\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function?.name || tc.name || \"\",\n\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\ttypeof (tc.function?.arguments ?? tc.arguments) === \"string\"\n\t\t\t\t\t\t\t\t? ((tc.function?.arguments ?? tc.arguments) as string)\n\t\t\t\t\t\t\t\t: JSON.stringify(tc.function?.arguments ?? tc.arguments ?? {}),\n\t\t\t\t\t},\n\t\t\t\t}),\n\t\t\t);\n\t\t}\n\n\t\tconst openAiResponse = {\n\t\t\tid: `workers-ai-${crypto.randomUUID()}`,\n\t\t\tobject: \"chat.completion\",\n\t\t\tcreated: Math.floor(Date.now() / 1000),\n\t\t\tmodel,\n\t\t\tchoices: [{ index: 0, message, finish_reason: finishReason }],\n\t\t};\n\n\t\treturn new Response(JSON.stringify(openAiResponse), {\n\t\t\theaders: { \"Content-Type\": \"application/json\" },\n\t\t});\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// Stream transformer: Workers AI SSE -> OpenAI-compatible SSE\n// Uses TransformStream for proper backpressure.\n// ---------------------------------------------------------------------------\n\n/**\n * Transforms a Workers AI SSE stream (data: {\"response\":\"chunk\"}) into\n * an OpenAI-compatible SSE stream (data: {\"choices\":[{\"delta\":{\"content\":\"chunk\"}}]}).\n *\n * Workers AI binding streams tool calls in an OpenAI-like nested format:\n * { tool_calls: [{ id, type, index, function: { name, arguments } }] }\n * Arguments are streamed incrementally across multiple SSE chunks, so the\n * transformer must forward them as incremental deltas rather than a single blob.\n */\nfunction transformWorkersAiStream(\n\tsource: ReadableStream<Uint8Array>,\n\tmodel: string,\n): ReadableStream<Uint8Array> {\n\tconst decoder = new TextDecoder();\n\tconst encoder = new TextEncoder();\n\t// Generate a stable ID and timestamp for the entire stream, matching OpenAI's\n\t// convention where all chunks in a single response share the same id/created.\n\tconst streamId = `workers-ai-${crypto.randomUUID()}`;\n\tconst created = Math.floor(Date.now() / 1000);\n\tlet buffer = \"\";\n\tlet hasToolCalls = false;\n\t// When true, the source stream is already in OpenAI format (some models\n\t// like Qwen3, Kimi K2.5 stream OpenAI-compatible SSE through the binding).\n\t// In that case, flush() should only emit [DONE] and skip the finish chunk.\n\tlet isOpenAiFormat = false;\n\t// Track tool call state per index: store the generated/assigned ID so that\n\t// subsequent argument deltas use the same ID (matching the working streaming.ts pattern).\n\tconst toolCallState = new Map<number, { id: string; name: string }>();\n\n\treturn source.pipeThrough(\n\t\tnew TransformStream<Uint8Array, Uint8Array>({\n\t\t\ttransform(chunk, controller) {\n\t\t\t\tbuffer += decoder.decode(chunk, { stream: true });\n\t\t\t\tconst lines = buffer.split(\"\\n\");\n\t\t\t\tbuffer = lines.pop() || \"\";\n\n\t\t\t\tfor (const line of lines) {\n\t\t\t\t\tconst trimmed = line.trim();\n\t\t\t\t\tif (!trimmed || !trimmed.startsWith(\"data: \")) continue;\n\t\t\t\t\tconst data = trimmed.slice(6);\n\n\t\t\t\t\t// Swallow source [DONE]; we emit our own in flush()\n\t\t\t\t\tif (data === \"[DONE]\") continue;\n\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst parsed = JSON.parse(data);\n\n\t\t\t\t\t\t// Some models (Qwen3, Kimi K2.5) return OpenAI-compatible format\n\t\t\t\t\t\t// directly through the binding, with `choices[].delta.content` and\n\t\t\t\t\t\t// optional `reasoning_content`. Detect this and pass through as-is.\n\t\t\t\t\t\tif (parsed.choices !== undefined) {\n\t\t\t\t\t\t\t// Already OpenAI format — pass through but ensure each tool call\n\t\t\t\t\t\t\t// index gets a unique, stable ID across all chunks.\n\t\t\t\t\t\t\tisOpenAiFormat = true;\n\t\t\t\t\t\t\tconst choice = parsed.choices?.[0];\n\t\t\t\t\t\t\tif (choice?.delta?.tool_calls) {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t\tfor (const tc of choice.delta.tool_calls) {\n\t\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\t\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t\t// First chunk for this index — generate/store unique ID\n\t\t\t\t\t\t\t\t\t\tconst id = tc.id || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, {\n\t\t\t\t\t\t\t\t\t\t\tid,\n\t\t\t\t\t\t\t\t\t\t\tname: tc.function?.name || \"\",\n\t\t\t\t\t\t\t\t\t\t});\n\t\t\t\t\t\t\t\t\t\ttc.id = id;\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\t// Subsequent chunk — reuse stored ID, remove id from delta\n\t\t\t\t\t\t\t\t\t\t// (OpenAI format only sends id in first chunk)\n\t\t\t\t\t\t\t\t\t\tdelete tc.id;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (choice?.finish_reason === \"tool_calls\") {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(parsed)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// --- Workers AI native format handling below ---\n\n\t\t\t\t\t\t// Text content\n\t\t\t\t\t\tif (parsed.response != null && parsed.response !== \"\") {\n\t\t\t\t\t\t\tconst openAiChunk = {\n\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\tdelta: { content: parsed.response },\n\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(openAiChunk)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Tool calls — Workers AI binding streams these incrementally:\n\t\t\t\t\t\t// Chunk A: { id, type, index, function: { name } } — start\n\t\t\t\t\t\t// Chunk B: { index, function: { arguments: \"partial...\" } } — args delta\n\t\t\t\t\t\t// Chunk C: { index, function: { arguments: \"rest...\" } } — args delta\n\t\t\t\t\t\t// Chunk D: { id: null, type: null, index, function: { name: null, arguments: \"\" } } — finalize (skip)\n\t\t\t\t\t\tif (Array.isArray(parsed.tool_calls) && parsed.tool_calls.length > 0) {\n\t\t\t\t\t\t\tfor (const tc of parsed.tool_calls) {\n\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\n\t\t\t\t\t\t\t\t// Resolve name and arguments from either nested or flat format\n\t\t\t\t\t\t\t\tconst tcName = tc.function?.name ?? tc.name ?? null;\n\t\t\t\t\t\t\t\tconst tcArgs = tc.function?.arguments ?? tc.arguments ?? null;\n\t\t\t\t\t\t\t\tconst tcId = tc.id ?? null;\n\n\t\t\t\t\t\t\t\t// Skip finalization chunks where everything is null/empty\n\t\t\t\t\t\t\t\tif (!tcId && !tcName && (!tcArgs || tcArgs === \"\")) continue;\n\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\n\t\t\t\t\t\t\t\t// Build the OpenAI-compatible tool_calls delta\n\t\t\t\t\t\t\t\tconst toolCallDelta: Record<string, unknown> = {\n\t\t\t\t\t\t\t\t\tindex: tcIndex,\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t// First chunk for this tool call index — emit id, type, name.\n\t\t\t\t\t\t\t\t\tconst id = tcId || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, { id, name: tcName || \"\" });\n\t\t\t\t\t\t\t\t\ttoolCallDelta.id = id;\n\t\t\t\t\t\t\t\t\ttoolCallDelta.type = \"function\";\n\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\tname: tcName || \"\",\n\t\t\t\t\t\t\t\t\t\t// Include arguments if they arrive in the same chunk\n\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\ttcArgs != null\n\t\t\t\t\t\t\t\t\t\t\t\t? typeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs)\n\t\t\t\t\t\t\t\t\t\t\t\t: \"\",\n\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t// Subsequent chunks — only include arguments delta\n\t\t\t\t\t\t\t\t\tif (tcArgs != null && tcArgs !== \"\") {\n\t\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\t\ttypeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs),\n\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\tcontinue; // Nothing useful to forward\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst toolChunk = {\n\t\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\t\tdelta: { tool_calls: [toolCallDelta] },\n\t\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(toolChunk)}\\n\\n`),\n\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (e) {\n\t\t\t\t\t\t// Log malformed SSE events for debugging; don't break the stream.\n\t\t\t\t\t\tconsole.warn(\"[tanstack-ai] failed to parse SSE event:\", data, e);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t\tflush(controller) {\n\t\t\t\tif (!isOpenAiFormat) {\n\t\t\t\t\t// Workers AI native format: emit a finish chunk with stop/tool_calls\n\t\t\t\t\tconst finalChunk = {\n\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\tcreated,\n\t\t\t\t\t\tmodel,\n\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\tdelta: {},\n\t\t\t\t\t\t\t\tfinish_reason: hasToolCalls ? \"tool_calls\" : \"stop\",\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t};\n\t\t\t\t\tcontroller.enqueue(encoder.encode(`data: ${JSON.stringify(finalChunk)}\\n\\n`));\n\t\t\t\t}\n\t\t\t\t// OpenAI format already includes its own finish_reason in the stream.\n\t\t\t\t// Either way, emit a [DONE] sentinel.\n\t\t\t\tcontroller.enqueue(encoder.encode(\"data: [DONE]\\n\\n\"));\n\t\t\t},\n\t\t}),\n\t);\n}\n"],"mappings":"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"}
@@ -109,6 +109,8 @@ function createWorkersAiBindingFetch(binding, options) {
109
109
  if (typeof body.max_tokens === "number") inputs.max_tokens = body.max_tokens;
110
110
  if (body.response_format) inputs.response_format = body.response_format;
111
111
  if (stream) inputs.stream = true;
112
+ if (body.reasoning_effort !== void 0) inputs.reasoning_effort = body.reasoning_effort;
113
+ if (body.chat_template_kwargs !== void 0) inputs.chat_template_kwargs = body.chat_template_kwargs;
112
114
  const runOptions = {};
113
115
  if (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;
114
116
  if (init?.signal) runOptions.signal = init.signal;
@@ -278,4 +280,4 @@ function transformWorkersAiStream(source, model) {
278
280
  //#endregion
279
281
  export { validateWorkersAiConfig as a, isDirectCredentialsConfig as i, createWorkersAiBindingFetch as n, isDirectBindingConfig as r, createGatewayFetch as t };
280
282
 
281
- //# sourceMappingURL=create-fetcher-DY7wfYYy.mjs.map
283
+ //# sourceMappingURL=create-fetcher-CeUOJgrh.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-fetcher-CeUOJgrh.mjs","names":[],"sources":["../src/utils/create-fetcher.ts"],"sourcesContent":["// ---------------------------------------------------------------------------\n// AI Gateway types (for third-party providers + Workers AI through gateway)\n// ---------------------------------------------------------------------------\n\nexport interface CloudflareAiGateway {\n\trun(request: unknown): Promise<Response>;\n}\n\nexport interface AiGatewayBindingConfig {\n\t/**\n\t * The AI Gateway binding\n\t * @example\n\t * env.AI.gateway('my-gateway-id')\n\t */\n\tbinding: CloudflareAiGateway;\n\t/**\n\t * The Provider API Key if you want to manually pass it, ignore if using Unified Billing or BYOK.\n\t */\n\tapiKey?: string;\n}\n\nexport type AiGatewayCredentialsConfig = {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The AI Gateway ID\n\t */\n\tgatewayId: string;\n} & (\n\t| {\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey: string;\n\t\t\tapiKey?: string;\n\t }\n\t| {\n\t\t\t/** Provider API Key */\n\t\t\tapiKey: string;\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey?: string;\n\t }\n);\n\nexport interface AiGatewayConfig {\n\tskipCache?: boolean;\n\tcacheTtl?: number;\n\tcustomCacheKey?: string;\n\tmetadata?: Record<string, unknown>;\n}\n\nexport type AiGatewayAdapterConfig = (AiGatewayBindingConfig | AiGatewayCredentialsConfig) &\n\tAiGatewayConfig;\n\n// ---------------------------------------------------------------------------\n// Plain Workers AI types (direct binding or REST, no gateway)\n// ---------------------------------------------------------------------------\n\n/**\n * The Workers AI binding interface (env.AI).\n * Accepts a model name and inputs, returns results directly.\n * Includes `gateway()` which is present on `env.AI` but not on `env.AI.gateway(id)`,\n * enabling structural discrimination from `CloudflareAiGateway`.\n */\nexport interface WorkersAiBinding {\n\trun(\n\t\tmodel: string,\n\t\tinputs: Record<string, unknown>,\n\t\toptions?: Record<string, unknown>,\n\t): Promise<unknown>;\n\tgateway(gatewayId: string): CloudflareAiGateway;\n}\n\nexport interface WorkersAiDirectBindingConfig {\n\t/**\n\t * The Workers AI binding (env.AI).\n\t * @example\n\t * { binding: env.AI }\n\t */\n\tbinding: WorkersAiBinding;\n}\n\nexport interface WorkersAiDirectCredentialsConfig {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The Cloudflare API key for Workers AI\n\t */\n\tapiKey: string;\n}\n\n/**\n * Config for Workers AI adapters. Supports four modes:\n * - Plain binding: `{ binding: env.AI }`\n * - Plain REST: `{ accountId, apiKey }`\n * - AI Gateway binding: `{ binding: env.AI.gateway(id) }`\n * - AI Gateway REST: `{ accountId, gatewayId, ... }`\n *\n * The third union member intersects `AiGatewayAdapterConfig` with `{ apiKey?: string }`.\n * For the gateway binding variant, `AiGatewayBindingConfig` already includes `apiKey?`,\n * so the intersection is redundant there. For the gateway credentials variant, this\n * `apiKey` represents the Workers AI token (used in the `Authorization` header to the\n * upstream provider), distinct from `cfApiKey` (used in the `cf-aig-authorization`\n * header for authenticated gateways).\n */\nexport type WorkersAiAdapterConfig = (\n\t| WorkersAiDirectBindingConfig\n\t| WorkersAiDirectCredentialsConfig\n\t| (AiGatewayAdapterConfig & { apiKey?: string })\n) & {\n\t/**\n\t * Session affinity key for prefix-cache optimization.\n\t * Routes requests with the same key to the same backend replica.\n\t */\n\tsessionAffinity?: string;\n};\n\n// ---------------------------------------------------------------------------\n// Config detection helpers\n// ---------------------------------------------------------------------------\n\n/** Returns true if this is a plain Workers AI binding config (`{ binding: env.AI }`) */\nexport function isDirectBindingConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectBindingConfig {\n\t// env.AI has a .gateway() method; env.AI.gateway(id) does not.\n\t// This distinguishes direct bindings from AI Gateway bindings.\n\treturn (\n\t\t\"binding\" in config &&\n\t\ttypeof (config.binding as unknown as Record<string, unknown>).gateway === \"function\"\n\t);\n}\n\n/** Returns true if this is a plain Workers AI REST config (accountId + apiKey, no gatewayId) */\nexport function isDirectCredentialsConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectCredentialsConfig {\n\treturn \"accountId\" in config && \"apiKey\" in config && !(\"gatewayId\" in config);\n}\n\n/** Returns true if this is an AI Gateway config (has gateway binding or `gatewayId`) */\nexport function isGatewayConfig(config: WorkersAiAdapterConfig): config is AiGatewayAdapterConfig {\n\tif (\"gatewayId\" in config) return true;\n\t// Has `binding` but NOT a direct Workers AI binding (no .gateway method)\n\treturn \"binding\" in config && !isDirectBindingConfig(config);\n}\n\n// ---------------------------------------------------------------------------\n// Config validation\n// ---------------------------------------------------------------------------\n\n/**\n * Validates that a WorkersAiAdapterConfig contains a valid configuration.\n * Throws an error if neither a binding, credentials (accountId + apiKey),\n * nor a gateway configuration is provided.\n */\nexport function validateWorkersAiConfig(config: WorkersAiAdapterConfig): void {\n\tif (\n\t\t!isDirectBindingConfig(config) &&\n\t\t!isDirectCredentialsConfig(config) &&\n\t\t!isGatewayConfig(config)\n\t) {\n\t\tthrow new Error(\n\t\t\t\"Invalid Workers AI configuration: you must provide either a binding (e.g. { binding: env.AI }), \" +\n\t\t\t\t\"credentials ({ accountId, apiKey }), or a gateway configuration ({ binding: env.AI.gateway(id) } \" +\n\t\t\t\t\"or { accountId, gatewayId }).\",\n\t\t);\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// createGatewayFetch -- for routing through AI Gateway\n// ---------------------------------------------------------------------------\n\nexport function createGatewayFetch(\n\tprovider: string,\n\tconfig: AiGatewayAdapterConfig,\n\theaders: Record<string, string> = {},\n): typeof fetch {\n\treturn (input, init) => {\n\t\tlet query: Record<string, unknown> = {};\n\n\t\tconst url =\n\t\t\ttypeof input === \"string\" ? input : input instanceof URL ? input.href : input.url;\n\t\tconst urlObj = new URL(url);\n\n\t\t// Extract endpoint path (remove /v1/ prefix if present)\n\t\tconst endpoint = urlObj.pathname.replace(/^\\/v1\\//, \"\").replace(/^\\//, \"\") + urlObj.search;\n\n\t\tif (init?.body) {\n\t\t\ttry {\n\t\t\t\tquery = JSON.parse(init.body as string);\n\t\t\t} catch {\n\t\t\t\tquery = { _raw: init.body };\n\t\t\t}\n\t\t}\n\n\t\tconst cacheHeaders: Record<string, string> = {};\n\n\t\tif (\"skipCache\" in config && config.skipCache) {\n\t\t\tcacheHeaders[\"cf-aig-skip-cache\"] = \"true\";\n\t\t}\n\n\t\tif (typeof config.cacheTtl === \"number\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-ttl\"] = String(config.cacheTtl);\n\t\t}\n\n\t\tif (typeof config.customCacheKey === \"string\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-key\"] = config.customCacheKey;\n\t\t}\n\n\t\tif (typeof config.metadata === \"object\") {\n\t\t\tcacheHeaders[\"cf-aig-metadata\"] = JSON.stringify(config.metadata);\n\t\t}\n\n\t\tconst request: {\n\t\t\tprovider: string;\n\t\t\tendpoint: string;\n\t\t\theaders: Record<string, string>;\n\t\t\tquery: Record<string, unknown>;\n\t\t} = {\n\t\t\tprovider,\n\t\t\tendpoint,\n\t\t\theaders: {\n\t\t\t\t...(init?.headers as Record<string, string> | undefined),\n\t\t\t\t...headers,\n\t\t\t\t...cacheHeaders,\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tquery,\n\t\t};\n\n\t\tif (provider === \"workers-ai\") {\n\t\t\tif (!request.endpoint.startsWith(\"run/\")) {\n\t\t\t\trequest.endpoint = `run/${query.model}`;\n\t\t\t}\n\t\t\tdelete query.model;\n\t\t\tdelete query.instructions;\n\t\t}\n\n\t\tif (config.apiKey) {\n\t\t\trequest.headers[\"authorization\"] = `Bearer ${config.apiKey}`;\n\t\t}\n\n\t\tif (\"binding\" in config) {\n\t\t\treturn (\n\t\t\t\tconfig.binding as {\n\t\t\t\t\trun(req: unknown, opts?: { signal?: AbortSignal }): Promise<Response>;\n\t\t\t\t}\n\t\t\t).run(request, { signal: init?.signal ?? undefined });\n\t\t}\n\n\t\treturn fetch(\n\t\t\t`https://gateway.ai.cloudflare.com/v1/${config.accountId}/${config.gatewayId}`,\n\t\t\t{\n\t\t\t\t...init,\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t\t\t...headers,\n\t\t\t\t\t...cacheHeaders,\n\t\t\t\t\t...(config.cfApiKey\n\t\t\t\t\t\t? { \"cf-aig-authorization\": `Bearer ${config.cfApiKey}` }\n\t\t\t\t\t\t: {}),\n\t\t\t\t},\n\t\t\t\tbody: JSON.stringify(request),\n\t\t\t},\n\t\t);\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// createWorkersAiBindingFetch -- shim that makes env.AI look like an OpenAI endpoint\n// ---------------------------------------------------------------------------\n\n/**\n * Normalize messages before passing to Workers AI binding.\n *\n * The binding has strict schema validation that may differ from the OpenAI API:\n * - `content` must not be null\n *\n * Content arrays (with image_url parts) are passed through as-is since the\n * Workers AI binding accepts them at runtime for vision-capable models.\n */\nfunction normalizeMessagesForBinding(\n\tmessages: Record<string, unknown>[],\n): Record<string, unknown>[] {\n\treturn messages.map((msg) => {\n\t\tconst normalized = { ...msg };\n\n\t\t// content: null → content: \"\"\n\t\tif (normalized.content === null || normalized.content === undefined) {\n\t\t\tnormalized.content = \"\";\n\t\t}\n\n\t\treturn normalized;\n\t});\n}\n\n/**\n * Creates a fetch function that intercepts OpenAI SDK requests and translates them\n * to Workers AI binding calls (env.AI.run). This allows the WorkersAiTextAdapter\n * to use the OpenAI SDK against a plain Workers AI binding.\n *\n * NOTE: The `input` URL parameter is intentionally ignored. The model name and all\n * request parameters are extracted from the JSON body, matching Workers AI's\n * `binding.run(model, inputs)` calling convention.\n */\nexport function createWorkersAiBindingFetch(\n\tbinding: WorkersAiBinding,\n\toptions?: { extraHeaders?: Record<string, string> },\n): typeof fetch {\n\treturn async (_input, init) => {\n\t\tif (!init?.body) {\n\t\t\treturn new Response(\"No body\", { status: 400 });\n\t\t}\n\n\t\tlet body: Record<string, unknown>;\n\t\ttry {\n\t\t\tbody = JSON.parse(init.body as string);\n\t\t} catch {\n\t\t\treturn new Response(\"Invalid JSON body\", { status: 400 });\n\t\t}\n\n\t\tconst model = body.model as string;\n\t\tconst stream = body.stream as boolean | undefined;\n\n\t\t// Build Workers AI inputs from OpenAI format\n\t\tconst inputs: Record<string, unknown> = {};\n\t\tif (body.messages) {\n\t\t\tinputs.messages = normalizeMessagesForBinding(\n\t\t\t\tbody.messages as Record<string, unknown>[],\n\t\t\t);\n\t\t}\n\t\tif (body.tools) inputs.tools = body.tools;\n\t\tif (typeof body.temperature === \"number\") inputs.temperature = body.temperature;\n\t\tif (typeof body.max_tokens === \"number\") inputs.max_tokens = body.max_tokens;\n\t\tif (body.response_format) inputs.response_format = body.response_format;\n\t\tif (stream) inputs.stream = true;\n\n\t\t// Workers AI-specific reasoning controls. These belong on the INPUTS object\n\t\t// passed to binding.run(model, inputs), not on the options (3rd) arg.\n\t\t// See https://github.com/cloudflare/ai/issues/503.\n\t\t//\n\t\t// `reasoning_effort: null` is a valid value (disables reasoning on models\n\t\t// that support it), so we check `!== undefined` rather than truthiness.\n\t\tif (body.reasoning_effort !== undefined) {\n\t\t\tinputs.reasoning_effort = body.reasoning_effort;\n\t\t}\n\t\tif (body.chat_template_kwargs !== undefined) {\n\t\t\tinputs.chat_template_kwargs = body.chat_template_kwargs;\n\t\t}\n\n\t\tconst runOptions: Record<string, unknown> = {};\n\t\tif (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;\n\t\tif (init?.signal) runOptions.signal = init.signal;\n\n\t\tconst result = await binding.run(\n\t\t\tmodel,\n\t\t\tinputs,\n\t\t\tObject.keys(runOptions).length > 0 ? runOptions : undefined,\n\t\t);\n\n\t\tif (stream && result instanceof ReadableStream) {\n\t\t\t// Workers AI returns an SSE stream with `data: {\"response\":\"chunk\"}` format.\n\t\t\t// Transform it to OpenAI-compatible SSE format.\n\t\t\tconst transformed = transformWorkersAiStream(\n\t\t\t\tresult as ReadableStream<Uint8Array>,\n\t\t\t\tmodel,\n\t\t\t);\n\t\t\treturn new Response(transformed, {\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"text/event-stream\",\n\t\t\t\t\t\"Cache-Control\": \"no-cache\",\n\t\t\t\t},\n\t\t\t});\n\t\t}\n\n\t\t// Graceful degradation: some models return a complete (non-streaming)\n\t\t// response even when `stream: true` is requested. Fall through to the\n\t\t// non-streaming wrapper which produces a valid OpenAI Chat Completion\n\t\t// response that the SDK can consume.\n\n\t\t// Non-streaming: Workers AI returns { response: \"text\", tool_calls?: [...] }\n\t\t// Wrap into OpenAI Chat Completion format.\n\t\tconst responseObj =\n\t\t\ttypeof result === \"object\" && result !== null\n\t\t\t\t? (result as Record<string, unknown>)\n\t\t\t\t: { response: String(result) };\n\n\t\tconst responseText =\n\t\t\ttypeof responseObj.response === \"string\"\n\t\t\t\t? responseObj.response\n\t\t\t\t: typeof responseObj.response === \"object\" && responseObj.response !== null\n\t\t\t\t\t? JSON.stringify(responseObj.response)\n\t\t\t\t\t: \"\";\n\n\t\tconst message: Record<string, unknown> = {\n\t\t\trole: \"assistant\",\n\t\t\tcontent: responseText,\n\t\t};\n\t\tlet finishReason = \"stop\";\n\n\t\t// Handle tool calls if present in Workers AI response\n\t\tif (Array.isArray(responseObj.tool_calls) && responseObj.tool_calls.length > 0) {\n\t\t\tfinishReason = \"tool_calls\";\n\t\t\tmessage.tool_calls = responseObj.tool_calls.map(\n\t\t\t\t(tc: {\n\t\t\t\t\tid?: string;\n\t\t\t\t\tname?: string;\n\t\t\t\t\targuments: unknown;\n\t\t\t\t\tfunction?: { name: string; arguments?: unknown };\n\t\t\t\t}) => ({\n\t\t\t\t\tid: tc.id || crypto.randomUUID(),\n\t\t\t\t\ttype: \"function\",\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function?.name || tc.name || \"\",\n\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\ttypeof (tc.function?.arguments ?? tc.arguments) === \"string\"\n\t\t\t\t\t\t\t\t? ((tc.function?.arguments ?? tc.arguments) as string)\n\t\t\t\t\t\t\t\t: JSON.stringify(tc.function?.arguments ?? tc.arguments ?? {}),\n\t\t\t\t\t},\n\t\t\t\t}),\n\t\t\t);\n\t\t}\n\n\t\tconst openAiResponse = {\n\t\t\tid: `workers-ai-${crypto.randomUUID()}`,\n\t\t\tobject: \"chat.completion\",\n\t\t\tcreated: Math.floor(Date.now() / 1000),\n\t\t\tmodel,\n\t\t\tchoices: [{ index: 0, message, finish_reason: finishReason }],\n\t\t};\n\n\t\treturn new Response(JSON.stringify(openAiResponse), {\n\t\t\theaders: { \"Content-Type\": \"application/json\" },\n\t\t});\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// Stream transformer: Workers AI SSE -> OpenAI-compatible SSE\n// Uses TransformStream for proper backpressure.\n// ---------------------------------------------------------------------------\n\n/**\n * Transforms a Workers AI SSE stream (data: {\"response\":\"chunk\"}) into\n * an OpenAI-compatible SSE stream (data: {\"choices\":[{\"delta\":{\"content\":\"chunk\"}}]}).\n *\n * Workers AI binding streams tool calls in an OpenAI-like nested format:\n * { tool_calls: [{ id, type, index, function: { name, arguments } }] }\n * Arguments are streamed incrementally across multiple SSE chunks, so the\n * transformer must forward them as incremental deltas rather than a single blob.\n */\nfunction transformWorkersAiStream(\n\tsource: ReadableStream<Uint8Array>,\n\tmodel: string,\n): ReadableStream<Uint8Array> {\n\tconst decoder = new TextDecoder();\n\tconst encoder = new TextEncoder();\n\t// Generate a stable ID and timestamp for the entire stream, matching OpenAI's\n\t// convention where all chunks in a single response share the same id/created.\n\tconst streamId = `workers-ai-${crypto.randomUUID()}`;\n\tconst created = Math.floor(Date.now() / 1000);\n\tlet buffer = \"\";\n\tlet hasToolCalls = false;\n\t// When true, the source stream is already in OpenAI format (some models\n\t// like Qwen3, Kimi K2.5 stream OpenAI-compatible SSE through the binding).\n\t// In that case, flush() should only emit [DONE] and skip the finish chunk.\n\tlet isOpenAiFormat = false;\n\t// Track tool call state per index: store the generated/assigned ID so that\n\t// subsequent argument deltas use the same ID (matching the working streaming.ts pattern).\n\tconst toolCallState = new Map<number, { id: string; name: string }>();\n\n\treturn source.pipeThrough(\n\t\tnew TransformStream<Uint8Array, Uint8Array>({\n\t\t\ttransform(chunk, controller) {\n\t\t\t\tbuffer += decoder.decode(chunk, { stream: true });\n\t\t\t\tconst lines = buffer.split(\"\\n\");\n\t\t\t\tbuffer = lines.pop() || \"\";\n\n\t\t\t\tfor (const line of lines) {\n\t\t\t\t\tconst trimmed = line.trim();\n\t\t\t\t\tif (!trimmed || !trimmed.startsWith(\"data: \")) continue;\n\t\t\t\t\tconst data = trimmed.slice(6);\n\n\t\t\t\t\t// Swallow source [DONE]; we emit our own in flush()\n\t\t\t\t\tif (data === \"[DONE]\") continue;\n\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst parsed = JSON.parse(data);\n\n\t\t\t\t\t\t// Some models (Qwen3, Kimi K2.5) return OpenAI-compatible format\n\t\t\t\t\t\t// directly through the binding, with `choices[].delta.content` and\n\t\t\t\t\t\t// optional `reasoning_content`. Detect this and pass through as-is.\n\t\t\t\t\t\tif (parsed.choices !== undefined) {\n\t\t\t\t\t\t\t// Already OpenAI format — pass through but ensure each tool call\n\t\t\t\t\t\t\t// index gets a unique, stable ID across all chunks.\n\t\t\t\t\t\t\tisOpenAiFormat = true;\n\t\t\t\t\t\t\tconst choice = parsed.choices?.[0];\n\t\t\t\t\t\t\tif (choice?.delta?.tool_calls) {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t\tfor (const tc of choice.delta.tool_calls) {\n\t\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\t\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t\t// First chunk for this index — generate/store unique ID\n\t\t\t\t\t\t\t\t\t\tconst id = tc.id || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, {\n\t\t\t\t\t\t\t\t\t\t\tid,\n\t\t\t\t\t\t\t\t\t\t\tname: tc.function?.name || \"\",\n\t\t\t\t\t\t\t\t\t\t});\n\t\t\t\t\t\t\t\t\t\ttc.id = id;\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\t// Subsequent chunk — reuse stored ID, remove id from delta\n\t\t\t\t\t\t\t\t\t\t// (OpenAI format only sends id in first chunk)\n\t\t\t\t\t\t\t\t\t\tdelete tc.id;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (choice?.finish_reason === \"tool_calls\") {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(parsed)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// --- Workers AI native format handling below ---\n\n\t\t\t\t\t\t// Text content\n\t\t\t\t\t\tif (parsed.response != null && parsed.response !== \"\") {\n\t\t\t\t\t\t\tconst openAiChunk = {\n\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\tdelta: { content: parsed.response },\n\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(openAiChunk)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Tool calls — Workers AI binding streams these incrementally:\n\t\t\t\t\t\t// Chunk A: { id, type, index, function: { name } } — start\n\t\t\t\t\t\t// Chunk B: { index, function: { arguments: \"partial...\" } } — args delta\n\t\t\t\t\t\t// Chunk C: { index, function: { arguments: \"rest...\" } } — args delta\n\t\t\t\t\t\t// Chunk D: { id: null, type: null, index, function: { name: null, arguments: \"\" } } — finalize (skip)\n\t\t\t\t\t\tif (Array.isArray(parsed.tool_calls) && parsed.tool_calls.length > 0) {\n\t\t\t\t\t\t\tfor (const tc of parsed.tool_calls) {\n\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\n\t\t\t\t\t\t\t\t// Resolve name and arguments from either nested or flat format\n\t\t\t\t\t\t\t\tconst tcName = tc.function?.name ?? tc.name ?? null;\n\t\t\t\t\t\t\t\tconst tcArgs = tc.function?.arguments ?? tc.arguments ?? null;\n\t\t\t\t\t\t\t\tconst tcId = tc.id ?? null;\n\n\t\t\t\t\t\t\t\t// Skip finalization chunks where everything is null/empty\n\t\t\t\t\t\t\t\tif (!tcId && !tcName && (!tcArgs || tcArgs === \"\")) continue;\n\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\n\t\t\t\t\t\t\t\t// Build the OpenAI-compatible tool_calls delta\n\t\t\t\t\t\t\t\tconst toolCallDelta: Record<string, unknown> = {\n\t\t\t\t\t\t\t\t\tindex: tcIndex,\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t// First chunk for this tool call index — emit id, type, name.\n\t\t\t\t\t\t\t\t\tconst id = tcId || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, { id, name: tcName || \"\" });\n\t\t\t\t\t\t\t\t\ttoolCallDelta.id = id;\n\t\t\t\t\t\t\t\t\ttoolCallDelta.type = \"function\";\n\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\tname: tcName || \"\",\n\t\t\t\t\t\t\t\t\t\t// Include arguments if they arrive in the same chunk\n\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\ttcArgs != null\n\t\t\t\t\t\t\t\t\t\t\t\t? typeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs)\n\t\t\t\t\t\t\t\t\t\t\t\t: \"\",\n\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t// Subsequent chunks — only include arguments delta\n\t\t\t\t\t\t\t\t\tif (tcArgs != null && tcArgs !== \"\") {\n\t\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\t\ttypeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs),\n\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\tcontinue; // Nothing useful to forward\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst toolChunk = {\n\t\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\t\tdelta: { tool_calls: [toolCallDelta] },\n\t\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(toolChunk)}\\n\\n`),\n\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (e) {\n\t\t\t\t\t\t// Log malformed SSE events for debugging; don't break the stream.\n\t\t\t\t\t\tconsole.warn(\"[tanstack-ai] failed to parse SSE event:\", data, e);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t\tflush(controller) {\n\t\t\t\tif (!isOpenAiFormat) {\n\t\t\t\t\t// Workers AI native format: emit a finish chunk with stop/tool_calls\n\t\t\t\t\tconst finalChunk = {\n\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\tcreated,\n\t\t\t\t\t\tmodel,\n\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\tdelta: {},\n\t\t\t\t\t\t\t\tfinish_reason: hasToolCalls ? \"tool_calls\" : \"stop\",\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t};\n\t\t\t\t\tcontroller.enqueue(encoder.encode(`data: ${JSON.stringify(finalChunk)}\\n\\n`));\n\t\t\t\t}\n\t\t\t\t// OpenAI format already includes its own finish_reason in the stream.\n\t\t\t\t// Either way, emit a [DONE] sentinel.\n\t\t\t\tcontroller.enqueue(encoder.encode(\"data: [DONE]\\n\\n\"));\n\t\t\t},\n\t\t}),\n\t);\n}\n"],"mappings":"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"}
package/dist/index.cjs CHANGED
@@ -1,5 +1,5 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- const require_workers_ai = require("./workers-ai-Bm7Up4or.cjs");
2
+ const require_workers_ai = require("./workers-ai-BOZ5iyhY.cjs");
3
3
  const require_adapters_anthropic = require("./adapters/anthropic.cjs");
4
4
  const require_adapters_gemini = require("./adapters/gemini.cjs");
5
5
  const require_adapters_grok = require("./adapters/grok.cjs");
package/dist/index.d.cts CHANGED
@@ -8,5 +8,5 @@ import { WorkersAiImageModel, createWorkersAiImage } from "./adapters/workers-ai
8
8
  import { WorkersAiSummarizeModel, createWorkersAiSummarize } from "./adapters/workers-ai-summarize.cjs";
9
9
  import { WorkersAiTranscriptionModel, createWorkersAiTranscription } from "./adapters/workers-ai-transcription.cjs";
10
10
  import { WorkersAiTTSModel, createWorkersAiTts } from "./adapters/workers-ai-tts.cjs";
11
- import { WorkersAiTextModel, createWorkersAiChat } from "./adapters/workers-ai.cjs";
12
- export { ANTHROPIC_MODELS, type AiGatewayAdapterConfig, type AnthropicChatModel, type AnthropicGatewayConfig, GROK_CHAT_MODELS, GROK_IMAGE_MODELS, type GeminiChatModel, type GeminiGatewayConfig, type GeminiImageModel, GeminiImageModels, type GeminiSummarizeModel, GeminiSummarizeModels, type GeminiTTSModel, GeminiTTSModels, type GeminiTextModel, GeminiTextModels, type GrokChatModel, type GrokGatewayConfig, type GrokImageModel, type GrokSummarizeModel, OPENAI_CHAT_MODELS, OPENAI_IMAGE_MODELS, OPENAI_TRANSCRIPTION_MODELS, OPENAI_TTS_MODELS, OPENAI_VIDEO_MODELS, type OpenAIChatModel, type OpenAIImageModel, type OpenAITTSModel, type OpenAITranscriptionModel, type OpenAIVideoModel, type OpenAiGatewayConfig, type OpenRouterChatModel, type OpenRouterGatewayConfig, type OpenRouterImageModel, type OpenRouterSummarizeModel, type WorkersAiAdapterConfig, type WorkersAiImageModel, type WorkersAiSummarizeModel, type WorkersAiTTSModel, type WorkersAiTextModel, type WorkersAiTranscriptionModel, createAnthropicChat, createAnthropicSummarize, createGeminiChat, createGeminiImage, createGeminiSummarize, createGeminiTts, createGrokChat, createGrokImage, createGrokSummarize, createOpenAiChat, createOpenAiImage, createOpenAiSummarize, createOpenAiTranscription, createOpenAiTts, createOpenAiVideo, createOpenRouterChat, createOpenRouterImage, createOpenRouterSummarize, createWorkersAiChat, createWorkersAiImage, createWorkersAiSummarize, createWorkersAiTranscription, createWorkersAiTts };
11
+ import { WorkersAiTextModel, WorkersAiTextModelOptions, createWorkersAiChat } from "./adapters/workers-ai.cjs";
12
+ export { ANTHROPIC_MODELS, type AiGatewayAdapterConfig, type AnthropicChatModel, type AnthropicGatewayConfig, GROK_CHAT_MODELS, GROK_IMAGE_MODELS, type GeminiChatModel, type GeminiGatewayConfig, type GeminiImageModel, GeminiImageModels, type GeminiSummarizeModel, GeminiSummarizeModels, type GeminiTTSModel, GeminiTTSModels, type GeminiTextModel, GeminiTextModels, type GrokChatModel, type GrokGatewayConfig, type GrokImageModel, type GrokSummarizeModel, OPENAI_CHAT_MODELS, OPENAI_IMAGE_MODELS, OPENAI_TRANSCRIPTION_MODELS, OPENAI_TTS_MODELS, OPENAI_VIDEO_MODELS, type OpenAIChatModel, type OpenAIImageModel, type OpenAITTSModel, type OpenAITranscriptionModel, type OpenAIVideoModel, type OpenAiGatewayConfig, type OpenRouterChatModel, type OpenRouterGatewayConfig, type OpenRouterImageModel, type OpenRouterSummarizeModel, type WorkersAiAdapterConfig, type WorkersAiImageModel, type WorkersAiSummarizeModel, type WorkersAiTTSModel, type WorkersAiTextModel, type WorkersAiTextModelOptions, type WorkersAiTranscriptionModel, createAnthropicChat, createAnthropicSummarize, createGeminiChat, createGeminiImage, createGeminiSummarize, createGeminiTts, createGrokChat, createGrokImage, createGrokSummarize, createOpenAiChat, createOpenAiImage, createOpenAiSummarize, createOpenAiTranscription, createOpenAiTts, createOpenAiVideo, createOpenRouterChat, createOpenRouterImage, createOpenRouterSummarize, createWorkersAiChat, createWorkersAiImage, createWorkersAiSummarize, createWorkersAiTranscription, createWorkersAiTts };
package/dist/index.d.mts CHANGED
@@ -8,5 +8,5 @@ import { WorkersAiImageModel, createWorkersAiImage } from "./adapters/workers-ai
8
8
  import { WorkersAiSummarizeModel, createWorkersAiSummarize } from "./adapters/workers-ai-summarize.mjs";
9
9
  import { WorkersAiTranscriptionModel, createWorkersAiTranscription } from "./adapters/workers-ai-transcription.mjs";
10
10
  import { WorkersAiTTSModel, createWorkersAiTts } from "./adapters/workers-ai-tts.mjs";
11
- import { WorkersAiTextModel, createWorkersAiChat } from "./adapters/workers-ai.mjs";
12
- export { ANTHROPIC_MODELS, type AiGatewayAdapterConfig, type AnthropicChatModel, type AnthropicGatewayConfig, GROK_CHAT_MODELS, GROK_IMAGE_MODELS, type GeminiChatModel, type GeminiGatewayConfig, type GeminiImageModel, GeminiImageModels, type GeminiSummarizeModel, GeminiSummarizeModels, type GeminiTTSModel, GeminiTTSModels, type GeminiTextModel, GeminiTextModels, type GrokChatModel, type GrokGatewayConfig, type GrokImageModel, type GrokSummarizeModel, OPENAI_CHAT_MODELS, OPENAI_IMAGE_MODELS, OPENAI_TRANSCRIPTION_MODELS, OPENAI_TTS_MODELS, OPENAI_VIDEO_MODELS, type OpenAIChatModel, type OpenAIImageModel, type OpenAITTSModel, type OpenAITranscriptionModel, type OpenAIVideoModel, type OpenAiGatewayConfig, type OpenRouterChatModel, type OpenRouterGatewayConfig, type OpenRouterImageModel, type OpenRouterSummarizeModel, type WorkersAiAdapterConfig, type WorkersAiImageModel, type WorkersAiSummarizeModel, type WorkersAiTTSModel, type WorkersAiTextModel, type WorkersAiTranscriptionModel, createAnthropicChat, createAnthropicSummarize, createGeminiChat, createGeminiImage, createGeminiSummarize, createGeminiTts, createGrokChat, createGrokImage, createGrokSummarize, createOpenAiChat, createOpenAiImage, createOpenAiSummarize, createOpenAiTranscription, createOpenAiTts, createOpenAiVideo, createOpenRouterChat, createOpenRouterImage, createOpenRouterSummarize, createWorkersAiChat, createWorkersAiImage, createWorkersAiSummarize, createWorkersAiTranscription, createWorkersAiTts };
11
+ import { WorkersAiTextModel, WorkersAiTextModelOptions, createWorkersAiChat } from "./adapters/workers-ai.mjs";
12
+ export { ANTHROPIC_MODELS, type AiGatewayAdapterConfig, type AnthropicChatModel, type AnthropicGatewayConfig, GROK_CHAT_MODELS, GROK_IMAGE_MODELS, type GeminiChatModel, type GeminiGatewayConfig, type GeminiImageModel, GeminiImageModels, type GeminiSummarizeModel, GeminiSummarizeModels, type GeminiTTSModel, GeminiTTSModels, type GeminiTextModel, GeminiTextModels, type GrokChatModel, type GrokGatewayConfig, type GrokImageModel, type GrokSummarizeModel, OPENAI_CHAT_MODELS, OPENAI_IMAGE_MODELS, OPENAI_TRANSCRIPTION_MODELS, OPENAI_TTS_MODELS, OPENAI_VIDEO_MODELS, type OpenAIChatModel, type OpenAIImageModel, type OpenAITTSModel, type OpenAITranscriptionModel, type OpenAIVideoModel, type OpenAiGatewayConfig, type OpenRouterChatModel, type OpenRouterGatewayConfig, type OpenRouterImageModel, type OpenRouterSummarizeModel, type WorkersAiAdapterConfig, type WorkersAiImageModel, type WorkersAiSummarizeModel, type WorkersAiTTSModel, type WorkersAiTextModel, type WorkersAiTextModelOptions, type WorkersAiTranscriptionModel, createAnthropicChat, createAnthropicSummarize, createGeminiChat, createGeminiImage, createGeminiSummarize, createGeminiTts, createGrokChat, createGrokImage, createGrokSummarize, createOpenAiChat, createOpenAiImage, createOpenAiSummarize, createOpenAiTranscription, createOpenAiTts, createOpenAiVideo, createOpenRouterChat, createOpenRouterImage, createOpenRouterSummarize, createWorkersAiChat, createWorkersAiImage, createWorkersAiSummarize, createWorkersAiTranscription, createWorkersAiTts };
@@ -20,7 +20,7 @@ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__ge
20
20
  enumerable: true
21
21
  }) : target, mod));
22
22
  //#endregion
23
- const require_create_fetcher = require("./create-fetcher-5iL34e6H.cjs");
23
+ const require_create_fetcher = require("./create-fetcher-By-hTiT9.cjs");
24
24
  const require_defineProperty = require("./defineProperty-DQoAg20E.cjs");
25
25
  let _tanstack_ai_adapters = require("@tanstack/ai/adapters");
26
26
  let openai = require("openai");
@@ -151,6 +151,12 @@ function buildOpenAITools(tools) {
151
151
  function generateId(prefix = "chatcmpl") {
152
152
  return `${prefix}-${crypto.randomUUID().replace(/-/g, "").slice(0, 16)}`;
153
153
  }
154
+ function normalizeModelOptions(modelOptions) {
155
+ if (modelOptions === null || typeof modelOptions !== "object" || Array.isArray(modelOptions)) return {};
156
+ const out = {};
157
+ for (const [key, value] of Object.entries(modelOptions)) if (value !== void 0) out[key] = value;
158
+ return out;
159
+ }
154
160
  var WorkersAiTextAdapter = class extends _tanstack_ai_adapters.BaseTextAdapter {
155
161
  constructor(model, config) {
156
162
  super({ apiKey: "unused" }, model);
@@ -159,7 +165,8 @@ var WorkersAiTextAdapter = class extends _tanstack_ai_adapters.BaseTextAdapter {
159
165
  this.client = buildWorkersAiClient(config);
160
166
  }
161
167
  async *chatStream(options) {
162
- const { systemPrompts, messages, tools, temperature, maxTokens, model } = options;
168
+ const { systemPrompts, messages, tools, temperature, maxTokens, model, modelOptions } = options;
169
+ const extraBody = normalizeModelOptions(modelOptions);
163
170
  const openAIMessages = buildOpenAIMessages(systemPrompts, messages);
164
171
  const openAITools = buildOpenAITools(tools);
165
172
  const timestamp = Date.now();
@@ -177,6 +184,7 @@ var WorkersAiTextAdapter = class extends _tanstack_ai_adapters.BaseTextAdapter {
177
184
  let stream;
178
185
  try {
179
186
  stream = await this.client.chat.completions.create({
187
+ ...extraBody,
180
188
  model: model ?? this.model,
181
189
  messages: openAIMessages,
182
190
  tools: openAITools,
@@ -188,6 +196,7 @@ var WorkersAiTextAdapter = class extends _tanstack_ai_adapters.BaseTextAdapter {
188
196
  } catch (streamError) {
189
197
  console.warn("[tanstack-ai] Streaming failed, falling back to non-streaming:", streamError instanceof Error ? streamError.message : streamError);
190
198
  const nonStreamResult = await this.client.chat.completions.create({
199
+ ...extraBody,
191
200
  model: model ?? this.model,
192
201
  messages: openAIMessages,
193
202
  tools: openAITools,
@@ -446,9 +455,11 @@ var WorkersAiTextAdapter = class extends _tanstack_ai_adapters.BaseTextAdapter {
446
455
  }
447
456
  async structuredOutput(options) {
448
457
  const { outputSchema, chatOptions } = options;
449
- const { systemPrompts, messages, temperature, model } = chatOptions;
458
+ const { systemPrompts, messages, temperature, model, modelOptions } = chatOptions;
459
+ const extraBody = normalizeModelOptions(modelOptions);
450
460
  const openAIMessages = buildOpenAIMessages(systemPrompts, messages, { includeToolMessages: false });
451
461
  const response = await this.client.chat.completions.create({
462
+ ...extraBody,
452
463
  model: model ?? this.model,
453
464
  messages: openAIMessages,
454
465
  temperature,
@@ -507,4 +518,4 @@ Object.defineProperty(exports, "createWorkersAiChat", {
507
518
  }
508
519
  });
509
520
 
510
- //# sourceMappingURL=workers-ai-Bm7Up4or.cjs.map
521
+ //# sourceMappingURL=workers-ai-BOZ5iyhY.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workers-ai-BOZ5iyhY.cjs","names":["isDirectBindingConfig","OpenAI","createWorkersAiBindingFetch","isDirectCredentialsConfig","createGatewayFetch","BaseTextAdapter"],"sources":["../src/adapters/workers-ai.ts"],"sourcesContent":["import type { AiModels, BaseAiTextGeneration } from \"@cloudflare/workers-types\";\nimport type { ContentPart, ModelMessage, StreamChunk, TextOptions } from \"@tanstack/ai\";\nimport {\n\tBaseTextAdapter,\n\ttype StructuredOutputOptions,\n\ttype StructuredOutputResult,\n} from \"@tanstack/ai/adapters\";\nimport OpenAI from \"openai\";\nimport {\n\ttype WorkersAiAdapterConfig,\n\ttype AiGatewayAdapterConfig,\n\tcreateGatewayFetch,\n\tcreateWorkersAiBindingFetch,\n\tisDirectBindingConfig,\n\tisDirectCredentialsConfig,\n\tvalidateWorkersAiConfig,\n} from \"../utils/create-fetcher\";\n\n// ---------------------------------------------------------------------------\n// Model types derived from @cloudflare/workers-types\n// ---------------------------------------------------------------------------\n\nexport type WorkersAiTextModel =\n\t| {\n\t\t\t[K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never;\n\t }[keyof AiModels]\n\t| (string & {});\n\n// ---------------------------------------------------------------------------\n// Provider-specific options forwarded to Workers AI's chat completions inputs.\n//\n// These correspond to fields on `ChatCompletionsCommonOptions` in\n// `@cloudflare/workers-types`. They are passed verbatim into the request body\n// sent to the Workers AI binding / REST endpoint / AI Gateway, and ultimately\n// land on the `inputs` object of `binding.run(model, inputs)`.\n//\n// Pass via `modelOptions` on a per-call basis:\n//\n// await adapter.chatStream({\n// model, messages,\n// modelOptions: {\n// reasoning_effort: \"low\",\n// chat_template_kwargs: { enable_thinking: false },\n// },\n// });\n// ---------------------------------------------------------------------------\n\nexport interface WorkersAiTextModelOptions {\n\t/**\n\t * Controls the reasoning budget for reasoning-capable models\n\t * (e.g. `@cf/zai-org/glm-4.7-flash`, `@cf/moonshotai/kimi-k2.5`,\n\t * `@cf/openai/gpt-oss-120b`).\n\t *\n\t * `null` is a valid value and disables reasoning for models that support it.\n\t */\n\treasoning_effort?: \"low\" | \"medium\" | \"high\" | null;\n\t/**\n\t * Chat-template overrides for reasoning-capable models that expose\n\t * thinking toggles (e.g. GLM, Kimi).\n\t */\n\tchat_template_kwargs?: {\n\t\t/** Whether to enable reasoning. Enabled by default on reasoning models. */\n\t\tenable_thinking?: boolean;\n\t\t/** If false, preserves reasoning context between turns. */\n\t\tclear_thinking?: boolean;\n\t};\n\t/**\n\t * Escape hatch for other Workers AI inputs-level parameters.\n\t *\n\t * Keys placed here are merged into the outbound request body and forwarded\n\t * to the underlying transport (binding / REST / gateway). Only fields that\n\t * the binding shim knows about are extracted for direct `env.AI` bindings;\n\t * everything is passed through on REST and gateway paths.\n\t */\n\t[key: string]: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Helpers: build the right OpenAI client depending on config mode\n// ---------------------------------------------------------------------------\n\nfunction buildWorkersAiClient(config: WorkersAiAdapterConfig): OpenAI {\n\tvalidateWorkersAiConfig(config);\n\n\tconst sessionHeaders: Record<string, string> | undefined = config.sessionAffinity\n\t\t? { \"x-session-affinity\": config.sessionAffinity }\n\t\t: undefined;\n\n\tif (isDirectBindingConfig(config)) {\n\t\t// Plain binding mode: shim translates OpenAI fetch calls to env.AI.run()\n\t\treturn new OpenAI({\n\t\t\tapiKey: \"unused\",\n\t\t\tfetch: createWorkersAiBindingFetch(\n\t\t\t\tconfig.binding,\n\t\t\t\tsessionHeaders ? { extraHeaders: sessionHeaders } : undefined,\n\t\t\t),\n\t\t});\n\t}\n\n\tif (isDirectCredentialsConfig(config)) {\n\t\t// Plain REST mode: point OpenAI SDK at Workers AI's OpenAI-compatible endpoint\n\t\treturn new OpenAI({\n\t\t\tbaseURL: `https://api.cloudflare.com/client/v4/accounts/${config.accountId}/ai/v1`,\n\t\t\tapiKey: config.apiKey,\n\t\t\tdefaultHeaders: sessionHeaders,\n\t\t});\n\t}\n\n\t// Gateway mode (existing): use createGatewayFetch\n\tconst gatewayConfig = config as AiGatewayAdapterConfig;\n\treturn new OpenAI({\n\t\tfetch: createGatewayFetch(\"workers-ai\", gatewayConfig, sessionHeaders),\n\t\tapiKey: gatewayConfig.apiKey ?? \"unused\",\n\t});\n}\n\n// ---------------------------------------------------------------------------\n// Shared message-building helpers\n// ---------------------------------------------------------------------------\n\nfunction extractTextContent(content: ModelMessage[\"content\"]): string {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n}\n\n/**\n * Convert a single TanStack AI {@link ContentPart} to the OpenAI Chat\n * Completions multi-modal format.\n *\n * TODO: handle other content types (audio, video, document)\n */\nfunction convertContentPart(part: ContentPart): OpenAI.Chat.ChatCompletionContentPart | undefined {\n\tswitch (part.type) {\n\t\tcase \"text\":\n\t\t\tif (part.content) {\n\t\t\t\treturn { type: \"text\", text: part.content };\n\t\t\t}\n\t\t\treturn undefined;\n\t\tcase \"image\": {\n\t\t\tlet url: string;\n\t\t\tif (part.source.type === \"data\") {\n\t\t\t\turl = part.source.value.startsWith(\"data:\")\n\t\t\t\t\t? part.source.value\n\t\t\t\t\t: `data:${part.source.mimeType};base64,${part.source.value}`;\n\t\t\t} else {\n\t\t\t\turl = part.source.value;\n\t\t\t}\n\t\t\treturn { type: \"image_url\", image_url: { url } };\n\t\t}\n\t\tdefault:\n\t\t\t// audio, video, document — not supported for now\n\t\t\tconsole.warn(\n\t\t\t\t`[@cloudflare/tanstack-ai] Unsupported content part type \"${part.type}\" — skipping`,\n\t\t\t);\n\t\t\treturn undefined;\n\t}\n}\n\n/**\n * Build OpenAI-compatible user message content from TanStack AI content.\n *\n * If the content has only text parts, returns a plain string.\n * If it includes image parts, returns an array of content parts in\n * OpenAI's multi-modal format (text + image_url).\n */\nfunction buildUserContent(\n\tcontent: ModelMessage[\"content\"],\n): string | OpenAI.Chat.ChatCompletionContentPart[] {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\n\tconst hasImages = content.some((p) => p.type === \"image\");\n\tif (!hasImages) {\n\t\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n\t}\n\n\tconst parts: OpenAI.Chat.ChatCompletionContentPart[] = [];\n\tfor (const part of content) {\n\t\tconst converted = convertContentPart(part);\n\t\tif (converted) {\n\t\t\tparts.push(converted);\n\t\t}\n\t}\n\treturn parts;\n}\n\nfunction buildOpenAIMessages(\n\tsystemPrompts: string[] | undefined,\n\tmessages: ModelMessage[],\n\toptions?: { includeToolMessages?: boolean },\n): OpenAI.Chat.ChatCompletionMessageParam[] {\n\tconst includeTools = options?.includeToolMessages ?? true;\n\tconst openAIMessages: OpenAI.Chat.ChatCompletionMessageParam[] = [];\n\n\tif (systemPrompts && systemPrompts.length > 0) {\n\t\topenAIMessages.push({\n\t\t\trole: \"system\",\n\t\t\tcontent: systemPrompts.join(\"\\n\"),\n\t\t});\n\t}\n\n\tfor (const message of messages) {\n\t\tif (message.role === \"user\") {\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"user\",\n\t\t\t\tcontent: buildUserContent(message.content),\n\t\t\t});\n\t\t} else if (message.role === \"assistant\") {\n\t\t\tconst assistantMessage: OpenAI.Chat.ChatCompletionAssistantMessageParam = {\n\t\t\t\trole: \"assistant\",\n\t\t\t\tcontent: extractTextContent(message.content),\n\t\t\t};\n\t\t\tif (includeTools && message.toolCalls && message.toolCalls.length > 0) {\n\t\t\t\tassistantMessage.tool_calls = message.toolCalls.map((tc) => ({\n\t\t\t\t\tid: tc.id,\n\t\t\t\t\ttype: \"function\" as const,\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function.name,\n\t\t\t\t\t\targuments: tc.function.arguments,\n\t\t\t\t\t},\n\t\t\t\t}));\n\t\t\t}\n\t\t\topenAIMessages.push(assistantMessage);\n\t\t} else if (includeTools && message.role === \"tool\") {\n\t\t\tlet toolContent: string;\n\t\t\tif (typeof message.content === \"string\") {\n\t\t\t\ttry {\n\t\t\t\t\tJSON.parse(message.content);\n\t\t\t\t\ttoolContent = message.content;\n\t\t\t\t} catch {\n\t\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t}\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"tool\",\n\t\t\t\ttool_call_id: message.toolCallId || `tool_${crypto.randomUUID().slice(0, 8)}`,\n\t\t\t\tcontent: toolContent,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn openAIMessages;\n}\n\nfunction buildOpenAITools(\n\ttools: Array<{ name: string; description: string; inputSchema?: unknown }> | undefined,\n): OpenAI.Chat.ChatCompletionTool[] | undefined {\n\tif (!tools) return undefined;\n\treturn tools.map((tool) => ({\n\t\ttype: \"function\" as const,\n\t\tfunction: {\n\t\t\tname: tool.name,\n\t\t\tdescription: tool.description,\n\t\t\tparameters: tool.inputSchema as Record<string, unknown>,\n\t\t},\n\t}));\n}\n\n// ---------------------------------------------------------------------------\n// ID generation\n// ---------------------------------------------------------------------------\n\nfunction generateId(prefix = \"chatcmpl\"): string {\n\treturn `${prefix}-${crypto.randomUUID().replace(/-/g, \"\").slice(0, 16)}`;\n}\n\n// ---------------------------------------------------------------------------\n// modelOptions normalization\n//\n// Users pass Workers AI-specific chat params (e.g. `reasoning_effort`,\n// `chat_template_kwargs`) via `modelOptions`. We merge these into the outbound\n// request body so they reach the binding / REST / gateway transports.\n//\n// Spread order matters: these go FIRST in the request body so that TanStack-AI\n// managed fields (`model`, `messages`, `temperature`, `max_tokens`, `stream`,\n// `tools`, `response_format`, ...) always win if a user accidentally sets\n// them both at the top level and inside `modelOptions`.\n//\n// `undefined` values are stripped so that JSON.stringify (and our binding shim\n// which does `!== undefined` checks) see them as absent. `null` values are\n// preserved — they're meaningful for fields like `reasoning_effort: null`\n// which explicitly disables reasoning on some models.\n// ---------------------------------------------------------------------------\nfunction normalizeModelOptions(\n\tmodelOptions: WorkersAiTextModelOptions | undefined,\n): Record<string, unknown> {\n\t// Guard against runtime misuse. TanStack AI types this as an object, but\n\t// users can always bypass with `as any`. `Object.entries` on a string\n\t// surprisingly returns per-character tuples (e.g. `Object.entries(\"ab\") →\n\t// [[\"0\",\"a\"],[\"1\",\"b\"]]`), which would leak spurious keys into the body.\n\t// Arrays similarly become index-keyed. Only accept plain objects.\n\tif (modelOptions === null || typeof modelOptions !== \"object\" || Array.isArray(modelOptions)) {\n\t\treturn {};\n\t}\n\tconst out: Record<string, unknown> = {};\n\tfor (const [key, value] of Object.entries(modelOptions)) {\n\t\tif (value !== undefined) out[key] = value;\n\t}\n\treturn out;\n}\n\n// ---------------------------------------------------------------------------\n// WorkersAiTextAdapter: chat / structured output via OpenAI Chat Completions\n// ---------------------------------------------------------------------------\n\nexport class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<\n\tTModel,\n\tWorkersAiTextModelOptions,\n\t// eslint-disable-next-line @typescript-eslint/no-explicit-any -- remaining BaseTextAdapter generics are opaque\n\tany,\n\tany\n> {\n\tname = \"workers-ai\" as const;\n\n\tprivate client: OpenAI;\n\n\tconstructor(model: TModel, config: WorkersAiAdapterConfig) {\n\t\tsuper({ apiKey: \"unused\" }, model);\n\t\tthis.client = buildWorkersAiClient(config);\n\t}\n\n\tasync *chatStream(options: TextOptions<WorkersAiTextModelOptions>): AsyncIterable<StreamChunk> {\n\t\tconst { systemPrompts, messages, tools, temperature, maxTokens, model, modelOptions } =\n\t\t\toptions;\n\t\tconst extraBody = normalizeModelOptions(modelOptions);\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages);\n\t\tconst openAITools = buildOpenAITools(tools);\n\n\t\tconst timestamp = Date.now();\n\t\tconst runId = generateId();\n\t\tconst messageId = generateId();\n\t\tlet hasEmittedRunStarted = false;\n\t\tlet hasEmittedTextMessageStart = false;\n\t\tlet accumulatedContent = \"\";\n\t\tlet hasEmittedStepStarted = false;\n\t\tlet accumulatedReasoning = \"\";\n\t\tconst stepId = generateId();\n\t\tlet hasReceivedFinishReason = false;\n\t\tconst toolCallsInProgress = new Map<\n\t\t\tnumber,\n\t\t\t{ id: string; name: string; arguments: string; started: boolean }\n\t\t>();\n\n\t\ttry {\n\t\t\tlet stream: AsyncIterable<OpenAI.Chat.Completions.ChatCompletionChunk>;\n\t\t\ttry {\n\t\t\t\tstream = await this.client.chat.completions.create({\n\t\t\t\t\t...extraBody,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t\tstream: true,\n\t\t\t\t\tstream_options: { include_usage: true },\n\t\t\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsStreaming);\n\t\t\t} catch (streamError: unknown) {\n\t\t\t\t// Some models (e.g. GPT-OSS) don't support streaming via the REST API.\n\t\t\t\t// Fall back to a non-streaming call and yield the result as a single chunk.\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Streaming failed, falling back to non-streaming:\",\n\t\t\t\t\tstreamError instanceof Error ? streamError.message : streamError,\n\t\t\t\t);\n\t\t\t\tconst nonStreamResult = await this.client.chat.completions.create({\n\t\t\t\t\t...extraBody,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsNonStreaming);\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\n\t\t\t\tconst msg = nonStreamResult.choices[0]?.message;\n\t\t\t\tif (msg?.content) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: msg.content,\n\t\t\t\t\t\tcontent: msg.content,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tif (msg?.tool_calls) {\n\t\t\t\t\tfor (const tc of msg.tool_calls) {\n\t\t\t\t\t\tif (tc.type !== \"function\") continue;\n\t\t\t\t\t\tconst fn = tc.function;\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = fn.arguments ? JSON.parse(fn.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst finishReason = nonStreamResult.choices[0]?.finish_reason;\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tusage: nonStreamResult.usage\n\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\tpromptTokens: nonStreamResult.usage.prompt_tokens,\n\t\t\t\t\t\t\t\tcompletionTokens: nonStreamResult.usage.completion_tokens,\n\t\t\t\t\t\t\t\ttotalTokens: nonStreamResult.usage.total_tokens,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t: undefined,\n\t\t\t\t\tfinishReason:\n\t\t\t\t\t\tfinishReason === \"tool_calls\" || finishReason === \"function_call\"\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: ((finishReason as \"stop\" | \"length\" | \"content_filter\") ?? \"stop\"),\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor await (const chunk of stream) {\n\t\t\t\tif (!chunk.choices || chunk.choices.length === 0) continue;\n\t\t\t\tconst choice = chunk.choices[0];\n\t\t\t\tif (!choice) continue;\n\n\t\t\t\t// Emit RUN_STARTED on first chunk\n\t\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\t\thasEmittedRunStarted = true;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tconst delta = choice.delta;\n\n\t\t\t\t// Reasoning content (used by models like QwQ, DeepSeek R1, Kimi K2.5)\n\t\t\t\t// The OpenAI SDK doesn't type this field, but models send it as an extension.\n\t\t\t\tconst reasoningContent = ((delta as Record<string, unknown>).reasoning_content ??\n\t\t\t\t\t(delta as Record<string, unknown>).reasoning) as string | undefined;\n\t\t\t\tif (reasoningContent) {\n\t\t\t\t\t// RUN_STARTED is already guaranteed by the guard above\n\t\t\t\t\tif (!hasEmittedStepStarted) {\n\t\t\t\t\t\thasEmittedStepStarted = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"STEP_STARTED\",\n\t\t\t\t\t\t\tstepId,\n\t\t\t\t\t\t\tstepType: \"thinking\",\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t\taccumulatedReasoning += reasoningContent;\n\t\t\t\t\t// TODO: TanStack AI's StreamProcessor currently treats STEP_FINISHED as an\n\t\t\t\t\t// incremental reasoning event (with `delta` + accumulated `content`), so we\n\t\t\t\t\t// emit one per token. If TanStack AI adds a dedicated STEP_CONTENT event\n\t\t\t\t\t// type, this should be updated to emit STEP_CONTENT per token and a single\n\t\t\t\t\t// STEP_FINISHED when reasoning ends (i.e. when the first non-reasoning\n\t\t\t\t\t// content or finish_reason arrives).\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"STEP_FINISHED\",\n\t\t\t\t\t\tstepId,\n\t\t\t\t\t\tdelta: reasoningContent,\n\t\t\t\t\t\tcontent: accumulatedReasoning,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Text content\n\t\t\t\tif (delta.content) {\n\t\t\t\t\tif (!hasEmittedTextMessageStart) {\n\t\t\t\t\t\thasEmittedTextMessageStart = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\taccumulatedContent += delta.content;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: delta.content,\n\t\t\t\t\t\tcontent: accumulatedContent,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Tool calls\n\t\t\t\tif (delta.tool_calls) {\n\t\t\t\t\tfor (const toolCallDelta of delta.tool_calls) {\n\t\t\t\t\t\tconst index = toolCallDelta.index;\n\n\t\t\t\t\t\tif (!toolCallsInProgress.has(index)) {\n\t\t\t\t\t\t\t// Always generate a unique ID per tool call index.\n\t\t\t\t\t\t\t// The backend may send the same ID for multiple tool calls,\n\t\t\t\t\t\t\t// so we cannot trust toolCallDelta.id to be unique.\n\t\t\t\t\t\t\ttoolCallsInProgress.set(index, {\n\t\t\t\t\t\t\t\tid: generateId(\"chatcmpl-tool\"),\n\t\t\t\t\t\t\t\tname: toolCallDelta.function?.name || \"\",\n\t\t\t\t\t\t\t\targuments: \"\",\n\t\t\t\t\t\t\t\tstarted: false,\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst toolCall = toolCallsInProgress.get(index)!;\n\n\t\t\t\t\t\t// Only update name if provided (ID is already set at creation time\n\t\t\t\t\t\t// and should not be overwritten by subsequent chunks that may have\n\t\t\t\t\t\t// duplicate/shared IDs from the backend)\n\t\t\t\t\t\tif (toolCallDelta.function?.name) {\n\t\t\t\t\t\t\ttoolCall.name = toolCallDelta.function.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments) {\n\t\t\t\t\t\t\ttoolCall.arguments += toolCallDelta.function.arguments;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Emit TOOL_CALL_START once we have id and name\n\t\t\t\t\t\tif (toolCall.id && toolCall.name && !toolCall.started) {\n\t\t\t\t\t\t\ttoolCall.started = true;\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tindex,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Stream tool call arguments\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments && toolCall.started) {\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_ARGS\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tdelta: toolCallDelta.function.arguments,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Finish\n\t\t\t\tif (choice.finish_reason) {\n\t\t\t\t\thasReceivedFinishReason = true;\n\n\t\t\t\t\t// End tool calls\n\t\t\t\t\tif (choice.finish_reason === \"tool_calls\" || toolCallsInProgress.size > 0) {\n\t\t\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\t\tparsedInput = toolCall.arguments\n\t\t\t\t\t\t\t\t\t? JSON.parse(toolCall.arguments)\n\t\t\t\t\t\t\t\t\t: {};\n\t\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tconst computedFinishReason =\n\t\t\t\t\t\tchoice.finish_reason === \"tool_calls\" ||\n\t\t\t\t\t\tchoice.finish_reason === \"function_call\" ||\n\t\t\t\t\t\ttoolCallsInProgress.size > 0\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: (choice.finish_reason as \"stop\" | \"length\" | \"content_filter\");\n\n\t\t\t\t\t// End text message if started\n\t\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Emit RUN_FINISHED\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tusage: chunk.usage\n\t\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\t\tpromptTokens: chunk.usage.prompt_tokens,\n\t\t\t\t\t\t\t\t\tcompletionTokens: chunk.usage.completion_tokens,\n\t\t\t\t\t\t\t\t\ttotalTokens: chunk.usage.total_tokens,\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t: undefined,\n\t\t\t\t\t\tfinishReason: computedFinishReason,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Premature stream termination: the stream ended without a finish_reason.\n\t\t\t// This can happen when Workers AI truncates a response or the connection drops.\n\t\t\t// Emit proper closing events so the consumer doesn't hang.\n\t\t\tif (hasEmittedRunStarted && !hasReceivedFinishReason) {\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Stream ended without finish_reason — possible truncation or connection drop\",\n\t\t\t\t);\n\n\t\t\t\t// Close any open tool calls\n\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\tif (toolCall.started) {\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = toolCall.arguments ? JSON.parse(toolCall.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Close text message if open\n\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tfinishReason: \"stop\",\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\tconst code =\n\t\t\t\terror instanceof Error ? (error as Error & { code?: string }).code : undefined;\n\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t\tyield {\n\t\t\t\ttype: \"RUN_ERROR\",\n\t\t\t\trunId,\n\t\t\t\tmodel: model ?? this.model,\n\t\t\t\ttimestamp,\n\t\t\t\terror: {\n\t\t\t\t\tmessage: message || \"Unknown error\",\n\t\t\t\t\tcode,\n\t\t\t\t},\n\t\t\t} satisfies StreamChunk;\n\t\t}\n\t}\n\n\tasync structuredOutput(\n\t\toptions: StructuredOutputOptions<WorkersAiTextModelOptions>,\n\t): Promise<StructuredOutputResult<unknown>> {\n\t\tconst { outputSchema, chatOptions } = options;\n\t\tconst { systemPrompts, messages, temperature, model, modelOptions } = chatOptions;\n\t\tconst extraBody = normalizeModelOptions(modelOptions);\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages, {\n\t\t\tincludeToolMessages: false,\n\t\t});\n\n\t\tconst response = await this.client.chat.completions.create({\n\t\t\t...extraBody,\n\t\t\tmodel: model ?? this.model,\n\t\t\tmessages: openAIMessages,\n\t\t\ttemperature,\n\t\t\tstream: false,\n\t\t\tresponse_format: {\n\t\t\t\ttype: \"json_schema\",\n\t\t\t\tjson_schema: {\n\t\t\t\t\tname: \"structured_output\",\n\t\t\t\t\tstrict: true,\n\t\t\t\t\tschema: outputSchema,\n\t\t\t\t},\n\t\t\t},\n\t\t} as OpenAI.Chat.Completions.ChatCompletionCreateParamsNonStreaming);\n\n\t\tconst choice = response.choices?.[0];\n\n\t\tif (!choice) {\n\t\t\tthrow new Error(\n\t\t\t\t`Workers AI structured output returned no choices: ${JSON.stringify(response)}`,\n\t\t\t);\n\t\t}\n\n\t\tconst rawContent = choice.message?.content ?? \"\";\n\n\t\t// Workers AI REST endpoint may return `content` as an already-parsed object\n\t\t// when using json_schema response format, so normalise both cases.\n\t\tlet data: unknown;\n\t\tlet rawText: string;\n\n\t\tif (typeof rawContent === \"string\") {\n\t\t\trawText = rawContent;\n\t\t\ttry {\n\t\t\t\tdata = JSON.parse(rawText);\n\t\t\t} catch {\n\t\t\t\tdata = rawText;\n\t\t\t}\n\t\t} else {\n\t\t\t// Already an object — stringify for rawText, use directly for data\n\t\t\tdata = rawContent;\n\t\t\trawText = JSON.stringify(rawContent);\n\t\t}\n\n\t\treturn { data, rawText };\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// Factory functions\n// ---------------------------------------------------------------------------\n\nexport function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig) {\n\treturn new WorkersAiTextAdapter(model, config);\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@cloudflare/tanstack-ai",
3
- "version": "0.1.7",
3
+ "version": "0.1.8",
4
4
  "description": "Use TanStack AI with Cloudflare Workers AI and AI Gateway",
5
5
  "keywords": [
6
6
  "ai",
@@ -1 +0,0 @@
1
- {"version":3,"file":"create-fetcher-5iL34e6H.cjs","names":[],"sources":["../src/utils/create-fetcher.ts"],"sourcesContent":["// ---------------------------------------------------------------------------\n// AI Gateway types (for third-party providers + Workers AI through gateway)\n// ---------------------------------------------------------------------------\n\nexport interface CloudflareAiGateway {\n\trun(request: unknown): Promise<Response>;\n}\n\nexport interface AiGatewayBindingConfig {\n\t/**\n\t * The AI Gateway binding\n\t * @example\n\t * env.AI.gateway('my-gateway-id')\n\t */\n\tbinding: CloudflareAiGateway;\n\t/**\n\t * The Provider API Key if you want to manually pass it, ignore if using Unified Billing or BYOK.\n\t */\n\tapiKey?: string;\n}\n\nexport type AiGatewayCredentialsConfig = {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The AI Gateway ID\n\t */\n\tgatewayId: string;\n} & (\n\t| {\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey: string;\n\t\t\tapiKey?: string;\n\t }\n\t| {\n\t\t\t/** Provider API Key */\n\t\t\tapiKey: string;\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey?: string;\n\t }\n);\n\nexport interface AiGatewayConfig {\n\tskipCache?: boolean;\n\tcacheTtl?: number;\n\tcustomCacheKey?: string;\n\tmetadata?: Record<string, unknown>;\n}\n\nexport type AiGatewayAdapterConfig = (AiGatewayBindingConfig | AiGatewayCredentialsConfig) &\n\tAiGatewayConfig;\n\n// ---------------------------------------------------------------------------\n// Plain Workers AI types (direct binding or REST, no gateway)\n// ---------------------------------------------------------------------------\n\n/**\n * The Workers AI binding interface (env.AI).\n * Accepts a model name and inputs, returns results directly.\n * Includes `gateway()` which is present on `env.AI` but not on `env.AI.gateway(id)`,\n * enabling structural discrimination from `CloudflareAiGateway`.\n */\nexport interface WorkersAiBinding {\n\trun(\n\t\tmodel: string,\n\t\tinputs: Record<string, unknown>,\n\t\toptions?: Record<string, unknown>,\n\t): Promise<unknown>;\n\tgateway(gatewayId: string): CloudflareAiGateway;\n}\n\nexport interface WorkersAiDirectBindingConfig {\n\t/**\n\t * The Workers AI binding (env.AI).\n\t * @example\n\t * { binding: env.AI }\n\t */\n\tbinding: WorkersAiBinding;\n}\n\nexport interface WorkersAiDirectCredentialsConfig {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The Cloudflare API key for Workers AI\n\t */\n\tapiKey: string;\n}\n\n/**\n * Config for Workers AI adapters. Supports four modes:\n * - Plain binding: `{ binding: env.AI }`\n * - Plain REST: `{ accountId, apiKey }`\n * - AI Gateway binding: `{ binding: env.AI.gateway(id) }`\n * - AI Gateway REST: `{ accountId, gatewayId, ... }`\n *\n * The third union member intersects `AiGatewayAdapterConfig` with `{ apiKey?: string }`.\n * For the gateway binding variant, `AiGatewayBindingConfig` already includes `apiKey?`,\n * so the intersection is redundant there. For the gateway credentials variant, this\n * `apiKey` represents the Workers AI token (used in the `Authorization` header to the\n * upstream provider), distinct from `cfApiKey` (used in the `cf-aig-authorization`\n * header for authenticated gateways).\n */\nexport type WorkersAiAdapterConfig = (\n\t| WorkersAiDirectBindingConfig\n\t| WorkersAiDirectCredentialsConfig\n\t| (AiGatewayAdapterConfig & { apiKey?: string })\n) & {\n\t/**\n\t * Session affinity key for prefix-cache optimization.\n\t * Routes requests with the same key to the same backend replica.\n\t */\n\tsessionAffinity?: string;\n};\n\n// ---------------------------------------------------------------------------\n// Config detection helpers\n// ---------------------------------------------------------------------------\n\n/** Returns true if this is a plain Workers AI binding config (`{ binding: env.AI }`) */\nexport function isDirectBindingConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectBindingConfig {\n\t// env.AI has a .gateway() method; env.AI.gateway(id) does not.\n\t// This distinguishes direct bindings from AI Gateway bindings.\n\treturn (\n\t\t\"binding\" in config &&\n\t\ttypeof (config.binding as unknown as Record<string, unknown>).gateway === \"function\"\n\t);\n}\n\n/** Returns true if this is a plain Workers AI REST config (accountId + apiKey, no gatewayId) */\nexport function isDirectCredentialsConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectCredentialsConfig {\n\treturn \"accountId\" in config && \"apiKey\" in config && !(\"gatewayId\" in config);\n}\n\n/** Returns true if this is an AI Gateway config (has gateway binding or `gatewayId`) */\nexport function isGatewayConfig(config: WorkersAiAdapterConfig): config is AiGatewayAdapterConfig {\n\tif (\"gatewayId\" in config) return true;\n\t// Has `binding` but NOT a direct Workers AI binding (no .gateway method)\n\treturn \"binding\" in config && !isDirectBindingConfig(config);\n}\n\n// ---------------------------------------------------------------------------\n// Config validation\n// ---------------------------------------------------------------------------\n\n/**\n * Validates that a WorkersAiAdapterConfig contains a valid configuration.\n * Throws an error if neither a binding, credentials (accountId + apiKey),\n * nor a gateway configuration is provided.\n */\nexport function validateWorkersAiConfig(config: WorkersAiAdapterConfig): void {\n\tif (\n\t\t!isDirectBindingConfig(config) &&\n\t\t!isDirectCredentialsConfig(config) &&\n\t\t!isGatewayConfig(config)\n\t) {\n\t\tthrow new Error(\n\t\t\t\"Invalid Workers AI configuration: you must provide either a binding (e.g. { binding: env.AI }), \" +\n\t\t\t\t\"credentials ({ accountId, apiKey }), or a gateway configuration ({ binding: env.AI.gateway(id) } \" +\n\t\t\t\t\"or { accountId, gatewayId }).\",\n\t\t);\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// createGatewayFetch -- for routing through AI Gateway\n// ---------------------------------------------------------------------------\n\nexport function createGatewayFetch(\n\tprovider: string,\n\tconfig: AiGatewayAdapterConfig,\n\theaders: Record<string, string> = {},\n): typeof fetch {\n\treturn (input, init) => {\n\t\tlet query: Record<string, unknown> = {};\n\n\t\tconst url =\n\t\t\ttypeof input === \"string\" ? input : input instanceof URL ? input.href : input.url;\n\t\tconst urlObj = new URL(url);\n\n\t\t// Extract endpoint path (remove /v1/ prefix if present)\n\t\tconst endpoint = urlObj.pathname.replace(/^\\/v1\\//, \"\").replace(/^\\//, \"\") + urlObj.search;\n\n\t\tif (init?.body) {\n\t\t\ttry {\n\t\t\t\tquery = JSON.parse(init.body as string);\n\t\t\t} catch {\n\t\t\t\tquery = { _raw: init.body };\n\t\t\t}\n\t\t}\n\n\t\tconst cacheHeaders: Record<string, string> = {};\n\n\t\tif (\"skipCache\" in config && config.skipCache) {\n\t\t\tcacheHeaders[\"cf-aig-skip-cache\"] = \"true\";\n\t\t}\n\n\t\tif (typeof config.cacheTtl === \"number\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-ttl\"] = String(config.cacheTtl);\n\t\t}\n\n\t\tif (typeof config.customCacheKey === \"string\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-key\"] = config.customCacheKey;\n\t\t}\n\n\t\tif (typeof config.metadata === \"object\") {\n\t\t\tcacheHeaders[\"cf-aig-metadata\"] = JSON.stringify(config.metadata);\n\t\t}\n\n\t\tconst request: {\n\t\t\tprovider: string;\n\t\t\tendpoint: string;\n\t\t\theaders: Record<string, string>;\n\t\t\tquery: Record<string, unknown>;\n\t\t} = {\n\t\t\tprovider,\n\t\t\tendpoint,\n\t\t\theaders: {\n\t\t\t\t...(init?.headers as Record<string, string> | undefined),\n\t\t\t\t...headers,\n\t\t\t\t...cacheHeaders,\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tquery,\n\t\t};\n\n\t\tif (provider === \"workers-ai\") {\n\t\t\tif (!request.endpoint.startsWith(\"run/\")) {\n\t\t\t\trequest.endpoint = `run/${query.model}`;\n\t\t\t}\n\t\t\tdelete query.model;\n\t\t\tdelete query.instructions;\n\t\t}\n\n\t\tif (config.apiKey) {\n\t\t\trequest.headers[\"authorization\"] = `Bearer ${config.apiKey}`;\n\t\t}\n\n\t\tif (\"binding\" in config) {\n\t\t\treturn (\n\t\t\t\tconfig.binding as {\n\t\t\t\t\trun(req: unknown, opts?: { signal?: AbortSignal }): Promise<Response>;\n\t\t\t\t}\n\t\t\t).run(request, { signal: init?.signal ?? undefined });\n\t\t}\n\n\t\treturn fetch(\n\t\t\t`https://gateway.ai.cloudflare.com/v1/${config.accountId}/${config.gatewayId}`,\n\t\t\t{\n\t\t\t\t...init,\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t\t\t...headers,\n\t\t\t\t\t...cacheHeaders,\n\t\t\t\t\t...(config.cfApiKey\n\t\t\t\t\t\t? { \"cf-aig-authorization\": `Bearer ${config.cfApiKey}` }\n\t\t\t\t\t\t: {}),\n\t\t\t\t},\n\t\t\t\tbody: JSON.stringify(request),\n\t\t\t},\n\t\t);\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// createWorkersAiBindingFetch -- shim that makes env.AI look like an OpenAI endpoint\n// ---------------------------------------------------------------------------\n\n/**\n * Normalize messages before passing to Workers AI binding.\n *\n * The binding has strict schema validation that may differ from the OpenAI API:\n * - `content` must not be null\n *\n * Content arrays (with image_url parts) are passed through as-is since the\n * Workers AI binding accepts them at runtime for vision-capable models.\n */\nfunction normalizeMessagesForBinding(\n\tmessages: Record<string, unknown>[],\n): Record<string, unknown>[] {\n\treturn messages.map((msg) => {\n\t\tconst normalized = { ...msg };\n\n\t\t// content: null → content: \"\"\n\t\tif (normalized.content === null || normalized.content === undefined) {\n\t\t\tnormalized.content = \"\";\n\t\t}\n\n\t\treturn normalized;\n\t});\n}\n\n/**\n * Creates a fetch function that intercepts OpenAI SDK requests and translates them\n * to Workers AI binding calls (env.AI.run). This allows the WorkersAiTextAdapter\n * to use the OpenAI SDK against a plain Workers AI binding.\n *\n * NOTE: The `input` URL parameter is intentionally ignored. The model name and all\n * request parameters are extracted from the JSON body, matching Workers AI's\n * `binding.run(model, inputs)` calling convention.\n */\nexport function createWorkersAiBindingFetch(\n\tbinding: WorkersAiBinding,\n\toptions?: { extraHeaders?: Record<string, string> },\n): typeof fetch {\n\treturn async (_input, init) => {\n\t\tif (!init?.body) {\n\t\t\treturn new Response(\"No body\", { status: 400 });\n\t\t}\n\n\t\tlet body: Record<string, unknown>;\n\t\ttry {\n\t\t\tbody = JSON.parse(init.body as string);\n\t\t} catch {\n\t\t\treturn new Response(\"Invalid JSON body\", { status: 400 });\n\t\t}\n\n\t\tconst model = body.model as string;\n\t\tconst stream = body.stream as boolean | undefined;\n\n\t\t// Build Workers AI inputs from OpenAI format\n\t\tconst inputs: Record<string, unknown> = {};\n\t\tif (body.messages) {\n\t\t\tinputs.messages = normalizeMessagesForBinding(\n\t\t\t\tbody.messages as Record<string, unknown>[],\n\t\t\t);\n\t\t}\n\t\tif (body.tools) inputs.tools = body.tools;\n\t\tif (typeof body.temperature === \"number\") inputs.temperature = body.temperature;\n\t\tif (typeof body.max_tokens === \"number\") inputs.max_tokens = body.max_tokens;\n\t\tif (body.response_format) inputs.response_format = body.response_format;\n\t\tif (stream) inputs.stream = true;\n\n\t\tconst runOptions: Record<string, unknown> = {};\n\t\tif (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;\n\t\tif (init?.signal) runOptions.signal = init.signal;\n\n\t\tconst result = await binding.run(\n\t\t\tmodel,\n\t\t\tinputs,\n\t\t\tObject.keys(runOptions).length > 0 ? runOptions : undefined,\n\t\t);\n\n\t\tif (stream && result instanceof ReadableStream) {\n\t\t\t// Workers AI returns an SSE stream with `data: {\"response\":\"chunk\"}` format.\n\t\t\t// Transform it to OpenAI-compatible SSE format.\n\t\t\tconst transformed = transformWorkersAiStream(\n\t\t\t\tresult as ReadableStream<Uint8Array>,\n\t\t\t\tmodel,\n\t\t\t);\n\t\t\treturn new Response(transformed, {\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"text/event-stream\",\n\t\t\t\t\t\"Cache-Control\": \"no-cache\",\n\t\t\t\t},\n\t\t\t});\n\t\t}\n\n\t\t// Graceful degradation: some models return a complete (non-streaming)\n\t\t// response even when `stream: true` is requested. Fall through to the\n\t\t// non-streaming wrapper which produces a valid OpenAI Chat Completion\n\t\t// response that the SDK can consume.\n\n\t\t// Non-streaming: Workers AI returns { response: \"text\", tool_calls?: [...] }\n\t\t// Wrap into OpenAI Chat Completion format.\n\t\tconst responseObj =\n\t\t\ttypeof result === \"object\" && result !== null\n\t\t\t\t? (result as Record<string, unknown>)\n\t\t\t\t: { response: String(result) };\n\n\t\tconst responseText =\n\t\t\ttypeof responseObj.response === \"string\"\n\t\t\t\t? responseObj.response\n\t\t\t\t: typeof responseObj.response === \"object\" && responseObj.response !== null\n\t\t\t\t\t? JSON.stringify(responseObj.response)\n\t\t\t\t\t: \"\";\n\n\t\tconst message: Record<string, unknown> = {\n\t\t\trole: \"assistant\",\n\t\t\tcontent: responseText,\n\t\t};\n\t\tlet finishReason = \"stop\";\n\n\t\t// Handle tool calls if present in Workers AI response\n\t\tif (Array.isArray(responseObj.tool_calls) && responseObj.tool_calls.length > 0) {\n\t\t\tfinishReason = \"tool_calls\";\n\t\t\tmessage.tool_calls = responseObj.tool_calls.map(\n\t\t\t\t(tc: {\n\t\t\t\t\tid?: string;\n\t\t\t\t\tname?: string;\n\t\t\t\t\targuments: unknown;\n\t\t\t\t\tfunction?: { name: string; arguments?: unknown };\n\t\t\t\t}) => ({\n\t\t\t\t\tid: tc.id || crypto.randomUUID(),\n\t\t\t\t\ttype: \"function\",\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function?.name || tc.name || \"\",\n\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\ttypeof (tc.function?.arguments ?? tc.arguments) === \"string\"\n\t\t\t\t\t\t\t\t? ((tc.function?.arguments ?? tc.arguments) as string)\n\t\t\t\t\t\t\t\t: JSON.stringify(tc.function?.arguments ?? tc.arguments ?? {}),\n\t\t\t\t\t},\n\t\t\t\t}),\n\t\t\t);\n\t\t}\n\n\t\tconst openAiResponse = {\n\t\t\tid: `workers-ai-${crypto.randomUUID()}`,\n\t\t\tobject: \"chat.completion\",\n\t\t\tcreated: Math.floor(Date.now() / 1000),\n\t\t\tmodel,\n\t\t\tchoices: [{ index: 0, message, finish_reason: finishReason }],\n\t\t};\n\n\t\treturn new Response(JSON.stringify(openAiResponse), {\n\t\t\theaders: { \"Content-Type\": \"application/json\" },\n\t\t});\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// Stream transformer: Workers AI SSE -> OpenAI-compatible SSE\n// Uses TransformStream for proper backpressure.\n// ---------------------------------------------------------------------------\n\n/**\n * Transforms a Workers AI SSE stream (data: {\"response\":\"chunk\"}) into\n * an OpenAI-compatible SSE stream (data: {\"choices\":[{\"delta\":{\"content\":\"chunk\"}}]}).\n *\n * Workers AI binding streams tool calls in an OpenAI-like nested format:\n * { tool_calls: [{ id, type, index, function: { name, arguments } }] }\n * Arguments are streamed incrementally across multiple SSE chunks, so the\n * transformer must forward them as incremental deltas rather than a single blob.\n */\nfunction transformWorkersAiStream(\n\tsource: ReadableStream<Uint8Array>,\n\tmodel: string,\n): ReadableStream<Uint8Array> {\n\tconst decoder = new TextDecoder();\n\tconst encoder = new TextEncoder();\n\t// Generate a stable ID and timestamp for the entire stream, matching OpenAI's\n\t// convention where all chunks in a single response share the same id/created.\n\tconst streamId = `workers-ai-${crypto.randomUUID()}`;\n\tconst created = Math.floor(Date.now() / 1000);\n\tlet buffer = \"\";\n\tlet hasToolCalls = false;\n\t// When true, the source stream is already in OpenAI format (some models\n\t// like Qwen3, Kimi K2.5 stream OpenAI-compatible SSE through the binding).\n\t// In that case, flush() should only emit [DONE] and skip the finish chunk.\n\tlet isOpenAiFormat = false;\n\t// Track tool call state per index: store the generated/assigned ID so that\n\t// subsequent argument deltas use the same ID (matching the working streaming.ts pattern).\n\tconst toolCallState = new Map<number, { id: string; name: string }>();\n\n\treturn source.pipeThrough(\n\t\tnew TransformStream<Uint8Array, Uint8Array>({\n\t\t\ttransform(chunk, controller) {\n\t\t\t\tbuffer += decoder.decode(chunk, { stream: true });\n\t\t\t\tconst lines = buffer.split(\"\\n\");\n\t\t\t\tbuffer = lines.pop() || \"\";\n\n\t\t\t\tfor (const line of lines) {\n\t\t\t\t\tconst trimmed = line.trim();\n\t\t\t\t\tif (!trimmed || !trimmed.startsWith(\"data: \")) continue;\n\t\t\t\t\tconst data = trimmed.slice(6);\n\n\t\t\t\t\t// Swallow source [DONE]; we emit our own in flush()\n\t\t\t\t\tif (data === \"[DONE]\") continue;\n\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst parsed = JSON.parse(data);\n\n\t\t\t\t\t\t// Some models (Qwen3, Kimi K2.5) return OpenAI-compatible format\n\t\t\t\t\t\t// directly through the binding, with `choices[].delta.content` and\n\t\t\t\t\t\t// optional `reasoning_content`. Detect this and pass through as-is.\n\t\t\t\t\t\tif (parsed.choices !== undefined) {\n\t\t\t\t\t\t\t// Already OpenAI format — pass through but ensure each tool call\n\t\t\t\t\t\t\t// index gets a unique, stable ID across all chunks.\n\t\t\t\t\t\t\tisOpenAiFormat = true;\n\t\t\t\t\t\t\tconst choice = parsed.choices?.[0];\n\t\t\t\t\t\t\tif (choice?.delta?.tool_calls) {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t\tfor (const tc of choice.delta.tool_calls) {\n\t\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\t\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t\t// First chunk for this index — generate/store unique ID\n\t\t\t\t\t\t\t\t\t\tconst id = tc.id || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, {\n\t\t\t\t\t\t\t\t\t\t\tid,\n\t\t\t\t\t\t\t\t\t\t\tname: tc.function?.name || \"\",\n\t\t\t\t\t\t\t\t\t\t});\n\t\t\t\t\t\t\t\t\t\ttc.id = id;\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\t// Subsequent chunk — reuse stored ID, remove id from delta\n\t\t\t\t\t\t\t\t\t\t// (OpenAI format only sends id in first chunk)\n\t\t\t\t\t\t\t\t\t\tdelete tc.id;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (choice?.finish_reason === \"tool_calls\") {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(parsed)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// --- Workers AI native format handling below ---\n\n\t\t\t\t\t\t// Text content\n\t\t\t\t\t\tif (parsed.response != null && parsed.response !== \"\") {\n\t\t\t\t\t\t\tconst openAiChunk = {\n\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\tdelta: { content: parsed.response },\n\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(openAiChunk)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Tool calls — Workers AI binding streams these incrementally:\n\t\t\t\t\t\t// Chunk A: { id, type, index, function: { name } } — start\n\t\t\t\t\t\t// Chunk B: { index, function: { arguments: \"partial...\" } } — args delta\n\t\t\t\t\t\t// Chunk C: { index, function: { arguments: \"rest...\" } } — args delta\n\t\t\t\t\t\t// Chunk D: { id: null, type: null, index, function: { name: null, arguments: \"\" } } — finalize (skip)\n\t\t\t\t\t\tif (Array.isArray(parsed.tool_calls) && parsed.tool_calls.length > 0) {\n\t\t\t\t\t\t\tfor (const tc of parsed.tool_calls) {\n\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\n\t\t\t\t\t\t\t\t// Resolve name and arguments from either nested or flat format\n\t\t\t\t\t\t\t\tconst tcName = tc.function?.name ?? tc.name ?? null;\n\t\t\t\t\t\t\t\tconst tcArgs = tc.function?.arguments ?? tc.arguments ?? null;\n\t\t\t\t\t\t\t\tconst tcId = tc.id ?? null;\n\n\t\t\t\t\t\t\t\t// Skip finalization chunks where everything is null/empty\n\t\t\t\t\t\t\t\tif (!tcId && !tcName && (!tcArgs || tcArgs === \"\")) continue;\n\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\n\t\t\t\t\t\t\t\t// Build the OpenAI-compatible tool_calls delta\n\t\t\t\t\t\t\t\tconst toolCallDelta: Record<string, unknown> = {\n\t\t\t\t\t\t\t\t\tindex: tcIndex,\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t// First chunk for this tool call index — emit id, type, name.\n\t\t\t\t\t\t\t\t\tconst id = tcId || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, { id, name: tcName || \"\" });\n\t\t\t\t\t\t\t\t\ttoolCallDelta.id = id;\n\t\t\t\t\t\t\t\t\ttoolCallDelta.type = \"function\";\n\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\tname: tcName || \"\",\n\t\t\t\t\t\t\t\t\t\t// Include arguments if they arrive in the same chunk\n\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\ttcArgs != null\n\t\t\t\t\t\t\t\t\t\t\t\t? typeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs)\n\t\t\t\t\t\t\t\t\t\t\t\t: \"\",\n\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t// Subsequent chunks — only include arguments delta\n\t\t\t\t\t\t\t\t\tif (tcArgs != null && tcArgs !== \"\") {\n\t\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\t\ttypeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs),\n\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\tcontinue; // Nothing useful to forward\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst toolChunk = {\n\t\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\t\tdelta: { tool_calls: [toolCallDelta] },\n\t\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(toolChunk)}\\n\\n`),\n\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (e) {\n\t\t\t\t\t\t// Log malformed SSE events for debugging; don't break the stream.\n\t\t\t\t\t\tconsole.warn(\"[tanstack-ai] failed to parse SSE event:\", data, e);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t\tflush(controller) {\n\t\t\t\tif (!isOpenAiFormat) {\n\t\t\t\t\t// Workers AI native format: emit a finish chunk with stop/tool_calls\n\t\t\t\t\tconst finalChunk = {\n\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\tcreated,\n\t\t\t\t\t\tmodel,\n\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\tdelta: {},\n\t\t\t\t\t\t\t\tfinish_reason: hasToolCalls ? \"tool_calls\" : \"stop\",\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t};\n\t\t\t\t\tcontroller.enqueue(encoder.encode(`data: ${JSON.stringify(finalChunk)}\\n\\n`));\n\t\t\t\t}\n\t\t\t\t// OpenAI format already includes its own finish_reason in the stream.\n\t\t\t\t// Either way, emit a [DONE] sentinel.\n\t\t\t\tcontroller.enqueue(encoder.encode(\"data: [DONE]\\n\\n\"));\n\t\t\t},\n\t\t}),\n\t);\n}\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"create-fetcher-DY7wfYYy.mjs","names":[],"sources":["../src/utils/create-fetcher.ts"],"sourcesContent":["// ---------------------------------------------------------------------------\n// AI Gateway types (for third-party providers + Workers AI through gateway)\n// ---------------------------------------------------------------------------\n\nexport interface CloudflareAiGateway {\n\trun(request: unknown): Promise<Response>;\n}\n\nexport interface AiGatewayBindingConfig {\n\t/**\n\t * The AI Gateway binding\n\t * @example\n\t * env.AI.gateway('my-gateway-id')\n\t */\n\tbinding: CloudflareAiGateway;\n\t/**\n\t * The Provider API Key if you want to manually pass it, ignore if using Unified Billing or BYOK.\n\t */\n\tapiKey?: string;\n}\n\nexport type AiGatewayCredentialsConfig = {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The AI Gateway ID\n\t */\n\tgatewayId: string;\n} & (\n\t| {\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey: string;\n\t\t\tapiKey?: string;\n\t }\n\t| {\n\t\t\t/** Provider API Key */\n\t\t\tapiKey: string;\n\t\t\t/** Cloudflare API Key for AI Gateway */\n\t\t\tcfApiKey?: string;\n\t }\n);\n\nexport interface AiGatewayConfig {\n\tskipCache?: boolean;\n\tcacheTtl?: number;\n\tcustomCacheKey?: string;\n\tmetadata?: Record<string, unknown>;\n}\n\nexport type AiGatewayAdapterConfig = (AiGatewayBindingConfig | AiGatewayCredentialsConfig) &\n\tAiGatewayConfig;\n\n// ---------------------------------------------------------------------------\n// Plain Workers AI types (direct binding or REST, no gateway)\n// ---------------------------------------------------------------------------\n\n/**\n * The Workers AI binding interface (env.AI).\n * Accepts a model name and inputs, returns results directly.\n * Includes `gateway()` which is present on `env.AI` but not on `env.AI.gateway(id)`,\n * enabling structural discrimination from `CloudflareAiGateway`.\n */\nexport interface WorkersAiBinding {\n\trun(\n\t\tmodel: string,\n\t\tinputs: Record<string, unknown>,\n\t\toptions?: Record<string, unknown>,\n\t): Promise<unknown>;\n\tgateway(gatewayId: string): CloudflareAiGateway;\n}\n\nexport interface WorkersAiDirectBindingConfig {\n\t/**\n\t * The Workers AI binding (env.AI).\n\t * @example\n\t * { binding: env.AI }\n\t */\n\tbinding: WorkersAiBinding;\n}\n\nexport interface WorkersAiDirectCredentialsConfig {\n\t/**\n\t * The Cloudflare account ID\n\t */\n\taccountId: string;\n\t/**\n\t * The Cloudflare API key for Workers AI\n\t */\n\tapiKey: string;\n}\n\n/**\n * Config for Workers AI adapters. Supports four modes:\n * - Plain binding: `{ binding: env.AI }`\n * - Plain REST: `{ accountId, apiKey }`\n * - AI Gateway binding: `{ binding: env.AI.gateway(id) }`\n * - AI Gateway REST: `{ accountId, gatewayId, ... }`\n *\n * The third union member intersects `AiGatewayAdapterConfig` with `{ apiKey?: string }`.\n * For the gateway binding variant, `AiGatewayBindingConfig` already includes `apiKey?`,\n * so the intersection is redundant there. For the gateway credentials variant, this\n * `apiKey` represents the Workers AI token (used in the `Authorization` header to the\n * upstream provider), distinct from `cfApiKey` (used in the `cf-aig-authorization`\n * header for authenticated gateways).\n */\nexport type WorkersAiAdapterConfig = (\n\t| WorkersAiDirectBindingConfig\n\t| WorkersAiDirectCredentialsConfig\n\t| (AiGatewayAdapterConfig & { apiKey?: string })\n) & {\n\t/**\n\t * Session affinity key for prefix-cache optimization.\n\t * Routes requests with the same key to the same backend replica.\n\t */\n\tsessionAffinity?: string;\n};\n\n// ---------------------------------------------------------------------------\n// Config detection helpers\n// ---------------------------------------------------------------------------\n\n/** Returns true if this is a plain Workers AI binding config (`{ binding: env.AI }`) */\nexport function isDirectBindingConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectBindingConfig {\n\t// env.AI has a .gateway() method; env.AI.gateway(id) does not.\n\t// This distinguishes direct bindings from AI Gateway bindings.\n\treturn (\n\t\t\"binding\" in config &&\n\t\ttypeof (config.binding as unknown as Record<string, unknown>).gateway === \"function\"\n\t);\n}\n\n/** Returns true if this is a plain Workers AI REST config (accountId + apiKey, no gatewayId) */\nexport function isDirectCredentialsConfig(\n\tconfig: WorkersAiAdapterConfig,\n): config is WorkersAiDirectCredentialsConfig {\n\treturn \"accountId\" in config && \"apiKey\" in config && !(\"gatewayId\" in config);\n}\n\n/** Returns true if this is an AI Gateway config (has gateway binding or `gatewayId`) */\nexport function isGatewayConfig(config: WorkersAiAdapterConfig): config is AiGatewayAdapterConfig {\n\tif (\"gatewayId\" in config) return true;\n\t// Has `binding` but NOT a direct Workers AI binding (no .gateway method)\n\treturn \"binding\" in config && !isDirectBindingConfig(config);\n}\n\n// ---------------------------------------------------------------------------\n// Config validation\n// ---------------------------------------------------------------------------\n\n/**\n * Validates that a WorkersAiAdapterConfig contains a valid configuration.\n * Throws an error if neither a binding, credentials (accountId + apiKey),\n * nor a gateway configuration is provided.\n */\nexport function validateWorkersAiConfig(config: WorkersAiAdapterConfig): void {\n\tif (\n\t\t!isDirectBindingConfig(config) &&\n\t\t!isDirectCredentialsConfig(config) &&\n\t\t!isGatewayConfig(config)\n\t) {\n\t\tthrow new Error(\n\t\t\t\"Invalid Workers AI configuration: you must provide either a binding (e.g. { binding: env.AI }), \" +\n\t\t\t\t\"credentials ({ accountId, apiKey }), or a gateway configuration ({ binding: env.AI.gateway(id) } \" +\n\t\t\t\t\"or { accountId, gatewayId }).\",\n\t\t);\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// createGatewayFetch -- for routing through AI Gateway\n// ---------------------------------------------------------------------------\n\nexport function createGatewayFetch(\n\tprovider: string,\n\tconfig: AiGatewayAdapterConfig,\n\theaders: Record<string, string> = {},\n): typeof fetch {\n\treturn (input, init) => {\n\t\tlet query: Record<string, unknown> = {};\n\n\t\tconst url =\n\t\t\ttypeof input === \"string\" ? input : input instanceof URL ? input.href : input.url;\n\t\tconst urlObj = new URL(url);\n\n\t\t// Extract endpoint path (remove /v1/ prefix if present)\n\t\tconst endpoint = urlObj.pathname.replace(/^\\/v1\\//, \"\").replace(/^\\//, \"\") + urlObj.search;\n\n\t\tif (init?.body) {\n\t\t\ttry {\n\t\t\t\tquery = JSON.parse(init.body as string);\n\t\t\t} catch {\n\t\t\t\tquery = { _raw: init.body };\n\t\t\t}\n\t\t}\n\n\t\tconst cacheHeaders: Record<string, string> = {};\n\n\t\tif (\"skipCache\" in config && config.skipCache) {\n\t\t\tcacheHeaders[\"cf-aig-skip-cache\"] = \"true\";\n\t\t}\n\n\t\tif (typeof config.cacheTtl === \"number\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-ttl\"] = String(config.cacheTtl);\n\t\t}\n\n\t\tif (typeof config.customCacheKey === \"string\") {\n\t\t\tcacheHeaders[\"cf-aig-cache-key\"] = config.customCacheKey;\n\t\t}\n\n\t\tif (typeof config.metadata === \"object\") {\n\t\t\tcacheHeaders[\"cf-aig-metadata\"] = JSON.stringify(config.metadata);\n\t\t}\n\n\t\tconst request: {\n\t\t\tprovider: string;\n\t\t\tendpoint: string;\n\t\t\theaders: Record<string, string>;\n\t\t\tquery: Record<string, unknown>;\n\t\t} = {\n\t\t\tprovider,\n\t\t\tendpoint,\n\t\t\theaders: {\n\t\t\t\t...(init?.headers as Record<string, string> | undefined),\n\t\t\t\t...headers,\n\t\t\t\t...cacheHeaders,\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tquery,\n\t\t};\n\n\t\tif (provider === \"workers-ai\") {\n\t\t\tif (!request.endpoint.startsWith(\"run/\")) {\n\t\t\t\trequest.endpoint = `run/${query.model}`;\n\t\t\t}\n\t\t\tdelete query.model;\n\t\t\tdelete query.instructions;\n\t\t}\n\n\t\tif (config.apiKey) {\n\t\t\trequest.headers[\"authorization\"] = `Bearer ${config.apiKey}`;\n\t\t}\n\n\t\tif (\"binding\" in config) {\n\t\t\treturn (\n\t\t\t\tconfig.binding as {\n\t\t\t\t\trun(req: unknown, opts?: { signal?: AbortSignal }): Promise<Response>;\n\t\t\t\t}\n\t\t\t).run(request, { signal: init?.signal ?? undefined });\n\t\t}\n\n\t\treturn fetch(\n\t\t\t`https://gateway.ai.cloudflare.com/v1/${config.accountId}/${config.gatewayId}`,\n\t\t\t{\n\t\t\t\t...init,\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t\t\t...headers,\n\t\t\t\t\t...cacheHeaders,\n\t\t\t\t\t...(config.cfApiKey\n\t\t\t\t\t\t? { \"cf-aig-authorization\": `Bearer ${config.cfApiKey}` }\n\t\t\t\t\t\t: {}),\n\t\t\t\t},\n\t\t\t\tbody: JSON.stringify(request),\n\t\t\t},\n\t\t);\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// createWorkersAiBindingFetch -- shim that makes env.AI look like an OpenAI endpoint\n// ---------------------------------------------------------------------------\n\n/**\n * Normalize messages before passing to Workers AI binding.\n *\n * The binding has strict schema validation that may differ from the OpenAI API:\n * - `content` must not be null\n *\n * Content arrays (with image_url parts) are passed through as-is since the\n * Workers AI binding accepts them at runtime for vision-capable models.\n */\nfunction normalizeMessagesForBinding(\n\tmessages: Record<string, unknown>[],\n): Record<string, unknown>[] {\n\treturn messages.map((msg) => {\n\t\tconst normalized = { ...msg };\n\n\t\t// content: null → content: \"\"\n\t\tif (normalized.content === null || normalized.content === undefined) {\n\t\t\tnormalized.content = \"\";\n\t\t}\n\n\t\treturn normalized;\n\t});\n}\n\n/**\n * Creates a fetch function that intercepts OpenAI SDK requests and translates them\n * to Workers AI binding calls (env.AI.run). This allows the WorkersAiTextAdapter\n * to use the OpenAI SDK against a plain Workers AI binding.\n *\n * NOTE: The `input` URL parameter is intentionally ignored. The model name and all\n * request parameters are extracted from the JSON body, matching Workers AI's\n * `binding.run(model, inputs)` calling convention.\n */\nexport function createWorkersAiBindingFetch(\n\tbinding: WorkersAiBinding,\n\toptions?: { extraHeaders?: Record<string, string> },\n): typeof fetch {\n\treturn async (_input, init) => {\n\t\tif (!init?.body) {\n\t\t\treturn new Response(\"No body\", { status: 400 });\n\t\t}\n\n\t\tlet body: Record<string, unknown>;\n\t\ttry {\n\t\t\tbody = JSON.parse(init.body as string);\n\t\t} catch {\n\t\t\treturn new Response(\"Invalid JSON body\", { status: 400 });\n\t\t}\n\n\t\tconst model = body.model as string;\n\t\tconst stream = body.stream as boolean | undefined;\n\n\t\t// Build Workers AI inputs from OpenAI format\n\t\tconst inputs: Record<string, unknown> = {};\n\t\tif (body.messages) {\n\t\t\tinputs.messages = normalizeMessagesForBinding(\n\t\t\t\tbody.messages as Record<string, unknown>[],\n\t\t\t);\n\t\t}\n\t\tif (body.tools) inputs.tools = body.tools;\n\t\tif (typeof body.temperature === \"number\") inputs.temperature = body.temperature;\n\t\tif (typeof body.max_tokens === \"number\") inputs.max_tokens = body.max_tokens;\n\t\tif (body.response_format) inputs.response_format = body.response_format;\n\t\tif (stream) inputs.stream = true;\n\n\t\tconst runOptions: Record<string, unknown> = {};\n\t\tif (options?.extraHeaders) runOptions.extraHeaders = options.extraHeaders;\n\t\tif (init?.signal) runOptions.signal = init.signal;\n\n\t\tconst result = await binding.run(\n\t\t\tmodel,\n\t\t\tinputs,\n\t\t\tObject.keys(runOptions).length > 0 ? runOptions : undefined,\n\t\t);\n\n\t\tif (stream && result instanceof ReadableStream) {\n\t\t\t// Workers AI returns an SSE stream with `data: {\"response\":\"chunk\"}` format.\n\t\t\t// Transform it to OpenAI-compatible SSE format.\n\t\t\tconst transformed = transformWorkersAiStream(\n\t\t\t\tresult as ReadableStream<Uint8Array>,\n\t\t\t\tmodel,\n\t\t\t);\n\t\t\treturn new Response(transformed, {\n\t\t\t\theaders: {\n\t\t\t\t\t\"Content-Type\": \"text/event-stream\",\n\t\t\t\t\t\"Cache-Control\": \"no-cache\",\n\t\t\t\t},\n\t\t\t});\n\t\t}\n\n\t\t// Graceful degradation: some models return a complete (non-streaming)\n\t\t// response even when `stream: true` is requested. Fall through to the\n\t\t// non-streaming wrapper which produces a valid OpenAI Chat Completion\n\t\t// response that the SDK can consume.\n\n\t\t// Non-streaming: Workers AI returns { response: \"text\", tool_calls?: [...] }\n\t\t// Wrap into OpenAI Chat Completion format.\n\t\tconst responseObj =\n\t\t\ttypeof result === \"object\" && result !== null\n\t\t\t\t? (result as Record<string, unknown>)\n\t\t\t\t: { response: String(result) };\n\n\t\tconst responseText =\n\t\t\ttypeof responseObj.response === \"string\"\n\t\t\t\t? responseObj.response\n\t\t\t\t: typeof responseObj.response === \"object\" && responseObj.response !== null\n\t\t\t\t\t? JSON.stringify(responseObj.response)\n\t\t\t\t\t: \"\";\n\n\t\tconst message: Record<string, unknown> = {\n\t\t\trole: \"assistant\",\n\t\t\tcontent: responseText,\n\t\t};\n\t\tlet finishReason = \"stop\";\n\n\t\t// Handle tool calls if present in Workers AI response\n\t\tif (Array.isArray(responseObj.tool_calls) && responseObj.tool_calls.length > 0) {\n\t\t\tfinishReason = \"tool_calls\";\n\t\t\tmessage.tool_calls = responseObj.tool_calls.map(\n\t\t\t\t(tc: {\n\t\t\t\t\tid?: string;\n\t\t\t\t\tname?: string;\n\t\t\t\t\targuments: unknown;\n\t\t\t\t\tfunction?: { name: string; arguments?: unknown };\n\t\t\t\t}) => ({\n\t\t\t\t\tid: tc.id || crypto.randomUUID(),\n\t\t\t\t\ttype: \"function\",\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function?.name || tc.name || \"\",\n\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\ttypeof (tc.function?.arguments ?? tc.arguments) === \"string\"\n\t\t\t\t\t\t\t\t? ((tc.function?.arguments ?? tc.arguments) as string)\n\t\t\t\t\t\t\t\t: JSON.stringify(tc.function?.arguments ?? tc.arguments ?? {}),\n\t\t\t\t\t},\n\t\t\t\t}),\n\t\t\t);\n\t\t}\n\n\t\tconst openAiResponse = {\n\t\t\tid: `workers-ai-${crypto.randomUUID()}`,\n\t\t\tobject: \"chat.completion\",\n\t\t\tcreated: Math.floor(Date.now() / 1000),\n\t\t\tmodel,\n\t\t\tchoices: [{ index: 0, message, finish_reason: finishReason }],\n\t\t};\n\n\t\treturn new Response(JSON.stringify(openAiResponse), {\n\t\t\theaders: { \"Content-Type\": \"application/json\" },\n\t\t});\n\t};\n}\n\n// ---------------------------------------------------------------------------\n// Stream transformer: Workers AI SSE -> OpenAI-compatible SSE\n// Uses TransformStream for proper backpressure.\n// ---------------------------------------------------------------------------\n\n/**\n * Transforms a Workers AI SSE stream (data: {\"response\":\"chunk\"}) into\n * an OpenAI-compatible SSE stream (data: {\"choices\":[{\"delta\":{\"content\":\"chunk\"}}]}).\n *\n * Workers AI binding streams tool calls in an OpenAI-like nested format:\n * { tool_calls: [{ id, type, index, function: { name, arguments } }] }\n * Arguments are streamed incrementally across multiple SSE chunks, so the\n * transformer must forward them as incremental deltas rather than a single blob.\n */\nfunction transformWorkersAiStream(\n\tsource: ReadableStream<Uint8Array>,\n\tmodel: string,\n): ReadableStream<Uint8Array> {\n\tconst decoder = new TextDecoder();\n\tconst encoder = new TextEncoder();\n\t// Generate a stable ID and timestamp for the entire stream, matching OpenAI's\n\t// convention where all chunks in a single response share the same id/created.\n\tconst streamId = `workers-ai-${crypto.randomUUID()}`;\n\tconst created = Math.floor(Date.now() / 1000);\n\tlet buffer = \"\";\n\tlet hasToolCalls = false;\n\t// When true, the source stream is already in OpenAI format (some models\n\t// like Qwen3, Kimi K2.5 stream OpenAI-compatible SSE through the binding).\n\t// In that case, flush() should only emit [DONE] and skip the finish chunk.\n\tlet isOpenAiFormat = false;\n\t// Track tool call state per index: store the generated/assigned ID so that\n\t// subsequent argument deltas use the same ID (matching the working streaming.ts pattern).\n\tconst toolCallState = new Map<number, { id: string; name: string }>();\n\n\treturn source.pipeThrough(\n\t\tnew TransformStream<Uint8Array, Uint8Array>({\n\t\t\ttransform(chunk, controller) {\n\t\t\t\tbuffer += decoder.decode(chunk, { stream: true });\n\t\t\t\tconst lines = buffer.split(\"\\n\");\n\t\t\t\tbuffer = lines.pop() || \"\";\n\n\t\t\t\tfor (const line of lines) {\n\t\t\t\t\tconst trimmed = line.trim();\n\t\t\t\t\tif (!trimmed || !trimmed.startsWith(\"data: \")) continue;\n\t\t\t\t\tconst data = trimmed.slice(6);\n\n\t\t\t\t\t// Swallow source [DONE]; we emit our own in flush()\n\t\t\t\t\tif (data === \"[DONE]\") continue;\n\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst parsed = JSON.parse(data);\n\n\t\t\t\t\t\t// Some models (Qwen3, Kimi K2.5) return OpenAI-compatible format\n\t\t\t\t\t\t// directly through the binding, with `choices[].delta.content` and\n\t\t\t\t\t\t// optional `reasoning_content`. Detect this and pass through as-is.\n\t\t\t\t\t\tif (parsed.choices !== undefined) {\n\t\t\t\t\t\t\t// Already OpenAI format — pass through but ensure each tool call\n\t\t\t\t\t\t\t// index gets a unique, stable ID across all chunks.\n\t\t\t\t\t\t\tisOpenAiFormat = true;\n\t\t\t\t\t\t\tconst choice = parsed.choices?.[0];\n\t\t\t\t\t\t\tif (choice?.delta?.tool_calls) {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t\tfor (const tc of choice.delta.tool_calls) {\n\t\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\t\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t\t// First chunk for this index — generate/store unique ID\n\t\t\t\t\t\t\t\t\t\tconst id = tc.id || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, {\n\t\t\t\t\t\t\t\t\t\t\tid,\n\t\t\t\t\t\t\t\t\t\t\tname: tc.function?.name || \"\",\n\t\t\t\t\t\t\t\t\t\t});\n\t\t\t\t\t\t\t\t\t\ttc.id = id;\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\t// Subsequent chunk — reuse stored ID, remove id from delta\n\t\t\t\t\t\t\t\t\t\t// (OpenAI format only sends id in first chunk)\n\t\t\t\t\t\t\t\t\t\tdelete tc.id;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (choice?.finish_reason === \"tool_calls\") {\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(parsed)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// --- Workers AI native format handling below ---\n\n\t\t\t\t\t\t// Text content\n\t\t\t\t\t\tif (parsed.response != null && parsed.response !== \"\") {\n\t\t\t\t\t\t\tconst openAiChunk = {\n\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\tdelta: { content: parsed.response },\n\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(openAiChunk)}\\n\\n`),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Tool calls — Workers AI binding streams these incrementally:\n\t\t\t\t\t\t// Chunk A: { id, type, index, function: { name } } — start\n\t\t\t\t\t\t// Chunk B: { index, function: { arguments: \"partial...\" } } — args delta\n\t\t\t\t\t\t// Chunk C: { index, function: { arguments: \"rest...\" } } — args delta\n\t\t\t\t\t\t// Chunk D: { id: null, type: null, index, function: { name: null, arguments: \"\" } } — finalize (skip)\n\t\t\t\t\t\tif (Array.isArray(parsed.tool_calls) && parsed.tool_calls.length > 0) {\n\t\t\t\t\t\t\tfor (const tc of parsed.tool_calls) {\n\t\t\t\t\t\t\t\tconst tcIndex = tc.index ?? 0;\n\n\t\t\t\t\t\t\t\t// Resolve name and arguments from either nested or flat format\n\t\t\t\t\t\t\t\tconst tcName = tc.function?.name ?? tc.name ?? null;\n\t\t\t\t\t\t\t\tconst tcArgs = tc.function?.arguments ?? tc.arguments ?? null;\n\t\t\t\t\t\t\t\tconst tcId = tc.id ?? null;\n\n\t\t\t\t\t\t\t\t// Skip finalization chunks where everything is null/empty\n\t\t\t\t\t\t\t\tif (!tcId && !tcName && (!tcArgs || tcArgs === \"\")) continue;\n\n\t\t\t\t\t\t\t\thasToolCalls = true;\n\n\t\t\t\t\t\t\t\t// Build the OpenAI-compatible tool_calls delta\n\t\t\t\t\t\t\t\tconst toolCallDelta: Record<string, unknown> = {\n\t\t\t\t\t\t\t\t\tindex: tcIndex,\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\t\tif (!toolCallState.has(tcIndex)) {\n\t\t\t\t\t\t\t\t\t// First chunk for this tool call index — emit id, type, name.\n\t\t\t\t\t\t\t\t\tconst id = tcId || `call${streamId}${tcIndex}`;\n\t\t\t\t\t\t\t\t\ttoolCallState.set(tcIndex, { id, name: tcName || \"\" });\n\t\t\t\t\t\t\t\t\ttoolCallDelta.id = id;\n\t\t\t\t\t\t\t\t\ttoolCallDelta.type = \"function\";\n\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\tname: tcName || \"\",\n\t\t\t\t\t\t\t\t\t\t// Include arguments if they arrive in the same chunk\n\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\ttcArgs != null\n\t\t\t\t\t\t\t\t\t\t\t\t? typeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs)\n\t\t\t\t\t\t\t\t\t\t\t\t: \"\",\n\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t// Subsequent chunks — only include arguments delta\n\t\t\t\t\t\t\t\t\tif (tcArgs != null && tcArgs !== \"\") {\n\t\t\t\t\t\t\t\t\t\ttoolCallDelta.function = {\n\t\t\t\t\t\t\t\t\t\t\targuments:\n\t\t\t\t\t\t\t\t\t\t\t\ttypeof tcArgs === \"string\"\n\t\t\t\t\t\t\t\t\t\t\t\t\t? tcArgs\n\t\t\t\t\t\t\t\t\t\t\t\t\t: JSON.stringify(tcArgs),\n\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\tcontinue; // Nothing useful to forward\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst toolChunk = {\n\t\t\t\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\t\t\t\tcreated,\n\t\t\t\t\t\t\t\t\tmodel,\n\t\t\t\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\t\t\t\tdelta: { tool_calls: [toolCallDelta] },\n\t\t\t\t\t\t\t\t\t\t\tfinish_reason: null,\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t],\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tcontroller.enqueue(\n\t\t\t\t\t\t\t\t\tencoder.encode(`data: ${JSON.stringify(toolChunk)}\\n\\n`),\n\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (e) {\n\t\t\t\t\t\t// Log malformed SSE events for debugging; don't break the stream.\n\t\t\t\t\t\tconsole.warn(\"[tanstack-ai] failed to parse SSE event:\", data, e);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t\tflush(controller) {\n\t\t\t\tif (!isOpenAiFormat) {\n\t\t\t\t\t// Workers AI native format: emit a finish chunk with stop/tool_calls\n\t\t\t\t\tconst finalChunk = {\n\t\t\t\t\t\tid: streamId,\n\t\t\t\t\t\tobject: \"chat.completion.chunk\",\n\t\t\t\t\t\tcreated,\n\t\t\t\t\t\tmodel,\n\t\t\t\t\t\tchoices: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t\t\tdelta: {},\n\t\t\t\t\t\t\t\tfinish_reason: hasToolCalls ? \"tool_calls\" : \"stop\",\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t};\n\t\t\t\t\tcontroller.enqueue(encoder.encode(`data: ${JSON.stringify(finalChunk)}\\n\\n`));\n\t\t\t\t}\n\t\t\t\t// OpenAI format already includes its own finish_reason in the stream.\n\t\t\t\t// Either way, emit a [DONE] sentinel.\n\t\t\t\tcontroller.enqueue(encoder.encode(\"data: [DONE]\\n\\n\"));\n\t\t\t},\n\t\t}),\n\t);\n}\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"workers-ai-Bm7Up4or.cjs","names":["isDirectBindingConfig","OpenAI","createWorkersAiBindingFetch","isDirectCredentialsConfig","createGatewayFetch","BaseTextAdapter"],"sources":["../src/adapters/workers-ai.ts"],"sourcesContent":["import type { AiModels, BaseAiTextGeneration } from \"@cloudflare/workers-types\";\nimport type { ContentPart, ModelMessage, StreamChunk, TextOptions } from \"@tanstack/ai\";\nimport {\n\tBaseTextAdapter,\n\ttype StructuredOutputOptions,\n\ttype StructuredOutputResult,\n} from \"@tanstack/ai/adapters\";\nimport OpenAI from \"openai\";\nimport {\n\ttype WorkersAiAdapterConfig,\n\ttype AiGatewayAdapterConfig,\n\tcreateGatewayFetch,\n\tcreateWorkersAiBindingFetch,\n\tisDirectBindingConfig,\n\tisDirectCredentialsConfig,\n\tvalidateWorkersAiConfig,\n} from \"../utils/create-fetcher\";\n\n// ---------------------------------------------------------------------------\n// Model types derived from @cloudflare/workers-types\n// ---------------------------------------------------------------------------\n\nexport type WorkersAiTextModel =\n\t| {\n\t\t\t[K in keyof AiModels]: AiModels[K] extends BaseAiTextGeneration ? K : never;\n\t }[keyof AiModels]\n\t| (string & {});\n\n// ---------------------------------------------------------------------------\n// Helpers: build the right OpenAI client depending on config mode\n// ---------------------------------------------------------------------------\n\nfunction buildWorkersAiClient(config: WorkersAiAdapterConfig): OpenAI {\n\tvalidateWorkersAiConfig(config);\n\n\tconst sessionHeaders: Record<string, string> | undefined = config.sessionAffinity\n\t\t? { \"x-session-affinity\": config.sessionAffinity }\n\t\t: undefined;\n\n\tif (isDirectBindingConfig(config)) {\n\t\t// Plain binding mode: shim translates OpenAI fetch calls to env.AI.run()\n\t\treturn new OpenAI({\n\t\t\tapiKey: \"unused\",\n\t\t\tfetch: createWorkersAiBindingFetch(\n\t\t\t\tconfig.binding,\n\t\t\t\tsessionHeaders ? { extraHeaders: sessionHeaders } : undefined,\n\t\t\t),\n\t\t});\n\t}\n\n\tif (isDirectCredentialsConfig(config)) {\n\t\t// Plain REST mode: point OpenAI SDK at Workers AI's OpenAI-compatible endpoint\n\t\treturn new OpenAI({\n\t\t\tbaseURL: `https://api.cloudflare.com/client/v4/accounts/${config.accountId}/ai/v1`,\n\t\t\tapiKey: config.apiKey,\n\t\t\tdefaultHeaders: sessionHeaders,\n\t\t});\n\t}\n\n\t// Gateway mode (existing): use createGatewayFetch\n\tconst gatewayConfig = config as AiGatewayAdapterConfig;\n\treturn new OpenAI({\n\t\tfetch: createGatewayFetch(\"workers-ai\", gatewayConfig, sessionHeaders),\n\t\tapiKey: gatewayConfig.apiKey ?? \"unused\",\n\t});\n}\n\n// ---------------------------------------------------------------------------\n// Shared message-building helpers\n// ---------------------------------------------------------------------------\n\nfunction extractTextContent(content: ModelMessage[\"content\"]): string {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n}\n\n/**\n * Convert a single TanStack AI {@link ContentPart} to the OpenAI Chat\n * Completions multi-modal format.\n *\n * TODO: handle other content types (audio, video, document)\n */\nfunction convertContentPart(part: ContentPart): OpenAI.Chat.ChatCompletionContentPart | undefined {\n\tswitch (part.type) {\n\t\tcase \"text\":\n\t\t\tif (part.content) {\n\t\t\t\treturn { type: \"text\", text: part.content };\n\t\t\t}\n\t\t\treturn undefined;\n\t\tcase \"image\": {\n\t\t\tlet url: string;\n\t\t\tif (part.source.type === \"data\") {\n\t\t\t\turl = part.source.value.startsWith(\"data:\")\n\t\t\t\t\t? part.source.value\n\t\t\t\t\t: `data:${part.source.mimeType};base64,${part.source.value}`;\n\t\t\t} else {\n\t\t\t\turl = part.source.value;\n\t\t\t}\n\t\t\treturn { type: \"image_url\", image_url: { url } };\n\t\t}\n\t\tdefault:\n\t\t\t// audio, video, document — not supported for now\n\t\t\tconsole.warn(\n\t\t\t\t`[@cloudflare/tanstack-ai] Unsupported content part type \"${part.type}\" — skipping`,\n\t\t\t);\n\t\t\treturn undefined;\n\t}\n}\n\n/**\n * Build OpenAI-compatible user message content from TanStack AI content.\n *\n * If the content has only text parts, returns a plain string.\n * If it includes image parts, returns an array of content parts in\n * OpenAI's multi-modal format (text + image_url).\n */\nfunction buildUserContent(\n\tcontent: ModelMessage[\"content\"],\n): string | OpenAI.Chat.ChatCompletionContentPart[] {\n\tif (content === null) return \"\";\n\tif (typeof content === \"string\") return content;\n\n\tconst hasImages = content.some((p) => p.type === \"image\");\n\tif (!hasImages) {\n\t\treturn content.flatMap((p) => (p.type === \"text\" ? [p.content] : [])).join(\"\");\n\t}\n\n\tconst parts: OpenAI.Chat.ChatCompletionContentPart[] = [];\n\tfor (const part of content) {\n\t\tconst converted = convertContentPart(part);\n\t\tif (converted) {\n\t\t\tparts.push(converted);\n\t\t}\n\t}\n\treturn parts;\n}\n\nfunction buildOpenAIMessages(\n\tsystemPrompts: string[] | undefined,\n\tmessages: ModelMessage[],\n\toptions?: { includeToolMessages?: boolean },\n): OpenAI.Chat.ChatCompletionMessageParam[] {\n\tconst includeTools = options?.includeToolMessages ?? true;\n\tconst openAIMessages: OpenAI.Chat.ChatCompletionMessageParam[] = [];\n\n\tif (systemPrompts && systemPrompts.length > 0) {\n\t\topenAIMessages.push({\n\t\t\trole: \"system\",\n\t\t\tcontent: systemPrompts.join(\"\\n\"),\n\t\t});\n\t}\n\n\tfor (const message of messages) {\n\t\tif (message.role === \"user\") {\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"user\",\n\t\t\t\tcontent: buildUserContent(message.content),\n\t\t\t});\n\t\t} else if (message.role === \"assistant\") {\n\t\t\tconst assistantMessage: OpenAI.Chat.ChatCompletionAssistantMessageParam = {\n\t\t\t\trole: \"assistant\",\n\t\t\t\tcontent: extractTextContent(message.content),\n\t\t\t};\n\t\t\tif (includeTools && message.toolCalls && message.toolCalls.length > 0) {\n\t\t\t\tassistantMessage.tool_calls = message.toolCalls.map((tc) => ({\n\t\t\t\t\tid: tc.id,\n\t\t\t\t\ttype: \"function\" as const,\n\t\t\t\t\tfunction: {\n\t\t\t\t\t\tname: tc.function.name,\n\t\t\t\t\t\targuments: tc.function.arguments,\n\t\t\t\t\t},\n\t\t\t\t}));\n\t\t\t}\n\t\t\topenAIMessages.push(assistantMessage);\n\t\t} else if (includeTools && message.role === \"tool\") {\n\t\t\tlet toolContent: string;\n\t\t\tif (typeof message.content === \"string\") {\n\t\t\t\ttry {\n\t\t\t\t\tJSON.parse(message.content);\n\t\t\t\t\ttoolContent = message.content;\n\t\t\t\t} catch {\n\t\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\ttoolContent = JSON.stringify(message.content);\n\t\t\t}\n\t\t\topenAIMessages.push({\n\t\t\t\trole: \"tool\",\n\t\t\t\ttool_call_id: message.toolCallId || `tool_${crypto.randomUUID().slice(0, 8)}`,\n\t\t\t\tcontent: toolContent,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn openAIMessages;\n}\n\nfunction buildOpenAITools(\n\ttools: Array<{ name: string; description: string; inputSchema?: unknown }> | undefined,\n): OpenAI.Chat.ChatCompletionTool[] | undefined {\n\tif (!tools) return undefined;\n\treturn tools.map((tool) => ({\n\t\ttype: \"function\" as const,\n\t\tfunction: {\n\t\t\tname: tool.name,\n\t\t\tdescription: tool.description,\n\t\t\tparameters: tool.inputSchema as Record<string, unknown>,\n\t\t},\n\t}));\n}\n\n// ---------------------------------------------------------------------------\n// ID generation\n// ---------------------------------------------------------------------------\n\nfunction generateId(prefix = \"chatcmpl\"): string {\n\treturn `${prefix}-${crypto.randomUUID().replace(/-/g, \"\").slice(0, 16)}`;\n}\n\n// ---------------------------------------------------------------------------\n// WorkersAiTextAdapter: chat / structured output via OpenAI Chat Completions\n// ---------------------------------------------------------------------------\n\n// TODO: Replace `any` generic params with proper types once BaseTextAdapter's\n// provider-options generics stabilize. Workers AI doesn't have provider-specific\n// options in the TanStack sense, so `any` is pragmatic for now.\nexport class WorkersAiTextAdapter<TModel extends WorkersAiTextModel> extends BaseTextAdapter<\n\tTModel,\n\t// eslint-disable-next-line @typescript-eslint/no-explicit-any -- BaseTextAdapter generic params are opaque\n\tany,\n\tany,\n\tany\n> {\n\tname = \"workers-ai\" as const;\n\n\tprivate client: OpenAI;\n\n\tconstructor(model: TModel, config: WorkersAiAdapterConfig) {\n\t\tsuper({ apiKey: \"unused\" }, model);\n\t\tthis.client = buildWorkersAiClient(config);\n\t}\n\n\tasync *chatStream(options: TextOptions<any>): AsyncIterable<StreamChunk> {\n\t\tconst { systemPrompts, messages, tools, temperature, maxTokens, model } = options;\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages);\n\t\tconst openAITools = buildOpenAITools(tools);\n\n\t\tconst timestamp = Date.now();\n\t\tconst runId = generateId();\n\t\tconst messageId = generateId();\n\t\tlet hasEmittedRunStarted = false;\n\t\tlet hasEmittedTextMessageStart = false;\n\t\tlet accumulatedContent = \"\";\n\t\tlet hasEmittedStepStarted = false;\n\t\tlet accumulatedReasoning = \"\";\n\t\tconst stepId = generateId();\n\t\tlet hasReceivedFinishReason = false;\n\t\tconst toolCallsInProgress = new Map<\n\t\t\tnumber,\n\t\t\t{ id: string; name: string; arguments: string; started: boolean }\n\t\t>();\n\n\t\ttry {\n\t\t\tlet stream: AsyncIterable<OpenAI.Chat.Completions.ChatCompletionChunk>;\n\t\t\ttry {\n\t\t\t\tstream = await this.client.chat.completions.create({\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t\tstream: true,\n\t\t\t\t\tstream_options: { include_usage: true },\n\t\t\t\t});\n\t\t\t} catch (streamError: unknown) {\n\t\t\t\t// Some models (e.g. GPT-OSS) don't support streaming via the REST API.\n\t\t\t\t// Fall back to a non-streaming call and yield the result as a single chunk.\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Streaming failed, falling back to non-streaming:\",\n\t\t\t\t\tstreamError instanceof Error ? streamError.message : streamError,\n\t\t\t\t);\n\t\t\t\tconst nonStreamResult = await this.client.chat.completions.create({\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\tmessages: openAIMessages,\n\t\t\t\t\ttools: openAITools,\n\t\t\t\t\ttemperature,\n\t\t\t\t\tmax_tokens: maxTokens,\n\t\t\t\t});\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\n\t\t\t\tconst msg = nonStreamResult.choices[0]?.message;\n\t\t\t\tif (msg?.content) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: msg.content,\n\t\t\t\t\t\tcontent: msg.content,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tif (msg?.tool_calls) {\n\t\t\t\t\tfor (const tc of msg.tool_calls) {\n\t\t\t\t\t\tif (tc.type !== \"function\") continue;\n\t\t\t\t\t\tconst fn = tc.function;\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = fn.arguments ? JSON.parse(fn.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tindex: 0,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: tc.id,\n\t\t\t\t\t\t\ttoolName: fn.name,\n\t\t\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst finishReason = nonStreamResult.choices[0]?.finish_reason;\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: nonStreamResult.model || model || this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tusage: nonStreamResult.usage\n\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\tpromptTokens: nonStreamResult.usage.prompt_tokens,\n\t\t\t\t\t\t\t\tcompletionTokens: nonStreamResult.usage.completion_tokens,\n\t\t\t\t\t\t\t\ttotalTokens: nonStreamResult.usage.total_tokens,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t: undefined,\n\t\t\t\t\tfinishReason:\n\t\t\t\t\t\tfinishReason === \"tool_calls\" || finishReason === \"function_call\"\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: ((finishReason as \"stop\" | \"length\" | \"content_filter\") ?? \"stop\"),\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor await (const chunk of stream) {\n\t\t\t\tif (!chunk.choices || chunk.choices.length === 0) continue;\n\t\t\t\tconst choice = chunk.choices[0];\n\t\t\t\tif (!choice) continue;\n\n\t\t\t\t// Emit RUN_STARTED on first chunk\n\t\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\t\thasEmittedRunStarted = true;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tconst delta = choice.delta;\n\n\t\t\t\t// Reasoning content (used by models like QwQ, DeepSeek R1, Kimi K2.5)\n\t\t\t\t// The OpenAI SDK doesn't type this field, but models send it as an extension.\n\t\t\t\tconst reasoningContent = ((delta as Record<string, unknown>).reasoning_content ??\n\t\t\t\t\t(delta as Record<string, unknown>).reasoning) as string | undefined;\n\t\t\t\tif (reasoningContent) {\n\t\t\t\t\t// RUN_STARTED is already guaranteed by the guard above\n\t\t\t\t\tif (!hasEmittedStepStarted) {\n\t\t\t\t\t\thasEmittedStepStarted = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"STEP_STARTED\",\n\t\t\t\t\t\t\tstepId,\n\t\t\t\t\t\t\tstepType: \"thinking\",\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t\taccumulatedReasoning += reasoningContent;\n\t\t\t\t\t// TODO: TanStack AI's StreamProcessor currently treats STEP_FINISHED as an\n\t\t\t\t\t// incremental reasoning event (with `delta` + accumulated `content`), so we\n\t\t\t\t\t// emit one per token. If TanStack AI adds a dedicated STEP_CONTENT event\n\t\t\t\t\t// type, this should be updated to emit STEP_CONTENT per token and a single\n\t\t\t\t\t// STEP_FINISHED when reasoning ends (i.e. when the first non-reasoning\n\t\t\t\t\t// content or finish_reason arrives).\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"STEP_FINISHED\",\n\t\t\t\t\t\tstepId,\n\t\t\t\t\t\tdelta: reasoningContent,\n\t\t\t\t\t\tcontent: accumulatedReasoning,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Text content\n\t\t\t\tif (delta.content) {\n\t\t\t\t\tif (!hasEmittedTextMessageStart) {\n\t\t\t\t\t\thasEmittedTextMessageStart = true;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_START\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\trole: \"assistant\",\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\taccumulatedContent += delta.content;\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_CONTENT\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tdelta: delta.content,\n\t\t\t\t\t\tcontent: accumulatedContent,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\t// Tool calls\n\t\t\t\tif (delta.tool_calls) {\n\t\t\t\t\tfor (const toolCallDelta of delta.tool_calls) {\n\t\t\t\t\t\tconst index = toolCallDelta.index;\n\n\t\t\t\t\t\tif (!toolCallsInProgress.has(index)) {\n\t\t\t\t\t\t\t// Always generate a unique ID per tool call index.\n\t\t\t\t\t\t\t// The backend may send the same ID for multiple tool calls,\n\t\t\t\t\t\t\t// so we cannot trust toolCallDelta.id to be unique.\n\t\t\t\t\t\t\ttoolCallsInProgress.set(index, {\n\t\t\t\t\t\t\t\tid: generateId(\"chatcmpl-tool\"),\n\t\t\t\t\t\t\t\tname: toolCallDelta.function?.name || \"\",\n\t\t\t\t\t\t\t\targuments: \"\",\n\t\t\t\t\t\t\t\tstarted: false,\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst toolCall = toolCallsInProgress.get(index)!;\n\n\t\t\t\t\t\t// Only update name if provided (ID is already set at creation time\n\t\t\t\t\t\t// and should not be overwritten by subsequent chunks that may have\n\t\t\t\t\t\t// duplicate/shared IDs from the backend)\n\t\t\t\t\t\tif (toolCallDelta.function?.name) {\n\t\t\t\t\t\t\ttoolCall.name = toolCallDelta.function.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments) {\n\t\t\t\t\t\t\ttoolCall.arguments += toolCallDelta.function.arguments;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Emit TOOL_CALL_START once we have id and name\n\t\t\t\t\t\tif (toolCall.id && toolCall.name && !toolCall.started) {\n\t\t\t\t\t\t\ttoolCall.started = true;\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_START\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tindex,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Stream tool call arguments\n\t\t\t\t\t\tif (toolCallDelta.function?.arguments && toolCall.started) {\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_ARGS\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tdelta: toolCallDelta.function.arguments,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Finish\n\t\t\t\tif (choice.finish_reason) {\n\t\t\t\t\thasReceivedFinishReason = true;\n\n\t\t\t\t\t// End tool calls\n\t\t\t\t\tif (choice.finish_reason === \"tool_calls\" || toolCallsInProgress.size > 0) {\n\t\t\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\t\tparsedInput = toolCall.arguments\n\t\t\t\t\t\t\t\t\t? JSON.parse(toolCall.arguments)\n\t\t\t\t\t\t\t\t\t: {};\n\t\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tconst computedFinishReason =\n\t\t\t\t\t\tchoice.finish_reason === \"tool_calls\" ||\n\t\t\t\t\t\tchoice.finish_reason === \"function_call\" ||\n\t\t\t\t\t\ttoolCallsInProgress.size > 0\n\t\t\t\t\t\t\t? \"tool_calls\"\n\t\t\t\t\t\t\t: (choice.finish_reason as \"stop\" | \"length\" | \"content_filter\");\n\n\t\t\t\t\t// End text message if started\n\t\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Emit RUN_FINISHED\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\t\trunId,\n\t\t\t\t\t\tmodel: chunk.model || model || this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\tusage: chunk.usage\n\t\t\t\t\t\t\t? {\n\t\t\t\t\t\t\t\t\tpromptTokens: chunk.usage.prompt_tokens,\n\t\t\t\t\t\t\t\t\tcompletionTokens: chunk.usage.completion_tokens,\n\t\t\t\t\t\t\t\t\ttotalTokens: chunk.usage.total_tokens,\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t: undefined,\n\t\t\t\t\t\tfinishReason: computedFinishReason,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Premature stream termination: the stream ended without a finish_reason.\n\t\t\t// This can happen when Workers AI truncates a response or the connection drops.\n\t\t\t// Emit proper closing events so the consumer doesn't hang.\n\t\t\tif (hasEmittedRunStarted && !hasReceivedFinishReason) {\n\t\t\t\tconsole.warn(\n\t\t\t\t\t\"[tanstack-ai] Stream ended without finish_reason — possible truncation or connection drop\",\n\t\t\t\t);\n\n\t\t\t\t// Close any open tool calls\n\t\t\t\tfor (const [, toolCall] of toolCallsInProgress) {\n\t\t\t\t\tif (toolCall.started) {\n\t\t\t\t\t\tlet parsedInput: unknown = {};\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tparsedInput = toolCall.arguments ? JSON.parse(toolCall.arguments) : {};\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\tparsedInput = {};\n\t\t\t\t\t\t}\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\ttype: \"TOOL_CALL_END\",\n\t\t\t\t\t\t\ttoolCallId: toolCall.id,\n\t\t\t\t\t\t\ttoolName: toolCall.name,\n\t\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t\t\tinput: parsedInput,\n\t\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Close text message if open\n\t\t\t\tif (hasEmittedTextMessageStart) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\ttype: \"TEXT_MESSAGE_END\",\n\t\t\t\t\t\tmessageId,\n\t\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\t\ttimestamp,\n\t\t\t\t\t} satisfies StreamChunk;\n\t\t\t\t}\n\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_FINISHED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t\tfinishReason: \"stop\",\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\tconst code =\n\t\t\t\terror instanceof Error ? (error as Error & { code?: string }).code : undefined;\n\t\t\tif (!hasEmittedRunStarted) {\n\t\t\t\tyield {\n\t\t\t\t\ttype: \"RUN_STARTED\",\n\t\t\t\t\trunId,\n\t\t\t\t\tmodel: model ?? this.model,\n\t\t\t\t\ttimestamp,\n\t\t\t\t} satisfies StreamChunk;\n\t\t\t}\n\t\t\tyield {\n\t\t\t\ttype: \"RUN_ERROR\",\n\t\t\t\trunId,\n\t\t\t\tmodel: model ?? this.model,\n\t\t\t\ttimestamp,\n\t\t\t\terror: {\n\t\t\t\t\tmessage: message || \"Unknown error\",\n\t\t\t\t\tcode,\n\t\t\t\t},\n\t\t\t} satisfies StreamChunk;\n\t\t}\n\t}\n\n\tasync structuredOutput(\n\t\toptions: StructuredOutputOptions<any>,\n\t): Promise<StructuredOutputResult<unknown>> {\n\t\tconst { outputSchema, chatOptions } = options;\n\t\tconst { systemPrompts, messages, temperature, model } = chatOptions;\n\n\t\tconst openAIMessages = buildOpenAIMessages(systemPrompts, messages, {\n\t\t\tincludeToolMessages: false,\n\t\t});\n\n\t\tconst response = await this.client.chat.completions.create({\n\t\t\tmodel: model ?? this.model,\n\t\t\tmessages: openAIMessages,\n\t\t\ttemperature,\n\t\t\tstream: false,\n\t\t\tresponse_format: {\n\t\t\t\ttype: \"json_schema\",\n\t\t\t\tjson_schema: {\n\t\t\t\t\tname: \"structured_output\",\n\t\t\t\t\tstrict: true,\n\t\t\t\t\tschema: outputSchema,\n\t\t\t\t},\n\t\t\t},\n\t\t});\n\n\t\tconst choice = response.choices?.[0];\n\n\t\tif (!choice) {\n\t\t\tthrow new Error(\n\t\t\t\t`Workers AI structured output returned no choices: ${JSON.stringify(response)}`,\n\t\t\t);\n\t\t}\n\n\t\tconst rawContent = choice.message?.content ?? \"\";\n\n\t\t// Workers AI REST endpoint may return `content` as an already-parsed object\n\t\t// when using json_schema response format, so normalise both cases.\n\t\tlet data: unknown;\n\t\tlet rawText: string;\n\n\t\tif (typeof rawContent === \"string\") {\n\t\t\trawText = rawContent;\n\t\t\ttry {\n\t\t\t\tdata = JSON.parse(rawText);\n\t\t\t} catch {\n\t\t\t\tdata = rawText;\n\t\t\t}\n\t\t} else {\n\t\t\t// Already an object — stringify for rawText, use directly for data\n\t\t\tdata = rawContent;\n\t\t\trawText = JSON.stringify(rawContent);\n\t\t}\n\n\t\treturn { data, rawText };\n\t}\n}\n\n// ---------------------------------------------------------------------------\n// Factory functions\n// ---------------------------------------------------------------------------\n\nexport function createWorkersAiChat(model: WorkersAiTextModel, config: WorkersAiAdapterConfig) {\n\treturn new WorkersAiTextAdapter(model, config);\n}\n"],"mappings":"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"}