@langchain/anthropic 0.1.21 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -27,11 +27,23 @@ function _formatImage(imageUrl) {
27
27
  };
28
28
  }
29
29
  function anthropicResponseToChatMessages(messages, additionalKwargs) {
30
+ const usage = additionalKwargs.usage;
31
+ const usageMetadata = usage != null
32
+ ? {
33
+ input_tokens: usage.input_tokens ?? 0,
34
+ output_tokens: usage.output_tokens ?? 0,
35
+ total_tokens: (usage.input_tokens ?? 0) + (usage.output_tokens ?? 0),
36
+ }
37
+ : undefined;
30
38
  if (messages.length === 1 && messages[0].type === "text") {
31
39
  return [
32
40
  {
33
41
  text: messages[0].text,
34
- message: new messages_1.AIMessage(messages[0].text, additionalKwargs),
42
+ message: new messages_1.AIMessage({
43
+ content: messages[0].text,
44
+ additional_kwargs: additionalKwargs,
45
+ usage_metadata: usageMetadata,
46
+ }),
35
47
  },
36
48
  ];
37
49
  }
@@ -45,6 +57,7 @@ function anthropicResponseToChatMessages(messages, additionalKwargs) {
45
57
  content: messages,
46
58
  additional_kwargs: additionalKwargs,
47
59
  tool_calls: toolCalls,
60
+ usage_metadata: usageMetadata,
48
61
  }),
49
62
  },
50
63
  ];
@@ -590,6 +603,7 @@ class ChatAnthropicMessages extends chat_models_1.BaseChatModel {
590
603
  }, options);
591
604
  const { content, ...additionalKwargs } = response;
592
605
  const generations = anthropicResponseToChatMessages(content, additionalKwargs);
606
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
593
607
  const { role: _role, type: _type, ...rest } = additionalKwargs;
594
608
  return { generations, llmOutput: rest };
595
609
  }
@@ -193,10 +193,7 @@ export declare class ChatAnthropicMessages<CallOptions extends ChatAnthropicCall
193
193
  model: string;
194
194
  stop_reason: "max_tokens" | "stop_sequence" | "end_turn" | null;
195
195
  stop_sequence: string | null;
196
- usage: Anthropic.Messages.Usage; /** A list of strings upon which to stop generating.
197
- * You probably want `["\n\nHuman:"]`, as that's the cue for
198
- * the next turn in the dialog agent.
199
- */
196
+ usage: Anthropic.Messages.Usage;
200
197
  };
201
198
  }>;
202
199
  /** @ignore */
@@ -24,11 +24,23 @@ function _formatImage(imageUrl) {
24
24
  };
25
25
  }
26
26
  function anthropicResponseToChatMessages(messages, additionalKwargs) {
27
+ const usage = additionalKwargs.usage;
28
+ const usageMetadata = usage != null
29
+ ? {
30
+ input_tokens: usage.input_tokens ?? 0,
31
+ output_tokens: usage.output_tokens ?? 0,
32
+ total_tokens: (usage.input_tokens ?? 0) + (usage.output_tokens ?? 0),
33
+ }
34
+ : undefined;
27
35
  if (messages.length === 1 && messages[0].type === "text") {
28
36
  return [
29
37
  {
30
38
  text: messages[0].text,
31
- message: new AIMessage(messages[0].text, additionalKwargs),
39
+ message: new AIMessage({
40
+ content: messages[0].text,
41
+ additional_kwargs: additionalKwargs,
42
+ usage_metadata: usageMetadata,
43
+ }),
32
44
  },
33
45
  ];
34
46
  }
@@ -42,6 +54,7 @@ function anthropicResponseToChatMessages(messages, additionalKwargs) {
42
54
  content: messages,
43
55
  additional_kwargs: additionalKwargs,
44
56
  tool_calls: toolCalls,
57
+ usage_metadata: usageMetadata,
45
58
  }),
46
59
  },
47
60
  ];
@@ -586,6 +599,7 @@ export class ChatAnthropicMessages extends BaseChatModel {
586
599
  }, options);
587
600
  const { content, ...additionalKwargs } = response;
588
601
  const generations = anthropicResponseToChatMessages(content, additionalKwargs);
602
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
589
603
  const { role: _role, type: _type, ...rest } = additionalKwargs;
590
604
  return { generations, llmOutput: rest };
591
605
  }
@@ -1,5 +1,5 @@
1
1
  "use strict";
2
- // Auto-generated by `scripts/create-entrypoints.js`. Do not edit manually.
2
+ // Auto-generated by build script. Do not edit manually.
3
3
  var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
4
  if (k2 === undefined) k2 = k;
5
5
  var desc = Object.getOwnPropertyDescriptor(m, k);
@@ -1,3 +1,3 @@
1
- // Auto-generated by `scripts/create-entrypoints.js`. Do not edit manually.
1
+ // Auto-generated by build script. Do not edit manually.
2
2
  export * as index from "../index.js";
3
3
  export * as experimental from "../experimental/index.js";
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@langchain/anthropic",
3
- "version": "0.1.21",
3
+ "version": "0.2.0",
4
4
  "description": "Anthropic integrations for LangChain.js",
5
5
  "type": "module",
6
6
  "engines": {
@@ -14,7 +14,8 @@
14
14
  },
15
15
  "homepage": "https://github.com/langchain-ai/langchainjs/tree/main/libs/langchain-anthropic/",
16
16
  "scripts": {
17
- "build": "yarn run build:deps && yarn clean && yarn build:esm && yarn build:cjs && yarn build:scripts",
17
+ "build": "yarn turbo:command build:internal --filter=@langchain/anthropic",
18
+ "build:internal": "yarn lc-build:v2 --create-entrypoints --pre --tree-shaking --gen-maps",
18
19
  "build:deps": "yarn run turbo:command build --filter=@langchain/core",
19
20
  "build:esm": "NODE_OPTIONS=--max-old-space-size=4096 tsc --outDir dist/",
20
21
  "build:cjs": "NODE_OPTIONS=--max-old-space-size=4096 tsc --outDir dist-cjs/ -p tsconfig.cjs.json && yarn move-cjs-to-dist && rimraf dist-cjs",
@@ -24,7 +25,7 @@
24
25
  "lint:dpdm": "dpdm --exit-code circular:1 --no-warning --no-tree src/*.ts src/**/*.ts",
25
26
  "lint": "yarn lint:eslint && yarn lint:dpdm",
26
27
  "lint:fix": "yarn lint:eslint --fix && yarn lint:dpdm",
27
- "clean": "rimraf .turbo/ dist/ && NODE_OPTIONS=--max-old-space-size=4096 yarn lc-build --config ./langchain.config.js --create-entrypoints --pre --gen-maps",
28
+ "clean": "rm -rf .turbo dist/",
28
29
  "prepack": "yarn build",
29
30
  "test": "yarn run build:deps && NODE_OPTIONS=--experimental-vm-modules jest --testPathIgnorePatterns=\\.int\\.test.ts --testTimeout 30000 --maxWorkers=50%",
30
31
  "test:watch": "yarn run build:deps && NODE_OPTIONS=--experimental-vm-modules jest --watch --testPathIgnorePatterns=\\.int\\.test.ts",
@@ -40,7 +41,7 @@
40
41
  "license": "MIT",
41
42
  "dependencies": {
42
43
  "@anthropic-ai/sdk": "^0.21.0",
43
- "@langchain/core": ">0.1.56 <0.3.0",
44
+ "@langchain/core": ">=0.2.5 <0.3.0",
44
45
  "fast-xml-parser": "^4.3.5",
45
46
  "zod": "^3.22.4",
46
47
  "zod-to-json-schema": "^3.22.4"
@@ -48,7 +49,7 @@
48
49
  "devDependencies": {
49
50
  "@jest/globals": "^29.5.0",
50
51
  "@langchain/community": "workspace:*",
51
- "@langchain/scripts": "~0.0",
52
+ "@langchain/scripts": "~0.0.14",
52
53
  "@swc/core": "^1.3.90",
53
54
  "@swc/jest": "^0.2.29",
54
55
  "dpdm": "^3.12.0",
@@ -1 +0,0 @@
1
- export {};
@@ -1,39 +0,0 @@
1
- // import { test, expect } from "@jest/globals";
2
- // import { ChatPromptTemplate } from "@langchain/core/prompts";
3
- // import { TavilySearchResults } from "@langchain/community/tools/tavily_search";
4
- // import { AgentExecutor, createToolCallingAgent } from "langchain/agents";
5
- // import { ChatAnthropic } from "../index.js";
6
- // const tools = [new TavilySearchResults({ maxResults: 1 })];
7
- // TODO: This test breaks CI build due to dependencies. Figure out a way around it.
8
- test("createToolCallingAgent works", async () => {
9
- // const prompt = ChatPromptTemplate.fromMessages([
10
- // ["system", "You are a helpful assistant"],
11
- // ["placeholder", "{chat_history}"],
12
- // ["human", "{input}"],
13
- // ["placeholder", "{agent_scratchpad}"],
14
- // ]);
15
- // const llm = new ChatAnthropic({
16
- // modelName: "claude-3-sonnet-20240229",
17
- // temperature: 0,
18
- // });
19
- // const agent = await createToolCallingAgent({
20
- // llm,
21
- // tools,
22
- // prompt,
23
- // });
24
- // const agentExecutor = new AgentExecutor({
25
- // agent,
26
- // tools,
27
- // });
28
- // const input = "what is the current weather in SF?";
29
- // const result = await agentExecutor.invoke({
30
- // input,
31
- // });
32
- // console.log(result);
33
- // expect(result.input).toBe(input);
34
- // expect(typeof result.output).toBe("string");
35
- // // Length greater than 10 because any less than that would warrant
36
- // // an investigation into why such a short generation was returned.
37
- // expect(result.output.length).toBeGreaterThan(10);
38
- });
39
- export {};
@@ -1 +0,0 @@
1
- export {};
@@ -1,278 +0,0 @@
1
- /* eslint-disable no-process-env */
2
- import { expect, test } from "@jest/globals";
3
- import { AIMessage, HumanMessage, ToolMessage } from "@langchain/core/messages";
4
- import { StructuredTool } from "@langchain/core/tools";
5
- import { z } from "zod";
6
- import { zodToJsonSchema } from "zod-to-json-schema";
7
- import { ChatAnthropic } from "../chat_models.js";
8
- const zodSchema = z
9
- .object({
10
- location: z.string().describe("The name of city to get the weather for."),
11
- })
12
- .describe("Get the weather of a specific location and return the temperature in Celsius.");
13
- class WeatherTool extends StructuredTool {
14
- constructor() {
15
- super(...arguments);
16
- Object.defineProperty(this, "schema", {
17
- enumerable: true,
18
- configurable: true,
19
- writable: true,
20
- value: z.object({
21
- location: z.string().describe("The name of city to get the weather for."),
22
- })
23
- });
24
- Object.defineProperty(this, "description", {
25
- enumerable: true,
26
- configurable: true,
27
- writable: true,
28
- value: "Get the weather of a specific location and return the temperature in Celsius."
29
- });
30
- Object.defineProperty(this, "name", {
31
- enumerable: true,
32
- configurable: true,
33
- writable: true,
34
- value: "get_weather"
35
- });
36
- }
37
- async _call(input) {
38
- console.log(`WeatherTool called with input: ${input}`);
39
- return `The weather in ${input.location} is 25°C`;
40
- }
41
- }
42
- const model = new ChatAnthropic({
43
- modelName: "claude-3-sonnet-20240229",
44
- temperature: 0,
45
- });
46
- const anthropicTool = {
47
- name: "get_weather",
48
- description: "Get the weather of a specific location and return the temperature in Celsius.",
49
- input_schema: {
50
- type: "object",
51
- properties: {
52
- location: {
53
- type: "string",
54
- description: "The name of city to get the weather for.",
55
- },
56
- },
57
- required: ["location"],
58
- },
59
- };
60
- test("Few shotting with tool calls", async () => {
61
- const chat = model.bindTools([new WeatherTool()]);
62
- const res = await chat.invoke([
63
- new HumanMessage("What is the weather in SF?"),
64
- new AIMessage({
65
- content: "Let me look up the current weather.",
66
- tool_calls: [
67
- {
68
- id: "toolu_feiwjf9u98r389u498",
69
- name: "get_weather",
70
- args: {
71
- location: "SF",
72
- },
73
- },
74
- ],
75
- }),
76
- new ToolMessage({
77
- tool_call_id: "toolu_feiwjf9u98r389u498",
78
- content: "It is currently 24 degrees with hail in San Francisco.",
79
- }),
80
- new AIMessage("It is currently 24 degrees in San Francisco with hail in San Francisco."),
81
- new HumanMessage("What did you say the weather was?"),
82
- ]);
83
- console.log(res);
84
- expect(res.content).toContain("24");
85
- });
86
- test("Can bind & invoke StructuredTools", async () => {
87
- const tools = [new WeatherTool()];
88
- const modelWithTools = model.bindTools(tools);
89
- const result = await modelWithTools.invoke("What is the weather in SF today?");
90
- console.log({
91
- tool_calls: JSON.stringify(result.content, null, 2),
92
- }, "Can bind & invoke StructuredTools");
93
- expect(Array.isArray(result.content)).toBeTruthy();
94
- if (!Array.isArray(result.content)) {
95
- throw new Error("Content is not an array");
96
- }
97
- let toolCall;
98
- result.content.forEach((item) => {
99
- if (item.type === "tool_use") {
100
- toolCall = item;
101
- }
102
- });
103
- if (!toolCall) {
104
- throw new Error("No tool call found");
105
- }
106
- expect(toolCall).toBeTruthy();
107
- const { name, input } = toolCall;
108
- expect(toolCall.input).toEqual(result.tool_calls?.[0].args);
109
- expect(name).toBe("get_weather");
110
- expect(input).toBeTruthy();
111
- expect(input.location).toBeTruthy();
112
- const result2 = await modelWithTools.invoke([
113
- new HumanMessage("What is the weather in SF today?"),
114
- result,
115
- new ToolMessage({
116
- tool_call_id: result.tool_calls?.[0].id ?? "",
117
- content: "The weather in San Francisco is currently 59 degrees and sunny.",
118
- }),
119
- new AIMessage("The weather in San Francisco is currently 59 degrees and sunny."),
120
- new HumanMessage("What did you say the weather was?"),
121
- ]);
122
- console.log(result2);
123
- // This should work, but Anthorpic is too skeptical
124
- expect(result2.content).toContain("59");
125
- });
126
- test("Can bind & invoke AnthropicTools", async () => {
127
- const modelWithTools = model.bind({
128
- tools: [anthropicTool],
129
- });
130
- const result = await modelWithTools.invoke("What is the weather in London today?");
131
- console.log({
132
- tool_calls: JSON.stringify(result.content, null, 2),
133
- }, "Can bind & invoke StructuredTools");
134
- expect(Array.isArray(result.content)).toBeTruthy();
135
- if (!Array.isArray(result.content)) {
136
- throw new Error("Content is not an array");
137
- }
138
- let toolCall;
139
- result.content.forEach((item) => {
140
- if (item.type === "tool_use") {
141
- toolCall = item;
142
- }
143
- });
144
- if (!toolCall) {
145
- throw new Error("No tool call found");
146
- }
147
- expect(toolCall).toBeTruthy();
148
- const { name, input } = toolCall;
149
- expect(name).toBe("get_weather");
150
- expect(input).toBeTruthy();
151
- expect(input.location).toBeTruthy();
152
- });
153
- test("Can bind & stream AnthropicTools", async () => {
154
- const modelWithTools = model.bind({
155
- tools: [anthropicTool],
156
- });
157
- const result = await modelWithTools.stream("What is the weather in London today?");
158
- let finalMessage;
159
- for await (const item of result) {
160
- console.log("item", JSON.stringify(item, null, 2));
161
- finalMessage = item;
162
- }
163
- if (!finalMessage) {
164
- throw new Error("No final message returned");
165
- }
166
- console.log({
167
- tool_calls: JSON.stringify(finalMessage.content, null, 2),
168
- }, "Can bind & invoke StructuredTools");
169
- expect(Array.isArray(finalMessage.content)).toBeTruthy();
170
- if (!Array.isArray(finalMessage.content)) {
171
- throw new Error("Content is not an array");
172
- }
173
- let toolCall;
174
- finalMessage.content.forEach((item) => {
175
- if (item.type === "tool_use") {
176
- toolCall = item;
177
- }
178
- });
179
- if (!toolCall) {
180
- throw new Error("No tool call found");
181
- }
182
- expect(toolCall).toBeTruthy();
183
- const { name, input } = toolCall;
184
- expect(name).toBe("get_weather");
185
- expect(input).toBeTruthy();
186
- expect(input.location).toBeTruthy();
187
- });
188
- test("withStructuredOutput with zod schema", async () => {
189
- const modelWithTools = model.withStructuredOutput(zodSchema, {
190
- name: "get_weather",
191
- });
192
- const result = await modelWithTools.invoke("What is the weather in London today?");
193
- console.log({
194
- result,
195
- }, "withStructuredOutput with zod schema");
196
- expect(typeof result.location).toBe("string");
197
- });
198
- test("withStructuredOutput with AnthropicTool", async () => {
199
- const modelWithTools = model.withStructuredOutput(anthropicTool, {
200
- name: anthropicTool.name,
201
- });
202
- const result = await modelWithTools.invoke("What is the weather in London today?");
203
- console.log({
204
- result,
205
- }, "withStructuredOutput with AnthropicTool");
206
- expect(typeof result.location).toBe("string");
207
- });
208
- test("withStructuredOutput JSON Schema only", async () => {
209
- const jsonSchema = zodToJsonSchema(zodSchema);
210
- const modelWithTools = model.withStructuredOutput(jsonSchema, {
211
- name: "get_weather",
212
- });
213
- const result = await modelWithTools.invoke("What is the weather in London today?");
214
- console.log({
215
- result,
216
- }, "withStructuredOutput JSON Schema only");
217
- expect(typeof result.location).toBe("string");
218
- });
219
- test("Can pass tool_choice", async () => {
220
- const tool1 = {
221
- name: "get_weather",
222
- description: "Get the weather of a specific location and return the temperature in Celsius.",
223
- input_schema: {
224
- type: "object",
225
- properties: {
226
- location: {
227
- type: "string",
228
- description: "The name of city to get the weather for.",
229
- },
230
- },
231
- required: ["location"],
232
- },
233
- };
234
- const tool2 = {
235
- name: "calculator",
236
- description: "Calculate any math expression and return the result.",
237
- input_schema: {
238
- type: "object",
239
- properties: {
240
- expression: {
241
- type: "string",
242
- description: "The math expression to calculate.",
243
- },
244
- },
245
- required: ["expression"],
246
- },
247
- };
248
- const tools = [tool1, tool2];
249
- const modelWithTools = model.bindTools(tools, {
250
- tool_choice: {
251
- type: "tool",
252
- name: "get_weather",
253
- },
254
- });
255
- const result = await modelWithTools.invoke("What is the sum of 272818 and 281818?");
256
- console.log({
257
- tool_calls: JSON.stringify(result.content, null, 2),
258
- }, "Can bind & invoke StructuredTools");
259
- expect(Array.isArray(result.content)).toBeTruthy();
260
- if (!Array.isArray(result.content)) {
261
- throw new Error("Content is not an array");
262
- }
263
- let toolCall;
264
- result.content.forEach((item) => {
265
- if (item.type === "tool_use") {
266
- toolCall = item;
267
- }
268
- });
269
- if (!toolCall) {
270
- throw new Error("No tool call found");
271
- }
272
- expect(toolCall).toBeTruthy();
273
- const { name, input } = toolCall;
274
- expect(toolCall.input).toEqual(result.tool_calls?.[0].args);
275
- expect(name).toBe("get_weather");
276
- expect(input).toBeTruthy();
277
- expect(input.location).toBeTruthy();
278
- });
@@ -1 +0,0 @@
1
- export {};
@@ -1,275 +0,0 @@
1
- /* eslint-disable no-process-env */
2
- import { expect, test } from "@jest/globals";
3
- import { HumanMessage } from "@langchain/core/messages";
4
- import { ChatPromptValue } from "@langchain/core/prompt_values";
5
- import { PromptTemplate, ChatPromptTemplate, AIMessagePromptTemplate, HumanMessagePromptTemplate, SystemMessagePromptTemplate, } from "@langchain/core/prompts";
6
- import { CallbackManager } from "@langchain/core/callbacks/manager";
7
- import { ChatAnthropic } from "../chat_models.js";
8
- test("Test ChatAnthropic", async () => {
9
- const chat = new ChatAnthropic({
10
- modelName: "claude-3-sonnet-20240229",
11
- maxRetries: 0,
12
- });
13
- const message = new HumanMessage("Hello!");
14
- const res = await chat.invoke([message]);
15
- console.log({ res });
16
- expect(res.response_metadata.usage).toBeDefined();
17
- });
18
- test("Test ChatAnthropic Generate", async () => {
19
- const chat = new ChatAnthropic({
20
- modelName: "claude-3-sonnet-20240229",
21
- maxRetries: 0,
22
- });
23
- const message = new HumanMessage("Hello!");
24
- const res = await chat.generate([[message], [message]]);
25
- expect(res.generations.length).toBe(2);
26
- for (const generation of res.generations) {
27
- expect(generation.length).toBe(1);
28
- for (const message of generation) {
29
- console.log(message.text);
30
- }
31
- }
32
- console.log({ res });
33
- });
34
- test.skip("Test ChatAnthropic Generate w/ ClientOptions", async () => {
35
- const chat = new ChatAnthropic({
36
- modelName: "claude-3-sonnet-20240229",
37
- maxRetries: 0,
38
- clientOptions: {
39
- defaultHeaders: {
40
- "Helicone-Auth": "HELICONE_API_KEY",
41
- },
42
- },
43
- });
44
- const message = new HumanMessage("Hello!");
45
- const res = await chat.generate([[message], [message]]);
46
- expect(res.generations.length).toBe(2);
47
- for (const generation of res.generations) {
48
- expect(generation.length).toBe(1);
49
- for (const message of generation) {
50
- console.log(message.text);
51
- }
52
- }
53
- console.log({ res });
54
- });
55
- test("Test ChatAnthropic Generate with a signal in call options", async () => {
56
- const chat = new ChatAnthropic({
57
- modelName: "claude-3-sonnet-20240229",
58
- maxRetries: 0,
59
- });
60
- const controller = new AbortController();
61
- const message = new HumanMessage("How is your day going? Be extremely verbose!");
62
- await expect(() => {
63
- const res = chat.generate([[message], [message]], {
64
- signal: controller.signal,
65
- });
66
- setTimeout(() => {
67
- controller.abort();
68
- }, 1000);
69
- return res;
70
- }).rejects.toThrow();
71
- }, 10000);
72
- test("Test ChatAnthropic tokenUsage with a batch", async () => {
73
- const model = new ChatAnthropic({
74
- temperature: 0,
75
- maxRetries: 0,
76
- modelName: "claude-3-sonnet-20240229",
77
- });
78
- const res = await model.generate([
79
- [new HumanMessage(`Hello!`)],
80
- [new HumanMessage(`Hi!`)],
81
- ]);
82
- console.log({ res });
83
- });
84
- test("Test ChatAnthropic in streaming mode", async () => {
85
- let nrNewTokens = 0;
86
- let streamedCompletion = "";
87
- const model = new ChatAnthropic({
88
- modelName: "claude-3-sonnet-20240229",
89
- maxRetries: 0,
90
- streaming: true,
91
- callbacks: CallbackManager.fromHandlers({
92
- async handleLLMNewToken(token) {
93
- nrNewTokens += 1;
94
- streamedCompletion += token;
95
- },
96
- }),
97
- });
98
- const message = new HumanMessage("Hello!");
99
- const res = await model.invoke([message]);
100
- console.log({ res });
101
- expect(nrNewTokens > 0).toBe(true);
102
- expect(res.content).toBe(streamedCompletion);
103
- });
104
- test("Test ChatAnthropic in streaming mode with a signal", async () => {
105
- let nrNewTokens = 0;
106
- let streamedCompletion = "";
107
- const model = new ChatAnthropic({
108
- modelName: "claude-3-sonnet-20240229",
109
- maxRetries: 0,
110
- streaming: true,
111
- callbacks: CallbackManager.fromHandlers({
112
- async handleLLMNewToken(token) {
113
- nrNewTokens += 1;
114
- streamedCompletion += token;
115
- },
116
- }),
117
- });
118
- const controller = new AbortController();
119
- const message = new HumanMessage("Hello! Give me an extremely verbose response");
120
- await expect(() => {
121
- const res = model.invoke([message], {
122
- signal: controller.signal,
123
- });
124
- setTimeout(() => {
125
- controller.abort();
126
- }, 500);
127
- return res;
128
- }).rejects.toThrow();
129
- console.log({ nrNewTokens, streamedCompletion });
130
- }, 5000);
131
- test.skip("Test ChatAnthropic prompt value", async () => {
132
- const chat = new ChatAnthropic({
133
- modelName: "claude-3-sonnet-20240229",
134
- maxRetries: 0,
135
- });
136
- const message = new HumanMessage("Hello!");
137
- const res = await chat.generatePrompt([new ChatPromptValue([message])]);
138
- expect(res.generations.length).toBe(1);
139
- for (const generation of res.generations) {
140
- for (const g of generation) {
141
- console.log(g.text);
142
- }
143
- }
144
- console.log({ res });
145
- });
146
- test.skip("ChatAnthropic, docs, prompt templates", async () => {
147
- const chat = new ChatAnthropic({
148
- modelName: "claude-3-sonnet-20240229",
149
- maxRetries: 0,
150
- temperature: 0,
151
- });
152
- const systemPrompt = PromptTemplate.fromTemplate("You are a helpful assistant that translates {input_language} to {output_language}.");
153
- const chatPrompt = ChatPromptTemplate.fromMessages([
154
- new SystemMessagePromptTemplate(systemPrompt),
155
- HumanMessagePromptTemplate.fromTemplate("{text}"),
156
- ]);
157
- const responseA = await chat.generatePrompt([
158
- await chatPrompt.formatPromptValue({
159
- input_language: "English",
160
- output_language: "French",
161
- text: "I love programming.",
162
- }),
163
- ]);
164
- console.log(responseA.generations);
165
- });
166
- test.skip("ChatAnthropic, longer chain of messages", async () => {
167
- const chat = new ChatAnthropic({
168
- modelName: "claude-3-sonnet-20240229",
169
- maxRetries: 0,
170
- temperature: 0,
171
- });
172
- const chatPrompt = ChatPromptTemplate.fromMessages([
173
- HumanMessagePromptTemplate.fromTemplate(`Hi, my name is Joe!`),
174
- AIMessagePromptTemplate.fromTemplate(`Nice to meet you, Joe!`),
175
- HumanMessagePromptTemplate.fromTemplate("{text}"),
176
- ]);
177
- const responseA = await chat.generatePrompt([
178
- await chatPrompt.formatPromptValue({
179
- text: "What did I just say my name was?",
180
- }),
181
- ]);
182
- console.log(responseA.generations);
183
- });
184
- test.skip("ChatAnthropic, Anthropic apiUrl set manually via constructor", async () => {
185
- // Pass the default URL through (should use this, and work as normal)
186
- const anthropicApiUrl = "https://api.anthropic.com";
187
- const chat = new ChatAnthropic({
188
- modelName: "claude-3-sonnet-20240229",
189
- maxRetries: 0,
190
- anthropicApiUrl,
191
- });
192
- const message = new HumanMessage("Hello!");
193
- const res = await chat.call([message]);
194
- console.log({ res });
195
- });
196
- test("Test ChatAnthropic stream method", async () => {
197
- const model = new ChatAnthropic({
198
- maxTokens: 50,
199
- maxRetries: 0,
200
- modelName: "claude-3-sonnet-20240229",
201
- });
202
- const stream = await model.stream("Print hello world.");
203
- const chunks = [];
204
- for await (const chunk of stream) {
205
- console.log(chunk);
206
- chunks.push(chunk);
207
- }
208
- expect(chunks.length).toBeGreaterThan(1);
209
- });
210
- test("Test ChatAnthropic stream method with abort", async () => {
211
- await expect(async () => {
212
- const model = new ChatAnthropic({
213
- maxTokens: 500,
214
- maxRetries: 0,
215
- modelName: "claude-3-sonnet-20240229",
216
- });
217
- const stream = await model.stream("How is your day going? Be extremely verbose.", {
218
- signal: AbortSignal.timeout(1000),
219
- });
220
- for await (const chunk of stream) {
221
- console.log(chunk);
222
- }
223
- }).rejects.toThrow();
224
- });
225
- test("Test ChatAnthropic stream method with early break", async () => {
226
- const model = new ChatAnthropic({
227
- maxTokens: 50,
228
- maxRetries: 0,
229
- modelName: "claude-3-sonnet-20240229",
230
- });
231
- const stream = await model.stream("How is your day going? Be extremely verbose.");
232
- let i = 0;
233
- for await (const chunk of stream) {
234
- console.log(chunk);
235
- i += 1;
236
- if (i > 10) {
237
- break;
238
- }
239
- }
240
- });
241
- test("Test ChatAnthropic headers passed through", async () => {
242
- const chat = new ChatAnthropic({
243
- modelName: "claude-3-sonnet-20240229",
244
- maxRetries: 0,
245
- apiKey: "NOT_REAL",
246
- clientOptions: {
247
- defaultHeaders: {
248
- "X-Api-Key": process.env.ANTHROPIC_API_KEY,
249
- },
250
- },
251
- });
252
- const message = new HumanMessage("Hello!");
253
- const res = await chat.invoke([message]);
254
- console.log({ res });
255
- });
256
- test("Test ChatAnthropic multimodal", async () => {
257
- const chat = new ChatAnthropic({
258
- modelName: "claude-3-sonnet-20240229",
259
- maxRetries: 0,
260
- });
261
- const res = await chat.invoke([
262
- new HumanMessage({
263
- content: [
264
- {
265
- type: "image_url",
266
- image_url: {
267
- url: "",
268
- },
269
- },
270
- { type: "text", text: "What is this a logo for?" },
271
- ],
272
- }),
273
- ]);
274
- console.log(res);
275
- });
@@ -1 +0,0 @@
1
- export {};
@@ -1,88 +0,0 @@
1
- import { jest, test } from "@jest/globals";
2
- import { AIMessage } from "@langchain/core/messages";
3
- import { z } from "zod";
4
- import { OutputParserException } from "@langchain/core/output_parsers";
5
- import { ChatAnthropic } from "../chat_models.js";
6
- test("withStructuredOutput with output validation", async () => {
7
- const model = new ChatAnthropic({
8
- modelName: "claude-3-haiku-20240307",
9
- temperature: 0,
10
- anthropicApiKey: "testing",
11
- });
12
- jest
13
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
14
- .spyOn(model, "invoke")
15
- .mockResolvedValue(new AIMessage({
16
- content: [
17
- {
18
- type: "tool_use",
19
- id: "notreal",
20
- name: "Extractor",
21
- input: "Incorrect string tool call input",
22
- },
23
- ],
24
- }));
25
- const schema = z.object({
26
- alerts: z
27
- .array(z.object({
28
- description: z.string().describe("A description of the alert."),
29
- severity: z
30
- .enum(["HIGH", "MEDIUM", "LOW"])
31
- .describe("How severe the alert is."),
32
- }))
33
- .describe("Important security or infrastructure alerts present in the given text."),
34
- });
35
- const modelWithStructuredOutput = model.withStructuredOutput(schema, {
36
- name: "Extractor",
37
- });
38
- await expect(async () => {
39
- await modelWithStructuredOutput.invoke(`
40
- Enumeration of Kernel Modules via Proc
41
- Prompt for Credentials with OSASCRIPT
42
- User Login
43
- Modification of Standard Authentication Module
44
- Suspicious Automator Workflows Execution
45
- `);
46
- }).rejects.toThrowError(OutputParserException);
47
- });
48
- test("withStructuredOutput with proper output", async () => {
49
- const model = new ChatAnthropic({
50
- modelName: "claude-3-haiku-20240307",
51
- temperature: 0,
52
- anthropicApiKey: "testing",
53
- });
54
- jest
55
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
56
- .spyOn(model, "invoke")
57
- .mockResolvedValue(new AIMessage({
58
- content: [
59
- {
60
- type: "tool_use",
61
- id: "notreal",
62
- name: "Extractor",
63
- input: { alerts: [{ description: "test", severity: "LOW" }] },
64
- },
65
- ],
66
- }));
67
- const schema = z.object({
68
- alerts: z
69
- .array(z.object({
70
- description: z.string().describe("A description of the alert."),
71
- severity: z
72
- .enum(["HIGH", "MEDIUM", "LOW"])
73
- .describe("How severe the alert is."),
74
- }))
75
- .describe("Important security or infrastructure alerts present in the given text."),
76
- });
77
- const modelWithStructuredOutput = model.withStructuredOutput(schema, {
78
- name: "Extractor",
79
- });
80
- const result = await modelWithStructuredOutput.invoke(`
81
- Enumeration of Kernel Modules via Proc
82
- Prompt for Credentials with OSASCRIPT
83
- User Login
84
- Modification of Standard Authentication Module
85
- Suspicious Automator Workflows Execution
86
- `);
87
- console.log(result);
88
- });