@tyvm/knowhow 0.0.47 → 0.0.49

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 (459) hide show
  1. package/benchmarks/results/27b0a06/2025-09-27/xai/xai-grok-code-fast-1.json +2909 -0
  2. package/benchmarks/results/4057aed/2025-08-14/anthropic/anthropic-claude-sonnet-4-20250514.json +1671 -0
  3. package/jest.config.js +2 -2
  4. package/package.json +8 -3
  5. package/src/agents/base/base.ts +39 -25
  6. package/src/agents/patcher/patcher.ts +26 -5
  7. package/src/agents/tools/agentCall.ts +4 -2
  8. package/src/agents/tools/aiClient.ts +3 -11
  9. package/src/agents/tools/ast/astAppendNode.ts +90 -0
  10. package/src/agents/tools/ast/astDeleteNode.ts +88 -0
  11. package/src/agents/tools/ast/astEditNode.ts +95 -0
  12. package/src/agents/tools/ast/astGetPathForLine.ts +73 -0
  13. package/src/agents/tools/ast/astListPaths.ts +66 -0
  14. package/src/agents/tools/ast/index.ts +7 -0
  15. package/src/agents/tools/callPlugin.ts +8 -2
  16. package/src/agents/tools/embeddingSearch.ts +2 -1
  17. package/src/agents/tools/execCommand.ts +239 -94
  18. package/src/agents/tools/fileSearch.ts +15 -17
  19. package/src/agents/tools/index.ts +1 -0
  20. package/src/agents/tools/language/definitions.ts +10 -2
  21. package/src/agents/tools/language/index.ts +3 -2
  22. package/src/agents/tools/lintFile.ts +4 -2
  23. package/src/agents/tools/list.ts +203 -62
  24. package/src/agents/tools/patch.ts +48 -14
  25. package/src/agents/tools/readBlocks.ts +34 -0
  26. package/src/agents/tools/readFile.ts +23 -0
  27. package/src/agents/tools/stringReplace.ts +33 -9
  28. package/src/agents/tools/writeFile.ts +55 -0
  29. package/src/agents/tools/ycmd/server.ts +14 -4
  30. package/src/chat/CliChatService.ts +6 -1
  31. package/src/chat/modules/AgentModule.ts +129 -67
  32. package/src/chat/modules/AskModule.ts +0 -1
  33. package/src/chat/modules/SetupModule.ts +4 -4
  34. package/src/chat/modules/SystemModule.ts +28 -5
  35. package/src/chat/types.ts +2 -0
  36. package/src/chat-old.ts +2 -2
  37. package/src/clients/anthropic.ts +22 -1
  38. package/src/clients/openai.ts +1 -1
  39. package/src/clients/xai.ts +15 -5
  40. package/src/config.ts +17 -5
  41. package/src/dataset/diffs/generate.ts +2 -2
  42. package/src/dataset/diffs/jsonl.ts +0 -1
  43. package/src/embeddings.ts +8 -8
  44. package/src/index.ts +11 -5
  45. package/src/plugins/GitPlugin.ts +530 -0
  46. package/src/plugins/LinterPlugin.ts +89 -0
  47. package/src/plugins/PluginBase.ts +4 -2
  48. package/src/plugins/asana.ts +4 -2
  49. package/src/plugins/downloader/plugin.ts +5 -2
  50. package/src/plugins/embedding.ts +24 -4
  51. package/src/plugins/figma.ts +7 -3
  52. package/src/plugins/github.ts +4 -2
  53. package/src/plugins/jira.ts +4 -2
  54. package/src/plugins/language.ts +134 -27
  55. package/src/plugins/linear.ts +4 -2
  56. package/src/plugins/notion.ts +4 -2
  57. package/src/plugins/plugins.ts +27 -16
  58. package/src/plugins/tree-sitter/editor.ts +369 -0
  59. package/src/plugins/tree-sitter/lang-packs/index.ts +23 -0
  60. package/src/plugins/tree-sitter/lang-packs/java.ts +59 -0
  61. package/src/plugins/tree-sitter/lang-packs/javascript.ts +57 -0
  62. package/src/plugins/tree-sitter/lang-packs/python.ts +45 -0
  63. package/src/plugins/tree-sitter/lang-packs/types.ts +79 -0
  64. package/src/plugins/tree-sitter/lang-packs/typescript.ts +49 -0
  65. package/src/plugins/tree-sitter/parser.ts +444 -0
  66. package/src/plugins/tree-sitter/simple-paths.ts +467 -0
  67. package/src/plugins/types.ts +11 -0
  68. package/src/plugins/url.ts +5 -3
  69. package/src/plugins/vim.ts +8 -5
  70. package/src/processors/CustomVariables.ts +60 -70
  71. package/src/processors/TokenCompressor.ts +15 -14
  72. package/src/processors/ToolResponseCache.ts +20 -14
  73. package/src/services/EmbeddingService.ts +18 -9
  74. package/src/services/EventService.ts +80 -0
  75. package/src/services/Mcp.ts +5 -0
  76. package/src/services/S3.ts +4 -3
  77. package/src/services/Tools.ts +125 -53
  78. package/src/services/index.ts +16 -11
  79. package/src/services/types.ts +3 -3
  80. package/src/types.ts +7 -2
  81. package/src/worker.ts +14 -1
  82. package/test-comprehensive.ts +31 -0
  83. package/tests/clients/AIClient.test.ts +490 -0
  84. package/tests/manual/agent-events/run-test.ts +203 -0
  85. package/tests/{integration → manual/file-edits}/figma.test.ts +1 -1
  86. package/tests/{integration → manual/file-edits}/fileblocks/readwrite.test.ts +7 -3
  87. package/tests/{integration → manual/file-edits}/patching.test.ts +11 -8
  88. package/tests/plugins/language/languagePlugin-content-triggers.test.ts +332 -0
  89. package/tests/plugins/language/languagePlugin-integration.test.ts +456 -0
  90. package/tests/plugins/language/languagePlugin.test.ts +363 -0
  91. package/tests/processors/Base64ImageDetector.test.ts +403 -0
  92. package/tests/processors/CustomVariables.test.ts +485 -0
  93. package/tests/processors/HarmonyToolProcessor.test.ts +490 -0
  94. package/tests/processors/TokenCompressor.test.ts +390 -0
  95. package/tests/processors/ToolResponseCache.test.ts +736 -0
  96. package/tests/services/Tools.test.ts +1339 -0
  97. package/tests/test.spec.ts +162 -117
  98. package/tests/tree-sitter/editor.test.ts +113 -0
  99. package/tests/tree-sitter/invalid.test.ts +299 -0
  100. package/tests/tree-sitter/paths/common-edits.test.ts +564 -0
  101. package/tests/tree-sitter/paths/debug-exact-position.test.ts +44 -0
  102. package/tests/tree-sitter/paths/debug-line-indexing.test.ts +49 -0
  103. package/tests/tree-sitter/paths/debug-paths.test.ts +90 -0
  104. package/tests/tree-sitter/paths/paths.test.ts +170 -0
  105. package/tests/tree-sitter/paths/simple-paths.test.ts +367 -0
  106. package/tests/tree-sitter/sample-after.ts +48 -0
  107. package/tests/tree-sitter/sample-before.ts +25 -0
  108. package/tests/tree-sitter/test-files/completely-broken.ts +7 -0
  109. package/tests/tree-sitter/test-files/duplicate-braces.ts +39 -0
  110. package/tests/tree-sitter/test-files/invalid-nesting.ts +39 -0
  111. package/tests/tree-sitter/test-files/malformed-signature.ts +39 -0
  112. package/tests/tree-sitter/test-files/mismatched-parens.ts +39 -0
  113. package/tests/tree-sitter/test-files/missing-semicolon.ts +39 -0
  114. package/tests/tree-sitter/test-files/partially-broken.ts +20 -0
  115. package/tests/tree-sitter/test-files/specific-errors.ts +14 -0
  116. package/tests/tree-sitter/test-files/unclosed-string.ts +39 -0
  117. package/tests/tree-sitter/tree-sitter.test.ts +251 -0
  118. package/ts_build/package.json +8 -3
  119. package/ts_build/src/agents/base/base.d.ts +7 -2
  120. package/ts_build/src/agents/base/base.js +27 -21
  121. package/ts_build/src/agents/base/base.js.map +1 -1
  122. package/ts_build/src/agents/patcher/patcher.js +26 -5
  123. package/ts_build/src/agents/patcher/patcher.js.map +1 -1
  124. package/ts_build/src/agents/tools/agentCall.js +2 -1
  125. package/ts_build/src/agents/tools/agentCall.js.map +1 -1
  126. package/ts_build/src/agents/tools/aiClient.d.ts +7 -8
  127. package/ts_build/src/agents/tools/aiClient.js.map +1 -1
  128. package/ts_build/src/agents/tools/ast/astAppendNode.d.ts +1 -0
  129. package/ts_build/src/agents/tools/ast/astAppendNode.js +96 -0
  130. package/ts_build/src/agents/tools/ast/astAppendNode.js.map +1 -0
  131. package/ts_build/src/agents/tools/ast/astDeleteNode.d.ts +1 -0
  132. package/ts_build/src/agents/tools/ast/astDeleteNode.js +94 -0
  133. package/ts_build/src/agents/tools/ast/astDeleteNode.js.map +1 -0
  134. package/ts_build/src/agents/tools/ast/astEditNode.d.ts +1 -0
  135. package/ts_build/src/agents/tools/ast/astEditNode.js +96 -0
  136. package/ts_build/src/agents/tools/ast/astEditNode.js.map +1 -0
  137. package/ts_build/src/agents/tools/ast/astGetPathForLine.d.ts +1 -0
  138. package/ts_build/src/agents/tools/ast/astGetPathForLine.js +78 -0
  139. package/ts_build/src/agents/tools/ast/astGetPathForLine.js.map +1 -0
  140. package/ts_build/src/agents/tools/ast/astListPaths.d.ts +1 -0
  141. package/ts_build/src/agents/tools/ast/astListPaths.js +78 -0
  142. package/ts_build/src/agents/tools/ast/astListPaths.js.map +1 -0
  143. package/ts_build/src/agents/tools/ast/index.d.ts +5 -0
  144. package/ts_build/src/agents/tools/ast/index.js +14 -0
  145. package/ts_build/src/agents/tools/ast/index.js.map +1 -0
  146. package/ts_build/src/agents/tools/astAppendNode.d.ts +1 -0
  147. package/ts_build/src/agents/tools/astAppendNode.js +98 -0
  148. package/ts_build/src/agents/tools/astAppendNode.js.map +1 -0
  149. package/ts_build/src/agents/tools/astDeleteNode.d.ts +1 -0
  150. package/ts_build/src/agents/tools/astDeleteNode.js +95 -0
  151. package/ts_build/src/agents/tools/astDeleteNode.js.map +1 -0
  152. package/ts_build/src/agents/tools/astEditNode.d.ts +1 -0
  153. package/ts_build/src/agents/tools/astEditNode.js +98 -0
  154. package/ts_build/src/agents/tools/astEditNode.js.map +1 -0
  155. package/ts_build/src/agents/tools/astGetPathForLine.d.ts +1 -0
  156. package/ts_build/src/agents/tools/astGetPathForLine.js +89 -0
  157. package/ts_build/src/agents/tools/astGetPathForLine.js.map +1 -0
  158. package/ts_build/src/agents/tools/astListPaths.d.ts +1 -0
  159. package/ts_build/src/agents/tools/astListPaths.js +82 -0
  160. package/ts_build/src/agents/tools/astListPaths.js.map +1 -0
  161. package/ts_build/src/agents/tools/callPlugin.js +4 -2
  162. package/ts_build/src/agents/tools/callPlugin.js.map +1 -1
  163. package/ts_build/src/agents/tools/embeddingSearch.js +3 -2
  164. package/ts_build/src/agents/tools/embeddingSearch.js.map +1 -1
  165. package/ts_build/src/agents/tools/execCommand.d.ts +2 -2
  166. package/ts_build/src/agents/tools/execCommand.js +201 -67
  167. package/ts_build/src/agents/tools/execCommand.js.map +1 -1
  168. package/ts_build/src/agents/tools/fileSearch.d.ts +1 -1
  169. package/ts_build/src/agents/tools/fileSearch.js +11 -15
  170. package/ts_build/src/agents/tools/fileSearch.js.map +1 -1
  171. package/ts_build/src/agents/tools/github/index.d.ts +1 -1
  172. package/ts_build/src/agents/tools/index.d.ts +1 -0
  173. package/ts_build/src/agents/tools/index.js +1 -0
  174. package/ts_build/src/agents/tools/index.js.map +1 -1
  175. package/ts_build/src/agents/tools/language/definitions.js +11 -2
  176. package/ts_build/src/agents/tools/language/definitions.js.map +1 -1
  177. package/ts_build/src/agents/tools/language/index.js +4 -3
  178. package/ts_build/src/agents/tools/language/index.js.map +1 -1
  179. package/ts_build/src/agents/tools/lintFile.js +4 -2
  180. package/ts_build/src/agents/tools/lintFile.js.map +1 -1
  181. package/ts_build/src/agents/tools/list.js +185 -49
  182. package/ts_build/src/agents/tools/list.js.map +1 -1
  183. package/ts_build/src/agents/tools/patch.js +33 -10
  184. package/ts_build/src/agents/tools/patch.js.map +1 -1
  185. package/ts_build/src/agents/tools/readBlocks.js +23 -0
  186. package/ts_build/src/agents/tools/readBlocks.js.map +1 -1
  187. package/ts_build/src/agents/tools/readFile.js +14 -0
  188. package/ts_build/src/agents/tools/readFile.js.map +1 -1
  189. package/ts_build/src/agents/tools/stringReplace.js +19 -2
  190. package/ts_build/src/agents/tools/stringReplace.js.map +1 -1
  191. package/ts_build/src/agents/tools/writeFile.js +40 -0
  192. package/ts_build/src/agents/tools/writeFile.js.map +1 -1
  193. package/ts_build/src/agents/tools/ycmd/server.js +5 -0
  194. package/ts_build/src/agents/tools/ycmd/server.js.map +1 -1
  195. package/ts_build/src/chat/CliChatService.d.ts +1 -0
  196. package/ts_build/src/chat/CliChatService.js +6 -2
  197. package/ts_build/src/chat/CliChatService.js.map +1 -1
  198. package/ts_build/src/chat/modules/AgentModule.d.ts +5 -1
  199. package/ts_build/src/chat/modules/AgentModule.js +62 -32
  200. package/ts_build/src/chat/modules/AgentModule.js.map +1 -1
  201. package/ts_build/src/chat/modules/AskModule.js.map +1 -1
  202. package/ts_build/src/chat/modules/SetupModule.js +4 -3
  203. package/ts_build/src/chat/modules/SetupModule.js.map +1 -1
  204. package/ts_build/src/chat/modules/SystemModule.js +19 -4
  205. package/ts_build/src/chat/modules/SystemModule.js.map +1 -1
  206. package/ts_build/src/chat/modules/index.d.ts +5 -0
  207. package/ts_build/src/chat/modules/index.js +14 -0
  208. package/ts_build/src/chat/modules/index.js.map +1 -0
  209. package/ts_build/src/chat/types.d.ts +2 -0
  210. package/ts_build/src/chat-old.js +3 -3
  211. package/ts_build/src/chat-old.js.map +1 -1
  212. package/ts_build/src/clients/anthropic.d.ts +1 -0
  213. package/ts_build/src/clients/anthropic.js +22 -1
  214. package/ts_build/src/clients/anthropic.js.map +1 -1
  215. package/ts_build/src/clients/openai.js +1 -1
  216. package/ts_build/src/clients/openai.js.map +1 -1
  217. package/ts_build/src/clients/xai.d.ts +7 -0
  218. package/ts_build/src/clients/xai.js +13 -4
  219. package/ts_build/src/clients/xai.js.map +1 -1
  220. package/ts_build/src/config.js +14 -3
  221. package/ts_build/src/config.js.map +1 -1
  222. package/ts_build/src/dataset/diffs/generate.js +2 -2
  223. package/ts_build/src/dataset/diffs/generate.js.map +1 -1
  224. package/ts_build/src/dataset/diffs/jsonl.js.map +1 -1
  225. package/ts_build/src/embeddings.js +9 -13
  226. package/ts_build/src/embeddings.js.map +1 -1
  227. package/ts_build/src/index.js +10 -10
  228. package/ts_build/src/index.js.map +1 -1
  229. package/ts_build/src/plugins/GitPlugin.d.ts +39 -0
  230. package/ts_build/src/plugins/GitPlugin.js +439 -0
  231. package/ts_build/src/plugins/GitPlugin.js.map +1 -0
  232. package/ts_build/src/plugins/LinterPlugin.d.ts +15 -0
  233. package/ts_build/src/plugins/LinterPlugin.js +65 -0
  234. package/ts_build/src/plugins/LinterPlugin.js.map +1 -0
  235. package/ts_build/src/plugins/PluginBase.d.ts +4 -3
  236. package/ts_build/src/plugins/PluginBase.js +3 -3
  237. package/ts_build/src/plugins/PluginBase.js.map +1 -1
  238. package/ts_build/src/plugins/asana.d.ts +3 -1
  239. package/ts_build/src/plugins/asana.js +3 -2
  240. package/ts_build/src/plugins/asana.js.map +1 -1
  241. package/ts_build/src/plugins/downloader/plugin.d.ts +3 -1
  242. package/ts_build/src/plugins/downloader/plugin.js +3 -2
  243. package/ts_build/src/plugins/downloader/plugin.js.map +1 -1
  244. package/ts_build/src/plugins/embedding.d.ts +5 -1
  245. package/ts_build/src/plugins/embedding.js +15 -3
  246. package/ts_build/src/plugins/embedding.js.map +1 -1
  247. package/ts_build/src/plugins/figma.d.ts +3 -1
  248. package/ts_build/src/plugins/figma.js +28 -4
  249. package/ts_build/src/plugins/figma.js.map +1 -1
  250. package/ts_build/src/plugins/github.d.ts +3 -1
  251. package/ts_build/src/plugins/github.js +3 -2
  252. package/ts_build/src/plugins/github.js.map +1 -1
  253. package/ts_build/src/plugins/jira.d.ts +3 -1
  254. package/ts_build/src/plugins/jira.js +3 -2
  255. package/ts_build/src/plugins/jira.js.map +1 -1
  256. package/ts_build/src/plugins/language.d.ts +7 -4
  257. package/ts_build/src/plugins/language.js +85 -20
  258. package/ts_build/src/plugins/language.js.map +1 -1
  259. package/ts_build/src/plugins/linear.d.ts +3 -1
  260. package/ts_build/src/plugins/linear.js +3 -2
  261. package/ts_build/src/plugins/linear.js.map +1 -1
  262. package/ts_build/src/plugins/notion.d.ts +3 -1
  263. package/ts_build/src/plugins/notion.js +3 -2
  264. package/ts_build/src/plugins/notion.js.map +1 -1
  265. package/ts_build/src/plugins/plugins.d.ts +4 -3
  266. package/ts_build/src/plugins/plugins.js +24 -14
  267. package/ts_build/src/plugins/plugins.js.map +1 -1
  268. package/ts_build/src/plugins/tree-sitter/editor.d.ts +34 -0
  269. package/ts_build/src/plugins/tree-sitter/editor.js +218 -0
  270. package/ts_build/src/plugins/tree-sitter/editor.js.map +1 -0
  271. package/ts_build/src/plugins/tree-sitter/human-readable-paths-new.d.ts +29 -0
  272. package/ts_build/src/plugins/tree-sitter/human-readable-paths-new.js +538 -0
  273. package/ts_build/src/plugins/tree-sitter/human-readable-paths-new.js.map +1 -0
  274. package/ts_build/src/plugins/tree-sitter/human-readable-paths.d.ts +22 -0
  275. package/ts_build/src/plugins/tree-sitter/human-readable-paths.js +332 -0
  276. package/ts_build/src/plugins/tree-sitter/human-readable-paths.js.map +1 -0
  277. package/ts_build/src/plugins/tree-sitter/lang-packs/index.d.ts +8 -0
  278. package/ts_build/src/plugins/tree-sitter/lang-packs/index.js +26 -0
  279. package/ts_build/src/plugins/tree-sitter/lang-packs/index.js.map +1 -0
  280. package/ts_build/src/plugins/tree-sitter/lang-packs/java.d.ts +2 -0
  281. package/ts_build/src/plugins/tree-sitter/lang-packs/java.js +61 -0
  282. package/ts_build/src/plugins/tree-sitter/lang-packs/java.js.map +1 -0
  283. package/ts_build/src/plugins/tree-sitter/lang-packs/javascript.d.ts +2 -0
  284. package/ts_build/src/plugins/tree-sitter/lang-packs/javascript.js +59 -0
  285. package/ts_build/src/plugins/tree-sitter/lang-packs/javascript.js.map +1 -0
  286. package/ts_build/src/plugins/tree-sitter/lang-packs/python.d.ts +2 -0
  287. package/ts_build/src/plugins/tree-sitter/lang-packs/python.js +47 -0
  288. package/ts_build/src/plugins/tree-sitter/lang-packs/python.js.map +1 -0
  289. package/ts_build/src/plugins/tree-sitter/lang-packs/types.d.ts +43 -0
  290. package/ts_build/src/plugins/tree-sitter/lang-packs/types.js +3 -0
  291. package/ts_build/src/plugins/tree-sitter/lang-packs/types.js.map +1 -0
  292. package/ts_build/src/plugins/tree-sitter/lang-packs/typescript.d.ts +2 -0
  293. package/ts_build/src/plugins/tree-sitter/lang-packs/typescript.js +50 -0
  294. package/ts_build/src/plugins/tree-sitter/lang-packs/typescript.js.map +1 -0
  295. package/ts_build/src/plugins/tree-sitter/parser.d.ts +75 -0
  296. package/ts_build/src/plugins/tree-sitter/parser.js +306 -0
  297. package/ts_build/src/plugins/tree-sitter/parser.js.map +1 -0
  298. package/ts_build/src/plugins/tree-sitter/simple-paths.d.ts +22 -0
  299. package/ts_build/src/plugins/tree-sitter/simple-paths.js +332 -0
  300. package/ts_build/src/plugins/tree-sitter/simple-paths.js.map +1 -0
  301. package/ts_build/src/plugins/types.d.ts +10 -0
  302. package/ts_build/src/plugins/url.d.ts +3 -2
  303. package/ts_build/src/plugins/url.js +3 -2
  304. package/ts_build/src/plugins/url.js.map +1 -1
  305. package/ts_build/src/plugins/vim.d.ts +4 -2
  306. package/ts_build/src/plugins/vim.js +6 -8
  307. package/ts_build/src/plugins/vim.js.map +1 -1
  308. package/ts_build/src/processors/CustomVariables.js +45 -47
  309. package/ts_build/src/processors/CustomVariables.js.map +1 -1
  310. package/ts_build/src/processors/TokenCompressor.js +10 -13
  311. package/ts_build/src/processors/TokenCompressor.js.map +1 -1
  312. package/ts_build/src/processors/ToolResponseCache.d.ts +2 -2
  313. package/ts_build/src/processors/ToolResponseCache.js +18 -10
  314. package/ts_build/src/processors/ToolResponseCache.js.map +1 -1
  315. package/ts_build/src/services/EmbeddingService.d.ts +10 -1
  316. package/ts_build/src/services/EmbeddingService.js +12 -12
  317. package/ts_build/src/services/EmbeddingService.js.map +1 -1
  318. package/ts_build/src/services/EventService.d.ts +7 -0
  319. package/ts_build/src/services/EventService.js +49 -0
  320. package/ts_build/src/services/EventService.js.map +1 -1
  321. package/ts_build/src/services/Mcp.js +8 -0
  322. package/ts_build/src/services/Mcp.js.map +1 -1
  323. package/ts_build/src/services/S3.js +4 -3
  324. package/ts_build/src/services/S3.js.map +1 -1
  325. package/ts_build/src/services/Tools.d.ts +1 -0
  326. package/ts_build/src/services/Tools.js +97 -35
  327. package/ts_build/src/services/Tools.js.map +1 -1
  328. package/ts_build/src/services/index.d.ts +4 -5
  329. package/ts_build/src/services/index.js +14 -9
  330. package/ts_build/src/services/index.js.map +1 -1
  331. package/ts_build/src/services/types.js +3 -3
  332. package/ts_build/src/services/types.js.map +1 -1
  333. package/ts_build/src/types.d.ts +7 -1
  334. package/ts_build/src/types.js +4 -0
  335. package/ts_build/src/types.js.map +1 -1
  336. package/ts_build/src/worker.js +12 -1
  337. package/ts_build/src/worker.js.map +1 -1
  338. package/ts_build/tests/clients/AIClient.test.d.ts +1 -0
  339. package/ts_build/tests/clients/AIClient.test.js +377 -0
  340. package/ts_build/tests/clients/AIClient.test.js.map +1 -0
  341. package/ts_build/tests/languagePlugin.test.js +217 -11
  342. package/ts_build/tests/languagePlugin.test.js.map +1 -1
  343. package/ts_build/tests/manual/agent-events/event-handler-reliability.test.d.ts +1 -0
  344. package/ts_build/tests/manual/agent-events/event-handler-reliability.test.js +315 -0
  345. package/ts_build/tests/manual/agent-events/event-handler-reliability.test.js.map +1 -0
  346. package/ts_build/tests/manual/agent-events/run-test.d.ts +2 -0
  347. package/ts_build/tests/manual/agent-events/run-test.js +148 -0
  348. package/ts_build/tests/manual/agent-events/run-test.js.map +1 -0
  349. package/ts_build/tests/manual/file-edits/figma.test.d.ts +1 -0
  350. package/ts_build/tests/manual/file-edits/figma.test.js +47 -0
  351. package/ts_build/tests/manual/file-edits/figma.test.js.map +1 -0
  352. package/ts_build/tests/manual/file-edits/fileblocks/readwrite.test.d.ts +1 -0
  353. package/ts_build/tests/manual/file-edits/fileblocks/readwrite.test.js +100 -0
  354. package/ts_build/tests/manual/file-edits/fileblocks/readwrite.test.js.map +1 -0
  355. package/ts_build/tests/manual/file-edits/patching.test.d.ts +1 -0
  356. package/ts_build/tests/manual/file-edits/patching.test.js +119 -0
  357. package/ts_build/tests/manual/file-edits/patching.test.js.map +1 -0
  358. package/ts_build/tests/plugins/language/languagePlugin-content-triggers.test.d.ts +1 -0
  359. package/ts_build/tests/plugins/language/languagePlugin-content-triggers.test.js +277 -0
  360. package/ts_build/tests/plugins/language/languagePlugin-content-triggers.test.js.map +1 -0
  361. package/ts_build/tests/plugins/language/languagePlugin-integration.test.d.ts +1 -0
  362. package/ts_build/tests/plugins/language/languagePlugin-integration.test.js +331 -0
  363. package/ts_build/tests/plugins/language/languagePlugin-integration.test.js.map +1 -0
  364. package/ts_build/tests/plugins/language/languagePlugin.test.d.ts +1 -0
  365. package/ts_build/tests/plugins/language/languagePlugin.test.js +286 -0
  366. package/ts_build/tests/plugins/language/languagePlugin.test.js.map +1 -0
  367. package/ts_build/tests/processors/Base64ImageDetector.test.d.ts +1 -0
  368. package/ts_build/tests/processors/Base64ImageDetector.test.js +351 -0
  369. package/ts_build/tests/processors/Base64ImageDetector.test.js.map +1 -0
  370. package/ts_build/tests/processors/CustomVariables.test.d.ts +1 -0
  371. package/ts_build/tests/processors/CustomVariables.test.js +354 -0
  372. package/ts_build/tests/processors/CustomVariables.test.js.map +1 -0
  373. package/ts_build/tests/processors/HarmonyToolProcessor.test.d.ts +1 -0
  374. package/ts_build/tests/processors/HarmonyToolProcessor.test.js +382 -0
  375. package/ts_build/tests/processors/HarmonyToolProcessor.test.js.map +1 -0
  376. package/ts_build/tests/processors/TokenCompressor.test.d.ts +1 -0
  377. package/ts_build/tests/processors/TokenCompressor.test.js +299 -0
  378. package/ts_build/tests/processors/TokenCompressor.test.js.map +1 -0
  379. package/ts_build/tests/processors/ToolResponseCache.test.d.ts +1 -0
  380. package/ts_build/tests/processors/ToolResponseCache.test.js +550 -0
  381. package/ts_build/tests/processors/ToolResponseCache.test.js.map +1 -0
  382. package/ts_build/tests/services/Plugins/plugin-event-integration.test.d.ts +1 -0
  383. package/ts_build/tests/services/Plugins/plugin-event-integration.test.js +232 -0
  384. package/ts_build/tests/services/Plugins/plugin-event-integration.test.js.map +1 -0
  385. package/ts_build/tests/services/Tools.test.d.ts +1 -0
  386. package/ts_build/tests/services/Tools.test.js +1059 -0
  387. package/ts_build/tests/services/Tools.test.js.map +1 -0
  388. package/ts_build/tests/test.spec.js +110 -68
  389. package/ts_build/tests/test.spec.js.map +1 -1
  390. package/ts_build/tests/tree-sitter/editor.test.d.ts +1 -0
  391. package/ts_build/tests/tree-sitter/editor.test.js +85 -0
  392. package/ts_build/tests/tree-sitter/editor.test.js.map +1 -0
  393. package/ts_build/tests/tree-sitter/invalid.test.d.ts +1 -0
  394. package/ts_build/tests/tree-sitter/invalid.test.js +198 -0
  395. package/ts_build/tests/tree-sitter/invalid.test.js.map +1 -0
  396. package/ts_build/tests/tree-sitter/paths/common-edits.test.d.ts +1 -0
  397. package/ts_build/tests/tree-sitter/paths/common-edits.test.js +347 -0
  398. package/ts_build/tests/tree-sitter/paths/common-edits.test.js.map +1 -0
  399. package/ts_build/tests/tree-sitter/paths/debug-exact-position.test.d.ts +1 -0
  400. package/ts_build/tests/tree-sitter/paths/debug-exact-position.test.js +35 -0
  401. package/ts_build/tests/tree-sitter/paths/debug-exact-position.test.js.map +1 -0
  402. package/ts_build/tests/tree-sitter/paths/debug-line-indexing.test.d.ts +1 -0
  403. package/ts_build/tests/tree-sitter/paths/debug-line-indexing.test.js +38 -0
  404. package/ts_build/tests/tree-sitter/paths/debug-line-indexing.test.js.map +1 -0
  405. package/ts_build/tests/tree-sitter/paths/debug-paths.test.d.ts +1 -0
  406. package/ts_build/tests/tree-sitter/paths/debug-paths.test.js +74 -0
  407. package/ts_build/tests/tree-sitter/paths/debug-paths.test.js.map +1 -0
  408. package/ts_build/tests/tree-sitter/paths/human-readable-paths.test.d.ts +1 -0
  409. package/ts_build/tests/tree-sitter/paths/human-readable-paths.test.js +302 -0
  410. package/ts_build/tests/tree-sitter/paths/human-readable-paths.test.js.map +1 -0
  411. package/ts_build/tests/tree-sitter/paths/paths.test.d.ts +1 -0
  412. package/ts_build/tests/tree-sitter/paths/paths.test.js +116 -0
  413. package/ts_build/tests/tree-sitter/paths/paths.test.js.map +1 -0
  414. package/ts_build/tests/tree-sitter/paths/simple-paths.test.d.ts +1 -0
  415. package/ts_build/tests/tree-sitter/paths/simple-paths.test.js +302 -0
  416. package/ts_build/tests/tree-sitter/paths/simple-paths.test.js.map +1 -0
  417. package/ts_build/tests/tree-sitter/sample-after.d.ts +11 -0
  418. package/ts_build/tests/tree-sitter/sample-after.js +44 -0
  419. package/ts_build/tests/tree-sitter/sample-after.js.map +1 -0
  420. package/ts_build/tests/tree-sitter/sample-before.d.ts +9 -0
  421. package/ts_build/tests/tree-sitter/sample-before.js +28 -0
  422. package/ts_build/tests/tree-sitter/sample-before.js.map +1 -0
  423. package/ts_build/tests/tree-sitter/test-files/completely-broken.d.ts +2 -0
  424. package/ts_build/tests/tree-sitter/test-files/completely-broken.js +17 -0
  425. package/ts_build/tests/tree-sitter/test-files/completely-broken.js.map +1 -0
  426. package/ts_build/tests/tree-sitter/test-files/duplicate-braces.d.ts +8 -0
  427. package/ts_build/tests/tree-sitter/test-files/duplicate-braces.js +38 -0
  428. package/ts_build/tests/tree-sitter/test-files/duplicate-braces.js.map +1 -0
  429. package/ts_build/tests/tree-sitter/test-files/invalid-nesting.d.ts +8 -0
  430. package/ts_build/tests/tree-sitter/test-files/invalid-nesting.js +38 -0
  431. package/ts_build/tests/tree-sitter/test-files/invalid-nesting.js.map +1 -0
  432. package/ts_build/tests/tree-sitter/test-files/malformed-signature.d.ts +8 -0
  433. package/ts_build/tests/tree-sitter/test-files/malformed-signature.js +38 -0
  434. package/ts_build/tests/tree-sitter/test-files/malformed-signature.js.map +1 -0
  435. package/ts_build/tests/tree-sitter/test-files/mismatched-parens.d.ts +10 -0
  436. package/ts_build/tests/tree-sitter/test-files/mismatched-parens.js +38 -0
  437. package/ts_build/tests/tree-sitter/test-files/mismatched-parens.js.map +1 -0
  438. package/ts_build/tests/tree-sitter/test-files/missing-semicolon.d.ts +8 -0
  439. package/ts_build/tests/tree-sitter/test-files/missing-semicolon.js +38 -0
  440. package/ts_build/tests/tree-sitter/test-files/missing-semicolon.js.map +1 -0
  441. package/ts_build/tests/tree-sitter/test-files/partially-broken.d.ts +6 -0
  442. package/ts_build/tests/tree-sitter/test-files/partially-broken.js +20 -0
  443. package/ts_build/tests/tree-sitter/test-files/partially-broken.js.map +1 -0
  444. package/ts_build/tests/tree-sitter/test-files/specific-errors.d.ts +7 -0
  445. package/ts_build/tests/tree-sitter/test-files/specific-errors.js +14 -0
  446. package/ts_build/tests/tree-sitter/test-files/specific-errors.js.map +1 -0
  447. package/ts_build/tests/tree-sitter/test-files/unclosed-string.d.ts +8 -0
  448. package/ts_build/tests/tree-sitter/test-files/unclosed-string.js +38 -0
  449. package/ts_build/tests/tree-sitter/test-files/unclosed-string.js.map +1 -0
  450. package/ts_build/tests/tree-sitter/tree-sitter.test.d.ts +1 -0
  451. package/ts_build/tests/tree-sitter/tree-sitter.test.js +185 -0
  452. package/ts_build/tests/tree-sitter/tree-sitter.test.js.map +1 -0
  453. package/tsconfig.json +2 -1
  454. package/tests/languagePlugin.test.ts +0 -74
  455. /package/src/chat/modules/{index.js → index.ts} +0 -0
  456. /package/tests/{integration → manual/file-edits}/patching/input.txt +0 -0
  457. /package/tests/{integration → manual/file-edits}/patching/output.txt +0 -0
  458. /package/tests/{integration → manual/file-edits}/patching/patch.txt +0 -0
  459. /package/tests/{integration → manual/file-edits}/patching/unseen.txt +0 -0
@@ -0,0 +1,490 @@
1
+ import { HarmonyToolProcessor } from '../../src/processors/HarmonyToolProcessor';
2
+ import { Message } from '../../src/clients/types';
3
+
4
+ describe('HarmonyToolProcessor', () => {
5
+ let processor: HarmonyToolProcessor;
6
+
7
+ beforeEach(() => {
8
+ // Reset singleton instance before each test
9
+ (HarmonyToolProcessor as any).instance = null;
10
+ processor = HarmonyToolProcessor.getInstance();
11
+ });
12
+
13
+ afterEach(() => {
14
+ // Clean up singleton instance after each test
15
+ (HarmonyToolProcessor as any).instance = null;
16
+ });
17
+
18
+ describe('singleton pattern', () => {
19
+ it('should return the same instance when getInstance is called multiple times', () => {
20
+ const instance1 = HarmonyToolProcessor.getInstance();
21
+ const instance2 = HarmonyToolProcessor.getInstance();
22
+
23
+ expect(instance1).toBe(instance2);
24
+ expect(instance1).toBeInstanceOf(HarmonyToolProcessor);
25
+ });
26
+
27
+ it('should create new instance after reset', () => {
28
+ const instance1 = HarmonyToolProcessor.getInstance();
29
+ (HarmonyToolProcessor as any).instance = null;
30
+ const instance2 = HarmonyToolProcessor.getInstance();
31
+
32
+ expect(instance1).not.toBe(instance2);
33
+ expect(instance2).toBeInstanceOf(HarmonyToolProcessor);
34
+ });
35
+ });
36
+
37
+ describe('createProcessor', () => {
38
+ it('should return a function', () => {
39
+ const processorFunction = processor.createProcessor();
40
+ expect(typeof processorFunction).toBe('function');
41
+ });
42
+
43
+ it('should not modify messages without Harmony patterns', () => {
44
+ const message: Message = {
45
+ role: 'assistant',
46
+ content: 'This is a regular message without tool calls'
47
+ };
48
+
49
+ const processorFunction = processor.createProcessor();
50
+ const originalMessages = [message];
51
+ const modifiedMessages = [{ ...message }];
52
+
53
+ processorFunction(originalMessages, modifiedMessages);
54
+
55
+ expect(modifiedMessages[0]).toEqual(message);
56
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
57
+ });
58
+ });
59
+
60
+ describe('Harmony pattern detection and conversion', () => {
61
+ it('should detect and convert basic Harmony function call', () => {
62
+ const message: Message = {
63
+ role: 'assistant',
64
+ content: 'I will help you. <|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1"}'
65
+ };
66
+
67
+ const processorFunction = processor.createProcessor();
68
+ const originalMessages = [message];
69
+ const modifiedMessages = [{ ...message }];
70
+
71
+ processorFunction(originalMessages, modifiedMessages);
72
+
73
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
74
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
75
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('testTool');
76
+ expect(modifiedMessages[0].tool_calls?.[0].function.arguments).toBe('{"param1":"value1"}');
77
+ expect(modifiedMessages[0].tool_calls?.[0].type).toBe('function');
78
+ expect(modifiedMessages[0].tool_calls?.[0].id).toMatch(/^harmony_call_/);
79
+ });
80
+
81
+ it('should detect and convert multiple Harmony function calls', () => {
82
+ const message: Message = {
83
+ role: 'assistant',
84
+ content: `First tool: <|channel|>commentary to=functions.tool1 <|constrain|>json<|message|>{"param1": "value1"}
85
+ Second tool: <|channel|>commentary to=functions.tool2 <|constrain|>json<|message|>{"param2": "value2"}`
86
+ };
87
+
88
+ const processorFunction = processor.createProcessor();
89
+ const originalMessages = [message];
90
+ const modifiedMessages = [{ ...message }];
91
+
92
+ processorFunction(originalMessages, modifiedMessages);
93
+
94
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
95
+ expect(modifiedMessages[0].tool_calls?.length).toBe(2);
96
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('tool1');
97
+ expect(modifiedMessages[0].tool_calls?.[1].function.name).toBe('tool2');
98
+ });
99
+
100
+ it('should handle complex JSON arguments', () => {
101
+ const message: Message = {
102
+ role: 'assistant',
103
+ content: `<|channel|>commentary to=functions.complexTool <|constrain|>json<|message|>{
104
+ "param1": "value1",
105
+ "param2": {
106
+ "nested": "object",
107
+ "array": [1, 2, 3]
108
+ },
109
+ "param3": true
110
+ }`
111
+ };
112
+
113
+ const processorFunction = processor.createProcessor();
114
+ const originalMessages = [message];
115
+ const modifiedMessages = [{ ...message }];
116
+
117
+ processorFunction(originalMessages, modifiedMessages);
118
+
119
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
120
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
121
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('complexTool');
122
+
123
+ const args = JSON.parse(modifiedMessages[0].tool_calls?.[0].function.arguments || '{}');
124
+ expect(args.param1).toBe('value1');
125
+ expect(args.param2.nested).toBe('object');
126
+ expect(args.param2.array).toEqual([1, 2, 3]);
127
+ expect(args.param3).toBe(true);
128
+ });
129
+
130
+ it('should detect and convert Harmony final answer pattern', () => {
131
+ const message: Message = {
132
+ role: 'assistant',
133
+ content: 'Here is my response: <|channel|>final<|message|>This is the final answer'
134
+ };
135
+
136
+ const processorFunction = processor.createProcessor();
137
+ const originalMessages = [message];
138
+ const modifiedMessages = [{ ...message }];
139
+
140
+ processorFunction(originalMessages, modifiedMessages);
141
+
142
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
143
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
144
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('finalAnswer');
145
+
146
+ const args = JSON.parse(modifiedMessages[0].tool_calls?.[0].function.arguments || '{}');
147
+ expect(args.answer).toBe('This is the final answer');
148
+ });
149
+
150
+ it('should detect and convert <|final|> pattern', () => {
151
+ const message: Message = {
152
+ role: 'assistant',
153
+ content: 'Processing complete <|final|>Task completed successfully'
154
+ };
155
+
156
+ const processorFunction = processor.createProcessor();
157
+ const originalMessages = [message];
158
+ const modifiedMessages = [{ ...message }];
159
+
160
+ processorFunction(originalMessages, modifiedMessages);
161
+
162
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
163
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
164
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('finalAnswer');
165
+
166
+ const args = JSON.parse(modifiedMessages[0].tool_calls?.[0].function.arguments || '{}');
167
+ expect(args.answer).toBe('Task completed successfully');
168
+ });
169
+ });
170
+
171
+ describe('message processing rules', () => {
172
+ it('should only process assistant messages', () => {
173
+ const userMessage: Message = {
174
+ role: 'user',
175
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1"}'
176
+ };
177
+
178
+ const systemMessage: Message = {
179
+ role: 'system',
180
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1"}'
181
+ };
182
+
183
+ const processorFunction = processor.createProcessor();
184
+ const originalMessages = [userMessage, systemMessage];
185
+ const modifiedMessages = [{ ...userMessage }, { ...systemMessage }];
186
+
187
+ processorFunction(originalMessages, modifiedMessages);
188
+
189
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
190
+ expect(modifiedMessages[1].tool_calls).toBeUndefined();
191
+ });
192
+
193
+ it('should only process string content', () => {
194
+ const message: Message = {
195
+ role: 'assistant',
196
+ content: [
197
+ { type: 'text', text: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1"}' }
198
+ ]
199
+ };
200
+
201
+ const processorFunction = processor.createProcessor();
202
+ const originalMessages = [message];
203
+ const modifiedMessages = [{ ...message }];
204
+
205
+ processorFunction(originalMessages, modifiedMessages);
206
+
207
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
208
+ });
209
+
210
+ it('should skip messages that already have tool calls', () => {
211
+ const message: Message = {
212
+ role: 'assistant',
213
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1"}',
214
+ tool_calls: [{
215
+ id: 'existing-call',
216
+ type: 'function',
217
+ function: {
218
+ name: 'existingTool',
219
+ arguments: '{"existing": "param"}'
220
+ }
221
+ }]
222
+ };
223
+
224
+ const processorFunction = processor.createProcessor();
225
+ const originalMessages = [message];
226
+ const modifiedMessages = [{ ...message }];
227
+
228
+ processorFunction(originalMessages, modifiedMessages);
229
+
230
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
231
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('existingTool');
232
+ });
233
+
234
+ it('should skip messages without Harmony patterns', () => {
235
+ const message: Message = {
236
+ role: 'assistant',
237
+ content: 'This is just a regular message with some <brackets> but no Harmony patterns'
238
+ };
239
+
240
+ const processorFunction = processor.createProcessor();
241
+ const originalMessages = [message];
242
+ const modifiedMessages = [{ ...message }];
243
+
244
+ processorFunction(originalMessages, modifiedMessages);
245
+
246
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
247
+ });
248
+ });
249
+
250
+ describe('JSON validation and cleaning', () => {
251
+ it('should handle valid JSON arguments', () => {
252
+ const message: Message = {
253
+ role: 'assistant',
254
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{"param1": "value1", "param2": 42}'
255
+ };
256
+
257
+ const processorFunction = processor.createProcessor();
258
+ const originalMessages = [message];
259
+ const modifiedMessages = [{ ...message }];
260
+
261
+ processorFunction(originalMessages, modifiedMessages);
262
+
263
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
264
+ expect(modifiedMessages[0].tool_calls?.[0].function.arguments).toBe('{"param1":"value1","param2":42}');
265
+ });
266
+
267
+ it('should handle malformed JSON by extracting what it can', () => {
268
+ const message: Message = {
269
+ role: 'assistant',
270
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>some text {"param1": "value1"} more text'
271
+ };
272
+
273
+ const processorFunction = processor.createProcessor();
274
+ const originalMessages = [message];
275
+ const modifiedMessages = [{ ...message }];
276
+
277
+ processorFunction(originalMessages, modifiedMessages);
278
+
279
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
280
+ expect(modifiedMessages[0].tool_calls?.[0].function.arguments).toBe('{"param1":"value1"}');
281
+ });
282
+
283
+ it('should skip tool calls with invalid JSON that cannot be parsed', () => {
284
+ const message: Message = {
285
+ role: 'assistant',
286
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>invalid json without braces'
287
+ };
288
+
289
+ const processorFunction = processor.createProcessor();
290
+ const originalMessages = [message];
291
+ const modifiedMessages = [{ ...message }];
292
+
293
+ processorFunction(originalMessages, modifiedMessages);
294
+
295
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
296
+ });
297
+
298
+ it('should handle empty JSON object', () => {
299
+ const message: Message = {
300
+ role: 'assistant',
301
+ content: '<|channel|>commentary to=functions.testTool <|constrain|>json<|message|>{}'
302
+ };
303
+
304
+ const processorFunction = processor.createProcessor();
305
+ const originalMessages = [message];
306
+ const modifiedMessages = [{ ...message }];
307
+
308
+ processorFunction(originalMessages, modifiedMessages);
309
+
310
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
311
+ expect(modifiedMessages[0].tool_calls?.[0].function.arguments).toBe('{}');
312
+ });
313
+ });
314
+
315
+ describe('tool call ID generation', () => {
316
+ it('should generate unique IDs for each tool call', () => {
317
+ const message: Message = {
318
+ role: 'assistant',
319
+ content: `<|channel|>commentary to=functions.tool1 <|constrain|>json<|message|>{"param1": "value1"}
320
+ <|channel|>commentary to=functions.tool2 <|constrain|>json<|message|>{"param2": "value2"}`
321
+ };
322
+
323
+ const processorFunction = processor.createProcessor();
324
+ const originalMessages = [message];
325
+ const modifiedMessages = [{ ...message }];
326
+
327
+ processorFunction(originalMessages, modifiedMessages);
328
+
329
+ expect(modifiedMessages[0].tool_calls?.length).toBe(2);
330
+ const id1 = modifiedMessages[0].tool_calls?.[0].id;
331
+ const id2 = modifiedMessages[0].tool_calls?.[1].id;
332
+
333
+ expect(id1).toBeDefined();
334
+ expect(id2).toBeDefined();
335
+ expect(id1).not.toBe(id2);
336
+ expect(id1).toMatch(/^harmony_call_/);
337
+ expect(id2).toMatch(/^harmony_call_/);
338
+ });
339
+ });
340
+
341
+ describe('edge cases and error handling', () => {
342
+ it('should handle empty message content', () => {
343
+ const message: Message = {
344
+ role: 'assistant',
345
+ content: ''
346
+ };
347
+
348
+ const processorFunction = processor.createProcessor();
349
+ const originalMessages = [message];
350
+ const modifiedMessages = [{ ...message }];
351
+
352
+ processorFunction(originalMessages, modifiedMessages);
353
+
354
+ expect(modifiedMessages[0]).toEqual(message);
355
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
356
+ });
357
+
358
+ it('should handle null content gracefully', () => {
359
+ const message: Message = {
360
+ role: 'assistant',
361
+ content: null as any
362
+ };
363
+
364
+ const processorFunction = processor.createProcessor();
365
+ const originalMessages = [message];
366
+ const modifiedMessages = [{ ...message }];
367
+
368
+ processorFunction(originalMessages, modifiedMessages);
369
+
370
+ expect(modifiedMessages[0]).toEqual(message);
371
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
372
+ });
373
+
374
+ it('should handle undefined content gracefully', () => {
375
+ const message: Message = {
376
+ role: 'assistant',
377
+ content: undefined as any
378
+ };
379
+
380
+ const processorFunction = processor.createProcessor();
381
+ const originalMessages = [message];
382
+ const modifiedMessages = [{ ...message }];
383
+
384
+ processorFunction(originalMessages, modifiedMessages);
385
+
386
+ expect(modifiedMessages[0]).toEqual(message);
387
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
388
+ });
389
+
390
+ it('should handle partial Harmony patterns', () => {
391
+ const message: Message = {
392
+ role: 'assistant',
393
+ content: '<|channel|>commentary to=functions.testTool but missing the rest'
394
+ };
395
+
396
+ const processorFunction = processor.createProcessor();
397
+ const originalMessages = [message];
398
+ const modifiedMessages = [{ ...message }];
399
+
400
+ processorFunction(originalMessages, modifiedMessages);
401
+
402
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
403
+ });
404
+
405
+ it('should handle malformed tool names', () => {
406
+ const message: Message = {
407
+ role: 'assistant',
408
+ content: '<|channel|>commentary to=functions. <|constrain|>json<|message|>{"param1": "value1"}'
409
+ };
410
+
411
+ const processorFunction = processor.createProcessor();
412
+ const originalMessages = [message];
413
+ const modifiedMessages = [{ ...message }];
414
+
415
+ processorFunction(originalMessages, modifiedMessages);
416
+
417
+ expect(modifiedMessages[0].tool_calls).toBeUndefined();
418
+ });
419
+
420
+ it('should process multiple messages in array', () => {
421
+ const message1: Message = {
422
+ role: 'assistant',
423
+ content: '<|channel|>commentary to=functions.tool1 <|constrain|>json<|message|>{"param1": "value1"}'
424
+ };
425
+
426
+ const message2: Message = {
427
+ role: 'assistant',
428
+ content: 'Regular message without tools'
429
+ };
430
+
431
+ const message3: Message = {
432
+ role: 'assistant',
433
+ content: '<|channel|>commentary to=functions.tool2 <|constrain|>json<|message|>{"param2": "value2"}'
434
+ };
435
+
436
+ const processorFunction = processor.createProcessor();
437
+ const originalMessages = [message1, message2, message3];
438
+ const modifiedMessages = [{ ...message1 }, { ...message2 }, { ...message3 }];
439
+
440
+ processorFunction(originalMessages, modifiedMessages);
441
+
442
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
443
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('tool1');
444
+ expect(modifiedMessages[1].tool_calls).toBeUndefined();
445
+ expect(modifiedMessages[2].tool_calls).toBeDefined();
446
+ expect(modifiedMessages[2].tool_calls?.[0].function.name).toBe('tool2');
447
+ });
448
+ });
449
+
450
+ describe('mixed pattern detection', () => {
451
+ it('should detect both function calls and final answer patterns', () => {
452
+ const message: Message = {
453
+ role: 'assistant',
454
+ content: `First I'll call a tool: <|channel|>commentary to=functions.searchTool <|constrain|>json<|message|>{"query": "test"}
455
+ Then I'll provide the final answer: <|channel|>final<|message|>Here is the result`
456
+ };
457
+
458
+ const processorFunction = processor.createProcessor();
459
+ const originalMessages = [message];
460
+ const modifiedMessages = [{ ...message }];
461
+
462
+ processorFunction(originalMessages, modifiedMessages);
463
+
464
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
465
+ expect(modifiedMessages[0].tool_calls?.length).toBe(2);
466
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('searchTool');
467
+ expect(modifiedMessages[0].tool_calls?.[1].function.name).toBe('finalAnswer');
468
+ });
469
+
470
+ it('should handle final pattern with empty content', () => {
471
+ const message: Message = {
472
+ role: 'assistant',
473
+ content: 'Task completed <|final|>'
474
+ };
475
+
476
+ const processorFunction = processor.createProcessor();
477
+ const originalMessages = [message];
478
+ const modifiedMessages = [{ ...message }];
479
+
480
+ processorFunction(originalMessages, modifiedMessages);
481
+
482
+ expect(modifiedMessages[0].tool_calls).toBeDefined();
483
+ expect(modifiedMessages[0].tool_calls?.length).toBe(1);
484
+ expect(modifiedMessages[0].tool_calls?.[0].function.name).toBe('finalAnswer');
485
+
486
+ const args = JSON.parse(modifiedMessages[0].tool_calls?.[0].function.arguments || '{}');
487
+ expect(args.answer).toBe('Done');
488
+ });
489
+ });
490
+ });