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
@@ -0,0 +1,362 @@
1
+ import {
2
+ HumanMessage,
3
+ AIMessage,
4
+ SystemMessage,
5
+ } from '@langchain/core/messages';
6
+ import { formatContentStrings } from './content';
7
+ import { ContentTypes } from '@/common';
8
+
9
+ describe('formatContentStrings', () => {
10
+ describe('Human messages', () => {
11
+ it('should convert human message with all text blocks to string', () => {
12
+ const messages = [
13
+ new HumanMessage({
14
+ content: [
15
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Hello' },
16
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'World' },
17
+ ],
18
+ }),
19
+ ];
20
+
21
+ const result = formatContentStrings(messages);
22
+
23
+ expect(result).toHaveLength(1);
24
+ expect(result[0].content).toBe('Hello\nWorld');
25
+ });
26
+
27
+ it('should not convert human message with mixed content types (text + image)', () => {
28
+ const messages = [
29
+ new HumanMessage({
30
+ content: [
31
+ { type: ContentTypes.TEXT, text: 'what do you see' },
32
+ {
33
+ type: 'image_url',
34
+ image_url: {
35
+ url: '_SOME_BASE64_DATA=',
36
+ detail: 'auto',
37
+ },
38
+ },
39
+ ],
40
+ }),
41
+ ];
42
+
43
+ const result = formatContentStrings(messages);
44
+
45
+ expect(result).toHaveLength(1);
46
+ expect(result[0].content).toEqual([
47
+ { type: ContentTypes.TEXT, text: 'what do you see' },
48
+ {
49
+ type: 'image_url',
50
+ image_url: {
51
+ url: '_SOME_BASE64_DATA=',
52
+ detail: 'auto',
53
+ },
54
+ },
55
+ ]);
56
+ });
57
+
58
+ it('should leave string content unchanged', () => {
59
+ const messages = [
60
+ new HumanMessage({
61
+ content: 'Hello World',
62
+ }),
63
+ ];
64
+
65
+ const result = formatContentStrings(messages);
66
+
67
+ expect(result).toHaveLength(1);
68
+ expect(result[0].content).toBe('Hello World');
69
+ });
70
+
71
+ it('should handle empty text blocks', () => {
72
+ const messages = [
73
+ new HumanMessage({
74
+ content: [
75
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Hello' },
76
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: '' },
77
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'World' },
78
+ ],
79
+ }),
80
+ ];
81
+
82
+ const result = formatContentStrings(messages);
83
+
84
+ expect(result).toHaveLength(1);
85
+ expect(result[0].content).toBe('Hello\n\nWorld');
86
+ });
87
+
88
+ it('should handle null/undefined text values', () => {
89
+ const messages = [
90
+ new HumanMessage({
91
+ content: [
92
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Hello' },
93
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: null },
94
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: undefined },
95
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'World' },
96
+ ],
97
+ }),
98
+ ];
99
+
100
+ const result = formatContentStrings(messages);
101
+
102
+ expect(result).toHaveLength(1);
103
+ expect(result[0].content).toBe('Hello\n\n\nWorld');
104
+ });
105
+ });
106
+
107
+ describe('AI messages', () => {
108
+ it('should convert AI message with all text blocks to string', () => {
109
+ const messages = [
110
+ new AIMessage({
111
+ content: [
112
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Hello' },
113
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'World' },
114
+ ],
115
+ }),
116
+ ];
117
+
118
+ const result = formatContentStrings(messages);
119
+
120
+ expect(result).toHaveLength(1);
121
+ expect(result[0].content).toBe('Hello\nWorld');
122
+ expect(result[0].getType()).toBe('ai');
123
+ });
124
+
125
+ it('should not convert AI message with mixed content types', () => {
126
+ const messages = [
127
+ new AIMessage({
128
+ content: [
129
+ {
130
+ type: ContentTypes.TEXT,
131
+ [ContentTypes.TEXT]: 'Here is an image',
132
+ },
133
+ {
134
+ type: ContentTypes.TOOL_CALL,
135
+ tool_call: { name: 'generate_image' },
136
+ },
137
+ ],
138
+ }),
139
+ ];
140
+
141
+ const result = formatContentStrings(messages);
142
+
143
+ expect(result).toHaveLength(1);
144
+ expect(result[0].content).toEqual([
145
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Here is an image' },
146
+ { type: ContentTypes.TOOL_CALL, tool_call: { name: 'generate_image' } },
147
+ ]);
148
+ });
149
+ });
150
+
151
+ describe('System messages', () => {
152
+ it('should convert System message with all text blocks to string', () => {
153
+ const messages = [
154
+ new SystemMessage({
155
+ content: [
156
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'System' },
157
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Message' },
158
+ ],
159
+ }),
160
+ ];
161
+
162
+ const result = formatContentStrings(messages);
163
+
164
+ expect(result).toHaveLength(1);
165
+ expect(result[0].content).toBe('System\nMessage');
166
+ expect(result[0].getType()).toBe('system');
167
+ });
168
+ });
169
+
170
+ describe('Mixed message types', () => {
171
+ it('should process all valid message types in mixed array', () => {
172
+ const messages = [
173
+ new HumanMessage({
174
+ content: [
175
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Human' },
176
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Message' },
177
+ ],
178
+ }),
179
+ new AIMessage({
180
+ content: [
181
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'AI' },
182
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Response' },
183
+ ],
184
+ }),
185
+ new SystemMessage({
186
+ content: [
187
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'System' },
188
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: 'Prompt' },
189
+ ],
190
+ }),
191
+ ];
192
+
193
+ const result = formatContentStrings(messages);
194
+
195
+ expect(result).toHaveLength(3);
196
+ // All messages should be converted
197
+ expect(result[0].content).toBe('Human\nMessage');
198
+ expect(result[0].getType()).toBe('human');
199
+
200
+ expect(result[1].content).toBe('AI\nResponse');
201
+ expect(result[1].getType()).toBe('ai');
202
+
203
+ expect(result[2].content).toBe('System\nPrompt');
204
+ expect(result[2].getType()).toBe('system');
205
+ });
206
+ });
207
+
208
+ describe('Edge cases', () => {
209
+ it('should handle empty array', () => {
210
+ const result = formatContentStrings([]);
211
+ expect(result).toEqual([]);
212
+ });
213
+
214
+ it('should handle messages with non-array content', () => {
215
+ const messages = [
216
+ new HumanMessage({
217
+ content: 'This is a string content',
218
+ }),
219
+ ];
220
+
221
+ const result = formatContentStrings(messages);
222
+
223
+ expect(result).toHaveLength(1);
224
+ expect(result[0].content).toBe('This is a string content');
225
+ });
226
+
227
+ it('should trim the final concatenated string', () => {
228
+ const messages = [
229
+ new HumanMessage({
230
+ content: [
231
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: ' Hello ' },
232
+ { type: ContentTypes.TEXT, [ContentTypes.TEXT]: ' World ' },
233
+ ],
234
+ }),
235
+ ];
236
+
237
+ const result = formatContentStrings(messages);
238
+
239
+ expect(result).toHaveLength(1);
240
+ expect(result[0].content).toBe('Hello \n World');
241
+ });
242
+ });
243
+
244
+ describe('Real-world scenarios', () => {
245
+ it('should handle the exact scenario from the issue', () => {
246
+ const messages = [
247
+ new HumanMessage({
248
+ content: [
249
+ {
250
+ type: 'text',
251
+ text: 'hi there',
252
+ },
253
+ ],
254
+ }),
255
+ new AIMessage({
256
+ content: [
257
+ {
258
+ type: 'text',
259
+ text: 'Hi Danny! How can I help you today?',
260
+ },
261
+ ],
262
+ }),
263
+ new HumanMessage({
264
+ content: [
265
+ {
266
+ type: 'text',
267
+ text: 'what do you see',
268
+ },
269
+ {
270
+ type: 'image_url',
271
+ image_url: {
272
+ url: '_SOME_BASE64_DATA=',
273
+ detail: 'auto',
274
+ },
275
+ },
276
+ ],
277
+ }),
278
+ ];
279
+
280
+ const result = formatContentStrings(messages);
281
+
282
+ expect(result).toHaveLength(3);
283
+
284
+ // First human message (all text) should be converted
285
+ expect(result[0].content).toBe('hi there');
286
+ expect(result[0].getType()).toBe('human');
287
+
288
+ // AI message (all text) should now also be converted
289
+ expect(result[1].content).toBe('Hi Danny! How can I help you today?');
290
+ expect(result[1].getType()).toBe('ai');
291
+
292
+ // Third message (mixed content) should remain unchanged
293
+ expect(result[2].content).toEqual([
294
+ {
295
+ type: 'text',
296
+ text: 'what do you see',
297
+ },
298
+ {
299
+ type: 'image_url',
300
+ image_url: {
301
+ url: '_SOME_BASE64_DATA=',
302
+ detail: 'auto',
303
+ },
304
+ },
305
+ ]);
306
+ });
307
+
308
+ it('should handle messages with tool calls', () => {
309
+ const messages = [
310
+ new HumanMessage({
311
+ content: [
312
+ {
313
+ type: ContentTypes.TEXT,
314
+ [ContentTypes.TEXT]: 'Please use the calculator',
315
+ },
316
+ {
317
+ type: ContentTypes.TOOL_CALL,
318
+ tool_call: { name: 'calculator', args: '{"a": 1, "b": 2}' },
319
+ },
320
+ ],
321
+ }),
322
+ new AIMessage({
323
+ content: [
324
+ {
325
+ type: ContentTypes.TEXT,
326
+ [ContentTypes.TEXT]: 'I will calculate that for you',
327
+ },
328
+ {
329
+ type: ContentTypes.TOOL_CALL,
330
+ tool_call: { name: 'calculator', args: '{"a": 1, "b": 2}' },
331
+ },
332
+ ],
333
+ }),
334
+ ];
335
+
336
+ const result = formatContentStrings(messages);
337
+
338
+ expect(result).toHaveLength(2);
339
+ // Should not convert because not all blocks are text
340
+ expect(result[0].content).toEqual([
341
+ {
342
+ type: ContentTypes.TEXT,
343
+ [ContentTypes.TEXT]: 'Please use the calculator',
344
+ },
345
+ {
346
+ type: ContentTypes.TOOL_CALL,
347
+ tool_call: { name: 'calculator', args: '{"a": 1, "b": 2}' },
348
+ },
349
+ ]);
350
+ expect(result[1].content).toEqual([
351
+ {
352
+ type: ContentTypes.TEXT,
353
+ [ContentTypes.TEXT]: 'I will calculate that for you',
354
+ },
355
+ {
356
+ type: ContentTypes.TOOL_CALL,
357
+ tool_call: { name: 'calculator', args: '{"a": 1, "b": 2}' },
358
+ },
359
+ ]);
360
+ });
361
+ });
362
+ });
@@ -0,0 +1,63 @@
1
+ import { ContentTypes } from '@/common';
2
+ import type { BaseMessage } from '@langchain/core/messages';
3
+
4
+ /**
5
+ * Formats an array of messages for LangChain, making sure all content fields are strings
6
+ * @param {Array<HumanMessage | AIMessage | SystemMessage | ToolMessage>} payload - The array of messages to format.
7
+ * @returns {Array<HumanMessage | AIMessage | SystemMessage | ToolMessage>} - The array of formatted LangChain messages, including ToolMessages for tool calls.
8
+ */
9
+ export const formatContentStrings = (
10
+ payload: Array<BaseMessage>
11
+ ): Array<BaseMessage> => {
12
+ // Create a new array to store the processed messages
13
+ const result: Array<BaseMessage> = [];
14
+
15
+ for (const message of payload) {
16
+ const messageType = message.getType();
17
+ const isValidMessage =
18
+ messageType === 'human' ||
19
+ messageType === 'ai' ||
20
+ messageType === 'system';
21
+
22
+ if (!isValidMessage) {
23
+ result.push(message);
24
+ continue;
25
+ }
26
+
27
+ // If content is already a string, add as-is
28
+ if (typeof message.content === 'string') {
29
+ result.push(message);
30
+ continue;
31
+ }
32
+
33
+ // If content is not an array, add as-is
34
+ if (!Array.isArray(message.content)) {
35
+ result.push(message);
36
+ continue;
37
+ }
38
+
39
+ // Check if all content blocks are text type
40
+ const allTextBlocks = message.content.every(
41
+ (block) => block.type === ContentTypes.TEXT
42
+ );
43
+
44
+ // Only convert to string if all blocks are text type
45
+ if (!allTextBlocks) {
46
+ result.push(message);
47
+ continue;
48
+ }
49
+
50
+ // Reduce text types to a single string
51
+ const content = message.content.reduce((acc, curr) => {
52
+ if (curr.type === ContentTypes.TEXT) {
53
+ return `${acc}${curr[ContentTypes.TEXT] || ''}\n`;
54
+ }
55
+ return acc;
56
+ }, '');
57
+
58
+ message.content = content.trim();
59
+ result.push(message);
60
+ }
61
+
62
+ return result;
63
+ };