From 79acf470550af09d70c4ed2f0661194225d96190 Mon Sep 17 00:00:00 2001 From: David Duong Date: Fri, 16 Feb 2024 19:32:09 +0100 Subject: [PATCH] Add OpenAI traceable wrapper for JS (#451) --- README.md | 12 +- js/.gitignore | 3 + js/README.md | 99 ++++-------- js/package.json | 11 +- js/scripts/create-entrypoints.js | 1 + js/src/tests/wrapped_openai.int.test.ts | 128 ++++++++++++++++ js/src/wrappers.ts | 22 +++ js/tsconfig.json | 3 +- js/yarn.lock | 194 ++++++++++++++++++++++++ 9 files changed, 395 insertions(+), 78 deletions(-) create mode 100644 js/src/tests/wrapped_openai.int.test.ts create mode 100644 js/src/wrappers.ts diff --git a/README.md b/README.md index 2fcefafd4..72d020b84 100644 --- a/README.md +++ b/README.md @@ -43,17 +43,13 @@ export LANGCHAIN_API_KEY=ls_... Then start tracing your app! ```javascript -import { traceable } from "langsmith/traceable"; import { OpenAI } from "openai"; +import { traceable } from "langsmith/traceable"; +import { wrapOpenAI } from "langsmith/wrappers"; -const client = new OpenAI(); - -const createCompletion = traceable( - openai.chat.completions.create.bind(openai.chat.completions), - { name: "OpenAI Chat Completion", run_type: "llm" } -); +const client = wrapOpenAI(new OpenAI()); -await createCompletion({ +await client.chat.completions.create({ model: "gpt-3.5-turbo", messages: [{ content: "Hi there!", role: "user" }], }); diff --git a/js/.gitignore b/js/.gitignore index a87badf7c..0f5848f98 100644 --- a/js/.gitignore +++ b/js/.gitignore @@ -46,6 +46,9 @@ Chinook_Sqlite.sql /schemas.cjs /schemas.js /schemas.d.ts +/wrappers.cjs +/wrappers.js +/wrappers.d.ts /index.cjs /index.js /index.d.ts diff --git a/js/README.md b/js/README.md index de971791f..73eed5728 100644 --- a/js/README.md +++ b/js/README.md @@ -90,6 +90,7 @@ Langsmith's `traceable` wrapper function makes it easy to trace any function or ### OpenAI SDK + The easiest ways to trace calls from the [OpenAI SDK](https://platform.openai.com/docs/api-reference) with LangSmith is using the `traceable` wrapper function available in LangSmith 0.1.0 and up. @@ -105,72 +106,41 @@ Next, you will need to install the LangSmith SDK and the OpenAI SDK: npm install langsmith openai ``` -After that, initialize your OpenAI client: +After that, initialize your OpenAI client and wrap the client with `wrapOpenAI` method to enable tracing for Completion and Chat completion API: ```ts import { OpenAI } from "openai"; +import { wrapOpenAI } from "langsmith/wrappers"; + +const openai = wrapOpenAI(new OpenAI()); -const client = new OpenAI(); +await openai.chat.completions.create({ + model: "gpt-3.5-turbo", + messages: [{ content: "Hi there!", role: "user" }], +}); ``` -Then, you can wrap the client methods you want to use by passing it to the `traceable` function like this: +Alternatively, you can use the `traceable` function to wrap the client methods you want to use: ```ts import { traceable } from "langsmith/traceable"; +const openai = new OpenAI(); + const createCompletion = traceable( openai.chat.completions.create.bind(openai.chat.completions), { name: "OpenAI Chat Completion", run_type: "llm" } ); -``` - -Note the use of `.bind` to preserve the function's context. The `run_type` field in the extra config object -marks the function as an LLM call, and enables token usage tracking for OpenAI. - -This new method takes the same exact arguments and has the same return type as the original method, -but will log everything to LangSmith! -```ts await createCompletion({ model: "gpt-3.5-turbo", messages: [{ content: "Hi there!", role: "user" }], }); ``` -``` -{ - id: 'chatcmpl-8sOWEOYVyehDlyPcBiaDtTxWvr9v6', - object: 'chat.completion', - created: 1707974654, - model: 'gpt-3.5-turbo-0613', - choices: [ - { - index: 0, - message: { role: 'assistant', content: 'Hello! How can I help you today?' }, - logprobs: null, - finish_reason: 'stop' - } - ], - usage: { prompt_tokens: 10, completion_tokens: 9, total_tokens: 19 }, - system_fingerprint: null -} -``` - -This also works for streaming: - -```ts -const stream = await createCompletion({ - model: "gpt-3.5-turbo", - stream: true, - messages: [{ content: "Hi there!", role: "user" }], -}); -``` - -```ts -for await (const chunk of stream) { - console.log(chunk); -} -``` +Note the use of `.bind` to preserve the function's context. The `run_type` field in the +extra config object marks the function as an LLM call, and enables token usage tracking +for OpenAI. Oftentimes, you use the OpenAI client inside of other functions or as part of a longer sequence. You can automatically get nested traces by using this wrapped method @@ -178,7 +148,7 @@ within other functions wrapped with `traceable`. ```ts const nestedTrace = traceable(async (text: string) => { - const completion = await createCompletion({ + const completion = await openai.chat.completions.create({ model: "gpt-3.5-turbo", messages: [{ content: text, role: "user" }], }); @@ -230,25 +200,22 @@ import { NextRequest, NextResponse } from "next/server"; import { OpenAI } from "openai"; import { traceable } from "langsmith/traceable"; +import { wrapOpenAI } from "langsmith/wrappers"; export const runtime = "edge"; const handler = traceable( async function () { - const openai = new OpenAI(); - const createCompletion = traceable( - openai.chat.completions.create.bind(openai.chat.completions), - { name: "OpenAI Chat Completion", run_type: "llm" } - ); + const openai = wrapOpenAI(new OpenAI()); - const completion = await createCompletion({ + const completion = await openai.chat.completions.create({ model: "gpt-3.5-turbo", messages: [{ content: "Why is the sky blue?", role: "user" }], }); const response1 = completion.choices[0].message.content; - const completion2 = await createCompletion({ + const completion2 = await openai.chat.completions.create({ model: "gpt-3.5-turbo", messages: [ { content: "Why is the sky blue?", role: "user" }, @@ -287,28 +254,25 @@ The [Vercel AI SDK](https://sdk.vercel.ai/docs) contains integrations with a var Here's an example of how you can trace outputs in a Next.js handler: ```ts -import { traceable } from 'langsmith/traceable'; -import { OpenAIStream, StreamingTextResponse } from 'ai'; +import { traceable } from "langsmith/traceable"; +import { OpenAIStream, StreamingTextResponse } from "ai"; // Note: There are no types for the Mistral API client yet. -import MistralClient from '@mistralai/mistralai'; +import MistralClient from "@mistralai/mistralai"; -const client = new MistralClient(process.env.MISTRAL_API_KEY || ''); +const client = new MistralClient(process.env.MISTRAL_API_KEY || ""); export async function POST(req: Request) { // Extract the `messages` from the body of the request const { messages } = await req.json(); - const mistralChatStream = traceable( - client.chatStream.bind(client), - { - name: "Mistral Stream", - run_type: "llm", - } - ); + const mistralChatStream = traceable(client.chatStream.bind(client), { + name: "Mistral Stream", + run_type: "llm", + }); const response = await mistralChatStream({ - model: 'mistral-tiny', + model: "mistral-tiny", maxTokens: 1000, messages, }); @@ -324,7 +288,6 @@ export async function POST(req: Request) { See the [AI SDK docs](https://sdk.vercel.ai/docs) for more examples. - #### Alternatives: **Log traces using a RunTree.** A RunTree tracks your application. Each RunTree object is required to have a name and run_type. These and other important attributes are as follows: @@ -413,7 +376,7 @@ try { await childChainRun.end({ error: `I errored again ${e.message}`, }); - await childChainRun.patchRun(); + await childChainRun.patchRun(); throw e; } @@ -431,7 +394,7 @@ await parentRun.patchRun(); ## Evaluation -#### Create a Dataset from Existing Runs +#### Create a Dataset from Existing Runs Once your runs are stored in LangSmith, you can convert them into a dataset. For this example, we will do so using the Client, but you can also do this using diff --git a/js/package.json b/js/package.json index fc6c187da..24037d110 100644 --- a/js/package.json +++ b/js/package.json @@ -20,6 +20,9 @@ "schemas.cjs", "schemas.js", "schemas.d.ts", + "wrappers.cjs", + "wrappers.js", + "wrappers.d.ts", "index.cjs", "index.js", "index.d.ts" @@ -80,6 +83,7 @@ "eslint-plugin-no-instanceof": "^1.0.1", "eslint-plugin-prettier": "^4.2.1", "jest": "^29.5.0", + "openai": "^4.28.0", "prettier": "^2.8.8", "ts-jest": "^29.1.0", "ts-node": "^10.9.1", @@ -129,6 +133,11 @@ "import": "./schemas.js", "require": "./schemas.cjs" }, + "./wrappers": { + "types": "./wrappers.d.ts", + "import": "./wrappers.js", + "require": "./wrappers.cjs" + }, "./package.json": "./package.json" } -} \ No newline at end of file +} diff --git a/js/scripts/create-entrypoints.js b/js/scripts/create-entrypoints.js index ef8ade94f..778a1adc1 100644 --- a/js/scripts/create-entrypoints.js +++ b/js/scripts/create-entrypoints.js @@ -12,6 +12,7 @@ const entrypoints = { traceable: "traceable", evaluation: "evaluation/index", schemas: "schemas", + wrappers: "wrappers", }; const updateJsonFile = (relativePath, updateFunction) => { const contents = fs.readFileSync(relativePath).toString(); diff --git a/js/src/tests/wrapped_openai.int.test.ts b/js/src/tests/wrapped_openai.int.test.ts new file mode 100644 index 000000000..9335e2cb9 --- /dev/null +++ b/js/src/tests/wrapped_openai.int.test.ts @@ -0,0 +1,128 @@ +import { jest } from "@jest/globals"; +import { OpenAI } from "openai"; +import { wrapOpenAI } from "../wrappers.js"; +import { Client } from "../client.js"; + +test.concurrent("chat.completions", async () => { + const client = new Client(); + const callSpy = jest + // eslint-disable-next-line @typescript-eslint/no-explicit-any + .spyOn((client as any).caller, "call") + .mockResolvedValue({ ok: true, text: () => "" }); + + const originalClient = new OpenAI(); + const patchedClient = wrapOpenAI(new OpenAI(), { client }); + + // invoke + const original = await originalClient.chat.completions.create({ + messages: [{ role: "user", content: `Say 'foo'` }], + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo", + }); + + const patched = await patchedClient.chat.completions.create({ + messages: [{ role: "user", content: `Say 'foo'` }], + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo", + }); + + expect(patched.choices).toEqual(original.choices); + + // stream + const originalStream = await originalClient.chat.completions.create({ + messages: [{ role: "user", content: `Say 'foo'` }], + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo", + stream: true, + }); + + const originalChoices = []; + for await (const chunk of originalStream) { + originalChoices.push(chunk.choices); + } + + const patchedStream = await patchedClient.chat.completions.create({ + messages: [{ role: "user", content: `Say 'foo'` }], + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo", + stream: true, + }); + + const patchedChoices = []; + for await (const chunk of patchedStream) { + patchedChoices.push(chunk.choices); + } + + expect(patchedChoices).toEqual(originalChoices); + for (const call of callSpy.mock.calls) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + expect((call[2] as any)["method"]).toBe("POST"); + } +}); + +test.concurrent("completions", async () => { + const client = new Client(); + const callSpy = jest + // eslint-disable-next-line @typescript-eslint/no-explicit-any + .spyOn((client as any).caller, "call") + .mockResolvedValue({ ok: true, text: () => "" }); + + const originalClient = new OpenAI(); + const patchedClient = wrapOpenAI(new OpenAI(), { client }); + + const prompt = `Say 'Hi I'm ChatGPT' then stop.`; + + // invoke + const original = await originalClient.completions.create({ + prompt, + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo-instruct", + }); + + const patched = await patchedClient.completions.create({ + prompt, + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo-instruct", + }); + + expect(patched.choices).toEqual(original.choices); + + // stream + const originalStream = await originalClient.completions.create({ + prompt, + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo-instruct", + stream: true, + }); + + const originalChoices = []; + for await (const chunk of originalStream) { + originalChoices.push(chunk.choices); + } + + const patchedStream = await patchedClient.completions.create({ + prompt, + temperature: 0, + seed: 42, + model: "gpt-3.5-turbo-instruct", + stream: true, + }); + + const patchedChoices = []; + for await (const chunk of patchedStream) { + patchedChoices.push(chunk.choices); + } + + expect(patchedChoices).toEqual(originalChoices); + for (const call of callSpy.mock.calls) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + expect((call[2] as any)["method"]).toBe("POST"); + } +}); diff --git a/js/src/wrappers.ts b/js/src/wrappers.ts new file mode 100644 index 000000000..9a25c8d6b --- /dev/null +++ b/js/src/wrappers.ts @@ -0,0 +1,22 @@ +import type { OpenAI } from "openai"; +import type { Client } from "./index.js"; +import { traceable } from "./traceable.js"; + +export const wrapOpenAI = ( + openai: OpenAI, + options?: { client?: Client } +): OpenAI => { + // @ts-expect-error Promise> != APIPromise<...> + openai.chat.completions.create = traceable( + openai.chat.completions.create.bind(openai.chat.completions), + Object.assign({ name: "ChatOpenAI", run_type: "llm" }, options?.client) + ); + + // @ts-expect-error Promise> != APIPromise<...> + openai.completions.create = traceable( + openai.completions.create.bind(openai.completions), + Object.assign({ name: "OpenAI", run_type: "llm" }, options?.client) + ); + + return openai; +}; diff --git a/js/tsconfig.json b/js/tsconfig.json index 5edd93e56..cb73e47d0 100644 --- a/js/tsconfig.json +++ b/js/tsconfig.json @@ -36,7 +36,8 @@ "src/run_trees.ts", "src/traceable.ts", "src/evaluation/index.ts", - "src/schemas.ts" + "src/schemas.ts", + "src/wrappers.ts" ] } } diff --git a/js/yarn.lock b/js/yarn.lock index a153e3709..a10225cfc 100644 --- a/js/yarn.lock +++ b/js/yarn.lock @@ -1540,11 +1540,26 @@ resolved "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz" integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== +"@types/node-fetch@^2.6.4": + version "2.6.11" + resolved "https://registry.yarnpkg.com/@types/node-fetch/-/node-fetch-2.6.11.tgz#9b39b78665dae0e82a08f02f4967d62c66f95d24" + integrity sha512-24xFj9R5+rfQJLRyM56qh+wnVSYhyXC2tkoBndtY0U+vubqNsYXGjufB2nn8Q6gt0LrARwL6UBtMCSVCwl4B1g== + dependencies: + "@types/node" "*" + form-data "^4.0.0" + "@types/node@*": version "20.2.5" resolved "https://registry.npmjs.org/@types/node/-/node-20.2.5.tgz" integrity sha512-JJulVEQXmiY9Px5axXHeYGLSjhkZEnD+MDPDGbCbIAbMslkKwmygtZFy1X6s/075Yo94sf8GuSlFfPzysQrWZQ== +"@types/node@^18.11.18": + version "18.19.17" + resolved "https://registry.yarnpkg.com/@types/node/-/node-18.19.17.tgz#a581a9fb4b2cfdbc61f008804f4436b2d5c40354" + integrity sha512-SzyGKgwPzuWp2SHhlpXKzCX0pIOfcI4V2eF37nNBJOhwlegQ83omtVQ1XxZpDE06V/d6AQvfQdPfnw0tRC//Ng== + dependencies: + undici-types "~5.26.4" + "@types/prettier@^2.1.5": version "2.7.2" resolved "https://registry.npmjs.org/@types/prettier/-/prettier-2.7.2.tgz" @@ -1666,6 +1681,13 @@ "@typescript-eslint/types" "5.59.8" eslint-visitor-keys "^3.3.0" +abort-controller@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/abort-controller/-/abort-controller-3.0.0.tgz#eaf54d53b62bae4138e809ca225c8439a6efb392" + integrity sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg== + dependencies: + event-target-shim "^5.0.0" + acorn-jsx@^5.3.2: version "5.3.2" resolved "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz" @@ -1681,6 +1703,13 @@ acorn@^8.4.1, acorn@^8.8.0: resolved "https://registry.npmjs.org/acorn/-/acorn-8.8.2.tgz" integrity sha512-xjIYgE8HBrkpd/sJqOGNspf8uHG+NOHGOw6a/Urj8taM2EXfdNAH2oFcPeIFfsv3+kz/mJrS5VuMqbNLjCa2vw== +agentkeepalive@^4.2.1: + version "4.5.0" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.5.0.tgz#2673ad1389b3c418c5a20c5d7364f93ca04be923" + integrity sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew== + dependencies: + humanize-ms "^1.2.1" + ajv@^6.10.0, ajv@^6.12.4: version "6.12.6" resolved "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz" @@ -1791,6 +1820,11 @@ array.prototype.flatmap@^1.3.1: es-abstract "^1.20.4" es-shim-unscopables "^1.0.0" +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q== + available-typed-arrays@^1.0.5: version "1.0.5" resolved "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz" @@ -1885,6 +1919,11 @@ balanced-match@^1.0.0: resolved "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz" integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== +base-64@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/base-64/-/base-64-0.1.0.tgz#780a99c84e7d600260361511c4877613bf24f6bb" + integrity sha512-Y5gU45svrR5tI2Vt/X9GPd3L0HNIKzGu202EjxrXMpuc2V2CiKgemAbUUsqYmZJvPtCXoUKjNZwBJzsNScUbXA== + base64-js@^1.5.1: version "1.5.1" resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" @@ -1989,6 +2028,11 @@ char-regex@^1.0.2: resolved "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz" integrity sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw== +charenc@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/charenc/-/charenc-0.0.2.tgz#c0a1d2f3a7092e03774bfa83f14c0fc5790a8667" + integrity sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA== + ci-info@^3.2.0: version "3.8.0" resolved "https://registry.npmjs.org/ci-info/-/ci-info-3.8.0.tgz" @@ -2042,6 +2086,13 @@ color-name@~1.1.4: resolved "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== +combined-stream@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" + integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== + dependencies: + delayed-stream "~1.0.0" + commander@^10.0.1: version "10.0.1" resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06" @@ -2090,6 +2141,11 @@ cross-spawn@^7.0.1, cross-spawn@^7.0.2, cross-spawn@^7.0.3: shebang-command "^2.0.0" which "^2.0.1" +crypt@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/crypt/-/crypt-0.0.2.tgz#88d7ff7ec0dfb86f713dc87bbb42d044d3e6c41b" + integrity sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow== + debug@^3.2.7: version "3.2.7" resolved "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz" @@ -2132,6 +2188,11 @@ define-properties@^1.1.3, define-properties@^1.1.4, define-properties@^1.2.0: has-property-descriptors "^1.0.0" object-keys "^1.1.1" +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ== + detect-newline@^3.0.0: version "3.1.0" resolved "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz" @@ -2147,6 +2208,14 @@ diff@^4.0.1: resolved "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz" integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== +digest-fetch@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/digest-fetch/-/digest-fetch-1.3.0.tgz#898e69264d00012a23cf26e8a3e40320143fc661" + integrity sha512-CGJuv6iKNM7QyZlM2T3sPAdZWd/p9zQiRNS9G+9COUCwzWFTs0Xp8NF5iePx7wtvhDykReiRRrSeNb4oMmB8lA== + dependencies: + base-64 "^0.1.0" + md5 "^2.3.0" + dir-glob@^3.0.1: version "3.0.1" resolved "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz" @@ -2443,6 +2512,11 @@ esutils@^2.0.2: resolved "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz" integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== +event-target-shim@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/event-target-shim/-/event-target-shim-5.0.1.tgz#5d4d3ebdf9583d63a5333ce2deb7480ab2b05789" + integrity sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ== + eventemitter3@^4.0.4: version "4.0.7" resolved "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz" @@ -2574,6 +2648,28 @@ for-each@^0.3.3: dependencies: is-callable "^1.1.3" +form-data-encoder@1.7.2: + version "1.7.2" + resolved "https://registry.yarnpkg.com/form-data-encoder/-/form-data-encoder-1.7.2.tgz#1f1ae3dccf58ed4690b86d87e4f57c654fbab040" + integrity sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A== + +form-data@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.0.tgz#93919daeaf361ee529584b9b31664dc12c9fa452" + integrity sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww== + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.8" + mime-types "^2.1.12" + +formdata-node@^4.3.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/formdata-node/-/formdata-node-4.4.1.tgz#23f6a5cb9cb55315912cbec4ff7b0f59bbd191e2" + integrity sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ== + dependencies: + node-domexception "1.0.0" + web-streams-polyfill "4.0.0-beta.3" + fs.realpath@^1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz" @@ -2777,6 +2873,13 @@ human-signals@^2.1.0: resolved "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz" integrity sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw== +humanize-ms@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed" + integrity sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ== + dependencies: + ms "^2.0.0" + ignore@^5.2.0: version "5.2.4" resolved "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz" @@ -2859,6 +2962,11 @@ is-boolean-object@^1.1.0: call-bind "^1.0.2" has-tostringtag "^1.0.0" +is-buffer@~1.1.6: + version "1.1.6" + resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" + integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== + is-callable@^1.1.3, is-callable@^1.1.4, is-callable@^1.2.7: version "1.2.7" resolved "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz" @@ -3542,6 +3650,15 @@ makeerror@1.0.12: dependencies: tmpl "1.0.5" +md5@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/md5/-/md5-2.3.0.tgz#c3da9a6aae3a30b46b7b0c349b87b110dc3bda4f" + integrity sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g== + dependencies: + charenc "0.0.2" + crypt "0.0.2" + is-buffer "~1.1.6" + merge-stream@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz" @@ -3560,6 +3677,18 @@ micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" +mime-db@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.12: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + mimic-fn@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz" @@ -3618,6 +3747,11 @@ ms@2.1.2, ms@^2.1.1: resolved "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz" integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== +ms@^2.0.0: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== + natural-compare-lite@^1.4.0: version "1.4.0" resolved "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz" @@ -3628,6 +3762,18 @@ natural-compare@^1.4.0: resolved "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== +node-domexception@1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/node-domexception/-/node-domexception-1.0.0.tgz#6888db46a1f71c0b76b3f7555016b63fe64766e5" + integrity sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ== + +node-fetch@^2.6.7: + version "2.7.0" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d" + integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A== + dependencies: + whatwg-url "^5.0.0" + node-int64@^0.4.0: version "0.4.0" resolved "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz" @@ -3698,6 +3844,21 @@ onetime@^5.1.2: dependencies: mimic-fn "^2.1.0" +openai@^4.28.0: + version "4.28.0" + resolved "https://registry.yarnpkg.com/openai/-/openai-4.28.0.tgz#ded00e3d98c25758b5406c9675ec27a957e00930" + integrity sha512-JM8fhcpmpGN0vrUwGquYIzdcEQHtFuom6sRCbbCM6CfzZXNuRk33G7KfeRAIfnaCxSpzrP5iHtwJzIm6biUZ2Q== + dependencies: + "@types/node" "^18.11.18" + "@types/node-fetch" "^2.6.4" + abort-controller "^3.0.0" + agentkeepalive "^4.2.1" + digest-fetch "^1.3.0" + form-data-encoder "1.7.2" + formdata-node "^4.3.2" + node-fetch "^2.6.7" + web-streams-polyfill "^3.2.1" + optionator@^0.9.1: version "0.9.1" resolved "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz" @@ -4211,6 +4372,11 @@ to-regex-range@^5.0.1: dependencies: is-number "^7.0.0" +tr46@~0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" + integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw== + ts-jest@^29.1.0: version "29.1.0" resolved "https://registry.npmjs.org/ts-jest/-/ts-jest-29.1.0.tgz" @@ -4312,6 +4478,11 @@ unbox-primitive@^1.0.2: has-symbols "^1.0.3" which-boxed-primitive "^1.0.2" +undici-types@~5.26.4: + version "5.26.5" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" + integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== + unicode-canonical-property-names-ecmascript@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz" @@ -4376,6 +4547,29 @@ walker@^1.0.8: dependencies: makeerror "1.0.12" +web-streams-polyfill@4.0.0-beta.3: + version "4.0.0-beta.3" + resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-4.0.0-beta.3.tgz#2898486b74f5156095e473efe989dcf185047a38" + integrity sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug== + +web-streams-polyfill@^3.2.1: + version "3.3.2" + resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-3.3.2.tgz#32e26522e05128203a7de59519be3c648004343b" + integrity sha512-3pRGuxRF5gpuZc0W+EpwQRmCD7gRqcDOMt688KmdlDAgAyaB1XlN0zq2njfDNm44XVdIouE7pZ6GzbdyH47uIQ== + +webidl-conversions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" + integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ== + +whatwg-url@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d" + integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw== + dependencies: + tr46 "~0.0.3" + webidl-conversions "^3.0.0" + which-boxed-primitive@^1.0.2: version "1.0.2" resolved "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz"