@draht/ai 2026.3.2-2

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 (154) hide show
  1. package/README.md +1185 -0
  2. package/dist/api-registry.d.ts +20 -0
  3. package/dist/api-registry.d.ts.map +1 -0
  4. package/dist/api-registry.js +44 -0
  5. package/dist/api-registry.js.map +1 -0
  6. package/dist/cli.d.ts +3 -0
  7. package/dist/cli.d.ts.map +1 -0
  8. package/dist/cli.js +116 -0
  9. package/dist/cli.js.map +1 -0
  10. package/dist/env-api-keys.d.ts +9 -0
  11. package/dist/env-api-keys.d.ts.map +1 -0
  12. package/dist/env-api-keys.js +99 -0
  13. package/dist/env-api-keys.js.map +1 -0
  14. package/dist/index.d.ts +22 -0
  15. package/dist/index.d.ts.map +1 -0
  16. package/dist/index.js +21 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/models.d.ts +24 -0
  19. package/dist/models.d.ts.map +1 -0
  20. package/dist/models.generated.d.ts +13133 -0
  21. package/dist/models.generated.d.ts.map +1 -0
  22. package/dist/models.generated.js +12939 -0
  23. package/dist/models.generated.js.map +1 -0
  24. package/dist/models.js +55 -0
  25. package/dist/models.js.map +1 -0
  26. package/dist/providers/amazon-bedrock.d.ts +15 -0
  27. package/dist/providers/amazon-bedrock.d.ts.map +1 -0
  28. package/dist/providers/amazon-bedrock.js +585 -0
  29. package/dist/providers/amazon-bedrock.js.map +1 -0
  30. package/dist/providers/anthropic.d.ts +33 -0
  31. package/dist/providers/anthropic.d.ts.map +1 -0
  32. package/dist/providers/anthropic.js +729 -0
  33. package/dist/providers/anthropic.js.map +1 -0
  34. package/dist/providers/azure-openai-responses.d.ts +15 -0
  35. package/dist/providers/azure-openai-responses.d.ts.map +1 -0
  36. package/dist/providers/azure-openai-responses.js +184 -0
  37. package/dist/providers/azure-openai-responses.js.map +1 -0
  38. package/dist/providers/github-copilot-headers.d.ts +8 -0
  39. package/dist/providers/github-copilot-headers.d.ts.map +1 -0
  40. package/dist/providers/github-copilot-headers.js +29 -0
  41. package/dist/providers/github-copilot-headers.js.map +1 -0
  42. package/dist/providers/google-gemini-cli.d.ts +74 -0
  43. package/dist/providers/google-gemini-cli.d.ts.map +1 -0
  44. package/dist/providers/google-gemini-cli.js +735 -0
  45. package/dist/providers/google-gemini-cli.js.map +1 -0
  46. package/dist/providers/google-shared.d.ts +65 -0
  47. package/dist/providers/google-shared.d.ts.map +1 -0
  48. package/dist/providers/google-shared.js +306 -0
  49. package/dist/providers/google-shared.js.map +1 -0
  50. package/dist/providers/google-vertex.d.ts +15 -0
  51. package/dist/providers/google-vertex.d.ts.map +1 -0
  52. package/dist/providers/google-vertex.js +371 -0
  53. package/dist/providers/google-vertex.js.map +1 -0
  54. package/dist/providers/google.d.ts +13 -0
  55. package/dist/providers/google.d.ts.map +1 -0
  56. package/dist/providers/google.js +352 -0
  57. package/dist/providers/google.js.map +1 -0
  58. package/dist/providers/openai-codex-responses.d.ts +9 -0
  59. package/dist/providers/openai-codex-responses.d.ts.map +1 -0
  60. package/dist/providers/openai-codex-responses.js +699 -0
  61. package/dist/providers/openai-codex-responses.js.map +1 -0
  62. package/dist/providers/openai-completions.d.ts +15 -0
  63. package/dist/providers/openai-completions.d.ts.map +1 -0
  64. package/dist/providers/openai-completions.js +712 -0
  65. package/dist/providers/openai-completions.js.map +1 -0
  66. package/dist/providers/openai-responses-shared.d.ts +17 -0
  67. package/dist/providers/openai-responses-shared.d.ts.map +1 -0
  68. package/dist/providers/openai-responses-shared.js +427 -0
  69. package/dist/providers/openai-responses-shared.js.map +1 -0
  70. package/dist/providers/openai-responses.d.ts +13 -0
  71. package/dist/providers/openai-responses.d.ts.map +1 -0
  72. package/dist/providers/openai-responses.js +198 -0
  73. package/dist/providers/openai-responses.js.map +1 -0
  74. package/dist/providers/register-builtins.d.ts +3 -0
  75. package/dist/providers/register-builtins.d.ts.map +1 -0
  76. package/dist/providers/register-builtins.js +63 -0
  77. package/dist/providers/register-builtins.js.map +1 -0
  78. package/dist/providers/simple-options.d.ts +8 -0
  79. package/dist/providers/simple-options.d.ts.map +1 -0
  80. package/dist/providers/simple-options.js +35 -0
  81. package/dist/providers/simple-options.js.map +1 -0
  82. package/dist/providers/transform-messages.d.ts +8 -0
  83. package/dist/providers/transform-messages.d.ts.map +1 -0
  84. package/dist/providers/transform-messages.js +155 -0
  85. package/dist/providers/transform-messages.js.map +1 -0
  86. package/dist/stream.d.ts +9 -0
  87. package/dist/stream.d.ts.map +1 -0
  88. package/dist/stream.js +28 -0
  89. package/dist/stream.js.map +1 -0
  90. package/dist/types.d.ts +279 -0
  91. package/dist/types.d.ts.map +1 -0
  92. package/dist/types.js +2 -0
  93. package/dist/types.js.map +1 -0
  94. package/dist/utils/event-stream.d.ts +21 -0
  95. package/dist/utils/event-stream.d.ts.map +1 -0
  96. package/dist/utils/event-stream.js +81 -0
  97. package/dist/utils/event-stream.js.map +1 -0
  98. package/dist/utils/http-proxy.d.ts +2 -0
  99. package/dist/utils/http-proxy.d.ts.map +1 -0
  100. package/dist/utils/http-proxy.js +15 -0
  101. package/dist/utils/http-proxy.js.map +1 -0
  102. package/dist/utils/json-parse.d.ts +9 -0
  103. package/dist/utils/json-parse.d.ts.map +1 -0
  104. package/dist/utils/json-parse.js +29 -0
  105. package/dist/utils/json-parse.js.map +1 -0
  106. package/dist/utils/oauth/anthropic.d.ts +17 -0
  107. package/dist/utils/oauth/anthropic.d.ts.map +1 -0
  108. package/dist/utils/oauth/anthropic.js +104 -0
  109. package/dist/utils/oauth/anthropic.js.map +1 -0
  110. package/dist/utils/oauth/github-copilot.d.ts +30 -0
  111. package/dist/utils/oauth/github-copilot.d.ts.map +1 -0
  112. package/dist/utils/oauth/github-copilot.js +281 -0
  113. package/dist/utils/oauth/github-copilot.js.map +1 -0
  114. package/dist/utils/oauth/google-antigravity.d.ts +26 -0
  115. package/dist/utils/oauth/google-antigravity.d.ts.map +1 -0
  116. package/dist/utils/oauth/google-antigravity.js +373 -0
  117. package/dist/utils/oauth/google-antigravity.js.map +1 -0
  118. package/dist/utils/oauth/google-gemini-cli.d.ts +26 -0
  119. package/dist/utils/oauth/google-gemini-cli.d.ts.map +1 -0
  120. package/dist/utils/oauth/google-gemini-cli.js +478 -0
  121. package/dist/utils/oauth/google-gemini-cli.js.map +1 -0
  122. package/dist/utils/oauth/index.d.ts +62 -0
  123. package/dist/utils/oauth/index.d.ts.map +1 -0
  124. package/dist/utils/oauth/index.js +133 -0
  125. package/dist/utils/oauth/index.js.map +1 -0
  126. package/dist/utils/oauth/openai-codex.d.ts +34 -0
  127. package/dist/utils/oauth/openai-codex.d.ts.map +1 -0
  128. package/dist/utils/oauth/openai-codex.js +380 -0
  129. package/dist/utils/oauth/openai-codex.js.map +1 -0
  130. package/dist/utils/oauth/pkce.d.ts +13 -0
  131. package/dist/utils/oauth/pkce.d.ts.map +1 -0
  132. package/dist/utils/oauth/pkce.js +31 -0
  133. package/dist/utils/oauth/pkce.js.map +1 -0
  134. package/dist/utils/oauth/types.d.ts +47 -0
  135. package/dist/utils/oauth/types.d.ts.map +1 -0
  136. package/dist/utils/oauth/types.js +2 -0
  137. package/dist/utils/oauth/types.js.map +1 -0
  138. package/dist/utils/overflow.d.ts +52 -0
  139. package/dist/utils/overflow.d.ts.map +1 -0
  140. package/dist/utils/overflow.js +115 -0
  141. package/dist/utils/overflow.js.map +1 -0
  142. package/dist/utils/sanitize-unicode.d.ts +22 -0
  143. package/dist/utils/sanitize-unicode.d.ts.map +1 -0
  144. package/dist/utils/sanitize-unicode.js +26 -0
  145. package/dist/utils/sanitize-unicode.js.map +1 -0
  146. package/dist/utils/typebox-helpers.d.ts +17 -0
  147. package/dist/utils/typebox-helpers.d.ts.map +1 -0
  148. package/dist/utils/typebox-helpers.js +21 -0
  149. package/dist/utils/typebox-helpers.js.map +1 -0
  150. package/dist/utils/validation.d.ts +18 -0
  151. package/dist/utils/validation.d.ts.map +1 -0
  152. package/dist/utils/validation.js +72 -0
  153. package/dist/utils/validation.js.map +1 -0
  154. package/package.json +67 -0
@@ -0,0 +1,585 @@
1
+ import { BedrockRuntimeClient, StopReason as BedrockStopReason, CachePointType, CacheTTL, ConversationRole, ConverseStreamCommand, ImageFormat, ToolResultStatus, } from "@aws-sdk/client-bedrock-runtime";
2
+ import { calculateCost } from "../models.js";
3
+ import { AssistantMessageEventStream } from "../utils/event-stream.js";
4
+ import { parseStreamingJson } from "../utils/json-parse.js";
5
+ import { sanitizeSurrogates } from "../utils/sanitize-unicode.js";
6
+ import { adjustMaxTokensForThinking, buildBaseOptions, clampReasoning } from "./simple-options.js";
7
+ import { transformMessages } from "./transform-messages.js";
8
+ export const streamBedrock = (model, context, options = {}) => {
9
+ const stream = new AssistantMessageEventStream();
10
+ (async () => {
11
+ const output = {
12
+ role: "assistant",
13
+ content: [],
14
+ api: "bedrock-converse-stream",
15
+ provider: model.provider,
16
+ model: model.id,
17
+ usage: {
18
+ input: 0,
19
+ output: 0,
20
+ cacheRead: 0,
21
+ cacheWrite: 0,
22
+ totalTokens: 0,
23
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
24
+ },
25
+ stopReason: "stop",
26
+ timestamp: Date.now(),
27
+ };
28
+ const blocks = output.content;
29
+ const config = {
30
+ region: options.region,
31
+ profile: options.profile,
32
+ };
33
+ // in Node.js/Bun environment only
34
+ if (typeof process !== "undefined" && (process.versions?.node || process.versions?.bun)) {
35
+ config.region = config.region || process.env.AWS_REGION || process.env.AWS_DEFAULT_REGION;
36
+ // Support proxies that don't need authentication
37
+ if (process.env.AWS_BEDROCK_SKIP_AUTH === "1") {
38
+ config.credentials = {
39
+ accessKeyId: "dummy-access-key",
40
+ secretAccessKey: "dummy-secret-key",
41
+ };
42
+ }
43
+ if (process.env.HTTP_PROXY ||
44
+ process.env.HTTPS_PROXY ||
45
+ process.env.NO_PROXY ||
46
+ process.env.http_proxy ||
47
+ process.env.https_proxy ||
48
+ process.env.no_proxy) {
49
+ const nodeHttpHandler = await import("@smithy/node-http-handler");
50
+ const proxyAgent = await import("proxy-agent");
51
+ const agent = new proxyAgent.ProxyAgent();
52
+ // Bedrock runtime uses NodeHttp2Handler by default since v3.798.0, which is based
53
+ // on `http2` module and has no support for http agent.
54
+ // Use NodeHttpHandler to support http agent.
55
+ config.requestHandler = new nodeHttpHandler.NodeHttpHandler({
56
+ httpAgent: agent,
57
+ httpsAgent: agent,
58
+ });
59
+ }
60
+ else if (process.env.AWS_BEDROCK_FORCE_HTTP1 === "1") {
61
+ // Some custom endpoints require HTTP/1.1 instead of HTTP/2
62
+ const nodeHttpHandler = await import("@smithy/node-http-handler");
63
+ config.requestHandler = new nodeHttpHandler.NodeHttpHandler();
64
+ }
65
+ }
66
+ config.region = config.region || "us-east-1";
67
+ try {
68
+ const client = new BedrockRuntimeClient(config);
69
+ const cacheRetention = resolveCacheRetention(options.cacheRetention);
70
+ const commandInput = {
71
+ modelId: model.id,
72
+ messages: convertMessages(context, model, cacheRetention),
73
+ system: buildSystemPrompt(context.systemPrompt, model, cacheRetention),
74
+ inferenceConfig: { maxTokens: options.maxTokens, temperature: options.temperature },
75
+ toolConfig: convertToolConfig(context.tools, options.toolChoice),
76
+ additionalModelRequestFields: buildAdditionalModelRequestFields(model, options),
77
+ };
78
+ options?.onPayload?.(commandInput);
79
+ const command = new ConverseStreamCommand(commandInput);
80
+ const response = await client.send(command, { abortSignal: options.signal });
81
+ for await (const item of response.stream) {
82
+ if (item.messageStart) {
83
+ if (item.messageStart.role !== ConversationRole.ASSISTANT) {
84
+ throw new Error("Unexpected assistant message start but got user message start instead");
85
+ }
86
+ stream.push({ type: "start", partial: output });
87
+ }
88
+ else if (item.contentBlockStart) {
89
+ handleContentBlockStart(item.contentBlockStart, blocks, output, stream);
90
+ }
91
+ else if (item.contentBlockDelta) {
92
+ handleContentBlockDelta(item.contentBlockDelta, blocks, output, stream);
93
+ }
94
+ else if (item.contentBlockStop) {
95
+ handleContentBlockStop(item.contentBlockStop, blocks, output, stream);
96
+ }
97
+ else if (item.messageStop) {
98
+ output.stopReason = mapStopReason(item.messageStop.stopReason);
99
+ }
100
+ else if (item.metadata) {
101
+ handleMetadata(item.metadata, model, output);
102
+ }
103
+ else if (item.internalServerException) {
104
+ throw new Error(`Internal server error: ${item.internalServerException.message}`);
105
+ }
106
+ else if (item.modelStreamErrorException) {
107
+ throw new Error(`Model stream error: ${item.modelStreamErrorException.message}`);
108
+ }
109
+ else if (item.validationException) {
110
+ throw new Error(`Validation error: ${item.validationException.message}`);
111
+ }
112
+ else if (item.throttlingException) {
113
+ throw new Error(`Throttling error: ${item.throttlingException.message}`);
114
+ }
115
+ else if (item.serviceUnavailableException) {
116
+ throw new Error(`Service unavailable: ${item.serviceUnavailableException.message}`);
117
+ }
118
+ }
119
+ if (options.signal?.aborted) {
120
+ throw new Error("Request was aborted");
121
+ }
122
+ if (output.stopReason === "error" || output.stopReason === "aborted") {
123
+ throw new Error("An unknown error occurred");
124
+ }
125
+ stream.push({ type: "done", reason: output.stopReason, message: output });
126
+ stream.end();
127
+ }
128
+ catch (error) {
129
+ for (const block of output.content) {
130
+ delete block.index;
131
+ delete block.partialJson;
132
+ }
133
+ output.stopReason = options.signal?.aborted ? "aborted" : "error";
134
+ output.errorMessage = error instanceof Error ? error.message : JSON.stringify(error);
135
+ stream.push({ type: "error", reason: output.stopReason, error: output });
136
+ stream.end();
137
+ }
138
+ })();
139
+ return stream;
140
+ };
141
+ export const streamSimpleBedrock = (model, context, options) => {
142
+ const base = buildBaseOptions(model, options, undefined);
143
+ if (!options?.reasoning) {
144
+ return streamBedrock(model, context, { ...base, reasoning: undefined });
145
+ }
146
+ if (model.id.includes("anthropic.claude") || model.id.includes("anthropic/claude")) {
147
+ if (supportsAdaptiveThinking(model.id)) {
148
+ return streamBedrock(model, context, {
149
+ ...base,
150
+ reasoning: options.reasoning,
151
+ thinkingBudgets: options.thinkingBudgets,
152
+ });
153
+ }
154
+ const adjusted = adjustMaxTokensForThinking(base.maxTokens || 0, model.maxTokens, options.reasoning, options.thinkingBudgets);
155
+ return streamBedrock(model, context, {
156
+ ...base,
157
+ maxTokens: adjusted.maxTokens,
158
+ reasoning: options.reasoning,
159
+ thinkingBudgets: {
160
+ ...(options.thinkingBudgets || {}),
161
+ [clampReasoning(options.reasoning)]: adjusted.thinkingBudget,
162
+ },
163
+ });
164
+ }
165
+ return streamBedrock(model, context, {
166
+ ...base,
167
+ reasoning: options.reasoning,
168
+ thinkingBudgets: options.thinkingBudgets,
169
+ });
170
+ };
171
+ function handleContentBlockStart(event, blocks, output, stream) {
172
+ const index = event.contentBlockIndex;
173
+ const start = event.start;
174
+ if (start?.toolUse) {
175
+ const block = {
176
+ type: "toolCall",
177
+ id: start.toolUse.toolUseId || "",
178
+ name: start.toolUse.name || "",
179
+ arguments: {},
180
+ partialJson: "",
181
+ index,
182
+ };
183
+ output.content.push(block);
184
+ stream.push({ type: "toolcall_start", contentIndex: blocks.length - 1, partial: output });
185
+ }
186
+ }
187
+ function handleContentBlockDelta(event, blocks, output, stream) {
188
+ const contentBlockIndex = event.contentBlockIndex;
189
+ const delta = event.delta;
190
+ let index = blocks.findIndex((b) => b.index === contentBlockIndex);
191
+ let block = blocks[index];
192
+ if (delta?.text !== undefined) {
193
+ // If no text block exists yet, create one, as `handleContentBlockStart` is not sent for text blocks
194
+ if (!block) {
195
+ const newBlock = { type: "text", text: "", index: contentBlockIndex };
196
+ output.content.push(newBlock);
197
+ index = blocks.length - 1;
198
+ block = blocks[index];
199
+ stream.push({ type: "text_start", contentIndex: index, partial: output });
200
+ }
201
+ if (block.type === "text") {
202
+ block.text += delta.text;
203
+ stream.push({ type: "text_delta", contentIndex: index, delta: delta.text, partial: output });
204
+ }
205
+ }
206
+ else if (delta?.toolUse && block?.type === "toolCall") {
207
+ block.partialJson = (block.partialJson || "") + (delta.toolUse.input || "");
208
+ block.arguments = parseStreamingJson(block.partialJson);
209
+ stream.push({ type: "toolcall_delta", contentIndex: index, delta: delta.toolUse.input || "", partial: output });
210
+ }
211
+ else if (delta?.reasoningContent) {
212
+ let thinkingBlock = block;
213
+ let thinkingIndex = index;
214
+ if (!thinkingBlock) {
215
+ const newBlock = { type: "thinking", thinking: "", thinkingSignature: "", index: contentBlockIndex };
216
+ output.content.push(newBlock);
217
+ thinkingIndex = blocks.length - 1;
218
+ thinkingBlock = blocks[thinkingIndex];
219
+ stream.push({ type: "thinking_start", contentIndex: thinkingIndex, partial: output });
220
+ }
221
+ if (thinkingBlock?.type === "thinking") {
222
+ if (delta.reasoningContent.text) {
223
+ thinkingBlock.thinking += delta.reasoningContent.text;
224
+ stream.push({
225
+ type: "thinking_delta",
226
+ contentIndex: thinkingIndex,
227
+ delta: delta.reasoningContent.text,
228
+ partial: output,
229
+ });
230
+ }
231
+ if (delta.reasoningContent.signature) {
232
+ thinkingBlock.thinkingSignature =
233
+ (thinkingBlock.thinkingSignature || "") + delta.reasoningContent.signature;
234
+ }
235
+ }
236
+ }
237
+ }
238
+ function handleMetadata(event, model, output) {
239
+ if (event.usage) {
240
+ output.usage.input = event.usage.inputTokens || 0;
241
+ output.usage.output = event.usage.outputTokens || 0;
242
+ output.usage.cacheRead = event.usage.cacheReadInputTokens || 0;
243
+ output.usage.cacheWrite = event.usage.cacheWriteInputTokens || 0;
244
+ output.usage.totalTokens = event.usage.totalTokens || output.usage.input + output.usage.output;
245
+ calculateCost(model, output.usage);
246
+ }
247
+ }
248
+ function handleContentBlockStop(event, blocks, output, stream) {
249
+ const index = blocks.findIndex((b) => b.index === event.contentBlockIndex);
250
+ const block = blocks[index];
251
+ if (!block)
252
+ return;
253
+ delete block.index;
254
+ switch (block.type) {
255
+ case "text":
256
+ stream.push({ type: "text_end", contentIndex: index, content: block.text, partial: output });
257
+ break;
258
+ case "thinking":
259
+ stream.push({ type: "thinking_end", contentIndex: index, content: block.thinking, partial: output });
260
+ break;
261
+ case "toolCall":
262
+ block.arguments = parseStreamingJson(block.partialJson);
263
+ delete block.partialJson;
264
+ stream.push({ type: "toolcall_end", contentIndex: index, toolCall: block, partial: output });
265
+ break;
266
+ }
267
+ }
268
+ /**
269
+ * Check if the model supports adaptive thinking (Opus 4.6 and Sonnet 4.6).
270
+ */
271
+ function supportsAdaptiveThinking(modelId) {
272
+ return (modelId.includes("opus-4-6") ||
273
+ modelId.includes("opus-4.6") ||
274
+ modelId.includes("sonnet-4-6") ||
275
+ modelId.includes("sonnet-4.6"));
276
+ }
277
+ function mapThinkingLevelToEffort(level, modelId) {
278
+ switch (level) {
279
+ case "minimal":
280
+ case "low":
281
+ return "low";
282
+ case "medium":
283
+ return "medium";
284
+ case "high":
285
+ return "high";
286
+ case "xhigh":
287
+ return modelId.includes("opus-4-6") || modelId.includes("opus-4.6") ? "max" : "high";
288
+ default:
289
+ return "high";
290
+ }
291
+ }
292
+ /**
293
+ * Resolve cache retention preference.
294
+ * Defaults to "short" and uses DRAHT_CACHE_RETENTION for backward compatibility.
295
+ */
296
+ function resolveCacheRetention(cacheRetention) {
297
+ if (cacheRetention) {
298
+ return cacheRetention;
299
+ }
300
+ if (typeof process !== "undefined" && process.env.DRAHT_CACHE_RETENTION === "long") {
301
+ return "long";
302
+ }
303
+ return "short";
304
+ }
305
+ /**
306
+ * Check if the model supports prompt caching.
307
+ * Supported: Claude 3.5 Haiku, Claude 3.7 Sonnet, Claude 4.x models
308
+ */
309
+ function supportsPromptCaching(model) {
310
+ if (model.cost.cacheRead || model.cost.cacheWrite) {
311
+ return true;
312
+ }
313
+ const id = model.id.toLowerCase();
314
+ // Claude 4.x models (opus-4, sonnet-4, haiku-4)
315
+ if (id.includes("claude") && (id.includes("-4-") || id.includes("-4.")))
316
+ return true;
317
+ // Claude 3.7 Sonnet
318
+ if (id.includes("claude-3-7-sonnet"))
319
+ return true;
320
+ // Claude 3.5 Haiku
321
+ if (id.includes("claude-3-5-haiku"))
322
+ return true;
323
+ return false;
324
+ }
325
+ /**
326
+ * Check if the model supports thinking signatures in reasoningContent.
327
+ * Only Anthropic Claude models support the signature field.
328
+ * Other models (OpenAI, Qwen, Minimax, Moonshot, etc.) reject it with:
329
+ * "This model doesn't support the reasoningContent.reasoningText.signature field"
330
+ */
331
+ function supportsThinkingSignature(model) {
332
+ const id = model.id.toLowerCase();
333
+ return id.includes("anthropic.claude") || id.includes("anthropic/claude");
334
+ }
335
+ function buildSystemPrompt(systemPrompt, model, cacheRetention) {
336
+ if (!systemPrompt)
337
+ return undefined;
338
+ const blocks = [{ text: sanitizeSurrogates(systemPrompt) }];
339
+ // Add cache point for supported Claude models when caching is enabled
340
+ if (cacheRetention !== "none" && supportsPromptCaching(model)) {
341
+ blocks.push({
342
+ cachePoint: { type: CachePointType.DEFAULT, ...(cacheRetention === "long" ? { ttl: CacheTTL.ONE_HOUR } : {}) },
343
+ });
344
+ }
345
+ return blocks;
346
+ }
347
+ function normalizeToolCallId(id) {
348
+ const sanitized = id.replace(/[^a-zA-Z0-9_-]/g, "_");
349
+ return sanitized.length > 64 ? sanitized.slice(0, 64) : sanitized;
350
+ }
351
+ function convertMessages(context, model, cacheRetention) {
352
+ const result = [];
353
+ const transformedMessages = transformMessages(context.messages, model, normalizeToolCallId);
354
+ for (let i = 0; i < transformedMessages.length; i++) {
355
+ const m = transformedMessages[i];
356
+ switch (m.role) {
357
+ case "user":
358
+ result.push({
359
+ role: ConversationRole.USER,
360
+ content: typeof m.content === "string"
361
+ ? [{ text: sanitizeSurrogates(m.content) }]
362
+ : m.content.map((c) => {
363
+ switch (c.type) {
364
+ case "text":
365
+ return { text: sanitizeSurrogates(c.text) };
366
+ case "image":
367
+ return { image: createImageBlock(c.mimeType, c.data) };
368
+ default:
369
+ throw new Error("Unknown user content type");
370
+ }
371
+ }),
372
+ });
373
+ break;
374
+ case "assistant": {
375
+ // Skip assistant messages with empty content (e.g., from aborted requests)
376
+ // Bedrock rejects messages with empty content arrays
377
+ if (m.content.length === 0) {
378
+ continue;
379
+ }
380
+ const contentBlocks = [];
381
+ for (const c of m.content) {
382
+ switch (c.type) {
383
+ case "text":
384
+ // Skip empty text blocks
385
+ if (c.text.trim().length === 0)
386
+ continue;
387
+ contentBlocks.push({ text: sanitizeSurrogates(c.text) });
388
+ break;
389
+ case "toolCall":
390
+ contentBlocks.push({
391
+ toolUse: { toolUseId: c.id, name: c.name, input: c.arguments },
392
+ });
393
+ break;
394
+ case "thinking":
395
+ // Skip empty thinking blocks
396
+ if (c.thinking.trim().length === 0)
397
+ continue;
398
+ // Only Anthropic models support the signature field in reasoningText.
399
+ // For other models, we omit the signature to avoid errors like:
400
+ // "This model doesn't support the reasoningContent.reasoningText.signature field"
401
+ if (supportsThinkingSignature(model)) {
402
+ contentBlocks.push({
403
+ reasoningContent: {
404
+ reasoningText: { text: sanitizeSurrogates(c.thinking), signature: c.thinkingSignature },
405
+ },
406
+ });
407
+ }
408
+ else {
409
+ contentBlocks.push({
410
+ reasoningContent: {
411
+ reasoningText: { text: sanitizeSurrogates(c.thinking) },
412
+ },
413
+ });
414
+ }
415
+ break;
416
+ default:
417
+ throw new Error("Unknown assistant content type");
418
+ }
419
+ }
420
+ // Skip if all content blocks were filtered out
421
+ if (contentBlocks.length === 0) {
422
+ continue;
423
+ }
424
+ result.push({
425
+ role: ConversationRole.ASSISTANT,
426
+ content: contentBlocks,
427
+ });
428
+ break;
429
+ }
430
+ case "toolResult": {
431
+ // Collect all consecutive toolResult messages into a single user message
432
+ // Bedrock requires all tool results to be in one message
433
+ const toolResults = [];
434
+ // Add current tool result with all content blocks combined
435
+ toolResults.push({
436
+ toolResult: {
437
+ toolUseId: m.toolCallId,
438
+ content: m.content.map((c) => c.type === "image"
439
+ ? { image: createImageBlock(c.mimeType, c.data) }
440
+ : { text: sanitizeSurrogates(c.text) }),
441
+ status: m.isError ? ToolResultStatus.ERROR : ToolResultStatus.SUCCESS,
442
+ },
443
+ });
444
+ // Look ahead for consecutive toolResult messages
445
+ let j = i + 1;
446
+ while (j < transformedMessages.length && transformedMessages[j].role === "toolResult") {
447
+ const nextMsg = transformedMessages[j];
448
+ toolResults.push({
449
+ toolResult: {
450
+ toolUseId: nextMsg.toolCallId,
451
+ content: nextMsg.content.map((c) => c.type === "image"
452
+ ? { image: createImageBlock(c.mimeType, c.data) }
453
+ : { text: sanitizeSurrogates(c.text) }),
454
+ status: nextMsg.isError ? ToolResultStatus.ERROR : ToolResultStatus.SUCCESS,
455
+ },
456
+ });
457
+ j++;
458
+ }
459
+ // Skip the messages we've already processed
460
+ i = j - 1;
461
+ result.push({
462
+ role: ConversationRole.USER,
463
+ content: toolResults,
464
+ });
465
+ break;
466
+ }
467
+ default:
468
+ throw new Error("Unknown message role");
469
+ }
470
+ }
471
+ // Add cache point to the last user message for supported Claude models when caching is enabled
472
+ if (cacheRetention !== "none" && supportsPromptCaching(model) && result.length > 0) {
473
+ const lastMessage = result[result.length - 1];
474
+ if (lastMessage.role === ConversationRole.USER && lastMessage.content) {
475
+ lastMessage.content.push({
476
+ cachePoint: {
477
+ type: CachePointType.DEFAULT,
478
+ ...(cacheRetention === "long" ? { ttl: CacheTTL.ONE_HOUR } : {}),
479
+ },
480
+ });
481
+ }
482
+ }
483
+ return result;
484
+ }
485
+ function convertToolConfig(tools, toolChoice) {
486
+ if (!tools?.length || toolChoice === "none")
487
+ return undefined;
488
+ const bedrockTools = tools.map((tool) => ({
489
+ toolSpec: {
490
+ name: tool.name,
491
+ description: tool.description,
492
+ inputSchema: { json: tool.parameters },
493
+ },
494
+ }));
495
+ let bedrockToolChoice;
496
+ switch (toolChoice) {
497
+ case "auto":
498
+ bedrockToolChoice = { auto: {} };
499
+ break;
500
+ case "any":
501
+ bedrockToolChoice = { any: {} };
502
+ break;
503
+ default:
504
+ if (toolChoice?.type === "tool") {
505
+ bedrockToolChoice = { tool: { name: toolChoice.name } };
506
+ }
507
+ }
508
+ return { tools: bedrockTools, toolChoice: bedrockToolChoice };
509
+ }
510
+ function mapStopReason(reason) {
511
+ switch (reason) {
512
+ case BedrockStopReason.END_TURN:
513
+ case BedrockStopReason.STOP_SEQUENCE:
514
+ return "stop";
515
+ case BedrockStopReason.MAX_TOKENS:
516
+ case BedrockStopReason.MODEL_CONTEXT_WINDOW_EXCEEDED:
517
+ return "length";
518
+ case BedrockStopReason.TOOL_USE:
519
+ return "toolUse";
520
+ default:
521
+ return "error";
522
+ }
523
+ }
524
+ function buildAdditionalModelRequestFields(model, options) {
525
+ if (!options.reasoning || !model.reasoning) {
526
+ return undefined;
527
+ }
528
+ if (model.id.includes("anthropic.claude") || model.id.includes("anthropic/claude")) {
529
+ const result = supportsAdaptiveThinking(model.id)
530
+ ? {
531
+ thinking: { type: "adaptive" },
532
+ output_config: { effort: mapThinkingLevelToEffort(options.reasoning, model.id) },
533
+ }
534
+ : (() => {
535
+ const defaultBudgets = {
536
+ minimal: 1024,
537
+ low: 2048,
538
+ medium: 8192,
539
+ high: 16384,
540
+ xhigh: 16384, // Claude doesn't support xhigh, clamp to high
541
+ };
542
+ // Custom budgets override defaults (xhigh not in ThinkingBudgets, use high)
543
+ const level = options.reasoning === "xhigh" ? "high" : options.reasoning;
544
+ const budget = options.thinkingBudgets?.[level] ?? defaultBudgets[options.reasoning];
545
+ return {
546
+ thinking: {
547
+ type: "enabled",
548
+ budget_tokens: budget,
549
+ },
550
+ };
551
+ })();
552
+ if (!supportsAdaptiveThinking(model.id) && (options.interleavedThinking ?? true)) {
553
+ result.anthropic_beta = ["interleaved-thinking-2025-05-14"];
554
+ }
555
+ return result;
556
+ }
557
+ return undefined;
558
+ }
559
+ function createImageBlock(mimeType, data) {
560
+ let format;
561
+ switch (mimeType) {
562
+ case "image/jpeg":
563
+ case "image/jpg":
564
+ format = ImageFormat.JPEG;
565
+ break;
566
+ case "image/png":
567
+ format = ImageFormat.PNG;
568
+ break;
569
+ case "image/gif":
570
+ format = ImageFormat.GIF;
571
+ break;
572
+ case "image/webp":
573
+ format = ImageFormat.WEBP;
574
+ break;
575
+ default:
576
+ throw new Error(`Unknown image type: ${mimeType}`);
577
+ }
578
+ const binaryString = atob(data);
579
+ const bytes = new Uint8Array(binaryString.length);
580
+ for (let i = 0; i < binaryString.length; i++) {
581
+ bytes[i] = binaryString.charCodeAt(i);
582
+ }
583
+ return { source: { bytes }, format };
584
+ }
585
+ //# sourceMappingURL=amazon-bedrock.js.map