@gguf/coder 0.3.0 → 0.3.1

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 (414) hide show
  1. package/.editorconfig +16 -0
  2. package/.env.example +63 -0
  3. package/.gitattributes +1 -0
  4. package/.semgrepignore +19 -0
  5. package/coder-dummy-file.ts +52 -0
  6. package/coder.config.example.json +59 -0
  7. package/coder.config.json +13 -0
  8. package/color_picker.html +36 -0
  9. package/package.json +2 -14
  10. package/scripts/extract-changelog.js +73 -0
  11. package/scripts/fetch-models.js +143 -0
  12. package/scripts/test.sh +40 -0
  13. package/scripts/update-homebrew-formula.sh +125 -0
  14. package/scripts/update-nix-version.sh +157 -0
  15. package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
  16. package/source/ai-sdk-client/AISDKClient.ts +155 -0
  17. package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
  18. package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
  19. package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
  20. package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
  21. package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
  22. package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
  23. package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
  24. package/source/ai-sdk-client/converters/message-converter.ts +113 -0
  25. package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
  26. package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
  27. package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
  28. package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
  29. package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
  30. package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
  31. package/source/ai-sdk-client/index.ts +7 -0
  32. package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
  33. package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
  34. package/source/ai-sdk-client/types.ts +9 -0
  35. package/source/ai-sdk-client-empty-message.spec.ts +141 -0
  36. package/source/ai-sdk-client-error-handling.spec.ts +186 -0
  37. package/source/ai-sdk-client-maxretries.spec.ts +114 -0
  38. package/source/ai-sdk-client-preparestep.spec.ts +279 -0
  39. package/source/app/App.spec.tsx +32 -0
  40. package/source/app/App.tsx +480 -0
  41. package/source/app/components/AppContainer.spec.tsx +96 -0
  42. package/source/app/components/AppContainer.tsx +56 -0
  43. package/source/app/components/ChatInterface.spec.tsx +163 -0
  44. package/source/app/components/ChatInterface.tsx +144 -0
  45. package/source/app/components/ModalSelectors.spec.tsx +141 -0
  46. package/source/app/components/ModalSelectors.tsx +135 -0
  47. package/source/app/helpers.spec.ts +97 -0
  48. package/source/app/helpers.ts +63 -0
  49. package/source/app/index.ts +4 -0
  50. package/source/app/types.ts +39 -0
  51. package/source/app/utils/appUtils.ts +294 -0
  52. package/source/app/utils/conversationState.ts +310 -0
  53. package/source/app.spec.tsx +244 -0
  54. package/source/cli.spec.ts +73 -0
  55. package/source/cli.tsx +51 -0
  56. package/source/client-factory.spec.ts +48 -0
  57. package/source/client-factory.ts +178 -0
  58. package/source/command-parser.spec.ts +127 -0
  59. package/source/command-parser.ts +36 -0
  60. package/source/commands/checkpoint.spec.tsx +277 -0
  61. package/source/commands/checkpoint.tsx +366 -0
  62. package/source/commands/clear.tsx +22 -0
  63. package/source/commands/custom-commands.tsx +121 -0
  64. package/source/commands/exit.ts +21 -0
  65. package/source/commands/export.spec.tsx +131 -0
  66. package/source/commands/export.tsx +79 -0
  67. package/source/commands/help.tsx +120 -0
  68. package/source/commands/index.ts +17 -0
  69. package/source/commands/init.tsx +339 -0
  70. package/source/commands/lsp-command.spec.tsx +281 -0
  71. package/source/commands/lsp.tsx +120 -0
  72. package/source/commands/mcp-command.spec.tsx +313 -0
  73. package/source/commands/mcp.tsx +162 -0
  74. package/source/commands/model-database.spec.tsx +758 -0
  75. package/source/commands/model-database.tsx +418 -0
  76. package/source/commands/model.ts +12 -0
  77. package/source/commands/provider.ts +12 -0
  78. package/source/commands/setup-config.tsx +16 -0
  79. package/source/commands/simple-commands.spec.tsx +175 -0
  80. package/source/commands/status.ts +12 -0
  81. package/source/commands/theme.ts +12 -0
  82. package/source/commands/update.spec.tsx +261 -0
  83. package/source/commands/update.tsx +201 -0
  84. package/source/commands/usage.spec.tsx +495 -0
  85. package/source/commands/usage.tsx +100 -0
  86. package/source/commands.spec.ts +436 -0
  87. package/source/commands.ts +83 -0
  88. package/source/components/assistant-message.spec.tsx +796 -0
  89. package/source/components/assistant-message.tsx +34 -0
  90. package/source/components/bash-execution-indicator.tsx +21 -0
  91. package/source/components/cancelling-indicator.tsx +16 -0
  92. package/source/components/chat-queue.spec.tsx +83 -0
  93. package/source/components/chat-queue.tsx +36 -0
  94. package/source/components/checkpoint-display.spec.tsx +219 -0
  95. package/source/components/checkpoint-display.tsx +126 -0
  96. package/source/components/checkpoint-selector.spec.tsx +173 -0
  97. package/source/components/checkpoint-selector.tsx +173 -0
  98. package/source/components/development-mode-indicator.spec.tsx +268 -0
  99. package/source/components/development-mode-indicator.tsx +38 -0
  100. package/source/components/message-box.spec.tsx +427 -0
  101. package/source/components/message-box.tsx +87 -0
  102. package/source/components/model-selector.tsx +132 -0
  103. package/source/components/provider-selector.tsx +75 -0
  104. package/source/components/random-spinner.tsx +19 -0
  105. package/source/components/security-disclaimer.tsx +73 -0
  106. package/source/components/status-connection-display.spec.tsx +133 -0
  107. package/source/components/status.tsx +267 -0
  108. package/source/components/theme-selector.tsx +126 -0
  109. package/source/components/tool-confirmation.tsx +190 -0
  110. package/source/components/tool-execution-indicator.tsx +33 -0
  111. package/source/components/tool-message.tsx +85 -0
  112. package/source/components/ui/titled-box.spec.tsx +207 -0
  113. package/source/components/ui/titled-box.tsx +57 -0
  114. package/source/components/usage/progress-bar.spec.tsx +398 -0
  115. package/source/components/usage/progress-bar.tsx +30 -0
  116. package/source/components/usage/usage-display.spec.tsx +780 -0
  117. package/source/components/usage/usage-display.tsx +291 -0
  118. package/source/components/user-input.spec.tsx +327 -0
  119. package/source/components/user-input.tsx +533 -0
  120. package/source/components/user-message.spec.tsx +230 -0
  121. package/source/components/user-message.tsx +84 -0
  122. package/source/components/welcome-message.tsx +76 -0
  123. package/source/config/env-substitution.ts +65 -0
  124. package/source/config/index.spec.ts +171 -0
  125. package/source/config/index.ts +154 -0
  126. package/source/config/paths.spec.ts +241 -0
  127. package/source/config/paths.ts +55 -0
  128. package/source/config/preferences.ts +51 -0
  129. package/source/config/themes.ts +315 -0
  130. package/source/constants.ts +130 -0
  131. package/source/context/mode-context.spec.ts +79 -0
  132. package/source/context/mode-context.ts +24 -0
  133. package/source/custom-commands/executor.spec.ts +142 -0
  134. package/source/custom-commands/executor.ts +64 -0
  135. package/source/custom-commands/loader.spec.ts +314 -0
  136. package/source/custom-commands/loader.ts +153 -0
  137. package/source/custom-commands/parser.ts +196 -0
  138. package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
  139. package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
  140. package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
  141. package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
  142. package/source/hooks/chat-handler/index.ts +12 -0
  143. package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
  144. package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
  145. package/source/hooks/chat-handler/types.ts +38 -0
  146. package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
  147. package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
  148. package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
  149. package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
  150. package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
  151. package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
  152. package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
  153. package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
  154. package/source/hooks/useAppHandlers.tsx +291 -0
  155. package/source/hooks/useAppInitialization.tsx +422 -0
  156. package/source/hooks/useAppState.tsx +311 -0
  157. package/source/hooks/useDirectoryTrust.tsx +98 -0
  158. package/source/hooks/useInputState.ts +414 -0
  159. package/source/hooks/useModeHandlers.tsx +302 -0
  160. package/source/hooks/useNonInteractiveMode.ts +140 -0
  161. package/source/hooks/useTerminalWidth.tsx +81 -0
  162. package/source/hooks/useTheme.ts +18 -0
  163. package/source/hooks/useToolHandler.tsx +349 -0
  164. package/source/hooks/useUIState.ts +61 -0
  165. package/source/init/agents-template-generator.ts +421 -0
  166. package/source/init/existing-rules-extractor.ts +319 -0
  167. package/source/init/file-scanner.spec.ts +227 -0
  168. package/source/init/file-scanner.ts +238 -0
  169. package/source/init/framework-detector.ts +382 -0
  170. package/source/init/language-detector.ts +269 -0
  171. package/source/init/project-analyzer.spec.ts +231 -0
  172. package/source/init/project-analyzer.ts +458 -0
  173. package/source/lsp/index.ts +31 -0
  174. package/source/lsp/lsp-client.spec.ts +508 -0
  175. package/source/lsp/lsp-client.ts +487 -0
  176. package/source/lsp/lsp-manager.spec.ts +477 -0
  177. package/source/lsp/lsp-manager.ts +419 -0
  178. package/source/lsp/protocol.spec.ts +502 -0
  179. package/source/lsp/protocol.ts +360 -0
  180. package/source/lsp/server-discovery.spec.ts +654 -0
  181. package/source/lsp/server-discovery.ts +515 -0
  182. package/source/markdown-parser/html-entities.spec.ts +88 -0
  183. package/source/markdown-parser/html-entities.ts +45 -0
  184. package/source/markdown-parser/index.spec.ts +281 -0
  185. package/source/markdown-parser/index.ts +126 -0
  186. package/source/markdown-parser/table-parser.spec.ts +133 -0
  187. package/source/markdown-parser/table-parser.ts +114 -0
  188. package/source/markdown-parser/utils.spec.ts +70 -0
  189. package/source/markdown-parser/utils.ts +13 -0
  190. package/source/mcp/mcp-client.spec.ts +81 -0
  191. package/source/mcp/mcp-client.ts +625 -0
  192. package/source/mcp/transport-factory.spec.ts +406 -0
  193. package/source/mcp/transport-factory.ts +312 -0
  194. package/source/message-handler.ts +67 -0
  195. package/source/model-database/database-engine.spec.ts +494 -0
  196. package/source/model-database/database-engine.ts +50 -0
  197. package/source/model-database/model-database.spec.ts +363 -0
  198. package/source/model-database/model-database.ts +91 -0
  199. package/source/model-database/model-engine.spec.ts +447 -0
  200. package/source/model-database/model-engine.ts +65 -0
  201. package/source/model-database/model-fetcher.spec.ts +583 -0
  202. package/source/model-database/model-fetcher.ts +330 -0
  203. package/source/models/index.ts +1 -0
  204. package/source/models/models-cache.spec.ts +214 -0
  205. package/source/models/models-cache.ts +78 -0
  206. package/source/models/models-dev-client.spec.ts +379 -0
  207. package/source/models/models-dev-client.ts +329 -0
  208. package/source/models/models-types.ts +68 -0
  209. package/source/prompt-history.ts +155 -0
  210. package/source/security/command-injection.spec.ts +240 -0
  211. package/source/services/checkpoint-manager.spec.ts +523 -0
  212. package/source/services/checkpoint-manager.ts +466 -0
  213. package/source/services/file-snapshot.spec.ts +569 -0
  214. package/source/services/file-snapshot.ts +220 -0
  215. package/source/test-utils/render-with-theme.tsx +48 -0
  216. package/source/tokenization/index.ts +1 -0
  217. package/source/tokenization/tokenizer-factory.spec.ts +170 -0
  218. package/source/tokenization/tokenizer-factory.ts +125 -0
  219. package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
  220. package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
  221. package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
  222. package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
  223. package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
  224. package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
  225. package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
  226. package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
  227. package/source/tool-calling/index.ts +5 -0
  228. package/source/tool-calling/json-parser.spec.ts +639 -0
  229. package/source/tool-calling/json-parser.ts +247 -0
  230. package/source/tool-calling/tool-parser.spec.ts +395 -0
  231. package/source/tool-calling/tool-parser.ts +120 -0
  232. package/source/tool-calling/xml-parser.spec.ts +662 -0
  233. package/source/tool-calling/xml-parser.ts +289 -0
  234. package/source/tools/execute-bash.spec.tsx +353 -0
  235. package/source/tools/execute-bash.tsx +219 -0
  236. package/source/tools/execute-function.spec.ts +130 -0
  237. package/source/tools/fetch-url.spec.tsx +342 -0
  238. package/source/tools/fetch-url.tsx +172 -0
  239. package/source/tools/find-files.spec.tsx +924 -0
  240. package/source/tools/find-files.tsx +293 -0
  241. package/source/tools/index.ts +102 -0
  242. package/source/tools/lsp-get-diagnostics.tsx +192 -0
  243. package/source/tools/needs-approval.spec.ts +282 -0
  244. package/source/tools/read-file.spec.tsx +801 -0
  245. package/source/tools/read-file.tsx +387 -0
  246. package/source/tools/search-file-contents.spec.tsx +1273 -0
  247. package/source/tools/search-file-contents.tsx +293 -0
  248. package/source/tools/string-replace.spec.tsx +730 -0
  249. package/source/tools/string-replace.tsx +548 -0
  250. package/source/tools/tool-manager.ts +210 -0
  251. package/source/tools/tool-registry.spec.ts +415 -0
  252. package/source/tools/tool-registry.ts +228 -0
  253. package/source/tools/web-search.tsx +223 -0
  254. package/source/tools/write-file.spec.tsx +559 -0
  255. package/source/tools/write-file.tsx +228 -0
  256. package/source/types/app.ts +37 -0
  257. package/source/types/checkpoint.ts +48 -0
  258. package/source/types/commands.ts +46 -0
  259. package/source/types/components.ts +27 -0
  260. package/source/types/config.ts +103 -0
  261. package/source/types/core-connection-status.spec.ts +67 -0
  262. package/source/types/core.ts +181 -0
  263. package/source/types/hooks.ts +50 -0
  264. package/source/types/index.ts +12 -0
  265. package/source/types/markdown-parser.ts +11 -0
  266. package/source/types/mcp.ts +52 -0
  267. package/source/types/system.ts +16 -0
  268. package/source/types/tokenization.ts +41 -0
  269. package/source/types/ui.ts +40 -0
  270. package/source/types/usage.ts +58 -0
  271. package/source/types/utils.ts +16 -0
  272. package/source/usage/calculator.spec.ts +385 -0
  273. package/source/usage/calculator.ts +104 -0
  274. package/source/usage/storage.spec.ts +703 -0
  275. package/source/usage/storage.ts +238 -0
  276. package/source/usage/tracker.spec.ts +456 -0
  277. package/source/usage/tracker.ts +102 -0
  278. package/source/utils/atomic-deletion.spec.ts +194 -0
  279. package/source/utils/atomic-deletion.ts +127 -0
  280. package/source/utils/bounded-map.spec.ts +300 -0
  281. package/source/utils/bounded-map.ts +193 -0
  282. package/source/utils/checkpoint-utils.spec.ts +222 -0
  283. package/source/utils/checkpoint-utils.ts +92 -0
  284. package/source/utils/error-formatter.spec.ts +169 -0
  285. package/source/utils/error-formatter.ts +194 -0
  286. package/source/utils/file-autocomplete.spec.ts +173 -0
  287. package/source/utils/file-autocomplete.ts +196 -0
  288. package/source/utils/file-cache.spec.ts +309 -0
  289. package/source/utils/file-cache.ts +195 -0
  290. package/source/utils/file-content-loader.spec.ts +180 -0
  291. package/source/utils/file-content-loader.ts +179 -0
  292. package/source/utils/file-mention-handler.spec.ts +261 -0
  293. package/source/utils/file-mention-handler.ts +84 -0
  294. package/source/utils/file-mention-parser.spec.ts +182 -0
  295. package/source/utils/file-mention-parser.ts +170 -0
  296. package/source/utils/fuzzy-matching.spec.ts +149 -0
  297. package/source/utils/fuzzy-matching.ts +146 -0
  298. package/source/utils/indentation-normalizer.spec.ts +216 -0
  299. package/source/utils/indentation-normalizer.ts +76 -0
  300. package/source/utils/installation-detector.spec.ts +178 -0
  301. package/source/utils/installation-detector.ts +153 -0
  302. package/source/utils/logging/config.spec.ts +311 -0
  303. package/source/utils/logging/config.ts +210 -0
  304. package/source/utils/logging/console-facade.spec.ts +184 -0
  305. package/source/utils/logging/console-facade.ts +384 -0
  306. package/source/utils/logging/correlation.spec.ts +679 -0
  307. package/source/utils/logging/correlation.ts +474 -0
  308. package/source/utils/logging/formatters.spec.ts +464 -0
  309. package/source/utils/logging/formatters.ts +207 -0
  310. package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
  311. package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
  312. package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
  313. package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
  314. package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
  315. package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
  316. package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
  317. package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
  318. package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
  319. package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
  320. package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
  321. package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
  322. package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
  323. package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
  324. package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
  325. package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
  326. package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
  327. package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
  328. package/source/utils/logging/health-monitor/index.ts +15 -0
  329. package/source/utils/logging/health-monitor/instances.ts +48 -0
  330. package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
  331. package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
  332. package/source/utils/logging/health-monitor/types.ts +126 -0
  333. package/source/utils/logging/index.spec.ts +284 -0
  334. package/source/utils/logging/index.ts +236 -0
  335. package/source/utils/logging/integration.spec.ts +441 -0
  336. package/source/utils/logging/log-method-factory.spec.ts +573 -0
  337. package/source/utils/logging/log-method-factory.ts +233 -0
  338. package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
  339. package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
  340. package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
  341. package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
  342. package/source/utils/logging/log-query/index.ts +23 -0
  343. package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
  344. package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
  345. package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
  346. package/source/utils/logging/log-query/query/query-builder.ts +151 -0
  347. package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
  348. package/source/utils/logging/log-query/query/query-engine.ts +45 -0
  349. package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
  350. package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
  351. package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
  352. package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
  353. package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
  354. package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
  355. package/source/utils/logging/log-query/types.ts +163 -0
  356. package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
  357. package/source/utils/logging/log-query/utils/helpers.ts +72 -0
  358. package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
  359. package/source/utils/logging/log-query/utils/sorting.ts +61 -0
  360. package/source/utils/logging/logger-provider.spec.ts +262 -0
  361. package/source/utils/logging/logger-provider.ts +362 -0
  362. package/source/utils/logging/performance.spec.ts +209 -0
  363. package/source/utils/logging/performance.ts +757 -0
  364. package/source/utils/logging/pino-logger.spec.ts +425 -0
  365. package/source/utils/logging/pino-logger.ts +514 -0
  366. package/source/utils/logging/redaction.spec.ts +490 -0
  367. package/source/utils/logging/redaction.ts +267 -0
  368. package/source/utils/logging/request-tracker.spec.ts +1198 -0
  369. package/source/utils/logging/request-tracker.ts +803 -0
  370. package/source/utils/logging/transports.spec.ts +505 -0
  371. package/source/utils/logging/transports.ts +305 -0
  372. package/source/utils/logging/types.ts +216 -0
  373. package/source/utils/message-builder.spec.ts +179 -0
  374. package/source/utils/message-builder.ts +101 -0
  375. package/source/utils/message-queue.tsx +486 -0
  376. package/source/utils/paste-detection.spec.ts +69 -0
  377. package/source/utils/paste-detection.ts +124 -0
  378. package/source/utils/paste-roundtrip.spec.ts +442 -0
  379. package/source/utils/paste-utils.spec.ts +128 -0
  380. package/source/utils/paste-utils.ts +52 -0
  381. package/source/utils/programming-language-helper.spec.ts +74 -0
  382. package/source/utils/programming-language-helper.ts +32 -0
  383. package/source/utils/prompt-assembly.spec.ts +221 -0
  384. package/source/utils/prompt-processor.ts +173 -0
  385. package/source/utils/tool-args-parser.spec.ts +136 -0
  386. package/source/utils/tool-args-parser.ts +54 -0
  387. package/source/utils/tool-cancellation.spec.ts +230 -0
  388. package/source/utils/tool-cancellation.ts +28 -0
  389. package/source/utils/tool-result-display.spec.tsx +469 -0
  390. package/source/utils/tool-result-display.tsx +90 -0
  391. package/source/utils/update-checker.spec.ts +383 -0
  392. package/source/utils/update-checker.ts +183 -0
  393. package/source/wizard/config-wizard.spec.tsx +103 -0
  394. package/source/wizard/config-wizard.tsx +382 -0
  395. package/source/wizard/steps/location-step.spec.tsx +186 -0
  396. package/source/wizard/steps/location-step.tsx +147 -0
  397. package/source/wizard/steps/mcp-step.spec.tsx +607 -0
  398. package/source/wizard/steps/mcp-step.tsx +632 -0
  399. package/source/wizard/steps/provider-step.spec.tsx +342 -0
  400. package/source/wizard/steps/provider-step.tsx +957 -0
  401. package/source/wizard/steps/summary-step.spec.tsx +749 -0
  402. package/source/wizard/steps/summary-step.tsx +228 -0
  403. package/source/wizard/templates/mcp-templates.spec.ts +613 -0
  404. package/source/wizard/templates/mcp-templates.ts +570 -0
  405. package/source/wizard/templates/provider-templates.spec.ts +152 -0
  406. package/source/wizard/templates/provider-templates.ts +485 -0
  407. package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
  408. package/source/wizard/utils/fetch-cloud-models.ts +223 -0
  409. package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
  410. package/source/wizard/utils/fetch-local-models.ts +192 -0
  411. package/source/wizard/validation-array.spec.ts +264 -0
  412. package/source/wizard/validation.spec.ts +373 -0
  413. package/source/wizard/validation.ts +232 -0
  414. package/source/app/prompts/main-prompt.md +0 -122
@@ -0,0 +1,247 @@
1
+ import type {ToolCall} from '@/types/index';
2
+
3
+ /**
4
+ * Internal JSON tool call parser
5
+ * Note: This is now an internal utility. Use tool-parser.ts for public API.
6
+ */
7
+
8
+ /**
9
+ * Detects malformed JSON tool call attempts and returns error details
10
+ * Returns null if no malformed tool calls detected
11
+ */
12
+ export function detectMalformedJSONToolCall(
13
+ content: string,
14
+ ): {error: string; examples: string} | null {
15
+ // Check for incomplete JSON structures
16
+ const patterns = [
17
+ {
18
+ // Incomplete JSON with name but missing arguments
19
+ regex: /\{\s*"name"\s*:\s*"[^"]+"\s*,?\s*\}/,
20
+ error: 'Incomplete tool call: missing "arguments" field',
21
+ hint: 'Tool calls must include both "name" and "arguments" fields',
22
+ },
23
+ {
24
+ // Incomplete JSON with arguments but missing name
25
+ regex: /\{\s*"arguments"\s*:\s*\{[^}]*\}\s*\}/,
26
+ error: 'Incomplete tool call: missing "name" field',
27
+ hint: 'Tool calls must include both "name" and "arguments" fields',
28
+ },
29
+ {
30
+ // Malformed arguments (not an object)
31
+ regex: /\{\s*"name"\s*:\s*"[^"]+"\s*,\s*"arguments"\s*:\s*"[^"]*"\s*\}/,
32
+ error: 'Invalid tool call: "arguments" must be an object, not a string',
33
+ hint: 'Use {"name": "tool_name", "arguments": {...}} format',
34
+ },
35
+ ];
36
+
37
+ for (const pattern of patterns) {
38
+ const match = content.match(pattern.regex);
39
+ if (match) {
40
+ return {
41
+ error: pattern.error,
42
+ examples: getCorrectJSONFormatExamples(pattern.hint),
43
+ };
44
+ }
45
+ }
46
+
47
+ return null;
48
+ }
49
+
50
+ /**
51
+ * Generates correct format examples for JSON error messages
52
+ */
53
+ function getCorrectJSONFormatExamples(_specificHint: string): string {
54
+ return `Please use the native tool calling format provided by the system. The tools are already available to you - call them directly using the function calling interface.`;
55
+ }
56
+
57
+ /**
58
+ * Parses JSON-formatted tool calls from content
59
+ * This is an internal function - use tool-parser.ts for public API
60
+ */
61
+ export function parseJSONToolCalls(content: string): ToolCall[] {
62
+ const extractedCalls: ToolCall[] = [];
63
+ let trimmedContent = content.trim();
64
+
65
+ // Handle markdown code blocks
66
+ const codeBlockMatch = trimmedContent.match(
67
+ /^```(?:json)?\s*\n?([\s\S]*?)\n?```$/,
68
+ );
69
+ if (codeBlockMatch && codeBlockMatch[1]) {
70
+ trimmedContent = codeBlockMatch[1].trim();
71
+ }
72
+
73
+ // Try to parse entire content as single JSON tool call
74
+ if (trimmedContent.startsWith('{') && trimmedContent.endsWith('}')) {
75
+ // Skip empty or nearly empty JSON objects
76
+ if (trimmedContent === '{}' || trimmedContent.replace(/\s/g, '') === '{}') {
77
+ return extractedCalls;
78
+ }
79
+
80
+ try {
81
+ const parsed = JSON.parse(trimmedContent) as {
82
+ name?: string;
83
+ arguments?: Record<string, unknown>;
84
+ };
85
+
86
+ if (parsed.name && parsed.arguments !== undefined) {
87
+ const toolCall = {
88
+ id: `call_${Date.now()}`,
89
+ function: {
90
+ name: parsed.name || '',
91
+ arguments: parsed.arguments || {},
92
+ },
93
+ };
94
+ extractedCalls.push(toolCall);
95
+ return extractedCalls;
96
+ }
97
+ } catch {
98
+ // Failed to parse - will be caught by malformed detection
99
+ }
100
+ }
101
+
102
+ // Look for standalone JSON blocks in the content (multiline without code blocks)
103
+ const jsonBlockRegex =
104
+ /\{\s*\n\s*"name":\s*"([^"]+)",\s*\n\s*"arguments":\s*\{[\s\S]*?\}\s*\n\s*\}/g;
105
+ let jsonMatch;
106
+ while ((jsonMatch = jsonBlockRegex.exec(content)) !== null) {
107
+ try {
108
+ const parsed = JSON.parse(jsonMatch[0]) as {
109
+ name?: string;
110
+ arguments?: Record<string, unknown>;
111
+ };
112
+ if (parsed.name && parsed.arguments !== undefined) {
113
+ const toolCall = {
114
+ id: `call_${Date.now()}_${extractedCalls.length}`,
115
+ function: {
116
+ name: parsed.name || '',
117
+ arguments: parsed.arguments || {},
118
+ },
119
+ };
120
+ extractedCalls.push(toolCall);
121
+ }
122
+ } catch {
123
+ // Failed to parse - will be caught by malformed detection
124
+ }
125
+ }
126
+
127
+ // Look for embedded tool calls using regex patterns
128
+ const toolCallPatterns = [
129
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*(\{[^}]*\})\}/g,
130
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*(\{[^}]+\})\}/g,
131
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*"([^"]+)"\}/g,
132
+ ];
133
+
134
+ for (const pattern of toolCallPatterns) {
135
+ let match;
136
+ while ((match = pattern.exec(content)) !== null) {
137
+ const [, name, argsStr] = match;
138
+ try {
139
+ let args: Record<string, unknown> | string;
140
+ if (argsStr && argsStr.startsWith('{')) {
141
+ args = JSON.parse(argsStr || '{}') as Record<string, unknown>;
142
+ } else {
143
+ args = argsStr || '';
144
+ }
145
+ extractedCalls.push({
146
+ id: `call_${Date.now()}_${extractedCalls.length}`,
147
+ function: {
148
+ name: name || '',
149
+ arguments: args as Record<string, unknown>,
150
+ },
151
+ });
152
+ } catch {
153
+ // Failed to parse - will be caught by malformed detection
154
+ }
155
+ }
156
+ }
157
+
158
+ // Deduplicate identical tool calls
159
+ const uniqueCalls = deduplicateToolCalls(extractedCalls);
160
+
161
+ return uniqueCalls;
162
+ }
163
+
164
+ function deduplicateToolCalls(toolCalls: ToolCall[]): ToolCall[] {
165
+ const seen = new Set<string>();
166
+ const unique: ToolCall[] = [];
167
+
168
+ for (const call of toolCalls) {
169
+ // Create a hash of the function name and arguments for comparison
170
+ const hash = `${call.function.name}:${JSON.stringify(
171
+ call.function.arguments,
172
+ )}`;
173
+
174
+ if (!seen.has(hash)) {
175
+ seen.add(hash);
176
+ unique.push(call);
177
+ }
178
+ // Else: duplicate, skip it
179
+ }
180
+
181
+ return unique;
182
+ }
183
+
184
+ /**
185
+ * Cleans content by removing tool call JSON blocks
186
+ * This is an internal function - use tool-parser.ts for public API
187
+ */
188
+ export function cleanJSONToolCalls(
189
+ content: string,
190
+ toolCalls: ToolCall[],
191
+ ): string {
192
+ if (toolCalls.length === 0) return content;
193
+
194
+ let cleanedContent = content;
195
+
196
+ // Handle markdown code blocks that contain only tool calls
197
+ const codeBlockRegex = /```(?:json)?\s*\n?([\s\S]*?)\n?```/g;
198
+ cleanedContent = cleanedContent.replace(
199
+ codeBlockRegex,
200
+ (match, blockContent: string) => {
201
+ const trimmedBlock = blockContent.trim();
202
+
203
+ // Check if this block contains a tool call that we parsed
204
+ try {
205
+ const parsed = JSON.parse(trimmedBlock) as {
206
+ name?: string;
207
+ arguments?: unknown;
208
+ };
209
+ if (parsed.name && parsed.arguments !== undefined) {
210
+ // This code block contains only a tool call, remove the entire block
211
+ return '';
212
+ }
213
+ } catch {
214
+ // Not valid JSON, keep the code block
215
+ }
216
+
217
+ // Keep the code block as-is if it doesn't contain a tool call
218
+ return match;
219
+ },
220
+ );
221
+
222
+ // Remove JSON blocks that were parsed as tool calls (for non-code-block cases)
223
+ const toolCallPatterns = [
224
+ /\{\s*\n\s*"name":\s*"([^"]+)",\s*\n\s*"arguments":\s*\{[\s\S]*?\}\s*\n\s*\}/g, // Multiline JSON blocks
225
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*(\{[^}]*\})\}/g,
226
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*(\{[^}]+\})\}/g,
227
+ /\{"name":\s*"([^"]+)",\s*"arguments":\s*"([^"]+)"\}/g,
228
+ ];
229
+
230
+ for (const pattern of toolCallPatterns) {
231
+ cleanedContent = cleanedContent.replace(pattern, '');
232
+ }
233
+
234
+ // Clean up whitespace artifacts left by removed tool calls
235
+ cleanedContent = cleanedContent
236
+ // Remove trailing whitespace from each line
237
+ .replace(/[ \t]+$/gm, '')
238
+ // Collapse multiple spaces (but not at start of line for indentation)
239
+ .replace(/([^ \t\n]) {2,}/g, '$1 ')
240
+ // Remove lines that are only whitespace
241
+ .replace(/^[ \t]+$/gm, '')
242
+ // Collapse 2+ consecutive blank lines to a single blank line
243
+ .replace(/\n{3,}/g, '\n\n')
244
+ .trim();
245
+
246
+ return cleanedContent;
247
+ }
@@ -0,0 +1,395 @@
1
+ import test from 'ava';
2
+ import {parseToolCalls} from './tool-parser';
3
+
4
+ console.log(`\ntool-parser.spec.ts`);
5
+
6
+ // XML Parser Tests
7
+
8
+ test('parseToolCalls: successfully parses valid XML tool call', t => {
9
+ const content = `
10
+ <read_file>
11
+ <path>/path/to/file.txt</path>
12
+ </read_file>
13
+ `;
14
+
15
+ const result = parseToolCalls(content);
16
+
17
+ t.true(result.success);
18
+ if (result.success) {
19
+ t.is(result.toolCalls.length, 1);
20
+ t.is(result.toolCalls[0].function.name, 'read_file');
21
+ t.deepEqual(result.toolCalls[0].function.arguments, {
22
+ path: '/path/to/file.txt',
23
+ });
24
+ }
25
+ });
26
+
27
+ test('parseToolCalls: detects malformed XML with attribute syntax', t => {
28
+ const content = `
29
+ <function=read_file>
30
+ <parameter=path>/path/to/file.txt</parameter>
31
+ </function>
32
+
33
+ I want to read the file at /path/to/file.txt
34
+ `;
35
+
36
+ const result = parseToolCalls(content);
37
+
38
+ t.false(result.success);
39
+ if (!result.success) {
40
+ t.regex(result.error, /Invalid syntax/i);
41
+ t.regex(result.examples, /native tool calling/i);
42
+ }
43
+ });
44
+
45
+ test('parseToolCalls: handles multiple valid XML tool calls', t => {
46
+ const content = `
47
+ <read_file>
48
+ <path>/path/to/file1.txt</path>
49
+ </read_file>
50
+
51
+ <create_file>
52
+ <path>/path/to/file2.txt</path>
53
+ <content>Hello world</content>
54
+ </create_file>
55
+ `;
56
+
57
+ const result = parseToolCalls(content);
58
+
59
+ t.true(result.success);
60
+ if (result.success) {
61
+ t.is(result.toolCalls.length, 2);
62
+ t.is(result.toolCalls[0].function.name, 'read_file');
63
+ t.is(result.toolCalls[1].function.name, 'create_file');
64
+ }
65
+ });
66
+
67
+ test('parseToolCalls: cleans XML tool calls from content', t => {
68
+ const content = `
69
+ Here is some text before the tool call.
70
+
71
+ <read_file>
72
+ <path>/path/to/file.txt</path>
73
+ </read_file>
74
+
75
+ And some text after.
76
+ `;
77
+
78
+ const result = parseToolCalls(content);
79
+
80
+ t.true(result.success);
81
+ if (result.success) {
82
+ t.is(result.toolCalls.length, 1);
83
+ t.regex(result.cleanedContent, /Here is some text before/);
84
+ t.regex(result.cleanedContent, /And some text after/);
85
+ t.notRegex(result.cleanedContent, /<read_file>/);
86
+ }
87
+ });
88
+
89
+ // JSON Parser Tests
90
+
91
+ test('parseToolCalls: successfully parses valid JSON tool call', t => {
92
+ const content = `
93
+ {
94
+ "name": "read_file",
95
+ "arguments": {
96
+ "path": "/path/to/file.txt"
97
+ }
98
+ }
99
+ `;
100
+
101
+ const result = parseToolCalls(content);
102
+
103
+ t.true(result.success);
104
+ if (result.success) {
105
+ t.is(result.toolCalls.length, 1);
106
+ t.is(result.toolCalls[0].function.name, 'read_file');
107
+ t.deepEqual(result.toolCalls[0].function.arguments, {
108
+ path: '/path/to/file.txt',
109
+ });
110
+ }
111
+ });
112
+
113
+ test('parseToolCalls: detects malformed JSON missing arguments field', t => {
114
+ const content = `
115
+ {
116
+ "name": "read_file"
117
+ }
118
+ `;
119
+
120
+ const result = parseToolCalls(content);
121
+
122
+ t.false(result.success);
123
+ if (!result.success) {
124
+ t.regex(result.error, /missing "arguments" field/i);
125
+ t.regex(result.examples, /native tool calling format/i);
126
+ }
127
+ });
128
+
129
+ test('parseToolCalls: detects malformed JSON missing name field', t => {
130
+ const content = `
131
+ {
132
+ "arguments": {
133
+ "path": "/path/to/file.txt"
134
+ }
135
+ }
136
+ `;
137
+
138
+ const result = parseToolCalls(content);
139
+
140
+ t.false(result.success);
141
+ if (!result.success) {
142
+ t.regex(result.error, /missing "name" field/i);
143
+ t.regex(result.examples, /native tool calling format/i);
144
+ }
145
+ });
146
+
147
+ test('parseToolCalls: detects malformed JSON with string arguments', t => {
148
+ const content = `
149
+ {
150
+ "name": "read_file",
151
+ "arguments": "/path/to/file.txt"
152
+ }
153
+ `;
154
+
155
+ const result = parseToolCalls(content);
156
+
157
+ t.false(result.success);
158
+ if (!result.success) {
159
+ t.regex(result.error, /"arguments" must be an object/i);
160
+ t.regex(result.examples, /native tool calling format/i);
161
+ }
162
+ });
163
+
164
+ test('parseToolCalls: handles JSON in markdown code blocks', t => {
165
+ const content = `
166
+ \`\`\`json
167
+ {
168
+ "name": "read_file",
169
+ "arguments": {
170
+ "path": "/path/to/file.txt"
171
+ }
172
+ }
173
+ \`\`\`
174
+ `;
175
+
176
+ const result = parseToolCalls(content);
177
+
178
+ t.true(result.success);
179
+ if (result.success) {
180
+ t.is(result.toolCalls.length, 1);
181
+ t.is(result.toolCalls[0].function.name, 'read_file');
182
+ }
183
+ });
184
+
185
+ test('parseToolCalls: cleans JSON tool calls from content', t => {
186
+ const content = `
187
+ Here is some text before.
188
+
189
+ {
190
+ "name": "read_file",
191
+ "arguments": {
192
+ "path": "/path/to/file.txt"
193
+ }
194
+ }
195
+
196
+ And some text after.
197
+ `;
198
+
199
+ const result = parseToolCalls(content);
200
+
201
+ t.true(result.success);
202
+ if (result.success) {
203
+ t.is(result.toolCalls.length, 1);
204
+ t.regex(result.cleanedContent, /Here is some text before/);
205
+ t.regex(result.cleanedContent, /And some text after/);
206
+ t.notRegex(result.cleanedContent, /"name":\s*"read_file"/);
207
+ }
208
+ });
209
+
210
+ // Priority Tests (XML should be tried first)
211
+
212
+ test('parseToolCalls: prioritizes XML over JSON when both present', t => {
213
+ const content = `
214
+ <read_file>
215
+ <path>/xml/path.txt</path>
216
+ </read_file>
217
+
218
+ {
219
+ "name": "read_file",
220
+ "arguments": {
221
+ "path": "/json/path.txt"
222
+ }
223
+ }
224
+ `;
225
+
226
+ const result = parseToolCalls(content);
227
+
228
+ t.true(result.success);
229
+ if (result.success) {
230
+ // Should parse XML first and return immediately
231
+ t.is(result.toolCalls.length, 1);
232
+ t.is(result.toolCalls[0].function.name, 'read_file');
233
+ // Verify it's the XML call (xml_call_ prefix)
234
+ t.regex(result.toolCalls[0].id, /xml_call/);
235
+ t.is(
236
+ (result.toolCalls[0].function.arguments as {path: string}).path,
237
+ '/xml/path.txt',
238
+ );
239
+ }
240
+ });
241
+
242
+ // Edge Cases
243
+
244
+ test('parseToolCalls: handles empty content', t => {
245
+ const result = parseToolCalls('');
246
+
247
+ t.true(result.success);
248
+ if (result.success) {
249
+ t.is(result.toolCalls.length, 0);
250
+ t.is(result.cleanedContent, '');
251
+ }
252
+ });
253
+
254
+ test('parseToolCalls: handles content with no tool calls', t => {
255
+ const content = 'Just some plain text without any tool calls.';
256
+
257
+ const result = parseToolCalls(content);
258
+
259
+ t.true(result.success);
260
+ if (result.success) {
261
+ t.is(result.toolCalls.length, 0);
262
+ t.is(result.cleanedContent, content);
263
+ }
264
+ });
265
+
266
+ test('parseToolCalls: handles empty JSON object', t => {
267
+ const content = '{}';
268
+
269
+ const result = parseToolCalls(content);
270
+
271
+ t.true(result.success);
272
+ if (result.success) {
273
+ t.is(result.toolCalls.length, 0);
274
+ }
275
+ });
276
+
277
+ test('parseToolCalls: deduplicates identical tool calls', t => {
278
+ const content = `
279
+ {
280
+ "name": "read_file",
281
+ "arguments": {
282
+ "path": "/path/to/file.txt"
283
+ }
284
+ }
285
+
286
+ {
287
+ "name": "read_file",
288
+ "arguments": {
289
+ "path": "/path/to/file.txt"
290
+ }
291
+ }
292
+ `;
293
+
294
+ const result = parseToolCalls(content);
295
+
296
+ t.true(result.success);
297
+ if (result.success) {
298
+ // Should deduplicate to single call
299
+ t.is(result.toolCalls.length, 1);
300
+ }
301
+ });
302
+
303
+ // Think Tag Tests (models like GLM-4 emit these for chain-of-thought)
304
+
305
+ test('parseToolCalls: strips complete <think>...</think> tags', t => {
306
+ const content = `<think>
307
+ Let me think about this...
308
+ I should read the file first.
309
+ </think>
310
+
311
+ Here is my response to your question.`;
312
+
313
+ const result = parseToolCalls(content);
314
+
315
+ t.true(result.success);
316
+ if (result.success) {
317
+ t.is(result.toolCalls.length, 0);
318
+ t.notRegex(result.cleanedContent, /<think>/);
319
+ t.notRegex(result.cleanedContent, /<\/think>/);
320
+ t.regex(result.cleanedContent, /Here is my response/);
321
+ }
322
+ });
323
+
324
+ test('parseToolCalls: strips orphaned closing </think> tags', t => {
325
+ const content = `</think>
326
+
327
+ Here is my response after some thinking.`;
328
+
329
+ const result = parseToolCalls(content);
330
+
331
+ t.true(result.success);
332
+ if (result.success) {
333
+ t.notRegex(result.cleanedContent, /<\/think>/);
334
+ t.regex(result.cleanedContent, /Here is my response/);
335
+ }
336
+ });
337
+
338
+ test('parseToolCalls: strips incomplete opening <think> tags (streaming)', t => {
339
+ const content = `Here is my response.
340
+
341
+ <think>
342
+ I'm still thinking about this...`;
343
+
344
+ const result = parseToolCalls(content);
345
+
346
+ t.true(result.success);
347
+ if (result.success) {
348
+ t.notRegex(result.cleanedContent, /<think>/);
349
+ t.regex(result.cleanedContent, /Here is my response/);
350
+ // The incomplete thinking content should be removed
351
+ t.notRegex(result.cleanedContent, /still thinking/);
352
+ }
353
+ });
354
+
355
+ test('parseToolCalls: handles think tags with tool calls', t => {
356
+ const content = `<think>
357
+ Let me analyze this request...
358
+ I'll need to read the file first.
359
+ </think>
360
+
361
+ <read_file>
362
+ <path>/path/to/file.txt</path>
363
+ </read_file>`;
364
+
365
+ const result = parseToolCalls(content);
366
+
367
+ t.true(result.success);
368
+ if (result.success) {
369
+ t.is(result.toolCalls.length, 1);
370
+ t.is(result.toolCalls[0].function.name, 'read_file');
371
+ t.notRegex(result.cleanedContent, /<think>/);
372
+ t.notRegex(result.cleanedContent, /<\/think>/);
373
+ }
374
+ });
375
+
376
+ test('parseToolCalls: handles case-insensitive think tags', t => {
377
+ const content = `<THINK>
378
+ Some thinking...
379
+ </THINK>
380
+
381
+ <Think>
382
+ More thinking...
383
+ </Think>
384
+
385
+ The actual response.`;
386
+
387
+ const result = parseToolCalls(content);
388
+
389
+ t.true(result.success);
390
+ if (result.success) {
391
+ t.notRegex(result.cleanedContent, /<think>/i);
392
+ t.notRegex(result.cleanedContent, /<\/think>/i);
393
+ t.regex(result.cleanedContent, /The actual response/);
394
+ }
395
+ });