@gguf/coder 0.3.0 → 0.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (414) hide show
  1. package/.editorconfig +16 -0
  2. package/.env.example +63 -0
  3. package/.gitattributes +1 -0
  4. package/.semgrepignore +19 -0
  5. package/coder-dummy-file.ts +52 -0
  6. package/coder.config.example.json +59 -0
  7. package/coder.config.json +13 -0
  8. package/color_picker.html +36 -0
  9. package/package.json +2 -14
  10. package/scripts/extract-changelog.js +73 -0
  11. package/scripts/fetch-models.js +143 -0
  12. package/scripts/test.sh +40 -0
  13. package/scripts/update-homebrew-formula.sh +125 -0
  14. package/scripts/update-nix-version.sh +157 -0
  15. package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
  16. package/source/ai-sdk-client/AISDKClient.ts +155 -0
  17. package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
  18. package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
  19. package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
  20. package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
  21. package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
  22. package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
  23. package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
  24. package/source/ai-sdk-client/converters/message-converter.ts +113 -0
  25. package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
  26. package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
  27. package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
  28. package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
  29. package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
  30. package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
  31. package/source/ai-sdk-client/index.ts +7 -0
  32. package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
  33. package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
  34. package/source/ai-sdk-client/types.ts +9 -0
  35. package/source/ai-sdk-client-empty-message.spec.ts +141 -0
  36. package/source/ai-sdk-client-error-handling.spec.ts +186 -0
  37. package/source/ai-sdk-client-maxretries.spec.ts +114 -0
  38. package/source/ai-sdk-client-preparestep.spec.ts +279 -0
  39. package/source/app/App.spec.tsx +32 -0
  40. package/source/app/App.tsx +480 -0
  41. package/source/app/components/AppContainer.spec.tsx +96 -0
  42. package/source/app/components/AppContainer.tsx +56 -0
  43. package/source/app/components/ChatInterface.spec.tsx +163 -0
  44. package/source/app/components/ChatInterface.tsx +144 -0
  45. package/source/app/components/ModalSelectors.spec.tsx +141 -0
  46. package/source/app/components/ModalSelectors.tsx +135 -0
  47. package/source/app/helpers.spec.ts +97 -0
  48. package/source/app/helpers.ts +63 -0
  49. package/source/app/index.ts +4 -0
  50. package/source/app/types.ts +39 -0
  51. package/source/app/utils/appUtils.ts +294 -0
  52. package/source/app/utils/conversationState.ts +310 -0
  53. package/source/app.spec.tsx +244 -0
  54. package/source/cli.spec.ts +73 -0
  55. package/source/cli.tsx +51 -0
  56. package/source/client-factory.spec.ts +48 -0
  57. package/source/client-factory.ts +178 -0
  58. package/source/command-parser.spec.ts +127 -0
  59. package/source/command-parser.ts +36 -0
  60. package/source/commands/checkpoint.spec.tsx +277 -0
  61. package/source/commands/checkpoint.tsx +366 -0
  62. package/source/commands/clear.tsx +22 -0
  63. package/source/commands/custom-commands.tsx +121 -0
  64. package/source/commands/exit.ts +21 -0
  65. package/source/commands/export.spec.tsx +131 -0
  66. package/source/commands/export.tsx +79 -0
  67. package/source/commands/help.tsx +120 -0
  68. package/source/commands/index.ts +17 -0
  69. package/source/commands/init.tsx +339 -0
  70. package/source/commands/lsp-command.spec.tsx +281 -0
  71. package/source/commands/lsp.tsx +120 -0
  72. package/source/commands/mcp-command.spec.tsx +313 -0
  73. package/source/commands/mcp.tsx +162 -0
  74. package/source/commands/model-database.spec.tsx +758 -0
  75. package/source/commands/model-database.tsx +418 -0
  76. package/source/commands/model.ts +12 -0
  77. package/source/commands/provider.ts +12 -0
  78. package/source/commands/setup-config.tsx +16 -0
  79. package/source/commands/simple-commands.spec.tsx +175 -0
  80. package/source/commands/status.ts +12 -0
  81. package/source/commands/theme.ts +12 -0
  82. package/source/commands/update.spec.tsx +261 -0
  83. package/source/commands/update.tsx +201 -0
  84. package/source/commands/usage.spec.tsx +495 -0
  85. package/source/commands/usage.tsx +100 -0
  86. package/source/commands.spec.ts +436 -0
  87. package/source/commands.ts +83 -0
  88. package/source/components/assistant-message.spec.tsx +796 -0
  89. package/source/components/assistant-message.tsx +34 -0
  90. package/source/components/bash-execution-indicator.tsx +21 -0
  91. package/source/components/cancelling-indicator.tsx +16 -0
  92. package/source/components/chat-queue.spec.tsx +83 -0
  93. package/source/components/chat-queue.tsx +36 -0
  94. package/source/components/checkpoint-display.spec.tsx +219 -0
  95. package/source/components/checkpoint-display.tsx +126 -0
  96. package/source/components/checkpoint-selector.spec.tsx +173 -0
  97. package/source/components/checkpoint-selector.tsx +173 -0
  98. package/source/components/development-mode-indicator.spec.tsx +268 -0
  99. package/source/components/development-mode-indicator.tsx +38 -0
  100. package/source/components/message-box.spec.tsx +427 -0
  101. package/source/components/message-box.tsx +87 -0
  102. package/source/components/model-selector.tsx +132 -0
  103. package/source/components/provider-selector.tsx +75 -0
  104. package/source/components/random-spinner.tsx +19 -0
  105. package/source/components/security-disclaimer.tsx +73 -0
  106. package/source/components/status-connection-display.spec.tsx +133 -0
  107. package/source/components/status.tsx +267 -0
  108. package/source/components/theme-selector.tsx +126 -0
  109. package/source/components/tool-confirmation.tsx +190 -0
  110. package/source/components/tool-execution-indicator.tsx +33 -0
  111. package/source/components/tool-message.tsx +85 -0
  112. package/source/components/ui/titled-box.spec.tsx +207 -0
  113. package/source/components/ui/titled-box.tsx +57 -0
  114. package/source/components/usage/progress-bar.spec.tsx +398 -0
  115. package/source/components/usage/progress-bar.tsx +30 -0
  116. package/source/components/usage/usage-display.spec.tsx +780 -0
  117. package/source/components/usage/usage-display.tsx +291 -0
  118. package/source/components/user-input.spec.tsx +327 -0
  119. package/source/components/user-input.tsx +533 -0
  120. package/source/components/user-message.spec.tsx +230 -0
  121. package/source/components/user-message.tsx +84 -0
  122. package/source/components/welcome-message.tsx +76 -0
  123. package/source/config/env-substitution.ts +65 -0
  124. package/source/config/index.spec.ts +171 -0
  125. package/source/config/index.ts +154 -0
  126. package/source/config/paths.spec.ts +241 -0
  127. package/source/config/paths.ts +55 -0
  128. package/source/config/preferences.ts +51 -0
  129. package/source/config/themes.ts +315 -0
  130. package/source/constants.ts +130 -0
  131. package/source/context/mode-context.spec.ts +79 -0
  132. package/source/context/mode-context.ts +24 -0
  133. package/source/custom-commands/executor.spec.ts +142 -0
  134. package/source/custom-commands/executor.ts +64 -0
  135. package/source/custom-commands/loader.spec.ts +314 -0
  136. package/source/custom-commands/loader.ts +153 -0
  137. package/source/custom-commands/parser.ts +196 -0
  138. package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
  139. package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
  140. package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
  141. package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
  142. package/source/hooks/chat-handler/index.ts +12 -0
  143. package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
  144. package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
  145. package/source/hooks/chat-handler/types.ts +38 -0
  146. package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
  147. package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
  148. package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
  149. package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
  150. package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
  151. package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
  152. package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
  153. package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
  154. package/source/hooks/useAppHandlers.tsx +291 -0
  155. package/source/hooks/useAppInitialization.tsx +422 -0
  156. package/source/hooks/useAppState.tsx +311 -0
  157. package/source/hooks/useDirectoryTrust.tsx +98 -0
  158. package/source/hooks/useInputState.ts +414 -0
  159. package/source/hooks/useModeHandlers.tsx +302 -0
  160. package/source/hooks/useNonInteractiveMode.ts +140 -0
  161. package/source/hooks/useTerminalWidth.tsx +81 -0
  162. package/source/hooks/useTheme.ts +18 -0
  163. package/source/hooks/useToolHandler.tsx +349 -0
  164. package/source/hooks/useUIState.ts +61 -0
  165. package/source/init/agents-template-generator.ts +421 -0
  166. package/source/init/existing-rules-extractor.ts +319 -0
  167. package/source/init/file-scanner.spec.ts +227 -0
  168. package/source/init/file-scanner.ts +238 -0
  169. package/source/init/framework-detector.ts +382 -0
  170. package/source/init/language-detector.ts +269 -0
  171. package/source/init/project-analyzer.spec.ts +231 -0
  172. package/source/init/project-analyzer.ts +458 -0
  173. package/source/lsp/index.ts +31 -0
  174. package/source/lsp/lsp-client.spec.ts +508 -0
  175. package/source/lsp/lsp-client.ts +487 -0
  176. package/source/lsp/lsp-manager.spec.ts +477 -0
  177. package/source/lsp/lsp-manager.ts +419 -0
  178. package/source/lsp/protocol.spec.ts +502 -0
  179. package/source/lsp/protocol.ts +360 -0
  180. package/source/lsp/server-discovery.spec.ts +654 -0
  181. package/source/lsp/server-discovery.ts +515 -0
  182. package/source/markdown-parser/html-entities.spec.ts +88 -0
  183. package/source/markdown-parser/html-entities.ts +45 -0
  184. package/source/markdown-parser/index.spec.ts +281 -0
  185. package/source/markdown-parser/index.ts +126 -0
  186. package/source/markdown-parser/table-parser.spec.ts +133 -0
  187. package/source/markdown-parser/table-parser.ts +114 -0
  188. package/source/markdown-parser/utils.spec.ts +70 -0
  189. package/source/markdown-parser/utils.ts +13 -0
  190. package/source/mcp/mcp-client.spec.ts +81 -0
  191. package/source/mcp/mcp-client.ts +625 -0
  192. package/source/mcp/transport-factory.spec.ts +406 -0
  193. package/source/mcp/transport-factory.ts +312 -0
  194. package/source/message-handler.ts +67 -0
  195. package/source/model-database/database-engine.spec.ts +494 -0
  196. package/source/model-database/database-engine.ts +50 -0
  197. package/source/model-database/model-database.spec.ts +363 -0
  198. package/source/model-database/model-database.ts +91 -0
  199. package/source/model-database/model-engine.spec.ts +447 -0
  200. package/source/model-database/model-engine.ts +65 -0
  201. package/source/model-database/model-fetcher.spec.ts +583 -0
  202. package/source/model-database/model-fetcher.ts +330 -0
  203. package/source/models/index.ts +1 -0
  204. package/source/models/models-cache.spec.ts +214 -0
  205. package/source/models/models-cache.ts +78 -0
  206. package/source/models/models-dev-client.spec.ts +379 -0
  207. package/source/models/models-dev-client.ts +329 -0
  208. package/source/models/models-types.ts +68 -0
  209. package/source/prompt-history.ts +155 -0
  210. package/source/security/command-injection.spec.ts +240 -0
  211. package/source/services/checkpoint-manager.spec.ts +523 -0
  212. package/source/services/checkpoint-manager.ts +466 -0
  213. package/source/services/file-snapshot.spec.ts +569 -0
  214. package/source/services/file-snapshot.ts +220 -0
  215. package/source/test-utils/render-with-theme.tsx +48 -0
  216. package/source/tokenization/index.ts +1 -0
  217. package/source/tokenization/tokenizer-factory.spec.ts +170 -0
  218. package/source/tokenization/tokenizer-factory.ts +125 -0
  219. package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
  220. package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
  221. package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
  222. package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
  223. package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
  224. package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
  225. package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
  226. package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
  227. package/source/tool-calling/index.ts +5 -0
  228. package/source/tool-calling/json-parser.spec.ts +639 -0
  229. package/source/tool-calling/json-parser.ts +247 -0
  230. package/source/tool-calling/tool-parser.spec.ts +395 -0
  231. package/source/tool-calling/tool-parser.ts +120 -0
  232. package/source/tool-calling/xml-parser.spec.ts +662 -0
  233. package/source/tool-calling/xml-parser.ts +289 -0
  234. package/source/tools/execute-bash.spec.tsx +353 -0
  235. package/source/tools/execute-bash.tsx +219 -0
  236. package/source/tools/execute-function.spec.ts +130 -0
  237. package/source/tools/fetch-url.spec.tsx +342 -0
  238. package/source/tools/fetch-url.tsx +172 -0
  239. package/source/tools/find-files.spec.tsx +924 -0
  240. package/source/tools/find-files.tsx +293 -0
  241. package/source/tools/index.ts +102 -0
  242. package/source/tools/lsp-get-diagnostics.tsx +192 -0
  243. package/source/tools/needs-approval.spec.ts +282 -0
  244. package/source/tools/read-file.spec.tsx +801 -0
  245. package/source/tools/read-file.tsx +387 -0
  246. package/source/tools/search-file-contents.spec.tsx +1273 -0
  247. package/source/tools/search-file-contents.tsx +293 -0
  248. package/source/tools/string-replace.spec.tsx +730 -0
  249. package/source/tools/string-replace.tsx +548 -0
  250. package/source/tools/tool-manager.ts +210 -0
  251. package/source/tools/tool-registry.spec.ts +415 -0
  252. package/source/tools/tool-registry.ts +228 -0
  253. package/source/tools/web-search.tsx +223 -0
  254. package/source/tools/write-file.spec.tsx +559 -0
  255. package/source/tools/write-file.tsx +228 -0
  256. package/source/types/app.ts +37 -0
  257. package/source/types/checkpoint.ts +48 -0
  258. package/source/types/commands.ts +46 -0
  259. package/source/types/components.ts +27 -0
  260. package/source/types/config.ts +103 -0
  261. package/source/types/core-connection-status.spec.ts +67 -0
  262. package/source/types/core.ts +181 -0
  263. package/source/types/hooks.ts +50 -0
  264. package/source/types/index.ts +12 -0
  265. package/source/types/markdown-parser.ts +11 -0
  266. package/source/types/mcp.ts +52 -0
  267. package/source/types/system.ts +16 -0
  268. package/source/types/tokenization.ts +41 -0
  269. package/source/types/ui.ts +40 -0
  270. package/source/types/usage.ts +58 -0
  271. package/source/types/utils.ts +16 -0
  272. package/source/usage/calculator.spec.ts +385 -0
  273. package/source/usage/calculator.ts +104 -0
  274. package/source/usage/storage.spec.ts +703 -0
  275. package/source/usage/storage.ts +238 -0
  276. package/source/usage/tracker.spec.ts +456 -0
  277. package/source/usage/tracker.ts +102 -0
  278. package/source/utils/atomic-deletion.spec.ts +194 -0
  279. package/source/utils/atomic-deletion.ts +127 -0
  280. package/source/utils/bounded-map.spec.ts +300 -0
  281. package/source/utils/bounded-map.ts +193 -0
  282. package/source/utils/checkpoint-utils.spec.ts +222 -0
  283. package/source/utils/checkpoint-utils.ts +92 -0
  284. package/source/utils/error-formatter.spec.ts +169 -0
  285. package/source/utils/error-formatter.ts +194 -0
  286. package/source/utils/file-autocomplete.spec.ts +173 -0
  287. package/source/utils/file-autocomplete.ts +196 -0
  288. package/source/utils/file-cache.spec.ts +309 -0
  289. package/source/utils/file-cache.ts +195 -0
  290. package/source/utils/file-content-loader.spec.ts +180 -0
  291. package/source/utils/file-content-loader.ts +179 -0
  292. package/source/utils/file-mention-handler.spec.ts +261 -0
  293. package/source/utils/file-mention-handler.ts +84 -0
  294. package/source/utils/file-mention-parser.spec.ts +182 -0
  295. package/source/utils/file-mention-parser.ts +170 -0
  296. package/source/utils/fuzzy-matching.spec.ts +149 -0
  297. package/source/utils/fuzzy-matching.ts +146 -0
  298. package/source/utils/indentation-normalizer.spec.ts +216 -0
  299. package/source/utils/indentation-normalizer.ts +76 -0
  300. package/source/utils/installation-detector.spec.ts +178 -0
  301. package/source/utils/installation-detector.ts +153 -0
  302. package/source/utils/logging/config.spec.ts +311 -0
  303. package/source/utils/logging/config.ts +210 -0
  304. package/source/utils/logging/console-facade.spec.ts +184 -0
  305. package/source/utils/logging/console-facade.ts +384 -0
  306. package/source/utils/logging/correlation.spec.ts +679 -0
  307. package/source/utils/logging/correlation.ts +474 -0
  308. package/source/utils/logging/formatters.spec.ts +464 -0
  309. package/source/utils/logging/formatters.ts +207 -0
  310. package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
  311. package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
  312. package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
  313. package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
  314. package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
  315. package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
  316. package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
  317. package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
  318. package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
  319. package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
  320. package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
  321. package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
  322. package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
  323. package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
  324. package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
  325. package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
  326. package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
  327. package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
  328. package/source/utils/logging/health-monitor/index.ts +15 -0
  329. package/source/utils/logging/health-monitor/instances.ts +48 -0
  330. package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
  331. package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
  332. package/source/utils/logging/health-monitor/types.ts +126 -0
  333. package/source/utils/logging/index.spec.ts +284 -0
  334. package/source/utils/logging/index.ts +236 -0
  335. package/source/utils/logging/integration.spec.ts +441 -0
  336. package/source/utils/logging/log-method-factory.spec.ts +573 -0
  337. package/source/utils/logging/log-method-factory.ts +233 -0
  338. package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
  339. package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
  340. package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
  341. package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
  342. package/source/utils/logging/log-query/index.ts +23 -0
  343. package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
  344. package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
  345. package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
  346. package/source/utils/logging/log-query/query/query-builder.ts +151 -0
  347. package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
  348. package/source/utils/logging/log-query/query/query-engine.ts +45 -0
  349. package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
  350. package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
  351. package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
  352. package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
  353. package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
  354. package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
  355. package/source/utils/logging/log-query/types.ts +163 -0
  356. package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
  357. package/source/utils/logging/log-query/utils/helpers.ts +72 -0
  358. package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
  359. package/source/utils/logging/log-query/utils/sorting.ts +61 -0
  360. package/source/utils/logging/logger-provider.spec.ts +262 -0
  361. package/source/utils/logging/logger-provider.ts +362 -0
  362. package/source/utils/logging/performance.spec.ts +209 -0
  363. package/source/utils/logging/performance.ts +757 -0
  364. package/source/utils/logging/pino-logger.spec.ts +425 -0
  365. package/source/utils/logging/pino-logger.ts +514 -0
  366. package/source/utils/logging/redaction.spec.ts +490 -0
  367. package/source/utils/logging/redaction.ts +267 -0
  368. package/source/utils/logging/request-tracker.spec.ts +1198 -0
  369. package/source/utils/logging/request-tracker.ts +803 -0
  370. package/source/utils/logging/transports.spec.ts +505 -0
  371. package/source/utils/logging/transports.ts +305 -0
  372. package/source/utils/logging/types.ts +216 -0
  373. package/source/utils/message-builder.spec.ts +179 -0
  374. package/source/utils/message-builder.ts +101 -0
  375. package/source/utils/message-queue.tsx +486 -0
  376. package/source/utils/paste-detection.spec.ts +69 -0
  377. package/source/utils/paste-detection.ts +124 -0
  378. package/source/utils/paste-roundtrip.spec.ts +442 -0
  379. package/source/utils/paste-utils.spec.ts +128 -0
  380. package/source/utils/paste-utils.ts +52 -0
  381. package/source/utils/programming-language-helper.spec.ts +74 -0
  382. package/source/utils/programming-language-helper.ts +32 -0
  383. package/source/utils/prompt-assembly.spec.ts +221 -0
  384. package/source/utils/prompt-processor.ts +173 -0
  385. package/source/utils/tool-args-parser.spec.ts +136 -0
  386. package/source/utils/tool-args-parser.ts +54 -0
  387. package/source/utils/tool-cancellation.spec.ts +230 -0
  388. package/source/utils/tool-cancellation.ts +28 -0
  389. package/source/utils/tool-result-display.spec.tsx +469 -0
  390. package/source/utils/tool-result-display.tsx +90 -0
  391. package/source/utils/update-checker.spec.ts +383 -0
  392. package/source/utils/update-checker.ts +183 -0
  393. package/source/wizard/config-wizard.spec.tsx +103 -0
  394. package/source/wizard/config-wizard.tsx +382 -0
  395. package/source/wizard/steps/location-step.spec.tsx +186 -0
  396. package/source/wizard/steps/location-step.tsx +147 -0
  397. package/source/wizard/steps/mcp-step.spec.tsx +607 -0
  398. package/source/wizard/steps/mcp-step.tsx +632 -0
  399. package/source/wizard/steps/provider-step.spec.tsx +342 -0
  400. package/source/wizard/steps/provider-step.tsx +957 -0
  401. package/source/wizard/steps/summary-step.spec.tsx +749 -0
  402. package/source/wizard/steps/summary-step.tsx +228 -0
  403. package/source/wizard/templates/mcp-templates.spec.ts +613 -0
  404. package/source/wizard/templates/mcp-templates.ts +570 -0
  405. package/source/wizard/templates/provider-templates.spec.ts +152 -0
  406. package/source/wizard/templates/provider-templates.ts +485 -0
  407. package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
  408. package/source/wizard/utils/fetch-cloud-models.ts +223 -0
  409. package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
  410. package/source/wizard/utils/fetch-local-models.ts +192 -0
  411. package/source/wizard/validation-array.spec.ts +264 -0
  412. package/source/wizard/validation.spec.ts +373 -0
  413. package/source/wizard/validation.ts +232 -0
  414. package/source/app/prompts/main-prompt.md +0 -122
@@ -0,0 +1,182 @@
1
+ import path from 'node:path';
2
+ import test from 'ava';
3
+ import {
4
+ isValidFilePath,
5
+ parseFileMentions,
6
+ parseLineRange,
7
+ resolveFilePath,
8
+ } from './file-mention-parser.js';
9
+
10
+ console.log(`\nfile-mention-parser.spec.ts`);
11
+
12
+ // Test parseFileMentions()
13
+ test('parses single file mention', t => {
14
+ const result = parseFileMentions('Check @app.tsx');
15
+ t.is(result.length, 1);
16
+ t.is(result[0].filePath, 'app.tsx');
17
+ t.is(result[0].rawText, '@app.tsx');
18
+ t.is(result[0].startIndex, 6);
19
+ t.is(result[0].endIndex, 14);
20
+ t.is(result[0].lineRange, undefined);
21
+ });
22
+
23
+ test('parses file with path', t => {
24
+ const result = parseFileMentions('Check @src/components/Button.tsx');
25
+ t.is(result.length, 1);
26
+ t.is(result[0].filePath, 'src/components/Button.tsx');
27
+ t.is(result[0].rawText, '@src/components/Button.tsx');
28
+ });
29
+
30
+ test('parses file with single line number', t => {
31
+ const result = parseFileMentions('@app.tsx:10');
32
+ t.is(result.length, 1);
33
+ t.is(result[0].filePath, 'app.tsx');
34
+ t.deepEqual(result[0].lineRange, {start: 10, end: undefined});
35
+ });
36
+
37
+ test('parses file with line range', t => {
38
+ const result = parseFileMentions('@app.tsx:10-20');
39
+ t.is(result.length, 1);
40
+ t.is(result[0].filePath, 'app.tsx');
41
+ t.deepEqual(result[0].lineRange, {start: 10, end: 20});
42
+ });
43
+
44
+ test('parses multiple file mentions', t => {
45
+ const result = parseFileMentions('Compare @a.ts and @b.ts');
46
+ t.is(result.length, 2);
47
+ t.is(result[0].filePath, 'a.ts');
48
+ t.is(result[1].filePath, 'b.ts');
49
+ });
50
+
51
+ test('parses file mentions with different line ranges', t => {
52
+ const result = parseFileMentions('Check @app.tsx:1-5 and @utils.ts:10-15');
53
+ t.is(result.length, 2);
54
+ t.deepEqual(result[0].lineRange, {start: 1, end: 5});
55
+ t.deepEqual(result[1].lineRange, {start: 10, end: 15});
56
+ });
57
+
58
+ test('handles text without file mentions', t => {
59
+ const result = parseFileMentions('This is just regular text');
60
+ t.is(result.length, 0);
61
+ });
62
+
63
+ test('ignores @ symbols in email addresses', t => {
64
+ const result = parseFileMentions('Email: user@example.com');
65
+ // Should not match email addresses (they have spaces or are at word boundaries)
66
+ // However, this might match "example.com" - let's verify behavior
67
+ t.true(result.length <= 1);
68
+ if (result.length === 1) {
69
+ t.is(result[0].filePath, 'example.com');
70
+ }
71
+ });
72
+
73
+ test('rejects invalid line ranges in file mentions', t => {
74
+ // Line range where end < start should not have lineRange
75
+ const result = parseFileMentions('@app.tsx:20-10');
76
+ t.is(result.length, 1);
77
+ t.is(result[0].lineRange, undefined);
78
+ });
79
+
80
+ test('rejects zero or negative line numbers', t => {
81
+ const result1 = parseFileMentions('@app.tsx:0');
82
+ t.is(result1[0].lineRange, undefined);
83
+
84
+ const result2 = parseFileMentions('@app.tsx:0-10');
85
+ t.is(result2[0].lineRange, undefined);
86
+ });
87
+
88
+ test('handles complex file paths', t => {
89
+ const result = parseFileMentions(
90
+ '@src/components/ui/Button/index.tsx:100-200',
91
+ );
92
+ t.is(result.length, 1);
93
+ t.is(result[0].filePath, 'src/components/ui/Button/index.tsx');
94
+ t.deepEqual(result[0].lineRange, {start: 100, end: 200});
95
+ });
96
+
97
+ // Test isValidFilePath()
98
+ test('accepts valid relative paths', t => {
99
+ t.true(isValidFilePath('app.tsx'));
100
+ t.true(isValidFilePath('src/app.tsx'));
101
+ t.true(isValidFilePath('src/components/Button.tsx'));
102
+ });
103
+
104
+ test('rejects directory traversal attempts', t => {
105
+ t.false(isValidFilePath('../../etc/passwd'));
106
+ t.false(isValidFilePath('../../../secret.txt'));
107
+ t.false(isValidFilePath('src/../../../etc/passwd'));
108
+ });
109
+
110
+ test('rejects absolute paths', t => {
111
+ t.false(isValidFilePath('/etc/passwd'));
112
+ t.false(isValidFilePath('/home/user/file.txt'));
113
+ t.false(isValidFilePath('C:\\Windows\\System32\\file.txt'));
114
+ });
115
+
116
+ test('rejects empty or whitespace paths', t => {
117
+ t.false(isValidFilePath(''));
118
+ t.false(isValidFilePath(' '));
119
+ });
120
+
121
+ test('rejects paths with null bytes', t => {
122
+ t.false(isValidFilePath('file.txt\0'));
123
+ t.false(isValidFilePath('file\0.txt'));
124
+ });
125
+
126
+ test('rejects paths starting with slashes', t => {
127
+ t.false(isValidFilePath('/file.txt'));
128
+ t.false(isValidFilePath('\\file.txt'));
129
+ });
130
+
131
+ // Test resolveFilePath()
132
+ test('resolves relative path to absolute', t => {
133
+ const cwd = '/home/user/project';
134
+ const resolved = resolveFilePath('src/app.tsx', cwd);
135
+ t.is(resolved, path.join(cwd, 'src/app.tsx'));
136
+ });
137
+
138
+ test('throws on invalid file paths', t => {
139
+ const cwd = '/home/user/project';
140
+ t.throws(() => resolveFilePath('../../etc/passwd', cwd), {
141
+ message: /Invalid file path/,
142
+ });
143
+ });
144
+
145
+ test('throws when resolved path escapes project directory', t => {
146
+ const cwd = '/home/user/project';
147
+ // Even if validation passes, resolve should catch escaping
148
+ t.throws(() => resolveFilePath('/etc/passwd', cwd), {
149
+ message: /Invalid file path/,
150
+ });
151
+ });
152
+
153
+ test('resolves nested paths correctly', t => {
154
+ const cwd = '/home/user/project';
155
+ const resolved = resolveFilePath('src/components/ui/Button.tsx', cwd);
156
+ t.is(resolved, path.join(cwd, 'src/components/ui/Button.tsx'));
157
+ });
158
+
159
+ // Test parseLineRange()
160
+ test('parses single line number', t => {
161
+ const result = parseLineRange('10');
162
+ t.deepEqual(result, {start: 10, end: undefined});
163
+ });
164
+
165
+ test('parses line range', t => {
166
+ const result = parseLineRange('10-20');
167
+ t.deepEqual(result, {start: 10, end: 20});
168
+ });
169
+
170
+ test('rejects invalid line ranges', t => {
171
+ t.is(parseLineRange('20-10'), null); // end < start
172
+ t.is(parseLineRange('0'), null); // zero
173
+ t.is(parseLineRange('-5'), null); // negative
174
+ t.is(parseLineRange('abc'), null); // non-numeric
175
+ t.is(parseLineRange(''), null); // empty
176
+ });
177
+
178
+ test('rejects malformed ranges', t => {
179
+ t.is(parseLineRange('10-20-30'), null); // too many parts
180
+ t.is(parseLineRange('10-'), null); // missing end
181
+ t.is(parseLineRange('-20'), null); // missing start
182
+ });
@@ -0,0 +1,170 @@
1
+ import path from 'node:path';
2
+
3
+ /**
4
+ * Represents a parsed file mention from user input
5
+ * Supports:
6
+ * - @filename.ts
7
+ * - @src/components/Button.tsx
8
+ * - @file.ts:10-20 (line ranges)
9
+ * - @file.ts:10 (single line)
10
+ */
11
+ interface FileMention {
12
+ rawText: string; // "@src/app.tsx:10-20"
13
+ filePath: string; // "src/app.tsx"
14
+ lineRange?: {
15
+ start: number;
16
+ end?: number; // undefined for single line
17
+ };
18
+ startIndex: number; // Position in input string
19
+ endIndex: number;
20
+ }
21
+
22
+ /**
23
+ * Regex pattern to match @file mentions
24
+ * Matches:
25
+ * - @file.ts
26
+ * - @src/path/to/file.tsx
27
+ * - @file.ts:10
28
+ * - @file.ts:10-20
29
+ */
30
+ const FILE_MENTION_REGEX = /@([^\s:]+)(?::(\d+)(?:-(\d+))?)?/g;
31
+
32
+ /**
33
+ * Parse all @mentions from user input
34
+ */
35
+ export function parseFileMentions(input: string): FileMention[] {
36
+ const mentions: FileMention[] = [];
37
+ let match: RegExpExecArray | null;
38
+
39
+ // Reset regex state
40
+ FILE_MENTION_REGEX.lastIndex = 0;
41
+
42
+ while ((match = FILE_MENTION_REGEX.exec(input)) !== null) {
43
+ const rawText = match[0]; // Full match: "@src/app.tsx:10-20"
44
+ const filePath = match[1]; // Captured group 1: "src/app.tsx"
45
+ const lineStart = match[2]; // Captured group 2: "10"
46
+ const lineEnd = match[3]; // Captured group 3: "20"
47
+
48
+ // Skip if the file path is empty or invalid
49
+ if (!filePath || !isValidFilePath(filePath)) {
50
+ continue;
51
+ }
52
+
53
+ const mention: FileMention = {
54
+ rawText,
55
+ filePath,
56
+ startIndex: match.index,
57
+ endIndex: match.index + rawText.length,
58
+ };
59
+
60
+ // Parse line range if present
61
+ if (lineStart) {
62
+ const start = parseInt(lineStart, 10);
63
+ const end = lineEnd ? parseInt(lineEnd, 10) : undefined;
64
+
65
+ // Validate line numbers
66
+ if (start > 0 && (!end || end >= start)) {
67
+ mention.lineRange = {start, end};
68
+ }
69
+ }
70
+
71
+ mentions.push(mention);
72
+ }
73
+
74
+ return mentions;
75
+ }
76
+
77
+ /**
78
+ * Validate file path to prevent directory traversal attacks
79
+ * and ensure it's within the project directory
80
+ */
81
+ export function isValidFilePath(filePath: string): boolean {
82
+ // Reject empty paths
83
+ if (!filePath || filePath.trim().length === 0) {
84
+ return false;
85
+ }
86
+
87
+ // Reject paths that try to escape parent directories
88
+ if (filePath.includes('..')) {
89
+ return false;
90
+ }
91
+
92
+ // Reject absolute paths (outside project)
93
+ if (path.isAbsolute(filePath)) {
94
+ return false;
95
+ }
96
+
97
+ // Reject Windows absolute paths (C:\, D:\, etc.) even on Unix systems
98
+ if (/^[A-Za-z]:[/\\]/.test(filePath)) {
99
+ return false;
100
+ }
101
+
102
+ // Reject paths with null bytes (security)
103
+ if (filePath.includes('\0')) {
104
+ return false;
105
+ }
106
+
107
+ // Reject paths that start with special characters that could be problematic
108
+ if (filePath.startsWith('/') || filePath.startsWith('\\')) {
109
+ return false;
110
+ }
111
+
112
+ return true;
113
+ }
114
+
115
+ /**
116
+ * Resolve a relative file path to an absolute path within the project
117
+ */
118
+ export function resolveFilePath(filePath: string, cwd: string): string {
119
+ // Validate first
120
+ if (!isValidFilePath(filePath)) {
121
+ throw new Error(`Invalid file path: ${filePath}`);
122
+ }
123
+
124
+ // Resolve to absolute path
125
+ const absolutePath = path.resolve(cwd, filePath);
126
+
127
+ // Ensure the resolved path is still within the project directory
128
+ const normalizedCwd = path.resolve(cwd);
129
+ if (!absolutePath.startsWith(normalizedCwd)) {
130
+ throw new Error(
131
+ `File path escapes project directory: ${filePath} -> ${absolutePath}`,
132
+ );
133
+ }
134
+
135
+ return absolutePath;
136
+ }
137
+
138
+ /**
139
+ * Parse line range from a string like "10-20" or "10"
140
+ */
141
+ export function parseLineRange(
142
+ rangeStr: string,
143
+ ): {start: number; end?: number} | null {
144
+ if (!rangeStr) {
145
+ return null;
146
+ }
147
+
148
+ const parts = rangeStr.split('-');
149
+
150
+ if (parts.length === 1) {
151
+ // Single line: "10"
152
+ const line = parseInt(parts[0], 10);
153
+ if (isNaN(line) || line <= 0) {
154
+ return null;
155
+ }
156
+ return {start: line, end: undefined};
157
+ } else if (parts.length === 2) {
158
+ // Range: "10-20"
159
+ const start = parseInt(parts[0], 10);
160
+ const end = parseInt(parts[1], 10);
161
+
162
+ if (isNaN(start) || isNaN(end) || start <= 0 || end < start) {
163
+ return null;
164
+ }
165
+
166
+ return {start, end};
167
+ }
168
+
169
+ return null;
170
+ }
@@ -0,0 +1,149 @@
1
+ import test from 'ava';
2
+ import {fuzzyScore, fuzzyScoreFilePath} from './fuzzy-matching';
3
+
4
+ console.log(`\nfuzzy-matching.spec.ts`);
5
+
6
+ // Tests for fuzzyScore (general text matching)
7
+ test('fuzzyScore - exact match returns highest score', t => {
8
+ t.is(fuzzyScore('help', 'help'), 1000);
9
+ t.is(fuzzyScore('status', 'status'), 1000);
10
+ });
11
+
12
+ test('fuzzyScore - case insensitive matching', t => {
13
+ t.is(fuzzyScore('Help', 'help'), 1000);
14
+ t.is(fuzzyScore('STATUS', 'status'), 1000);
15
+ t.is(fuzzyScore('help', 'HELP'), 1000);
16
+ });
17
+
18
+ test('fuzzyScore - prefix match', t => {
19
+ const score = fuzzyScore('help', 'hel');
20
+ t.true(score > 0);
21
+ t.true(score >= 800);
22
+ });
23
+
24
+ test('fuzzyScore - substring match', t => {
25
+ const score = fuzzyScore('status', 'tat');
26
+ t.true(score > 0);
27
+ t.true(score >= 700);
28
+ });
29
+
30
+ test('fuzzyScore - sequential character match', t => {
31
+ const score = fuzzyScore('provider', 'pvdr');
32
+ t.true(score > 0);
33
+ t.true(score < 700); // Should be lower than substring match
34
+ });
35
+
36
+ test('fuzzyScore - no match returns 0', t => {
37
+ t.is(fuzzyScore('help', 'xyz'), 0);
38
+ t.is(fuzzyScore('status', 'abc'), 0);
39
+ });
40
+
41
+ test('fuzzyScore - empty query returns 0', t => {
42
+ t.is(fuzzyScore('help', ''), 0);
43
+ t.is(fuzzyScore('anything', ''), 0);
44
+ });
45
+
46
+ test('fuzzyScore - ranks exact match higher than prefix', t => {
47
+ const exactScore = fuzzyScore('help', 'help');
48
+ const prefixScore = fuzzyScore('helper', 'help');
49
+ t.true(exactScore > prefixScore);
50
+ });
51
+
52
+ test('fuzzyScore - ranks prefix higher than substring', t => {
53
+ const prefixScore = fuzzyScore('helper', 'help');
54
+ const substringScore = fuzzyScore('unhelpful', 'help');
55
+ t.true(prefixScore > substringScore);
56
+ });
57
+
58
+ test('fuzzyScore - ranks substring higher than fuzzy', t => {
59
+ const substringScore = fuzzyScore('helper', 'elp');
60
+ const fuzzyScore1 = fuzzyScore('enable-lp', 'elp');
61
+ t.true(substringScore > fuzzyScore1);
62
+ });
63
+
64
+ // Tests for fuzzyScoreFilePath (file path specific matching)
65
+ test('fuzzyScoreFilePath - exact filename match scores high', t => {
66
+ const score = fuzzyScoreFilePath('src/utils/helper.ts', 'helper.ts');
67
+ t.true(score >= 900);
68
+ });
69
+
70
+ test('fuzzyScoreFilePath - filename prefix scores higher than path prefix', t => {
71
+ const filenameScore = fuzzyScoreFilePath('src/utils/helper.ts', 'help');
72
+ const pathScore = fuzzyScoreFilePath('helper/src/utils/file.ts', 'help');
73
+ t.true(filenameScore > pathScore);
74
+ });
75
+
76
+ test('fuzzyScoreFilePath - filename substring matches correctly', t => {
77
+ const filenameScore = fuzzyScoreFilePath('src/utils/my-helper.ts', 'help');
78
+ const pathScore = fuzzyScoreFilePath('helper/src/utils/file.ts', 'help');
79
+ // Both should have scores, but path prefix might score higher than filename substring
80
+ t.true(filenameScore > 0);
81
+ t.true(pathScore > 0);
82
+ });
83
+
84
+ test('fuzzyScoreFilePath - handles paths with multiple slashes', t => {
85
+ const score = fuzzyScoreFilePath('a/b/c/d/file.ts', 'file');
86
+ t.true(score > 0);
87
+ });
88
+
89
+ test('fuzzyScoreFilePath - case insensitive for paths', t => {
90
+ const score1 = fuzzyScoreFilePath('src/Utils/Helper.ts', 'helper');
91
+ const score2 = fuzzyScoreFilePath('src/utils/helper.ts', 'HELPER');
92
+ t.true(score1 > 0);
93
+ t.true(score2 > 0);
94
+ });
95
+
96
+ // Command-specific test scenarios
97
+ test('command matching - /h matches help, status does not', t => {
98
+ const helpScore = fuzzyScore('help', 'h');
99
+ const statusScore = fuzzyScore('status', 'h');
100
+ t.true(helpScore > 0);
101
+ t.is(statusScore, 0);
102
+ });
103
+
104
+ test('command matching - /st matches status', t => {
105
+ const score = fuzzyScore('status', 'st');
106
+ t.true(score > 0);
107
+ t.true(score >= 800); // Should be prefix match
108
+ });
109
+
110
+ test('command matching - /mod matches model', t => {
111
+ const modelScore = fuzzyScore('model', 'mod');
112
+ const recommendationsScore = fuzzyScore('recommendations', 'mod');
113
+
114
+ t.true(modelScore > 0);
115
+ // recommendations doesn't contain 'mod' as substring, so it should be 0 or fuzzy match
116
+ // model should score higher (prefix match)
117
+ t.true(modelScore > recommendationsScore);
118
+ });
119
+
120
+ test('command matching - fuzzy matching for /pvd matches provider', t => {
121
+ const score = fuzzyScore('provider', 'pvd');
122
+ t.true(score > 0);
123
+ });
124
+
125
+ test('command matching - /init matches init higher than initialization', t => {
126
+ const initScore = fuzzyScore('init', 'init');
127
+ const initializationScore = fuzzyScore('initialization', 'init');
128
+
129
+ t.true(initScore > initializationScore);
130
+ });
131
+
132
+ // Edge cases
133
+ test('fuzzyScore - single character queries work', t => {
134
+ const score1 = fuzzyScore('help', 'h');
135
+ const score2 = fuzzyScore('status', 's');
136
+ t.true(score1 > 0);
137
+ t.true(score2 > 0);
138
+ });
139
+
140
+ test('fuzzyScore - longer text with short query', t => {
141
+ const score = fuzzyScore('very-long-command-name-here', 'vlc');
142
+ t.true(score > 0);
143
+ });
144
+
145
+ test('fuzzyScoreFilePath - works with Windows-style paths', t => {
146
+ // While the function uses /, testing that it handles edge cases
147
+ const score = fuzzyScoreFilePath('src\\utils\\file.ts', 'file');
148
+ t.true(score > 0);
149
+ });
@@ -0,0 +1,146 @@
1
+ /**
2
+ * Fuzzy match scoring algorithm
3
+ * Returns a score from 0 to 1000 (higher = better match)
4
+ *
5
+ * This can be used for matching file paths, command names, or any other strings.
6
+ */
7
+ export function fuzzyScore(text: string, query: string): number {
8
+ if (!query) {
9
+ return 0;
10
+ }
11
+
12
+ const lowerText = text.toLowerCase();
13
+ const lowerQuery = query.toLowerCase();
14
+
15
+ // Exact match (highest score)
16
+ if (lowerText === lowerQuery) {
17
+ return 1000;
18
+ }
19
+
20
+ // Text ends with query
21
+ if (lowerText.endsWith(lowerQuery)) {
22
+ return 850;
23
+ }
24
+
25
+ // Text starts with query
26
+ if (lowerText.startsWith(lowerQuery)) {
27
+ return 800;
28
+ }
29
+
30
+ // Text contains query as substring
31
+ if (lowerText.includes(lowerQuery)) {
32
+ return 700;
33
+ }
34
+
35
+ // Sequential character match (fuzzy)
36
+ // All query characters appear in order in the text
37
+ let textIndex = 0;
38
+ let queryIndex = 0;
39
+ let lastMatchIndex = -1;
40
+ let consecutiveMatches = 0;
41
+
42
+ while (textIndex < lowerText.length && queryIndex < lowerQuery.length) {
43
+ if (lowerText[textIndex] === lowerQuery[queryIndex]) {
44
+ // Bonus for consecutive matches
45
+ if (textIndex === lastMatchIndex + 1) {
46
+ consecutiveMatches++;
47
+ } else {
48
+ consecutiveMatches = 1;
49
+ }
50
+ lastMatchIndex = textIndex;
51
+ queryIndex++;
52
+ }
53
+ textIndex++;
54
+ }
55
+
56
+ // If all query characters matched
57
+ if (queryIndex === lowerQuery.length) {
58
+ // Score based on match density and consecutive matches
59
+ const matchDensity = lowerQuery.length / lowerText.length;
60
+ const consecutiveBonus = consecutiveMatches * 50;
61
+ return Math.min(500 + matchDensity * 100 + consecutiveBonus, 699);
62
+ }
63
+
64
+ // No match
65
+ return 0;
66
+ }
67
+
68
+ /**
69
+ * Fuzzy score specifically for file paths
70
+ * Gives higher priority to filename matches over directory matches
71
+ */
72
+ export function fuzzyScoreFilePath(filePath: string, query: string): number {
73
+ if (!query) {
74
+ return 0;
75
+ }
76
+
77
+ const lowerPath = filePath.toLowerCase();
78
+ const lowerQuery = query.toLowerCase();
79
+
80
+ // Exact match (highest score)
81
+ if (lowerPath === lowerQuery) {
82
+ return 1000;
83
+ }
84
+
85
+ // Exact match of filename (without path)
86
+ const filename = filePath.split('/').pop() || '';
87
+ if (filename.toLowerCase() === lowerQuery) {
88
+ return 900;
89
+ }
90
+
91
+ // Path ends with query
92
+ if (lowerPath.endsWith(lowerQuery)) {
93
+ return 850;
94
+ }
95
+
96
+ // Filename starts with query
97
+ if (filename.toLowerCase().startsWith(lowerQuery)) {
98
+ return 800;
99
+ }
100
+
101
+ // Path starts with query
102
+ if (lowerPath.startsWith(lowerQuery)) {
103
+ return 750;
104
+ }
105
+
106
+ // Filename contains query as substring
107
+ if (filename.toLowerCase().includes(lowerQuery)) {
108
+ return 700;
109
+ }
110
+
111
+ // Path contains query as substring
112
+ if (lowerPath.includes(lowerQuery)) {
113
+ return 600;
114
+ }
115
+
116
+ // Sequential character match (fuzzy)
117
+ let pathIndex = 0;
118
+ let queryIndex = 0;
119
+ let lastMatchIndex = -1;
120
+ let consecutiveMatches = 0;
121
+
122
+ while (pathIndex < lowerPath.length && queryIndex < lowerQuery.length) {
123
+ if (lowerPath[pathIndex] === lowerQuery[queryIndex]) {
124
+ // Bonus for consecutive matches
125
+ if (pathIndex === lastMatchIndex + 1) {
126
+ consecutiveMatches++;
127
+ } else {
128
+ consecutiveMatches = 1;
129
+ }
130
+ lastMatchIndex = pathIndex;
131
+ queryIndex++;
132
+ }
133
+ pathIndex++;
134
+ }
135
+
136
+ // If all query characters matched
137
+ if (queryIndex === lowerQuery.length) {
138
+ // Score based on match density and consecutive matches
139
+ const matchDensity = lowerQuery.length / lowerPath.length;
140
+ const consecutiveBonus = consecutiveMatches * 50;
141
+ return Math.min(500 + matchDensity * 100 + consecutiveBonus, 599);
142
+ }
143
+
144
+ // No match
145
+ return 0;
146
+ }