onbuzz 3.4.0 → 3.6.2

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 (562) hide show
  1. package/package.json +1 -1
  2. package/scripts/bump-version.js +116 -0
  3. package/src/__test-utils__/fixtures/malformedJson.js +31 -0
  4. package/src/__test-utils__/globalSetup.js +9 -0
  5. package/src/__test-utils__/globalTeardown.js +12 -0
  6. package/src/__test-utils__/mockFactories.js +101 -0
  7. package/src/analyzers/__tests__/CSSAnalyzer.test.js +41 -0
  8. package/src/analyzers/__tests__/ConfigValidator.test.js +362 -0
  9. package/src/analyzers/__tests__/ESLintAnalyzer.test.js +271 -0
  10. package/src/analyzers/__tests__/JavaScriptAnalyzer.test.js +40 -0
  11. package/src/analyzers/__tests__/PrettierFormatter.test.js +197 -0
  12. package/src/analyzers/__tests__/PythonAnalyzer.test.js +208 -0
  13. package/src/analyzers/__tests__/SecurityAnalyzer.test.js +303 -0
  14. package/src/analyzers/__tests__/SparrowAnalyzer.test.js +270 -0
  15. package/src/analyzers/__tests__/TypeScriptAnalyzer.test.js +187 -0
  16. package/src/core/__tests__/agentPool.test.js +601 -0
  17. package/src/core/__tests__/agentScheduler.test.js +576 -0
  18. package/src/core/__tests__/contextManager.test.js +252 -0
  19. package/src/core/__tests__/flowExecutor.test.js +262 -0
  20. package/src/core/__tests__/messageProcessor.test.js +627 -0
  21. package/src/core/__tests__/orchestrator.test.js +257 -0
  22. package/src/core/__tests__/stateManager.test.js +375 -0
  23. package/src/core/agentPool.js +26 -4
  24. package/src/core/agentScheduler.js +79 -21
  25. package/src/core/messageProcessor.js +110 -2
  26. package/src/index.js +27 -11
  27. package/src/interfaces/__tests__/imageServing.test.js +228 -0
  28. package/src/interfaces/terminal/__tests__/smoke/imports.test.js +3 -5
  29. package/src/interfaces/webServer.js +97 -13
  30. package/src/services/__tests__/agentActivityService.test.js +319 -0
  31. package/src/services/__tests__/apiKeyManager.test.js +206 -0
  32. package/src/services/__tests__/benchmarkService.test.js +184 -0
  33. package/src/services/__tests__/budgetService.test.js +211 -0
  34. package/src/services/__tests__/contextInjectionService.test.js +205 -0
  35. package/src/services/__tests__/conversationCompactionService.test.js +280 -0
  36. package/src/services/__tests__/credentialVault.test.js +469 -0
  37. package/src/services/__tests__/errorHandler.test.js +314 -0
  38. package/src/services/__tests__/fileAttachmentService.test.js +278 -0
  39. package/src/services/__tests__/flowContextService.test.js +199 -0
  40. package/src/services/__tests__/memoryService.test.js +450 -0
  41. package/src/services/__tests__/modelRouterService.test.js +388 -0
  42. package/src/services/__tests__/modelsService.test.js +261 -0
  43. package/src/services/__tests__/portRegistry.test.js +123 -0
  44. package/src/services/__tests__/projectDetector.test.js +34 -0
  45. package/src/services/__tests__/promptService.test.js +242 -0
  46. package/src/services/__tests__/qualityInspector.test.js +97 -0
  47. package/src/services/__tests__/scheduleService.test.js +308 -0
  48. package/src/services/__tests__/serviceRegistry.test.js +74 -0
  49. package/src/services/__tests__/skillsService.test.js +402 -0
  50. package/src/services/__tests__/tokenCountingService.test.js +48 -0
  51. package/src/services/conversationCompactionService.js +2 -2
  52. package/src/services/visualEditorServer.js +26 -7
  53. package/src/tools/__tests__/agentCommunicationTool.test.js +500 -0
  54. package/src/tools/__tests__/agentDelayTool.test.js +342 -0
  55. package/src/tools/__tests__/asyncToolManager.test.js +344 -0
  56. package/src/tools/__tests__/baseTool.test.js +420 -0
  57. package/src/tools/__tests__/codeMapTool.test.js +348 -0
  58. package/src/tools/__tests__/fileContentReplaceTool.test.js +309 -0
  59. package/src/tools/__tests__/fileTreeTool.test.js +274 -0
  60. package/src/tools/__tests__/filesystemTool.test.js +717 -0
  61. package/src/tools/__tests__/helpTool.test.js +204 -0
  62. package/src/tools/__tests__/jobDoneTool.test.js +296 -0
  63. package/src/tools/__tests__/memoryTool.test.js +297 -0
  64. package/src/tools/__tests__/seekTool.test.js +282 -0
  65. package/src/tools/__tests__/skillsTool.test.js +226 -0
  66. package/src/tools/__tests__/staticAnalysisTool.test.js +509 -0
  67. package/src/tools/__tests__/taskManagerTool.test.js +725 -0
  68. package/src/tools/__tests__/terminalTool.test.js +384 -0
  69. package/src/tools/__tests__/userPromptTool.test.js +297 -0
  70. package/src/tools/__tests__/webTool.e2e.test.js +25 -11
  71. package/src/tools/imageTool.js +41 -5
  72. package/src/tools/webTool.js +161 -48
  73. package/src/types/__tests__/agent.test.js +499 -0
  74. package/src/types/__tests__/contextReference.test.js +606 -0
  75. package/src/types/__tests__/conversation.test.js +555 -0
  76. package/src/types/__tests__/toolCommand.test.js +584 -0
  77. package/src/types/contextReference.js +1 -1
  78. package/src/utilities/__tests__/attachmentValidator.test.js +80 -0
  79. package/src/utilities/__tests__/configManager.test.js +397 -0
  80. package/src/utilities/__tests__/constants.test.js +49 -0
  81. package/src/utilities/__tests__/directoryAccessManager.test.js +388 -0
  82. package/src/utilities/__tests__/fileProcessor.test.js +104 -0
  83. package/src/utilities/__tests__/jsonRepair.test.js +104 -0
  84. package/src/utilities/__tests__/logger.test.js +129 -0
  85. package/src/utilities/__tests__/platformUtils.test.js +87 -0
  86. package/src/utilities/__tests__/structuredFileValidator.test.js +263 -0
  87. package/src/utilities/__tests__/tagParser.test.js +887 -0
  88. package/src/utilities/__tests__/toolConstants.test.js +94 -0
  89. package/src/utilities/tagParser.js +2 -2
  90. package/web-ui/build/index.html +2 -2
  91. package/web-ui/build/static/1c-8PZzOTzp.js +1 -0
  92. package/web-ui/build/static/abap-Bcx_Au1F.js +1 -0
  93. package/web-ui/build/static/abnf-BKTLqpWA.js +1 -0
  94. package/web-ui/build/static/abnf-J05BAvJt.js +1 -0
  95. package/web-ui/build/static/accesslog-Cp8_lqVY.js +1 -0
  96. package/web-ui/build/static/actionscript-BK0UaMrm.js +1 -0
  97. package/web-ui/build/static/actionscript-CyqZUddh.js +1 -0
  98. package/web-ui/build/static/ada-BNirS6Nr.js +1 -0
  99. package/web-ui/build/static/ada-BSFWcT1O.js +1 -0
  100. package/web-ui/build/static/agda-D0NJDJg7.js +1 -0
  101. package/web-ui/build/static/al-rWARKtwb.js +1 -0
  102. package/web-ui/build/static/angelscript-fCehtOYk.js +1 -0
  103. package/web-ui/build/static/antlr4-Dn9jrnZN.js +1 -0
  104. package/web-ui/build/static/apache-DaQCsvNW.js +1 -0
  105. package/web-ui/build/static/apacheconf-dY4i0Xvz.js +1 -0
  106. package/web-ui/build/static/apex-vhS4SI46.js +1 -0
  107. package/web-ui/build/static/apl-CKRkxH90.js +1 -0
  108. package/web-ui/build/static/applescript-CWmpQIEB.js +1 -0
  109. package/web-ui/build/static/applescript-DBaX7Uqo.js +1 -0
  110. package/web-ui/build/static/aql-8s41lrIa.js +1 -0
  111. package/web-ui/build/static/arcade-w2_RhAcq.js +1 -0
  112. package/web-ui/build/static/arduino-I7BtZTu6.js +1 -0
  113. package/web-ui/build/static/arduino-h2LZErKQ.js +1 -0
  114. package/web-ui/build/static/arff-C543-5a1.js +1 -0
  115. package/web-ui/build/static/armasm-DyZdFOzz.js +1 -0
  116. package/web-ui/build/static/asciidoc-ZzENlACu.js +1 -0
  117. package/web-ui/build/static/asciidoc-_j9x9bUz.js +1 -0
  118. package/web-ui/build/static/asm6502-CsNsmBfq.js +1 -0
  119. package/web-ui/build/static/asmatmel-CkIVf_tD.js +1 -0
  120. package/web-ui/build/static/aspectj-C6AQLme_.js +1 -0
  121. package/web-ui/build/static/aspnet-5AkdiVyL.js +1 -0
  122. package/web-ui/build/static/autohotkey-BRZVABiS.js +1 -0
  123. package/web-ui/build/static/autohotkey-DVTmfk_f.js +1 -0
  124. package/web-ui/build/static/autoit-3UEcWu5a.js +1 -0
  125. package/web-ui/build/static/autoit-BDByIKSH.js +1 -0
  126. package/web-ui/build/static/avisynth-BHc4uUkP.js +1 -0
  127. package/web-ui/build/static/avrasm-BAPq8_aI.js +1 -0
  128. package/web-ui/build/static/avro-idl-BKEBYUtv.js +1 -0
  129. package/web-ui/build/static/awk-CBCkArRT.js +1 -0
  130. package/web-ui/build/static/axapta-DlOgnXSZ.js +1 -0
  131. package/web-ui/build/static/bash-C6Brp5OE.js +1 -0
  132. package/web-ui/build/static/bash-DkEO7JRq.js +1 -0
  133. package/web-ui/build/static/basic-DG6TYB0R.js +1 -0
  134. package/web-ui/build/static/basic-DRPcNfAn.js +1 -0
  135. package/web-ui/build/static/batch-DdjZ5KC1.js +1 -0
  136. package/web-ui/build/static/bbcode-DCXEEs2w.js +1 -0
  137. package/web-ui/build/static/bicep-CpLhfOwt.js +1 -0
  138. package/web-ui/build/static/birb-DNWkqgQm.js +1 -0
  139. package/web-ui/build/static/bison-DwxbQHJ9.js +1 -0
  140. package/web-ui/build/static/bnf-Cgnt7npj.js +1 -0
  141. package/web-ui/build/static/bnf-DSTq_eu9.js +1 -0
  142. package/web-ui/build/static/brainfuck-Bi8mGutW.js +1 -0
  143. package/web-ui/build/static/brainfuck-DOWfqVtR.js +1 -0
  144. package/web-ui/build/static/brightscript-D95pbP-v.js +1 -0
  145. package/web-ui/build/static/bro-BrDVwXeg.js +1 -0
  146. package/web-ui/build/static/bsl-BMoXI84g.js +1 -0
  147. package/web-ui/build/static/c-CKH4C7-Z.js +1 -0
  148. package/web-ui/build/static/c-Z0txyaeJ.js +1 -0
  149. package/web-ui/build/static/c-like-Dzm9dMmR.js +1 -0
  150. package/web-ui/build/static/cal-DoyAwiUt.js +1 -0
  151. package/web-ui/build/static/capnproto-DeIi9LOH.js +1 -0
  152. package/web-ui/build/static/ceylon-Coim6DIe.js +1 -0
  153. package/web-ui/build/static/cfscript-CwsndC-j.js +1 -0
  154. package/web-ui/build/static/chaiscript-D6Aq-PSv.js +1 -0
  155. package/web-ui/build/static/cil-vi56VRk_.js +1 -0
  156. package/web-ui/build/static/clean-BfpKrTdp.js +1 -0
  157. package/web-ui/build/static/clojure-DUtl6BaB.js +1 -0
  158. package/web-ui/build/static/clojure-DXJHtDlY.js +1 -0
  159. package/web-ui/build/static/clojure-repl-BxwP5C3g.js +1 -0
  160. package/web-ui/build/static/cmake-C9_VZ1vH.js +1 -0
  161. package/web-ui/build/static/cmake-dplO-PGD.js +1 -0
  162. package/web-ui/build/static/cobol-DsZhu02V.js +1 -0
  163. package/web-ui/build/static/coffeescript-Cw9jtGNP.js +1 -0
  164. package/web-ui/build/static/coffeescript-DvDt4T2l.js +1 -0
  165. package/web-ui/build/static/concurnas-Bzc_Dcdd.js +1 -0
  166. package/web-ui/build/static/coq-Cv-5BqGo.js +1 -0
  167. package/web-ui/build/static/coq-DWFe2ssK.js +1 -0
  168. package/web-ui/build/static/cos-D6Lc6Cah.js +1 -0
  169. package/web-ui/build/static/cpp-BFmLjd76.js +1 -0
  170. package/web-ui/build/static/cpp-DVQgbHji.js +1 -0
  171. package/web-ui/build/static/crmsh-Cqveth9p.js +1 -0
  172. package/web-ui/build/static/crystal-0syYaH4Y.js +1 -0
  173. package/web-ui/build/static/crystal-Noptp-kr.js +1 -0
  174. package/web-ui/build/static/csharp-B799cFMH.js +1 -0
  175. package/web-ui/build/static/csharp-_HlvMZzJ.js +1 -0
  176. package/web-ui/build/static/cshtml-CnwOXlhP.js +1 -0
  177. package/web-ui/build/static/csp-1ffxIG_-.js +1 -0
  178. package/web-ui/build/static/csp-Bws60bPu.js +1 -0
  179. package/web-ui/build/static/css-BGdwXzpm.js +1 -0
  180. package/web-ui/build/static/css-extras-DZCECiOa.js +1 -0
  181. package/web-ui/build/static/csv-FMFGT0T4.js +1 -0
  182. package/web-ui/build/static/cypher-DnXoEwRp.js +1 -0
  183. package/web-ui/build/static/d-CBrts1xB.js +1 -0
  184. package/web-ui/build/static/d-qrJLxk2L.js +1 -0
  185. package/web-ui/build/static/dart-3vBSXJVV.js +1 -0
  186. package/web-ui/build/static/dart-Cj5b7BV9.js +1 -0
  187. package/web-ui/build/static/dataweave-BuFf63rk.js +1 -0
  188. package/web-ui/build/static/dax-Cl-se1JI.js +1 -0
  189. package/web-ui/build/static/delphi-CLkRb26y.js +1 -0
  190. package/web-ui/build/static/dhall--TIL2Z--.js +1 -0
  191. package/web-ui/build/static/diff-Bn-XL2om.js +1 -0
  192. package/web-ui/build/static/diff-BsTwly4w.js +1 -0
  193. package/web-ui/build/static/django-BfRtHnTS.js +1 -0
  194. package/web-ui/build/static/django-Dm9O4e3A.js +1 -0
  195. package/web-ui/build/static/dns-BIVEp3uD.js +1 -0
  196. package/web-ui/build/static/dns-zone-file-CO7LnOdh.js +1 -0
  197. package/web-ui/build/static/docker-BhwMip1R.js +1 -0
  198. package/web-ui/build/static/dockerfile-8Tjw9_jF.js +1 -0
  199. package/web-ui/build/static/dos-CRMiAo46.js +1 -0
  200. package/web-ui/build/static/dot-DPpW7LrJ.js +1 -0
  201. package/web-ui/build/static/dsconfig-D0zbYilI.js +1 -0
  202. package/web-ui/build/static/dts-C_-yqWEL.js +1 -0
  203. package/web-ui/build/static/dust-CucJgqnE.js +1 -0
  204. package/web-ui/build/static/ebnf-CCHK0H6j.js +1 -0
  205. package/web-ui/build/static/ebnf-CDdAcveH.js +1 -0
  206. package/web-ui/build/static/editorconfig-f-5b95UM.js +1 -0
  207. package/web-ui/build/static/eiffel-MmghFce7.js +1 -0
  208. package/web-ui/build/static/ejs-CPMN4-jk.js +1 -0
  209. package/web-ui/build/static/elixir-57Ldyw8h.js +1 -0
  210. package/web-ui/build/static/elixir-DTxnmhIt.js +1 -0
  211. package/web-ui/build/static/elm-ClV9zQoT.js +1 -0
  212. package/web-ui/build/static/elm-pX-i6o7U.js +1 -0
  213. package/web-ui/build/static/erb-BPeO9smT.js +1 -0
  214. package/web-ui/build/static/erb-BnZQ4STz.js +1 -0
  215. package/web-ui/build/static/erlang-DPtI7VK_.js +1 -0
  216. package/web-ui/build/static/erlang-KG5mg5wN.js +1 -0
  217. package/web-ui/build/static/erlang-repl-BqPVXZ3Y.js +1 -0
  218. package/web-ui/build/static/etlua-BRc0Qbbs.js +1 -0
  219. package/web-ui/build/static/excel-BAlZ9Hkj.js +1 -0
  220. package/web-ui/build/static/excel-formula-BOW-bnHh.js +1 -0
  221. package/web-ui/build/static/factor-DCCsCpGM.js +1 -0
  222. package/web-ui/build/static/false-CnqnCzBU.js +1 -0
  223. package/web-ui/build/static/firestore-security-rules-DtkQ3uEq.js +1 -0
  224. package/web-ui/build/static/fix-CfPjl4Xr.js +1 -0
  225. package/web-ui/build/static/flix-BCA3BceS.js +1 -0
  226. package/web-ui/build/static/flow-C-5ewqYW.js +1 -0
  227. package/web-ui/build/static/fortran-CfDtl8An.js +1 -0
  228. package/web-ui/build/static/fortran-DQ_knNPt.js +1 -0
  229. package/web-ui/build/static/fsharp-CcVQ3IqX.js +1 -0
  230. package/web-ui/build/static/fsharp-olQ6ojCa.js +1 -0
  231. package/web-ui/build/static/ftl-DIWHDyWt.js +1 -0
  232. package/web-ui/build/static/gams-_NVFTSj5.js +1 -0
  233. package/web-ui/build/static/gap-DIG5TV2b.js +1 -0
  234. package/web-ui/build/static/gauss-C8rjPjTG.js +1 -0
  235. package/web-ui/build/static/gcode-8wJu4gcL.js +1 -0
  236. package/web-ui/build/static/gcode-DjHf417I.js +1 -0
  237. package/web-ui/build/static/gdscript-BZdmRCYu.js +1 -0
  238. package/web-ui/build/static/gedcom-BVFJ8C_Q.js +1 -0
  239. package/web-ui/build/static/gherkin-7NQkoaub.js +1 -0
  240. package/web-ui/build/static/gherkin-bSpNbJ48.js +1 -0
  241. package/web-ui/build/static/git-BRY_UXsc.js +1 -0
  242. package/web-ui/build/static/glsl-Ck6ShGRC.js +1 -0
  243. package/web-ui/build/static/glsl-jwCJ0Pmg.js +1 -0
  244. package/web-ui/build/static/gml-BAjG4Kr2.js +1 -0
  245. package/web-ui/build/static/gml-BnhKb5Da.js +1 -0
  246. package/web-ui/build/static/gn-Dux09iX8.js +1 -0
  247. package/web-ui/build/static/go-BOG-9Cqk.js +1 -0
  248. package/web-ui/build/static/go-BWZB_3b5.js +1 -0
  249. package/web-ui/build/static/go-module-BVLW7KBE.js +1 -0
  250. package/web-ui/build/static/golo-BD_SOfwL.js +1 -0
  251. package/web-ui/build/static/gradle-zSadWOD2.js +1 -0
  252. package/web-ui/build/static/graphql-BQlyj78B.js +1 -0
  253. package/web-ui/build/static/groovy-BurRMqQS.js +1 -0
  254. package/web-ui/build/static/groovy-SP58zwlE.js +1 -0
  255. package/web-ui/build/static/haml-eZ5ah5KY.js +1 -0
  256. package/web-ui/build/static/haml-lFC47IZb.js +1 -0
  257. package/web-ui/build/static/handlebars-B4UXrB-Q.js +1 -0
  258. package/web-ui/build/static/handlebars-CQ-Q5HnC.js +1 -0
  259. package/web-ui/build/static/haskell-AQrRyTSy.js +1 -0
  260. package/web-ui/build/static/haskell-BZTSbaB_.js +1 -0
  261. package/web-ui/build/static/haxe-5l1X6ESp.js +1 -0
  262. package/web-ui/build/static/haxe-DBn90muG.js +1 -0
  263. package/web-ui/build/static/hcl-CnMewPLM.js +1 -0
  264. package/web-ui/build/static/hlsl-RAtuBzr5.js +1 -0
  265. package/web-ui/build/static/hoon-CSqRU_4M.js +1 -0
  266. package/web-ui/build/static/hpkp-_gNbXcHt.js +1 -0
  267. package/web-ui/build/static/hsp-DY1V4au3.js +1 -0
  268. package/web-ui/build/static/hsts-j5z2jJo8.js +1 -0
  269. package/web-ui/build/static/htmlbars-C5EHvatr.js +1 -0
  270. package/web-ui/build/static/http-DgWgQrZh.js +1 -0
  271. package/web-ui/build/static/http-DphJL0q2.js +1 -0
  272. package/web-ui/build/static/hy-DnBqjPsB.js +1 -0
  273. package/web-ui/build/static/ichigojam-DeiCOKyF.js +1 -0
  274. package/web-ui/build/static/icon-CWANFWY5.js +1 -0
  275. package/web-ui/build/static/icu-message-format-C7w3vpgC.js +1 -0
  276. package/web-ui/build/static/idris-BeD8eULz.js +1 -0
  277. package/web-ui/build/static/iecst-BIznHXqY.js +1 -0
  278. package/web-ui/build/static/ignore-BcFgcNaS.js +1 -0
  279. package/web-ui/build/static/index-D8uVofpo.js +13 -0
  280. package/web-ui/build/static/index-DPFadqM6.css +1 -0
  281. package/web-ui/build/static/index-SkOgWEAU.js +1 -0
  282. package/web-ui/build/static/index-Vd3WlhtC.js +747 -0
  283. package/web-ui/build/static/inform7-CkQD_jz-.js +1 -0
  284. package/web-ui/build/static/inform7-phQiuDty.js +1 -0
  285. package/web-ui/build/static/ini-Bw_QAbzV.js +1 -0
  286. package/web-ui/build/static/ini-CB8ZxX7y.js +1 -0
  287. package/web-ui/build/static/io-D6IgpCmL.js +1 -0
  288. package/web-ui/build/static/irpf90-Ctj0koST.js +1 -0
  289. package/web-ui/build/static/isbl-D2mGcH87.js +1 -0
  290. package/web-ui/build/static/j-KvHmDBWH.js +1 -0
  291. package/web-ui/build/static/java-BeBIdo5i.js +1 -0
  292. package/web-ui/build/static/java-llFZkHLe.js +1 -0
  293. package/web-ui/build/static/javadoc-DXvQGu0s.js +1 -0
  294. package/web-ui/build/static/javadoclike-B5qdA9KZ.js +1 -0
  295. package/web-ui/build/static/javascript-De6HzHvc.js +1 -0
  296. package/web-ui/build/static/javastacktrace-C5MolKiP.js +1 -0
  297. package/web-ui/build/static/jboss-cli-2TXd54zo.js +1 -0
  298. package/web-ui/build/static/jexl-W4AVA9fi.js +1 -0
  299. package/web-ui/build/static/jolie-DTJKRMZN.js +1 -0
  300. package/web-ui/build/static/jq-BYg-QQKh.js +1 -0
  301. package/web-ui/build/static/js-extras-BrYWd2VE.js +1 -0
  302. package/web-ui/build/static/js-templates-DorYpbHq.js +1 -0
  303. package/web-ui/build/static/jsdoc-CRF8n9pZ.js +1 -0
  304. package/web-ui/build/static/json-Dlcd7rla.js +1 -0
  305. package/web-ui/build/static/json-rhOJZzzZ.js +1 -0
  306. package/web-ui/build/static/json5-hTq1nNIW.js +1 -0
  307. package/web-ui/build/static/jsonp-CMg9Xvol.js +1 -0
  308. package/web-ui/build/static/jsstacktrace-hEeYxHtB.js +1 -0
  309. package/web-ui/build/static/jsx-B7PtA8WD.js +1 -0
  310. package/web-ui/build/static/julia-CNiEEY-n.js +1 -0
  311. package/web-ui/build/static/julia-eE0_SJlc.js +1 -0
  312. package/web-ui/build/static/julia-repl-CUJTTT4C.js +1 -0
  313. package/web-ui/build/static/keepalived-GUWJBqyD.js +1 -0
  314. package/web-ui/build/static/keyman-Bdf9MJyu.js +1 -0
  315. package/web-ui/build/static/kotlin-Ch6Bej5W.js +1 -0
  316. package/web-ui/build/static/kotlin-DFJ7D7Lw.js +1 -0
  317. package/web-ui/build/static/kumir-DJLIjcCs.js +1 -0
  318. package/web-ui/build/static/kusto-BM0YTwU4.js +1 -0
  319. package/web-ui/build/static/lasso-Z1DVS84K.js +1 -0
  320. package/web-ui/build/static/latex-BWbw71RK.js +1 -0
  321. package/web-ui/build/static/latex-CMzqmbhw.js +1 -0
  322. package/web-ui/build/static/latte-C1g8_grc.js +1 -0
  323. package/web-ui/build/static/ldif-C6QH3OIL.js +1 -0
  324. package/web-ui/build/static/leaf-CbR--ZsH.js +1 -0
  325. package/web-ui/build/static/less-ClVrKh2Z.js +1 -0
  326. package/web-ui/build/static/less-DNSxm8UA.js +1 -0
  327. package/web-ui/build/static/lilypond-lTzf_sWt.js +1 -0
  328. package/web-ui/build/static/liquid-Bn91mVfC.js +1 -0
  329. package/web-ui/build/static/lisp-CU3bHohQ.js +1 -0
  330. package/web-ui/build/static/lisp-DbgzE9W8.js +1 -0
  331. package/web-ui/build/static/livecodeserver-Cse1Uz3H.js +1 -0
  332. package/web-ui/build/static/livescript-BaxbgzWP.js +1 -0
  333. package/web-ui/build/static/livescript-nJt61DBy.js +1 -0
  334. package/web-ui/build/static/llvm-DBboo6UI.js +1 -0
  335. package/web-ui/build/static/llvm-vIy7XYVy.js +1 -0
  336. package/web-ui/build/static/log-CT7nfoDW.js +1 -0
  337. package/web-ui/build/static/lolcode-CUKVytZh.js +1 -0
  338. package/web-ui/build/static/lsl-CsAOlGF2.js +1 -0
  339. package/web-ui/build/static/lua-BuU2FFxP.js +1 -0
  340. package/web-ui/build/static/lua-CiDuKQaa.js +1 -0
  341. package/web-ui/build/static/magma-7vR0zcmS.js +1 -0
  342. package/web-ui/build/static/makefile-Buiz-Msh.js +1 -0
  343. package/web-ui/build/static/makefile-DXW_-6OY.js +1 -0
  344. package/web-ui/build/static/markdown-Bk5DUoGY.js +1 -0
  345. package/web-ui/build/static/markdown-CRS5W_Ai.js +1 -0
  346. package/web-ui/build/static/markup-templating-24odpmF3.js +1 -0
  347. package/web-ui/build/static/mathematica-BxcwhJUp.js +1 -0
  348. package/web-ui/build/static/matlab-3pJYx_Fb.js +1 -0
  349. package/web-ui/build/static/matlab-BqlRrzMf.js +1 -0
  350. package/web-ui/build/static/maxima-DlCfUpcj.js +1 -0
  351. package/web-ui/build/static/maxscript-Cu_gCaFU.js +1 -0
  352. package/web-ui/build/static/mel-D7iQ-5Up.js +1 -0
  353. package/web-ui/build/static/mel-DzBKNpoN.js +1 -0
  354. package/web-ui/build/static/mercury-Dfrb-i8A.js +1 -0
  355. package/web-ui/build/static/mermaid-WN7V2_Eq.js +1 -0
  356. package/web-ui/build/static/mipsasm-CcijzL0q.js +1 -0
  357. package/web-ui/build/static/mizar-Bk68zACP.js +1 -0
  358. package/web-ui/build/static/mizar-Twc2-iZ4.js +1 -0
  359. package/web-ui/build/static/mojolicious-DBbo2S7X.js +1 -0
  360. package/web-ui/build/static/mongodb-2RsFIjgg.js +1 -0
  361. package/web-ui/build/static/monkey-CPXtQ0Bf.js +1 -0
  362. package/web-ui/build/static/monkey-DjV7Wcek.js +1 -0
  363. package/web-ui/build/static/moonscript-B5M5as70.js +1 -0
  364. package/web-ui/build/static/moonscript-D1BHW4Il.js +1 -0
  365. package/web-ui/build/static/n1ql-D0heNDBD.js +1 -0
  366. package/web-ui/build/static/n1ql-DfHqXQD7.js +1 -0
  367. package/web-ui/build/static/n4js-CaPf44Dz.js +1 -0
  368. package/web-ui/build/static/nand2tetris-hdl-D1nf9mn4.js +1 -0
  369. package/web-ui/build/static/naniscript-DnCnu5ZX.js +1 -0
  370. package/web-ui/build/static/nasm-BZrSaMsK.js +1 -0
  371. package/web-ui/build/static/neon-D29Grm2v.js +1 -0
  372. package/web-ui/build/static/nevod-DgSNbQkE.js +1 -0
  373. package/web-ui/build/static/nginx-BAaDGDfT.js +1 -0
  374. package/web-ui/build/static/nginx-BvJ1lrHX.js +1 -0
  375. package/web-ui/build/static/nim--9zzVe5F.js +1 -0
  376. package/web-ui/build/static/nim-Br1relpU.js +1 -0
  377. package/web-ui/build/static/nix--0ftErCy.js +1 -0
  378. package/web-ui/build/static/nix-104ztQqr.js +1 -0
  379. package/web-ui/build/static/node-repl-BUMAf7_p.js +1 -0
  380. package/web-ui/build/static/nsis-BaeKybNA.js +1 -0
  381. package/web-ui/build/static/nsis-CdZEv2iA.js +1 -0
  382. package/web-ui/build/static/objectivec-DBB4ymdg.js +1 -0
  383. package/web-ui/build/static/objectivec-kFYXC6g4.js +1 -0
  384. package/web-ui/build/static/ocaml-D1GXvN7c.js +1 -0
  385. package/web-ui/build/static/ocaml-D80jRMPE.js +1 -0
  386. package/web-ui/build/static/opencl-fb7BfRdO.js +1 -0
  387. package/web-ui/build/static/openqasm-CWUBrR2w.js +1 -0
  388. package/web-ui/build/static/openscad-Dim7ILSL.js +1 -0
  389. package/web-ui/build/static/oxygene-BSwApkwz.js +1 -0
  390. package/web-ui/build/static/oz-CMtRoi5F.js +1 -0
  391. package/web-ui/build/static/parigp-AH8cZ38D.js +1 -0
  392. package/web-ui/build/static/parser-bBNjuhG3.js +1 -0
  393. package/web-ui/build/static/parser3-DUtoWEAd.js +1 -0
  394. package/web-ui/build/static/pascal-Cr3DPIYT.js +1 -0
  395. package/web-ui/build/static/pascaligo-pWW12jfs.js +1 -0
  396. package/web-ui/build/static/pcaxis-DBw9rxmr.js +1 -0
  397. package/web-ui/build/static/peoplecode-aCpMPm_s.js +1 -0
  398. package/web-ui/build/static/perl-BpZ7GmJ3.js +1 -0
  399. package/web-ui/build/static/perl-fnHTrqJL.js +1 -0
  400. package/web-ui/build/static/pf-Dz352ty7.js +1 -0
  401. package/web-ui/build/static/pgsql-CHPUdlI_.js +1 -0
  402. package/web-ui/build/static/php-BRwItjmS.js +1 -0
  403. package/web-ui/build/static/php-CrX_kswO.js +1 -0
  404. package/web-ui/build/static/php-extras-BmeRXDSO.js +1 -0
  405. package/web-ui/build/static/php-template-B0MFJ9RR.js +1 -0
  406. package/web-ui/build/static/phpdoc-wAPkJj9X.js +1 -0
  407. package/web-ui/build/static/plaintext-CmPk1gvP.js +1 -0
  408. package/web-ui/build/static/plsql-pWVw0sCJ.js +1 -0
  409. package/web-ui/build/static/pony-B4SXhyDS.js +1 -0
  410. package/web-ui/build/static/powerquery-ZJ28bdRR.js +1 -0
  411. package/web-ui/build/static/powershell-CWg1ca6z.js +1 -0
  412. package/web-ui/build/static/powershell-Dnl0aBXc.js +1 -0
  413. package/web-ui/build/static/processing-CbYVU7hZ.js +1 -0
  414. package/web-ui/build/static/processing-DnroirEw.js +1 -0
  415. package/web-ui/build/static/profile-DLNc-MTA.js +1 -0
  416. package/web-ui/build/static/prolog-4KlPFQus.js +1 -0
  417. package/web-ui/build/static/prolog-CtUicY87.js +1 -0
  418. package/web-ui/build/static/promql-C_i6OJVg.js +1 -0
  419. package/web-ui/build/static/properties-Cj0lBOSP.js +1 -0
  420. package/web-ui/build/static/properties-vGFibcz9.js +1 -0
  421. package/web-ui/build/static/protobuf-BOIGxbSP.js +1 -0
  422. package/web-ui/build/static/protobuf-CQ3su-J7.js +1 -0
  423. package/web-ui/build/static/psl-DeG5_YUF.js +1 -0
  424. package/web-ui/build/static/pug-BieVVXYz.js +1 -0
  425. package/web-ui/build/static/puppet-Ba40SVKU.js +1 -0
  426. package/web-ui/build/static/puppet-D7BzrcGt.js +1 -0
  427. package/web-ui/build/static/pure-DZnkz1iT.js +1 -0
  428. package/web-ui/build/static/purebasic-CLLZW_6G.js +1 -0
  429. package/web-ui/build/static/purebasic-CYPZo_H6.js +1 -0
  430. package/web-ui/build/static/purescript-Dyjfu5Id.js +1 -0
  431. package/web-ui/build/static/python-BdIWKxdN.js +1 -0
  432. package/web-ui/build/static/python-ofKsqxv7.js +1 -0
  433. package/web-ui/build/static/python-repl-DiTYb1xK.js +1 -0
  434. package/web-ui/build/static/q-B4P0If_I.js +1 -0
  435. package/web-ui/build/static/q-t_17xfY8.js +1 -0
  436. package/web-ui/build/static/qml-B5WhiN48.js +1 -0
  437. package/web-ui/build/static/qml-Dq0cESXJ.js +1 -0
  438. package/web-ui/build/static/qore-DCx30XRf.js +1 -0
  439. package/web-ui/build/static/qsharp-UrBScekp.js +1 -0
  440. package/web-ui/build/static/r-B0Ty1RKQ.js +1 -0
  441. package/web-ui/build/static/r-B0za8QKS.js +1 -0
  442. package/web-ui/build/static/racket-Dj6WEyhS.js +1 -0
  443. package/web-ui/build/static/reason-dj9hJSfr.js +1 -0
  444. package/web-ui/build/static/reasonml-B-q5_wag.js +1 -0
  445. package/web-ui/build/static/regex-4HEc5C1m.js +1 -0
  446. package/web-ui/build/static/rego-BdQe18RK.js +1 -0
  447. package/web-ui/build/static/renpy-CVMA2llL.js +1 -0
  448. package/web-ui/build/static/rest-9B4JWVGr.js +1 -0
  449. package/web-ui/build/static/rib-DR-U8OaT.js +1 -0
  450. package/web-ui/build/static/rip-Bu2t_rFZ.js +1 -0
  451. package/web-ui/build/static/roboconf-CJeXD5-I.js +1 -0
  452. package/web-ui/build/static/roboconf-DzDTVrdM.js +1 -0
  453. package/web-ui/build/static/robotframework-CR7KyPpN.js +1 -0
  454. package/web-ui/build/static/routeros-B2741z2k.js +1 -0
  455. package/web-ui/build/static/rsl-B9F_ZCgv.js +1 -0
  456. package/web-ui/build/static/ruby-I2JTNgyY.js +1 -0
  457. package/web-ui/build/static/ruby-QGDPOmJX.js +1 -0
  458. package/web-ui/build/static/ruleslanguage-CGzXEUCO.js +1 -0
  459. package/web-ui/build/static/rust-BxW5-WOm.js +1 -0
  460. package/web-ui/build/static/rust-CSOA43di.js +1 -0
  461. package/web-ui/build/static/sas-Bclfx4g3.js +1 -0
  462. package/web-ui/build/static/sas-xbQaiYjT.js +1 -0
  463. package/web-ui/build/static/sass-DJPbdNwd.js +1 -0
  464. package/web-ui/build/static/scala-Bo18NtHQ.js +1 -0
  465. package/web-ui/build/static/scala-Cy0JH-SG.js +1 -0
  466. package/web-ui/build/static/scheme-BjcWWjIx.js +1 -0
  467. package/web-ui/build/static/scheme-DQdj8PzN.js +1 -0
  468. package/web-ui/build/static/scilab-Bn1KHdK-.js +1 -0
  469. package/web-ui/build/static/scss-B1twkZBz.js +1 -0
  470. package/web-ui/build/static/scss-DmOuMI4v.js +1 -0
  471. package/web-ui/build/static/shell-BUlkJG0S.js +1 -0
  472. package/web-ui/build/static/shell-session-Bke-svxA.js +1 -0
  473. package/web-ui/build/static/smali-Ch9S16HV.js +1 -0
  474. package/web-ui/build/static/smali-D_yDr_Aj.js +1 -0
  475. package/web-ui/build/static/smalltalk-B9TfQ5Md.js +1 -0
  476. package/web-ui/build/static/smalltalk-EwbZxZsR.js +1 -0
  477. package/web-ui/build/static/smarty-9kDPpeSm.js +1 -0
  478. package/web-ui/build/static/sml-2fEfT7rd.js +1 -0
  479. package/web-ui/build/static/sml-BiwoLNk7.js +1 -0
  480. package/web-ui/build/static/solidity-n_x8Oe0h.js +1 -0
  481. package/web-ui/build/static/solution-file-B2mvjI3e.js +1 -0
  482. package/web-ui/build/static/soy-DPkgKBIS.js +1 -0
  483. package/web-ui/build/static/sparql-Cy95tds0.js +1 -0
  484. package/web-ui/build/static/splunk-spl-Ym3z9ouN.js +1 -0
  485. package/web-ui/build/static/sqf-CXZTG8WE.js +1 -0
  486. package/web-ui/build/static/sqf-Cwi3yg7f.js +1 -0
  487. package/web-ui/build/static/sql-DPxSQY4S.js +1 -0
  488. package/web-ui/build/static/sql-peh7ijGj.js +1 -0
  489. package/web-ui/build/static/sql_more-0YAbAuPw.js +1 -0
  490. package/web-ui/build/static/squirrel-CphzjV0e.js +1 -0
  491. package/web-ui/build/static/stan-0-xZ95-O.js +1 -0
  492. package/web-ui/build/static/stan-CaI4__2g.js +1 -0
  493. package/web-ui/build/static/stata-BrbzrGSs.js +1 -0
  494. package/web-ui/build/static/step21-C_qeyVLw.js +1 -0
  495. package/web-ui/build/static/stylus-Btycb2sZ.js +1 -0
  496. package/web-ui/build/static/stylus-FoBJ7jki.js +1 -0
  497. package/web-ui/build/static/subunit-Dpg-m04-.js +1 -0
  498. package/web-ui/build/static/swift-Cr9uZmgb.js +1 -0
  499. package/web-ui/build/static/swift-hGLFtD7e.js +1 -0
  500. package/web-ui/build/static/systemd-Bls2D9Vj.js +1 -0
  501. package/web-ui/build/static/t4-cs-C4qDO-jJ.js +1 -0
  502. package/web-ui/build/static/t4-templating-BbCFPMPO.js +1 -0
  503. package/web-ui/build/static/t4-vb-D1zoEccT.js +1 -0
  504. package/web-ui/build/static/taggerscript-CWHk9Gih.js +1 -0
  505. package/web-ui/build/static/tap-Bjt0UnzV.js +1 -0
  506. package/web-ui/build/static/tap-BnHKwLQs.js +1 -0
  507. package/web-ui/build/static/tcl-Zo9kx4y-.js +1 -0
  508. package/web-ui/build/static/tcl-fzLmefkt.js +1 -0
  509. package/web-ui/build/static/textile-9lIlUPH5.js +1 -0
  510. package/web-ui/build/static/thrift-M3K6r5Cy.js +1 -0
  511. package/web-ui/build/static/toml-HpaKqckc.js +1 -0
  512. package/web-ui/build/static/tp-DFKuxrKj.js +1 -0
  513. package/web-ui/build/static/tremor-D4_bUtMB.js +1 -0
  514. package/web-ui/build/static/tsx-o1RT-T90.js +1 -0
  515. package/web-ui/build/static/tt2-1xDqcN_2.js +1 -0
  516. package/web-ui/build/static/turtle-Dlt-aGky.js +1 -0
  517. package/web-ui/build/static/twig-CJ_BnGSR.js +1 -0
  518. package/web-ui/build/static/twig-CjsiSQb6.js +1 -0
  519. package/web-ui/build/static/typescript-B8B9zUn-.js +1 -0
  520. package/web-ui/build/static/typescript-D0Jgo8O7.js +1 -0
  521. package/web-ui/build/static/typoscript-C8Qke4ZB.js +1 -0
  522. package/web-ui/build/static/unrealscript-YxJdDNZ3.js +1 -0
  523. package/web-ui/build/static/uorazor-CtEVnqBv.js +1 -0
  524. package/web-ui/build/static/uri-YdaiQl4c.js +1 -0
  525. package/web-ui/build/static/v-CIyttMDD.js +1 -0
  526. package/web-ui/build/static/vala-DGslcym_.js +1 -0
  527. package/web-ui/build/static/vala-GFPx3uEJ.js +1 -0
  528. package/web-ui/build/static/vbnet-B20itab-.js +1 -0
  529. package/web-ui/build/static/vbnet-BdoN6egk.js +1 -0
  530. package/web-ui/build/static/vbscript-PHVh6Fp_.js +1 -0
  531. package/web-ui/build/static/vbscript-html-woH1VZ7U.js +1 -0
  532. package/web-ui/build/static/velocity-DtVfCZeg.js +1 -0
  533. package/web-ui/build/static/verilog-Bt6edXvM.js +1 -0
  534. package/web-ui/build/static/verilog-k_7lr9Zq.js +1 -0
  535. package/web-ui/build/static/vhdl-BMzOgOeK.js +1 -0
  536. package/web-ui/build/static/vhdl-BcAbtPG6.js +1 -0
  537. package/web-ui/build/static/vim-DrinG9a4.js +1 -0
  538. package/web-ui/build/static/vim-WihLATJL.js +1 -0
  539. package/web-ui/build/static/visual-basic-CJnvgPjM.js +1 -0
  540. package/web-ui/build/static/warpscript-zMlbUoZs.js +1 -0
  541. package/web-ui/build/static/wasm-GUnfTBUL.js +1 -0
  542. package/web-ui/build/static/web-idl-CfaLTG_r.js +1 -0
  543. package/web-ui/build/static/wiki-13AlLoOc.js +1 -0
  544. package/web-ui/build/static/wolfram-zHocYNXW.js +1 -0
  545. package/web-ui/build/static/wren-Byq862Iu.js +1 -0
  546. package/web-ui/build/static/x86asm-CLcOnePY.js +1 -0
  547. package/web-ui/build/static/xeora-BVHqWOFS.js +1 -0
  548. package/web-ui/build/static/xl-lXi8OYfr.js +1 -0
  549. package/web-ui/build/static/xml-KZjGBKxi.js +1 -0
  550. package/web-ui/build/static/xml-doc-DrQSDcEW.js +1 -0
  551. package/web-ui/build/static/xojo-DosHeFXU.js +1 -0
  552. package/web-ui/build/static/xquery-BZN1F14Q.js +1 -0
  553. package/web-ui/build/static/xquery-Cnz7ZLFr.js +1 -0
  554. package/web-ui/build/static/yaml-BzXOcy9u.js +1 -0
  555. package/web-ui/build/static/yaml-C207y5bt.js +1 -0
  556. package/web-ui/build/static/yang-ByrBdDIg.js +1 -0
  557. package/web-ui/build/static/zephir-bahTa7of.js +1 -0
  558. package/web-ui/build/static/zig-BlFYhdtC.js +1 -0
  559. package/src/tools/browserTool.js +0 -897
  560. package/src/utilities/platformUtils.test.js +0 -98
  561. package/web-ui/build/static/index-SmQFfvBs.js +0 -746
  562. package/web-ui/build/static/index-V2ySwjHp.css +0 -1
@@ -0,0 +1,314 @@
1
+ import { jest, describe, test, expect, beforeEach, afterEach } from '@jest/globals';
2
+ import { createMockLogger, createMockConfig } from '../../__test-utils__/mockFactories.js';
3
+
4
+ // Mock constants before importing the module under test
5
+ const ERROR_TYPES = {
6
+ FILE_NOT_FOUND: 'FILE_NOT_FOUND',
7
+ PERMISSION_DENIED: 'PERMISSION_DENIED',
8
+ OPERATION_TIMEOUT: 'OPERATION_TIMEOUT',
9
+ RATE_LIMIT_EXCEEDED: 'RATE_LIMIT_EXCEEDED',
10
+ AUTHENTICATION_FAILED: 'AUTHENTICATION_FAILED',
11
+ UNKNOWN_ERROR: 'UNKNOWN_ERROR',
12
+ VALIDATION_ERROR: 'VALIDATION_ERROR',
13
+ CONFIGURATION_ERROR: 'CONFIGURATION_ERROR'
14
+ };
15
+
16
+ const HTTP_STATUS = {
17
+ OK: 200,
18
+ BAD_REQUEST: 400,
19
+ UNAUTHORIZED: 401,
20
+ FORBIDDEN: 403,
21
+ NOT_FOUND: 404,
22
+ TOO_MANY_REQUESTS: 429,
23
+ INTERNAL_SERVER_ERROR: 500,
24
+ };
25
+
26
+ jest.unstable_mockModule('../../utilities/constants.js', () => ({
27
+ ERROR_TYPES,
28
+ HTTP_STATUS,
29
+ AGENT_STATUS: {}
30
+ }));
31
+
32
+ const { ErrorHandler } = await import('../errorHandler.js');
33
+
34
+ describe('ErrorHandler', () => {
35
+ let handler;
36
+ let logger;
37
+
38
+ beforeEach(() => {
39
+ jest.useFakeTimers();
40
+ logger = createMockLogger();
41
+ handler = new ErrorHandler(createMockConfig(), logger);
42
+ });
43
+
44
+ afterEach(() => {
45
+ jest.useRealTimers();
46
+ });
47
+
48
+ // --- classifyError ---
49
+
50
+ test('classifyError returns FILE_NOT_FOUND for ENOENT errors', () => {
51
+ const error = new Error('File missing');
52
+ error.code = 'ENOENT';
53
+ const info = handler.classifyError(error);
54
+ expect(info.type).toBe(ERROR_TYPES.FILE_NOT_FOUND);
55
+ expect(info.message).toBe('File missing');
56
+ expect(info.id).toMatch(/^err_/);
57
+ expect(info.timestamp).toBeDefined();
58
+ });
59
+
60
+ test('classifyError returns PERMISSION_DENIED for EACCES errors', () => {
61
+ const error = new Error('Access denied');
62
+ error.code = 'EACCES';
63
+ const info = handler.classifyError(error);
64
+ expect(info.type).toBe(ERROR_TYPES.PERMISSION_DENIED);
65
+ });
66
+
67
+ test('classifyError returns AUTHENTICATION_FAILED for 401 status', () => {
68
+ const error = new Error('Unauthorized');
69
+ error.status = HTTP_STATUS.UNAUTHORIZED;
70
+ const info = handler.classifyError(error);
71
+ expect(info.type).toBe(ERROR_TYPES.AUTHENTICATION_FAILED);
72
+ });
73
+
74
+ test('classifyError returns RATE_LIMIT_EXCEEDED for 429 status', () => {
75
+ const error = new Error('Rate limit');
76
+ error.status = HTTP_STATUS.TOO_MANY_REQUESTS;
77
+ const info = handler.classifyError(error);
78
+ expect(info.type).toBe(ERROR_TYPES.RATE_LIMIT_EXCEEDED);
79
+ });
80
+
81
+ test('classifyError returns OPERATION_TIMEOUT for timeout message', () => {
82
+ const error = new Error('Request timed out');
83
+ const info = handler.classifyError(error);
84
+ expect(info.type).toBe(ERROR_TYPES.OPERATION_TIMEOUT);
85
+ });
86
+
87
+ test('classifyError returns VALIDATION_ERROR for validation message', () => {
88
+ const error = new Error('Input validation failed');
89
+ const info = handler.classifyError(error);
90
+ expect(info.type).toBe(ERROR_TYPES.VALIDATION_ERROR);
91
+ });
92
+
93
+ test('classifyError returns CONFIGURATION_ERROR for config message', () => {
94
+ const error = new Error('Bad configuration');
95
+ const info = handler.classifyError(error);
96
+ expect(info.type).toBe(ERROR_TYPES.CONFIGURATION_ERROR);
97
+ });
98
+
99
+ test('classifyError uses context.operation for additional classification', () => {
100
+ const error = new Error('Something generic');
101
+ const info = handler.classifyError(error, { operation: 'api_request' });
102
+ expect(info.type).toBe(ERROR_TYPES.RATE_LIMIT_EXCEEDED);
103
+ });
104
+
105
+ test('classifyError sets agentId, toolId, operationId from context', () => {
106
+ const error = new Error('fail');
107
+ const info = handler.classifyError(error, { agentId: 'a1', toolId: 't1', operationId: 'op1' });
108
+ expect(info.agentId).toBe('a1');
109
+ expect(info.toolId).toBe('t1');
110
+ expect(info.operationId).toBe('op1');
111
+ });
112
+
113
+ test('classifyError returns UNKNOWN_ERROR for unrecognizable errors', () => {
114
+ const error = new Error('something completely random with no keywords');
115
+ const info = handler.classifyError(error);
116
+ expect(info.type).toBe(ERROR_TYPES.UNKNOWN_ERROR);
117
+ });
118
+
119
+ // --- determineSeverity ---
120
+
121
+ test('determineSeverity returns critical for authentication errors', () => {
122
+ const error = new Error('Auth');
123
+ error.status = HTTP_STATUS.UNAUTHORIZED;
124
+ expect(handler.determineSeverity(error, {})).toBe('critical');
125
+ });
126
+
127
+ test('determineSeverity returns high for rate limit errors', () => {
128
+ const error = new Error('Rate limit');
129
+ error.status = HTTP_STATUS.TOO_MANY_REQUESTS;
130
+ expect(handler.determineSeverity(error, {})).toBe('high');
131
+ });
132
+
133
+ test('determineSeverity returns high when retryCount >= 3', () => {
134
+ const error = new Error('generic random error no keywords');
135
+ expect(handler.determineSeverity(error, { retryCount: 3 })).toBe('high');
136
+ });
137
+
138
+ test('determineSeverity returns medium for agent communication', () => {
139
+ const error = new Error('something random no keywords');
140
+ expect(handler.determineSeverity(error, { agentId: 'a1', operation: 'agent_communication' })).toBe('medium');
141
+ });
142
+
143
+ test('determineSeverity returns low for generic low-priority errors', () => {
144
+ const error = new Error('some problem');
145
+ error.code = 'ENOENT';
146
+ expect(handler.determineSeverity(error, {})).toBe('low');
147
+ });
148
+
149
+ // --- isRecoverable ---
150
+
151
+ test('isRecoverable returns false for AUTHENTICATION_FAILED', () => {
152
+ const info = handler.classifyError(Object.assign(new Error('auth'), { status: HTTP_STATUS.UNAUTHORIZED }));
153
+ expect(info.recoverable).toBe(false);
154
+ });
155
+
156
+ test('isRecoverable returns false when retryCount >= maxRetries', () => {
157
+ const info = { type: ERROR_TYPES.FILE_NOT_FOUND, retryCount: 5, maxRetries: 2, severity: 'low' };
158
+ expect(handler.isRecoverable(info)).toBe(false);
159
+ });
160
+
161
+ test('isRecoverable returns true for transient recoverable errors', () => {
162
+ const info = { type: ERROR_TYPES.OPERATION_TIMEOUT, retryCount: 0, maxRetries: 3, severity: 'high' };
163
+ expect(handler.isRecoverable(info)).toBe(true);
164
+ });
165
+
166
+ // --- calculateRetryDelay ---
167
+
168
+ test('calculateRetryDelay returns increasing delays with exponential backoff', () => {
169
+ const delay0 = handler.calculateRetryDelay({ retryCount: 0 });
170
+ // Base delay = 1000 * 2^0 + jitter(0-1000) = 1000-2000
171
+ expect(delay0).toBeGreaterThanOrEqual(1000);
172
+ expect(delay0).toBeLessThanOrEqual(2000);
173
+ });
174
+
175
+ test('calculateRetryDelay caps at 30000ms', () => {
176
+ const delay = handler.calculateRetryDelay({ retryCount: 20 });
177
+ expect(delay).toBeLessThanOrEqual(30000);
178
+ });
179
+
180
+ // --- subscribe / notifySubscribers ---
181
+
182
+ test('subscribe registers a callback and unsubscribe removes it', () => {
183
+ const cb = jest.fn();
184
+ const unsub = handler.subscribe(cb);
185
+ handler.notifySubscribers({ id: 'err1' }, { success: true });
186
+ expect(cb).toHaveBeenCalledTimes(1);
187
+ expect(cb).toHaveBeenCalledWith({ id: 'err1' }, { success: true });
188
+
189
+ unsub();
190
+ handler.notifySubscribers({ id: 'err2' }, { success: false });
191
+ expect(cb).toHaveBeenCalledTimes(1); // not called again
192
+ });
193
+
194
+ test('notifySubscribers logs error if callback throws', () => {
195
+ handler.subscribe(() => { throw new Error('boom'); });
196
+ handler.notifySubscribers({}, {});
197
+ expect(logger.error).toHaveBeenCalledWith('Error subscriber callback failed', expect.any(Object));
198
+ });
199
+
200
+ // --- getErrorStats / clearErrorStats / updateErrorStats ---
201
+
202
+ test('getErrorStats returns accumulated stats with type and agent counts', () => {
203
+ handler.updateErrorStats({ type: ERROR_TYPES.FILE_NOT_FOUND, agentId: 'a1' });
204
+ handler.updateErrorStats({ type: ERROR_TYPES.FILE_NOT_FOUND });
205
+ handler.updateErrorStats({ type: ERROR_TYPES.OPERATION_TIMEOUT, agentId: 'a1' });
206
+ const stats = handler.getErrorStats();
207
+ expect(stats.totalErrors).toBe(3);
208
+ expect(stats.errorsByType[ERROR_TYPES.FILE_NOT_FOUND]).toBe(2);
209
+ expect(stats.errorsByType[ERROR_TYPES.OPERATION_TIMEOUT]).toBe(1);
210
+ expect(stats.errorsByAgent['a1']).toBe(2);
211
+ });
212
+
213
+ test('clearErrorStats resets all counters and empties queue', () => {
214
+ handler.updateErrorStats({ type: ERROR_TYPES.FILE_NOT_FOUND });
215
+ handler.errorQueue.push({ test: true });
216
+ handler.clearErrorStats();
217
+ const stats = handler.getErrorStats();
218
+ expect(stats.totalErrors).toBe(0);
219
+ expect(stats.queueLength).toBe(0);
220
+ expect(stats.recoveryAttempts).toBe(0);
221
+ expect(stats.successfulRecoveries).toBe(0);
222
+ expect(stats.criticalErrors).toBe(0);
223
+ });
224
+
225
+ test('getErrorStats computes recoverySuccessRate correctly', () => {
226
+ handler.errorStats.recoveryAttempts = 10;
227
+ handler.errorStats.successfulRecoveries = 7;
228
+ const stats = handler.getErrorStats();
229
+ expect(stats.recoverySuccessRate).toBeCloseTo(0.7);
230
+ });
231
+
232
+ // --- handleError full pipeline ---
233
+
234
+ test('handleError classifies, processes, updates stats and notifies subscribers', async () => {
235
+ const cb = jest.fn();
236
+ handler.subscribe(cb);
237
+
238
+ const error = new Error('Request timed out');
239
+ const result = await handler.handleError(error, { retryCount: 0 });
240
+
241
+ expect(result).toBeDefined();
242
+ expect(result.errorType).toBe(ERROR_TYPES.OPERATION_TIMEOUT);
243
+ expect(result.severity).toBe('high');
244
+ expect(result.errorId).toMatch(/^err_/);
245
+ expect(cb).toHaveBeenCalledTimes(1);
246
+
247
+ const stats = handler.getErrorStats();
248
+ expect(stats.totalErrors).toBe(1);
249
+ });
250
+
251
+ test('handleError returns fallback result when internal processing throws', async () => {
252
+ handler.processError = jest.fn().mockRejectedValue(new Error('processing failed'));
253
+
254
+ const result = await handler.handleError(new Error('test'), {});
255
+ expect(result.success).toBe(false);
256
+ expect(result.severity).toBe('critical');
257
+ });
258
+
259
+ // --- shouldRetry ---
260
+
261
+ test('shouldRetry returns false when recovery succeeded', () => {
262
+ const info = { recoverable: true, retryCount: 0, maxRetries: 3 };
263
+ expect(handler.shouldRetry(info, { success: true })).toBe(false);
264
+ });
265
+
266
+ test('shouldRetry returns true when recoverable and recovery failed', () => {
267
+ const info = { recoverable: true, retryCount: 0, maxRetries: 3 };
268
+ expect(handler.shouldRetry(info, { success: false })).toBe(true);
269
+ });
270
+
271
+ test('shouldRetry returns false when not recoverable', () => {
272
+ const info = { recoverable: false, retryCount: 0, maxRetries: 3 };
273
+ expect(handler.shouldRetry(info, null)).toBe(false);
274
+ });
275
+
276
+ // --- getMaxRetries ---
277
+
278
+ test('getMaxRetries returns 5 for RATE_LIMIT_EXCEEDED', () => {
279
+ const error = new Error('rate');
280
+ error.status = HTTP_STATUS.TOO_MANY_REQUESTS;
281
+ expect(handler.getMaxRetries(error, {})).toBe(5);
282
+ });
283
+
284
+ test('getMaxRetries returns 3 for OPERATION_TIMEOUT', () => {
285
+ const error = new Error('request timed out');
286
+ expect(handler.getMaxRetries(error, {})).toBe(3);
287
+ });
288
+
289
+ // --- handleCriticalError / processError ---
290
+
291
+ test('processError handles critical error by calling handleCriticalError', async () => {
292
+ const error = new Error('auth fail');
293
+ error.status = HTTP_STATUS.UNAUTHORIZED;
294
+ const errorInfo = handler.classifyError(error);
295
+ // severity should be critical
296
+ expect(errorInfo.severity).toBe('critical');
297
+
298
+ const spy = jest.spyOn(handler, 'handleCriticalError');
299
+ await handler.processError(errorInfo);
300
+ expect(spy).toHaveBeenCalledWith(errorInfo);
301
+ });
302
+
303
+ test('processError returns failure result when internal error occurs', async () => {
304
+ const errorInfo = {
305
+ id: 'err_test', type: ERROR_TYPES.FILE_NOT_FOUND, severity: 'low',
306
+ message: 'test', recoverable: true, retryCount: 0, maxRetries: 3
307
+ };
308
+ // Force logError to throw
309
+ handler.logError = jest.fn().mockRejectedValue(new Error('log broken'));
310
+ const result = await handler.processError(errorInfo);
311
+ expect(result.success).toBe(false);
312
+ expect(result.severity).toBe('critical');
313
+ });
314
+ });
@@ -0,0 +1,278 @@
1
+ import { jest, describe, test, expect, beforeEach } from '@jest/globals';
2
+ import { createMockLogger } from '../../__test-utils__/mockFactories.js';
3
+
4
+ // Mock dependencies
5
+ const mockFileProcessor = {
6
+ createDirectory: jest.fn().mockResolvedValue(undefined),
7
+ fileExists: jest.fn().mockResolvedValue(true),
8
+ readFile: jest.fn().mockResolvedValue('{}'),
9
+ writeFile: jest.fn().mockResolvedValue(undefined),
10
+ getFileStats: jest.fn().mockResolvedValue({ size: 1024, modified: new Date() }),
11
+ calculateHash: jest.fn().mockResolvedValue('abc123hash'),
12
+ processFile: jest.fn().mockResolvedValue({ content: 'processed content' }),
13
+ estimateTokens: jest.fn().mockReturnValue(100),
14
+ deleteDirectory: jest.fn().mockResolvedValue(undefined)
15
+ };
16
+
17
+ const mockValidator = {
18
+ validate: jest.fn().mockReturnValue({ valid: true, errors: [], warnings: [], sizeLevel: 'small' }),
19
+ getContentType: jest.fn().mockReturnValue('text'),
20
+ getMimeType: jest.fn().mockReturnValue('text/plain')
21
+ };
22
+
23
+ jest.unstable_mockModule('../../utilities/fileProcessor.js', () => ({
24
+ default: jest.fn().mockImplementation(() => mockFileProcessor)
25
+ }));
26
+
27
+ jest.unstable_mockModule('../../utilities/attachmentValidator.js', () => ({
28
+ default: jest.fn().mockImplementation(() => mockValidator)
29
+ }));
30
+
31
+ jest.unstable_mockModule('../../utilities/userDataDir.js', () => ({
32
+ getUserDataPaths: jest.fn(() => ({
33
+ settings: '/fake/settings',
34
+ attachments: '/fake/attachments',
35
+ skills: '/fake/skills'
36
+ })),
37
+ ensureUserDataDirs: jest.fn(async () => {})
38
+ }));
39
+
40
+ jest.unstable_mockModule('crypto', () => ({
41
+ randomUUID: jest.fn(() => 'test-uuid-1234')
42
+ }));
43
+
44
+ const { default: FileAttachmentService } = await import('../fileAttachmentService.js');
45
+
46
+ describe('FileAttachmentService', () => {
47
+ let service;
48
+ let logger;
49
+
50
+ beforeEach(() => {
51
+ jest.clearAllMocks();
52
+ logger = createMockLogger();
53
+ service = new FileAttachmentService({}, logger);
54
+ // Reset index to avoid stale state
55
+ service.index = null;
56
+ });
57
+
58
+ test('constructor initializes with null index', () => {
59
+ expect(service.index).toBeNull();
60
+ });
61
+
62
+ test('initialize creates directory and loads index', async () => {
63
+ mockFileProcessor.fileExists.mockResolvedValueOnce(false);
64
+ mockFileProcessor.writeFile.mockResolvedValue(undefined);
65
+
66
+ await service.initialize();
67
+ expect(mockFileProcessor.createDirectory).toHaveBeenCalled();
68
+ expect(logger.info).toHaveBeenCalled();
69
+ });
70
+
71
+ test('initialize throws on error', async () => {
72
+ mockFileProcessor.createDirectory.mockRejectedValueOnce(new Error('no dir'));
73
+ await expect(service.initialize()).rejects.toThrow('no dir');
74
+ });
75
+
76
+ test('loadIndex creates new index when file does not exist', async () => {
77
+ mockFileProcessor.fileExists.mockResolvedValueOnce(false);
78
+ mockFileProcessor.writeFile.mockResolvedValue(undefined);
79
+
80
+ const index = await service.loadIndex();
81
+ expect(index).toEqual({ attachments: {}, agentRefs: {} });
82
+ });
83
+
84
+ test('loadIndex loads existing index', async () => {
85
+ const existingIndex = { attachments: { f1: {} }, agentRefs: { a1: ['f1'] } };
86
+ mockFileProcessor.fileExists.mockResolvedValueOnce(true);
87
+ mockFileProcessor.readFile.mockResolvedValueOnce(JSON.stringify(existingIndex));
88
+
89
+ const index = await service.loadIndex();
90
+ expect(index.attachments.f1).toBeDefined();
91
+ });
92
+
93
+ test('loadIndex handles read error with default', async () => {
94
+ mockFileProcessor.fileExists.mockRejectedValueOnce(new Error('read fail'));
95
+ const index = await service.loadIndex();
96
+ expect(index).toEqual({ attachments: {}, agentRefs: {} });
97
+ });
98
+
99
+ test('saveIndex writes JSON to file', async () => {
100
+ service.index = { attachments: {}, agentRefs: {} };
101
+ await service.saveIndex();
102
+ expect(mockFileProcessor.writeFile).toHaveBeenCalled();
103
+ });
104
+
105
+ test('uploadFile creates attachment metadata', async () => {
106
+ service.index = { attachments: {}, agentRefs: {} };
107
+ mockFileProcessor.fileExists.mockResolvedValue(true);
108
+ mockFileProcessor.getFileStats.mockResolvedValue({ size: 512, modified: new Date() });
109
+ mockFileProcessor.processFile.mockResolvedValue({ content: 'text content' });
110
+ mockFileProcessor.estimateTokens.mockReturnValue(50);
111
+ mockFileProcessor.writeFile.mockResolvedValue(undefined);
112
+ mockFileProcessor.createDirectory.mockResolvedValue(undefined);
113
+ mockValidator.validate.mockReturnValue({ valid: true, errors: [], warnings: [], sizeLevel: 'small' });
114
+
115
+ const result = await service.uploadFile({
116
+ agentId: 'agent-1',
117
+ filePath: '/path/to/file.txt',
118
+ mode: 'content'
119
+ });
120
+
121
+ expect(result.fileId).toBe('test-uuid-1234');
122
+ expect(result.agentId).toBe('agent-1');
123
+ expect(result.mode).toBe('content');
124
+ expect(service.index.attachments['test-uuid-1234']).toBeDefined();
125
+ });
126
+
127
+ test('uploadFile throws for non-existent file', async () => {
128
+ service.index = { attachments: {}, agentRefs: {} };
129
+ mockFileProcessor.fileExists.mockResolvedValueOnce(false);
130
+
131
+ await expect(service.uploadFile({
132
+ agentId: 'agent-1',
133
+ filePath: '/missing.txt'
134
+ })).rejects.toThrow('File not found');
135
+ });
136
+
137
+ test('uploadFile throws on validation failure', async () => {
138
+ service.index = { attachments: {}, agentRefs: {} };
139
+ mockFileProcessor.fileExists.mockResolvedValueOnce(true);
140
+ mockFileProcessor.getFileStats.mockResolvedValue({ size: 999999999, modified: new Date() });
141
+ mockValidator.validate.mockReturnValue({ valid: false, errors: ['Too large'], warnings: [] });
142
+
143
+ await expect(service.uploadFile({
144
+ agentId: 'agent-1',
145
+ filePath: '/big.txt'
146
+ })).rejects.toThrow('Validation failed');
147
+ });
148
+
149
+ test('uploadFile handles reference mode', async () => {
150
+ service.index = { attachments: {}, agentRefs: {} };
151
+ mockFileProcessor.fileExists.mockResolvedValue(true);
152
+ mockFileProcessor.getFileStats.mockResolvedValue({ size: 512, modified: new Date() });
153
+ mockValidator.validate.mockReturnValue({ valid: true, errors: [], warnings: [], sizeLevel: 'small' });
154
+
155
+ const result = await service.uploadFile({
156
+ agentId: 'agent-1',
157
+ filePath: '/path/ref.js',
158
+ mode: 'reference'
159
+ });
160
+
161
+ expect(result.mode).toBe('reference');
162
+ expect(result.tokenEstimate).toBe(0);
163
+ });
164
+
165
+ test('getAttachments returns filtered attachments', async () => {
166
+ service.index = {
167
+ attachments: { f1: { fileId: 'f1', agentId: 'a1', fileName: 'a.txt', mode: 'content', active: true } },
168
+ agentRefs: { a1: ['f1'] }
169
+ };
170
+ mockFileProcessor.fileExists.mockResolvedValue(true);
171
+ mockFileProcessor.readFile.mockResolvedValue(JSON.stringify({
172
+ fileId: 'f1', agentId: 'a1', active: true, mode: 'content'
173
+ }));
174
+
175
+ const results = await service.getAttachments('a1');
176
+ expect(results).toHaveLength(1);
177
+ });
178
+
179
+ test('getAttachments loads index if null', async () => {
180
+ service.index = null;
181
+ mockFileProcessor.fileExists.mockResolvedValueOnce(false);
182
+ mockFileProcessor.writeFile.mockResolvedValue(undefined);
183
+
184
+ const results = await service.getAttachments('a1');
185
+ expect(results).toEqual([]);
186
+ });
187
+
188
+ test('getAttachment returns null for unknown fileId', async () => {
189
+ service.index = { attachments: {}, agentRefs: {} };
190
+ const result = await service.getAttachment('unknown');
191
+ expect(result).toBeNull();
192
+ });
193
+
194
+ test('getAttachment returns null when metadata file missing', async () => {
195
+ service.index = { attachments: { f1: { fileId: 'f1', agentId: 'a1' } }, agentRefs: {} };
196
+ mockFileProcessor.fileExists.mockResolvedValueOnce(false);
197
+ const result = await service.getAttachment('f1');
198
+ expect(result).toBeNull();
199
+ });
200
+
201
+ test('getAttachmentContent returns null for reference mode', async () => {
202
+ service.index = { attachments: { f1: { fileId: 'f1', agentId: 'a1' } }, agentRefs: {} };
203
+ mockFileProcessor.fileExists.mockResolvedValueOnce(true);
204
+ mockFileProcessor.readFile.mockResolvedValueOnce(JSON.stringify({ mode: 'reference', agentId: 'a1' }));
205
+
206
+ const content = await service.getAttachmentContent('f1');
207
+ expect(content).toBeNull();
208
+ });
209
+
210
+ test('toggleActive flips active state', async () => {
211
+ service.index = { attachments: { f1: { fileId: 'f1', agentId: 'a1', active: true } }, agentRefs: {} };
212
+ mockFileProcessor.fileExists.mockResolvedValue(true);
213
+ mockFileProcessor.readFile.mockResolvedValueOnce(JSON.stringify({
214
+ fileId: 'f1', agentId: 'a1', active: true
215
+ }));
216
+
217
+ const result = await service.toggleActive('f1');
218
+ expect(result.active).toBe(false);
219
+ });
220
+
221
+ test('toggleActive throws for missing attachment', async () => {
222
+ service.index = { attachments: {}, agentRefs: {} };
223
+ await expect(service.toggleActive('unknown')).rejects.toThrow('Attachment not found');
224
+ });
225
+
226
+ test('deleteAttachment removes when no other references', async () => {
227
+ service.index = {
228
+ attachments: { f1: { fileId: 'f1', agentId: 'a1' } },
229
+ agentRefs: { a1: ['f1'] }
230
+ };
231
+ mockFileProcessor.fileExists.mockResolvedValue(true);
232
+ mockFileProcessor.readFile.mockResolvedValueOnce(JSON.stringify({
233
+ fileId: 'f1', agentId: 'a1', referencedBy: ['a1']
234
+ }));
235
+
236
+ const deleted = await service.deleteAttachment('f1', 'a1');
237
+ expect(deleted).toBe(true);
238
+ expect(mockFileProcessor.deleteDirectory).toHaveBeenCalled();
239
+ });
240
+
241
+ test('deleteAttachment derefs when still referenced', async () => {
242
+ service.index = {
243
+ attachments: { f1: { fileId: 'f1', agentId: 'a1' } },
244
+ agentRefs: { a1: ['f1'] }
245
+ };
246
+ mockFileProcessor.fileExists.mockResolvedValue(true);
247
+ mockFileProcessor.readFile.mockResolvedValueOnce(JSON.stringify({
248
+ fileId: 'f1', agentId: 'a1', referencedBy: ['a1', 'a2']
249
+ }));
250
+
251
+ const deleted = await service.deleteAttachment('f1', 'a1');
252
+ expect(deleted).toBe(false);
253
+ });
254
+
255
+ test('getAttachmentPreview returns truncated content', async () => {
256
+ service.index = { attachments: { f1: { fileId: 'f1', agentId: 'a1' } }, agentRefs: {} };
257
+ mockFileProcessor.fileExists.mockResolvedValue(true);
258
+ const metadata = { fileId: 'f1', agentId: 'a1', mode: 'content', contentFileName: 'content.txt' };
259
+ mockFileProcessor.readFile
260
+ .mockResolvedValueOnce(JSON.stringify(metadata)) // getAttachment
261
+ .mockResolvedValueOnce('x'.repeat(2000)); // getAttachmentContent
262
+
263
+ const preview = await service.getAttachmentPreview('f1');
264
+ expect(preview.length).toBe(1003); // 1000 + '...'
265
+ });
266
+
267
+ test('getAttachmentPreview returns image placeholder', async () => {
268
+ service.index = { attachments: { f1: { fileId: 'f1', agentId: 'a1' } }, agentRefs: {} };
269
+ mockFileProcessor.fileExists.mockResolvedValue(true);
270
+ const metadata = { fileId: 'f1', agentId: 'a1', mode: 'content', contentFileName: 'content.base64' };
271
+ mockFileProcessor.readFile
272
+ .mockResolvedValueOnce(JSON.stringify(metadata))
273
+ .mockResolvedValueOnce('data:image/png;base64,abc');
274
+
275
+ const preview = await service.getAttachmentPreview('f1');
276
+ expect(preview).toBe('[Image content - base64 encoded]');
277
+ });
278
+ });