@librechat/agents 1.8.8 → 1.9.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.
Files changed (74) hide show
  1. package/dist/cjs/common/enum.cjs +1 -0
  2. package/dist/cjs/common/enum.cjs.map +1 -1
  3. package/dist/cjs/events.cjs +8 -1
  4. package/dist/cjs/events.cjs.map +1 -1
  5. package/dist/cjs/llm/anthropic/llm.cjs +119 -0
  6. package/dist/cjs/llm/anthropic/llm.cjs.map +1 -0
  7. package/dist/cjs/llm/anthropic/utils/message_inputs.cjs +277 -0
  8. package/dist/cjs/llm/anthropic/utils/message_inputs.cjs.map +1 -0
  9. package/dist/cjs/llm/anthropic/utils/message_outputs.cjs +135 -0
  10. package/dist/cjs/llm/anthropic/utils/message_outputs.cjs.map +1 -0
  11. package/dist/cjs/llm/providers.cjs +5 -4
  12. package/dist/cjs/llm/providers.cjs.map +1 -1
  13. package/dist/cjs/llm/text.cjs +58 -0
  14. package/dist/cjs/llm/text.cjs.map +1 -0
  15. package/dist/cjs/main.cjs +3 -0
  16. package/dist/cjs/main.cjs.map +1 -1
  17. package/dist/cjs/messages.cjs +4 -4
  18. package/dist/cjs/messages.cjs.map +1 -1
  19. package/dist/cjs/stream.cjs +63 -48
  20. package/dist/cjs/stream.cjs.map +1 -1
  21. package/dist/cjs/tools/ToolNode.cjs +20 -5
  22. package/dist/cjs/tools/ToolNode.cjs.map +1 -1
  23. package/dist/cjs/utils/misc.cjs +49 -0
  24. package/dist/cjs/utils/misc.cjs.map +1 -0
  25. package/dist/esm/common/enum.mjs +1 -0
  26. package/dist/esm/common/enum.mjs.map +1 -1
  27. package/dist/esm/events.mjs +8 -1
  28. package/dist/esm/events.mjs.map +1 -1
  29. package/dist/esm/llm/anthropic/llm.mjs +117 -0
  30. package/dist/esm/llm/anthropic/llm.mjs.map +1 -0
  31. package/dist/esm/llm/anthropic/utils/message_inputs.mjs +274 -0
  32. package/dist/esm/llm/anthropic/utils/message_inputs.mjs.map +1 -0
  33. package/dist/esm/llm/anthropic/utils/message_outputs.mjs +133 -0
  34. package/dist/esm/llm/anthropic/utils/message_outputs.mjs.map +1 -0
  35. package/dist/esm/llm/providers.mjs +5 -4
  36. package/dist/esm/llm/providers.mjs.map +1 -1
  37. package/dist/esm/llm/text.mjs +56 -0
  38. package/dist/esm/llm/text.mjs.map +1 -0
  39. package/dist/esm/main.mjs +2 -1
  40. package/dist/esm/main.mjs.map +1 -1
  41. package/dist/esm/messages.mjs +4 -4
  42. package/dist/esm/messages.mjs.map +1 -1
  43. package/dist/esm/stream.mjs +63 -49
  44. package/dist/esm/stream.mjs.map +1 -1
  45. package/dist/esm/tools/ToolNode.mjs +22 -7
  46. package/dist/esm/tools/ToolNode.mjs.map +1 -1
  47. package/dist/esm/utils/misc.mjs +47 -0
  48. package/dist/esm/utils/misc.mjs.map +1 -0
  49. package/dist/types/common/enum.d.ts +2 -1
  50. package/dist/types/llm/anthropic/types.d.ts +4 -0
  51. package/dist/types/llm/anthropic/utils/message_inputs.d.ts +1 -1
  52. package/dist/types/llm/text.d.ts +6 -6
  53. package/dist/types/stream.d.ts +2 -0
  54. package/dist/types/types/llm.d.ts +6 -1
  55. package/dist/types/utils/index.d.ts +1 -0
  56. package/dist/types/utils/misc.d.ts +6 -0
  57. package/package.json +7 -6
  58. package/src/common/enum.ts +1 -0
  59. package/src/events.ts +9 -1
  60. package/src/llm/anthropic/llm.ts +9 -6
  61. package/src/llm/anthropic/types.ts +7 -1
  62. package/src/llm/anthropic/utils/message_inputs.ts +86 -8
  63. package/src/llm/providers.ts +6 -4
  64. package/src/llm/text.ts +30 -45
  65. package/src/messages.ts +4 -4
  66. package/src/scripts/args.ts +1 -1
  67. package/src/scripts/code_exec.ts +4 -0
  68. package/src/scripts/simple.ts +5 -1
  69. package/src/stream.ts +68 -50
  70. package/src/tools/ToolNode.ts +25 -9
  71. package/src/types/llm.ts +6 -1
  72. package/src/utils/index.ts +1 -0
  73. package/src/utils/llmConfig.ts +6 -0
  74. package/src/utils/misc.ts +45 -0
@@ -1,11 +1,11 @@
1
- import { END, MessagesAnnotation } from '@langchain/langgraph';
1
+ import { END, MessagesAnnotation, isCommand, isGraphInterrupt } from '@langchain/langgraph';
2
2
  import { ToolMessage, isBaseMessage } from '@langchain/core/messages';
3
3
  import type { RunnableConfig, RunnableToolLike } from '@langchain/core/runnables';
4
4
  import type { BaseMessage, AIMessage } from '@langchain/core/messages';
5
5
  import type { StructuredToolInterface } from '@langchain/core/tools';
6
6
  import type * as t from '@/types';
7
- import{ RunnableCallable } from '@/utils';
8
- import { GraphNodeKeys } from '@/common';
7
+ import{ RunnableCallable, unescapeObject } from '@/utils';
8
+ import { GraphNodeKeys, Providers } from '@/common';
9
9
 
10
10
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
11
  export class ToolNode<T = any> extends RunnableCallable<T, T> {
@@ -46,7 +46,6 @@ export class ToolNode<T = any> extends RunnableCallable<T, T> {
46
46
  this.tools = tools;
47
47
  this.toolMap = toolMap ?? new Map(tools.map(tool => [tool.name, tool]));
48
48
  }
49
-
50
49
  const outputs = await Promise.all(
51
50
  (message as AIMessage).tool_calls?.map(async (call) => {
52
51
  const tool = this.toolMap.get(call.name);
@@ -54,11 +53,15 @@ export class ToolNode<T = any> extends RunnableCallable<T, T> {
54
53
  if (tool === undefined) {
55
54
  throw new Error(`Tool "${call.name}" not found.`);
56
55
  }
56
+ const args = config.metadata?.provider === Providers.GOOGLE ? unescapeObject(call.args) : call.args;
57
57
  const output = await tool.invoke(
58
- { ...call, type: 'tool_call' },
58
+ { ...call, args, type: 'tool_call' },
59
59
  config
60
60
  );
61
- if (isBaseMessage(output) && output._getType() === 'tool') {
61
+ if (
62
+ (isBaseMessage(output) && output._getType() === 'tool') ||
63
+ isCommand(output)
64
+ ) {
62
65
  return output;
63
66
  } else {
64
67
  return new ToolMessage({
@@ -68,11 +71,14 @@ export class ToolNode<T = any> extends RunnableCallable<T, T> {
68
71
  tool_call_id: call.id!,
69
72
  });
70
73
  }
71
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
72
- } catch (e: any) {
74
+ } catch (_e: unknown) {
75
+ const e = _e as Error;
73
76
  if (!this.handleToolErrors) {
74
77
  throw e;
75
78
  }
79
+ if (isGraphInterrupt(e)) {
80
+ throw e;
81
+ }
76
82
  return new ToolMessage({
77
83
  content: `Error: ${e.message}\n Please fix your mistakes.`,
78
84
  name: call.name,
@@ -82,7 +88,17 @@ export class ToolNode<T = any> extends RunnableCallable<T, T> {
82
88
  }) ?? []
83
89
  );
84
90
 
85
- return (Array.isArray(input) ? outputs : { messages: outputs }) as T;
91
+ if (!outputs.some(isCommand)) {
92
+ return (Array.isArray(input) ? outputs : { messages: outputs }) as T;
93
+ }
94
+
95
+ const combinedOutputs = outputs.map((output) => {
96
+ if (isCommand(output)) {
97
+ return output;
98
+ }
99
+ return Array.isArray(input) ? [output] : { messages: [output] };
100
+ });
101
+ return combinedOutputs as T;
86
102
  }
87
103
  }
88
104
 
package/src/types/llm.ts CHANGED
@@ -6,12 +6,14 @@ import { ChatMistralAI } from '@langchain/mistralai';
6
6
  import { ChatBedrockConverse } from '@langchain/aws';
7
7
  import { ChatVertexAI } from '@langchain/google-vertexai';
8
8
  import { BedrockChat } from '@langchain/community/chat_models/bedrock/web';
9
+ import { ChatGoogleGenerativeAI } from '@langchain/google-genai';
9
10
  import type { Runnable } from '@langchain/core/runnables';
10
11
  import type { StructuredTool } from '@langchain/core/tools';
11
12
  import type { BindToolsInput } from '@langchain/core/language_models/chat_models';
12
13
  import type { BedrockChatFields } from '@langchain/community/chat_models/bedrock/web';
13
14
  import type { ChatOpenAIFields } from '@langchain/openai';
14
15
  import type { OpenAI as OpenAIClient } from 'openai';
16
+ import type { GoogleGenerativeAIChatInput } from '@langchain/google-genai';
15
17
  import type { ChatVertexAIInput } from '@langchain/google-vertexai';
16
18
  import type { ChatBedrockConverseInput } from '@langchain/aws';
17
19
  import type { ChatMistralAIInput } from '@langchain/mistralai';
@@ -29,8 +31,9 @@ export type MistralAIClientOptions = ChatMistralAIInput;
29
31
  export type VertexAIClientOptions = ChatVertexAIInput;
30
32
  export type BedrockClientOptions = BedrockChatFields;
31
33
  export type BedrockConverseClientOptions = ChatBedrockConverseInput;
34
+ export type GoogleClientOptions = GoogleGenerativeAIChatInput;
32
35
 
33
- export type ClientOptions = OpenAIClientOptions | OllamaClientOptions | AnthropicClientOptions | MistralAIClientOptions | VertexAIClientOptions | BedrockClientOptions | BedrockConverseClientOptions;
36
+ export type ClientOptions = OpenAIClientOptions | OllamaClientOptions | AnthropicClientOptions | MistralAIClientOptions | VertexAIClientOptions | BedrockClientOptions | BedrockConverseClientOptions | GoogleClientOptions;
34
37
 
35
38
  export type LLMConfig = {
36
39
  provider: Providers;
@@ -44,6 +47,7 @@ export type ProviderOptionsMap = {
44
47
  [Providers.VERTEXAI]: VertexAIClientOptions;
45
48
  [Providers.BEDROCK_LEGACY]: BedrockClientOptions;
46
49
  [Providers.BEDROCK]: BedrockConverseClientOptions;
50
+ [Providers.GOOGLE]: GoogleClientOptions;
47
51
  };
48
52
 
49
53
  export type ChatModelMap = {
@@ -54,6 +58,7 @@ export type ChatModelMap = {
54
58
  [Providers.VERTEXAI]: ChatVertexAI;
55
59
  [Providers.BEDROCK_LEGACY]: BedrockChat;
56
60
  [Providers.BEDROCK]: ChatBedrockConverse;
61
+ [Providers.GOOGLE]: ChatGoogleGenerativeAI;
57
62
  };
58
63
 
59
64
  export type ChatModelConstructorMap = {
@@ -1,2 +1,3 @@
1
1
  export * from './graph';
2
+ export * from './misc';
2
3
  export * from './run';
@@ -35,6 +35,12 @@ const llmConfigs: Record<string, t.LLMConfig | undefined> = {
35
35
  streaming: true,
36
36
  streamUsage: true,
37
37
  },
38
+ [Providers.GOOGLE]: {
39
+ provider: Providers.GOOGLE,
40
+ model: 'gemini-2.0-flash-exp',
41
+ streaming: true,
42
+ streamUsage: true,
43
+ },
38
44
  [Providers.BEDROCK]: {
39
45
  provider: Providers.BEDROCK,
40
46
  model: 'anthropic.claude-3-sonnet-20240229-v1:0',
@@ -0,0 +1,45 @@
1
+ /**
2
+ * Unescapes a c-escaped string
3
+ * @param str The string to unescape
4
+ * @returns The unescaped string
5
+ */
6
+ const unescapeString = (string: string): string => string.replace(/\\(.)/g, (_, char) => {
7
+ switch (char) {
8
+ case 'n':
9
+ return '\n';
10
+ case 't':
11
+ return '\t';
12
+ case 'r':
13
+ return '\r';
14
+ case '"':
15
+ return '"';
16
+ case '\'':
17
+ return '\'';
18
+ case '\\':
19
+ return '\\';
20
+ default:
21
+ return char;
22
+ }
23
+ });
24
+
25
+ /**
26
+ * Recursively unescapes all string values in an object
27
+ * @param obj The object to unescape
28
+ * @returns The unescaped object
29
+ */
30
+ export function unescapeObject(obj: unknown, key?: string): unknown {
31
+ if (typeof obj === 'string') {
32
+ let unescaped = unescapeString(obj);
33
+ if (key === 'filePath' && unescaped.match(/^"(.+)"$/)) {
34
+ unescaped = unescaped.substring(1, unescaped.length - 1);
35
+ }
36
+ return unescaped;
37
+ }
38
+ if (Array.isArray(obj)) {
39
+ return obj.map((value) => unescapeObject(value, key === 'contextPaths' ? 'filePath' : ''));
40
+ }
41
+ if (typeof obj === 'object' && obj !== null) {
42
+ return Object.fromEntries(Object.entries(obj).map(([key, value]) => [key, unescapeObject(value, key)]));
43
+ }
44
+ return obj;
45
+ }