diff --git a/README.md b/README.md index 351d9c0dc..c24fdefbf 100644 --- a/README.md +++ b/README.md @@ -124,6 +124,91 @@ await client.files.create({ }); ``` +## Streaming Helpers + +The SDK makes it easy to stream responses, by providing an emitter helper via `.stream()`: + +```ts +import OpenAI from 'openai'; + +const client = new OpenAI(); + +async function main() { + const stream = client.chat.completions + .stream({ + model: 'gpt-4o', + max_tokens: 1024, + messages: [ + { + role: 'user', + content: 'Say hi!', + }, + ], + }) + .on('chunk', (text) => { + console.log(text); + }); + + const message = await stream.finalMessage(); + console.log(message); +} + +main(); +``` + +With `.stream()` you get event handlers, accumulation, and an async iterable. + +Alternatively, you can use `client.chat.completions({ ..., stream: true })` which only returns an async iterable of the events in the stream and thus uses less memory (it does not build up a final message object for you). + +## Tool Helpers + +The SDK makes it easy to create and run [function tools with the chats API](https://platform.openai.com/docs/guides/function-calling). You can use Zod schemas or direct JSON schemas to describe the shape of tool input, and then you can run the tools using the `client.beta.messages.toolRunner` method. This method will automatically handle passing the inputs generated by the model into your tools and providing the results back to the model. + +```ts +import OpenAI from 'openai'; + +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; +import { z } from 'zod/v4'; + +const client = new OpenAI(); + +async function main() { + const addTool = betaZodFunctionTool({ + name: 'add', + parameters: z.object({ + a: z.number(), + b: z.number(), + }), + description: 'Add two numbers together', + run: (input) => { + return String(input.a + input.b); + }, + }); + + const multiplyTool = betaZodFunctionTool({ + name: 'multiply', + parameters: z.object({ + a: z.number(), + b: z.number(), + }), + description: 'Multiply two numbers together', + run: (input) => { + return String(input.a * input.b); + }, + }); + + const finalMessage = await client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [{ role: 'user', content: 'What is 5 plus 3, and then multiply that result by 4?' }], + tools: [addTool, multiplyTool], + }); + console.log(finalMessage); +} + +main(); +``` + ## Webhook Verification Verifying webhook signatures is _optional but encouraged_. diff --git a/examples/tool-calls-beta-zod.ts b/examples/tool-calls-beta-zod.ts new file mode 100755 index 000000000..8a5619f17 --- /dev/null +++ b/examples/tool-calls-beta-zod.ts @@ -0,0 +1,105 @@ +#!/usr/bin/env -S npm run tsn -T + +import OpenAI from 'openai'; +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; +import { z } from 'zod'; + +const client = new OpenAI(); + +async function main() { + const runner = client.beta.chat.completions.toolRunner({ + messages: [ + { + role: 'user', + content: `I'm planning a trip to San Francisco and I need some information. Can you help me with the weather, current time, and currency exchange rates (from EUR)? Please use parallel tool use.`, + }, + ], + tools: [ + betaZodFunctionTool({ + name: 'getWeather', + description: 'Get the weather at a specific location', + parameters: z.object({ + location: z.string().describe('The city and state, e.g. San Francisco, CA'), + }), + run: ({ location }) => { + return `The weather is sunny with a temperature of 20°C in ${location}.`; + }, + }), + betaZodFunctionTool({ + name: 'getTime', + description: 'Get the current time in a specific timezone', + parameters: z.object({ + timezone: z.string().describe('The timezone, e.g. America/Los_Angeles'), + }), + run: ({ timezone }) => { + return `The current time in ${timezone} is 3:00 PM.`; + }, + }), + betaZodFunctionTool({ + name: 'getCurrencyExchangeRate', + description: 'Get the exchange rate between two currencies', + parameters: z.object({ + from_currency: z.string().describe('The currency to convert from, e.g. USD'), + to_currency: z.string().describe('The currency to convert to, e.g. EUR'), + }), + run: ({ from_currency, to_currency }) => { + return `The exchange rate from ${from_currency} to ${to_currency} is 0.85.`; + }, + }), + ], + model: 'gpt-4o', + max_tokens: 1024, + // This limits the conversation to at most 10 back and forth between the API. + max_iterations: 10, + }); + + console.log(`\nšŸš€ Running tools...\n`); + + for await (const message of runner) { + if (!message) continue; + + console.log(`ā”Œā”€ Message ${message.id} `.padEnd(process.stdout.columns, '─')); + console.log(); + + const { choices } = message; + const firstChoice = choices.at(0)!; + + // When we get a tool call request it's null + if (firstChoice.message.content !== null) { + console.log(`${firstChoice.message.content}\n`); + } else { + // each tool call (could be many) + for (const toolCall of firstChoice.message.tool_calls ?? []) { + if (toolCall.type === 'function') { + console.log(`${toolCall.function.name}(${JSON.stringify(toolCall.function.arguments, null, 2)})\n`); + } + } + } + + console.log(`└─`.padEnd(process.stdout.columns, '─')); + console.log(); + console.log(); + + const defaultResponse = await runner.generateToolResponse(); + if (defaultResponse && Array.isArray(defaultResponse)) { + console.log(`ā”Œā”€ Response `.padEnd(process.stdout.columns, '─')); + console.log(); + + for (const toolResponse of defaultResponse) { + if (toolResponse.role === 'tool') { + const toolCall = firstChoice.message.tool_calls?.find((tc) => tc.id === toolResponse.tool_call_id); + if (toolCall && toolCall.type === 'function') { + console.log(`${toolCall.function.name}(): ${toolResponse.content}`); + } + } + } + + console.log(); + console.log(`└─`.padEnd(process.stdout.columns, '─')); + console.log(); + console.log(); + } + } +} + +main(); diff --git a/examples/tool-helpers-advanced-streaming.ts b/examples/tool-helpers-advanced-streaming.ts new file mode 100755 index 000000000..458ef38dc --- /dev/null +++ b/examples/tool-helpers-advanced-streaming.ts @@ -0,0 +1,109 @@ +#!/usr/bin/env -S npm run tsn -T + +import OpenAI from 'openai'; +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; +import { z } from 'zod'; + +const client = new OpenAI(); + +async function main() { + const runner = client.beta.chat.completions.toolRunner({ + messages: [ + { + role: 'user', + content: `I'm planning a trip to San Francisco and I need some information. Can you help me with the weather, current time, and currency exchange rates (from EUR)? Please use parallel tool use`, + }, + ], + tools: [ + betaZodFunctionTool({ + name: 'getWeather', + description: 'Get the weather at a specific location', + parameters: z.object({ + location: z.string().describe('The city and state, e.g. San Francisco, CA'), + }), + run: ({ location }) => { + return `The weather is sunny with a temperature of 20°C in ${location}.`; + }, + }), + betaZodFunctionTool({ + name: 'getTime', + description: 'Get the current time in a specific timezone', + parameters: z.object({ + timezone: z.string().describe('The timezone, e.g. America/Los_Angeles'), + }), + run: ({ timezone }) => { + return `The current time in ${timezone} is 3:00 PM.`; + }, + }), + betaZodFunctionTool({ + name: 'getCurrencyExchangeRate', + description: 'Get the exchange rate between two currencies', + parameters: z.object({ + from_currency: z.string().describe('The currency to convert from, e.g. USD'), + to_currency: z.string().describe('The currency to convert to, e.g. EUR'), + }), + run: ({ from_currency, to_currency }) => { + return `The exchange rate from ${from_currency} to ${to_currency} is 0.85.`; + }, + }), + ], + model: 'gpt-4o', + max_tokens: 1024, + // This limits the conversation to at most 10 back and forth between the API. + max_iterations: 10, + stream: true, + }); + + console.log(`\nšŸš€ Running tools...\n`); + + let prevMessageStarted = ''; + let prevToolStarted = ''; + let prevWasToolCall = false; + + for await (const messageStream of runner) { + for await (const event of messageStream) { + const hadToolCalls = !!event.choices?.[0]?.delta?.tool_calls; + + if (hadToolCalls) { + if (!prevMessageStarted) { + console.log(`ā”Œā”€ Message ${event.id} `.padEnd(process.stdout.columns, '─')); + prevMessageStarted = event.id; + } + + prevWasToolCall = true; + const toolCalls = event.choices[0]!.delta.tool_calls!; + + for (const toolCall of toolCalls) { + if (toolCall.function?.name && prevToolStarted !== toolCall.function.name) { + process.stdout.write(`\n${toolCall.function.name}: `); + prevToolStarted = toolCall.function.name; + } else if (toolCall.function?.arguments) { + process.stdout.write(toolCall.function.arguments); + } + } + } else if (event.choices?.[0]?.delta?.content) { + if (prevWasToolCall) { + console.log(); + console.log(); + console.log(`└─`.padEnd(process.stdout.columns, '─')); + console.log(); + prevWasToolCall = false; + } + + if (prevMessageStarted !== event.id) { + console.log(`ā”Œā”€ Message ${event.id} `.padEnd(process.stdout.columns, '─')); + console.log(); + prevMessageStarted = event.id; + } + + process.stdout.write(event.choices[0].delta.content); + } + } + } + + console.log(); + console.log(); + console.log(`└─`.padEnd(process.stdout.columns, '─')); +} + +main(); diff --git a/examples/tool-helpers-advanced.ts b/examples/tool-helpers-advanced.ts new file mode 100755 index 000000000..43d67c9b8 --- /dev/null +++ b/examples/tool-helpers-advanced.ts @@ -0,0 +1,38 @@ +#!/usr/bin/env -S npm run tsn -T + +import OpenAI from 'openai'; +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; +import { z } from 'zod'; + +const client = new OpenAI(); + +async function main() { + const message = await client.beta.chat.completions.toolRunner({ + messages: [ + { + role: 'user', + content: `What is the weather in SF?`, + }, + ], + tools: [ + betaZodFunctionTool({ + name: 'getWeather', + description: 'Get the weather at a specific location', + parameters: z.object({ + location: z.string().describe('The city and state, e.g. San Francisco, CA'), + }), + run: ({ location }) => { + return `The weather is foggy with a temperature of 20°C in ${location}.`; + }, + }), + ], + model: 'gpt-4o', + max_tokens: 1024, + // the maximum number of iterations to run the tool + max_iterations: 10, + }); + + console.log('Final response:', message.content); +} + +main(); diff --git a/examples/tool-helpers-json-schema.ts b/examples/tool-helpers-json-schema.ts new file mode 100755 index 000000000..aef9bd52f --- /dev/null +++ b/examples/tool-helpers-json-schema.ts @@ -0,0 +1,44 @@ +#!/usr/bin/env -S npm run tsn -T + +import OpenAI from 'openai'; +import { betaFunctionTool } from 'openai/helpers/beta/json-schema'; + +const client = new OpenAI(); + +async function main() { + const message = await client.beta.chat.completions.toolRunner({ + messages: [ + { + role: 'user', + content: `What is the weather in SF?`, + }, + ], + tools: [ + betaFunctionTool({ + name: 'getWeather', + description: 'Get the weather at a specific location', + parameters: { + type: 'object', + properties: { + location: { + type: 'string', + description: 'The city and state, e.g. San Francisco, CA', + }, + }, + required: ['location'], + }, + run: ({ location }) => { + return `The weather is foggy with a temperature of 20°C in ${location}.`; + }, + }), + ], + model: 'gpt-4o', + max_tokens: 1024, + // the maximum number of iterations to run the tool + max_iterations: 10, + }); + + console.log('Final response:', message.content); +} + +main(); diff --git a/helpers.md b/helpers.md index 268a0abe9..fd9d7af34 100644 --- a/helpers.md +++ b/helpers.md @@ -302,9 +302,133 @@ If you need to cancel a stream, you can `break` from a `for await` loop or call See an example of streaming helpers in action in [`examples/stream.ts`](examples/stream.ts). -### Automated function calls +### Automated function calls via Beta Tool Runner -We provide the `openai.chat.completions.runTools({…})` +The SDK provides a convenient tool runner helper at `openai.beta.chat.completions.toolRunner({…})` that simplifies [function tool calling](https://platform.openai.com/docs/guides/function-calling). This helper allows you to define functions with their associated schemas, and the SDK will automatically invoke them as the AI requests and validate the parameters for you. + +#### Usage + +```ts +import OpenAI from 'openai'; + +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; +import { z } from 'zod/v4'; + +const client = new OpenAI(); + +async function main() { + const addTool = betaZodFunctionTool({ + name: 'add', + parameters: z.object({ + a: z.number(), + b: z.number(), + }), + description: 'Add two numbers together', + run: (input) => { + return String(input.a + input.b); + }, + }); + + const multiplyTool = betaZodFunctionTool({ + name: 'multiply', + parameters: z.object({ + a: z.number(), + b: z.number(), + }), + description: 'Multiply two numbers together', + run: (input) => { + return String(input.a * input.b); + }, + }); + + const finalMessage = await client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [{ role: 'user', content: 'What is 5 plus 3, and then multiply that result by 4?' }], + tools: [addTool, multiplyTool], + }); + console.log(finalMessage); +} + +main(); +``` + +#### Advanced Usage + +You can also use the `toolRunner` as an async generator to act as the logic runs in. + +```ts +// or, instead of using "await client.beta.messages.toolRunner", you can use: +const toolRunner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [{ role: 'user', content: 'What is 5 plus 3, and then multiply that result by 4?' }], + tools: [addTool, multiplyTool], +}); + +for await (const event of toolRunner) { + console.log(event.choices[0]!.message.content); + + // If the most recent message triggered a tool call, you can get the result of + // that tool call + const toolResponse = await toolRunner.generateToolResponse(); + console.log(toolResponse); +} +``` + +The tool runner will invoke the AI with your tool offering and initial message history, and then the AI may respond by invoking your tools. Eventually the AI will respond without a tool call message, which is the "final" message in the chain. As the AI repeatedly invokes tools, you can view the responses via the async iterator. + +When you just "await" the `toolRunner`, it simply automatically iterates until the end of the async generator. + +#### Streaming + +```ts +const runner = anthropic.beta.messages.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [{ role: 'user', content: 'What is the weather in San Francisco?' }], + tools: [calculatorTool], + stream: true, +}); + +// When streaming, the runner returns ChatCompletionStream +for await (const messageStream of runner) { + for await (const event of messageStream) { + console.log('event:', event); + } + console.log('message:', await messageStream.finalMessage()); +} + +console.log(await runner); +``` + +See [./examples/tool-helpers-advanced-streaming.ts] for a more in-depth example. + +#### Beta Zod Tool + +Zod schemas can be used to define the input schema for your tools: + +```ts +import { betaZodFunctionTool } from 'openai/helpers/beta/zod'; + +const weatherTool = betaZodFunctionTool({ + name: 'get_weather', + inputSchema: z.object({ + location: z.string().describe('The city and state, e.g. San Francisco, CA'), + unit: z.enum(['celsius', 'fahrenheit']).default('fahrenheit'), + }), + description: 'Get the current weather in a given location', + run: async (input) => { + return `The weather in ${input.location} is ${input.unit === 'celsius' ? '22°C' : '72°F'}`; + }, +}); +``` + +The AI's generated inputs will be directly validated and fed into your function automatically. + +### Legacy Automated function calls + +We also provide the `openai.chat.completions.runTools({…})` convenience helper for using function tool calls with the `/chat/completions` endpoint which automatically call the JavaScript functions you provide and sends their results back to the `/chat/completions` endpoint, diff --git a/package.json b/package.json index f53c15aae..40333357e 100644 --- a/package.json +++ b/package.json @@ -26,18 +26,20 @@ "lint": "./scripts/lint", "fix": "./scripts/format" }, - "dependencies": {}, + "dependencies": { + "json-schema-to-ts": "^3.1.1" + }, "devDependencies": { "@arethetypeswrong/cli": "^0.17.0", "@swc/core": "^1.3.102", "@swc/jest": "^0.2.29", "@types/jest": "^29.4.0", - "@types/ws": "^8.5.13", "@types/node": "^20.17.6", + "@types/ws": "^8.5.13", "@typescript-eslint/eslint-plugin": "8.31.1", "@typescript-eslint/parser": "8.31.1", - "deep-object-diff": "^1.1.9", "@typescript-eslint/typescript-estree": "8.31.1", + "deep-object-diff": "^1.1.9", "eslint": "^9.39.1", "eslint-plugin-prettier": "^5.4.1", "eslint-plugin-unused-imports": "^4.1.4", @@ -45,6 +47,7 @@ "fast-check": "^3.22.0", "iconv-lite": "^0.6.3", "jest": "^29.4.0", + "nock": "^14.0.10", "prettier": "^3.0.0", "publint": "^0.2.12", "ts-jest": "^29.1.0", @@ -53,9 +56,9 @@ "tsconfig-paths": "^4.0.0", "tslib": "^2.8.1", "typescript": "5.8.3", + "typescript-eslint": "8.31.1", "ws": "^8.18.0", - "zod": "^3.25 || ^4.0", - "typescript-eslint": "8.31.1" + "zod": "^3.25 || ^4.0" }, "bin": { "openai": "bin/cli" diff --git a/src/helpers/beta/json-schema.ts b/src/helpers/beta/json-schema.ts new file mode 100644 index 000000000..35af1d313 --- /dev/null +++ b/src/helpers/beta/json-schema.ts @@ -0,0 +1,30 @@ +import type { FromSchema, JSONSchema } from 'json-schema-to-ts'; +import type { BetaRunnableChatCompletionFunctionTool, Promisable } from '../../lib/beta/BetaRunnableTool'; +import type { FunctionTool } from '../../resources/beta'; + +type NoInfer = T extends infer R ? R : never; + +/** + * Creates a Tool with a provided JSON schema that can be passed + * to the `.toolRunner()` method. The schema is used to automatically validate + * the input arguments for the tool. + */ +export function betaFunctionTool< + const Schema extends Exclude & { type: 'object' }, +>(options: { + name: string; + parameters: Schema; + description: string; + run: (args: NoInfer>) => Promisable>; +}): BetaRunnableChatCompletionFunctionTool>> { + return { + type: 'function', + function: { + name: options.name, + parameters: options.parameters, + description: options.description, + }, + run: options.run, + parse: (content: unknown) => content as FromSchema, + } as any; // For some reason this causes infinite inference so we cast to any to not crash lsp +} diff --git a/src/helpers/beta/zod.ts b/src/helpers/beta/zod.ts new file mode 100644 index 000000000..d776f0c80 --- /dev/null +++ b/src/helpers/beta/zod.ts @@ -0,0 +1,30 @@ +import type { infer as zodInfer, ZodType } from 'zod/v4'; +import * as z from 'zod/v4'; +import type { BetaRunnableChatCompletionFunctionTool, Promisable } from '../../lib/beta/BetaRunnableTool'; +import type { ChatCompletionContentPart } from '../../resources'; + +/** + * Creates a tool using the provided Zod schema that can be passed + * into the `.toolRunner()` method. The Zod schema will automatically be + * converted into JSON Schema when passed to the API. The provided function's + * input arguments will also be validated against the provided schema. + */ +export function betaZodFunctionTool(options: { + name: string; + parameters: InputSchema; + description: string; + run: (args: zodInfer) => Promisable; +}): BetaRunnableChatCompletionFunctionTool> { + const jsonSchema = z.toJSONSchema(options.parameters, { reused: 'ref' }); + + return { + type: 'function', + function: { + name: options.name, + description: options.description, + parameters: jsonSchema, + }, + run: options.run, + parse: (args: unknown) => options.parameters.parse(args), + }; +} diff --git a/src/internal/utils/values.ts b/src/internal/utils/values.ts index 284ff5cde..d3bce3e1b 100644 --- a/src/internal/utils/values.ts +++ b/src/internal/utils/values.ts @@ -103,3 +103,10 @@ export const safeJSON = (text: string) => { return undefined; } }; + +// Gets a value from an object, deletes the key, and returns the value (or undefined if not found) +export const pop = , K extends string>(obj: T, key: K): T[K] => { + const value = obj[key]; + delete obj[key]; + return value; +}; diff --git a/src/lib/beta/BetaRunnableTool.ts b/src/lib/beta/BetaRunnableTool.ts new file mode 100644 index 000000000..21888a006 --- /dev/null +++ b/src/lib/beta/BetaRunnableTool.ts @@ -0,0 +1,10 @@ +import type { ChatCompletionContentPart, ChatCompletionFunctionTool } from '../../resources'; + +export type Promisable = T | Promise; + +// this type is just an extension of ChatCompletionFunctionTool with a run and parse method +// that will be called by `toolRunner()` helpers +export type BetaRunnableChatCompletionFunctionTool = ChatCompletionFunctionTool & { + run: (args: Input) => Promisable; + parse: (content: unknown) => Input; +}; diff --git a/src/lib/beta/BetaToolRunner.ts b/src/lib/beta/BetaToolRunner.ts new file mode 100644 index 000000000..4e6a0f379 --- /dev/null +++ b/src/lib/beta/BetaToolRunner.ts @@ -0,0 +1,433 @@ +import { OpenAI } from '../..'; +import { OpenAIError } from '../../core/error'; +import { buildHeaders } from '../../internal/headers'; +import type { RequestOptions } from '../../internal/request-options'; +import type { + ChatCompletion, + ChatCompletionCreateParams, + ChatCompletionMessage, + ChatCompletionMessageParam, + ChatCompletionStream, + ChatCompletionTool, + ChatCompletionToolMessageParam, +} from '../../resources/chat/completions'; +import type { BetaRunnableChatCompletionFunctionTool } from './BetaRunnableTool'; + +/** + * Just Promise.withResolvers(), which is not available in all environments. + */ +function promiseWithResolvers(): { + promise: Promise; + resolve: (value: T) => void; + reject: (reason?: any) => void; +} { + let resolve: (value: T) => void; + let reject: (reason?: any) => void; + const promise = new Promise((res, rej) => { + resolve = res; + reject = rej; + }); + return { promise, resolve: resolve!, reject: reject! }; +} + +/** + * A `BetaToolRunner` handles the automatic conversation loop between the assistant and tools. + * + * A `BetaToolRunner` is an async iterable that yields either `ChatCompletion` or + * `ChatCompletionStream` objects depending on the streaming configuration. + */ +export class BetaToolRunner + implements AsyncIterable +{ + /** Whether the async iterator has been consumed */ + #consumed = false; + /** Whether parameters have been mutated since the last API call */ + #mutated = false; + /** Current state containing the request parameters */ + #state: { params: BetaToolRunnerParams }; + #options: BetaToolRunnerRequestOptions; + /** + * Promise for the last message received from the assistant. + * + * This resolves to undefined in non-streaming mode if there are no choices provided. + */ + #message?: Promise | undefined; + /** + * Resolves to the last (entire) chat completion received from the assistant. + * We want to return an attribute of ourself so that the promise keeps running + * after the yield, and we can access it later. + */ + #chatCompletion?: Promise; + /** Cached tool response to avoid redundant executions */ + #toolResponse?: Promise | undefined; + /** Promise resolvers for waiting on completion */ + #completion: { + promise: Promise; + resolve: (value: ChatCompletionMessage) => void; + reject: (reason?: any) => void; + }; + /** Number of iterations (API requests) made so far */ + #iterationCount = 0; + + constructor( + private client: OpenAI, + params: BetaToolRunnerParams, + options?: BetaToolRunnerRequestOptions, + ) { + if (params.n && params.n > 1) { + throw new Error('BetaToolRunner does not support n > 1'); + } + + this.#state = { + params: { + // You can't clone the entire params since there are functions as handlers. + // You also don't really need to clone params.messages, but it probably will prevent a foot gun + // somewhere. + ...params, + messages: structuredClone(params.messages), + }, + }; + + this.#options = { + ...options, + headers: buildHeaders([{ 'x-stainless-helper': 'BetaToolRunner' }, options?.headers]), + }; + this.#completion = promiseWithResolvers(); + } + + async *[Symbol.asyncIterator](): AsyncIterator< + Stream extends true ? ChatCompletionStream : ChatCompletion + > { + if (this.#consumed) { + throw new OpenAIError('Cannot iterate over a consumed stream'); + } + + this.#consumed = true; + this.#mutated = true; + this.#toolResponse = undefined; + + try { + while (true) { + let stream: ChatCompletionStream | undefined; + try { + if ( + this.#state.params.max_iterations && + this.#iterationCount >= this.#state.params.max_iterations + ) { + break; + } + + this.#mutated = false; + this.#message = undefined; + this.#toolResponse = undefined; + this.#iterationCount++; + + const { max_iterations: _, ...params } = this.#state.params; + + if (params.stream) { + stream = this.client.chat.completions.stream({ ...params, stream: true }, this.#options); + this.#message = stream.finalMessage(); + + // Make sure that this promise doesn't throw before we get the option to do something about it. + // Error will be caught when we call await this.#message ultimately + this.#message?.catch(() => {}); + yield stream as any; + } else { + this.#chatCompletion = this.client.chat.completions.create( + { + ...params, // spread and explicit so we get better types + stream: false, + }, + this.#options, + ); + + this.#message = this.#chatCompletion.then((resp) => resp.choices.at(0)!.message); + this.#message.catch(() => {}); + + yield this.#chatCompletion as any; + } + + const prevMessage = await this.#message; + + if (!this.#mutated) { + this.#state.params.messages.push(prevMessage); + } + + const toolMessages = await this.#generateToolResponse(prevMessage); + if (toolMessages) { + for (const toolMessage of toolMessages) { + this.#state.params.messages.push(toolMessage); + } + } + + if (!toolMessages && !this.#mutated) { + break; + } + } finally { + if (stream) { + stream.abort(); + } + } + } + + if (!this.#message) { + throw new OpenAIError('ToolRunner concluded without a message from the server'); + } + + this.#completion.resolve(await this.#message); + } catch (error) { + this.#consumed = false; + // Silence unhandled promise errors + this.#completion.promise.catch(() => {}); + this.#completion.reject(error); + this.#completion = promiseWithResolvers(); + throw error; + } + } + + /** + * Update the parameters for the next API call. This invalidates any cached tool responses. + * + * @param paramsOrMutator - Either new parameters or a function to mutate existing parameters + * + * @example + * // Direct parameter update + * runner.setChatParams({ + * model: 'gpt-4o', + * max_tokens: 500, + * }); + * + * @example + * // Using a mutator function + * runner.setChatParams((params) => ({ + * ...params, + * max_tokens: 100, + * })); + * + * @example + * // Appending a user message + * runner.setChatParams((params) => ({ + * ...params, + * messages: [...params.messages, { role: 'user', content: 'What colour is the sky?' }], + * })); + */ + setChatParams(params: BetaToolRunnerParams): void; + setChatParams(mutator: (prevParams: BetaToolRunnerParams) => BetaToolRunnerParams): void; + setChatParams( + paramsOrMutator: BetaToolRunnerParams | ((prevParams: BetaToolRunnerParams) => BetaToolRunnerParams), + ) { + if (typeof paramsOrMutator === 'function') { + this.#state.params = paramsOrMutator(this.#state.params); + } else { + this.#state.params = paramsOrMutator; + } + this.#mutated = true; + // Invalidate cached tool response since parameters changed + this.#toolResponse = undefined; + } + + /** + * Get the tool response for the last message from the assistant. + * Avoids redundant tool executions by caching results. + * + * @returns A promise that resolves to a `ChatCompletionToolMessageParam` containing tool results, or null if no tools need to be executed + * + * @example + * const toolResponse = await runner.generateToolResponse(); + * if (toolResponse) { + * console.log('Tool results:', toolResponse.content); + * } + */ + async generateToolResponse(): Promise { + // The most recent message from the assistant. + const message = await this.#message; + if (!message) { + return null; + } + return this.#generateToolResponse(message); + } + + async #generateToolResponse(lastMessage: ChatCompletionMessage) { + if (this.#toolResponse) { + return this.#toolResponse; + } + const toolsResponse = generateToolResponse( + lastMessage, + this.#state.params.tools.filter( + (tool): tool is BetaRunnableChatCompletionFunctionTool => + 'run' in tool && tool.type === 'function', + ), + ); + this.#toolResponse = toolsResponse; + return toolsResponse; + } + + /** + * Wait for the async iterator to complete. This works even if the async iterator hasn't yet started, and + * will wait for an instance to start and go to completion. + * + * @returns A promise that resolves to the final `ChatComletionMessage` when the iterator completes + * + * @example + * // Start consuming the iterator + * for await (const message of runner) { + * console.log('Message:', message.content); + * } + * + * // Meanwhile, wait for completion from another part of the code + * const finalMessage = await runner.done(); + * console.log('Final response:', finalMessage.content); + */ + done(): Promise { + return this.#completion.promise; + } + + /** + * Returns a promise indicating that the stream is done. Unlike .done(), this + * will eagerly read the stream: + * + * - If the iterator has not been consumed, consume the entire iterator and + * return the final message from the assistant. + * - If the iterator has been consumed, waits for it to complete and returns the final message. + * + * @returns A promise that resolves to the final `ChatCompletionMessage` from the conversation + * @throws {OpenAIError} If no messages were processed during the conversation + * + * @example + * const finalMessage = await runner.runUntilDone(); + * console.log('Final response:', finalMessage.content); + */ + async runUntilDone(): Promise { + // If not yet consumed, start consuming and wait for completion + if (!this.#consumed) { + for await (const _ of this) { + // Iterator naturally populates this.#message + } + } + + // If consumed but not completed, wait for completion + return this.done(); + } + + /** + * Get the current parameters being used by the `BetaToolRunner`. + * + * @returns A readonly view of the current `BetaToolRunnerParams` + * + * @example + * const currentParams = runner.params; + * console.log('Current model:', currentParams.model); + * console.log('Message count:', currentParams.messages.length); + */ + get params(): Readonly { + return this.#state.params as Readonly; + } + + /** + * Add one or more messages to the conversation history. + * + * @param messages - One or more `ChatCompletionMessageParam` objects to add to the conversation + * + * @example + * runner.pushMessages( + * { role: 'user', content: 'Also, what about the weather in NYC?' } + * ); + * + * @example + * // Adding multiple messages + * runner.pushMessages( + * { role: 'user', content: 'What about NYC?' }, + * { role: 'user', content: 'And Boston?' } + * ); + */ + pushMessages(...messages: ChatCompletionMessageParam[]) { + this.setChatParams((params) => ({ + ...params, + messages: [...params.messages, ...messages], + })); + } + + /** + * Makes the ToolRunner directly awaitable, equivalent to calling .runUntilDone() + * This allows using `await runner` instead of `await runner.runUntilDone()` + */ + then( + onfulfilled?: ((value: ChatCompletionMessage) => TResult1 | PromiseLike) | undefined | null, + onrejected?: ((reason: any) => TResult2 | PromiseLike) | undefined | null, + ): Promise { + return this.runUntilDone().then(onfulfilled, onrejected); + } +} + +async function generateToolResponse( + lastMessage: ChatCompletionMessage, + tools: BetaRunnableChatCompletionFunctionTool[], +): Promise { + // Only process if the last message is from the assistant and has tool use blocks + if (!lastMessage || lastMessage.role !== 'assistant' || typeof lastMessage.content === 'string') { + return null; + } + + const { tool_calls: prevToolCalls = [] } = lastMessage; + + if ((lastMessage.tool_calls ?? []).length === 0) { + return null; + } + + return ( + await Promise.all( + prevToolCalls.map( + async (toolUse): Promise => { + if (toolUse.type !== 'function') return null; // TODO: eventually we should support additional tool call types + + const tool = tools.find( + (t) => t.type === 'function' && toolUse.function.name === t.function.name, + ) as BetaRunnableChatCompletionFunctionTool; + + if (!tool || !('run' in tool)) { + return { + role: 'tool', + tool_call_id: toolUse.id, + content: `Error: Tool '${toolUse.function.name}' not found`, + }; + } + + try { + const result = await tool.run(tool.parse(JSON.parse(toolUse.function.arguments))); + return { + role: 'tool', + tool_call_id: toolUse.id, + content: typeof result === 'string' ? result : JSON.stringify(result), + }; + } catch (error) { + return { + role: 'tool', + tool_call_id: toolUse.id, + content: `Error: ${error instanceof Error ? error.message : String(error)}`, + }; + } + }, + ), + ) + ).filter((result): result is NonNullable => result != null); +} + +// vendored from typefest just to make things look a bit nicer on hover +type Simplify = { [KeyType in keyof T]: T[KeyType] } & {}; + +/** + * Parameters for creating a ToolRunner, extending MessageCreateParams with runnable tools. + */ +export type BetaToolRunnerParams = Simplify< + Omit & { + tools: (ChatCompletionTool | BetaRunnableChatCompletionFunctionTool)[]; + /** + * Maximum number of iterations (API requests) to make in the tool execution loop. + * Each iteration consists of: assistant response → tool execution → tool results. + * When exceeded, the loop will terminate even if tools are still being requested. + */ + max_iterations?: number; + } +>; + +export type BetaToolRunnerRequestOptions = Pick; diff --git a/src/resources/beta/beta.ts b/src/resources/beta/beta.ts index 7bbca11a6..c7dbb7df5 100644 --- a/src/resources/beta/beta.ts +++ b/src/resources/beta/beta.ts @@ -74,6 +74,7 @@ import { TranscriptionSessionUpdatedEvent, } from './realtime/realtime'; import * as ChatKitAPI from './chatkit/chatkit'; +import { BetaChat as BetaChatAPI } from './chat'; import { ChatKit, ChatKitWorkflow } from './chatkit/chatkit'; import * as ThreadsAPI from './threads/threads'; import { @@ -92,20 +93,25 @@ import { ThreadUpdateParams, Threads, } from './threads/threads'; +import { Chat } from './../chat'; export class Beta extends APIResource { realtime: RealtimeAPI.Realtime = new RealtimeAPI.Realtime(this._client); chatkit: ChatKitAPI.ChatKit = new ChatKitAPI.ChatKit(this._client); assistants: AssistantsAPI.Assistants = new AssistantsAPI.Assistants(this._client); threads: ThreadsAPI.Threads = new ThreadsAPI.Threads(this._client); + chat: BetaChatAPI = new BetaChatAPI(this._client); } Beta.Realtime = Realtime; Beta.ChatKit = ChatKit; Beta.Assistants = Assistants; Beta.Threads = Threads; +Beta.Chat = Chat; export declare namespace Beta { + export { Chat }; + export { Realtime as Realtime, type ConversationCreatedEvent as ConversationCreatedEvent, diff --git a/src/resources/beta/chat.ts b/src/resources/beta/chat.ts new file mode 100644 index 000000000..af9ebee4b --- /dev/null +++ b/src/resources/beta/chat.ts @@ -0,0 +1,57 @@ +import { APIResource } from '../../resource'; +import { + BetaToolRunner, + type BetaToolRunnerParams, + type BetaToolRunnerRequestOptions, +} from '../../lib/beta/BetaToolRunner'; + +export class BetaCompletions extends APIResource { + /** + * Creates a tool runner that automates the back-and-forth conversation between the model and tools. + * + * The tool runner handles the following workflow: + * 1. Sends a request to the model with some initial messages and available tools + * 2. If the model calls tools, executes them automatically + * 3. Sends tool results back to the model + * 4. Repeats until the model provides a final response or max_iterations is reached + * + * @see [helpers.md](https://github.com/openai/openai-node/blob/master/helpers.md) + * + * @example + * ```typescript + * const finalMessage = await client.beta.chat.completions.toolRunner({ + * messages: [{ role: 'user', content: 'What's the weather in San Francisco?' }], + * tools: [ + * betaZodFunctionTool({ + * name: 'get_weather', + * inputSchema: z.object({ + * location: z.string().describe('The city and state, e.g. San Francisco, CA'), + * unit: z.enum(['celsius', 'fahrenheit']).default('fahrenheit'), + * }), + * description: 'Get the current weather in a given location', + * run: async (input) => { + * return `The weather in ${input.location} is ${input.unit === 'celsius' ? '22°C' : '72°F'}`; + * }, + * }) + * ], + * model: 'gpt-4o' + * }); + * console.log(finalMessage.content); + * ``` + */ + toolRunner( + body: BetaToolRunnerParams & { stream?: false }, + options?: BetaToolRunnerRequestOptions, + ): BetaToolRunner; + toolRunner( + body: BetaToolRunnerParams & { stream: true }, + options?: BetaToolRunnerRequestOptions, + ): BetaToolRunner; + toolRunner(body: BetaToolRunnerParams, options?: BetaToolRunnerRequestOptions): BetaToolRunner { + return new BetaToolRunner(this._client, body, options); + } +} + +export class BetaChat extends APIResource { + completions: BetaCompletions = new BetaCompletions(this._client); +} diff --git a/src/resources/chat/completions/completions.ts b/src/resources/chat/completions/completions.ts index 8665d43bb..76670bccf 100644 --- a/src/resources/chat/completions/completions.ts +++ b/src/resources/chat/completions/completions.ts @@ -19,6 +19,7 @@ import { ChatCompletionToolRunnerParams } from '../../../lib/ChatCompletionRunne import { ChatCompletionStreamingToolRunnerParams } from '../../../lib/ChatCompletionStreamingRunner'; import { ChatCompletionStream, type ChatCompletionStreamParams } from '../../../lib/ChatCompletionStream'; import { ExtractParsedContentFromParams, parseChatCompletion, validateInputTools } from '../../../lib/parser'; +import { BetaToolRunner } from '../../../lib/beta/BetaToolRunner'; export class Completions extends APIResource { messages: MessagesAPI.Messages = new MessagesAPI.Messages(this._client); @@ -1984,6 +1985,7 @@ export interface ChatCompletionListParams extends CursorPageParams { } Completions.Messages = Messages; +Completions.BetaToolRunner = BetaToolRunner; export declare namespace Completions { export { @@ -2033,4 +2035,6 @@ export declare namespace Completions { }; export { Messages as Messages, type MessageListParams as MessageListParams }; + + export { BetaToolRunner }; } diff --git a/tests/lib/tools/BetaToolRunner.test.ts b/tests/lib/tools/BetaToolRunner.test.ts new file mode 100644 index 000000000..da49cad6c --- /dev/null +++ b/tests/lib/tools/BetaToolRunner.test.ts @@ -0,0 +1,1266 @@ +import OpenAI from 'openai'; +import { mockFetch } from '../../utils/mock-fetch'; +import type { BetaRunnableChatCompletionFunctionTool } from 'openai/lib/beta/BetaRunnableTool'; +import type { + ChatCompletion, + ChatCompletionChunk, + ChatCompletionMessage, + ChatCompletionMessageFunctionToolCall, + ChatCompletionMessageToolCall, + ChatCompletionToolMessageParam, +} from 'openai/resources'; +import type { Fetch } from 'openai/internal/builtin-types'; +import type { BetaToolRunnerParams } from 'openai/lib/beta/BetaToolRunner'; + +const weatherTool: BetaRunnableChatCompletionFunctionTool<{ location: string }> = { + type: 'function', + function: { + name: 'getWeather', + description: 'Get weather', + parameters: { + type: 'object', + properties: { + location: { type: 'string' }, + }, + }, + }, + run: async ({ location }) => `Sunny in ${location}`, + parse: (input: unknown) => input as { location: string }, +}; + +const calculatorTool: BetaRunnableChatCompletionFunctionTool<{ a: number; b: number; operation: string }> = { + type: 'function', + function: { + name: 'calculate', + description: 'Perform calculations', + parameters: { + type: 'object', + properties: { + a: { type: 'number' }, + b: { type: 'number' }, + operation: { type: 'string', enum: ['add', 'multiply'] }, + }, + }, + }, + run: async ({ a, b, operation }) => { + if (operation === 'add') return String(a + b); + if (operation === 'multiply') return String(a * b); + throw new Error(`Unknown operation: ${operation}`); + }, + parse: (input: unknown) => input as { a: number; b: number; operation: string }, +}; + +// Helper functions to create content blocks +function getWeatherToolUse(location: string, id: string = 'tool_1'): ChatCompletionMessageToolCall { + return { + id: id, + type: 'function', + function: { + name: 'getWeather', + arguments: JSON.stringify({ location }), + }, + }; +} + +function getWeatherToolResult(location: string, id: string = 'tool_1'): ChatCompletionToolMessageParam { + return { role: 'tool', tool_call_id: id, content: `Sunny in ${location}` }; +} + +function getCalculatorToolUse( + a: number, + b: number, + operation: string, + id: string = 'tool_2', +): ChatCompletionMessageToolCall { + return { + id: id, + type: 'function', + function: { + name: 'calculate', + arguments: JSON.stringify({ a, b, operation }), + }, + }; +} + +function getCalculatorToolResult( + a: number, + b: number, + operation: string, + id: string = 'tool_2', +): ChatCompletionToolMessageParam { + let result: string; + if (operation === 'add') { + result = String(a + b); + } else if (operation === 'multiply') { + result = String(a * b); + } else { + result = `Error: Unknown operation: ${operation}`; + } + return { + role: 'tool', + tool_call_id: id, + content: result, + }; +} + +function getTextContent(text?: string): ChatCompletionMessage { + return { + role: 'assistant', + content: text || 'Some text content', + refusal: null, + }; +} + +function chunkChatCompletion(message: ChatCompletion): ChatCompletionChunk[] { + const events: ChatCompletionChunk[] = []; + + const messageContent = message.choices[0]!.message; + + // Check if it's a text message + if (messageContent.content) { + // Initial chunk with role only (no content in first chunk for text messages) + events.push({ + choices: message.choices.map( + (choice): ChatCompletionChunk.Choice => ({ + delta: { + content: null, + refusal: null, + role: choice.message.role, + }, + finish_reason: null, + index: choice.index, + }), + ), + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + }); + + // Text deltas - always chunked + // Simulate chunked streaming by splitting text + const words = messageContent.content.split(' '); + const chunks = []; + for (let i = 0; i < words.length; i += 2) { + chunks.push(words.slice(i, i + 2).join(' ') + (i + 2 < words.length ? ' ' : '')); + } + + // Create a chunk for each piece of text + chunks.forEach((chunk) => { + if (chunk) { + events.push({ + choices: [ + { + delta: { + content: chunk, + refusal: null, + }, + index: 0, + finish_reason: null, + }, + ], + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + }); + } + }); + } else if (messageContent.tool_calls && messageContent.tool_calls.length > 0) { + // Initial chunk with role only for tool calls + events.push({ + choices: message.choices.map( + (choice): ChatCompletionChunk.Choice => ({ + delta: { + content: null, + refusal: null, + role: choice.message.role, + }, + finish_reason: null, + index: choice.index, + }), + ), + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + }); + + // Handle tool calls + messageContent.tool_calls.forEach((toolCall, toolIndex) => { + // Initial tool call chunk + if (toolCall.type === 'function') { + const functionToolCall = toolCall as ChatCompletionMessageFunctionToolCall; + + // Create a chunk for the function name + events.push({ + choices: [ + { + delta: { + tool_calls: [ + { + index: toolIndex, + id: toolCall.id, + type: 'function', + function: { + name: functionToolCall.function.name, + }, + }, + ], + content: null, + refusal: null, + }, + index: 0, + finish_reason: null, + }, + ], + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + }); + + // Input JSON deltas - always chunked for arguments + const jsonStr = functionToolCall.function.arguments; + // Simulate chunked JSON streaming + const chunkSize = Math.ceil(jsonStr.length / 3); + for (let i = 0; i < jsonStr.length; i += chunkSize) { + const argChunk = jsonStr.slice(i, i + chunkSize); + events.push({ + choices: [ + { + delta: { + tool_calls: [ + { + index: toolIndex, + function: { + arguments: argChunk, + }, + }, + ], + content: null, + refusal: null, + }, + index: 0, + finish_reason: null, + }, + ], + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + }); + } + } + }); + } + + // Final chunk with finish reason + events.push({ + choices: [ + { + delta: { + content: null, + role: 'assistant', + refusal: null, + }, + index: 0, + finish_reason: message.choices[0]!.finish_reason, + }, + ], + id: message.id, + created: message.created, + model: message.model, + object: 'chat.completion.chunk', + usage: message.usage ?? null, + }); + + return events; +} + +// Overloaded setupTest function for both streaming and non-streaming +interface SetupTestResult { + runner: OpenAI.Beta.Chat.Completions.BetaToolRunner; + fetch: ReturnType['fetch']; + handleRequest: (fetch: Fetch) => void; + handleAssistantMessage: (messageContentOrToolCalls: ToolCallsOrMessage) => ChatCompletion; + handleAssistantMessageStream: (messageContentOrToolCalls?: ToolCallsOrMessage) => ChatCompletion; +} + +type ToolCallsOrMessage = ChatCompletionMessageToolCall[] | ChatCompletionMessage; + +function setupTest(params?: Partial & { stream?: false }): SetupTestResult; +function setupTest(params: Partial & { stream: true }): SetupTestResult; +function setupTest(params: Partial = {}): SetupTestResult { + const { handleRequest, handleStreamEvents, fetch } = mockFetch(); + let messageIdCounter = 0; + const handleAssistantMessage: SetupTestResult['handleAssistantMessage'] = ( + messageContentOrToolCalls, + ) => { + const isToolCalls = Array.isArray(messageContentOrToolCalls); + + const messageContent = + isToolCalls ? + { + role: 'assistant' as const, + tool_calls: messageContentOrToolCalls, + refusal: null, + content: null, + } + : (messageContentOrToolCalls as ChatCompletionMessage); // TODO: check that this is right + + const message: ChatCompletion = { + id: `msg_${messageIdCounter++}`, + object: 'chat.completion', + created: Math.floor(Date.now() / 1000), + model: 'gpt-4', + choices: [ + { + index: 0, + message: messageContent, + finish_reason: 'stop', + logprobs: null, + }, + ], + usage: { + prompt_tokens: 10, + completion_tokens: 20, + total_tokens: 30, + }, + }; + + handleRequest(async () => { + return new Response(JSON.stringify(message), { + status: 200, + headers: { 'content-type': 'application/json' }, + }); + }); + return message; + }; + + const handleAssistantMessageStream: SetupTestResult['handleAssistantMessageStream'] = ( + messageContentOrToolCalls, + ) => { + const isToolCalls = Array.isArray(messageContentOrToolCalls); + + const messageContent = + isToolCalls ? + { + role: 'assistant' as const, + tool_calls: messageContentOrToolCalls, + refusal: null, + content: null, + } + : (messageContentOrToolCalls as ChatCompletionMessage); + + const message: ChatCompletion = { + id: `msg_${messageIdCounter++}`, + object: 'chat.completion', + created: Math.floor(Date.now() / 1000), + model: 'gpt-4', + choices: [ + { + index: 0, + message: messageContent, + finish_reason: isToolCalls ? 'tool_calls' : 'stop', + logprobs: null, + }, + ], + usage: { + prompt_tokens: 10, + completion_tokens: 20, + total_tokens: 30, + }, + }; + + handleStreamEvents(chunkChatCompletion(message)); + return message; + }; + + const client = new OpenAI({ apiKey: 'test-key', fetch: fetch, maxRetries: 0 }); + + const runnerParams: BetaToolRunnerParams = { + messages: params.messages || [{ role: 'user', content: 'What is the weather?' }], + model: params.model || 'gpt-4o', + max_tokens: params.max_tokens || 1000, + tools: params.tools || [weatherTool], + ...params, + }; + + const runner = client.beta.chat.completions.toolRunner(runnerParams); + + return { + runner, + fetch, + handleRequest, + handleAssistantMessage, + handleAssistantMessageStream, + }; +} + +async function expectEvent(iterator: AsyncIterator, assertions?: (event: T) => void | Promise) { + const result = await iterator.next(); + expect(result.done).toBe(false); + if (!result.done) { + await assertions?.(result.value); + } +} + +async function expectDone(iterator: AsyncIterator) { + const result = await iterator.next(); + expect(result.done).toBe(true); + expect(result.value).toBeUndefined(); +} + +describe('ToolRunner', () => { + it('throws when consumed multiple times', async () => { + const { runner, handleAssistantMessage } = setupTest(); + + // First consumption - get the iterator explicitly + handleAssistantMessage(getTextContent()); + await runner[Symbol.asyncIterator]().next(); + + // Second attempt to get iterator should throw + handleAssistantMessage(getTextContent()); + expect(async () => await runner[Symbol.asyncIterator]().next()).rejects.toThrow( + 'Cannot iterate over a consumed stream', + ); + }); + + it('throws when constructed with n>1', async () => { + expect(() => { + setupTest({ n: 999 }); + }).toThrow('BetaToolRunner does not support n > 1'); + + expect(() => { + setupTest({ n: null }); + }).not.toThrow(); + }); + + describe('iterator.next()', () => { + it('yields CompletionMessage', async () => { + const { runner, handleAssistantMessage } = setupTest(); + + const iterator = runner[Symbol.asyncIterator](); + + handleAssistantMessage([getWeatherToolUse('SF')]); + + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message.tool_calls).toMatchObject([getWeatherToolUse('SF')]); + }); + + handleAssistantMessage(getTextContent()); + + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message).toMatchObject(getTextContent()); + }); + + await expectDone(iterator); + }); + + it('yields ChatCompletionStream when stream=true', async () => { + const { runner, handleAssistantMessageStream } = setupTest({ stream: true }); + + const iterator = runner[Symbol.asyncIterator](); + + // First iteration: assistant requests tool (using helper that generates proper stream events) + handleAssistantMessageStream([getWeatherToolUse('SF')]); + await expectEvent(iterator, async (stream) => { + expect(stream.constructor.name).toBe('ChatCompletionStream'); + const events = []; + for await (const event of stream) { + events.push(event); + } + + // 1. Initial chunk with role only (no content, no tool_calls) + expect(events[0]).toMatchObject({ + choices: [ + { + delta: { + content: null, + refusal: null, + role: 'assistant', + }, + finish_reason: null, + index: 0, + }, + ], + object: 'chat.completion.chunk', + }); + + // 2. Tool call chunk with function name + expect(events[1]).toMatchObject({ + choices: [ + { + delta: { + tool_calls: [ + { + index: 0, + id: 'tool_1', + type: 'function', + function: { + name: 'getWeather', + }, + }, + ], + content: null, + refusal: null, + }, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + // 3-5. Argument chunks (3 chunks for the JSON string) + expect(events[2]).toMatchObject({ + choices: [ + { + delta: { + tool_calls: [ + { + index: 0, + function: { + arguments: expect.any(String), + }, + }, + ], + content: null, + refusal: null, + }, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + expect(events[3]).toMatchObject({ + choices: [ + { + delta: { + tool_calls: [ + { + index: 0, + function: { + arguments: expect.any(String), + }, + }, + ], + content: null, + refusal: null, + }, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + expect(events[4]).toMatchObject({ + choices: [ + { + delta: { + tool_calls: [ + { + index: 0, + function: { + arguments: expect.any(String), + }, + }, + ], + content: null, + refusal: null, + }, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + // 6. Final chunk with finish_reason + expect(events[5]).toMatchObject({ + choices: [ + { + delta: { + content: null, + role: 'assistant', + refusal: null, + }, + finish_reason: 'tool_calls', + }, + ], + object: 'chat.completion.chunk', + usage: { + prompt_tokens: 10, + completion_tokens: 20, + total_tokens: 30, + }, + }); + + expect(events.length).toBe(6); + }); + + // Second iteration: assistant provides final response + handleAssistantMessageStream(getTextContent()); + const result2 = await iterator.next(); + expect(result2.done).toBe(false); + + const stream2 = result2.value; + const events2: ChatCompletionChunk[] = []; + for await (const event of stream2) { + events2.push(event); + } + // Assert the expected structure of events2 + expect(events2).toHaveLength(4); + + // 1. Initial chunk with role only + expect(events2[0]).toMatchObject({ + choices: [ + { + delta: { + content: null, + refusal: null, + role: 'assistant', + }, + finish_reason: null, + index: 0, + }, + ], + object: 'chat.completion.chunk', + }); + + // 2. First text content delta + expect(events2[1]).toMatchObject({ + choices: [ + { + delta: { + content: 'Some text ', + refusal: null, + }, + index: 0, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + // 3. Second text content delta + expect(events2[2]).toMatchObject({ + choices: [ + { + delta: { + content: 'content', + refusal: null, + }, + index: 0, + finish_reason: null, + }, + ], + object: 'chat.completion.chunk', + }); + + // 4. Final chunk with finish_reason and usage + expect(events2[3]).toMatchObject({ + choices: [ + { + delta: { + content: null, + role: 'assistant', + refusal: null, + }, + index: 0, + finish_reason: 'stop', + }, + ], + object: 'chat.completion.chunk', + usage: { + prompt_tokens: 10, + completion_tokens: 20, + total_tokens: 30, + }, + }); + + await expectDone(iterator); + }); + + it('handles multiple tools', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Get weather and calculate 2+3' }], + tools: [weatherTool, calculatorTool], + }); + + const iterator = runner[Symbol.asyncIterator](); + + handleAssistantMessage([getWeatherToolUse('NYC'), getCalculatorToolUse(2, 3, 'add')]); + await expectEvent(iterator, (message) => { + expect(message?.choices).toHaveLength(1); + expect(message?.choices[0]?.message.tool_calls).toHaveLength(2); + expect(message?.choices[0]?.message.tool_calls).toMatchObject([ + getWeatherToolUse('NYC'), + getCalculatorToolUse(2, 3, 'add'), + ]); + }); + + // Assistant provides final response + handleAssistantMessage(getTextContent()); + await expectEvent(iterator, (message) => { + expect(message?.choices).toHaveLength(1); + expect(message?.choices[0]?.message).toMatchObject(getTextContent()); + }); + + // Check that we have both tool results in the messages + // Second message should be assistant with tool uses + // Third message should be user with both tool results + const messages = runner.params.messages; + expect(messages).toHaveLength(4); // user message, assistant with tools, tool result 1, tool result 2, assistant final + expect(messages[1]).toMatchObject({ + role: 'assistant', + tool_calls: [getWeatherToolUse('NYC'), getCalculatorToolUse(2, 3, 'add')], + }); + expect(messages[2]).toMatchObject(getWeatherToolResult('NYC')); + expect(messages[3]).toMatchObject(getCalculatorToolResult(2, 3, 'add', 'tool_2')); + await expectDone(iterator); + }); + + it('handles missing tool', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Use a tool' }], + tools: [weatherTool], // Only weather tool available + }); + + const iterator = runner[Symbol.asyncIterator](); + + // Assistant requests a tool that doesn't exist + handleAssistantMessage([ + { + type: 'function', + id: 'tool', + function: { + name: 'unknownTool', + arguments: JSON.stringify({ param: 'value' }), + }, + }, + ]); + + await expectEvent(iterator, (message) => { + expect(message?.choices?.[0]?.message.tool_calls).toMatchObject([ + { + type: 'function', + id: 'tool', + function: { + name: 'unknownTool', + arguments: JSON.stringify({ param: 'value' }), + }, + }, + ]); + }); + + // The tool response should contain an error + handleAssistantMessage(getTextContent()); + await expectEvent(iterator, (message) => { + // TODO: this seems sketchy + expect(message?.choices?.[0]?.message).toMatchObject(getTextContent()); + }); + + await expectDone(iterator); + }); + + it('handles tool execution errors', async () => { + const errorTool: BetaRunnableChatCompletionFunctionTool<{ shouldFail: boolean }> = { + type: 'function', + function: { + name: 'errorTool', + description: 'Tool that can fail', + parameters: { type: 'object', properties: { shouldFail: { type: 'boolean' } } }, + }, + run: async ({ shouldFail }) => { + if (shouldFail) throw new Error('Tool execution failed'); + return 'Success'; + }, + parse: (input: unknown) => input as { shouldFail: boolean }, + }; + + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Test error handling' }], + tools: [errorTool], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // Assistant requests the error tool with failure flag + handleAssistantMessage([ + { + id: 'tool_1', + type: 'function', + function: { + name: 'errorTool', + arguments: JSON.stringify({ shouldFail: true }), + }, + }, + ]); + + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message.tool_calls?.[0]).toMatchObject({ + type: 'function', + function: { + name: 'errorTool', + }, + }); + }); + + // Assistant handles the error + handleAssistantMessage(getTextContent()); + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message).toMatchObject(getTextContent()); + }); + + // Check that the tool error was properly added to the messages + expect(runner.params.messages).toHaveLength(3); + expect(runner.params.messages[2]).toMatchObject({ + role: 'tool', + tool_call_id: 'tool_1', + content: expect.stringContaining('Error: Tool execution failed'), + }); + + await expectDone(iterator); + }); + + it('handles api errors streaming', async () => { + const { runner, handleRequest, handleAssistantMessageStream } = setupTest({ + messages: [{ role: 'user', content: 'Test error handling' }], + tools: [weatherTool], + stream: true, + }); + + handleRequest(async () => { + return new Response(null, { + status: 400, + }); + }); + const iterator1 = runner[Symbol.asyncIterator](); + await expectEvent(iterator1, async (stream) => { + await expect(stream.finalMessage()).rejects.toThrow('400'); + }); + await expect(iterator1.next()).rejects.toThrow('400'); + await expectDone(iterator1); + + // We let you consume the iterator again to continue the conversation when there is an error. + handleAssistantMessageStream(getTextContent()); + const iterator2 = runner[Symbol.asyncIterator](); + await expectEvent(iterator2, (message) => { + expect(message.finalMessage()).resolves.toMatchObject(getTextContent()); + }); + await expectDone(iterator2); + }); + + it('handles api errors', async () => { + const { runner, handleRequest, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Test error handling' }], + tools: [weatherTool], + }); + + handleRequest(async () => { + return new Response(null, { + status: 500, + }); + }); + const iterator1 = runner[Symbol.asyncIterator](); + await expect(iterator1.next()).rejects.toThrow('500'); + await expectDone(iterator1); + + // We let you consume the iterator again to continue the conversation when there is an error. + handleAssistantMessage(getTextContent()); + const iterator2 = runner[Symbol.asyncIterator](); + await expectEvent(iterator2, (message) => { + expect(message?.choices?.[0]?.message).toMatchObject(getTextContent()); + }); + await expectDone(iterator2); + }); + + it('respects max_iterations parameter', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Use tools repeatedly, one at a time' }], + max_iterations: 2, // Limit to 2 iterations + }); + + const iterator = runner[Symbol.asyncIterator](); + + // First iteration + handleAssistantMessage([getWeatherToolUse('Paris')]); + await expectEvent(iterator, (message) => { + expect(message?.choices?.[0]?.message.tool_calls).toMatchObject([getWeatherToolUse('Paris')]); + }); + + // Second iteration (should be the last) + handleAssistantMessage([getWeatherToolUse('Berlin', 'tool_2')]); + await expectEvent(iterator, (message) => { + expect(message?.choices?.[0]?.message.tool_calls).toMatchObject([ + getWeatherToolUse('Berlin', 'tool_2'), + ]); + }); + + // Should stop here due to max_iterations + await expectDone(iterator); + + // When max_iterations is reached, the iterator completes even if tools were requested. + // The final message would be the last tool_use message from the assistant, + // but no further iterations occur to execute those tools. + const messages = runner.params.messages; + expect(messages).toHaveLength(5); + await expect(runner.runUntilDone()).resolves.toMatchObject({ + role: 'assistant', + tool_calls: [getWeatherToolUse('Berlin', 'tool_2')], + }); + }); + }); + + describe('iterator.return()', () => { + it('stops iteration', async () => { + const { runner, handleAssistantMessage } = setupTest(); + + const iterator = runner[Symbol.asyncIterator](); + + handleAssistantMessage([getWeatherToolUse('SF')]); + + // Get first message + await expectEvent(iterator); + + // Call return to cleanup + const returnResult = await iterator.return?.(); + expect(returnResult?.done).toBe(true); + expect(returnResult?.value).toBeUndefined(); + + // Further calls should indicate done + await expectDone(iterator); + }); + }); + + describe('.setMessagesParams()', () => { + it('updates parameters for next iteration', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Initial message' }], + max_tokens: 100, + }); + + // Update parameters before iteration + runner.setChatParams({ + messages: [{ role: 'user', content: 'Updated message' }], + model: 'gpt-4o', + max_tokens: 200, + tools: [weatherTool], + }); + + const iterator = runner[Symbol.asyncIterator](); + + handleAssistantMessage(getTextContent()); + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message).toMatchObject(getTextContent()); + }); + + // Verify params were updated + expect(runner.params.max_tokens).toBe(200); + expect(runner.params.messages[0]?.content).toBe('Updated message'); + + await expectDone(iterator); + }); + + it('allows you to update append custom tool_use blocks', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Get weather' }], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // First iteration: assistant requests tool + handleAssistantMessage([getWeatherToolUse('Paris')]); + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message?.tool_calls).toMatchObject([getWeatherToolUse('Paris')]); + }); + + // Verify generateToolResponse returns the tool result for Paris + const toolResponse = await runner.generateToolResponse(); + expect(toolResponse).toMatchObject([getWeatherToolResult('Paris')]); + + // Update params to append a custom tool_use block to messages + runner.setChatParams({ + ...runner.params, + messages: [ + ...runner.params.messages, + { role: 'assistant', tool_calls: [getWeatherToolUse('London', 'tool_1')] }, + ], + }); + + // Assistant provides final response incorporating both tool results + handleAssistantMessage(getTextContent()); + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message).toMatchObject(getTextContent()); + }); + + // Verify the messages were properly appended + // The messages array should have: initial user message + custom assistant + custom tool_use + expect(runner.params.messages).toHaveLength(3); + expect(runner.params.messages[1]).toMatchObject({ + role: 'assistant', + tool_calls: [getWeatherToolUse('London', 'tool_1')], + }); + // Verify the third message has the London tool_result + // (responded to automatically by the ToolRunner) + expect(runner.params.messages[2]).toMatchObject(getWeatherToolResult('Paris', 'tool_1')); + await expectDone(iterator); + }); + + it('allows you to use non-string returning custom tools', async () => { + const customTool: BetaRunnableChatCompletionFunctionTool<{ location: string }> = { + type: 'function', + function: { + name: 'getWeather', + description: 'Get the weather in a given location', + parameters: { + type: 'object', + properties: { + location: { type: 'string' }, + }, + }, + }, + run: async ({ location }) => { + return [ + { + type: 'image_url' as const, + image_url: { + url: `https://example.com/weather-${location}.jpg`, + }, + }, + ]; + }, + parse: (input: unknown) => input as { location: string }, + }; + + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Test done method' }], + tools: [customTool], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // Assistant requests the custom tool + handleAssistantMessage([getWeatherToolUse('Paris')]); + await expectEvent(iterator, (message) => { + expect(message?.choices[0]?.message?.tool_calls).toMatchObject([getWeatherToolUse('Paris')]); + }); + + // Verify generateToolResponse returns the custom tool result + const toolResponse = await runner.generateToolResponse(); + expect(toolResponse).toMatchObject([ + { + role: 'tool', + tool_call_id: 'tool_1', + content: JSON.stringify([ + { + type: 'image_url', + image_url: { + url: 'https://example.com/weather-Paris.jpg', + }, + }, + ]), + }, + ]); + }); + }); + + describe('.runUntilDone()', () => { + it('consumes iterator if not started', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Test done method' }], + }); + + handleAssistantMessage(getTextContent()); + const finalMessage = await runner.runUntilDone(); + expect(finalMessage).toMatchObject(getTextContent()); + }); + }); + + describe('.done()', () => { + it('waits for completion when iterator is consumed', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Test done method' }], + }); + + // Start consuming in background + const consumePromise = (async () => { + for await (const _ of runner) { + // Just consume + } + })(); + + handleAssistantMessage(getTextContent()); + const finalMessage = await runner.done(); + expect(finalMessage).toMatchObject(getTextContent()); + + await consumePromise; + }); + }); + + describe('.generateToolResponse()', () => { + it('returns tool response for last message', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Get weather' }], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // First message create call should respond with a tool use. + handleAssistantMessage([getWeatherToolUse('Miami')]); + const firstResult = await iterator.next(); + // Make sure we got the message + expect(firstResult.value).toMatchObject({ + choices: [ + { + message: { + role: 'assistant', + content: null, + refusal: null, + tool_calls: [getWeatherToolUse('Miami')], + }, + }, + ], + }); + + // When we call generateToolResponse, it should use the previous message + const toolResponse = await runner.generateToolResponse(); // this is the cached prev tool response (aka Miami weather) + expect(toolResponse?.[0]).toMatchObject(getWeatherToolResult('Miami')); + // At this point we should still only have the initial user message + // The assistant message gets added after the yield completes + expect(runner.params.messages.length).toBe(1); + + // Ending the tool loop with an assistant message should work as expected. + handleAssistantMessage(getTextContent()); + await iterator.next(); + await expectDone(iterator); + }); + + it('calls tools at most once', async () => { + let weatherToolCallCount = 0; + const trackingWeatherTool: BetaRunnableChatCompletionFunctionTool<{ location: string }> = { + type: 'function', + function: { + name: 'getWeather', + description: 'Get weather', + parameters: { type: 'object', properties: { location: { type: 'string' } } }, + }, + run: async ({ location }) => { + weatherToolCallCount++; + return `Sunny in ${location}`; + }, + parse: (input: unknown) => input as { location: string }, + }; + + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Get weather' }], + tools: [trackingWeatherTool], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // Assistant requests tool + handleAssistantMessage([getWeatherToolUse('Boston')]); + await iterator.next(); + + // Tools are executed automatically in the ToolRunner after receiving tool_use blocks + // The generateToolResponse is called internally, which should trigger the tool + // Let's call it manually to verify caching behavior + const response1 = await runner.generateToolResponse(); + expect(weatherToolCallCount).toBe(1); // Tool should be called once + expect(response1).toMatchObject([getWeatherToolResult('Boston')]); + const response2 = await runner.generateToolResponse(); + expect(weatherToolCallCount).toBe(1); // Still 1, cached + expect(response2).toMatchObject([getWeatherToolResult('Boston')]); + + // Final response should be an assistant response. + handleAssistantMessage(getTextContent()); + await iterator.next(); + + // At this point, the iterator should be completely consumed. + await expectDone(iterator); + + // Since we've never called setMessagesParams(), we should expect the tool to only be called once since it should + // all be cached. Note, that the caching mechanism here should be async-safe. + expect(weatherToolCallCount).toBe(1); + }); + + it('returns null when no tools need execution', async () => { + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'Just chat' }], + }); + + const iterator = runner[Symbol.asyncIterator](); + + handleAssistantMessage(getTextContent()); + await iterator.next(); + + // Since the previous block is a text response, we should expect generateToolResponse to return null + const toolResponse = await runner.generateToolResponse(); + expect(toolResponse).toBeNull(); + await expectDone(iterator); + }); + }); + + it('caches tool responses properly and handles text messages', async () => { + let callCount = 0; + + const { runner, handleAssistantMessage } = setupTest({ + messages: [{ role: 'user', content: 'What is the weather?' }], + tools: [ + { + type: 'function', + function: { + name: 'getWeather', + description: 'Get weather information', + parameters: { + type: 'object', + properties: { + location: { + type: 'string', + description: 'The location to get weather information for', + }, + }, + }, + }, + run: async ({ location }: { location: string }) => { + callCount++; + return `Sunny in ${location}`; + }, + parse: (input: unknown) => input as { location: string }, + }, + ], + }); + + const iterator = runner[Symbol.asyncIterator](); + + // Initial tool response should be null (no assistant message yet) + const initialResponse = await runner.generateToolResponse(); + expect(initialResponse).toBeNull(); + expect(callCount).toBe(0); + + // Assistant requests weather tool + handleAssistantMessage([getWeatherToolUse('Paris')]); + await iterator.next(); + + // Tool response should be cached + const toolResponse1 = await runner.generateToolResponse(); + expect(toolResponse1).toMatchObject([getWeatherToolResult('Paris')]); + expect(callCount).toBe(1); + + const toolResponse2 = await runner.generateToolResponse(); + expect(toolResponse2).toMatchObject([getWeatherToolResult('Paris')]); + expect(callCount).toBe(1); // Still 1 - cached response + + // Now test with a text message + handleAssistantMessage(getTextContent()); + await iterator.next(); + + const textResponse = await runner.generateToolResponse(); + expect(textResponse).toBeNull(); // Text messages return null + expect(callCount).toBe(1); // Still 1 - no new tool calls + + await expectDone(iterator); + }); +}); diff --git a/tests/lib/tools/BetaToolRunnerE2E.test.ts b/tests/lib/tools/BetaToolRunnerE2E.test.ts new file mode 100644 index 000000000..4d57ee071 --- /dev/null +++ b/tests/lib/tools/BetaToolRunnerE2E.test.ts @@ -0,0 +1,351 @@ +import { OpenAI } from '../../../src'; +import { betaZodFunctionTool } from '../../../src/helpers/beta/zod'; +import { z } from 'zod/v4'; +import nock from 'nock'; +import { gunzipSync } from 'zlib'; +import { RequestInfo } from 'openai/internal/builtin-types'; +import * as fs from 'node:fs/promises'; + +describe('toolRunner integration tests', () => { + let client: OpenAI; + let globalNockDone: (() => void) | undefined; + + beforeAll(async () => { + // Configure nock for recording/playback + nock.back.fixtures = __dirname + '/nockFixtures'; + + const isRecording = process.env['NOCK_RECORD'] === 'true'; + let apiKey = ''; + if (isRecording) { + apiKey = process.env['OPENAI_API_KEY']!; + if (!apiKey) { + throw new Error('you have to have an API key to run new snapshots'); + } + + nock.back.setMode('record'); + + // Configure nock to save readable JSON responses + nock.back.setMode('record'); + nock.recorder.rec({ + dont_print: true, + output_objects: true, + enable_reqheaders_recording: true, + }); + } else { + apiKey = 'test-api-key'; + nock.back.setMode('lockdown'); + } + + // Set up global nock recording/playback with custom transformer + const nockBack = await nock.back('ToolRunner.json', { + // Custom transformer to decompress gzipped responses + afterRecord: (scopes) => { + return scopes.map((scope) => { + const rawHeaders = (scope as any).rawHeaders as Record | undefined; + if ( + scope.response && + Array.isArray(scope.response) && + rawHeaders && + rawHeaders['content-encoding'] === 'gzip' + ) { + try { + // Decompress the gzipped response + const compressed = Buffer.from(scope.response[0], 'hex'); + const decompressed = gunzipSync(compressed); + const jsonResponse = JSON.parse(decompressed.toString()); + + // Replace with readable JSON + scope.response = jsonResponse; + + // Remove gzip header since we decompressed + delete rawHeaders['content-encoding']; + } catch (e) { + // Keep original if decompression fails + console.error('Failed to decompress response:', e); + } + } + return scope; + }); + }, + }); + globalNockDone = nockBack.nockDone; + + // Create a nock-compatible fetch function + const nockCompatibleFetch = async (input: RequestInfo | URL, init?: RequestInit): Promise => { + // Use the global fetch (Node.js 18+ or undici) which nock can intercept + const globalFetch = globalThis.fetch; + if (!globalFetch) { + throw new Error( + 'Global fetch is not available. Ensure you are using Node.js 18+ or have undici available.', + ); + } + return await globalFetch(input, init); + }; + + client = new OpenAI({ + apiKey: apiKey, + fetch: nockCompatibleFetch, + }); + }); + + afterAll(() => { + if (globalNockDone) { + globalNockDone(); + } + }); + + // Helper functions for creating common test tools + function createTestTool( + customConfig: Partial<{ + name: string; + inputSchema: z.ZodType; + description: string; + run: (args: any) => any; + }> = {}, + ) { + return betaZodFunctionTool({ + name: 'test_tool', + parameters: z.object({ value: z.string() }), + description: 'A test tool', + run: () => 'Tool result', + ...customConfig, + }); + } + + function createCounterTool() { + return betaZodFunctionTool({ + name: 'test_tool', + parameters: z.object({ count: z.number() }), + description: 'A test tool', + run: (args) => `Called with ${args.count}`, + }); + } + + it('should answer tools and run until completion', async () => { + const tool = createTestTool(); + + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + max_iterations: 5, // High limit, should stop before reaching it + messages: [ + { + role: 'user', + content: + 'Use the test_tool with value "test", then provide a final response that includes the word \'foo\'.', + }, + ], + tools: [tool], + }); + + const messages = []; + for await (const message of runner) { + messages.push(message); + } + + // Should have exactly 2 messages: tool use + final response + expect(messages).toHaveLength(2); + + // First message should contain one tool use + const firstMessage = messages[0]!.choices[0]!; + expect(firstMessage.message.role).toBe('assistant'); + expect(firstMessage.message.content).toBeNull(); // openai only responds with tool use and null content + expect(firstMessage.message.tool_calls).toHaveLength(1); // the tool call should be present + expect(firstMessage.finish_reason).toBe('tool_calls'); + + // Second message should be final response with text + expect(messages[1]!.choices).toHaveLength(1); + const secondMessage = messages[1]!.choices[0]!; + expect(secondMessage.message.role).toBe('assistant'); + expect(secondMessage.message.content).toContain('foo'); + expect(secondMessage.finish_reason).toBe('stop'); + }); + + describe('max_iterations', () => { + it('should respect max_iterations limit', async () => { + const tool = createCounterTool(); + + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + max_iterations: 2, + messages: [ + { + role: 'user', + content: + "Use the test_tool with count 1, then use it again with count 2, then say '231' in the final message", + }, + ], + tools: [tool], + }); + + const messages = []; + for await (const message of runner) { + messages.push(message); + } + + // Should have exactly 2 messages due to max_iterations limit + expect(messages).toHaveLength(2); + + // First message should contain tool uses + const firstMessage = messages[0]!.choices[0]!; + expect(firstMessage.message.role).toBe('assistant'); + expect(firstMessage.message.content).toBeNull(); + expect(firstMessage.message.tool_calls).toHaveLength(2); + + const { tool_calls: toolUseBlocks } = firstMessage.message; + expect(toolUseBlocks).toBeDefined(); + expect(toolUseBlocks).toHaveLength(2); + + if (toolUseBlocks && toolUseBlocks[0] && toolUseBlocks[0].type === 'function') { + expect(toolUseBlocks[0].function).toBeDefined(); + expect(toolUseBlocks[0].function.name).toBe('test_tool'); + expect(JSON.parse(toolUseBlocks[0].function.arguments)).toEqual({ count: 1 }); + } else { + // Doing it with an if else to get nice type inference + throw new Error('Expected tool call at index 0 to be a function'); + } + + if (toolUseBlocks && toolUseBlocks[1] && toolUseBlocks[1].type === 'function') { + expect(toolUseBlocks[1].function).toBeDefined(); + expect(toolUseBlocks[1].function.name).toBe('test_tool'); + expect(JSON.parse(toolUseBlocks[1].function.arguments)).toEqual({ count: 2 }); + } else { + throw new Error('Expected tool call at index 1 to be a function'); + } + + // Second message should be final response (not a tool call) + const secondMessage = messages[1]!.choices[0]!; + expect(secondMessage.message.role).toBe('assistant'); + expect(secondMessage.message.content).toContain('231'); + expect(secondMessage.finish_reason).toBe('stop'); + }); + }); + + describe('done()', () => { + it('should consume the iterator and return final message', async () => { + const tool = createTestTool({ inputSchema: z.object({ input: z.string() }) }); + + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [ + { + role: 'user', + content: + 'Use the test_tool with input "test", then provide a final response with the word \'231\'', + }, + ], + tools: [tool], + }); + + const finalMessage = await runner.runUntilDone(); + + // Final message should be the last text-only response + expect(finalMessage.role).toBe('assistant'); + expect(finalMessage.tool_calls).toBeUndefined(); + expect(finalMessage.content).toContain('231'); + }); + }); + + describe('setMessagesParams()', () => { + it('should update parameters using direct assignment', async () => { + const tool = createTestTool(); + + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [{ role: 'user', content: 'Hello' }], + tools: [tool], + }); + + // Update parameters + runner.setChatParams({ + model: 'gpt-4o', + max_tokens: 500, + messages: [{ role: 'user', content: 'Updated message' }], + tools: [tool], + }); + + const params = runner.params; + expect(params.model).toBe('gpt-4o'); + expect(params.max_tokens).toBe(500); + expect(params.messages).toEqual([{ role: 'user', content: 'Updated message' }]); + }); + }); + + describe('Non string returning tools', () => { + it('should handle non-string returning tools', async () => { + const exampleImageBuffer = await fs.readFile(__dirname + '/logo.png'); + const exampleImageBase64 = exampleImageBuffer.toString('base64'); + const exampleImageUrl = `data:image/png;base64,${exampleImageBase64}`; + + const tool = betaZodFunctionTool({ + name: 'cool_logo_getter_tool', + description: 'query for a company logo', + parameters: z.object({ + name: z.string().min(1).max(100).describe('the name of the company whose logo you want'), + }), + run: async () => { + return [ + { + type: 'image_url' as const, + image_url: { + url: exampleImageUrl, + }, + }, + ]; + }, + }); + + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + messages: [ + { + role: 'user', + content: + 'what is the dominant colour of the logo of the company "Stainless"? One word response nothing else', + }, + ], + tools: [tool], + }); + + const finalMessage = await runner.runUntilDone(); + const color = finalMessage.content?.toLowerCase(); + expect(['blue', 'black', 'gray', 'grey']).toContain(color); // ai is bad at colours apparently + }); + }); + + it('top level non-object schemas throw', async () => { + // At least for now, openai does not support non-object schemas at top level + const runner = client.beta.chat.completions.toolRunner({ + model: 'gpt-4o', + max_tokens: 1000, + max_iterations: 5, + messages: [ + { + role: 'user', + content: + 'Use the array_tool with the array ["hello", "world"], then provide a final response that includes the word \'foo\'.', + }, + ], + tools: [ + betaZodFunctionTool({ + name: 'array_tool', + description: 'Tool for array operations', + parameters: z.array(z.string()).describe('Array of strings'), + run: async (input: string[]) => { + return input.map((item) => item.toUpperCase()).join(', '); + }, + }), + ], + }); + await expect(async () => { + const messages = []; + for await (const message of runner) { + messages.push(message); + } + }).rejects.toThrow(); + }); +}); diff --git a/tests/lib/tools/logo.png b/tests/lib/tools/logo.png new file mode 100644 index 000000000..5072d80ad Binary files /dev/null and b/tests/lib/tools/logo.png differ diff --git a/tests/lib/tools/nockFixtures/ToolRunner.json b/tests/lib/tools/nockFixtures/ToolRunner.json new file mode 100644 index 000000000..2d70510bc --- /dev/null +++ b/tests/lib/tools/nockFixtures/ToolRunner.json @@ -0,0 +1,743 @@ +[ + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with value \"test\", then provide a final response that includes the word 'foo'." + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "value": { + "type": "string" + } + }, + "required": [ + "value" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b3404201c07ce392f9674370bc94efe64819ed37e73c4b1103f64410af5bc4297575e8996b68e024bdf83cbfdf3db428be9667be07b9bfb31b1c153342c4b38a56cb31134ea1311d469737adab37af71f8a3d8048fecccdab8c6d257abb37c886ba7b6b2d16ca64fd749fbfd6f3b3bc698f01363444510c7d400249703f9f81a86b8aa119aa246d6823ccdb61f56095924dd3292b18c7a6b671fe9bbba781bf2f68dc4b20301779141390dba25b994d1dc4fd08c325c3fab03c8c01431a442deddba6fc4fd55520fb0fd4bc459be951694e8daaa3268654ef1b9bcd6adfc1b48058b048f8e24414d2918461082ea8037997f428cbac85160d33882efe6784d83fc3d37b3ee3582f5ea7e5f506ffd0e0053b7353889ad85adb3a179297b232ac6ea573d76e56819a942e46d4e1cd6c612b4e7bc6b45bb5fa79f5a064dfbb034d9a4b312e847ba2822912823c5d8bdeb045fc0f7bbe5186b73c9928db30bbc318591e6f4fdadb9303ae8e8b70574c0a428b9e0fa16ea292fca2a68ad4cdd671d61fd223f66f00751471aa0b2d7a830f961ba4efd78a7c43347d59910d075d" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf2080f9d6e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:22 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "416", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=rXPneXdVZ1o5jQllF6SDrXFVxTPUc3cZnvKmc4fOGFA-1764868522844-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "541", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999974", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_5568cd3e444146f0987483ecb85a557d" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with value \"test\", then provide a final response that includes the word 'foo'." + }, + { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_BrhUNZ4BN8pE5j82qXAUUSXE", + "type": "function", + "function": { + "name": "test_tool", + "arguments": "{\"value\":\"test\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + { + "role": "tool", + "content": "Tool result", + "tool_call_id": "call_BrhUNZ4BN8pE5j82qXAUUSXE" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "value": { + "type": "string" + } + }, + "required": [ + "value" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b6303081c07ceb92e88a4d17588fa095439af5053f5133d5bb46bda33b73d617b68157380538145e77f2ef78e8f269dd0811fcbdf37769b03ebf95a96f71248c3b284d3cc0db5ed8bea4158cd3e3d5aecc14273fecdcdab8c6e126d7b3c3f4fc96ebd52edd3129e275b7b9d4ec2186f2708ad9c1a7aeb15098c4962df61054d9575555744c9a68ca2b41a561d954b4ad539e5756fa5da320a388b0f5af16359e1819ff284214519c1b0a8d840988c35fe1d0449c9511faed1d150db97d049d5d03253f99426cd18da0e2f4c83b264f1cb4f003d3b23626c015b6d4a25abcf6c5796b7b232ac9ad269a3a81aba15f633a7809e38026a825edcf6d5d05569cbbcea901e18f1585795ab1a690b8a04857ad536fb112f10b1c63eb35de965a4a83d70b7858810db00e77f7fae8e4d832be0c9b808dd6d4007518d53ea76c97770b9dc289555b63b21ca3a76b1dfc0c2d4431053a9c65b6f688822fa3ba8d0d705c397644977c000" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf20c18206e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:23 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "474", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=LYGly6rnpgNHCOxOFA7QeHPRYzTurihkxtXwtc9CepA-1764868523447-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "491", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999969", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_3ac4e8a7e8aa46f39fc0e1d9c04bcddd" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with count 1, then use it again with count 2, then say '231' in the final message" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "count": { + "type": "number" + } + }, + "required": [ + "count" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b1205001c0776d356336ef452185b7fa2989c27747bf5885a0b7a5e7fee8ae720a600c6d46151f0ffff6a333c1c5a98c25bfefd6f566e3b9e2fbcafde4e561625102165cd9606141f45a4d89d983dc7a7e9ab1bf22d0042f93333ce22b217c8fdf1d62cd4d33ebcf8dc7510476b27e6e6ad7ac91578c846df736006090e8281de03ac2c098aa4881c7fa892b2b6bb59450a2959a7359251be72e90001849db7813bb9a9e161ca40f557985105c00605a70540e00f6000ac6e00693e28941449d1c136383e8b776dbe12f0b7b85be98e1beff28ec075c4e37ebcec0d3f1fb39d80bb4966c9eee3370a63323365502ed195fe92bd29ea791286599c05dbfcdecbe1f4eefbe37ff8c3ef7afbcdfc5b6856697edaf2eb05634edfe67a55331ce39493ee5231676d71638dbf6b48eb982418283cdd4e6dceba2f4add6942ac2abbe7f16b8e64100ca7539e3250e3b83a638b6639dfcc2aa4b2ee1290144238504f0380922028010cc89eb12c47dcb2c0f21ef47a1681151668b89894bc06854ef7d11a4d7a7fe782b3fdc43d4de8f9aec0c71fc61909c57ba8ca3247470f52ebc5cab70b001e130a70b457e00b8a944d35753f66682bac6af302af18d007" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf20fa8706e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:24 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "628", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=InX4_raFLpPD6so5YDSanVIUeYwhE61r4XvchlbUrMI-1764868524174-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "645", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999973", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_6679f034ed8847fdb661fc2177df2e81" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with count 1, then use it again with count 2, then say '231' in the final message" + }, + { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_SOafgmSqA42A5VTHNCFzzclP", + "type": "function", + "function": { + "name": "test_tool", + "arguments": "{\"count\": 1}" + } + }, + { + "id": "call_vOoqWR3PM0cJycPOu020fsIB", + "type": "function", + "function": { + "name": "test_tool", + "arguments": "{\"count\": 2}" + } + } + ], + "refusal": null, + "annotations": [] + }, + { + "role": "tool", + "content": "Called with 1", + "tool_call_id": "call_SOafgmSqA42A5VTHNCFzzclP" + }, + { + "role": "tool", + "content": "Called with 2", + "tool_call_id": "call_vOoqWR3PM0cJycPOu020fsIB" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "count": { + "type": "number" + } + }, + "required": [ + "count" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b2503001c07ceb92e88a0d57588fa8962729ed0edd5236a19fcbc4ffdf2c3c0ba0f6176eab0284cddbce736271a602cdf188303eb7c2d4b2749362cee206f13cb077f574f37940600aebcf7cba6a0934c9fae3607f5c3dfd7571447d336cccef3ca7237c3e186877494302bbf2041001119bd0351d71443335449397921b9211f3703556a9aa709158c27433bc50200d2cf38e81b7a8ac33612c8a7a21a3100e6a8ab8bd686e3171840d49a5b57922cdec4fff147b9051fd795a1e2115e2e733bfb25b954d471d3d55e8f6e9478861d2540431cf9688683be1acf47259c9a252b9bc18e6e660f44599964e30a84294d7c4cd41031ad4ed2b43d5e35995e4c9bc1bd6e5f3e9eab5f0ab2370bd495bea0fcc9b48aefd52fa51430ed8a5855ba062a5e9a88956c68e4c5382fda431a76009583071032c1b4af59c3271807f61d3576688866282bb2e1a10f" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf21439a56e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:24 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "276", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=fk.YZ1nN6lHx9Jk384yzaRNH9.qjOCnmr57tKCKBneA-1764868524562-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "294", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999963", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_fb4e2bb27818409ea6c2bdf6d1e3a979" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with input \"test\", then provide a final response with the word '231'" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "value": { + "type": "string" + } + }, + "required": [ + "value" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b3404201c07ce392f9674370bc94efe64815e55fdd411cb03dfd840a050cf2b7479e5956869eb28b0f43db8dc3fbf2db4986eb607beb7b91f131b3c45c3b28453ca361b41a33e11419d36a7a73dab77eb50ec0144f2676e5e65ec2ed1f717f5e96539cf07e5c5df5db7a7a37bfbf51d852f9f26363444510c7d400249703f9f81a86b8aa119aaa46c6823ccdb61d5b14ac9a6e994158c6353db38ffcddd9f81bf2f68dc4b20301779141390dba2a6cca60ee27e84e192617d581ec680210da296f66d53fea7ea2a90fd077a9a0a69f4ea2548ec34b8df180fb95c0e7d72c4b48058b048f8e24414d2918461082ea8037997d49565d6428b86194417eb2f42ecffc2d37bfec5b15ebc4ecbeb0dfea1c10b76e6a61035b1b5b6752e242f65655835a5d36a37ab404d4a1723eaf066b6b015a7f531ed56ad7e5e7594ec7b2dd0c4b914e342b8272a982221c8d3d5e80d5bc4ffb0e71b6578cb9389b20db33b8c91e5f1f6a4bd3d39e0eab80877c5a420b4e8f910ea262ac92f6aaa48dd6c1d67fd2175b1df00d451c4a92eb4e80d3e586e90be5f2bf20dd1f46545361c7401" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf216aea26e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:25 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "511", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=lIOQ2BIO9WKpMpbKLYb2RseYMGvqbZaceIA4DTRhlng-1764868525176-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "531", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999975", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_85d8af06b7a047e8bc14caf4c00d6c1b" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the test_tool with input \"test\", then provide a final response with the word '231'" + }, + { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_Qxn2wdqzfjakfNA8Om3ovujF", + "type": "function", + "function": { + "name": "test_tool", + "arguments": "{\"value\":\"test\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + { + "role": "tool", + "content": "Tool result", + "tool_call_id": "call_Qxn2wdqzfjakfNA8Om3ovujF" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "test_tool", + "description": "A test tool", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "value": { + "type": "string" + } + }, + "required": [ + "value" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b5e03081c07768cb38df016ccc7f95f6b4bf5cf3bb7850ac99ac99e2162a9c0a2930a205d3ecfcd5e1c4d4a28f058c698a2e9fc8d65792f824ecf8ef814d41bf535edc897f8d71711d6250ec52abb15fbd3ceaa6d927aa328564bbfdbbcbe8e97d6f53e3ab59fbd6cb5eb319888904fdd37bc2241c5b4c8bee3bc240e59cc223fb29584ca527d79ba59e1d1eaaba5e532af1bdbda2040546364f15fe420f3129fbb2641cb4a81e11a112ec71d1fa0f22f08d2722eeb71677574d4f47cb651f1f05b7e6d445aa93cb5f85eaaa4f881a7d8416b9f00a4b29dd2290080ea5432e56d06ff6c7cb43b2e4f97637e0dc245af757913236c34c2045401a269cef71268e07439ee4f37de35e48053162907d40f409c172b0d0ac977ae678a1e6f2e58de0e3a845782e3ca513a03026afaa753b5340eb5001eae0f4b1701bdff787c38c59c8f70ed5e23514661937fad93b8fcb3877285e8c38e02a9ddc8e317de28bd53467f2f4e7ccebcca3c080396c31f" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf21a7ed76e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:25 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "479", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=qNyLPFUGXiBDkXw56Stqbq0WT1NdlcADn5_3l4Z5Hog-1764868525749-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "493", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999971", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_54ad70544bc24bbe9083778f9fa71eb3" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "what is the dominant colour of the logo of the company \"Stainless\"? One word response nothing else" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "cool_logo_getter_tool", + "description": "query for a company logo", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "name": { + "description": "the name of the company whose logo you want", + "type": "string", + "minLength": 1, + "maxLength": 100 + } + }, + "required": [ + "name" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b4504001c07ce393d16a9ec8a3af713a8723ea1a6ca3fd1b2459d5ebfee1028fd98375688ed12a5028bcef1f7aecf7168b1dc6e0fb1b27def3571f1140d7f96704ad966228566799532328d9adce99eddd21a300b6100ca9f05152fc5766e77469776b19db2284f0b66e1577266e919cf56f300055ca3ce74bf208155c4d16718d5b60cb4d0d4cc8b82c47332e127d160629e0e4405df2ad625052099fe09406f3868e1494f113e9565f46045ec080eba30e04f20054db1d6d0dc0b912265e29639e7d771c7b657a8fc0f54f43eb4a9375d33f511ecd1aad573357362650d4d43666157fc623a26b3a57839a24b4e314424c40c25d1e9141d28663366dd30873686e79948f985c6b93e7959cdbbd75f783bf3ea8c8f7f7be41f0e5ef1ca7d735ca556fd561f466f952ce107e61fa7572c4915ca9f4c8840453012c6b96790e7e40756f113653f9aae3a843892824e21a722e58c975595163159a58cf05894cdf96aa814f28ff2dd0b39f67844dbdae72db839ab93db845a22aaf48de1fc84aaf2e8a6bc6ca3b3933a30a053364048b4b29202578a2a3dd3870a1011fa2ba7085527d20d1d7d30" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf21e1e4f6e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:26 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "954", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=l0u8cvI3o1.I.XRbD.N5T7Bn.pyy3BakqRsfTHR68i0-1764868526807-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "977", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999973", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_97729a7b267f411bb257ccd8c7353602" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "what is the dominant colour of the logo of the company \"Stainless\"? One word response nothing else" + }, + { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_kFW2MaMSo1zcq86n1umN9f0S", + "type": "function", + "function": { + "name": "cool_logo_getter_tool", + "arguments": "{\"name\":\"Stainless\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + { + "role": "tool", + "content": "[{\"type\":\"image_url\",\"image_url\":{\"url\":\"\"}}]", + "tool_call_id": "call_kFW2MaMSo1zcq86n1umN9f0S" + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "cool_logo_getter_tool", + "description": "query for a company logo", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "name": { + "description": "the name of the company whose logo you want", + "type": "string", + "minLength": 1, + "maxLength": 100 + } + }, + "required": [ + "name" + ], + "additionalProperties": false + } + } + } + ], + "stream": false + }, + "status": 200, + "response": [ + "1b2e03a01c07762c637abc12dcb8c9a86da57ab319d3f8628a7b62681085aebea9c3a23075f39edb9c692c3f9b9bd3a8f3b52c8c382c4b389b6b6babaca59f7e220872f8b59efa10442e80177f73b232a527b13add968bdd28719f6bebac2ef60bfdea6d36dbf635297008cc634cbd8f472054dac73be05545d2144d16d482f5437b33285b5a2ae8102d684e7b1da5d0d807e52759ed63c366b8651c55412d88a10d4dc82f48baeef313a65183d18de036c8f41ffe142bc598ee9d3d6f609ddc3dfea0c5adb408cabab01b4e0741e7180801163014a35485fcac3d76060428af2a2b5b03d9485e2d4a22af1e3b8c708fec0a248c88112e411f4a740d8ff787cc06a68f2e6fce4946604cf7680ea7feb84673c513fbaf511e089923bd5fcfc2c0356645dac5682bba3c91267dacb98162bd6db4915703cc1e9846a2c135d6adc141865a7cfba5e169bcee8992a8d9e801" + ], + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf224bc6c6e28-EWR", + "connection": "keep-alive", + "content-encoding": "br", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:27 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "398", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=LOvRoAWCrQ6T5M3T5_CH7YLU854JepQKfWzlSncIces-1764868527317-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "transfer-encoding": "chunked", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "418", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29987343", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "25ms", + "x-request-id": "req_8d58af50dc1b4c32853fe9a53a2a76d1" + }, + "responseIsBinary": false + }, + { + "scope": "https://api.openai.com:443", + "method": "POST", + "path": "/v1/chat/completions", + "body": { + "model": "gpt-4o", + "max_tokens": 1000, + "messages": [ + { + "role": "user", + "content": "Use the array_tool with the array [\"hello\", \"world\"], then provide a final response that includes the word 'foo'." + } + ], + "tools": [ + { + "type": "function", + "function": { + "name": "array_tool", + "description": "Tool for array operations", + "parameters": { + "$schema": "https://json-schema.org/draft/2020-12/schema", + "description": "Array of strings", + "type": "array", + "items": { + "type": "string" + } + } + } + } + ], + "stream": false + }, + "status": 400, + "response": { + "error": { + "message": "Invalid schema for function 'array_tool': schema must be a JSON Schema of 'type: \"object\"', got 'type: \"array\"'.", + "type": "invalid_request_error", + "param": "tools[0].function.parameters", + "code": "invalid_function_parameters" + } + }, + "rawHeaders": { + "access-control-expose-headers": "X-Request-ID", + "alt-svc": "h3=\":443\"; ma=86400", + "cf-cache-status": "DYNAMIC", + "cf-ray": "9a8cf2284c9f6e28-EWR", + "connection": "keep-alive", + "content-length": "279", + "content-type": "application/json", + "date": "Thu, 04 Dec 2025 17:15:27 GMT", + "openai-organization": "stainless-lo3pdl", + "openai-processing-ms": "13", + "openai-project": "proj_FSQrMfywhpvjH41RhwSodie7", + "openai-version": "2020-10-01", + "server": "cloudflare", + "set-cookie": "_cfuvid=yaovjnUKaiTMtUADfGnuZOBaRER36255ThgNEffXMOQ-1764868527488-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None", + "strict-transport-security": "max-age=31536000; includeSubDomains; preload", + "x-content-type-options": "nosniff", + "x-envoy-upstream-service-time": "29", + "x-openai-proxy-wasm": "v0.1", + "x-ratelimit-limit-requests": "10000", + "x-ratelimit-limit-tokens": "30000000", + "x-ratelimit-remaining-requests": "9999", + "x-ratelimit-remaining-tokens": "29999969", + "x-ratelimit-reset-requests": "6ms", + "x-ratelimit-reset-tokens": "0s", + "x-request-id": "req_6c49242948f64e559435a8b58c423b3f" + }, + "responseIsBinary": false + } +] \ No newline at end of file diff --git a/tests/utils/mock-fetch.ts b/tests/utils/mock-fetch.ts index f8b2184f5..6349de911 100644 --- a/tests/utils/mock-fetch.ts +++ b/tests/utils/mock-fetch.ts @@ -1,4 +1,5 @@ import { type Fetch, type RequestInfo, type RequestInit, type Response } from 'openai/internal/builtin-types'; +import { PassThrough } from 'stream'; /** * Creates a mock `fetch` function and a `handleRequest` function for intercepting `fetch` calls. @@ -9,7 +10,12 @@ import { type Fetch, type RequestInfo, type RequestInit, type Response } from 'o * - calls the callback with the `fetch` arguments * - resolves `fetch` with the callback output */ -export function mockFetch(): { fetch: Fetch; handleRequest: (handle: Fetch) => Promise } { +export function mockFetch(): { + fetch: Fetch; + handleRequest: (handle: Fetch) => void; + handleStreamEvents: (events: any[]) => void; + handleMessageStreamEvents: (iter: AsyncIterable) => void; +} { const fetchQueue: ((handler: typeof fetch) => void)[] = []; const handlerQueue: Promise[] = []; @@ -61,5 +67,43 @@ export function mockFetch(): { fetch: Fetch; handleRequest: (handle: Fetch) => P }); } - return { fetch, handleRequest }; + function handleStreamEvents(events: any[]) { + handleRequest(async () => { + const stream = new PassThrough(); + (async () => { + for (const event of events) { + stream.write(`event: ${event.type}\n`); + stream.write(`data: ${JSON.stringify(event)}\n\n`); + } + stream.end(`\n`); + })(); + return new Response(stream, { + headers: { + 'Content-Type': 'text/event-stream', + 'Transfer-Encoding': 'chunked', + }, + }); + }); + } + + function handleMessageStreamEvents(iter: AsyncIterable) { + handleRequest(async () => { + const stream = new PassThrough(); + (async () => { + for await (const chunk of iter) { + stream.write(`event: ${chunk.type}\n`); + stream.write(`data: ${JSON.stringify(chunk)}\n\n`); + } + stream.end(`\n`); + })(); + return new Response(stream, { + headers: { + 'Content-Type': 'text/event-stream', + 'Transfer-Encoding': 'chunked', + }, + }); + }); + } + + return { fetch: fetch as any, handleRequest, handleStreamEvents, handleMessageStreamEvents }; } diff --git a/yarn.lock b/yarn.lock index 67b47f1a1..57107d2b6 100644 --- a/yarn.lock +++ b/yarn.lock @@ -294,6 +294,11 @@ dependencies: "@babel/helper-plugin-utils" "^7.22.5" +"@babel/runtime@^7.18.3": + version "7.28.4" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.28.4.tgz#a70226016fabe25c5783b2f22d3e1c9bc5ca3326" + integrity sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ== + "@babel/template@^7.22.15", "@babel/template@^7.3.3": version "7.22.15" resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.22.15.tgz#09576efc3830f0430f4548ef971dde1350ef2f38" @@ -700,6 +705,18 @@ "@jridgewell/resolve-uri" "^3.1.0" "@jridgewell/sourcemap-codec" "^1.4.14" +"@mswjs/interceptors@^0.39.5": + version "0.39.8" + resolved "https://registry.yarnpkg.com/@mswjs/interceptors/-/interceptors-0.39.8.tgz#0a2cf4cf26a731214ca4156273121f67dff7ebf8" + integrity sha512-2+BzZbjRO7Ct61k8fMNHEtoKjeWI9pIlHFTqBwZ5icHpqszIgEZbjb1MW5Z0+bITTCTl3gk4PDBxs9tA/csXvA== + dependencies: + "@open-draft/deferred-promise" "^2.2.0" + "@open-draft/logger" "^0.3.0" + "@open-draft/until" "^2.0.0" + is-node-process "^1.2.0" + outvariant "^1.4.3" + strict-event-emitter "^0.5.1" + "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -721,6 +738,24 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" +"@open-draft/deferred-promise@^2.2.0": + version "2.2.0" + resolved "https://registry.yarnpkg.com/@open-draft/deferred-promise/-/deferred-promise-2.2.0.tgz#4a822d10f6f0e316be4d67b4d4f8c9a124b073bd" + integrity sha512-CecwLWx3rhxVQF6V4bAgPS5t+So2sTbPgAzafKkVizyi7tlwpcFpdFqq+wqF2OwNBmqFuu6tOyouTuxgpMfzmA== + +"@open-draft/logger@^0.3.0": + version "0.3.0" + resolved "https://registry.yarnpkg.com/@open-draft/logger/-/logger-0.3.0.tgz#2b3ab1242b360aa0adb28b85f5d7da1c133a0954" + integrity sha512-X2g45fzhxH238HKO4xbSr7+wBS8Fvw6ixhTDuvLd5mqh6bJJCFAPwU9mPDxbcrRtfxv4u5IHCEH77BmxvXmmxQ== + dependencies: + is-node-process "^1.2.0" + outvariant "^1.4.0" + +"@open-draft/until@^2.0.0": + version "2.1.0" + resolved "https://registry.yarnpkg.com/@open-draft/until/-/until-2.1.0.tgz#0acf32f470af2ceaf47f095cdecd40d68666efda" + integrity sha512-U69T3ItWHvLwGg5eJ0n3I62nWuE6ilHlmz7zM0npLBRvPRd7e6NYmg54vvRtP5mZG7kZqZCFVdsTWo7BPtBujg== + "@pkgr/core@^0.2.4": version "0.2.4" resolved "https://registry.yarnpkg.com/@pkgr/core/-/core-0.2.4.tgz#d897170a2b0ba51f78a099edccd968f7b103387c" @@ -2046,6 +2081,11 @@ is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3: dependencies: is-extglob "^2.1.1" +is-node-process@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/is-node-process/-/is-node-process-1.2.0.tgz#ea02a1b90ddb3934a19aea414e88edef7e11d134" + integrity sha512-Vg4o6/fqPxIjtxgUH5QLJhwZ7gW5diGCVlXpuUfELC62CuxM1iHcRe51f2W1FDy04Ai4KJkagKjx3XaqyfRKXw== + is-number@^7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" @@ -2507,6 +2547,14 @@ json-parse-even-better-errors@^2.3.0: resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== +json-schema-to-ts@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/json-schema-to-ts/-/json-schema-to-ts-3.1.1.tgz#81f3acaf5a34736492f6f5f51870ef9ece1ca853" + integrity sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g== + dependencies: + "@babel/runtime" "^7.18.3" + ts-algebra "^2.0.0" + json-schema-traverse@^0.4.1: version "0.4.1" resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" @@ -2517,6 +2565,11 @@ json-stable-stringify-without-jsonify@^1.0.1: resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== +json-stringify-safe@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" + integrity sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA== + json5@^2.2.2, json5@^2.2.3: version "2.2.3" resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.3.tgz#78cd6f1a19bdc12b73db5ad0c61efd66c1e29283" @@ -2715,6 +2768,15 @@ natural-compare@^1.4.0: resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== +nock@^14.0.10: + version "14.0.10" + resolved "https://registry.yarnpkg.com/nock/-/nock-14.0.10.tgz#d6f4e73e1c6b4b7aa19d852176e68940e15cd19d" + integrity sha512-Q7HjkpyPeLa0ZVZC5qpxBt5EyLczFJ91MEewQiIi9taWuA0KB/MDJlUWtON+7dGouVdADTQsf9RA7TZk6D8VMw== + dependencies: + "@mswjs/interceptors" "^0.39.5" + json-stringify-safe "^5.0.1" + propagate "^2.0.0" + node-emoji@^2.1.3: version "2.1.3" resolved "https://registry.yarnpkg.com/node-emoji/-/node-emoji-2.1.3.tgz#93cfabb5cc7c3653aa52f29d6ffb7927d8047c06" @@ -2800,6 +2862,11 @@ optionator@^0.9.3: prelude-ls "^1.2.1" type-check "^0.4.0" +outvariant@^1.4.0, outvariant@^1.4.3: + version "1.4.3" + resolved "https://registry.yarnpkg.com/outvariant/-/outvariant-1.4.3.tgz#221c1bfc093e8fec7075497e7799fdbf43d14873" + integrity sha512-+Sl2UErvtsoajRDKCE5/dBz4DIvHXQQnAxtQTF04OJxY0+DyZXSo5P5Bb7XYWOh81syohlYL24hbDwxedPUJCA== + p-all@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/p-all/-/p-all-3.0.0.tgz#077c023c37e75e760193badab2bad3ccd5782bfb" @@ -2962,6 +3029,11 @@ prompts@^2.0.1: kleur "^3.0.3" sisteransi "^1.0.5" +propagate@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/propagate/-/propagate-2.0.1.tgz#40cdedab18085c792334e64f0ac17256d38f9a45" + integrity sha512-vGrhOavPSTz4QVNuBNdcNXePNdNMaO1xj9yBeH1ScQPjk/rhg9sSlCXPhMkFuaNNW/syTvYqsnbIJxMBfRbbag== + publint@^0.2.12: version "0.2.12" resolved "https://registry.yarnpkg.com/publint/-/publint-0.2.12.tgz#d25cd6bd243d5bdd640344ecdddb3eeafdcc4059" @@ -3151,6 +3223,11 @@ stack-utils@^2.0.3: dependencies: escape-string-regexp "^2.0.0" +strict-event-emitter@^0.5.1: + version "0.5.1" + resolved "https://registry.yarnpkg.com/strict-event-emitter/-/strict-event-emitter-0.5.1.tgz#1602ece81c51574ca39c6815e09f1a3e8550bd93" + integrity sha512-vMgjE/GGEPEFnhFub6pa4FmJBRBVOLpIII2hvCZ8Kzb7K0hlHo7mQv6xYrBvCL2LtAIBwFUK8wvuJgTVSQ5MFQ== + string-length@^4.0.1: version "4.0.2" resolved "https://registry.yarnpkg.com/string-length/-/string-length-4.0.2.tgz#a8a8dc7bd5c1a82b9b3c8b87e125f66871b6e57a" @@ -3295,6 +3372,11 @@ to-regex-range@^5.0.1: dependencies: is-number "^7.0.0" +ts-algebra@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ts-algebra/-/ts-algebra-2.0.0.tgz#4e3e0953878f26518fce7f6bb115064a65388b7a" + integrity sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw== + ts-api-utils@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-2.0.1.tgz#660729385b625b939aaa58054f45c058f33f10cd"