@smythos/sre 1.5.53 → 1.5.54

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 (223) hide show
  1. package/CHANGELOG +98 -98
  2. package/LICENSE +18 -18
  3. package/README.md +135 -135
  4. package/dist/bundle-analysis-lazy.html +4949 -0
  5. package/dist/bundle-analysis.html +4949 -0
  6. package/dist/index.js +3 -3
  7. package/dist/index.js.map +1 -1
  8. package/dist/types/subsystems/LLMManager/LLM.service/connectors/openai/OpenAIConnector.class.d.ts +1 -6
  9. package/dist/types/utils/package-manager.utils.d.ts +26 -0
  10. package/package.json +1 -1
  11. package/src/Components/APICall/APICall.class.ts +157 -157
  12. package/src/Components/APICall/AccessTokenManager.ts +166 -166
  13. package/src/Components/APICall/ArrayBufferResponse.helper.ts +58 -58
  14. package/src/Components/APICall/OAuth.helper.ts +447 -447
  15. package/src/Components/APICall/mimeTypeCategories.ts +46 -46
  16. package/src/Components/APICall/parseData.ts +167 -167
  17. package/src/Components/APICall/parseHeaders.ts +41 -41
  18. package/src/Components/APICall/parseProxy.ts +68 -68
  19. package/src/Components/APICall/parseUrl.ts +91 -91
  20. package/src/Components/APIEndpoint.class.ts +234 -234
  21. package/src/Components/APIOutput.class.ts +58 -58
  22. package/src/Components/AgentPlugin.class.ts +102 -102
  23. package/src/Components/Async.class.ts +155 -155
  24. package/src/Components/Await.class.ts +90 -90
  25. package/src/Components/Classifier.class.ts +158 -158
  26. package/src/Components/Component.class.ts +132 -132
  27. package/src/Components/ComponentHost.class.ts +38 -38
  28. package/src/Components/DataSourceCleaner.class.ts +92 -92
  29. package/src/Components/DataSourceIndexer.class.ts +181 -181
  30. package/src/Components/DataSourceLookup.class.ts +161 -161
  31. package/src/Components/ECMASandbox.class.ts +71 -71
  32. package/src/Components/FEncDec.class.ts +29 -29
  33. package/src/Components/FHash.class.ts +33 -33
  34. package/src/Components/FSign.class.ts +80 -80
  35. package/src/Components/FSleep.class.ts +25 -25
  36. package/src/Components/FTimestamp.class.ts +25 -25
  37. package/src/Components/FileStore.class.ts +78 -78
  38. package/src/Components/ForEach.class.ts +97 -97
  39. package/src/Components/GPTPlugin.class.ts +70 -70
  40. package/src/Components/GenAILLM.class.ts +586 -586
  41. package/src/Components/HuggingFace.class.ts +314 -314
  42. package/src/Components/Image/imageSettings.config.ts +70 -70
  43. package/src/Components/ImageGenerator.class.ts +502 -502
  44. package/src/Components/JSONFilter.class.ts +54 -54
  45. package/src/Components/LLMAssistant.class.ts +213 -213
  46. package/src/Components/LogicAND.class.ts +28 -28
  47. package/src/Components/LogicAtLeast.class.ts +85 -85
  48. package/src/Components/LogicAtMost.class.ts +86 -86
  49. package/src/Components/LogicOR.class.ts +29 -29
  50. package/src/Components/LogicXOR.class.ts +34 -34
  51. package/src/Components/MCPClient.class.ts +138 -138
  52. package/src/Components/MemoryDeleteKeyVal.class.ts +70 -70
  53. package/src/Components/MemoryReadKeyVal.class.ts +66 -66
  54. package/src/Components/MemoryWriteKeyVal.class.ts +62 -62
  55. package/src/Components/MemoryWriteObject.class.ts +97 -97
  56. package/src/Components/MultimodalLLM.class.ts +128 -128
  57. package/src/Components/OpenAPI.class.ts +72 -72
  58. package/src/Components/PromptGenerator.class.ts +122 -122
  59. package/src/Components/ScrapflyWebScrape.class.ts +159 -159
  60. package/src/Components/ServerlessCode.class.ts +123 -123
  61. package/src/Components/TavilyWebSearch.class.ts +98 -98
  62. package/src/Components/VisionLLM.class.ts +104 -104
  63. package/src/Components/ZapierAction.class.ts +127 -127
  64. package/src/Components/index.ts +97 -97
  65. package/src/Core/AgentProcess.helper.ts +240 -240
  66. package/src/Core/Connector.class.ts +123 -123
  67. package/src/Core/ConnectorsService.ts +197 -197
  68. package/src/Core/DummyConnector.ts +49 -49
  69. package/src/Core/HookService.ts +105 -105
  70. package/src/Core/SmythRuntime.class.ts +235 -235
  71. package/src/Core/SystemEvents.ts +16 -16
  72. package/src/Core/boot.ts +56 -56
  73. package/src/config.ts +15 -15
  74. package/src/constants.ts +126 -126
  75. package/src/data/hugging-face.params.json +579 -579
  76. package/src/helpers/AWSLambdaCode.helper.ts +590 -590
  77. package/src/helpers/BinaryInput.helper.ts +331 -331
  78. package/src/helpers/Conversation.helper.ts +1119 -1119
  79. package/src/helpers/ECMASandbox.helper.ts +54 -54
  80. package/src/helpers/JsonContent.helper.ts +97 -97
  81. package/src/helpers/LocalCache.helper.ts +97 -97
  82. package/src/helpers/Log.helper.ts +274 -274
  83. package/src/helpers/OpenApiParser.helper.ts +150 -150
  84. package/src/helpers/S3Cache.helper.ts +147 -147
  85. package/src/helpers/SmythURI.helper.ts +5 -5
  86. package/src/helpers/Sysconfig.helper.ts +77 -77
  87. package/src/helpers/TemplateString.helper.ts +243 -243
  88. package/src/helpers/TypeChecker.helper.ts +329 -329
  89. package/src/index.ts +3 -3
  90. package/src/index.ts.bak +3 -3
  91. package/src/subsystems/AgentManager/Agent.class.ts +1114 -1114
  92. package/src/subsystems/AgentManager/Agent.helper.ts +3 -3
  93. package/src/subsystems/AgentManager/AgentData.service/AgentDataConnector.ts +230 -230
  94. package/src/subsystems/AgentManager/AgentData.service/connectors/CLIAgentDataConnector.class.ts +66 -66
  95. package/src/subsystems/AgentManager/AgentData.service/connectors/LocalAgentDataConnector.class.ts +142 -142
  96. package/src/subsystems/AgentManager/AgentData.service/connectors/NullAgentData.class.ts +39 -39
  97. package/src/subsystems/AgentManager/AgentData.service/index.ts +18 -18
  98. package/src/subsystems/AgentManager/AgentLogger.class.ts +301 -297
  99. package/src/subsystems/AgentManager/AgentRequest.class.ts +51 -51
  100. package/src/subsystems/AgentManager/AgentRuntime.class.ts +559 -559
  101. package/src/subsystems/AgentManager/AgentSSE.class.ts +101 -101
  102. package/src/subsystems/AgentManager/AgentSettings.class.ts +52 -52
  103. package/src/subsystems/AgentManager/Component.service/ComponentConnector.ts +32 -32
  104. package/src/subsystems/AgentManager/Component.service/connectors/LocalComponentConnector.class.ts +60 -60
  105. package/src/subsystems/AgentManager/Component.service/index.ts +11 -11
  106. package/src/subsystems/AgentManager/EmbodimentSettings.class.ts +47 -47
  107. package/src/subsystems/AgentManager/ForkedAgent.class.ts +154 -154
  108. package/src/subsystems/AgentManager/OSResourceMonitor.ts +77 -77
  109. package/src/subsystems/ComputeManager/Code.service/CodeConnector.ts +98 -98
  110. package/src/subsystems/ComputeManager/Code.service/connectors/AWSLambdaCode.class.ts +172 -172
  111. package/src/subsystems/ComputeManager/Code.service/connectors/ECMASandbox.class.ts +131 -131
  112. package/src/subsystems/ComputeManager/Code.service/index.ts +13 -13
  113. package/src/subsystems/IO/CLI.service/CLIConnector.ts +47 -47
  114. package/src/subsystems/IO/CLI.service/index.ts +9 -9
  115. package/src/subsystems/IO/Log.service/LogConnector.ts +32 -32
  116. package/src/subsystems/IO/Log.service/connectors/ConsoleLog.class.ts +28 -28
  117. package/src/subsystems/IO/Log.service/index.ts +13 -13
  118. package/src/subsystems/IO/NKV.service/NKVConnector.ts +43 -43
  119. package/src/subsystems/IO/NKV.service/connectors/NKVLocalStorage.class.ts +234 -234
  120. package/src/subsystems/IO/NKV.service/connectors/NKVRAM.class.ts +204 -204
  121. package/src/subsystems/IO/NKV.service/connectors/NKVRedis.class.ts +182 -182
  122. package/src/subsystems/IO/NKV.service/index.ts +14 -14
  123. package/src/subsystems/IO/Router.service/RouterConnector.ts +21 -21
  124. package/src/subsystems/IO/Router.service/connectors/ExpressRouter.class.ts +48 -48
  125. package/src/subsystems/IO/Router.service/connectors/NullRouter.class.ts +40 -40
  126. package/src/subsystems/IO/Router.service/index.ts +11 -11
  127. package/src/subsystems/IO/Storage.service/SmythFS.class.ts +489 -489
  128. package/src/subsystems/IO/Storage.service/StorageConnector.ts +66 -66
  129. package/src/subsystems/IO/Storage.service/connectors/LocalStorage.class.ts +327 -327
  130. package/src/subsystems/IO/Storage.service/connectors/S3Storage.class.ts +482 -482
  131. package/src/subsystems/IO/Storage.service/index.ts +13 -13
  132. package/src/subsystems/IO/VectorDB.service/VectorDBConnector.ts +108 -108
  133. package/src/subsystems/IO/VectorDB.service/connectors/MilvusVectorDB.class.ts +454 -454
  134. package/src/subsystems/IO/VectorDB.service/connectors/PineconeVectorDB.class.ts +384 -384
  135. package/src/subsystems/IO/VectorDB.service/connectors/RAMVecrtorDB.class.ts +421 -421
  136. package/src/subsystems/IO/VectorDB.service/embed/BaseEmbedding.ts +107 -107
  137. package/src/subsystems/IO/VectorDB.service/embed/OpenAIEmbedding.ts +109 -109
  138. package/src/subsystems/IO/VectorDB.service/embed/index.ts +21 -21
  139. package/src/subsystems/IO/VectorDB.service/index.ts +14 -14
  140. package/src/subsystems/LLMManager/LLM.helper.ts +251 -251
  141. package/src/subsystems/LLMManager/LLM.inference.ts +339 -339
  142. package/src/subsystems/LLMManager/LLM.service/LLMConnector.ts +489 -489
  143. package/src/subsystems/LLMManager/LLM.service/LLMCredentials.helper.ts +171 -171
  144. package/src/subsystems/LLMManager/LLM.service/connectors/Anthropic.class.ts +659 -659
  145. package/src/subsystems/LLMManager/LLM.service/connectors/Bedrock.class.ts +400 -400
  146. package/src/subsystems/LLMManager/LLM.service/connectors/Echo.class.ts +77 -77
  147. package/src/subsystems/LLMManager/LLM.service/connectors/GoogleAI.class.ts +757 -757
  148. package/src/subsystems/LLMManager/LLM.service/connectors/Groq.class.ts +304 -304
  149. package/src/subsystems/LLMManager/LLM.service/connectors/Perplexity.class.ts +250 -250
  150. package/src/subsystems/LLMManager/LLM.service/connectors/VertexAI.class.ts +423 -423
  151. package/src/subsystems/LLMManager/LLM.service/connectors/openai/OpenAIConnector.class.ts +488 -488
  152. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ChatCompletionsApiInterface.ts +524 -524
  153. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterface.ts +100 -100
  154. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterfaceFactory.ts +81 -81
  155. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ResponsesApiInterface.ts +1145 -1145
  156. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/constants.ts +13 -13
  157. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/index.ts +4 -4
  158. package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/utils.ts +11 -11
  159. package/src/subsystems/LLMManager/LLM.service/connectors/openai/types.ts +32 -32
  160. package/src/subsystems/LLMManager/LLM.service/connectors/xAI.class.ts +471 -471
  161. package/src/subsystems/LLMManager/LLM.service/index.ts +44 -44
  162. package/src/subsystems/LLMManager/ModelsProvider.service/ModelsProviderConnector.ts +300 -300
  163. package/src/subsystems/LLMManager/ModelsProvider.service/connectors/JSONModelsProvider.class.ts +252 -252
  164. package/src/subsystems/LLMManager/ModelsProvider.service/index.ts +11 -11
  165. package/src/subsystems/LLMManager/custom-models.ts +854 -854
  166. package/src/subsystems/LLMManager/models.ts +2540 -2540
  167. package/src/subsystems/LLMManager/paramMappings.ts +69 -69
  168. package/src/subsystems/MemoryManager/Cache.service/CacheConnector.ts +86 -86
  169. package/src/subsystems/MemoryManager/Cache.service/connectors/LocalStorageCache.class.ts +297 -297
  170. package/src/subsystems/MemoryManager/Cache.service/connectors/RAMCache.class.ts +201 -201
  171. package/src/subsystems/MemoryManager/Cache.service/connectors/RedisCache.class.ts +252 -252
  172. package/src/subsystems/MemoryManager/Cache.service/connectors/S3Cache.class.ts +373 -373
  173. package/src/subsystems/MemoryManager/Cache.service/index.ts +15 -15
  174. package/src/subsystems/MemoryManager/LLMCache.ts +72 -72
  175. package/src/subsystems/MemoryManager/LLMContext.ts +124 -124
  176. package/src/subsystems/MemoryManager/LLMMemory.service/LLMMemoryConnector.ts +26 -26
  177. package/src/subsystems/MemoryManager/RuntimeContext.ts +266 -266
  178. package/src/subsystems/Security/AccessControl/ACL.class.ts +208 -208
  179. package/src/subsystems/Security/AccessControl/AccessCandidate.class.ts +82 -82
  180. package/src/subsystems/Security/AccessControl/AccessRequest.class.ts +52 -52
  181. package/src/subsystems/Security/Account.service/AccountConnector.ts +44 -44
  182. package/src/subsystems/Security/Account.service/connectors/AWSAccount.class.ts +76 -76
  183. package/src/subsystems/Security/Account.service/connectors/DummyAccount.class.ts +130 -130
  184. package/src/subsystems/Security/Account.service/connectors/JSONFileAccount.class.ts +159 -159
  185. package/src/subsystems/Security/Account.service/index.ts +14 -14
  186. package/src/subsystems/Security/Credentials.helper.ts +62 -62
  187. package/src/subsystems/Security/ManagedVault.service/ManagedVaultConnector.ts +38 -38
  188. package/src/subsystems/Security/ManagedVault.service/connectors/NullManagedVault.class.ts +53 -53
  189. package/src/subsystems/Security/ManagedVault.service/connectors/SecretManagerManagedVault.ts +154 -154
  190. package/src/subsystems/Security/ManagedVault.service/index.ts +12 -12
  191. package/src/subsystems/Security/SecureConnector.class.ts +110 -110
  192. package/src/subsystems/Security/Vault.service/Vault.helper.ts +30 -30
  193. package/src/subsystems/Security/Vault.service/VaultConnector.ts +29 -29
  194. package/src/subsystems/Security/Vault.service/connectors/HashicorpVault.class.ts +46 -46
  195. package/src/subsystems/Security/Vault.service/connectors/JSONFileVault.class.ts +221 -221
  196. package/src/subsystems/Security/Vault.service/connectors/NullVault.class.ts +54 -54
  197. package/src/subsystems/Security/Vault.service/connectors/SecretsManager.class.ts +140 -140
  198. package/src/subsystems/Security/Vault.service/index.ts +12 -12
  199. package/src/types/ACL.types.ts +104 -104
  200. package/src/types/AWS.types.ts +10 -10
  201. package/src/types/Agent.types.ts +61 -61
  202. package/src/types/AgentLogger.types.ts +17 -17
  203. package/src/types/Cache.types.ts +1 -1
  204. package/src/types/Common.types.ts +2 -2
  205. package/src/types/LLM.types.ts +496 -496
  206. package/src/types/Redis.types.ts +8 -8
  207. package/src/types/SRE.types.ts +64 -64
  208. package/src/types/Security.types.ts +14 -14
  209. package/src/types/Storage.types.ts +5 -5
  210. package/src/types/VectorDB.types.ts +86 -86
  211. package/src/utils/base64.utils.ts +275 -275
  212. package/src/utils/cli.utils.ts +68 -68
  213. package/src/utils/data.utils.ts +322 -322
  214. package/src/utils/date-time.utils.ts +22 -22
  215. package/src/utils/general.utils.ts +238 -238
  216. package/src/utils/index.ts +12 -12
  217. package/src/utils/lazy-client.ts +261 -261
  218. package/src/utils/numbers.utils.ts +13 -13
  219. package/src/utils/oauth.utils.ts +35 -35
  220. package/src/utils/string.utils.ts +414 -414
  221. package/src/utils/url.utils.ts +19 -19
  222. package/src/utils/validation.utils.ts +74 -74
  223. package/dist/types/subsystems/LLMManager/ModelsProvider.service/connectors/SmythModelsProvider.class.d.ts +0 -39
@@ -1,471 +1,471 @@
1
- import EventEmitter from 'events';
2
- import axios, { AxiosInstance } from 'axios';
3
-
4
- import { JSON_RESPONSE_INSTRUCTION, BUILT_IN_MODEL_PREFIX } from '@sre/constants';
5
-
6
- import {
7
- TLLMParams,
8
- ToolData,
9
- TLLMMessageBlock,
10
- TLLMToolResultMessageBlock,
11
- TLLMMessageRole,
12
- APIKeySource,
13
- BasicCredentials,
14
- ILLMRequestFuncParams,
15
- TLLMChatResponse,
16
- ILLMRequestContext,
17
- } from '@sre/types/LLM.types';
18
- import { LLMHelper } from '@sre/LLMManager/LLM.helper';
19
-
20
- import { LLMConnector } from '../LLMConnector';
21
- import { SystemEvents } from '@sre/Core/SystemEvents';
22
-
23
- type ChatCompletionParams = {
24
- model: string;
25
- messages: any[];
26
- max_tokens?: number;
27
- temperature?: number;
28
- top_p?: number;
29
- frequency_penalty?: number;
30
- presence_penalty?: number;
31
- stop?: string | string[];
32
- response_format?: { type: string };
33
- stream?: boolean;
34
- stream_options?: { include_usage: boolean };
35
- tools?: any[];
36
- tool_choice?: any;
37
- // xAI search parameters - nested structure
38
- search_parameters?: {
39
- mode?: 'auto' | 'on' | 'off';
40
- return_citations?: boolean;
41
- max_search_results?: number;
42
- sources?: Array<{
43
- type: 'web' | 'x' | 'news' | 'rss';
44
- country?: string;
45
- excluded_websites?: string[];
46
- allowed_websites?: string[];
47
- safe_search?: boolean;
48
- included_x_handles?: string[];
49
- excluded_x_handles?: string[];
50
- post_favorite_count?: number;
51
- post_view_count?: number;
52
- links?: string[];
53
- }>;
54
- from_date?: string;
55
- to_date?: string;
56
- };
57
- };
58
-
59
- type TUsage = {
60
- prompt_tokens: number;
61
- completion_tokens: number;
62
- total_tokens: number;
63
- prompt_tokens_details?: {
64
- text_tokens?: number;
65
- audio_tokens?: number;
66
- image_tokens?: number;
67
- cached_tokens?: number;
68
- };
69
- completion_tokens_details?: {
70
- reasoning_tokens?: number;
71
- audio_tokens?: number;
72
- accepted_prediction_tokens?: number;
73
- rejected_prediction_tokens?: number;
74
- };
75
- reasoning_tokens?: number; // for backward compatibility
76
- num_sources_used?: number;
77
- };
78
-
79
- export class xAIConnector extends LLMConnector {
80
- public name = 'LLM:Grok';
81
-
82
- private async getClient(params: ILLMRequestContext): Promise<AxiosInstance> {
83
- const apiKey = (params.credentials as BasicCredentials)?.apiKey;
84
- const baseURL = params?.modelInfo?.baseURL || 'https://api.x.ai/v1';
85
-
86
- if (!apiKey) throw new Error('Please provide an API key for Grok');
87
-
88
- return axios.create({
89
- baseURL,
90
- headers: {
91
- Authorization: `Bearer ${apiKey}`,
92
- 'Content-Type': 'application/json',
93
- },
94
- });
95
- }
96
-
97
- protected async request({ acRequest, body, context }: ILLMRequestFuncParams): Promise<TLLMChatResponse> {
98
- try {
99
- const grok = await this.getClient(context);
100
- const response = await grok.post('/chat/completions', body);
101
-
102
- const message = response?.data?.choices?.[0]?.message;
103
- const finishReason = response?.data?.choices?.[0]?.finish_reason;
104
- const usage = response?.data?.usage as TUsage;
105
- const citations = response?.data?.citations;
106
-
107
- let toolsData: ToolData[] = [];
108
- let useTool = false;
109
-
110
- if (finishReason === 'tool_calls') {
111
- toolsData =
112
- message?.tool_calls?.map((tool, index) => ({
113
- index,
114
- id: tool?.id,
115
- type: tool?.type,
116
- name: tool?.function?.name,
117
- arguments: tool?.function?.arguments,
118
- role: 'tool',
119
- })) || [];
120
-
121
- useTool = true;
122
- }
123
-
124
- // Handle citations from live search
125
- let content = message?.content ?? '';
126
- if (citations && citations.length > 0) {
127
- const citationsText = '\n\n**Sources:**\n' + citations.map((url, index) => `${index + 1}. ${url}`).join('\n');
128
- content += citationsText;
129
- }
130
-
131
- this.reportUsage(usage, {
132
- modelEntryName: context.modelEntryName,
133
- keySource: context.isUserKey ? APIKeySource.User : APIKeySource.Smyth,
134
- agentId: context.agentId,
135
- teamId: context.teamId,
136
- });
137
-
138
- return {
139
- content,
140
- finishReason,
141
- useTool,
142
- toolsData,
143
- message,
144
- usage,
145
- };
146
- } catch (error) {
147
- throw error;
148
- }
149
- }
150
-
151
- protected async streamRequest({ acRequest, body, context }: ILLMRequestFuncParams): Promise<EventEmitter> {
152
- const emitter = new EventEmitter();
153
-
154
- try {
155
- const grok = await this.getClient(context);
156
- const response = await grok.post(
157
- '/chat/completions',
158
- { ...body, stream: true, stream_options: { include_usage: true } },
159
- {
160
- responseType: 'stream',
161
- }
162
- );
163
-
164
- const reportedUsage: any[] = [];
165
- let finishReason = 'stop';
166
- let toolsData: any[] = [];
167
- let usage: any = {};
168
- let citations: any[] = [];
169
-
170
- response.data.on('data', (chunk) => {
171
- const lines = chunk.toString().split('\n');
172
-
173
- for (const line of lines) {
174
- if (line.startsWith('data: ')) {
175
- const data = line.slice(6);
176
- if (data === '[DONE]') {
177
- // Handle stream completion
178
- continue;
179
- }
180
-
181
- try {
182
- const parsed = JSON.parse(data);
183
- const delta = parsed.choices?.[0]?.delta;
184
-
185
- // Usage data comes in final chunk when stream_options.include_usage is true
186
- if (parsed?.usage) {
187
- usage = parsed.usage;
188
- }
189
-
190
- // Handle citations from xAI - they come at the top level as an array of URLs
191
- if (parsed.citations) {
192
- citations = parsed.citations;
193
- }
194
-
195
- if (delta) {
196
- emitter.emit('data', delta);
197
-
198
- if (delta.content) {
199
- emitter.emit('content', delta.content, delta.role);
200
- }
201
-
202
- if (delta.tool_calls) {
203
- const toolCall = delta.tool_calls[0];
204
- const index = toolCall?.index;
205
-
206
- toolsData[index] = {
207
- index,
208
- role: 'tool',
209
- id: (toolsData?.[index]?.id || '') + (toolCall?.id || ''),
210
- type: (toolsData?.[index]?.type || '') + (toolCall?.type || ''),
211
- name: (toolsData?.[index]?.name || '') + (toolCall?.function?.name || ''),
212
- arguments: (toolsData?.[index]?.arguments || '') + (toolCall?.function?.arguments || ''),
213
- };
214
- }
215
- }
216
-
217
- if (parsed.choices?.[0]?.finish_reason) {
218
- finishReason = parsed.choices[0].finish_reason;
219
- }
220
- } catch (e) {
221
- // Ignore parsing errors for incomplete chunks
222
- }
223
- }
224
- }
225
- });
226
-
227
- response.data.on('end', () => {
228
- // Include citations in content if available
229
- if (citations && citations.length > 0) {
230
- const citationsText = '\n\n**Sources:**\n' + citations.map((url, index) => `${index + 1}. ${url}`).join('\n');
231
-
232
- emitter.emit('content', citationsText, 'assistant');
233
- }
234
-
235
- if (toolsData.length > 0) {
236
- emitter.emit('toolInfo', toolsData);
237
- }
238
-
239
- // Report usage if available
240
- if (Object.keys(usage).length > 0) {
241
- const _reported = this.reportUsage(usage, {
242
- modelEntryName: context.modelEntryName,
243
- keySource: context.isUserKey ? APIKeySource.User : APIKeySource.Smyth,
244
- agentId: context.agentId,
245
- teamId: context.teamId,
246
- });
247
- reportedUsage.push(_reported);
248
- }
249
-
250
- if (finishReason !== 'stop') {
251
- emitter.emit('interrupted', finishReason);
252
- }
253
-
254
- setTimeout(() => {
255
- emitter.emit('end', toolsData, reportedUsage, finishReason);
256
- }, 100);
257
- });
258
-
259
- response.data.on('error', (error) => {
260
- emitter.emit('error', error);
261
- });
262
- } catch (error) {
263
- emitter.emit('error', error);
264
- }
265
-
266
- return emitter;
267
- }
268
-
269
- protected async reqBodyAdapter(params: TLLMParams): Promise<ChatCompletionParams> {
270
- const messages = params?.messages || [];
271
- const modelName = params.model as string;
272
-
273
- // Handle JSON response format
274
- const responseFormat = params?.responseFormat || '';
275
- if (responseFormat === 'json') {
276
- if (messages?.[0]?.role === TLLMMessageRole.System) {
277
- messages[0].content += JSON_RESPONSE_INSTRUCTION;
278
- } else {
279
- messages.unshift({ role: TLLMMessageRole.System, content: JSON_RESPONSE_INSTRUCTION });
280
- }
281
-
282
- params.responseFormat = { type: 'json_object' };
283
- }
284
-
285
- const body: ChatCompletionParams = {
286
- model: modelName,
287
- messages,
288
- };
289
-
290
- // Add parameters if they're not undefined and not 0 (for reasoning models)
291
- if (params?.maxTokens !== undefined) body.max_tokens = params.maxTokens;
292
- if (params?.temperature !== undefined) body.temperature = params.temperature;
293
- if (params?.topP !== undefined) body.top_p = params.topP;
294
-
295
- if (params?.responseFormat) {
296
- body.response_format = params.responseFormat;
297
- }
298
-
299
- // Add tools configuration if available
300
- if (params?.toolsConfig?.tools && params?.toolsConfig?.tools?.length > 0) {
301
- body.tools = params.toolsConfig.tools;
302
- }
303
-
304
- if (params?.toolsConfig?.tool_choice) {
305
- body.tool_choice = params.toolsConfig.tool_choice;
306
- }
307
-
308
- // Add xAI search configuration if useSearch is enabled
309
- if (params?.useSearch) {
310
- body.search_parameters = {};
311
-
312
- // Basic search parameters
313
- if (params.searchMode) body.search_parameters.mode = params.searchMode;
314
- if (params.returnCitations !== undefined) body.search_parameters.return_citations = params.returnCitations;
315
- if (params.maxSearchResults !== undefined) body.search_parameters.max_search_results = params.maxSearchResults;
316
-
317
- // Date filtering
318
- if (params.fromDate) body.search_parameters.from_date = params.fromDate;
319
- if (params.toDate) body.search_parameters.to_date = params.toDate;
320
-
321
- // Create sources array
322
- const sources: any[] = [];
323
-
324
- // If searchDataSources is provided, use it as source types
325
- if (params.searchDataSources && params.searchDataSources.length > 0) {
326
- params.searchDataSources.forEach((sourceType) => {
327
- const source: any = { type: sourceType };
328
-
329
- // Add parameters based on source type
330
- if (sourceType === 'web' || sourceType === 'news') {
331
- if (params.searchCountry) source.country = params.searchCountry;
332
-
333
- // Website filtering (mutually exclusive)
334
- if (params.excludedWebsites && params.excludedWebsites.length > 0) {
335
- source.excluded_websites = params.excludedWebsites;
336
- } else if (params.allowedWebsites && params.allowedWebsites.length > 0) {
337
- source.allowed_websites = params.allowedWebsites;
338
- }
339
-
340
- if (params.safeSearch !== undefined) source.safe_search = params.safeSearch;
341
- }
342
-
343
- if (sourceType === 'x') {
344
- if (params.includedXHandles && params.includedXHandles.length > 0) {
345
- source.included_x_handles = params.includedXHandles;
346
- } else if (params.excludedXHandles && params.excludedXHandles.length > 0) {
347
- source.excluded_x_handles = params.excludedXHandles;
348
- }
349
- if (params.postFavoriteCount !== undefined && params.postFavoriteCount > 0) {
350
- source.post_favorite_count = params.postFavoriteCount;
351
- }
352
- if (params.postViewCount !== undefined && params.postViewCount > 0) {
353
- source.post_view_count = params.postViewCount;
354
- }
355
- }
356
-
357
- if (sourceType === 'rss') {
358
- if (params.rssLinks) source.links = params.rssLinks;
359
- }
360
-
361
- sources.push(source);
362
- });
363
- }
364
-
365
- if (sources.length > 0) {
366
- body.search_parameters.sources = sources;
367
- }
368
- }
369
-
370
- return body;
371
- }
372
-
373
- protected reportUsage(usage: TUsage, metadata: { modelEntryName: string; keySource: APIKeySource; agentId: string; teamId: string }) {
374
- // SmythOS (built-in) models have a prefix, so we need to remove it to get the model name
375
- const modelName = metadata.modelEntryName.replace(BUILT_IN_MODEL_PREFIX, '');
376
-
377
- const usageData = {
378
- sourceId: `llm:${modelName}`,
379
- input_tokens: usage?.prompt_tokens - (usage?.prompt_tokens_details?.cached_tokens || 0),
380
- output_tokens: usage?.completion_tokens,
381
- input_tokens_cache_write: 0,
382
- input_tokens_cache_read: usage?.prompt_tokens_details?.cached_tokens || 0,
383
- reasoning_tokens: usage?.completion_tokens_details?.reasoning_tokens || usage?.reasoning_tokens || 0,
384
- keySource: metadata.keySource,
385
- agentId: metadata.agentId,
386
- teamId: metadata.teamId,
387
- };
388
- SystemEvents.emit('USAGE:LLM', usageData);
389
-
390
- return usageData;
391
- }
392
-
393
- public formatToolsConfig({ type = 'function', toolDefinitions, toolChoice = 'auto' }) {
394
- let tools: any[] = [];
395
-
396
- if (type === 'function') {
397
- tools = toolDefinitions.map((tool) => {
398
- const { name, description, properties, requiredFields } = tool;
399
-
400
- return {
401
- type: 'function',
402
- function: {
403
- name,
404
- description,
405
- parameters: {
406
- type: 'object',
407
- properties,
408
- required: requiredFields,
409
- },
410
- },
411
- };
412
- });
413
- }
414
-
415
- return tools?.length > 0 ? { tools, tool_choice: toolChoice || 'auto' } : {};
416
- }
417
-
418
- public transformToolMessageBlocks({
419
- messageBlock,
420
- toolsData,
421
- }: {
422
- messageBlock: TLLMMessageBlock;
423
- toolsData: ToolData[];
424
- }): TLLMToolResultMessageBlock[] {
425
- const messageBlocks: TLLMToolResultMessageBlock[] = [];
426
-
427
- if (messageBlock) {
428
- const transformedMessageBlock = {
429
- ...messageBlock,
430
- content: typeof messageBlock.content === 'object' ? JSON.stringify(messageBlock.content) : messageBlock.content,
431
- };
432
- if (transformedMessageBlock.tool_calls) {
433
- for (let toolCall of transformedMessageBlock.tool_calls) {
434
- toolCall.function.arguments =
435
- typeof toolCall.function.arguments === 'object' ? JSON.stringify(toolCall.function.arguments) : toolCall.function.arguments;
436
- }
437
- }
438
- messageBlocks.push(transformedMessageBlock);
439
- }
440
-
441
- const transformedToolsData = toolsData.map((toolData) => ({
442
- tool_call_id: toolData.id,
443
- role: TLLMMessageRole.Tool,
444
- name: toolData.name,
445
- content: typeof toolData.result === 'string' ? toolData.result : JSON.stringify(toolData.result),
446
- }));
447
-
448
- return [...messageBlocks, ...transformedToolsData];
449
- }
450
-
451
- public getConsistentMessages(messages) {
452
- const _messages = LLMHelper.removeDuplicateUserMessages(messages);
453
-
454
- return _messages.map((message) => {
455
- const _message = { ...message };
456
- let textContent = '';
457
-
458
- if (message?.parts) {
459
- textContent = message.parts.map((textBlock) => textBlock?.text || '').join(' ');
460
- } else if (Array.isArray(message?.content)) {
461
- textContent = message.content.map((textBlock) => textBlock?.text || '').join(' ');
462
- } else if (message?.content) {
463
- textContent = message.content;
464
- }
465
-
466
- _message.content = textContent;
467
-
468
- return _message;
469
- });
470
- }
471
- }
1
+ import EventEmitter from 'events';
2
+ import axios, { AxiosInstance } from 'axios';
3
+
4
+ import { JSON_RESPONSE_INSTRUCTION, BUILT_IN_MODEL_PREFIX } from '@sre/constants';
5
+
6
+ import {
7
+ TLLMParams,
8
+ ToolData,
9
+ TLLMMessageBlock,
10
+ TLLMToolResultMessageBlock,
11
+ TLLMMessageRole,
12
+ APIKeySource,
13
+ BasicCredentials,
14
+ ILLMRequestFuncParams,
15
+ TLLMChatResponse,
16
+ ILLMRequestContext,
17
+ } from '@sre/types/LLM.types';
18
+ import { LLMHelper } from '@sre/LLMManager/LLM.helper';
19
+
20
+ import { LLMConnector } from '../LLMConnector';
21
+ import { SystemEvents } from '@sre/Core/SystemEvents';
22
+
23
+ type ChatCompletionParams = {
24
+ model: string;
25
+ messages: any[];
26
+ max_tokens?: number;
27
+ temperature?: number;
28
+ top_p?: number;
29
+ frequency_penalty?: number;
30
+ presence_penalty?: number;
31
+ stop?: string | string[];
32
+ response_format?: { type: string };
33
+ stream?: boolean;
34
+ stream_options?: { include_usage: boolean };
35
+ tools?: any[];
36
+ tool_choice?: any;
37
+ // xAI search parameters - nested structure
38
+ search_parameters?: {
39
+ mode?: 'auto' | 'on' | 'off';
40
+ return_citations?: boolean;
41
+ max_search_results?: number;
42
+ sources?: Array<{
43
+ type: 'web' | 'x' | 'news' | 'rss';
44
+ country?: string;
45
+ excluded_websites?: string[];
46
+ allowed_websites?: string[];
47
+ safe_search?: boolean;
48
+ included_x_handles?: string[];
49
+ excluded_x_handles?: string[];
50
+ post_favorite_count?: number;
51
+ post_view_count?: number;
52
+ links?: string[];
53
+ }>;
54
+ from_date?: string;
55
+ to_date?: string;
56
+ };
57
+ };
58
+
59
+ type TUsage = {
60
+ prompt_tokens: number;
61
+ completion_tokens: number;
62
+ total_tokens: number;
63
+ prompt_tokens_details?: {
64
+ text_tokens?: number;
65
+ audio_tokens?: number;
66
+ image_tokens?: number;
67
+ cached_tokens?: number;
68
+ };
69
+ completion_tokens_details?: {
70
+ reasoning_tokens?: number;
71
+ audio_tokens?: number;
72
+ accepted_prediction_tokens?: number;
73
+ rejected_prediction_tokens?: number;
74
+ };
75
+ reasoning_tokens?: number; // for backward compatibility
76
+ num_sources_used?: number;
77
+ };
78
+
79
+ export class xAIConnector extends LLMConnector {
80
+ public name = 'LLM:Grok';
81
+
82
+ private async getClient(params: ILLMRequestContext): Promise<AxiosInstance> {
83
+ const apiKey = (params.credentials as BasicCredentials)?.apiKey;
84
+ const baseURL = params?.modelInfo?.baseURL || 'https://api.x.ai/v1';
85
+
86
+ if (!apiKey) throw new Error('Please provide an API key for Grok');
87
+
88
+ return axios.create({
89
+ baseURL,
90
+ headers: {
91
+ Authorization: `Bearer ${apiKey}`,
92
+ 'Content-Type': 'application/json',
93
+ },
94
+ });
95
+ }
96
+
97
+ protected async request({ acRequest, body, context }: ILLMRequestFuncParams): Promise<TLLMChatResponse> {
98
+ try {
99
+ const grok = await this.getClient(context);
100
+ const response = await grok.post('/chat/completions', body);
101
+
102
+ const message = response?.data?.choices?.[0]?.message;
103
+ const finishReason = response?.data?.choices?.[0]?.finish_reason;
104
+ const usage = response?.data?.usage as TUsage;
105
+ const citations = response?.data?.citations;
106
+
107
+ let toolsData: ToolData[] = [];
108
+ let useTool = false;
109
+
110
+ if (finishReason === 'tool_calls') {
111
+ toolsData =
112
+ message?.tool_calls?.map((tool, index) => ({
113
+ index,
114
+ id: tool?.id,
115
+ type: tool?.type,
116
+ name: tool?.function?.name,
117
+ arguments: tool?.function?.arguments,
118
+ role: 'tool',
119
+ })) || [];
120
+
121
+ useTool = true;
122
+ }
123
+
124
+ // Handle citations from live search
125
+ let content = message?.content ?? '';
126
+ if (citations && citations.length > 0) {
127
+ const citationsText = '\n\n**Sources:**\n' + citations.map((url, index) => `${index + 1}. ${url}`).join('\n');
128
+ content += citationsText;
129
+ }
130
+
131
+ this.reportUsage(usage, {
132
+ modelEntryName: context.modelEntryName,
133
+ keySource: context.isUserKey ? APIKeySource.User : APIKeySource.Smyth,
134
+ agentId: context.agentId,
135
+ teamId: context.teamId,
136
+ });
137
+
138
+ return {
139
+ content,
140
+ finishReason,
141
+ useTool,
142
+ toolsData,
143
+ message,
144
+ usage,
145
+ };
146
+ } catch (error) {
147
+ throw error;
148
+ }
149
+ }
150
+
151
+ protected async streamRequest({ acRequest, body, context }: ILLMRequestFuncParams): Promise<EventEmitter> {
152
+ const emitter = new EventEmitter();
153
+
154
+ try {
155
+ const grok = await this.getClient(context);
156
+ const response = await grok.post(
157
+ '/chat/completions',
158
+ { ...body, stream: true, stream_options: { include_usage: true } },
159
+ {
160
+ responseType: 'stream',
161
+ }
162
+ );
163
+
164
+ const reportedUsage: any[] = [];
165
+ let finishReason = 'stop';
166
+ let toolsData: any[] = [];
167
+ let usage: any = {};
168
+ let citations: any[] = [];
169
+
170
+ response.data.on('data', (chunk) => {
171
+ const lines = chunk.toString().split('\n');
172
+
173
+ for (const line of lines) {
174
+ if (line.startsWith('data: ')) {
175
+ const data = line.slice(6);
176
+ if (data === '[DONE]') {
177
+ // Handle stream completion
178
+ continue;
179
+ }
180
+
181
+ try {
182
+ const parsed = JSON.parse(data);
183
+ const delta = parsed.choices?.[0]?.delta;
184
+
185
+ // Usage data comes in final chunk when stream_options.include_usage is true
186
+ if (parsed?.usage) {
187
+ usage = parsed.usage;
188
+ }
189
+
190
+ // Handle citations from xAI - they come at the top level as an array of URLs
191
+ if (parsed.citations) {
192
+ citations = parsed.citations;
193
+ }
194
+
195
+ if (delta) {
196
+ emitter.emit('data', delta);
197
+
198
+ if (delta.content) {
199
+ emitter.emit('content', delta.content, delta.role);
200
+ }
201
+
202
+ if (delta.tool_calls) {
203
+ const toolCall = delta.tool_calls[0];
204
+ const index = toolCall?.index;
205
+
206
+ toolsData[index] = {
207
+ index,
208
+ role: 'tool',
209
+ id: (toolsData?.[index]?.id || '') + (toolCall?.id || ''),
210
+ type: (toolsData?.[index]?.type || '') + (toolCall?.type || ''),
211
+ name: (toolsData?.[index]?.name || '') + (toolCall?.function?.name || ''),
212
+ arguments: (toolsData?.[index]?.arguments || '') + (toolCall?.function?.arguments || ''),
213
+ };
214
+ }
215
+ }
216
+
217
+ if (parsed.choices?.[0]?.finish_reason) {
218
+ finishReason = parsed.choices[0].finish_reason;
219
+ }
220
+ } catch (e) {
221
+ // Ignore parsing errors for incomplete chunks
222
+ }
223
+ }
224
+ }
225
+ });
226
+
227
+ response.data.on('end', () => {
228
+ // Include citations in content if available
229
+ if (citations && citations.length > 0) {
230
+ const citationsText = '\n\n**Sources:**\n' + citations.map((url, index) => `${index + 1}. ${url}`).join('\n');
231
+
232
+ emitter.emit('content', citationsText, 'assistant');
233
+ }
234
+
235
+ if (toolsData.length > 0) {
236
+ emitter.emit('toolInfo', toolsData);
237
+ }
238
+
239
+ // Report usage if available
240
+ if (Object.keys(usage).length > 0) {
241
+ const _reported = this.reportUsage(usage, {
242
+ modelEntryName: context.modelEntryName,
243
+ keySource: context.isUserKey ? APIKeySource.User : APIKeySource.Smyth,
244
+ agentId: context.agentId,
245
+ teamId: context.teamId,
246
+ });
247
+ reportedUsage.push(_reported);
248
+ }
249
+
250
+ if (finishReason !== 'stop') {
251
+ emitter.emit('interrupted', finishReason);
252
+ }
253
+
254
+ setTimeout(() => {
255
+ emitter.emit('end', toolsData, reportedUsage, finishReason);
256
+ }, 100);
257
+ });
258
+
259
+ response.data.on('error', (error) => {
260
+ emitter.emit('error', error);
261
+ });
262
+ } catch (error) {
263
+ emitter.emit('error', error);
264
+ }
265
+
266
+ return emitter;
267
+ }
268
+
269
+ protected async reqBodyAdapter(params: TLLMParams): Promise<ChatCompletionParams> {
270
+ const messages = params?.messages || [];
271
+ const modelName = params.model as string;
272
+
273
+ // Handle JSON response format
274
+ const responseFormat = params?.responseFormat || '';
275
+ if (responseFormat === 'json') {
276
+ if (messages?.[0]?.role === TLLMMessageRole.System) {
277
+ messages[0].content += JSON_RESPONSE_INSTRUCTION;
278
+ } else {
279
+ messages.unshift({ role: TLLMMessageRole.System, content: JSON_RESPONSE_INSTRUCTION });
280
+ }
281
+
282
+ params.responseFormat = { type: 'json_object' };
283
+ }
284
+
285
+ const body: ChatCompletionParams = {
286
+ model: modelName,
287
+ messages,
288
+ };
289
+
290
+ // Add parameters if they're not undefined and not 0 (for reasoning models)
291
+ if (params?.maxTokens !== undefined) body.max_tokens = params.maxTokens;
292
+ if (params?.temperature !== undefined) body.temperature = params.temperature;
293
+ if (params?.topP !== undefined) body.top_p = params.topP;
294
+
295
+ if (params?.responseFormat) {
296
+ body.response_format = params.responseFormat;
297
+ }
298
+
299
+ // Add tools configuration if available
300
+ if (params?.toolsConfig?.tools && params?.toolsConfig?.tools?.length > 0) {
301
+ body.tools = params.toolsConfig.tools;
302
+ }
303
+
304
+ if (params?.toolsConfig?.tool_choice) {
305
+ body.tool_choice = params.toolsConfig.tool_choice;
306
+ }
307
+
308
+ // Add xAI search configuration if useSearch is enabled
309
+ if (params?.useSearch) {
310
+ body.search_parameters = {};
311
+
312
+ // Basic search parameters
313
+ if (params.searchMode) body.search_parameters.mode = params.searchMode;
314
+ if (params.returnCitations !== undefined) body.search_parameters.return_citations = params.returnCitations;
315
+ if (params.maxSearchResults !== undefined) body.search_parameters.max_search_results = params.maxSearchResults;
316
+
317
+ // Date filtering
318
+ if (params.fromDate) body.search_parameters.from_date = params.fromDate;
319
+ if (params.toDate) body.search_parameters.to_date = params.toDate;
320
+
321
+ // Create sources array
322
+ const sources: any[] = [];
323
+
324
+ // If searchDataSources is provided, use it as source types
325
+ if (params.searchDataSources && params.searchDataSources.length > 0) {
326
+ params.searchDataSources.forEach((sourceType) => {
327
+ const source: any = { type: sourceType };
328
+
329
+ // Add parameters based on source type
330
+ if (sourceType === 'web' || sourceType === 'news') {
331
+ if (params.searchCountry) source.country = params.searchCountry;
332
+
333
+ // Website filtering (mutually exclusive)
334
+ if (params.excludedWebsites && params.excludedWebsites.length > 0) {
335
+ source.excluded_websites = params.excludedWebsites;
336
+ } else if (params.allowedWebsites && params.allowedWebsites.length > 0) {
337
+ source.allowed_websites = params.allowedWebsites;
338
+ }
339
+
340
+ if (params.safeSearch !== undefined) source.safe_search = params.safeSearch;
341
+ }
342
+
343
+ if (sourceType === 'x') {
344
+ if (params.includedXHandles && params.includedXHandles.length > 0) {
345
+ source.included_x_handles = params.includedXHandles;
346
+ } else if (params.excludedXHandles && params.excludedXHandles.length > 0) {
347
+ source.excluded_x_handles = params.excludedXHandles;
348
+ }
349
+ if (params.postFavoriteCount !== undefined && params.postFavoriteCount > 0) {
350
+ source.post_favorite_count = params.postFavoriteCount;
351
+ }
352
+ if (params.postViewCount !== undefined && params.postViewCount > 0) {
353
+ source.post_view_count = params.postViewCount;
354
+ }
355
+ }
356
+
357
+ if (sourceType === 'rss') {
358
+ if (params.rssLinks) source.links = params.rssLinks;
359
+ }
360
+
361
+ sources.push(source);
362
+ });
363
+ }
364
+
365
+ if (sources.length > 0) {
366
+ body.search_parameters.sources = sources;
367
+ }
368
+ }
369
+
370
+ return body;
371
+ }
372
+
373
+ protected reportUsage(usage: TUsage, metadata: { modelEntryName: string; keySource: APIKeySource; agentId: string; teamId: string }) {
374
+ // SmythOS (built-in) models have a prefix, so we need to remove it to get the model name
375
+ const modelName = metadata.modelEntryName.replace(BUILT_IN_MODEL_PREFIX, '');
376
+
377
+ const usageData = {
378
+ sourceId: `llm:${modelName}`,
379
+ input_tokens: usage?.prompt_tokens - (usage?.prompt_tokens_details?.cached_tokens || 0),
380
+ output_tokens: usage?.completion_tokens,
381
+ input_tokens_cache_write: 0,
382
+ input_tokens_cache_read: usage?.prompt_tokens_details?.cached_tokens || 0,
383
+ reasoning_tokens: usage?.completion_tokens_details?.reasoning_tokens || usage?.reasoning_tokens || 0,
384
+ keySource: metadata.keySource,
385
+ agentId: metadata.agentId,
386
+ teamId: metadata.teamId,
387
+ };
388
+ SystemEvents.emit('USAGE:LLM', usageData);
389
+
390
+ return usageData;
391
+ }
392
+
393
+ public formatToolsConfig({ type = 'function', toolDefinitions, toolChoice = 'auto' }) {
394
+ let tools: any[] = [];
395
+
396
+ if (type === 'function') {
397
+ tools = toolDefinitions.map((tool) => {
398
+ const { name, description, properties, requiredFields } = tool;
399
+
400
+ return {
401
+ type: 'function',
402
+ function: {
403
+ name,
404
+ description,
405
+ parameters: {
406
+ type: 'object',
407
+ properties,
408
+ required: requiredFields,
409
+ },
410
+ },
411
+ };
412
+ });
413
+ }
414
+
415
+ return tools?.length > 0 ? { tools, tool_choice: toolChoice || 'auto' } : {};
416
+ }
417
+
418
+ public transformToolMessageBlocks({
419
+ messageBlock,
420
+ toolsData,
421
+ }: {
422
+ messageBlock: TLLMMessageBlock;
423
+ toolsData: ToolData[];
424
+ }): TLLMToolResultMessageBlock[] {
425
+ const messageBlocks: TLLMToolResultMessageBlock[] = [];
426
+
427
+ if (messageBlock) {
428
+ const transformedMessageBlock = {
429
+ ...messageBlock,
430
+ content: typeof messageBlock.content === 'object' ? JSON.stringify(messageBlock.content) : messageBlock.content,
431
+ };
432
+ if (transformedMessageBlock.tool_calls) {
433
+ for (let toolCall of transformedMessageBlock.tool_calls) {
434
+ toolCall.function.arguments =
435
+ typeof toolCall.function.arguments === 'object' ? JSON.stringify(toolCall.function.arguments) : toolCall.function.arguments;
436
+ }
437
+ }
438
+ messageBlocks.push(transformedMessageBlock);
439
+ }
440
+
441
+ const transformedToolsData = toolsData.map((toolData) => ({
442
+ tool_call_id: toolData.id,
443
+ role: TLLMMessageRole.Tool,
444
+ name: toolData.name,
445
+ content: typeof toolData.result === 'string' ? toolData.result : JSON.stringify(toolData.result),
446
+ }));
447
+
448
+ return [...messageBlocks, ...transformedToolsData];
449
+ }
450
+
451
+ public getConsistentMessages(messages) {
452
+ const _messages = LLMHelper.removeDuplicateUserMessages(messages);
453
+
454
+ return _messages.map((message) => {
455
+ const _message = { ...message };
456
+ let textContent = '';
457
+
458
+ if (message?.parts) {
459
+ textContent = message.parts.map((textBlock) => textBlock?.text || '').join(' ');
460
+ } else if (Array.isArray(message?.content)) {
461
+ textContent = message.content.map((textBlock) => textBlock?.text || '').join(' ');
462
+ } else if (message?.content) {
463
+ textContent = message.content;
464
+ }
465
+
466
+ _message.content = textContent;
467
+
468
+ return _message;
469
+ });
470
+ }
471
+ }