@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,383 @@
1
+ import { readFileSync } from 'fs';
2
+ import { dirname, join } from 'path';
3
+ import { fileURLToPath } from 'url';
4
+ import test from 'ava';
5
+ import { checkForUpdates } from './update-checker';
6
+
7
+ console.log(`\nupdate-checker.spec.ts`);
8
+
9
+ // Get current version from package.json dynamically
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dirname = dirname(__filename);
12
+ const packageJsonPath = join(__dirname, '../../package.json');
13
+ const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
14
+ const CURRENT_VERSION = packageJson.version as string;
15
+
16
+ // Mock fetch globally for testing
17
+ const originalFetch = globalThis.fetch;
18
+
19
+ // Helper to create mock fetch responses
20
+ function createMockFetch(
21
+ status: number,
22
+ data: unknown,
23
+ shouldReject = false,
24
+ ): typeof fetch {
25
+ return (async () => {
26
+ if (shouldReject) {
27
+ throw new Error('Network error');
28
+ }
29
+
30
+ return {
31
+ ok: status >= 200 && status < 300,
32
+ status,
33
+ statusText: status === 200 ? 'OK' : 'Error',
34
+ json: async () => data,
35
+ } as Response;
36
+ }) as typeof fetch;
37
+ }
38
+
39
+ test.beforeEach(() => {
40
+ // Reset fetch before each test
41
+ globalThis.fetch = originalFetch;
42
+ // Default to npm install override
43
+ process.env.CODER_INSTALL_METHOD = 'npm';
44
+ });
45
+
46
+ test.afterEach(() => {
47
+ // Restore original fetch and env after each test
48
+ globalThis.fetch = originalFetch;
49
+ delete process.env.CODER_INSTALL_METHOD;
50
+ });
51
+
52
+ // Version Comparison Tests
53
+
54
+ test('checkForUpdates: detects newer major version', async t => {
55
+ // Calculate a newer major version dynamically
56
+ const currentParts = CURRENT_VERSION.split('.');
57
+ const newerMajorVersion = `${parseInt(currentParts[0]) + 1}.0.0`;
58
+
59
+ globalThis.fetch = createMockFetch(200, {
60
+ version: newerMajorVersion,
61
+ name: '@gguf/coder',
62
+ });
63
+
64
+ const result = await checkForUpdates();
65
+
66
+ t.true(result.hasUpdate);
67
+ t.is(result.currentVersion, CURRENT_VERSION);
68
+ t.is(result.latestVersion, newerMajorVersion);
69
+ t.truthy(result.updateCommand);
70
+ });
71
+
72
+ test('checkForUpdates: detects newer minor version', async t => {
73
+ // Calculate a newer minor version dynamically
74
+ const currentParts = CURRENT_VERSION.split('.');
75
+ const newerMinorVersion = `${currentParts[0]}.${parseInt(currentParts[1]) + 1
76
+ }.0`;
77
+
78
+ globalThis.fetch = createMockFetch(200, {
79
+ version: newerMinorVersion,
80
+ name: '@gguf/coder',
81
+ });
82
+
83
+ const result = await checkForUpdates();
84
+
85
+ t.true(result.hasUpdate);
86
+ t.is(result.latestVersion, newerMinorVersion);
87
+ });
88
+
89
+ test('checkForUpdates: detects newer patch version', async t => {
90
+ // Calculate a newer patch version dynamically
91
+ const currentParts = CURRENT_VERSION.split('.');
92
+ const newerPatchVersion = `${currentParts[0]}.${currentParts[1]}.${parseInt(currentParts[2]) + 1
93
+ }`;
94
+
95
+ globalThis.fetch = createMockFetch(200, {
96
+ version: newerPatchVersion,
97
+ name: '@gguf/coder',
98
+ });
99
+
100
+ const result = await checkForUpdates();
101
+
102
+ t.true(result.hasUpdate);
103
+ t.is(result.latestVersion, newerPatchVersion);
104
+ });
105
+
106
+ test('checkForUpdates: detects same version (no update)', async t => {
107
+ globalThis.fetch = createMockFetch(200, {
108
+ version: CURRENT_VERSION,
109
+ name: '@gguf/coder',
110
+ });
111
+
112
+ const result = await checkForUpdates();
113
+
114
+ t.false(result.hasUpdate);
115
+ t.is(result.currentVersion, CURRENT_VERSION);
116
+ t.is(result.latestVersion, CURRENT_VERSION);
117
+ t.is(result.updateCommand, undefined);
118
+ });
119
+
120
+ test('checkForUpdates: detects older version (no update)', async t => {
121
+ // Calculate an older patch version dynamically
122
+ const currentParts = CURRENT_VERSION.split('.');
123
+ const patchNum = parseInt(currentParts[2]);
124
+ // Use 0 if current patch is already 0, otherwise decrement
125
+ const olderPatchVersion = `${currentParts[0]}.${currentParts[1]}.${Math.max(
126
+ 0,
127
+ patchNum - 1,
128
+ )}`;
129
+
130
+ globalThis.fetch = createMockFetch(200, {
131
+ version: olderPatchVersion,
132
+ name: '@gguf/coder',
133
+ });
134
+
135
+ const result = await checkForUpdates();
136
+
137
+ t.false(result.hasUpdate);
138
+ t.is(result.latestVersion, olderPatchVersion);
139
+ });
140
+
141
+ test('checkForUpdates: handles version with v prefix', async t => {
142
+ // Use a newer major version with v prefix
143
+ const currentParts = CURRENT_VERSION.split('.');
144
+ const newerMajorVersion = `v${parseInt(currentParts[0]) + 1}.0.0`;
145
+
146
+ globalThis.fetch = createMockFetch(200, {
147
+ version: newerMajorVersion,
148
+ name: '@gguf/coder',
149
+ });
150
+
151
+ const result = await checkForUpdates();
152
+
153
+ t.true(result.hasUpdate);
154
+ t.is(result.latestVersion, newerMajorVersion);
155
+ });
156
+
157
+ test('checkForUpdates: handles pre-release versions', async t => {
158
+ // Use a newer major version with pre-release tag
159
+ const currentParts = CURRENT_VERSION.split('.');
160
+ const newerPreReleaseVersion = `${parseInt(currentParts[0]) + 1}.0.0-beta.1`;
161
+
162
+ globalThis.fetch = createMockFetch(200, {
163
+ version: newerPreReleaseVersion,
164
+ name: '@gguf/coder',
165
+ });
166
+
167
+ const result = await checkForUpdates();
168
+
169
+ // Pre-release info is stripped during comparison
170
+ t.true(result.hasUpdate);
171
+ t.is(result.latestVersion, newerPreReleaseVersion);
172
+ });
173
+
174
+ // Network Error Handling Tests
175
+
176
+ test('checkForUpdates: handles network errors gracefully', async t => {
177
+ globalThis.fetch = createMockFetch(200, {}, true);
178
+
179
+ const result = await checkForUpdates();
180
+
181
+ t.false(result.hasUpdate);
182
+ t.truthy(result.currentVersion);
183
+ t.is(result.latestVersion, undefined);
184
+ });
185
+
186
+ test('checkForUpdates: handles HTTP 404 error', async t => {
187
+ globalThis.fetch = createMockFetch(404, {
188
+ error: 'Not found',
189
+ });
190
+
191
+ const result = await checkForUpdates();
192
+
193
+ t.false(result.hasUpdate);
194
+ t.truthy(result.currentVersion);
195
+ });
196
+
197
+ test('checkForUpdates: handles HTTP 500 error', async t => {
198
+ globalThis.fetch = createMockFetch(500, {
199
+ error: 'Internal server error',
200
+ });
201
+
202
+ const result = await checkForUpdates();
203
+
204
+ t.false(result.hasUpdate);
205
+ t.truthy(result.currentVersion);
206
+ });
207
+
208
+ test('checkForUpdates: handles timeout (via AbortSignal)', async t => {
209
+ // Simulate timeout by throwing AbortError
210
+ globalThis.fetch = (async () => {
211
+ const error = new Error('The operation was aborted');
212
+ error.name = 'AbortError';
213
+ throw error;
214
+ }) as typeof fetch;
215
+
216
+ const result = await checkForUpdates();
217
+
218
+ // Should handle timeout gracefully
219
+ t.false(result.hasUpdate);
220
+ });
221
+
222
+ // Response Format Tests
223
+
224
+ test('checkForUpdates: returns correct update command', async t => {
225
+ globalThis.fetch = createMockFetch(200, {
226
+ version: '2.0.0',
227
+ name: '@gguf/coder',
228
+ });
229
+
230
+ const result = await checkForUpdates();
231
+
232
+ t.is(result.updateCommand, 'npm update -g @gguf/coder');
233
+ });
234
+
235
+ test('checkForUpdates: returns correct Homebrew command when installed via Homebrew', async t => {
236
+ globalThis.fetch = createMockFetch(200, {
237
+ version: '2.0.0',
238
+ name: '@gguf/coder',
239
+ });
240
+
241
+ process.env.CODER_INSTALL_METHOD = 'homebrew';
242
+
243
+ const result = await checkForUpdates();
244
+
245
+ t.is(
246
+ result.updateCommand,
247
+ 'brew list coder >/dev/null 2>&1 && brew upgrade coder || (echo "Error: coder not found in Homebrew. Please install it first with: brew install coder" && exit 1)',
248
+ );
249
+ });
250
+
251
+ test('checkForUpdates: returns message for Nix installations (no executable command)', async t => {
252
+ globalThis.fetch = createMockFetch(200, {
253
+ version: '2.0.0',
254
+ name: '@gguf/coder',
255
+ });
256
+
257
+ process.env.CODER_INSTALL_METHOD = 'nix';
258
+
259
+ const result = await checkForUpdates();
260
+
261
+ t.is(
262
+ result.updateMessage,
263
+ 'To update, re-run: nix run github:gguf/coder (or update your flake).',
264
+ );
265
+ });
266
+
267
+ test('checkForUpdates: includes current version in response', async t => {
268
+ globalThis.fetch = createMockFetch(200, {
269
+ version: '2.0.0',
270
+ name: '@gguf/coder',
271
+ });
272
+
273
+ const result = await checkForUpdates();
274
+
275
+ t.truthy(result.currentVersion);
276
+ t.regex(result.currentVersion, /^\d+\.\d+\.\d+/);
277
+ });
278
+
279
+ test('checkForUpdates: handles missing version field in response', async t => {
280
+ globalThis.fetch = createMockFetch(200, {
281
+ name: '@gguf/coder',
282
+ // version field missing
283
+ });
284
+
285
+ const result = await checkForUpdates();
286
+
287
+ t.false(result.hasUpdate);
288
+ });
289
+
290
+ test('checkForUpdates: handles malformed JSON response', async t => {
291
+ globalThis.fetch = (async () => {
292
+ return {
293
+ ok: true,
294
+ status: 200,
295
+ statusText: 'OK',
296
+ json: async () => {
297
+ throw new Error('Invalid JSON');
298
+ },
299
+ } as unknown as Response;
300
+ }) as typeof fetch;
301
+
302
+ const result = await checkForUpdates();
303
+
304
+ t.false(result.hasUpdate);
305
+ });
306
+
307
+ // Edge Cases
308
+
309
+ test('checkForUpdates: handles empty version string', async t => {
310
+ globalThis.fetch = createMockFetch(200, {
311
+ version: '',
312
+ name: '@gguf/coder',
313
+ });
314
+
315
+ const result = await checkForUpdates();
316
+
317
+ t.false(result.hasUpdate);
318
+ });
319
+
320
+ test('checkForUpdates: handles version with extra segments', async t => {
321
+ globalThis.fetch = createMockFetch(200, {
322
+ version: '2.0.0.1',
323
+ name: '@gguf/coder',
324
+ });
325
+
326
+ const result = await checkForUpdates();
327
+
328
+ // Should compare first 3 segments
329
+ t.true(result.hasUpdate);
330
+ });
331
+
332
+ test('checkForUpdates: handles invalid version format', async t => {
333
+ globalThis.fetch = createMockFetch(200, {
334
+ version: 'not-a-version',
335
+ name: '@gguf/coder',
336
+ });
337
+
338
+ const result = await checkForUpdates();
339
+
340
+ // Should handle gracefully
341
+ t.false(result.hasUpdate);
342
+ });
343
+
344
+ // Integration Tests
345
+
346
+ test('checkForUpdates: complete workflow for update available', async t => {
347
+ globalThis.fetch = createMockFetch(200, {
348
+ version: '99.99.99',
349
+ name: '@gguf/coder',
350
+ });
351
+
352
+ const result = await checkForUpdates();
353
+
354
+ t.true(result.hasUpdate);
355
+ t.truthy(result.currentVersion);
356
+ t.is(result.latestVersion, '99.99.99');
357
+ t.is(result.updateCommand, 'npm update -g @gguf/coder');
358
+ });
359
+
360
+ test('checkForUpdates: complete workflow for no update', async t => {
361
+ globalThis.fetch = createMockFetch(200, {
362
+ version: '0.0.1',
363
+ name: '@gguf/coder',
364
+ });
365
+
366
+ const result = await checkForUpdates();
367
+
368
+ t.false(result.hasUpdate);
369
+ t.truthy(result.currentVersion);
370
+ t.is(result.latestVersion, '0.0.1');
371
+ t.is(result.updateCommand, undefined);
372
+ });
373
+
374
+ test('checkForUpdates: complete workflow for network failure', async t => {
375
+ globalThis.fetch = createMockFetch(200, {}, true);
376
+
377
+ const result = await checkForUpdates();
378
+
379
+ t.false(result.hasUpdate);
380
+ t.truthy(result.currentVersion);
381
+ t.is(result.latestVersion, undefined);
382
+ t.is(result.updateCommand, undefined);
383
+ });
@@ -0,0 +1,183 @@
1
+ import { readFileSync } from 'fs';
2
+ import { dirname, join } from 'path';
3
+ import { fileURLToPath } from 'url';
4
+ import { loadPreferences, savePreferences } from '@/config/preferences';
5
+ import { TIMEOUT_UPDATE_CHECK_MS } from '@/constants';
6
+ import type { NpmRegistryResponse, UpdateInfo } from '@/types/index';
7
+ import { logError } from '@/utils/message-queue';
8
+ import { detectInstallationMethod } from './installation-detector';
9
+
10
+ const UPDATE_COMMANDS = {
11
+ NPM: 'npm update -g @gguf/coder',
12
+ // Check if package exists before upgrading to provide better error messages
13
+ HOMEBREW:
14
+ 'brew list coder >/dev/null 2>&1 && brew upgrade coder || (echo "Error: coder not found in Homebrew. Please install it first with: brew install coder" && exit 1)',
15
+ } as const;
16
+
17
+ const UPDATE_MESSAGES = {
18
+ NIX: 'To update, re-run: nix run github:gguf/coder (or update your flake).',
19
+ UNKNOWN:
20
+ 'A new version is available. Please update using your package manager.',
21
+ } as const;
22
+
23
+ const __filename = fileURLToPath(import.meta.url);
24
+ const __dirname = dirname(__filename);
25
+
26
+ /**
27
+ * Compare two semver version strings
28
+ * Returns true if latest is greater than current
29
+ */
30
+ function isNewerVersion(current: string, latest: string): boolean {
31
+ const parseVersion = (version: string) => {
32
+ const clean = version.replace(/^v/, '').split('-')[0]; // Remove 'v' prefix and pre-release info
33
+ return clean.split('.').map(num => parseInt(num) || 0);
34
+ };
35
+
36
+ const currentParts = parseVersion(current);
37
+ const latestParts = parseVersion(latest);
38
+
39
+ const maxLength = Math.max(currentParts.length, latestParts.length);
40
+
41
+ for (let i = 0; i < maxLength; i++) {
42
+ const currentPart = currentParts[i] || 0;
43
+ const latestPart = latestParts[i] || 0;
44
+
45
+ if (latestPart > currentPart) {
46
+ return true;
47
+ } else if (latestPart < currentPart) {
48
+ return false;
49
+ }
50
+ }
51
+
52
+ return false;
53
+ }
54
+
55
+ /**
56
+ * Get the current package version from package.json
57
+ */
58
+ interface PackageJson {
59
+ version: string;
60
+ [key: string]: unknown;
61
+ }
62
+
63
+ function getCurrentVersion(): string {
64
+ try {
65
+ const packageJsonPath = join(__dirname, '../../package.json');
66
+ const packageJson = JSON.parse(
67
+ readFileSync(packageJsonPath, 'utf-8'),
68
+ ) as PackageJson;
69
+ return packageJson.version;
70
+ } catch (error) {
71
+ const errorMessage = error instanceof Error ? error.message : String(error);
72
+ logError(`Failed to read current version: ${errorMessage}`);
73
+ return '0.0.0';
74
+ }
75
+ }
76
+
77
+ /**
78
+ * Fetch the latest version from npm registry
79
+ */
80
+ async function fetchLatestVersion(): Promise<string | null> {
81
+ try {
82
+ const response = await fetch(
83
+ 'https://registry.npmjs.org/@gguf/coder/latest',
84
+ {
85
+ method: 'GET',
86
+ headers: {
87
+ Accept: 'application/json',
88
+ 'User-Agent': 'coder-update-checker',
89
+ },
90
+ // Add timeout
91
+ signal: AbortSignal.timeout(TIMEOUT_UPDATE_CHECK_MS),
92
+ },
93
+ );
94
+
95
+ if (!response.ok) {
96
+ throw new Error(`HTTP ${response.status}: ${response.statusText}`);
97
+ }
98
+
99
+ const data = (await response.json()) as NpmRegistryResponse;
100
+ return data.version;
101
+ } catch (error) {
102
+ const errorMessage = error instanceof Error ? error.message : String(error);
103
+ logError(`Failed to fetch latest version: ${errorMessage}`);
104
+ return null;
105
+ }
106
+ }
107
+
108
+ /**
109
+ * Update the last update check timestamp in preferences
110
+ */
111
+ function updateLastCheckTime(): void {
112
+ const preferences = loadPreferences();
113
+ preferences.lastUpdateCheck = Date.now();
114
+ savePreferences(preferences);
115
+ }
116
+
117
+ /**
118
+ * Check for package updates
119
+ */
120
+ export async function checkForUpdates(): Promise<UpdateInfo> {
121
+ const currentVersion = getCurrentVersion();
122
+
123
+ try {
124
+ const latestVersion = await fetchLatestVersion();
125
+ updateLastCheckTime();
126
+
127
+ if (!latestVersion) {
128
+ return {
129
+ hasUpdate: false,
130
+ currentVersion,
131
+ };
132
+ }
133
+
134
+ const hasUpdate = isNewerVersion(currentVersion, latestVersion);
135
+
136
+ function getUpdateDetails(hasUpdate: boolean): {
137
+ command?: string;
138
+ message?: string;
139
+ } {
140
+ if (!hasUpdate) {
141
+ return {};
142
+ }
143
+
144
+ const method = detectInstallationMethod();
145
+
146
+ // Use constants defined at top of file for maintainability
147
+
148
+ switch (method) {
149
+ case 'npm':
150
+ return { command: UPDATE_COMMANDS.NPM };
151
+ case 'homebrew':
152
+ return { command: UPDATE_COMMANDS.HOMEBREW };
153
+ case 'nix':
154
+ return { message: UPDATE_MESSAGES.NIX };
155
+ default:
156
+ // For 'unknown' fallback to a general message (do not attempt to run a command)
157
+ return { message: UPDATE_MESSAGES.UNKNOWN };
158
+ }
159
+ }
160
+
161
+ const updateDetails = getUpdateDetails(hasUpdate);
162
+
163
+ return {
164
+ hasUpdate,
165
+ currentVersion,
166
+ latestVersion,
167
+ updateCommand: updateDetails.command,
168
+ updateMessage: updateDetails.message,
169
+ };
170
+ } catch (error) {
171
+ const errorMessage = error instanceof Error ? error.message : String(error);
172
+ logError(`Update check failed: ${errorMessage}`);
173
+
174
+ // Still update the timestamp to prevent hammering the API on repeated failures
175
+ updateLastCheckTime();
176
+
177
+ return {
178
+ hasUpdate: false,
179
+ currentVersion,
180
+ error: errorMessage,
181
+ };
182
+ }
183
+ }
@@ -0,0 +1,103 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {ConfigWizard} from './config-wizard.js';
5
+
6
+ // ============================================================================
7
+ // Tests for ConfigWizard Component Rendering
8
+ // ============================================================================
9
+
10
+ console.log(`\nconfig-wizard.spec.tsx – ${React.version}`);
11
+
12
+ test('ConfigWizard renders with title', t => {
13
+ const {lastFrame} = render(
14
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
15
+ );
16
+
17
+ const output = lastFrame();
18
+ t.truthy(output);
19
+ t.regex(output!, /Configuration Wizard/);
20
+ });
21
+
22
+ test('ConfigWizard shows initial location step', t => {
23
+ const {lastFrame} = render(
24
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
25
+ );
26
+
27
+ const output = lastFrame();
28
+ t.regex(output!, /Where would you like to create your configuration/);
29
+ });
30
+
31
+ test('ConfigWizard shows keyboard shortcuts', t => {
32
+ const {lastFrame} = render(
33
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
34
+ );
35
+
36
+ const output = lastFrame();
37
+ t.regex(output!, /Esc.*Exit wizard/);
38
+ t.regex(output!, /Shift\+Tab.*Go back/);
39
+ });
40
+
41
+ test('ConfigWizard shows location options', t => {
42
+ const {lastFrame} = render(
43
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
44
+ );
45
+
46
+ const output = lastFrame();
47
+ t.regex(output!, /Current project directory/);
48
+ t.regex(output!, /Global user config/);
49
+ });
50
+
51
+ test('ConfigWizard renders without crashing when onCancel is provided', t => {
52
+ let cancelCalled = false;
53
+
54
+ const {lastFrame} = render(
55
+ <ConfigWizard
56
+ projectDir="/tmp/test-project"
57
+ onComplete={() => {}}
58
+ onCancel={() => {
59
+ cancelCalled = true;
60
+ }}
61
+ />,
62
+ );
63
+
64
+ t.truthy(lastFrame());
65
+ t.false(cancelCalled); // Should not be called on render
66
+ });
67
+
68
+ test('ConfigWizard accepts projectDir prop', t => {
69
+ const projectDir = '/custom/project/path';
70
+
71
+ const {lastFrame} = render(
72
+ <ConfigWizard projectDir={projectDir} onComplete={() => {}} />,
73
+ );
74
+
75
+ // Component should render without errors
76
+ t.truthy(lastFrame());
77
+ });
78
+
79
+ test('ConfigWizard renders TitledBox with correct border', t => {
80
+ const {lastFrame} = render(
81
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
82
+ );
83
+
84
+ const output = lastFrame();
85
+ // Check for rounded border characters
86
+ t.regex(output!, /╭/); // Top-left corner
87
+ t.regex(output!, /╮/); // Top-right corner
88
+ t.regex(output!, /╰/); // Bottom-left corner
89
+ t.regex(output!, /╯/); // Bottom-right corner
90
+ });
91
+
92
+ test('ConfigWizard renders with correct initial state', t => {
93
+ const {frames} = render(
94
+ <ConfigWizard projectDir="/tmp/test-project" onComplete={() => {}} />,
95
+ );
96
+
97
+ // Should have rendered at least one frame
98
+ t.true(frames.length > 0);
99
+
100
+ // First frame should show location step
101
+ const firstFrame = frames[0];
102
+ t.regex(firstFrame, /Configuration Wizard/);
103
+ });