illuma-agents 1.0.2 → 1.0.3

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 (225) hide show
  1. package/LICENSE +25 -21
  2. package/dist/cjs/agents/AgentContext.cjs +222 -0
  3. package/dist/cjs/agents/AgentContext.cjs.map +1 -0
  4. package/dist/cjs/common/enum.cjs +7 -6
  5. package/dist/cjs/common/enum.cjs.map +1 -1
  6. package/dist/cjs/events.cjs +7 -5
  7. package/dist/cjs/events.cjs.map +1 -1
  8. package/dist/cjs/graphs/Graph.cjs +328 -207
  9. package/dist/cjs/graphs/Graph.cjs.map +1 -1
  10. package/dist/cjs/graphs/MultiAgentGraph.cjs +507 -0
  11. package/dist/cjs/graphs/MultiAgentGraph.cjs.map +1 -0
  12. package/dist/cjs/llm/anthropic/index.cjs.map +1 -1
  13. package/dist/cjs/llm/google/index.cjs.map +1 -1
  14. package/dist/cjs/llm/ollama/index.cjs.map +1 -1
  15. package/dist/cjs/llm/openai/index.cjs +35 -0
  16. package/dist/cjs/llm/openai/index.cjs.map +1 -1
  17. package/dist/cjs/llm/openai/utils/index.cjs +3 -1
  18. package/dist/cjs/llm/openai/utils/index.cjs.map +1 -1
  19. package/dist/cjs/llm/openrouter/index.cjs.map +1 -1
  20. package/dist/cjs/llm/providers.cjs +0 -2
  21. package/dist/cjs/llm/providers.cjs.map +1 -1
  22. package/dist/cjs/llm/vertexai/index.cjs.map +1 -1
  23. package/dist/cjs/main.cjs +12 -1
  24. package/dist/cjs/main.cjs.map +1 -1
  25. package/dist/cjs/messages/cache.cjs +123 -0
  26. package/dist/cjs/messages/cache.cjs.map +1 -0
  27. package/dist/cjs/messages/content.cjs +53 -0
  28. package/dist/cjs/messages/content.cjs.map +1 -0
  29. package/dist/cjs/messages/format.cjs +17 -29
  30. package/dist/cjs/messages/format.cjs.map +1 -1
  31. package/dist/cjs/run.cjs +119 -74
  32. package/dist/cjs/run.cjs.map +1 -1
  33. package/dist/cjs/stream.cjs +77 -73
  34. package/dist/cjs/stream.cjs.map +1 -1
  35. package/dist/cjs/tools/Calculator.cjs +45 -0
  36. package/dist/cjs/tools/Calculator.cjs.map +1 -0
  37. package/dist/cjs/tools/CodeExecutor.cjs +22 -22
  38. package/dist/cjs/tools/CodeExecutor.cjs.map +1 -1
  39. package/dist/cjs/tools/ToolNode.cjs +5 -3
  40. package/dist/cjs/tools/ToolNode.cjs.map +1 -1
  41. package/dist/cjs/tools/handlers.cjs +20 -20
  42. package/dist/cjs/tools/handlers.cjs.map +1 -1
  43. package/dist/cjs/utils/events.cjs +31 -0
  44. package/dist/cjs/utils/events.cjs.map +1 -0
  45. package/dist/cjs/utils/handlers.cjs +70 -0
  46. package/dist/cjs/utils/handlers.cjs.map +1 -0
  47. package/dist/cjs/utils/tokens.cjs +54 -7
  48. package/dist/cjs/utils/tokens.cjs.map +1 -1
  49. package/dist/esm/agents/AgentContext.mjs +220 -0
  50. package/dist/esm/agents/AgentContext.mjs.map +1 -0
  51. package/dist/esm/common/enum.mjs +7 -6
  52. package/dist/esm/common/enum.mjs.map +1 -1
  53. package/dist/esm/events.mjs +7 -5
  54. package/dist/esm/events.mjs.map +1 -1
  55. package/dist/esm/graphs/Graph.mjs +330 -209
  56. package/dist/esm/graphs/Graph.mjs.map +1 -1
  57. package/dist/esm/graphs/MultiAgentGraph.mjs +505 -0
  58. package/dist/esm/graphs/MultiAgentGraph.mjs.map +1 -0
  59. package/dist/esm/llm/anthropic/index.mjs.map +1 -1
  60. package/dist/esm/llm/google/index.mjs.map +1 -1
  61. package/dist/esm/llm/ollama/index.mjs.map +1 -1
  62. package/dist/esm/llm/openai/index.mjs +35 -0
  63. package/dist/esm/llm/openai/index.mjs.map +1 -1
  64. package/dist/esm/llm/openai/utils/index.mjs +3 -1
  65. package/dist/esm/llm/openai/utils/index.mjs.map +1 -1
  66. package/dist/esm/llm/openrouter/index.mjs.map +1 -1
  67. package/dist/esm/llm/providers.mjs +0 -2
  68. package/dist/esm/llm/providers.mjs.map +1 -1
  69. package/dist/esm/llm/vertexai/index.mjs.map +1 -1
  70. package/dist/esm/main.mjs +7 -2
  71. package/dist/esm/main.mjs.map +1 -1
  72. package/dist/esm/messages/cache.mjs +120 -0
  73. package/dist/esm/messages/cache.mjs.map +1 -0
  74. package/dist/esm/messages/content.mjs +51 -0
  75. package/dist/esm/messages/content.mjs.map +1 -0
  76. package/dist/esm/messages/format.mjs +18 -29
  77. package/dist/esm/messages/format.mjs.map +1 -1
  78. package/dist/esm/run.mjs +119 -74
  79. package/dist/esm/run.mjs.map +1 -1
  80. package/dist/esm/stream.mjs +77 -73
  81. package/dist/esm/stream.mjs.map +1 -1
  82. package/dist/esm/tools/Calculator.mjs +24 -0
  83. package/dist/esm/tools/Calculator.mjs.map +1 -0
  84. package/dist/esm/tools/CodeExecutor.mjs +22 -22
  85. package/dist/esm/tools/CodeExecutor.mjs.map +1 -1
  86. package/dist/esm/tools/ToolNode.mjs +5 -3
  87. package/dist/esm/tools/ToolNode.mjs.map +1 -1
  88. package/dist/esm/tools/handlers.mjs +20 -20
  89. package/dist/esm/tools/handlers.mjs.map +1 -1
  90. package/dist/esm/utils/events.mjs +29 -0
  91. package/dist/esm/utils/events.mjs.map +1 -0
  92. package/dist/esm/utils/handlers.mjs +68 -0
  93. package/dist/esm/utils/handlers.mjs.map +1 -0
  94. package/dist/esm/utils/tokens.mjs +54 -8
  95. package/dist/esm/utils/tokens.mjs.map +1 -1
  96. package/dist/types/agents/AgentContext.d.ts +94 -0
  97. package/dist/types/common/enum.d.ts +9 -7
  98. package/dist/types/events.d.ts +3 -3
  99. package/dist/types/graphs/Graph.d.ts +60 -66
  100. package/dist/types/graphs/MultiAgentGraph.d.ts +47 -0
  101. package/dist/types/graphs/index.d.ts +1 -0
  102. package/dist/types/index.d.ts +1 -0
  103. package/dist/types/llm/openai/index.d.ts +10 -0
  104. package/dist/types/messages/cache.d.ts +20 -0
  105. package/dist/types/messages/content.d.ts +7 -0
  106. package/dist/types/messages/format.d.ts +1 -7
  107. package/dist/types/messages/index.d.ts +2 -0
  108. package/dist/types/messages/reducer.d.ts +9 -0
  109. package/dist/types/run.d.ts +16 -10
  110. package/dist/types/stream.d.ts +4 -3
  111. package/dist/types/tools/Calculator.d.ts +8 -0
  112. package/dist/types/tools/ToolNode.d.ts +1 -1
  113. package/dist/types/tools/handlers.d.ts +9 -7
  114. package/dist/types/tools/search/tool.d.ts +4 -4
  115. package/dist/types/types/graph.d.ts +124 -11
  116. package/dist/types/types/llm.d.ts +13 -9
  117. package/dist/types/types/messages.d.ts +4 -0
  118. package/dist/types/types/run.d.ts +46 -8
  119. package/dist/types/types/stream.d.ts +3 -2
  120. package/dist/types/utils/events.d.ts +6 -0
  121. package/dist/types/utils/handlers.d.ts +34 -0
  122. package/dist/types/utils/index.d.ts +1 -0
  123. package/dist/types/utils/tokens.d.ts +24 -0
  124. package/package.json +162 -145
  125. package/src/agents/AgentContext.ts +323 -0
  126. package/src/common/enum.ts +177 -176
  127. package/src/events.ts +197 -191
  128. package/src/graphs/Graph.ts +1058 -846
  129. package/src/graphs/MultiAgentGraph.ts +598 -0
  130. package/src/graphs/index.ts +2 -1
  131. package/src/index.ts +25 -24
  132. package/src/llm/anthropic/index.ts +413 -413
  133. package/src/llm/google/index.ts +222 -222
  134. package/src/llm/google/utils/zod_to_genai_parameters.ts +86 -88
  135. package/src/llm/ollama/index.ts +92 -92
  136. package/src/llm/openai/index.ts +894 -853
  137. package/src/llm/openai/utils/index.ts +920 -918
  138. package/src/llm/openrouter/index.ts +60 -60
  139. package/src/llm/providers.ts +55 -57
  140. package/src/llm/vertexai/index.ts +360 -360
  141. package/src/messages/cache.test.ts +461 -0
  142. package/src/messages/cache.ts +151 -0
  143. package/src/messages/content.test.ts +362 -0
  144. package/src/messages/content.ts +63 -0
  145. package/src/messages/format.ts +611 -625
  146. package/src/messages/formatAgentMessages.test.ts +1144 -917
  147. package/src/messages/index.ts +6 -4
  148. package/src/messages/reducer.ts +80 -0
  149. package/src/run.ts +447 -381
  150. package/src/scripts/abort.ts +157 -138
  151. package/src/scripts/ant_web_search.ts +158 -158
  152. package/src/scripts/cli.ts +172 -167
  153. package/src/scripts/cli2.ts +133 -125
  154. package/src/scripts/cli3.ts +184 -178
  155. package/src/scripts/cli4.ts +191 -184
  156. package/src/scripts/cli5.ts +191 -184
  157. package/src/scripts/code_exec.ts +213 -214
  158. package/src/scripts/code_exec_simple.ts +147 -129
  159. package/src/scripts/content.ts +138 -120
  160. package/src/scripts/handoff-test.ts +135 -0
  161. package/src/scripts/multi-agent-chain.ts +278 -0
  162. package/src/scripts/multi-agent-conditional.ts +220 -0
  163. package/src/scripts/multi-agent-document-review-chain.ts +197 -0
  164. package/src/scripts/multi-agent-hybrid-flow.ts +310 -0
  165. package/src/scripts/multi-agent-parallel.ts +343 -0
  166. package/src/scripts/multi-agent-sequence.ts +212 -0
  167. package/src/scripts/multi-agent-supervisor.ts +364 -0
  168. package/src/scripts/multi-agent-test.ts +186 -0
  169. package/src/scripts/search.ts +146 -150
  170. package/src/scripts/simple.ts +225 -225
  171. package/src/scripts/stream.ts +140 -122
  172. package/src/scripts/test-custom-prompt-key.ts +145 -0
  173. package/src/scripts/test-handoff-input.ts +170 -0
  174. package/src/scripts/test-multi-agent-list-handoff.ts +261 -0
  175. package/src/scripts/test-tools-before-handoff.ts +222 -0
  176. package/src/scripts/tools.ts +153 -155
  177. package/src/specs/agent-handoffs.test.ts +889 -0
  178. package/src/specs/anthropic.simple.test.ts +320 -317
  179. package/src/specs/azure.simple.test.ts +325 -316
  180. package/src/specs/openai.simple.test.ts +311 -316
  181. package/src/specs/openrouter.simple.test.ts +107 -0
  182. package/src/specs/prune.test.ts +758 -763
  183. package/src/specs/reasoning.test.ts +201 -165
  184. package/src/specs/thinking-prune.test.ts +769 -703
  185. package/src/specs/token-memoization.test.ts +39 -0
  186. package/src/stream.ts +664 -651
  187. package/src/tools/Calculator.test.ts +278 -0
  188. package/src/tools/Calculator.ts +25 -0
  189. package/src/tools/CodeExecutor.ts +220 -220
  190. package/src/tools/ToolNode.ts +170 -170
  191. package/src/tools/handlers.ts +341 -336
  192. package/src/types/graph.ts +372 -185
  193. package/src/types/llm.ts +141 -140
  194. package/src/types/messages.ts +4 -0
  195. package/src/types/run.ts +128 -89
  196. package/src/types/stream.ts +401 -400
  197. package/src/utils/events.ts +32 -0
  198. package/src/utils/handlers.ts +107 -0
  199. package/src/utils/index.ts +6 -5
  200. package/src/utils/llmConfig.ts +183 -183
  201. package/src/utils/tokens.ts +129 -70
  202. package/dist/types/scripts/abort.d.ts +0 -1
  203. package/dist/types/scripts/ant_web_search.d.ts +0 -1
  204. package/dist/types/scripts/args.d.ts +0 -7
  205. package/dist/types/scripts/caching.d.ts +0 -1
  206. package/dist/types/scripts/cli.d.ts +0 -1
  207. package/dist/types/scripts/cli2.d.ts +0 -1
  208. package/dist/types/scripts/cli3.d.ts +0 -1
  209. package/dist/types/scripts/cli4.d.ts +0 -1
  210. package/dist/types/scripts/cli5.d.ts +0 -1
  211. package/dist/types/scripts/code_exec.d.ts +0 -1
  212. package/dist/types/scripts/code_exec_files.d.ts +0 -1
  213. package/dist/types/scripts/code_exec_simple.d.ts +0 -1
  214. package/dist/types/scripts/content.d.ts +0 -1
  215. package/dist/types/scripts/empty_input.d.ts +0 -1
  216. package/dist/types/scripts/image.d.ts +0 -1
  217. package/dist/types/scripts/memory.d.ts +0 -1
  218. package/dist/types/scripts/search.d.ts +0 -1
  219. package/dist/types/scripts/simple.d.ts +0 -1
  220. package/dist/types/scripts/stream.d.ts +0 -1
  221. package/dist/types/scripts/thinking.d.ts +0 -1
  222. package/dist/types/scripts/tools.d.ts +0 -1
  223. package/dist/types/specs/spec.utils.d.ts +0 -1
  224. package/dist/types/tools/example.d.ts +0 -78
  225. package/src/tools/example.ts +0 -129
@@ -1,222 +1,222 @@
1
- /* eslint-disable @typescript-eslint/ban-ts-comment */
2
- import { AIMessageChunk } from '@langchain/core/messages';
3
- import { ChatGenerationChunk } from '@langchain/core/outputs';
4
- import { ChatGoogleGenerativeAI } from '@langchain/google-genai';
5
- import { getEnvironmentVariable } from '@langchain/core/utils/env';
6
- import { GoogleGenerativeAI as GenerativeAI } from '@google/generative-ai';
7
- import type {
8
- GenerateContentRequest,
9
- SafetySetting,
10
- } from '@google/generative-ai';
11
- import type { CallbackManagerForLLMRun } from '@langchain/core/callbacks/manager';
12
- import type { BaseMessage, UsageMetadata } from '@langchain/core/messages';
13
- import type { GeminiGenerationConfig } from '@langchain/google-common';
14
- import type { GeminiApiUsageMetadata } from './types';
15
- import type { GoogleClientOptions } from '@/types';
16
- import {
17
- convertResponseContentToChatGenerationChunk,
18
- convertBaseMessagesToContent,
19
- } from './utils/common';
20
-
21
- export class CustomChatGoogleGenerativeAI extends ChatGoogleGenerativeAI {
22
- thinkingConfig?: GeminiGenerationConfig['thinkingConfig'];
23
- constructor(fields: GoogleClientOptions) {
24
- super(fields);
25
-
26
- this.model = fields.model.replace(/^models\//, '');
27
-
28
- this.maxOutputTokens = fields.maxOutputTokens ?? this.maxOutputTokens;
29
-
30
- if (this.maxOutputTokens != null && this.maxOutputTokens < 0) {
31
- throw new Error('`maxOutputTokens` must be a positive integer');
32
- }
33
-
34
- this.temperature = fields.temperature ?? this.temperature;
35
- if (
36
- this.temperature != null &&
37
- (this.temperature < 0 || this.temperature > 2)
38
- ) {
39
- throw new Error('`temperature` must be in the range of [0.0,2.0]');
40
- }
41
-
42
- this.topP = fields.topP ?? this.topP;
43
- if (this.topP != null && this.topP < 0) {
44
- throw new Error('`topP` must be a positive integer');
45
- }
46
-
47
- if (this.topP != null && this.topP > 1) {
48
- throw new Error('`topP` must be below 1.');
49
- }
50
-
51
- this.topK = fields.topK ?? this.topK;
52
- if (this.topK != null && this.topK < 0) {
53
- throw new Error('`topK` must be a positive integer');
54
- }
55
-
56
- this.stopSequences = fields.stopSequences ?? this.stopSequences;
57
-
58
- this.apiKey = fields.apiKey ?? getEnvironmentVariable('GOOGLE_API_KEY');
59
- if (this.apiKey == null || this.apiKey === '') {
60
- throw new Error(
61
- 'Please set an API key for Google GenerativeAI ' +
62
- 'in the environment variable GOOGLE_API_KEY ' +
63
- 'or in the `apiKey` field of the ' +
64
- 'ChatGoogleGenerativeAI constructor'
65
- );
66
- }
67
-
68
- this.safetySettings = fields.safetySettings ?? this.safetySettings;
69
- if (this.safetySettings && this.safetySettings.length > 0) {
70
- const safetySettingsSet = new Set(
71
- this.safetySettings.map((s) => s.category)
72
- );
73
- if (safetySettingsSet.size !== this.safetySettings.length) {
74
- throw new Error(
75
- 'The categories in `safetySettings` array must be unique'
76
- );
77
- }
78
- }
79
-
80
- this.thinkingConfig = fields.thinkingConfig ?? this.thinkingConfig;
81
-
82
- this.streaming = fields.streaming ?? this.streaming;
83
- this.json = fields.json;
84
-
85
- // @ts-ignore - Accessing private property from parent class
86
- this.client = new GenerativeAI(this.apiKey).getGenerativeModel(
87
- {
88
- model: this.model,
89
- safetySettings: this.safetySettings as SafetySetting[],
90
- generationConfig: {
91
- stopSequences: this.stopSequences,
92
- maxOutputTokens: this.maxOutputTokens,
93
- temperature: this.temperature,
94
- topP: this.topP,
95
- topK: this.topK,
96
- ...(this.json != null
97
- ? { responseMimeType: 'application/json' }
98
- : {}),
99
- },
100
- },
101
- {
102
- apiVersion: fields.apiVersion,
103
- baseUrl: fields.baseUrl,
104
- customHeaders: fields.customHeaders,
105
- }
106
- );
107
- this.streamUsage = fields.streamUsage ?? this.streamUsage;
108
- }
109
-
110
- static lc_name(): 'IllumaGoogleGenerativeAI' {
111
- return 'IllumaGoogleGenerativeAI';
112
- }
113
-
114
- invocationParams(
115
- options?: this['ParsedCallOptions']
116
- ): Omit<GenerateContentRequest, 'contents'> {
117
- const params = super.invocationParams(options);
118
- if (this.thinkingConfig) {
119
- /** @ts-ignore */
120
- this.client.generationConfig = {
121
- /** @ts-ignore */
122
- ...this.client.generationConfig,
123
- /** @ts-ignore */
124
- thinkingConfig: this.thinkingConfig,
125
- };
126
- }
127
- return params;
128
- }
129
-
130
- async *_streamResponseChunks(
131
- messages: BaseMessage[],
132
- options: this['ParsedCallOptions'],
133
- runManager?: CallbackManagerForLLMRun
134
- ): AsyncGenerator<ChatGenerationChunk> {
135
- const prompt = convertBaseMessagesToContent(
136
- messages,
137
- this._isMultimodalModel,
138
- this.useSystemInstruction
139
- );
140
- let actualPrompt = prompt;
141
- if (prompt?.[0].role === 'system') {
142
- const [systemInstruction] = prompt;
143
- /** @ts-ignore */
144
- this.client.systemInstruction = systemInstruction;
145
- actualPrompt = prompt.slice(1);
146
- }
147
- const parameters = this.invocationParams(options);
148
- const request = {
149
- ...parameters,
150
- contents: actualPrompt,
151
- };
152
- const stream = await this.caller.callWithOptions(
153
- { signal: options.signal },
154
- async () => {
155
- /** @ts-ignore */
156
- const { stream } = await this.client.generateContentStream(request);
157
- return stream;
158
- }
159
- );
160
-
161
- let index = 0;
162
- let lastUsageMetadata: UsageMetadata | undefined;
163
- for await (const response of stream) {
164
- if (
165
- 'usageMetadata' in response &&
166
- this.streamUsage !== false &&
167
- options.streamUsage !== false
168
- ) {
169
- const genAIUsageMetadata = response.usageMetadata as
170
- | GeminiApiUsageMetadata
171
- | undefined;
172
-
173
- const output_tokens =
174
- (genAIUsageMetadata?.candidatesTokenCount ?? 0) +
175
- (genAIUsageMetadata?.thoughtsTokenCount ?? 0);
176
- lastUsageMetadata = {
177
- input_tokens: genAIUsageMetadata?.promptTokenCount ?? 0,
178
- output_tokens,
179
- total_tokens: genAIUsageMetadata?.totalTokenCount ?? 0,
180
- };
181
- }
182
-
183
- const chunk = convertResponseContentToChatGenerationChunk(response, {
184
- usageMetadata: undefined,
185
- index,
186
- });
187
- index += 1;
188
- if (!chunk) {
189
- continue;
190
- }
191
-
192
- yield chunk;
193
- await runManager?.handleLLMNewToken(
194
- chunk.text || '',
195
- undefined,
196
- undefined,
197
- undefined,
198
- undefined,
199
- { chunk }
200
- );
201
- }
202
-
203
- if (lastUsageMetadata) {
204
- const finalChunk = new ChatGenerationChunk({
205
- text: '',
206
- message: new AIMessageChunk({
207
- content: '',
208
- usage_metadata: lastUsageMetadata,
209
- }),
210
- });
211
- yield finalChunk;
212
- await runManager?.handleLLMNewToken(
213
- finalChunk.text || '',
214
- undefined,
215
- undefined,
216
- undefined,
217
- undefined,
218
- { chunk: finalChunk }
219
- );
220
- }
221
- }
222
- }
1
+ /* eslint-disable @typescript-eslint/ban-ts-comment */
2
+ import { AIMessageChunk } from '@langchain/core/messages';
3
+ import { ChatGenerationChunk } from '@langchain/core/outputs';
4
+ import { ChatGoogleGenerativeAI } from '@langchain/google-genai';
5
+ import { getEnvironmentVariable } from '@langchain/core/utils/env';
6
+ import { GoogleGenerativeAI as GenerativeAI } from '@google/generative-ai';
7
+ import type {
8
+ GenerateContentRequest,
9
+ SafetySetting,
10
+ } from '@google/generative-ai';
11
+ import type { CallbackManagerForLLMRun } from '@langchain/core/callbacks/manager';
12
+ import type { BaseMessage, UsageMetadata } from '@langchain/core/messages';
13
+ import type { GeminiGenerationConfig } from '@langchain/google-common';
14
+ import type { GeminiApiUsageMetadata } from './types';
15
+ import type { GoogleClientOptions } from '@/types';
16
+ import {
17
+ convertResponseContentToChatGenerationChunk,
18
+ convertBaseMessagesToContent,
19
+ } from './utils/common';
20
+
21
+ export class CustomChatGoogleGenerativeAI extends ChatGoogleGenerativeAI {
22
+ thinkingConfig?: GeminiGenerationConfig['thinkingConfig'];
23
+ constructor(fields: GoogleClientOptions) {
24
+ super(fields);
25
+
26
+ this.model = fields.model.replace(/^models\//, '');
27
+
28
+ this.maxOutputTokens = fields.maxOutputTokens ?? this.maxOutputTokens;
29
+
30
+ if (this.maxOutputTokens != null && this.maxOutputTokens < 0) {
31
+ throw new Error('`maxOutputTokens` must be a positive integer');
32
+ }
33
+
34
+ this.temperature = fields.temperature ?? this.temperature;
35
+ if (
36
+ this.temperature != null &&
37
+ (this.temperature < 0 || this.temperature > 2)
38
+ ) {
39
+ throw new Error('`temperature` must be in the range of [0.0,2.0]');
40
+ }
41
+
42
+ this.topP = fields.topP ?? this.topP;
43
+ if (this.topP != null && this.topP < 0) {
44
+ throw new Error('`topP` must be a positive integer');
45
+ }
46
+
47
+ if (this.topP != null && this.topP > 1) {
48
+ throw new Error('`topP` must be below 1.');
49
+ }
50
+
51
+ this.topK = fields.topK ?? this.topK;
52
+ if (this.topK != null && this.topK < 0) {
53
+ throw new Error('`topK` must be a positive integer');
54
+ }
55
+
56
+ this.stopSequences = fields.stopSequences ?? this.stopSequences;
57
+
58
+ this.apiKey = fields.apiKey ?? getEnvironmentVariable('GOOGLE_API_KEY');
59
+ if (this.apiKey == null || this.apiKey === '') {
60
+ throw new Error(
61
+ 'Please set an API key for Google GenerativeAI ' +
62
+ 'in the environment variable GOOGLE_API_KEY ' +
63
+ 'or in the `apiKey` field of the ' +
64
+ 'ChatGoogleGenerativeAI constructor'
65
+ );
66
+ }
67
+
68
+ this.safetySettings = fields.safetySettings ?? this.safetySettings;
69
+ if (this.safetySettings && this.safetySettings.length > 0) {
70
+ const safetySettingsSet = new Set(
71
+ this.safetySettings.map((s) => s.category)
72
+ );
73
+ if (safetySettingsSet.size !== this.safetySettings.length) {
74
+ throw new Error(
75
+ 'The categories in `safetySettings` array must be unique'
76
+ );
77
+ }
78
+ }
79
+
80
+ this.thinkingConfig = fields.thinkingConfig ?? this.thinkingConfig;
81
+
82
+ this.streaming = fields.streaming ?? this.streaming;
83
+ this.json = fields.json;
84
+
85
+ // @ts-ignore - Accessing private property from parent class
86
+ this.client = new GenerativeAI(this.apiKey).getGenerativeModel(
87
+ {
88
+ model: this.model,
89
+ safetySettings: this.safetySettings as SafetySetting[],
90
+ generationConfig: {
91
+ stopSequences: this.stopSequences,
92
+ maxOutputTokens: this.maxOutputTokens,
93
+ temperature: this.temperature,
94
+ topP: this.topP,
95
+ topK: this.topK,
96
+ ...(this.json != null
97
+ ? { responseMimeType: 'application/json' }
98
+ : {}),
99
+ },
100
+ },
101
+ {
102
+ apiVersion: fields.apiVersion,
103
+ baseUrl: fields.baseUrl,
104
+ customHeaders: fields.customHeaders,
105
+ }
106
+ );
107
+ this.streamUsage = fields.streamUsage ?? this.streamUsage;
108
+ }
109
+
110
+ static lc_name(): 'IllumaGoogleGenerativeAI' {
111
+ return 'IllumaGoogleGenerativeAI';
112
+ }
113
+
114
+ invocationParams(
115
+ options?: this['ParsedCallOptions']
116
+ ): Omit<GenerateContentRequest, 'contents'> {
117
+ const params = super.invocationParams(options);
118
+ if (this.thinkingConfig) {
119
+ /** @ts-ignore */
120
+ this.client.generationConfig = {
121
+ /** @ts-ignore */
122
+ ...this.client.generationConfig,
123
+ /** @ts-ignore */
124
+ thinkingConfig: this.thinkingConfig,
125
+ };
126
+ }
127
+ return params;
128
+ }
129
+
130
+ async *_streamResponseChunks(
131
+ messages: BaseMessage[],
132
+ options: this['ParsedCallOptions'],
133
+ runManager?: CallbackManagerForLLMRun
134
+ ): AsyncGenerator<ChatGenerationChunk> {
135
+ const prompt = convertBaseMessagesToContent(
136
+ messages,
137
+ this._isMultimodalModel,
138
+ this.useSystemInstruction
139
+ );
140
+ let actualPrompt = prompt;
141
+ if (prompt?.[0].role === 'system') {
142
+ const [systemInstruction] = prompt;
143
+ /** @ts-ignore */
144
+ this.client.systemInstruction = systemInstruction;
145
+ actualPrompt = prompt.slice(1);
146
+ }
147
+ const parameters = this.invocationParams(options);
148
+ const request = {
149
+ ...parameters,
150
+ contents: actualPrompt,
151
+ };
152
+ const stream = await this.caller.callWithOptions(
153
+ { signal: options.signal },
154
+ async () => {
155
+ /** @ts-ignore */
156
+ const { stream } = await this.client.generateContentStream(request);
157
+ return stream;
158
+ }
159
+ );
160
+
161
+ let index = 0;
162
+ let lastUsageMetadata: UsageMetadata | undefined;
163
+ for await (const response of stream) {
164
+ if (
165
+ 'usageMetadata' in response &&
166
+ this.streamUsage !== false &&
167
+ options.streamUsage !== false
168
+ ) {
169
+ const genAIUsageMetadata = response.usageMetadata as
170
+ | GeminiApiUsageMetadata
171
+ | undefined;
172
+
173
+ const output_tokens =
174
+ (genAIUsageMetadata?.candidatesTokenCount ?? 0) +
175
+ (genAIUsageMetadata?.thoughtsTokenCount ?? 0);
176
+ lastUsageMetadata = {
177
+ input_tokens: genAIUsageMetadata?.promptTokenCount ?? 0,
178
+ output_tokens,
179
+ total_tokens: genAIUsageMetadata?.totalTokenCount ?? 0,
180
+ };
181
+ }
182
+
183
+ const chunk = convertResponseContentToChatGenerationChunk(response, {
184
+ usageMetadata: undefined,
185
+ index,
186
+ });
187
+ index += 1;
188
+ if (!chunk) {
189
+ continue;
190
+ }
191
+
192
+ yield chunk;
193
+ await runManager?.handleLLMNewToken(
194
+ chunk.text || '',
195
+ undefined,
196
+ undefined,
197
+ undefined,
198
+ undefined,
199
+ { chunk }
200
+ );
201
+ }
202
+
203
+ if (lastUsageMetadata) {
204
+ const finalChunk = new ChatGenerationChunk({
205
+ text: '',
206
+ message: new AIMessageChunk({
207
+ content: '',
208
+ usage_metadata: lastUsageMetadata,
209
+ }),
210
+ });
211
+ yield finalChunk;
212
+ await runManager?.handleLLMNewToken(
213
+ finalChunk.text || '',
214
+ undefined,
215
+ undefined,
216
+ undefined,
217
+ undefined,
218
+ { chunk: finalChunk }
219
+ );
220
+ }
221
+ }
222
+ }
@@ -1,88 +1,86 @@
1
- /* eslint-disable @typescript-eslint/no-unused-vars */
2
-
3
- import {
4
- type FunctionDeclarationSchema as GenerativeAIFunctionDeclarationSchema,
5
- type SchemaType as FunctionDeclarationSchemaType,
6
- } from '@google/generative-ai';
7
- import {
8
- InteropZodType,
9
- isInteropZodSchema,
10
- } from '@langchain/core/utils/types';
11
- import {
12
- type JsonSchema7Type,
13
- toJsonSchema,
14
- } from '@langchain/core/utils/json_schema';
15
-
16
- export interface GenerativeAIJsonSchema extends Record<string, unknown> {
17
- properties?: Record<string, GenerativeAIJsonSchema>;
18
- type: FunctionDeclarationSchemaType;
19
- }
20
-
21
- export interface GenerativeAIJsonSchemaDirty extends GenerativeAIJsonSchema {
22
- properties?: Record<string, GenerativeAIJsonSchemaDirty>;
23
- additionalProperties?: boolean;
24
- }
25
-
26
- export function removeAdditionalProperties(
27
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
28
- obj: Record<string, any>
29
- ): GenerativeAIJsonSchema {
30
- if (typeof obj === 'object' && obj !== null) {
31
- const newObj = { ...obj };
32
-
33
- if ('additionalProperties' in newObj) {
34
- delete newObj.additionalProperties;
35
- }
36
- if ('$schema' in newObj) {
37
- delete newObj.$schema;
38
- }
39
- if ('strict' in newObj) {
40
- delete newObj.strict;
41
- }
42
-
43
- for (const key in newObj) {
44
- if (key in newObj) {
45
- if (Array.isArray(newObj[key])) {
46
- newObj[key] = newObj[key].map(removeAdditionalProperties);
47
- } else if (typeof newObj[key] === 'object' && newObj[key] !== null) {
48
- newObj[key] = removeAdditionalProperties(newObj[key]);
49
- }
50
- }
51
- }
52
-
53
- return newObj as GenerativeAIJsonSchema;
54
- }
55
-
56
- return obj as GenerativeAIJsonSchema;
57
- }
58
-
59
- export function schemaToGenerativeAIParameters<
60
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
61
- RunOutput extends Record<string, any> = Record<string, any>,
62
- >(
63
- schema: InteropZodType<RunOutput> | JsonSchema7Type
64
- ): GenerativeAIFunctionDeclarationSchema {
65
- // GenerativeAI doesn't accept either the $schema or additionalProperties
66
- // attributes, so we need to explicitly remove them.
67
- const jsonSchema = removeAdditionalProperties(
68
- isInteropZodSchema(schema) ? toJsonSchema(schema) : schema
69
- );
70
- const { $schema, ...rest } = jsonSchema;
71
-
72
- return rest as GenerativeAIFunctionDeclarationSchema;
73
- }
74
-
75
- export function jsonSchemaToGeminiParameters(
76
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
77
- schema: Record<string, any>
78
- ): GenerativeAIFunctionDeclarationSchema {
79
- // Gemini doesn't accept either the $schema or additionalProperties
80
- // attributes, so we need to explicitly remove them.
81
-
82
- const jsonSchema = removeAdditionalProperties(
83
- schema as GenerativeAIJsonSchemaDirty
84
- );
85
- const { $schema, ...rest } = jsonSchema;
86
-
87
- return rest as GenerativeAIFunctionDeclarationSchema;
88
- }
1
+ import {
2
+ type FunctionDeclarationSchema as GenerativeAIFunctionDeclarationSchema,
3
+ type SchemaType as FunctionDeclarationSchemaType,
4
+ } from '@google/generative-ai';
5
+ import {
6
+ InteropZodType,
7
+ isInteropZodSchema,
8
+ } from '@langchain/core/utils/types';
9
+ import {
10
+ type JsonSchema7Type,
11
+ toJsonSchema,
12
+ } from '@langchain/core/utils/json_schema';
13
+
14
+ export interface GenerativeAIJsonSchema extends Record<string, unknown> {
15
+ properties?: Record<string, GenerativeAIJsonSchema>;
16
+ type: FunctionDeclarationSchemaType;
17
+ }
18
+
19
+ export interface GenerativeAIJsonSchemaDirty extends GenerativeAIJsonSchema {
20
+ properties?: Record<string, GenerativeAIJsonSchemaDirty>;
21
+ additionalProperties?: boolean;
22
+ }
23
+
24
+ export function removeAdditionalProperties(
25
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
26
+ obj: Record<string, any>
27
+ ): GenerativeAIJsonSchema {
28
+ if (typeof obj === 'object' && obj !== null) {
29
+ const newObj = { ...obj };
30
+
31
+ if ('additionalProperties' in newObj) {
32
+ delete newObj.additionalProperties;
33
+ }
34
+ if ('$schema' in newObj) {
35
+ delete newObj.$schema;
36
+ }
37
+ if ('strict' in newObj) {
38
+ delete newObj.strict;
39
+ }
40
+
41
+ for (const key in newObj) {
42
+ if (key in newObj) {
43
+ if (Array.isArray(newObj[key])) {
44
+ newObj[key] = newObj[key].map(removeAdditionalProperties);
45
+ } else if (typeof newObj[key] === 'object' && newObj[key] !== null) {
46
+ newObj[key] = removeAdditionalProperties(newObj[key]);
47
+ }
48
+ }
49
+ }
50
+
51
+ return newObj as GenerativeAIJsonSchema;
52
+ }
53
+
54
+ return obj as GenerativeAIJsonSchema;
55
+ }
56
+
57
+ export function schemaToGenerativeAIParameters<
58
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
59
+ RunOutput extends Record<string, any> = Record<string, any>,
60
+ >(
61
+ schema: InteropZodType<RunOutput> | JsonSchema7Type
62
+ ): GenerativeAIFunctionDeclarationSchema {
63
+ // GenerativeAI doesn't accept either the $schema or additionalProperties
64
+ // attributes, so we need to explicitly remove them.
65
+ const jsonSchema = removeAdditionalProperties(
66
+ isInteropZodSchema(schema) ? toJsonSchema(schema) : schema
67
+ );
68
+ const { $schema: _s, ...rest } = jsonSchema;
69
+
70
+ return rest as GenerativeAIFunctionDeclarationSchema;
71
+ }
72
+
73
+ export function jsonSchemaToGeminiParameters(
74
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
75
+ schema: Record<string, any>
76
+ ): GenerativeAIFunctionDeclarationSchema {
77
+ // Gemini doesn't accept either the $schema or additionalProperties
78
+ // attributes, so we need to explicitly remove them.
79
+
80
+ const jsonSchema = removeAdditionalProperties(
81
+ schema as GenerativeAIJsonSchemaDirty
82
+ );
83
+ const { $schema: _s, ...rest } = jsonSchema;
84
+
85
+ return rest as GenerativeAIFunctionDeclarationSchema;
86
+ }