@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,796 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {themes} from '../config/themes';
5
+ import {ThemeContext} from '../hooks/useTheme';
6
+ import {
7
+ type Colors,
8
+ decodeHtmlEntities,
9
+ parseMarkdown,
10
+ parseMarkdownTable,
11
+ } from '../markdown-parser/index';
12
+ import AssistantMessage from './assistant-message';
13
+
14
+ // Mock theme colors for testing
15
+ const mockColors: any = {
16
+ primary: '#3b82f6',
17
+ secondary: '#6b7280',
18
+ success: '#10b981',
19
+ error: '#ef4444',
20
+ warning: '#f59e0b',
21
+ info: '#3b82f6',
22
+ white: '#ffffff',
23
+ black: '#000000',
24
+ tool: '#8b5cf6',
25
+ diffAdded: '#10b981',
26
+ diffRemoved: '#ef4444',
27
+ diffAddedText: '#d1fae5',
28
+ diffRemovedText: '#fee2e2',
29
+ };
30
+
31
+ console.log(`\nassistant-message.spec.tsx – ${React.version}`);
32
+
33
+ // Mock ThemeProvider for testing
34
+ const MockThemeProvider = ({children}: {children: React.ReactNode}) => {
35
+ const mockTheme = {
36
+ currentTheme: 'tokyo-night' as const,
37
+ colors: themes['tokyo-night'].colors,
38
+ setCurrentTheme: () => {},
39
+ };
40
+
41
+ return (
42
+ <ThemeContext.Provider value={mockTheme}>{children}</ThemeContext.Provider>
43
+ );
44
+ };
45
+
46
+ // ============================================================================
47
+ // Component Rendering Tests
48
+ // ============================================================================
49
+
50
+ test('AssistantMessage renders with basic message', t => {
51
+ const {lastFrame} = render(
52
+ <MockThemeProvider>
53
+ <AssistantMessage message="Hello world" model="test-model" />
54
+ </MockThemeProvider>,
55
+ );
56
+
57
+ const output = lastFrame();
58
+ t.truthy(output);
59
+ t.regex(output!, /test-model:/);
60
+ t.regex(output!, /Hello world/);
61
+ });
62
+
63
+ test('AssistantMessage renders with bold text', t => {
64
+ const {lastFrame} = render(
65
+ <MockThemeProvider>
66
+ <AssistantMessage message="This is **bold** text" model="test-model" />
67
+ </MockThemeProvider>,
68
+ );
69
+
70
+ const output = lastFrame();
71
+ t.truthy(output);
72
+ t.regex(output!, /bold/);
73
+ });
74
+
75
+ test('AssistantMessage renders with inline code', t => {
76
+ const {lastFrame} = render(
77
+ <MockThemeProvider>
78
+ <AssistantMessage
79
+ message="Use `const` for constants"
80
+ model="test-model"
81
+ />
82
+ </MockThemeProvider>,
83
+ );
84
+
85
+ const output = lastFrame();
86
+ t.truthy(output);
87
+ t.regex(output!, /const/);
88
+ t.regex(output!, /for constants/);
89
+ });
90
+
91
+ test('AssistantMessage renders with HTML entities', t => {
92
+ const {lastFrame} = render(
93
+ <MockThemeProvider>
94
+ <AssistantMessage
95
+ message="Price: &euro;100&nbsp;only"
96
+ model="test-model"
97
+ />
98
+ </MockThemeProvider>,
99
+ );
100
+
101
+ const output = lastFrame();
102
+ t.truthy(output);
103
+ // Should have decoded entities
104
+ t.regex(output!, /Price:/);
105
+ t.regex(output!, /100/);
106
+ t.regex(output!, /only/);
107
+ });
108
+
109
+ test('AssistantMessage renders with markdown table', t => {
110
+ const message = `| Name | Age |
111
+ |------|-----|
112
+ | John | 30 |
113
+ | Jane | 25 |`;
114
+
115
+ const {lastFrame} = render(
116
+ <MockThemeProvider>
117
+ <AssistantMessage message={message} model="test-model" />
118
+ </MockThemeProvider>,
119
+ );
120
+
121
+ const output = lastFrame();
122
+ t.truthy(output);
123
+ t.regex(output!, /Name/);
124
+ t.regex(output!, /Age/);
125
+ t.regex(output!, /John/);
126
+ t.regex(output!, /Jane/);
127
+ // Should contain table separators
128
+ t.regex(output!, /│/);
129
+ });
130
+
131
+ test('AssistantMessage renders with headings', t => {
132
+ const {lastFrame} = render(
133
+ <MockThemeProvider>
134
+ <AssistantMessage message="# Main Heading" model="test-model" />
135
+ </MockThemeProvider>,
136
+ );
137
+
138
+ const output = lastFrame();
139
+ t.truthy(output);
140
+ t.regex(output!, /Main Heading/);
141
+ });
142
+
143
+ test('AssistantMessage renders with lists', t => {
144
+ const message = `- Item 1
145
+ - Item 2
146
+ - Item 3`;
147
+
148
+ const {lastFrame} = render(
149
+ <MockThemeProvider>
150
+ <AssistantMessage message={message} model="test-model" />
151
+ </MockThemeProvider>,
152
+ );
153
+
154
+ const output = lastFrame();
155
+ t.truthy(output);
156
+ t.regex(output!, /Item 1/);
157
+ t.regex(output!, /Item 2/);
158
+ t.regex(output!, /Item 3/);
159
+ // Should contain bullets
160
+ t.regex(output!, /•/);
161
+ });
162
+
163
+ test('AssistantMessage renders with blockquotes', t => {
164
+ const {lastFrame} = render(
165
+ <MockThemeProvider>
166
+ <AssistantMessage message="> This is a quote" model="test-model" />
167
+ </MockThemeProvider>,
168
+ );
169
+
170
+ const output = lastFrame();
171
+ t.truthy(output);
172
+ t.regex(output!, /This is a quote/);
173
+ });
174
+
175
+ test('AssistantMessage renders with links', t => {
176
+ const {lastFrame} = render(
177
+ <MockThemeProvider>
178
+ <AssistantMessage
179
+ message="Check [this link](https://example.com)"
180
+ model="test-model"
181
+ />
182
+ </MockThemeProvider>,
183
+ );
184
+
185
+ const output = lastFrame();
186
+ t.truthy(output);
187
+ t.regex(output!, /this link/);
188
+ t.regex(output!, /https:\/\/example\.com/);
189
+ });
190
+
191
+ test('AssistantMessage renders with mixed markdown', t => {
192
+ const message = `# Title
193
+
194
+ This has **bold** and *italic* text.
195
+
196
+ - List item
197
+
198
+ Price: &euro;50`;
199
+
200
+ const {lastFrame} = render(
201
+ <MockThemeProvider>
202
+ <AssistantMessage message={message} model="test-model" />
203
+ </MockThemeProvider>,
204
+ );
205
+
206
+ const output = lastFrame();
207
+ t.truthy(output);
208
+ t.regex(output!, /Title/);
209
+ t.regex(output!, /bold/);
210
+ t.regex(output!, /italic/);
211
+ t.regex(output!, /List item/);
212
+ t.regex(output!, /50/);
213
+ });
214
+
215
+ test('AssistantMessage renders without crashing with empty message', t => {
216
+ const {lastFrame} = render(
217
+ <MockThemeProvider>
218
+ <AssistantMessage message="" model="test-model" />
219
+ </MockThemeProvider>,
220
+ );
221
+
222
+ const output = lastFrame();
223
+ t.truthy(output);
224
+ t.regex(output!, /test-model:/);
225
+ });
226
+
227
+ test('AssistantMessage renders model name correctly', t => {
228
+ const {lastFrame} = render(
229
+ <MockThemeProvider>
230
+ <AssistantMessage message="Test" model="gpt-4" />
231
+ </MockThemeProvider>,
232
+ );
233
+
234
+ const output = lastFrame();
235
+ t.truthy(output);
236
+ t.regex(output!, /gpt-4:/);
237
+ });
238
+
239
+ // ============================================================================
240
+ // HTML Entity Decoding Tests
241
+ // ============================================================================
242
+
243
+ test('decodeHtmlEntities handles common entities', t => {
244
+ const input = 'Hello&nbsp;world&amp;test';
245
+ const result = decodeHtmlEntities(input);
246
+ t.is(result, 'Hello world&test');
247
+ });
248
+
249
+ test('decodeHtmlEntities handles less-than and greater-than', t => {
250
+ const input = '&lt;div&gt;content&lt;/div&gt;';
251
+ const result = decodeHtmlEntities(input);
252
+ t.is(result, '<div>content</div>');
253
+ });
254
+
255
+ test('decodeHtmlEntities handles quotes', t => {
256
+ const input = '&quot;Hello&quot; &apos;World&apos;';
257
+ const result = decodeHtmlEntities(input);
258
+ t.is(result, '"Hello" \'World\'');
259
+ });
260
+
261
+ test('decodeHtmlEntities handles copyright and trademark symbols', t => {
262
+ const input = '&copy; 2024 &reg; &trade;';
263
+ const result = decodeHtmlEntities(input);
264
+ t.is(result, '© 2024 ® ™');
265
+ });
266
+
267
+ test('decodeHtmlEntities handles currency symbols', t => {
268
+ const input = '&euro;100 &pound;50 &yen;1000 &cent;25';
269
+ const result = decodeHtmlEntities(input);
270
+ t.is(result, '€100 £50 ¥1000 ¢25');
271
+ });
272
+
273
+ test('decodeHtmlEntities handles mathematical symbols', t => {
274
+ const input = '45&deg; &plusmn;5 2&times;3 10&divide;2';
275
+ const result = decodeHtmlEntities(input);
276
+ t.is(result, '45° ±5 2×3 10÷2');
277
+ });
278
+
279
+ test('decodeHtmlEntities handles typography symbols', t => {
280
+ const input =
281
+ '&ndash; &mdash; &lsquo;text&rsquo; &ldquo;quote&rdquo; &hellip; &bull;';
282
+ const result = decodeHtmlEntities(input);
283
+ t.is(result, '– — \u2018text\u2019 \u201Cquote\u201D … •');
284
+ });
285
+
286
+ test('decodeHtmlEntities handles numeric entities (decimal)', t => {
287
+ const input = 'Space&#160;here and&#32;there';
288
+ const result = decodeHtmlEntities(input);
289
+ // &#160; is non-breaking space, &#32; is regular space
290
+ t.is(result, 'Space\u00A0here and there');
291
+ });
292
+
293
+ test('decodeHtmlEntities handles numeric entities (hexadecimal)', t => {
294
+ const input = 'Unicode&#xA0;space and&#x20;space';
295
+ const result = decodeHtmlEntities(input);
296
+ // &#xA0; is non-breaking space, &#x20; is regular space
297
+ t.is(result, 'Unicode\u00A0space and space');
298
+ });
299
+
300
+ test('decodeHtmlEntities handles mixed entities', t => {
301
+ const input =
302
+ '&lt;p&gt;Price: &euro;100&nbsp;&plusmn;&nbsp;&#8364;5&lt;/p&gt;';
303
+ const result = decodeHtmlEntities(input);
304
+ t.is(result, '<p>Price: €100 ± €5</p>');
305
+ });
306
+
307
+ test('decodeHtmlEntities leaves normal text unchanged', t => {
308
+ const input = 'Normal text without entities';
309
+ const result = decodeHtmlEntities(input);
310
+ t.is(result, input);
311
+ });
312
+
313
+ test('decodeHtmlEntities handles multiple occurrences of same entity', t => {
314
+ const input = '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;';
315
+ const result = decodeHtmlEntities(input);
316
+ t.is(result, ' ');
317
+ });
318
+
319
+ // ============================================================================
320
+ // Markdown Table Parsing Tests
321
+ // ============================================================================
322
+
323
+ test('parseMarkdownTable handles simple two-column table', t => {
324
+ const table = `| Header 1 | Header 2 |
325
+ |----------|----------|
326
+ | Cell 1 | Cell 2 |
327
+ | Cell 3 | Cell 4 |
328
+ `;
329
+ const result = parseMarkdownTable(table, mockColors);
330
+ t.true(result.includes('Header 1'));
331
+ t.true(result.includes('Header 2'));
332
+ t.true(result.includes('Cell 1'));
333
+ t.true(result.includes('Cell 4'));
334
+ t.true(result.includes('│')); // Contains column separator
335
+ t.true(result.includes('─')); // Contains row separator
336
+ });
337
+
338
+ test('parseMarkdownTable handles table with varying cell lengths', t => {
339
+ const table = `| Short | Very Long Header Text |
340
+ |-------|----------------------|
341
+ | A | B |
342
+ | Long text here | X |
343
+ `;
344
+ const result = parseMarkdownTable(table, mockColors);
345
+ t.true(result.includes('Short'));
346
+ t.true(result.includes('Very Long Header Text'));
347
+ t.true(result.includes('Long text here'));
348
+ });
349
+
350
+ test('parseMarkdownTable returns original text for invalid table', t => {
351
+ const notATable = 'Just some text without table structure';
352
+ const result = parseMarkdownTable(notATable, mockColors);
353
+ t.is(result, notATable);
354
+ });
355
+
356
+ test('parseMarkdownTable returns original text for table without data rows', t => {
357
+ const table = `| Header 1 | Header 2 |
358
+ |----------|----------|
359
+ `;
360
+ const result = parseMarkdownTable(table, mockColors);
361
+ t.is(result, table);
362
+ });
363
+
364
+ test('parseMarkdownTable handles table with empty cells', t => {
365
+ const table = `| Col1 | Col2 |
366
+ |------|------|
367
+ | Text | |
368
+ | | Text |
369
+ `;
370
+ const result = parseMarkdownTable(table, mockColors);
371
+ t.true(result.includes('Col1'));
372
+ t.true(result.includes('Col2'));
373
+ t.true(result.includes('Text'));
374
+ });
375
+
376
+ test('parseMarkdownTable normalizes column count', t => {
377
+ const table = `| A | B |
378
+ |---|---|
379
+ | 1 | 2 |
380
+ | 3 |
381
+ `;
382
+ const result = parseMarkdownTable(table, mockColors);
383
+ // Should handle the missing cell gracefully
384
+ t.true(result.includes('A'));
385
+ t.true(result.includes('B'));
386
+ t.true(result.includes('1'));
387
+ t.true(result.includes('3'));
388
+ });
389
+
390
+ test('parseMarkdownTable handles markdown formatting in cells', t => {
391
+ const table = `| Command | Description |
392
+ |---------|-------------|
393
+ | \`npm run build\` | Compile TypeScript |
394
+ | **Important** | Do this first |
395
+ | [Link](url) | External reference |
396
+ `;
397
+ const result = parseMarkdownTable(table, mockColors);
398
+ // Should strip markdown for proper alignment
399
+ t.true(result.includes('Command'));
400
+ t.true(result.includes('Description'));
401
+ t.true(result.includes('npm run build'));
402
+ t.true(result.includes('Important'));
403
+ t.true(result.includes('Link'));
404
+ // Should have proper table structure
405
+ t.true(result.includes('│'));
406
+ t.true(result.includes('─'));
407
+ });
408
+
409
+ // ============================================================================
410
+ // Markdown Parsing Tests
411
+ // ============================================================================
412
+
413
+ test('parseMarkdown handles inline code', t => {
414
+ const text = 'Use `const` for constants';
415
+ const result = parseMarkdown(text, mockColors);
416
+ t.true(result.includes('const'));
417
+ // Should still contain the word even if styled
418
+ t.true(result.includes('for constants'));
419
+ });
420
+
421
+ test('parseMarkdown handles bold text', t => {
422
+ const text = 'This is **bold** text';
423
+ const result = parseMarkdown(text, mockColors);
424
+ t.true(result.includes('bold'));
425
+ t.true(result.includes('This is'));
426
+ });
427
+
428
+ test('parseMarkdown handles italic text', t => {
429
+ const text = 'This is *italic* text';
430
+ const result = parseMarkdown(text, mockColors);
431
+ t.true(result.includes('italic'));
432
+ t.true(result.includes('text'));
433
+ });
434
+
435
+ test('parseMarkdown preserves underscores in identifiers', t => {
436
+ const text =
437
+ 'Use `create_file`, `read_file`, or `search_file_contents` functions';
438
+ const result = parseMarkdown(text, mockColors);
439
+ // Underscores should be preserved in code identifiers
440
+ t.true(result.includes('create_file'));
441
+ t.true(result.includes('read_file'));
442
+ t.true(result.includes('search_file_contents'));
443
+ });
444
+
445
+ test('parseMarkdown preserves underscores in regular text', t => {
446
+ const text = 'The variable_name and function_call should remain intact';
447
+ const result = parseMarkdown(text, mockColors);
448
+ // Underscores should NOT be treated as markdown formatting
449
+ t.true(result.includes('variable_name'));
450
+ t.true(result.includes('function_call'));
451
+ });
452
+
453
+ test('parseMarkdown handles headings', t => {
454
+ const text = '# Main Heading\n## Subheading';
455
+ const result = parseMarkdown(text, mockColors);
456
+ t.true(result.includes('Main Heading'));
457
+ t.true(result.includes('Subheading'));
458
+ });
459
+
460
+ test('parseMarkdown handles links', t => {
461
+ const text = 'Check [this link](https://example.com)';
462
+ const result = parseMarkdown(text, mockColors);
463
+ t.true(result.includes('this link'));
464
+ t.true(result.includes('https://example.com'));
465
+ });
466
+
467
+ test('parseMarkdown handles blockquotes', t => {
468
+ const text = '> This is a quote\n> Another line';
469
+ const result = parseMarkdown(text, mockColors);
470
+ t.true(result.includes('This is a quote'));
471
+ t.true(result.includes('Another line'));
472
+ });
473
+
474
+ test('parseMarkdown handles unordered lists', t => {
475
+ const text = '- Item 1\n- Item 2\n* Item 3';
476
+ const result = parseMarkdown(text, mockColors);
477
+ t.true(result.includes('Item 1'));
478
+ t.true(result.includes('Item 2'));
479
+ t.true(result.includes('Item 3'));
480
+ // Should convert to bullets
481
+ t.true(result.includes('•'));
482
+ });
483
+
484
+ test('parseMarkdown handles ordered lists', t => {
485
+ const text = '1. First item\n2. Second item\n3. Third item';
486
+ const result = parseMarkdown(text, mockColors);
487
+ // Should preserve the numbers
488
+ t.true(result.includes('1.'));
489
+ t.true(result.includes('2.'));
490
+ t.true(result.includes('3.'));
491
+ t.true(result.includes('First item'));
492
+ t.true(result.includes('Second item'));
493
+ t.true(result.includes('Third item'));
494
+ });
495
+
496
+ test('parseMarkdown decodes HTML entities', t => {
497
+ const text = 'Price: &euro;100&nbsp;only';
498
+ const result = parseMarkdown(text, mockColors);
499
+ t.true(result.includes('€'));
500
+ t.true(result.includes('100'));
501
+ t.true(result.includes('only'));
502
+ t.false(result.includes('&euro;'));
503
+ t.false(result.includes('&nbsp;'));
504
+ });
505
+
506
+ test('parseMarkdown handles tables', t => {
507
+ const text = `
508
+ Here is a table:
509
+
510
+ | Name | Age |
511
+ |------|-----|
512
+ | John | 30 |
513
+ | Jane | 25 |
514
+
515
+ That was the table.
516
+ `;
517
+ const result = parseMarkdown(text, mockColors);
518
+ t.true(result.includes('Name'));
519
+ t.true(result.includes('Age'));
520
+ t.true(result.includes('John'));
521
+ t.true(result.includes('Jane'));
522
+ t.true(result.includes('│'));
523
+ });
524
+
525
+ test('parseMarkdown handles mixed markdown features', t => {
526
+ const text = `
527
+ # Title
528
+
529
+ This has **bold** and *italic* text.
530
+
531
+ - List item 1
532
+ - List item 2
533
+
534
+ \`code here\`
535
+
536
+ > A quote
537
+
538
+ Price: &euro;50
539
+ `;
540
+ const result = parseMarkdown(text, mockColors);
541
+ t.true(result.includes('Title'));
542
+ t.true(result.includes('bold'));
543
+ t.true(result.includes('italic'));
544
+ t.true(result.includes('List item 1'));
545
+ t.true(result.includes('code here'));
546
+ t.true(result.includes('A quote'));
547
+ t.true(result.includes('€'));
548
+ t.false(result.includes('&euro;'));
549
+ });
550
+
551
+ test('parseMarkdown handles plain text without markdown', t => {
552
+ const text = 'Just plain text with no special formatting';
553
+ const result = parseMarkdown(text, mockColors);
554
+ t.is(result, text);
555
+ });
556
+
557
+ test('parseMarkdown handles empty string', t => {
558
+ const result = parseMarkdown('', mockColors);
559
+ t.is(result, '');
560
+ });
561
+
562
+ test('parseMarkdown handles code blocks with language', t => {
563
+ const text = '```javascript\nconst x = 5;\n```';
564
+ const result = parseMarkdown(text, mockColors);
565
+ t.true(result.includes('const'));
566
+ t.true(result.includes('x'));
567
+ });
568
+
569
+ test('parseMarkdown handles code blocks without language', t => {
570
+ const text = '```\nplain code\n```';
571
+ const result = parseMarkdown(text, mockColors);
572
+ t.true(result.includes('plain code'));
573
+ });
574
+
575
+ // ============================================================================
576
+ // Edge Case Tests - Things That Should NOT Be Formatted
577
+ // ============================================================================
578
+
579
+ test('parseMarkdown does not create bullet list from hyphen in middle of line', t => {
580
+ const text = 'The file path is C:\\Users\\John - Documents\\file.txt';
581
+ const result = parseMarkdown(text, mockColors);
582
+ // Should not have bullet point character
583
+ t.false(result.includes('•'));
584
+ t.true(result.includes('C:\\Users\\John - Documents\\file.txt'));
585
+ });
586
+
587
+ test('parseMarkdown does not format asterisks in math expressions', t => {
588
+ const text = 'Calculate 5 * 3 * 2 = 30';
589
+ const result = parseMarkdown(text, mockColors);
590
+ // Should preserve the asterisks and not apply formatting
591
+ t.true(result.includes('5 * 3 * 2'));
592
+ });
593
+
594
+ test('parseMarkdown does not format asterisks in file globs', t => {
595
+ const text = 'Use glob pattern *.tsx or **/*.js to match files';
596
+ const result = parseMarkdown(text, mockColors);
597
+ // Should preserve asterisks in glob patterns
598
+ t.true(result.includes('*.tsx'));
599
+ t.true(result.includes('**/*.js'));
600
+ });
601
+
602
+ test('parseMarkdown does not create heading from hash in middle of line', t => {
603
+ const text = 'The commit hash is abc123 #main branch';
604
+ const result = parseMarkdown(text, mockColors);
605
+ // Should not be formatted as heading since # is not at line start
606
+ t.true(result.includes('#main'));
607
+ });
608
+
609
+ test('parseMarkdown does not create heading from hex color', t => {
610
+ const text = 'Use color #3b82f6 for the button';
611
+ const result = parseMarkdown(text, mockColors);
612
+ // Should not format as heading
613
+ t.true(result.includes('#3b82f6'));
614
+ });
615
+
616
+ test('parseMarkdown preserves asterisks inside code blocks', t => {
617
+ const text = '```javascript\nconst pattern = /\\*\\*/g;\n```';
618
+ const result = parseMarkdown(text, mockColors);
619
+ // Asterisks inside code block should not trigger bold formatting
620
+ t.true(result.includes('pattern'));
621
+ // Code block content should be preserved
622
+ t.true(result.includes('/\\*\\*/g'));
623
+ });
624
+
625
+ test('parseMarkdown preserves markdown-like syntax inside inline code', t => {
626
+ const text = 'Use the pattern `**bold**` to make text bold';
627
+ const result = parseMarkdown(text, mockColors);
628
+ // The **bold** inside backticks should NOT be formatted
629
+ t.true(result.includes('**bold**'));
630
+ });
631
+
632
+ test('parseMarkdown does not format bullet with no space after hyphen', t => {
633
+ const text = 'The range is 1-10 and 20-30';
634
+ const result = parseMarkdown(text, mockColors);
635
+ // Should not create bullets for hyphens without spaces
636
+ t.false(result.includes('•'));
637
+ t.true(result.includes('1-10'));
638
+ t.true(result.includes('20-30'));
639
+ });
640
+
641
+ test('parseMarkdown handles mixed asterisks and bullet points', t => {
642
+ const text = '* First item\n* Second item with 2 * 3 = 6 calculation';
643
+ const result = parseMarkdown(text, mockColors);
644
+ // Should have bullets for list items
645
+ t.true(result.includes('•'));
646
+ t.true(result.includes('First item'));
647
+ // But preserve asterisks in math
648
+ t.true(result.includes('2 * 3'));
649
+ });
650
+
651
+ test('parseMarkdown does not format single asterisk surrounded by word chars', t => {
652
+ const text = 'The pointer syntax is char*ptr or int*value';
653
+ const result = parseMarkdown(text, mockColors);
654
+ // Should preserve pointer syntax
655
+ t.true(result.includes('char*ptr'));
656
+ t.true(result.includes('int*value'));
657
+ });
658
+
659
+ test('parseMarkdown preserves double asterisks in code comments', t => {
660
+ const text = 'In C: /* comment */ and /** doc comment */';
661
+ const result = parseMarkdown(text, mockColors);
662
+ // Should not treat /* or /** as bold markers
663
+ t.true(result.includes('/*'));
664
+ t.true(result.includes('/**'));
665
+ });
666
+
667
+ test('parseMarkdown handles hyphen in URL correctly', t => {
668
+ const text = 'Visit [my-site](https://example-domain.com/my-page)';
669
+ const result = parseMarkdown(text, mockColors);
670
+ // Should not create bullets from hyphens in URLs
671
+ t.false(result.includes('•'));
672
+ t.true(result.includes('example-domain.com/my-page'));
673
+ });
674
+
675
+ test('parseMarkdown handles bullet list with bold text correctly', t => {
676
+ const text = `* **Reading and understanding code** – view files
677
+ * **Creating new files** – scaffold components
678
+ * **Editing existing code** – insert, replace, or delete`;
679
+ const result = parseMarkdown(text, mockColors);
680
+ // Should have bullets
681
+ t.true(result.includes('•'));
682
+ // Should have the text (bold markers may be removed)
683
+ t.true(result.includes('Reading and understanding code'));
684
+ t.true(result.includes('Creating new files'));
685
+ t.true(result.includes('Editing existing code'));
686
+ // Should NOT have italic formatting bleeding across lines
687
+ // (checking that all items have similar formatting)
688
+ });
689
+
690
+ test('parseMarkdown handles nested/indented bullet lists', t => {
691
+ const text = `- Top level item
692
+ - Nested item 1
693
+ - Nested item 2
694
+ - Double nested
695
+ - Another top level`;
696
+ const result = parseMarkdown(text, mockColors);
697
+ // Should have bullets
698
+ t.true(result.includes('•'));
699
+ t.true(result.includes('Top level item'));
700
+ t.true(result.includes('Nested item 1'));
701
+ t.true(result.includes('Double nested'));
702
+ // Check indentation is preserved (should have 2 spaces before nested bullets)
703
+ t.true(result.includes(' • Nested item 1'));
704
+ t.true(result.includes(' • Double nested'));
705
+ });
706
+
707
+ test('parseMarkdown handles nested numbered lists', t => {
708
+ const text = `1. First item
709
+ 1. Nested first
710
+ 2. Nested second
711
+ 2. Second item`;
712
+ const result = parseMarkdown(text, mockColors);
713
+ t.true(result.includes('1. First item'));
714
+ t.true(result.includes('2. Second item'));
715
+ // Check nested numbering preserved
716
+ t.true(result.includes(' 1. Nested first'));
717
+ t.true(result.includes(' 2. Nested second'));
718
+ });
719
+
720
+ test('parseMarkdown restores inline code placeholders correctly', t => {
721
+ const text = 'Use `npm install` and `npm start` commands';
722
+ const result = parseMarkdown(text, mockColors);
723
+ // Should have the code content
724
+ t.true(result.includes('npm install'));
725
+ t.true(result.includes('npm start'));
726
+ // Should NOT have placeholder remnants
727
+ t.false(result.includes('__INLINE_CODE'));
728
+ t.false(result.includes('_INLINE'));
729
+ t.false(result.includes('CODE_'));
730
+ });
731
+
732
+ test('parseMarkdown renders tables with plain text (no markdown)', t => {
733
+ const text = `| Command | Description |
734
+ |---------|-------------|
735
+ | \`npm install\` | Install dependencies |
736
+ | \`npm start\` | Start the application |`;
737
+ const result = parseMarkdown(text, mockColors);
738
+ // Should have the plain text content (backticks removed)
739
+ t.true(result.includes('npm install'));
740
+ t.true(result.includes('npm start'));
741
+ t.true(result.includes('Install dependencies'));
742
+ // Should NOT have backticks in table
743
+ t.false(result.includes('`npm install`'));
744
+ t.false(result.includes('`npm start`'));
745
+ // Should NOT have placeholder remnants or corruption
746
+ t.false(result.includes('__INLINE_CODE'));
747
+ t.false(result.includes('_INLINE'));
748
+ t.false(result.includes('CODE_'));
749
+ t.false(result.includes('INLINECODE'));
750
+ });
751
+
752
+ test('parseMarkdown converts <br> tags to newlines', t => {
753
+ const text = 'Line one<br>Line two<br/>Line three<BR>Line four';
754
+ const result = parseMarkdown(text, mockColors);
755
+ const lines = result.split('\n');
756
+ t.true(lines.length >= 4);
757
+ t.true(result.includes('Line one'));
758
+ t.true(result.includes('Line two'));
759
+ t.true(result.includes('Line three'));
760
+ t.true(result.includes('Line four'));
761
+ t.false(result.includes('<br'));
762
+ t.false(result.includes('<BR'));
763
+ });
764
+
765
+ test('parseMarkdown preserves spacing before bullet lists', t => {
766
+ const text = `I can assist with tasks such as:
767
+
768
+ - First item
769
+ - Second item
770
+
771
+ Let me know what you'd like to work on.`;
772
+ const result = parseMarkdown(text, mockColors);
773
+
774
+ // Should have the paragraph text
775
+ t.true(result.includes('I can assist with tasks such as:'));
776
+
777
+ // Should have bullets
778
+ t.true(result.includes('•'));
779
+ t.true(result.includes('First item'));
780
+ t.true(result.includes('Second item'));
781
+
782
+ // Should have the closing text
783
+ t.true(result.includes("Let me know what you'd like to work on."));
784
+
785
+ // Should have proper spacing - check for double newline before list
786
+ // (The blank line should be preserved)
787
+ const lines = result.split('\n');
788
+ const suchAsIndex = lines.findIndex(l => l.includes('I can assist'));
789
+ const firstBulletIndex = lines.findIndex(l => l.includes('• First'));
790
+
791
+ // There should be at least one empty line between them
792
+ t.true(
793
+ firstBulletIndex - suchAsIndex >= 2,
794
+ 'Should have blank line before list',
795
+ );
796
+ });