@gguf/coder 0.2.9 → 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 (439) 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/dist/tools/execute-bash.js +3 -3
  10. package/dist/tools/execute-bash.js.map +1 -1
  11. package/dist/tools/fetch-url.js +3 -3
  12. package/dist/tools/fetch-url.js.map +1 -1
  13. package/dist/tools/find-files.d.ts.map +1 -1
  14. package/dist/tools/find-files.js +1 -1
  15. package/dist/tools/find-files.js.map +1 -1
  16. package/dist/tools/lsp-get-diagnostics.js +1 -1
  17. package/dist/tools/lsp-get-diagnostics.js.map +1 -1
  18. package/dist/tools/read-file.d.ts.map +1 -1
  19. package/dist/tools/read-file.js +6 -6
  20. package/dist/tools/read-file.js.map +1 -1
  21. package/dist/tools/search-file-contents.d.ts.map +1 -1
  22. package/dist/tools/search-file-contents.js +1 -1
  23. package/dist/tools/search-file-contents.js.map +1 -1
  24. package/dist/tools/string-replace.js +11 -11
  25. package/dist/tools/string-replace.js.map +1 -1
  26. package/dist/tools/web-search.d.ts.map +1 -1
  27. package/dist/tools/web-search.js +3 -3
  28. package/dist/tools/web-search.js.map +1 -1
  29. package/dist/tools/write-file.js +4 -4
  30. package/dist/tools/write-file.js.map +1 -1
  31. package/dist/utils/tool-result-display.d.ts.map +1 -1
  32. package/dist/utils/tool-result-display.js +3 -3
  33. package/dist/utils/tool-result-display.js.map +1 -1
  34. package/package.json +2 -14
  35. package/scripts/extract-changelog.js +73 -0
  36. package/scripts/fetch-models.js +143 -0
  37. package/scripts/test.sh +40 -0
  38. package/scripts/update-homebrew-formula.sh +125 -0
  39. package/scripts/update-nix-version.sh +157 -0
  40. package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
  41. package/source/ai-sdk-client/AISDKClient.ts +155 -0
  42. package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
  43. package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
  44. package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
  45. package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
  46. package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
  47. package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
  48. package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
  49. package/source/ai-sdk-client/converters/message-converter.ts +113 -0
  50. package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
  51. package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
  52. package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
  53. package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
  54. package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
  55. package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
  56. package/source/ai-sdk-client/index.ts +7 -0
  57. package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
  58. package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
  59. package/source/ai-sdk-client/types.ts +9 -0
  60. package/source/ai-sdk-client-empty-message.spec.ts +141 -0
  61. package/source/ai-sdk-client-error-handling.spec.ts +186 -0
  62. package/source/ai-sdk-client-maxretries.spec.ts +114 -0
  63. package/source/ai-sdk-client-preparestep.spec.ts +279 -0
  64. package/source/app/App.spec.tsx +32 -0
  65. package/source/app/App.tsx +480 -0
  66. package/source/app/components/AppContainer.spec.tsx +96 -0
  67. package/source/app/components/AppContainer.tsx +56 -0
  68. package/source/app/components/ChatInterface.spec.tsx +163 -0
  69. package/source/app/components/ChatInterface.tsx +144 -0
  70. package/source/app/components/ModalSelectors.spec.tsx +141 -0
  71. package/source/app/components/ModalSelectors.tsx +135 -0
  72. package/source/app/helpers.spec.ts +97 -0
  73. package/source/app/helpers.ts +63 -0
  74. package/source/app/index.ts +4 -0
  75. package/source/app/types.ts +39 -0
  76. package/source/app/utils/appUtils.ts +294 -0
  77. package/source/app/utils/conversationState.ts +310 -0
  78. package/source/app.spec.tsx +244 -0
  79. package/source/cli.spec.ts +73 -0
  80. package/source/cli.tsx +51 -0
  81. package/source/client-factory.spec.ts +48 -0
  82. package/source/client-factory.ts +178 -0
  83. package/source/command-parser.spec.ts +127 -0
  84. package/source/command-parser.ts +36 -0
  85. package/source/commands/checkpoint.spec.tsx +277 -0
  86. package/source/commands/checkpoint.tsx +366 -0
  87. package/source/commands/clear.tsx +22 -0
  88. package/source/commands/custom-commands.tsx +121 -0
  89. package/source/commands/exit.ts +21 -0
  90. package/source/commands/export.spec.tsx +131 -0
  91. package/source/commands/export.tsx +79 -0
  92. package/source/commands/help.tsx +120 -0
  93. package/source/commands/index.ts +17 -0
  94. package/source/commands/init.tsx +339 -0
  95. package/source/commands/lsp-command.spec.tsx +281 -0
  96. package/source/commands/lsp.tsx +120 -0
  97. package/source/commands/mcp-command.spec.tsx +313 -0
  98. package/source/commands/mcp.tsx +162 -0
  99. package/source/commands/model-database.spec.tsx +758 -0
  100. package/source/commands/model-database.tsx +418 -0
  101. package/source/commands/model.ts +12 -0
  102. package/source/commands/provider.ts +12 -0
  103. package/source/commands/setup-config.tsx +16 -0
  104. package/source/commands/simple-commands.spec.tsx +175 -0
  105. package/source/commands/status.ts +12 -0
  106. package/source/commands/theme.ts +12 -0
  107. package/source/commands/update.spec.tsx +261 -0
  108. package/source/commands/update.tsx +201 -0
  109. package/source/commands/usage.spec.tsx +495 -0
  110. package/source/commands/usage.tsx +100 -0
  111. package/source/commands.spec.ts +436 -0
  112. package/source/commands.ts +83 -0
  113. package/source/components/assistant-message.spec.tsx +796 -0
  114. package/source/components/assistant-message.tsx +34 -0
  115. package/source/components/bash-execution-indicator.tsx +21 -0
  116. package/source/components/cancelling-indicator.tsx +16 -0
  117. package/source/components/chat-queue.spec.tsx +83 -0
  118. package/source/components/chat-queue.tsx +36 -0
  119. package/source/components/checkpoint-display.spec.tsx +219 -0
  120. package/source/components/checkpoint-display.tsx +126 -0
  121. package/source/components/checkpoint-selector.spec.tsx +173 -0
  122. package/source/components/checkpoint-selector.tsx +173 -0
  123. package/source/components/development-mode-indicator.spec.tsx +268 -0
  124. package/source/components/development-mode-indicator.tsx +38 -0
  125. package/source/components/message-box.spec.tsx +427 -0
  126. package/source/components/message-box.tsx +87 -0
  127. package/source/components/model-selector.tsx +132 -0
  128. package/source/components/provider-selector.tsx +75 -0
  129. package/source/components/random-spinner.tsx +19 -0
  130. package/source/components/security-disclaimer.tsx +73 -0
  131. package/source/components/status-connection-display.spec.tsx +133 -0
  132. package/source/components/status.tsx +267 -0
  133. package/source/components/theme-selector.tsx +126 -0
  134. package/source/components/tool-confirmation.tsx +190 -0
  135. package/source/components/tool-execution-indicator.tsx +33 -0
  136. package/source/components/tool-message.tsx +85 -0
  137. package/source/components/ui/titled-box.spec.tsx +207 -0
  138. package/source/components/ui/titled-box.tsx +57 -0
  139. package/source/components/usage/progress-bar.spec.tsx +398 -0
  140. package/source/components/usage/progress-bar.tsx +30 -0
  141. package/source/components/usage/usage-display.spec.tsx +780 -0
  142. package/source/components/usage/usage-display.tsx +291 -0
  143. package/source/components/user-input.spec.tsx +327 -0
  144. package/source/components/user-input.tsx +533 -0
  145. package/source/components/user-message.spec.tsx +230 -0
  146. package/source/components/user-message.tsx +84 -0
  147. package/source/components/welcome-message.tsx +76 -0
  148. package/source/config/env-substitution.ts +65 -0
  149. package/source/config/index.spec.ts +171 -0
  150. package/source/config/index.ts +154 -0
  151. package/source/config/paths.spec.ts +241 -0
  152. package/source/config/paths.ts +55 -0
  153. package/source/config/preferences.ts +51 -0
  154. package/source/config/themes.ts +315 -0
  155. package/source/constants.ts +130 -0
  156. package/source/context/mode-context.spec.ts +79 -0
  157. package/source/context/mode-context.ts +24 -0
  158. package/source/custom-commands/executor.spec.ts +142 -0
  159. package/source/custom-commands/executor.ts +64 -0
  160. package/source/custom-commands/loader.spec.ts +314 -0
  161. package/source/custom-commands/loader.ts +153 -0
  162. package/source/custom-commands/parser.ts +196 -0
  163. package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
  164. package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
  165. package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
  166. package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
  167. package/source/hooks/chat-handler/index.ts +12 -0
  168. package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
  169. package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
  170. package/source/hooks/chat-handler/types.ts +38 -0
  171. package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
  172. package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
  173. package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
  174. package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
  175. package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
  176. package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
  177. package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
  178. package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
  179. package/source/hooks/useAppHandlers.tsx +291 -0
  180. package/source/hooks/useAppInitialization.tsx +422 -0
  181. package/source/hooks/useAppState.tsx +311 -0
  182. package/source/hooks/useDirectoryTrust.tsx +98 -0
  183. package/source/hooks/useInputState.ts +414 -0
  184. package/source/hooks/useModeHandlers.tsx +302 -0
  185. package/source/hooks/useNonInteractiveMode.ts +140 -0
  186. package/source/hooks/useTerminalWidth.tsx +81 -0
  187. package/source/hooks/useTheme.ts +18 -0
  188. package/source/hooks/useToolHandler.tsx +349 -0
  189. package/source/hooks/useUIState.ts +61 -0
  190. package/source/init/agents-template-generator.ts +421 -0
  191. package/source/init/existing-rules-extractor.ts +319 -0
  192. package/source/init/file-scanner.spec.ts +227 -0
  193. package/source/init/file-scanner.ts +238 -0
  194. package/source/init/framework-detector.ts +382 -0
  195. package/source/init/language-detector.ts +269 -0
  196. package/source/init/project-analyzer.spec.ts +231 -0
  197. package/source/init/project-analyzer.ts +458 -0
  198. package/source/lsp/index.ts +31 -0
  199. package/source/lsp/lsp-client.spec.ts +508 -0
  200. package/source/lsp/lsp-client.ts +487 -0
  201. package/source/lsp/lsp-manager.spec.ts +477 -0
  202. package/source/lsp/lsp-manager.ts +419 -0
  203. package/source/lsp/protocol.spec.ts +502 -0
  204. package/source/lsp/protocol.ts +360 -0
  205. package/source/lsp/server-discovery.spec.ts +654 -0
  206. package/source/lsp/server-discovery.ts +515 -0
  207. package/source/markdown-parser/html-entities.spec.ts +88 -0
  208. package/source/markdown-parser/html-entities.ts +45 -0
  209. package/source/markdown-parser/index.spec.ts +281 -0
  210. package/source/markdown-parser/index.ts +126 -0
  211. package/source/markdown-parser/table-parser.spec.ts +133 -0
  212. package/source/markdown-parser/table-parser.ts +114 -0
  213. package/source/markdown-parser/utils.spec.ts +70 -0
  214. package/source/markdown-parser/utils.ts +13 -0
  215. package/source/mcp/mcp-client.spec.ts +81 -0
  216. package/source/mcp/mcp-client.ts +625 -0
  217. package/source/mcp/transport-factory.spec.ts +406 -0
  218. package/source/mcp/transport-factory.ts +312 -0
  219. package/source/message-handler.ts +67 -0
  220. package/source/model-database/database-engine.spec.ts +494 -0
  221. package/source/model-database/database-engine.ts +50 -0
  222. package/source/model-database/model-database.spec.ts +363 -0
  223. package/source/model-database/model-database.ts +91 -0
  224. package/source/model-database/model-engine.spec.ts +447 -0
  225. package/source/model-database/model-engine.ts +65 -0
  226. package/source/model-database/model-fetcher.spec.ts +583 -0
  227. package/source/model-database/model-fetcher.ts +330 -0
  228. package/source/models/index.ts +1 -0
  229. package/source/models/models-cache.spec.ts +214 -0
  230. package/source/models/models-cache.ts +78 -0
  231. package/source/models/models-dev-client.spec.ts +379 -0
  232. package/source/models/models-dev-client.ts +329 -0
  233. package/source/models/models-types.ts +68 -0
  234. package/source/prompt-history.ts +155 -0
  235. package/source/security/command-injection.spec.ts +240 -0
  236. package/source/services/checkpoint-manager.spec.ts +523 -0
  237. package/source/services/checkpoint-manager.ts +466 -0
  238. package/source/services/file-snapshot.spec.ts +569 -0
  239. package/source/services/file-snapshot.ts +220 -0
  240. package/source/test-utils/render-with-theme.tsx +48 -0
  241. package/source/tokenization/index.ts +1 -0
  242. package/source/tokenization/tokenizer-factory.spec.ts +170 -0
  243. package/source/tokenization/tokenizer-factory.ts +125 -0
  244. package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
  245. package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
  246. package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
  247. package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
  248. package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
  249. package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
  250. package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
  251. package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
  252. package/source/tool-calling/index.ts +5 -0
  253. package/source/tool-calling/json-parser.spec.ts +639 -0
  254. package/source/tool-calling/json-parser.ts +247 -0
  255. package/source/tool-calling/tool-parser.spec.ts +395 -0
  256. package/source/tool-calling/tool-parser.ts +120 -0
  257. package/source/tool-calling/xml-parser.spec.ts +662 -0
  258. package/source/tool-calling/xml-parser.ts +289 -0
  259. package/source/tools/execute-bash.spec.tsx +353 -0
  260. package/source/tools/execute-bash.tsx +219 -0
  261. package/source/tools/execute-function.spec.ts +130 -0
  262. package/source/tools/fetch-url.spec.tsx +342 -0
  263. package/source/tools/fetch-url.tsx +172 -0
  264. package/source/tools/find-files.spec.tsx +924 -0
  265. package/source/tools/find-files.tsx +293 -0
  266. package/source/tools/index.ts +102 -0
  267. package/source/tools/lsp-get-diagnostics.tsx +192 -0
  268. package/source/tools/needs-approval.spec.ts +282 -0
  269. package/source/tools/read-file.spec.tsx +801 -0
  270. package/source/tools/read-file.tsx +387 -0
  271. package/source/tools/search-file-contents.spec.tsx +1273 -0
  272. package/source/tools/search-file-contents.tsx +293 -0
  273. package/source/tools/string-replace.spec.tsx +730 -0
  274. package/source/tools/string-replace.tsx +548 -0
  275. package/source/tools/tool-manager.ts +210 -0
  276. package/source/tools/tool-registry.spec.ts +415 -0
  277. package/source/tools/tool-registry.ts +228 -0
  278. package/source/tools/web-search.tsx +223 -0
  279. package/source/tools/write-file.spec.tsx +559 -0
  280. package/source/tools/write-file.tsx +228 -0
  281. package/source/types/app.ts +37 -0
  282. package/source/types/checkpoint.ts +48 -0
  283. package/source/types/commands.ts +46 -0
  284. package/source/types/components.ts +27 -0
  285. package/source/types/config.ts +103 -0
  286. package/source/types/core-connection-status.spec.ts +67 -0
  287. package/source/types/core.ts +181 -0
  288. package/source/types/hooks.ts +50 -0
  289. package/source/types/index.ts +12 -0
  290. package/source/types/markdown-parser.ts +11 -0
  291. package/source/types/mcp.ts +52 -0
  292. package/source/types/system.ts +16 -0
  293. package/source/types/tokenization.ts +41 -0
  294. package/source/types/ui.ts +40 -0
  295. package/source/types/usage.ts +58 -0
  296. package/source/types/utils.ts +16 -0
  297. package/source/usage/calculator.spec.ts +385 -0
  298. package/source/usage/calculator.ts +104 -0
  299. package/source/usage/storage.spec.ts +703 -0
  300. package/source/usage/storage.ts +238 -0
  301. package/source/usage/tracker.spec.ts +456 -0
  302. package/source/usage/tracker.ts +102 -0
  303. package/source/utils/atomic-deletion.spec.ts +194 -0
  304. package/source/utils/atomic-deletion.ts +127 -0
  305. package/source/utils/bounded-map.spec.ts +300 -0
  306. package/source/utils/bounded-map.ts +193 -0
  307. package/source/utils/checkpoint-utils.spec.ts +222 -0
  308. package/source/utils/checkpoint-utils.ts +92 -0
  309. package/source/utils/error-formatter.spec.ts +169 -0
  310. package/source/utils/error-formatter.ts +194 -0
  311. package/source/utils/file-autocomplete.spec.ts +173 -0
  312. package/source/utils/file-autocomplete.ts +196 -0
  313. package/source/utils/file-cache.spec.ts +309 -0
  314. package/source/utils/file-cache.ts +195 -0
  315. package/source/utils/file-content-loader.spec.ts +180 -0
  316. package/source/utils/file-content-loader.ts +179 -0
  317. package/source/utils/file-mention-handler.spec.ts +261 -0
  318. package/source/utils/file-mention-handler.ts +84 -0
  319. package/source/utils/file-mention-parser.spec.ts +182 -0
  320. package/source/utils/file-mention-parser.ts +170 -0
  321. package/source/utils/fuzzy-matching.spec.ts +149 -0
  322. package/source/utils/fuzzy-matching.ts +146 -0
  323. package/source/utils/indentation-normalizer.spec.ts +216 -0
  324. package/source/utils/indentation-normalizer.ts +76 -0
  325. package/source/utils/installation-detector.spec.ts +178 -0
  326. package/source/utils/installation-detector.ts +153 -0
  327. package/source/utils/logging/config.spec.ts +311 -0
  328. package/source/utils/logging/config.ts +210 -0
  329. package/source/utils/logging/console-facade.spec.ts +184 -0
  330. package/source/utils/logging/console-facade.ts +384 -0
  331. package/source/utils/logging/correlation.spec.ts +679 -0
  332. package/source/utils/logging/correlation.ts +474 -0
  333. package/source/utils/logging/formatters.spec.ts +464 -0
  334. package/source/utils/logging/formatters.ts +207 -0
  335. package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
  336. package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
  337. package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
  338. package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
  339. package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
  340. package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
  341. package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
  342. package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
  343. package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
  344. package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
  345. package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
  346. package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
  347. package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
  348. package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
  349. package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
  350. package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
  351. package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
  352. package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
  353. package/source/utils/logging/health-monitor/index.ts +15 -0
  354. package/source/utils/logging/health-monitor/instances.ts +48 -0
  355. package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
  356. package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
  357. package/source/utils/logging/health-monitor/types.ts +126 -0
  358. package/source/utils/logging/index.spec.ts +284 -0
  359. package/source/utils/logging/index.ts +236 -0
  360. package/source/utils/logging/integration.spec.ts +441 -0
  361. package/source/utils/logging/log-method-factory.spec.ts +573 -0
  362. package/source/utils/logging/log-method-factory.ts +233 -0
  363. package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
  364. package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
  365. package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
  366. package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
  367. package/source/utils/logging/log-query/index.ts +23 -0
  368. package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
  369. package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
  370. package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
  371. package/source/utils/logging/log-query/query/query-builder.ts +151 -0
  372. package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
  373. package/source/utils/logging/log-query/query/query-engine.ts +45 -0
  374. package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
  375. package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
  376. package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
  377. package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
  378. package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
  379. package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
  380. package/source/utils/logging/log-query/types.ts +163 -0
  381. package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
  382. package/source/utils/logging/log-query/utils/helpers.ts +72 -0
  383. package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
  384. package/source/utils/logging/log-query/utils/sorting.ts +61 -0
  385. package/source/utils/logging/logger-provider.spec.ts +262 -0
  386. package/source/utils/logging/logger-provider.ts +362 -0
  387. package/source/utils/logging/performance.spec.ts +209 -0
  388. package/source/utils/logging/performance.ts +757 -0
  389. package/source/utils/logging/pino-logger.spec.ts +425 -0
  390. package/source/utils/logging/pino-logger.ts +514 -0
  391. package/source/utils/logging/redaction.spec.ts +490 -0
  392. package/source/utils/logging/redaction.ts +267 -0
  393. package/source/utils/logging/request-tracker.spec.ts +1198 -0
  394. package/source/utils/logging/request-tracker.ts +803 -0
  395. package/source/utils/logging/transports.spec.ts +505 -0
  396. package/source/utils/logging/transports.ts +305 -0
  397. package/source/utils/logging/types.ts +216 -0
  398. package/source/utils/message-builder.spec.ts +179 -0
  399. package/source/utils/message-builder.ts +101 -0
  400. package/source/utils/message-queue.tsx +486 -0
  401. package/source/utils/paste-detection.spec.ts +69 -0
  402. package/source/utils/paste-detection.ts +124 -0
  403. package/source/utils/paste-roundtrip.spec.ts +442 -0
  404. package/source/utils/paste-utils.spec.ts +128 -0
  405. package/source/utils/paste-utils.ts +52 -0
  406. package/source/utils/programming-language-helper.spec.ts +74 -0
  407. package/source/utils/programming-language-helper.ts +32 -0
  408. package/source/utils/prompt-assembly.spec.ts +221 -0
  409. package/source/utils/prompt-processor.ts +173 -0
  410. package/source/utils/tool-args-parser.spec.ts +136 -0
  411. package/source/utils/tool-args-parser.ts +54 -0
  412. package/source/utils/tool-cancellation.spec.ts +230 -0
  413. package/source/utils/tool-cancellation.ts +28 -0
  414. package/source/utils/tool-result-display.spec.tsx +469 -0
  415. package/source/utils/tool-result-display.tsx +90 -0
  416. package/source/utils/update-checker.spec.ts +383 -0
  417. package/source/utils/update-checker.ts +183 -0
  418. package/source/wizard/config-wizard.spec.tsx +103 -0
  419. package/source/wizard/config-wizard.tsx +382 -0
  420. package/source/wizard/steps/location-step.spec.tsx +186 -0
  421. package/source/wizard/steps/location-step.tsx +147 -0
  422. package/source/wizard/steps/mcp-step.spec.tsx +607 -0
  423. package/source/wizard/steps/mcp-step.tsx +632 -0
  424. package/source/wizard/steps/provider-step.spec.tsx +342 -0
  425. package/source/wizard/steps/provider-step.tsx +957 -0
  426. package/source/wizard/steps/summary-step.spec.tsx +749 -0
  427. package/source/wizard/steps/summary-step.tsx +228 -0
  428. package/source/wizard/templates/mcp-templates.spec.ts +613 -0
  429. package/source/wizard/templates/mcp-templates.ts +570 -0
  430. package/source/wizard/templates/provider-templates.spec.ts +152 -0
  431. package/source/wizard/templates/provider-templates.ts +485 -0
  432. package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
  433. package/source/wizard/utils/fetch-cloud-models.ts +223 -0
  434. package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
  435. package/source/wizard/utils/fetch-local-models.ts +192 -0
  436. package/source/wizard/validation-array.spec.ts +264 -0
  437. package/source/wizard/validation.spec.ts +373 -0
  438. package/source/wizard/validation.ts +232 -0
  439. package/source/app/prompts/main-prompt.md +0 -122
@@ -0,0 +1,281 @@
1
+ import test from 'ava';
2
+ import type {Colors} from '../types/markdown-parser.js';
3
+ import {parseMarkdown} from './index.js';
4
+
5
+ console.log(`\nindex.spec.ts`);
6
+
7
+ const mockColors: Colors = {
8
+ primary: '#3b82f6',
9
+ secondary: '#6b7280',
10
+ success: '#10b981',
11
+ error: '#ef4444',
12
+ warning: '#f59e0b',
13
+ info: '#3b82f6',
14
+ white: '#ffffff',
15
+ tool: '#8b5cf6',
16
+ };
17
+
18
+ // Inline code tests
19
+ test('parseMarkdown handles inline code', t => {
20
+ const text = 'Use `npm install` to install packages';
21
+ const result = parseMarkdown(text, mockColors);
22
+ t.true(result.includes('npm install'));
23
+ t.false(result.includes('`npm install`'));
24
+ });
25
+
26
+ // Bold and italic tests
27
+ test('parseMarkdown handles bold text', t => {
28
+ const text = 'This is **bold** text';
29
+ const result = parseMarkdown(text, mockColors);
30
+ t.true(result.includes('bold'));
31
+ });
32
+
33
+ test('parseMarkdown handles italic text', t => {
34
+ const text = 'This is *italic* text';
35
+ const result = parseMarkdown(text, mockColors);
36
+ t.true(result.includes('italic'));
37
+ });
38
+
39
+ // Underscore preservation tests
40
+ test('parseMarkdown preserves underscores in identifiers', t => {
41
+ const text =
42
+ 'Use `create_file`, `read_file`, or `search_file_contents` functions';
43
+ const result = parseMarkdown(text, mockColors);
44
+ t.true(result.includes('create_file'));
45
+ t.true(result.includes('read_file'));
46
+ t.true(result.includes('search_file_contents'));
47
+ });
48
+
49
+ test('parseMarkdown preserves underscores in regular text', t => {
50
+ const text = 'The variable_name and function_call should remain intact';
51
+ const result = parseMarkdown(text, mockColors);
52
+ t.true(result.includes('variable_name'));
53
+ t.true(result.includes('function_call'));
54
+ });
55
+
56
+ // Heading tests
57
+ test('parseMarkdown handles headings', t => {
58
+ const text = '# Heading 1\n## Heading 2\n### Heading 3';
59
+ const result = parseMarkdown(text, mockColors);
60
+ t.true(result.includes('Heading 1'));
61
+ t.true(result.includes('Heading 2'));
62
+ t.true(result.includes('Heading 3'));
63
+ });
64
+
65
+ // Link tests
66
+ test('parseMarkdown handles links', t => {
67
+ const text = 'Visit [Google](https://google.com) for search';
68
+ const result = parseMarkdown(text, mockColors);
69
+ t.true(result.includes('Google'));
70
+ t.true(result.includes('https://google.com'));
71
+ });
72
+
73
+ // Blockquote tests
74
+ test('parseMarkdown handles blockquotes', t => {
75
+ const text = '> This is a quote\n> Another line';
76
+ const result = parseMarkdown(text, mockColors);
77
+ t.true(result.includes('This is a quote'));
78
+ t.true(result.includes('Another line'));
79
+ });
80
+
81
+ // List tests
82
+ test('parseMarkdown handles unordered lists', t => {
83
+ const text = '- Item 1\n- Item 2\n- Item 3';
84
+ const result = parseMarkdown(text, mockColors);
85
+ t.true(result.includes('•'));
86
+ t.true(result.includes('Item 1'));
87
+ t.true(result.includes('Item 2'));
88
+ });
89
+
90
+ test('parseMarkdown handles ordered lists', t => {
91
+ const text = '1. First\n2. Second\n3. Third';
92
+ const result = parseMarkdown(text, mockColors);
93
+ t.true(result.includes('1.'));
94
+ t.true(result.includes('2.'));
95
+ t.true(result.includes('3.'));
96
+ t.true(result.includes('First'));
97
+ t.true(result.includes('Second'));
98
+ });
99
+
100
+ test('parseMarkdown handles nested/indented bullet lists', t => {
101
+ const text = `- Top level item
102
+ - Nested item 1
103
+ - Nested item 2
104
+ - Double nested
105
+ - Another top level`;
106
+ const result = parseMarkdown(text, mockColors);
107
+ t.true(result.includes(' • Nested item 1'));
108
+ t.true(result.includes(' • Double nested'));
109
+ });
110
+
111
+ test('parseMarkdown handles nested numbered lists', t => {
112
+ const text = `1. First level
113
+ 1. Nested first
114
+ 2. Nested second
115
+ 2. Back to first level`;
116
+ const result = parseMarkdown(text, mockColors);
117
+ t.true(result.includes(' 1. Nested first'));
118
+ t.true(result.includes(' 2. Nested second'));
119
+ });
120
+
121
+ // HTML entity tests
122
+ test('parseMarkdown decodes HTML entities', t => {
123
+ const text = '<div> & ©';
124
+ const result = parseMarkdown(text, mockColors);
125
+ t.true(result.includes('<div>'));
126
+ t.true(result.includes('&'));
127
+ t.true(result.includes('©'));
128
+ });
129
+
130
+ // Table tests
131
+ test('parseMarkdown handles tables', t => {
132
+ const text = `| Name | Age |
133
+ |------|-----|
134
+ | John | 25 |`;
135
+ const result = parseMarkdown(text, mockColors);
136
+ t.true(result.includes('Name'));
137
+ t.true(result.includes('John'));
138
+ });
139
+
140
+ test('parseMarkdown renders tables with plain text (no markdown)', t => {
141
+ const text = `| Command | Description |
142
+ |---------|-------------|
143
+ | \`npm install\` | Install dependencies |
144
+ | \`npm start\` | Start the application |`;
145
+ const result = parseMarkdown(text, mockColors);
146
+ t.true(result.includes('npm install'));
147
+ t.true(result.includes('npm start'));
148
+ t.false(result.includes('`npm install`'));
149
+ });
150
+
151
+ // Code block tests
152
+ test('parseMarkdown handles code blocks with language', t => {
153
+ const text = '```javascript\nconst x = 5;\n```';
154
+ const result = parseMarkdown(text, mockColors);
155
+ t.true(result.includes('const x = 5'));
156
+ });
157
+
158
+ test('parseMarkdown handles code blocks without language', t => {
159
+ const text = '```\nPlain code\n```';
160
+ const result = parseMarkdown(text, mockColors);
161
+ t.true(result.includes('Plain code'));
162
+ });
163
+
164
+ // Edge case tests
165
+ test('parseMarkdown does not create bullet list from hyphen in middle of line', t => {
166
+ const text = 'This is not - a list';
167
+ const result = parseMarkdown(text, mockColors);
168
+ t.false(result.includes('•'));
169
+ t.true(result.includes('This is not - a list'));
170
+ });
171
+
172
+ test('parseMarkdown does not format asterisks in math expressions', t => {
173
+ const text = 'Calculate 5 * 3 * 2 = 30';
174
+ const result = parseMarkdown(text, mockColors);
175
+ t.true(result.includes('5 * 3 * 2'));
176
+ });
177
+
178
+ test('parseMarkdown does not format asterisks in file globs', t => {
179
+ const text = 'Match files with *.js or *.tsx patterns';
180
+ const result = parseMarkdown(text, mockColors);
181
+ t.true(result.includes('*.js'));
182
+ t.true(result.includes('*.tsx'));
183
+ });
184
+
185
+ test('parseMarkdown does not create heading from hash in middle of line', t => {
186
+ const text = 'This # is not a heading';
187
+ const result = parseMarkdown(text, mockColors);
188
+ t.true(result.includes('This # is not a heading'));
189
+ });
190
+
191
+ test('parseMarkdown preserves asterisks inside code blocks', t => {
192
+ const text = '```javascript\nconst pattern = /\\*\\*/g;\n```';
193
+ const result = parseMarkdown(text, mockColors);
194
+ t.true(result.includes('/\\*\\*/g') || result.includes('**'));
195
+ });
196
+
197
+ test('parseMarkdown preserves markdown-like syntax inside inline code', t => {
198
+ const text = 'Use `**bold**` and `*italic*` syntax';
199
+ const result = parseMarkdown(text, mockColors);
200
+ // Code content should be preserved (though formatting may be applied by chalk)
201
+ t.false(result.includes('`**bold**`'));
202
+ t.false(result.includes('`*italic*`'));
203
+ });
204
+
205
+ test('parseMarkdown does not format single asterisk surrounded by word chars', t => {
206
+ const text = 'The pointer syntax is char*ptr or int*value';
207
+ const result = parseMarkdown(text, mockColors);
208
+ t.true(result.includes('char*ptr'));
209
+ t.true(result.includes('int*value'));
210
+ });
211
+
212
+ // BR tag test
213
+ test('parseMarkdown converts <br> tags to newlines', t => {
214
+ const text = 'Line one<br>Line two<br/>Line three<BR>Line four';
215
+ const result = parseMarkdown(text, mockColors);
216
+ const lines = result.split('\n');
217
+ t.true(lines.length >= 4);
218
+ t.true(result.includes('Line one'));
219
+ t.true(result.includes('Line two'));
220
+ t.false(result.includes('<br'));
221
+ });
222
+
223
+ // Spacing tests
224
+ test('parseMarkdown preserves spacing before bullet lists', t => {
225
+ const text = `I can assist with tasks such as:
226
+
227
+ - First item
228
+ - Second item
229
+
230
+ Let me know what you'd like to work on.`;
231
+ const result = parseMarkdown(text, mockColors);
232
+ const lines = result.split('\n');
233
+ const suchAsIndex = lines.findIndex(l => l.includes('I can assist'));
234
+ const firstBulletIndex = lines.findIndex(l => l.includes('• First'));
235
+ t.true(
236
+ firstBulletIndex - suchAsIndex >= 2,
237
+ 'Should have blank line before list',
238
+ );
239
+ });
240
+
241
+ // Mixed content tests
242
+ test('parseMarkdown handles mixed markdown features', t => {
243
+ const text = `# Title
244
+
245
+ This is **bold** and *italic* text with \`code\`.
246
+
247
+ - List item 1
248
+ - List item 2
249
+
250
+ | Name | Value |
251
+ |------|-------|
252
+ | Test | 123 |`;
253
+ const result = parseMarkdown(text, mockColors);
254
+ t.true(result.includes('Title'));
255
+ t.true(result.includes('bold'));
256
+ t.true(result.includes('italic'));
257
+ t.true(result.includes('•'));
258
+ t.true(result.includes('Name'));
259
+ });
260
+
261
+ test('parseMarkdown handles plain text without markdown', t => {
262
+ const text = 'This is just plain text without any formatting';
263
+ const result = parseMarkdown(text, mockColors);
264
+ t.true(result.includes('This is just plain text'));
265
+ });
266
+
267
+ test('parseMarkdown handles empty string', t => {
268
+ const result = parseMarkdown('', mockColors);
269
+ t.is(result, '');
270
+ });
271
+
272
+ // Placeholder restoration test
273
+ test('parseMarkdown restores inline code placeholders correctly', t => {
274
+ const text = 'Use `npm install` and `npm start` commands';
275
+ const result = parseMarkdown(text, mockColors);
276
+ t.true(result.includes('npm install'));
277
+ t.true(result.includes('npm start'));
278
+ t.false(result.includes('__INLINE_CODE'));
279
+ t.false(result.includes('_INLINE'));
280
+ t.false(result.includes('CODE_'));
281
+ });
@@ -0,0 +1,126 @@
1
+ import chalk from 'chalk';
2
+ import {highlight} from 'cli-highlight';
3
+ import type {Colors} from '../types/markdown-parser.js';
4
+ import {decodeHtmlEntities} from './html-entities.js';
5
+ import {parseMarkdownTable} from './table-parser.js';
6
+
7
+ // Basic markdown parser for terminal
8
+ export function parseMarkdown(text: string, themeColors: Colors): string {
9
+ // First decode HTML entities
10
+ let result = decodeHtmlEntities(text);
11
+
12
+ // Step 1: Parse tables FIRST (before <br> conversion and code extraction)
13
+ // Tables should have plain text only, no markdown formatting
14
+ // Note: We parse tables before converting <br> tags so that multi-line
15
+ // cells don't break the table regex
16
+ result = result.replace(
17
+ /(?:^|\n)((?:\|.+\|\n)+)/gm,
18
+ (_match, tableText: string) => {
19
+ return '\n' + parseMarkdownTable(tableText, themeColors) + '\n';
20
+ },
21
+ );
22
+
23
+ // Step 2: Convert <br> and <br/> tags to newlines (AFTER table parsing)
24
+ result = result.replace(/<br\s*\/?>/gi, '\n');
25
+
26
+ // Step 3: Extract and protect code blocks and inline code with placeholders
27
+ const codeBlocks: string[] = [];
28
+ const inlineCodes: string[] = [];
29
+
30
+ // Extract code blocks first (```language\ncode\n```)
31
+ result = result.replace(
32
+ /```([a-zA-Z0-9\-+#]+)?\n([\s\S]*?)```/g,
33
+ (_match, lang: string | undefined, code: string) => {
34
+ try {
35
+ const codeStr = String(code).trim();
36
+ // Apply syntax highlighting with detected language
37
+ const highlighted = highlight(codeStr, {
38
+ language: lang || 'plaintext',
39
+ theme: 'default',
40
+ });
41
+ const placeholder = `__CODE_BLOCK_${codeBlocks.length}__`;
42
+ codeBlocks.push(highlighted);
43
+ return placeholder;
44
+ } catch {
45
+ // Fallback to plain colored text if highlighting fails
46
+ const formatted = chalk.hex(themeColors.tool)(String(code).trim());
47
+ const placeholder = `__CODE_BLOCK_${codeBlocks.length}__`;
48
+ codeBlocks.push(formatted);
49
+ return placeholder;
50
+ }
51
+ },
52
+ );
53
+
54
+ // Extract inline code (`code`)
55
+ result = result.replace(/`([^`]+)`/g, (_match, code: string) => {
56
+ const formatted = chalk.hex(themeColors.tool)(String(code).trim());
57
+ const placeholder = `__INLINE_CODE_${inlineCodes.length}__`;
58
+ inlineCodes.push(formatted);
59
+ return placeholder;
60
+ });
61
+
62
+ // Step 4: Process markdown formatting (now safe from code interference)
63
+ // Process lists FIRST before italic, since * at start of line is a list, not italic
64
+ // List items (- item or * item or 1. item)
65
+ // Use [ \t]* instead of \s* to avoid consuming newlines before the list
66
+ // Preserve indentation for nested lists
67
+ result = result.replace(/^([ \t]*)[-*]\s+(.+)$/gm, (_match, indent, text) => {
68
+ return indent + chalk.hex(themeColors.white)(`• ${text}`);
69
+ });
70
+ result = result.replace(
71
+ /^([ \t]*)(\d+)\.\s+(.+)$/gm,
72
+ (_match, indent, num, text) => {
73
+ return indent + chalk.hex(themeColors.white)(`${num}. ${text}`);
74
+ },
75
+ );
76
+
77
+ // Bold (**text** only - avoid __ to prevent conflicts with snake_case)
78
+ result = result.replace(/\*\*([^*]+)\*\*/g, (_match, text) => {
79
+ return chalk.hex(themeColors.white).bold(text);
80
+ });
81
+
82
+ // Italic (*text* only - avoid _ to prevent conflicts with snake_case)
83
+ // Require whitespace or line boundaries around asterisks to avoid matching char*ptr
84
+ // Use [^*\n] to prevent matching across lines
85
+ // Only match if content contains at least one letter to avoid matching math like "5 * 3 * 2"
86
+ result = result.replace(
87
+ /(^|\s)\*([^*\n]*[a-zA-Z][^*\n]*)\*($|\s)/gm,
88
+ (_match, before, text, after) => {
89
+ return before + chalk.hex(themeColors.white).italic(text) + after;
90
+ },
91
+ );
92
+
93
+ // Headings (# Heading)
94
+ result = result.replace(/^(#{1,6})\s+(.+)$/gm, (_match, _hashes, text) => {
95
+ return chalk.hex(themeColors.primary).bold(text);
96
+ });
97
+
98
+ // Links [text](url)
99
+ result = result.replace(/\[([^\]]+)\]\(([^)]+)\)/g, (_match, text, url) => {
100
+ return (
101
+ chalk.hex(themeColors.info).underline(text) +
102
+ ' ' +
103
+ chalk.hex(themeColors.secondary)(`(${url})`)
104
+ );
105
+ });
106
+
107
+ // Blockquotes (> text)
108
+ result = result.replace(/^>\s+(.+)$/gm, (_match, text) => {
109
+ return chalk.hex(themeColors.secondary).italic(`> ${text}`);
110
+ });
111
+
112
+ // Step 5: Restore code blocks and inline code from placeholders
113
+ result = result.replace(/__CODE_BLOCK_(\d+)__/g, (_match, index: string) => {
114
+ return codeBlocks[parseInt(index, 10)] || '';
115
+ });
116
+ result = result.replace(/__INLINE_CODE_(\d+)__/g, (_match, index: string) => {
117
+ return inlineCodes[parseInt(index, 10)] || '';
118
+ });
119
+
120
+ return result;
121
+ }
122
+
123
+ // Re-export utilities for convenience
124
+ export {decodeHtmlEntities} from './html-entities.js';
125
+ export {parseMarkdownTable} from './table-parser.js';
126
+ export type {Colors} from '../types/markdown-parser.js';
@@ -0,0 +1,133 @@
1
+ import test from 'ava';
2
+ import type {Colors} from '../types/markdown-parser';
3
+ import {parseMarkdownTable} from './table-parser';
4
+
5
+ console.log(`\ntable-parser.spec.ts`);
6
+
7
+ const mockColors: Colors = {
8
+ primary: '#3b82f6',
9
+ secondary: '#6b7280',
10
+ success: '#10b981',
11
+ error: '#ef4444',
12
+ warning: '#f59e0b',
13
+ info: '#3b82f6',
14
+ white: '#ffffff',
15
+ tool: '#8b5cf6',
16
+ };
17
+
18
+ test('parseMarkdownTable handles simple two-column table', t => {
19
+ const table = `| Name | Age |
20
+ |------|-----|
21
+ | John | 25 |
22
+ | Jane | 30 |`;
23
+ const result = parseMarkdownTable(table, mockColors);
24
+ t.true(result.includes('Name'));
25
+ t.true(result.includes('Age'));
26
+ t.true(result.includes('John'));
27
+ t.true(result.includes('Jane'));
28
+ t.true(result.includes('25'));
29
+ t.true(result.includes('30'));
30
+ });
31
+
32
+ test('parseMarkdownTable handles table with varying cell lengths', t => {
33
+ const table = `| Short | Very Long Column Name |
34
+ |-------|----------------------|
35
+ | A | This is a much longer text |
36
+ | B | Short |`;
37
+ const result = parseMarkdownTable(table, mockColors);
38
+ t.true(result.includes('Short'));
39
+ t.true(result.includes('Very Long Column Name'));
40
+ t.true(result.includes('This is a much longer text'));
41
+ });
42
+
43
+ test('parseMarkdownTable returns original text for invalid table', t => {
44
+ const table = `Not a table at all`;
45
+ const result = parseMarkdownTable(table, mockColors);
46
+ t.is(result, table);
47
+ });
48
+
49
+ test('parseMarkdownTable returns original text for table without data rows', t => {
50
+ const table = `| Name | Age |
51
+ |------|-----|`;
52
+ const result = parseMarkdownTable(table, mockColors);
53
+ t.is(result, table);
54
+ });
55
+
56
+ test('parseMarkdownTable handles table with empty cells', t => {
57
+ const table = `| Name | Age |
58
+ |------|-----|
59
+ | John | |
60
+ | | 30 |`;
61
+ const result = parseMarkdownTable(table, mockColors);
62
+ t.true(result.includes('John'));
63
+ t.true(result.includes('30'));
64
+ });
65
+
66
+ test('parseMarkdownTable normalizes column count', t => {
67
+ const table = `| A | B | C |
68
+ |---|---|---|
69
+ | 1 | 2 |
70
+ | 3 | 4 | 5 | 6 |`;
71
+ const result = parseMarkdownTable(table, mockColors);
72
+ // Should handle mismatched columns gracefully
73
+ t.true(typeof result === 'string');
74
+ t.true(result.length > 0);
75
+ });
76
+
77
+ test('parseMarkdownTable strips markdown from cells', t => {
78
+ const table = `| Command | Description |
79
+ |---------|-------------|
80
+ | \`npm install\` | Install **dependencies** |
81
+ | \`npm start\` | Start the *application* |`;
82
+ const result = parseMarkdownTable(table, mockColors);
83
+ // Should have plain text (backticks and formatting removed)
84
+ t.true(result.includes('npm install'));
85
+ t.true(result.includes('npm start'));
86
+ t.true(result.includes('dependencies'));
87
+ t.true(result.includes('application'));
88
+ // Should NOT have markdown syntax
89
+ t.false(result.includes('`npm install`'));
90
+ t.false(result.includes('**dependencies**'));
91
+ t.false(result.includes('*application*'));
92
+ });
93
+
94
+ test('parseMarkdownTable strips HTML from cells', t => {
95
+ const table = `| Name | Description |
96
+ |------|-------------|
97
+ | Test | Text with <br> tag |
98
+ | Another | Text with <b>bold</b> |`;
99
+ const result = parseMarkdownTable(table, mockColors);
100
+ t.true(result.includes('Text with tag')); // HTML tag removed
101
+ t.true(result.includes('Text with bold')); // HTML tags removed
102
+ t.false(result.includes('<br>'));
103
+ t.false(result.includes('<b>'));
104
+ });
105
+
106
+ test('parseMarkdownTable handles links in cells', t => {
107
+ const table = `| Name | Link |
108
+ |------|------|
109
+ | Google | [Visit](https://google.com) |`;
110
+ const result = parseMarkdownTable(table, mockColors);
111
+ t.true(result.includes('Visit'));
112
+ t.false(result.includes('[Visit]'));
113
+ t.false(result.includes('(https://google.com)'));
114
+ });
115
+
116
+ test('parseMarkdownTable handles alignment markers', t => {
117
+ const table = `| Left | Center | Right |
118
+ |:-----|:------:|------:|
119
+ | A | B | C |`;
120
+ const result = parseMarkdownTable(table, mockColors);
121
+ t.true(result.includes('Left'));
122
+ t.true(result.includes('Center'));
123
+ t.true(result.includes('Right'));
124
+ });
125
+
126
+ test('parseMarkdownTable creates table with borders', t => {
127
+ const table = `| A | B |
128
+ |---|---|
129
+ | 1 | 2 |`;
130
+ const result = parseMarkdownTable(table, mockColors);
131
+ // Should contain table border characters
132
+ t.true(result.includes('─') || result.includes('│'));
133
+ });
@@ -0,0 +1,114 @@
1
+ import {DEFAULT_TERMINAL_WIDTH, TABLE_COLUMN_MIN_WIDTH} from '@/constants';
2
+ import chalk from 'chalk';
3
+ import Table from 'cli-table3';
4
+ import type {Colors} from '../types/markdown-parser';
5
+ import {stripMarkdown} from './utils';
6
+
7
+ // Parse markdown tables using cli-table3
8
+ export function parseMarkdownTable(
9
+ tableText: string,
10
+ themeColors: Colors,
11
+ ): string {
12
+ const lines = tableText.trim().split('\n');
13
+ if (lines.length < 2) return tableText;
14
+
15
+ // Extract rows
16
+ const rows = lines.map(line =>
17
+ line
18
+ .split('|')
19
+ .map(cell => cell.trim())
20
+ .filter(cell => cell.length > 0),
21
+ );
22
+
23
+ // Check if second row is separator (e.g., |---|---|)
24
+ const separatorRow = rows[1];
25
+ const isSeparator = separatorRow?.every(cell => /^:?-+:?$/.test(cell));
26
+ if (!isSeparator || rows.length < 3) return tableText;
27
+
28
+ // Helper to clean cell content - remove all markdown and HTML
29
+ const cleanCell = (text: string): string => {
30
+ let result = text;
31
+ // Remove inline code backticks (extract the content)
32
+ result = result.replace(/`([^`]+)`/g, '$1');
33
+ // Remove HTML tags
34
+ result = result.replace(/<[^>]+>/g, '');
35
+ // Strip markdown formatting (bold, italic, links)
36
+ result = stripMarkdown(result);
37
+ return result.trim();
38
+ };
39
+
40
+ // Extract header and data rows - clean all content
41
+ const header = rows[0].map(cell => cleanCell(cell));
42
+ const dataRows = rows.slice(2).map(row => row.map(cell => cleanCell(cell)));
43
+
44
+ // Calculate widths from cleaned content
45
+ const headerWidths = header.map(cell => cell.length);
46
+ const dataWidths = dataRows.map(row => row.map(cell => cell.length));
47
+
48
+ // Calculate column widths properly
49
+ const terminalWidth = process.stdout.columns || DEFAULT_TERMINAL_WIDTH;
50
+ const numCols = header.length;
51
+
52
+ // Get max content width for each column
53
+ const contentWidths = headerWidths.map((headerWidth, colIdx) => {
54
+ let maxWidth = headerWidth;
55
+ for (const rowWidths of dataWidths) {
56
+ if (rowWidths[colIdx]) {
57
+ maxWidth = Math.max(maxWidth, rowWidths[colIdx]);
58
+ }
59
+ }
60
+ return maxWidth;
61
+ });
62
+
63
+ // Calculate available width (accounting for borders and padding)
64
+ const borderWidth = numCols + 1; // vertical bars
65
+ const paddingWidth = numCols * 2; // 1 space on each side of each column
66
+ const availableWidth = terminalWidth - borderWidth - paddingWidth;
67
+
68
+ // Distribute width proportionally
69
+ const totalContentWidth = contentWidths.reduce((a, b) => a + b, 0);
70
+ const colWidths = contentWidths.map(width =>
71
+ Math.max(
72
+ TABLE_COLUMN_MIN_WIDTH,
73
+ Math.floor((width / totalContentWidth) * availableWidth),
74
+ ),
75
+ );
76
+
77
+ // Create table with cli-table3 - full borders, proper alignment
78
+ const table = new Table({
79
+ head: header.map(cell => chalk.hex(themeColors.primary).bold(cell)),
80
+ colWidths: colWidths,
81
+ style: {
82
+ head: [], // Don't apply default styles, we're using chalk
83
+ border: ['gray'], // Subtle border color
84
+ 'padding-left': 1,
85
+ 'padding-right': 1,
86
+ },
87
+ chars: {
88
+ top: '─',
89
+ 'top-mid': '┬',
90
+ 'top-left': '┌',
91
+ 'top-right': '┐',
92
+ bottom: '─',
93
+ 'bottom-mid': '┴',
94
+ 'bottom-left': '└',
95
+ 'bottom-right': '┘',
96
+ left: '│',
97
+ 'left-mid': '├',
98
+ mid: '─',
99
+ 'mid-mid': '┼',
100
+ right: '│',
101
+ 'right-mid': '┤',
102
+ middle: '│',
103
+ },
104
+ wordWrap: true,
105
+ wrapOnWordBoundary: true,
106
+ });
107
+
108
+ // Add data rows - don't style them, let cli-table3 handle layout
109
+ for (const row of dataRows) {
110
+ table.push(row);
111
+ }
112
+
113
+ return table.toString();
114
+ }