@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,736 @@
1
+ import { Message } from "../../src/clients/types";
2
+ import {
3
+ ToolResponseCache,
4
+ jqToolResponseDefinition,
5
+ } from "../../src/processors/ToolResponseCache";
6
+ import { ToolsService } from "../../src/services";
7
+
8
+ // Mock node-jq
9
+ jest.mock("node-jq", () => ({
10
+ run: jest
11
+ .fn()
12
+ .mockImplementation(async (query: string, data: any, options: any) => {
13
+ // Simulate common JQ queries based on the test data
14
+
15
+ // Handle .test query on {"test": "value"}
16
+ if (query === ".test") {
17
+ if (data && data.test !== undefined) {
18
+ return JSON.stringify(data.test);
19
+ }
20
+ }
21
+
22
+ // Handle .id query for extracting id values
23
+ if (query === ".id") {
24
+ return data && data.id !== undefined ? data.id.toString() : "null";
25
+ }
26
+
27
+ // Handle .data | length query for counting array elements
28
+ if (query === ".data | length") {
29
+ if (data && Array.isArray(data.data)) {
30
+ return data.data.length.toString();
31
+ }
32
+ }
33
+
34
+ // Handle .data | add query for summing array elements
35
+ if (query === ".data | add") {
36
+ if (data && Array.isArray(data.data)) {
37
+ const sum = data.data.reduce((a, b) => a + b, 0);
38
+ return sum.toString();
39
+ }
40
+ }
41
+
42
+ // Handle .unicode query for special characters test
43
+ if (query === ".unicode") {
44
+ if (data && data.unicode !== undefined) {
45
+ return JSON.stringify(data.unicode);
46
+ }
47
+ }
48
+
49
+ // Handle .empty query for empty string test
50
+ if (query === ".empty") {
51
+ if (data && data.empty !== undefined) {
52
+ return JSON.stringify(data.empty);
53
+ }
54
+ }
55
+
56
+ // Handle .nullValue query for null value test
57
+ if (query === ".nullValue") {
58
+ if (data && data.nullValue !== undefined) {
59
+ return JSON.stringify(data.nullValue);
60
+ }
61
+ }
62
+
63
+ // Handle .name query on {name: "test", data: [1, 2, 3]}
64
+ if (query === ".name") {
65
+ if (data && data.name) {
66
+ return JSON.stringify(data.name);
67
+ }
68
+ }
69
+
70
+ // Handle .data[] query on {name: "test", data: [1, 2, 3]}
71
+ if (query === ".data[]") {
72
+ if (data && Array.isArray(data.data)) {
73
+ return data.data.join("\n");
74
+ }
75
+ }
76
+
77
+ // Handle map(.value) on [{id: 1, value: "a"}, {id: 2, value: "b"}]
78
+ if (query === "map(.value)") {
79
+ if (Array.isArray(data)) {
80
+ const values = data.map((item) => item.value);
81
+ return JSON.stringify(values);
82
+ }
83
+ }
84
+
85
+ // Handle map({identifier: .id, content: .value}) transformation
86
+ if (query === "map({identifier: .id, content: .value})") {
87
+ if (Array.isArray(data)) {
88
+ const transformed = data.map((item) => ({
89
+ identifier: item.id,
90
+ content: item.value,
91
+ }));
92
+ return JSON.stringify(transformed);
93
+ }
94
+ }
95
+
96
+ // Handle "." query (return entire object, formatted)
97
+ if (query === ".") {
98
+ return JSON.stringify(data, null, 2);
99
+ }
100
+
101
+ // Handle .nested.inner query for nested JSON
102
+ if (query === ".nested.inner") {
103
+ if (data && data.nested && data.nested.inner) {
104
+ return JSON.stringify(data.nested.inner);
105
+ }
106
+ }
107
+
108
+ // Handle map(select(.id > 10)) - empty result
109
+ if (query === "map(select(.id > 10))") {
110
+ return JSON.stringify([]);
111
+ }
112
+
113
+ // Handle invalid queries - throw error
114
+ if (query === ".invalid[") {
115
+ throw new Error("Invalid JQ query syntax");
116
+ }
117
+
118
+ // Handle deep nested queries
119
+ if (query === ".level1.level2.level3.level4.level5.deepValue") {
120
+ if (
121
+ data &&
122
+ data.level1 &&
123
+ data.level1.level2 &&
124
+ data.level1.level2.level3 &&
125
+ data.level1.level2.level3.level4 &&
126
+ data.level1.level2.level3.level4.level5
127
+ ) {
128
+ return JSON.stringify(
129
+ data.level1.level2.level3.level4.level5.deepValue
130
+ );
131
+ }
132
+ }
133
+
134
+ // Handle queries on invalid JSON data
135
+ if (typeof data === "string" && data === "invalid json string") {
136
+ throw new Error("Invalid JSON input");
137
+ }
138
+
139
+ // Default fallback - return stringified data
140
+ try {
141
+ return JSON.stringify(data);
142
+ } catch (error) {
143
+ throw new Error(`JQ query failed: ${error}`);
144
+ }
145
+ }),
146
+ }));
147
+
148
+ const mockJq = require("node-jq");
149
+
150
+ describe("ToolResponseCache", () => {
151
+ let cache: ToolResponseCache;
152
+ let mockToolsService: jest.Mocked<ToolsService>;
153
+
154
+ beforeEach(() => {
155
+ jest.clearAllMocks();
156
+
157
+ mockToolsService = {
158
+ addTools: jest.fn(),
159
+ addFunctions: jest.fn(),
160
+ getTool: jest.fn().mockReturnValue(undefined),
161
+ callTool: jest.fn(),
162
+ } as any;
163
+
164
+ cache = new ToolResponseCache(mockToolsService);
165
+ });
166
+
167
+ describe("constructor", () => {
168
+ it("should create an instance and register tool with ToolsService", () => {
169
+ expect(cache).toBeDefined();
170
+ expect(cache).toBeInstanceOf(ToolResponseCache);
171
+ expect(mockToolsService.addTools).toHaveBeenCalledWith([
172
+ jqToolResponseDefinition,
173
+ ]);
174
+ expect(mockToolsService.addFunctions).toHaveBeenCalledWith({
175
+ jqToolResponse: expect.any(Function),
176
+ });
177
+ });
178
+
179
+ it("should overwrite tool if it already exists", () => {
180
+ mockToolsService.getTool.mockReturnValue(jqToolResponseDefinition);
181
+ const newCache = new ToolResponseCache(mockToolsService);
182
+
183
+ // Should be called each time
184
+ expect(mockToolsService.addTools).toHaveBeenCalledTimes(2);
185
+ });
186
+ });
187
+
188
+ describe("createProcessor", () => {
189
+ it("should create a message processor function", () => {
190
+ const processor = cache.createProcessor();
191
+ expect(typeof processor).toBe("function");
192
+ });
193
+
194
+ it("should process tool response messages", async () => {
195
+ const processor = cache.createProcessor();
196
+
197
+ const messages: Message[] = [
198
+ {
199
+ role: "tool",
200
+ tool_call_id: "call_123",
201
+ content: '{"data": [{"name": "test", "value": 42}]}',
202
+ },
203
+ ];
204
+
205
+ await processor(messages, messages);
206
+
207
+ // Verify message was stored
208
+ expect(cache.getStorageKeys()).toContain("call_123");
209
+ expect(cache.retrieveRawResponse("call_123")).toBe(
210
+ '{"data": [{"name": "test", "value": 42}]}'
211
+ );
212
+ });
213
+
214
+ it("should ignore non-tool messages", async () => {
215
+ const processor = cache.createProcessor();
216
+
217
+ const messages: Message[] = [
218
+ {
219
+ role: "user",
220
+ content: "This is a user message",
221
+ },
222
+ {
223
+ role: "assistant",
224
+ content: "This is an assistant message",
225
+ },
226
+ ];
227
+
228
+ await processor(messages, messages);
229
+
230
+ expect(cache.getStorageSize()).toBe(0);
231
+ });
232
+ it("should ignore tool messages without tool_call_id", async () => {
233
+ const processor = cache.createProcessor();
234
+
235
+ const messages: Message[] = [
236
+ {
237
+ role: "tool",
238
+ content: "Tool response without call ID",
239
+ } as Message,
240
+ ];
241
+
242
+ await processor(messages, messages);
243
+
244
+ expect(cache.getStorageSize()).toBe(0);
245
+ });
246
+
247
+ it("should ignore tool messages with non-string content", async () => {
248
+ const processor = cache.createProcessor();
249
+
250
+ const messages: Message[] = [
251
+ {
252
+ role: "tool",
253
+ tool_call_id: "call_123",
254
+ content: null,
255
+ } as Message,
256
+ {
257
+ role: "tool",
258
+ tool_call_id: "call_456",
259
+ content: undefined,
260
+ } as Message,
261
+ {
262
+ role: "tool",
263
+ tool_call_id: "call_789",
264
+ content: 42 as any,
265
+ } as Message,
266
+ ];
267
+
268
+ await processor(messages, messages);
269
+
270
+ expect(cache.getStorageSize()).toBe(0);
271
+ });
272
+
273
+ it("should apply filter function when provided", async () => {
274
+ const filterFn = (msg: Message) => msg.tool_call_id === "call_allowed";
275
+ const processor = cache.createProcessor(filterFn);
276
+
277
+ const messages: Message[] = [
278
+ {
279
+ role: "tool",
280
+ tool_call_id: "call_allowed",
281
+ content: "Allowed content",
282
+ },
283
+ {
284
+ role: "tool",
285
+ tool_call_id: "call_blocked",
286
+ content: "Blocked content",
287
+ },
288
+ ];
289
+
290
+ await processor(messages, messages);
291
+
292
+ expect(cache.getStorageKeys()).toContain("call_allowed");
293
+ expect(cache.getStorageKeys()).not.toContain("call_blocked");
294
+ expect(cache.getStorageSize()).toBe(1);
295
+ });
296
+ });
297
+
298
+ describe("storeToolResponse", () => {
299
+ it("should store tool response content with metadata", () => {
300
+ const content = '{"test": "data"}';
301
+ const toolCallId = "call_123";
302
+
303
+ // Access private method for testing
304
+ (cache as any).storeToolResponse(content, toolCallId);
305
+
306
+ expect(cache.retrieveRawResponse(toolCallId)).toBe(content);
307
+ expect(cache.getStorageKeys()).toContain(toolCallId);
308
+
309
+ // Check metadata
310
+ const metadata = (cache as any).metadataStorage[toolCallId];
311
+ expect(metadata.toolCallId).toBe(toolCallId);
312
+ expect(metadata.originalLength).toBe(content.length);
313
+ expect(metadata.storedAt).toBeGreaterThan(0);
314
+ });
315
+ });
316
+ describe("queryToolResponse", () => {
317
+ beforeEach(() => {
318
+ // Store some test data
319
+ cache.storeToolResponse(
320
+ '{"name": "test", "data": [1, 2, 3]}',
321
+ "call_123"
322
+ );
323
+ cache.storeToolResponse(
324
+ '[{"id": 1, "value": "a"}, {"id": 2, "value": "b"}]',
325
+ "call_456"
326
+ );
327
+ cache.storeToolResponse(
328
+ '{"nested": "{\\"inner\\": \\"value\\"}"}',
329
+ "call_789"
330
+ );
331
+ cache.storeToolResponse("invalid json string", "call_invalid");
332
+ });
333
+
334
+ it("should execute simple JQ queries successfully", async () => {
335
+ const result = await cache.queryToolResponse("call_123", ".name");
336
+ expect(result).toBe('"test"');
337
+ });
338
+
339
+ it("should execute array queries", async () => {
340
+ const result = await cache.queryToolResponse("call_123", ".data[]");
341
+ expect(result).toBe("1\n2\n3");
342
+ });
343
+
344
+ it("should execute complex JQ queries", async () => {
345
+ const result = await cache.queryToolResponse("call_456", "map(.value)");
346
+ expect(result).toBe('["a","b"]');
347
+ });
348
+
349
+ it("should handle object transformation queries", async () => {
350
+ const result = await cache.queryToolResponse(
351
+ "call_456",
352
+ "map({identifier: .id, content: .value})"
353
+ );
354
+ const parsed = JSON.parse(result);
355
+ expect(parsed).toEqual([
356
+ { identifier: 1, content: "a" },
357
+ { identifier: 2, content: "b" },
358
+ ]);
359
+ });
360
+
361
+ it("should return error for missing tool call ID", async () => {
362
+ const result = await cache.queryToolResponse("missing_id", ".test");
363
+ expect(result).toContain("Error: No tool response found");
364
+ expect(result).toContain("missing_id");
365
+ expect(result).toContain("Available IDs:");
366
+ });
367
+
368
+ it("should handle invalid JQ queries with error message", async () => {
369
+ const result = await cache.queryToolResponse("call_123", ".invalid[");
370
+ expect(result).toContain("JQ Query Error:");
371
+ });
372
+
373
+ it("should handle non-JSON data with helpful error", async () => {
374
+ const result = await cache.queryToolResponse("call_invalid", ".test");
375
+ expect(result).toContain("Error: Tool response data is not valid JSON");
376
+ expect(result).toContain('toolCallId "call_invalid"');
377
+ });
378
+
379
+ it("should return formatted JSON for complex results", async () => {
380
+ const result = await cache.queryToolResponse("call_456", ".");
381
+ expect(result).toContain("[\n");
382
+ expect(result).toContain(" {");
383
+ expect(result).toContain(' "id":');
384
+ });
385
+
386
+ it("should handle nested JSON strings parsing", async () => {
387
+ const result = await cache.queryToolResponse("call_789", ".nested.inner");
388
+ expect(result).toBe('"value"');
389
+ });
390
+
391
+ it("should handle empty query results", async () => {
392
+ const result = await cache.queryToolResponse(
393
+ "call_456",
394
+ "map(select(.id > 10))"
395
+ );
396
+ expect(result).toBe("[]");
397
+ });
398
+ });
399
+
400
+ describe("parseNestedJsonStrings", () => {
401
+ it("should parse simple JSON strings", () => {
402
+ const input = '{"test": "value"}';
403
+ const result = cache.parseNestedJsonStrings(input);
404
+ expect(result).toEqual({ test: "value" });
405
+ });
406
+
407
+ it("should parse nested JSON strings recursively", () => {
408
+ const input = '{"outer": "{\\"inner\\": \\"value\\"}"}';
409
+ const result = cache.parseNestedJsonStrings(input);
410
+ expect(result).toEqual({
411
+ outer: { inner: "value" },
412
+ });
413
+ });
414
+
415
+ it("should handle arrays with JSON strings", () => {
416
+ const input = ['{"test": "value1"}', '{"test": "value2"}'];
417
+ const result = cache.parseNestedJsonStrings(input);
418
+ expect(result).toEqual([{ test: "value1" }, { test: "value2" }]);
419
+ });
420
+
421
+ it("should handle mixed nested structures", () => {
422
+ const input = {
423
+ stringField: '{"nested": "value"}',
424
+ arrayField: ['{"item": 1}', '{"item": 2}'],
425
+ objectField: {
426
+ deepString: '{"deep": "nested"}',
427
+ },
428
+ };
429
+ const result = cache.parseNestedJsonStrings(input);
430
+ expect(result).toEqual({
431
+ stringField: { nested: "value" },
432
+ arrayField: [{ item: 1 }, { item: 2 }],
433
+ objectField: {
434
+ deepString: { deep: "nested" },
435
+ },
436
+ });
437
+ });
438
+
439
+ it("should leave non-JSON strings unchanged", () => {
440
+ const input = {
441
+ jsonString: '{"test": "value"}',
442
+ regularString: "just a string",
443
+ number: 42,
444
+ boolean: true,
445
+ };
446
+ const result = cache.parseNestedJsonStrings(input);
447
+ expect(result).toEqual({
448
+ jsonString: { test: "value" },
449
+ regularString: "just a string",
450
+ number: 42,
451
+ boolean: true,
452
+ });
453
+ });
454
+
455
+ it("should handle malformed JSON strings gracefully", () => {
456
+ const input = '{"invalid": json}';
457
+ const result = cache.parseNestedJsonStrings(input);
458
+ expect(result).toBe(input); // Should return original string
459
+ });
460
+ });
461
+ describe("retrieveRawResponse", () => {
462
+ it("should return stored raw content", () => {
463
+ const content = '{"test": "data"}';
464
+ cache.storeToolResponse(content, "call_123");
465
+
466
+ const result = cache.retrieveRawResponse("call_123");
467
+ expect(result).toBe(content);
468
+ });
469
+
470
+ it("should return null for missing tool call ID", () => {
471
+ const result = cache.retrieveRawResponse("missing_id");
472
+ expect(result).toBeNull();
473
+ });
474
+ });
475
+
476
+ describe("clearStorage", () => {
477
+ it("should clear all stored data and metadata", () => {
478
+ cache.storeToolResponse('{"test": "data"}', "call_123");
479
+ cache.storeToolResponse('{"more": "data"}', "call_456");
480
+
481
+ expect(cache.getStorageSize()).toBe(2);
482
+
483
+ cache.clearStorage();
484
+
485
+ expect(cache.getStorageSize()).toBe(0);
486
+ expect(cache.getStorageKeys()).toEqual([]);
487
+ expect(cache.retrieveRawResponse("call_123")).toBeNull();
488
+ expect(cache.retrieveRawResponse("call_456")).toBeNull();
489
+ });
490
+ });
491
+
492
+ describe("getStorageKeys", () => {
493
+ it("should return empty array when no data stored", () => {
494
+ expect(cache.getStorageKeys()).toEqual([]);
495
+ });
496
+
497
+ it("should return all stored tool call IDs", () => {
498
+ cache.storeToolResponse('{"test": "data1"}', "call_123");
499
+ cache.storeToolResponse('{"test": "data2"}', "call_456");
500
+ cache.storeToolResponse('{"test": "data3"}', "call_789");
501
+
502
+ const keys = cache.getStorageKeys();
503
+ expect(keys).toContain("call_123");
504
+ expect(keys).toContain("call_456");
505
+ expect(keys).toContain("call_789");
506
+ expect(keys).toHaveLength(3);
507
+ });
508
+ });
509
+
510
+ describe("getStorageSize", () => {
511
+ it("should return 0 for empty storage", () => {
512
+ expect(cache.getStorageSize()).toBe(0);
513
+ });
514
+
515
+ it("should return correct count after storing responses", () => {
516
+ expect(cache.getStorageSize()).toBe(0);
517
+
518
+ cache.storeToolResponse('{"test": "data1"}', "call_123");
519
+ expect(cache.getStorageSize()).toBe(1);
520
+
521
+ cache.storeToolResponse('{"test": "data2"}', "call_456");
522
+ expect(cache.getStorageSize()).toBe(2);
523
+
524
+ cache.clearStorage();
525
+ expect(cache.getStorageSize()).toBe(0);
526
+ });
527
+ });
528
+
529
+ describe("registerTool", () => {
530
+ let mockToolsService: jest.Mocked<ToolsService>;
531
+
532
+ beforeEach(() => {
533
+ mockToolsService = {
534
+ getTool: jest.fn(),
535
+ addTools: jest.fn(),
536
+ addFunctions: jest.fn(),
537
+ } as any;
538
+ });
539
+
540
+ it("should register tool when not already present", () => {
541
+ mockToolsService.getTool.mockReturnValue(null);
542
+
543
+ cache.registerTool(mockToolsService);
544
+
545
+ expect(mockToolsService.addTools).toHaveBeenCalledWith([
546
+ {
547
+ type: "function",
548
+ function: expect.objectContaining({
549
+ name: "jqToolResponse",
550
+ }),
551
+ },
552
+ ]);
553
+ expect(mockToolsService.addFunctions).toHaveBeenCalledWith({
554
+ jqToolResponse: expect.any(Function),
555
+ });
556
+ });
557
+
558
+ it("should overwrite tool when already present", () => {
559
+ mockToolsService.getTool.mockReturnValue({} as any);
560
+
561
+ cache.registerTool(mockToolsService);
562
+
563
+ expect(mockToolsService.addTools).toHaveBeenCalled();
564
+ expect(mockToolsService.addFunctions).toHaveBeenCalled();
565
+ });
566
+
567
+ it("should register function that calls queryToolResponse", async () => {
568
+ mockToolsService.getTool.mockReturnValue(null);
569
+
570
+ cache.registerTool(mockToolsService);
571
+
572
+ const addFunctionsCall = mockToolsService.addFunctions.mock.calls[0][0];
573
+ const jqFunction = addFunctionsCall.jqToolResponse;
574
+
575
+ // Store test data
576
+ cache.storeToolResponse('{"test": "value"}', "call_123");
577
+
578
+ // Test the registered function
579
+ const result = await jqFunction("call_123", ".test");
580
+ expect(result).toBe('"value"');
581
+ });
582
+ });
583
+
584
+ describe("Edge Cases and Error Handling", () => {
585
+ it("should handle very large JSON objects", async () => {
586
+ const largeObject = {
587
+ data: Array(1000)
588
+ .fill(0)
589
+ .map((_, i) => ({
590
+ id: i,
591
+ name: `item_${i}`,
592
+ description: `This is item number ${i} with some additional text to make it larger`,
593
+ })),
594
+ };
595
+ const largeContent = JSON.stringify(largeObject);
596
+
597
+ cache.storeToolResponse(largeContent, "call_large");
598
+
599
+ const result = await cache.queryToolResponse(
600
+ "call_large",
601
+ ".data | length"
602
+ );
603
+ expect(result).toBe("1000");
604
+ });
605
+
606
+ it("should handle special characters in JSON", async () => {
607
+ const specialContent = JSON.stringify({
608
+ unicode: "Hello 世界 🌍",
609
+ escaped: "Line 1\nLine 2\tTabbed",
610
+ quotes: 'He said "Hello" to me',
611
+ });
612
+
613
+ cache.storeToolResponse(specialContent, "call_special");
614
+
615
+ const result = await cache.queryToolResponse("call_special", ".unicode");
616
+ expect(result).toBe('"Hello 世界 🌍"');
617
+ });
618
+
619
+ it("should handle empty strings and null values", async () => {
620
+ const content = JSON.stringify({
621
+ empty: "",
622
+ nullValue: null,
623
+ zero: 0,
624
+ false: false,
625
+ });
626
+
627
+ cache.storeToolResponse(content, "call_empty");
628
+
629
+ const emptyResult = await cache.queryToolResponse("call_empty", ".empty");
630
+ expect(emptyResult).toBe('""');
631
+
632
+ const nullResult = await cache.queryToolResponse(
633
+ "call_empty",
634
+ ".nullValue"
635
+ );
636
+ expect(nullResult).toBe("null");
637
+ });
638
+
639
+ it("should handle concurrent storage operations", async () => {
640
+ const promises = Array(10)
641
+ .fill(0)
642
+ .map((_, i) =>
643
+ Promise.resolve(cache.storeToolResponse(`{"id": ${i}}`, `call_${i}`))
644
+ );
645
+
646
+ await Promise.all(promises);
647
+
648
+ expect(cache.getStorageSize()).toBe(10);
649
+
650
+ const results = await Promise.all(
651
+ Array(10)
652
+ .fill(0)
653
+ .map((_, i) => cache.queryToolResponse(`call_${i}`, ".id"))
654
+ );
655
+
656
+ results.forEach((result, i) => {
657
+ expect(result).toBe(i.toString());
658
+ });
659
+ });
660
+
661
+ it("should handle deeply nested JSON structures", async () => {
662
+ const deepObject = {
663
+ level1: {
664
+ level2: {
665
+ level3: {
666
+ level4: {
667
+ level5: {
668
+ deepValue: "found it!",
669
+ },
670
+ },
671
+ },
672
+ },
673
+ },
674
+ };
675
+
676
+ cache.storeToolResponse(JSON.stringify(deepObject), "call_deep");
677
+
678
+ const result = await cache.queryToolResponse(
679
+ "call_deep",
680
+ ".level1.level2.level3.level4.level5.deepValue"
681
+ );
682
+ expect(result).toBe('"found it!"');
683
+ });
684
+ });
685
+
686
+ describe("Integration with Message Processing", () => {
687
+ it("should integrate with complete message processing workflow", async () => {
688
+ const processor = cache.createProcessor();
689
+
690
+ const messages: Message[] = [
691
+ {
692
+ role: "user",
693
+ content: "Test request",
694
+ },
695
+ {
696
+ role: "assistant",
697
+ content: "Processing...",
698
+ tool_calls: [
699
+ {
700
+ id: "call_123",
701
+ type: "function",
702
+ function: {
703
+ name: "testTool",
704
+ arguments: "{}",
705
+ },
706
+ },
707
+ ],
708
+ },
709
+ {
710
+ role: "tool",
711
+ tool_call_id: "call_123",
712
+ content: JSON.stringify({
713
+ result: "success",
714
+ data: [1, 2, 3, 4, 5],
715
+ }),
716
+ },
717
+ ];
718
+
719
+ await processor(messages, messages);
720
+
721
+ expect(cache.getStorageSize()).toBe(1);
722
+
723
+ const result = await cache.queryToolResponse(
724
+ "call_123",
725
+ ".data | length"
726
+ );
727
+ expect(result).toBe("5");
728
+
729
+ const sumResult = await cache.queryToolResponse(
730
+ "call_123",
731
+ ".data | add"
732
+ );
733
+ expect(result).toBe("5");
734
+ });
735
+ });
736
+ });