@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,382 @@
1
+ import { spawnSync } from 'node:child_process';
2
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
3
+ import { dirname } from 'node:path';
4
+ import { TitledBox } from '@/components/ui/titled-box';
5
+ import { colors } from '@/config/index';
6
+ import { useResponsiveTerminal } from '@/hooks/useTerminalWidth';
7
+ import { logError } from '@/utils/message-queue';
8
+ import { Box, Text, useFocus, useInput } from 'ink';
9
+ import RandomSpinner from '@/components/random-spinner';
10
+ import { useEffect, useState } from 'react';
11
+ import type { ProviderConfig } from '../types/config';
12
+ import { type ConfigLocation, LocationStep } from './steps/location-step';
13
+ import { McpStep } from './steps/mcp-step';
14
+ import { ProviderStep } from './steps/provider-step';
15
+ import { SummaryStep } from './steps/summary-step';
16
+ import type { McpServerConfig } from './templates/mcp-templates';
17
+ import { buildConfigObject } from './validation';
18
+
19
+ interface ConfigWizardProps {
20
+ projectDir: string;
21
+ onComplete: (configPath: string) => void;
22
+ onCancel?: () => void;
23
+ }
24
+
25
+ type WizardStep =
26
+ | 'location'
27
+ | 'providers'
28
+ | 'mcp'
29
+ | 'summary'
30
+ | 'editing'
31
+ | 'saving'
32
+ | 'complete';
33
+
34
+ export function ConfigWizard({
35
+ projectDir,
36
+ onComplete,
37
+ onCancel,
38
+ }: ConfigWizardProps) {
39
+ const [step, setStep] = useState<WizardStep>('location');
40
+ const [configPath, setConfigPath] = useState('');
41
+ const [providers, setProviders] = useState<ProviderConfig[]>([]);
42
+ const [mcpServers, setMcpServers] = useState<Record<string, McpServerConfig>>(
43
+ {},
44
+ );
45
+ const [error, setError] = useState<string | null>(null);
46
+ const { boxWidth, isNarrow } = useResponsiveTerminal();
47
+
48
+ // Capture focus to ensure keyboard handling works properly
49
+ useFocus({ autoFocus: true, id: 'config-wizard' });
50
+
51
+ // Load existing config if editing
52
+ useEffect(() => {
53
+ if (!configPath || !existsSync(configPath)) {
54
+ return;
55
+ }
56
+
57
+ // Use a microtask to defer state updates
58
+ void Promise.resolve().then(() => {
59
+ try {
60
+ const configContent = readFileSync(configPath, 'utf-8');
61
+ const config = JSON.parse(configContent) as {
62
+ coder?: {
63
+ providers?: ProviderConfig[];
64
+ mcpServers?: Record<string, McpServerConfig>;
65
+ };
66
+ };
67
+
68
+ const newProviders = config.coder?.providers || [];
69
+ const newMcpServers = config.coder?.mcpServers || {};
70
+
71
+ setProviders(newProviders);
72
+ setMcpServers(newMcpServers);
73
+ } catch (err) {
74
+ logError('Failed to load existing configuration', true, {
75
+ context: { configPath },
76
+ error: err instanceof Error ? err.message : String(err),
77
+ });
78
+ }
79
+ });
80
+ }, [configPath]);
81
+
82
+ const handleLocationComplete = (_location: ConfigLocation, path: string) => {
83
+ setConfigPath(path);
84
+ setStep('providers');
85
+ };
86
+
87
+ const handleProvidersComplete = (newProviders: ProviderConfig[]) => {
88
+ setProviders(newProviders);
89
+ setStep('mcp');
90
+ };
91
+
92
+ const handleMcpComplete = (
93
+ newMcpServers: Record<string, McpServerConfig>,
94
+ ) => {
95
+ setMcpServers(newMcpServers);
96
+ setStep('summary');
97
+ };
98
+
99
+ const handleSave = () => {
100
+ setStep('saving');
101
+ setError(null);
102
+
103
+ try {
104
+ // Build config object
105
+ const config = buildConfigObject(providers, mcpServers);
106
+
107
+ // Ensure directory exists
108
+ const dir = dirname(configPath);
109
+ if (!existsSync(dir)) {
110
+ mkdirSync(dir, { recursive: true });
111
+ }
112
+
113
+ // Write config file
114
+ writeFileSync(configPath, JSON.stringify(config, null, 2), 'utf-8');
115
+
116
+ setStep('complete');
117
+ // Don't auto-complete - wait for user to press Enter
118
+ } catch (err) {
119
+ setError(
120
+ err instanceof Error ? err.message : 'Failed to save configuration',
121
+ );
122
+ setStep('summary');
123
+ }
124
+ };
125
+
126
+ const handleAddProviders = () => {
127
+ setStep('providers');
128
+ };
129
+
130
+ const handleAddMcpServers = () => {
131
+ setStep('mcp');
132
+ };
133
+
134
+ const handleCancel = () => {
135
+ if (onCancel) {
136
+ onCancel();
137
+ }
138
+ };
139
+
140
+ const openInEditor = () => {
141
+ try {
142
+ // Save current progress to file
143
+ const config = buildConfigObject(providers, mcpServers);
144
+
145
+ // Ensure directory exists
146
+ const dir = dirname(configPath);
147
+ if (!existsSync(dir)) {
148
+ mkdirSync(dir, { recursive: true });
149
+ }
150
+
151
+ // Write config file
152
+ writeFileSync(configPath, JSON.stringify(config, null, 2), 'utf-8');
153
+
154
+ // Detect editor (respect $EDITOR or $VISUAL environment variables)
155
+ // Fall back to nano on Unix/Mac (much friendlier than vi!)
156
+ // On Windows, use notepad
157
+ const editor =
158
+ process.env.EDITOR ||
159
+ process.env.VISUAL ||
160
+ (process.platform === 'win32' ? 'notepad' : 'nano');
161
+
162
+ // Show cursor and restore terminal for editor
163
+ process.stdout.write('\x1B[?25h'); // Show cursor
164
+ process.stdin.setRawMode?.(false); // Disable raw mode
165
+
166
+ // Open editor and wait for it to close
167
+ const result = spawnSync(editor, [configPath], {
168
+ stdio: 'inherit', // Give editor full control of terminal
169
+ });
170
+
171
+ // Restore terminal state after editor closes
172
+ process.stdin.setRawMode?.(true); // Re-enable raw mode
173
+ process.stdout.write('\x1B[?25l'); // Hide cursor (Ink will manage it)
174
+
175
+ if (result.status === 0) {
176
+ // Reload the edited config
177
+ try {
178
+ const editedContent = readFileSync(configPath, 'utf-8');
179
+ const editedConfig = JSON.parse(editedContent) as {
180
+ coder?: {
181
+ providers?: ProviderConfig[];
182
+ mcpServers?: Record<string, McpServerConfig>;
183
+ };
184
+ };
185
+
186
+ // Update state with edited values
187
+ if (editedConfig.coder) {
188
+ setProviders(editedConfig.coder.providers || []);
189
+ setMcpServers(editedConfig.coder.mcpServers || {});
190
+ }
191
+
192
+ // Return to summary to review changes
193
+ setStep('summary');
194
+ setError(null);
195
+ } catch (parseErr) {
196
+ setError(
197
+ parseErr instanceof Error
198
+ ? `Invalid JSON: ${parseErr.message}`
199
+ : 'Failed to parse edited configuration',
200
+ );
201
+ setStep('summary');
202
+ }
203
+ } else {
204
+ setError('Editor exited with an error. Changes may not be saved.');
205
+ setStep('summary');
206
+ }
207
+ } catch (err) {
208
+ // Restore terminal state on error
209
+ process.stdin.setRawMode?.(true);
210
+ process.stdout.write('\x1B[?25l');
211
+
212
+ setError(
213
+ err instanceof Error
214
+ ? `Failed to open editor: ${err.message}`
215
+ : 'Failed to open editor',
216
+ );
217
+ setStep('summary');
218
+ }
219
+ };
220
+
221
+ // Handle global keyboard shortcuts
222
+ useInput((input, key) => {
223
+ // In complete step, wait for Enter to finish
224
+ if (step === 'complete' && key.return) {
225
+ onComplete(configPath);
226
+ return;
227
+ }
228
+
229
+ // Escape - cancel/exit wizard completely
230
+ if (key.escape) {
231
+ if (onCancel) {
232
+ onCancel();
233
+ }
234
+ return;
235
+ }
236
+
237
+ // Ctrl+E to open editor (available after location is chosen)
238
+ if (
239
+ key.ctrl &&
240
+ input === 'e' &&
241
+ configPath &&
242
+ (step === 'providers' || step === 'mcp' || step === 'summary')
243
+ ) {
244
+ openInEditor();
245
+ }
246
+ });
247
+
248
+ const renderStep = () => {
249
+ switch (step) {
250
+ case 'location': {
251
+ return (
252
+ <LocationStep
253
+ projectDir={projectDir}
254
+ onComplete={handleLocationComplete}
255
+ onBack={onCancel}
256
+ />
257
+ );
258
+ }
259
+ case 'providers': {
260
+ return (
261
+ <ProviderStep
262
+ existingProviders={providers}
263
+ onComplete={handleProvidersComplete}
264
+ onBack={() => setStep('location')}
265
+ />
266
+ );
267
+ }
268
+ case 'mcp': {
269
+ return (
270
+ <McpStep
271
+ existingServers={mcpServers}
272
+ onComplete={handleMcpComplete}
273
+ onBack={() => setStep('providers')}
274
+ />
275
+ );
276
+ }
277
+ case 'summary': {
278
+ return (
279
+ <SummaryStep
280
+ configPath={configPath}
281
+ providers={providers}
282
+ mcpServers={mcpServers}
283
+ onSave={handleSave}
284
+ onAddProviders={handleAddProviders}
285
+ onAddMcpServers={handleAddMcpServers}
286
+ onCancel={handleCancel}
287
+ onBack={() => setStep('mcp')}
288
+ />
289
+ );
290
+ }
291
+ case 'editing': {
292
+ return (
293
+ <Box flexDirection="column">
294
+ <Box marginBottom={1}>
295
+ <Text color={colors.primary}>Opening editor...</Text>
296
+ </Box>
297
+ <Box marginBottom={1}>
298
+ <Text dimColor>Configuration saved to: {configPath}</Text>
299
+ </Box>
300
+ <Box>
301
+ <Text color={colors.secondary}>
302
+ Save and close your editor to return to the wizard.
303
+ </Text>
304
+ </Box>
305
+ </Box>
306
+ );
307
+ }
308
+ case 'saving': {
309
+ return (
310
+ <Box flexDirection="column">
311
+ <Box>
312
+ <Text color={colors.success}>
313
+ <RandomSpinner /> Saving configuration...
314
+ </Text>
315
+ </Box>
316
+ </Box>
317
+ );
318
+ }
319
+ case 'complete': {
320
+ return (
321
+ <Box flexDirection="column">
322
+ <Box marginBottom={1}>
323
+ <Text color={colors.success} bold>
324
+ ✓ Configuration saved!
325
+ </Text>
326
+ </Box>
327
+ <Box marginBottom={1}>
328
+ <Text dimColor>Saved to: {configPath}</Text>
329
+ </Box>
330
+ <Box>
331
+ <Text color={colors.secondary}>Press Enter to continue</Text>
332
+ </Box>
333
+ </Box>
334
+ );
335
+ }
336
+ default: {
337
+ return null;
338
+ }
339
+ }
340
+ };
341
+
342
+ return (
343
+ <TitledBox
344
+ title="Configuration Wizard"
345
+ width={boxWidth}
346
+ borderColor={colors.primary}
347
+ paddingX={2}
348
+ paddingY={1}
349
+ flexDirection="column"
350
+ marginBottom={1}
351
+ >
352
+ {error && (
353
+ <Box marginBottom={1}>
354
+ <Text color={colors.error}>Error: {error}</Text>
355
+ </Box>
356
+ )}
357
+
358
+ {renderStep()}
359
+
360
+ {(step === 'location' ||
361
+ step === 'providers' ||
362
+ step === 'mcp' ||
363
+ step === 'summary') &&
364
+ (isNarrow ? (
365
+ <Box marginTop={1} flexDirection="column">
366
+ <Text color={colors.secondary}>Esc: Exit wizard</Text>
367
+ <Text color={colors.secondary}>Shift+Tab: Go back</Text>
368
+ {configPath && (
369
+ <Text color={colors.secondary}>Ctrl+E: Edit manually</Text>
370
+ )}
371
+ </Box>
372
+ ) : (
373
+ <Box marginTop={1}>
374
+ <Text color={colors.secondary}>
375
+ Esc: Exit wizard | Shift+Tab: Go back
376
+ {configPath && ' | Ctrl+E: Edit manually'}
377
+ </Text>
378
+ </Box>
379
+ ))}
380
+ </TitledBox>
381
+ );
382
+ }
@@ -0,0 +1,186 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {LocationStep} from './location-step.js';
5
+
6
+ // ============================================================================
7
+ // Tests for LocationStep Component Rendering
8
+ // ============================================================================
9
+
10
+ console.log(`\nlocation-step.spec.tsx – ${React.version}`);
11
+
12
+ test('LocationStep renders with location selection options', t => {
13
+ const {lastFrame} = render(
14
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
15
+ );
16
+
17
+ const output = lastFrame();
18
+ t.truthy(output);
19
+ t.regex(output!, /Where would you like to create your configuration\?/);
20
+ });
21
+
22
+ test('LocationStep shows project directory option', t => {
23
+ const {lastFrame} = render(
24
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
25
+ );
26
+
27
+ const output = lastFrame();
28
+ t.regex(output!, /Current project directory/);
29
+ });
30
+
31
+ test('LocationStep shows global config option', t => {
32
+ const {lastFrame} = render(
33
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
34
+ );
35
+
36
+ const output = lastFrame();
37
+ t.regex(output!, /Global user config/);
38
+ });
39
+
40
+ test('LocationStep shows tip about config types', t => {
41
+ const {lastFrame} = render(
42
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
43
+ );
44
+
45
+ const output = lastFrame();
46
+ t.regex(output!, /Project configs are useful for team settings/);
47
+ });
48
+
49
+ // ============================================================================
50
+ // Tests for LocationStep Component Callbacks
51
+ // ============================================================================
52
+
53
+ test('LocationStep calls onComplete when provided', t => {
54
+ let completeCalled = false;
55
+
56
+ const {lastFrame} = render(
57
+ <LocationStep
58
+ onComplete={() => {
59
+ completeCalled = true;
60
+ }}
61
+ projectDir="/test/project"
62
+ />,
63
+ );
64
+
65
+ t.truthy(lastFrame());
66
+ t.false(completeCalled); // Should not be called on render
67
+ });
68
+
69
+ test('LocationStep calls onBack when provided', t => {
70
+ let backCalled = false;
71
+
72
+ const {lastFrame} = render(
73
+ <LocationStep
74
+ onComplete={() => {}}
75
+ onBack={() => {
76
+ backCalled = true;
77
+ }}
78
+ projectDir="/test/project"
79
+ />,
80
+ );
81
+
82
+ t.truthy(lastFrame());
83
+ t.false(backCalled); // Should not be called on render
84
+ });
85
+
86
+ // ============================================================================
87
+ // Tests for LocationStep Props Validation
88
+ // ============================================================================
89
+
90
+ test('LocationStep requires onComplete prop', t => {
91
+ const {lastFrame} = render(
92
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
93
+ );
94
+
95
+ t.truthy(lastFrame());
96
+ });
97
+
98
+ test('LocationStep requires projectDir prop', t => {
99
+ const {lastFrame} = render(
100
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
101
+ );
102
+
103
+ t.truthy(lastFrame());
104
+ });
105
+
106
+ test('LocationStep handles optional onBack prop', t => {
107
+ const {lastFrame} = render(
108
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
109
+ );
110
+
111
+ t.truthy(lastFrame());
112
+ });
113
+
114
+ // ============================================================================
115
+ // Tests for LocationStep UI Elements
116
+ // ============================================================================
117
+
118
+ test('LocationStep renders SelectInput component', t => {
119
+ const {lastFrame} = render(
120
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
121
+ );
122
+
123
+ const output = lastFrame();
124
+ t.truthy(output);
125
+ t.regex(output!, /Current project directory|Global user config/);
126
+ });
127
+
128
+ test('LocationStep renders with correct initial state', t => {
129
+ const {frames} = render(
130
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
131
+ );
132
+
133
+ t.true(frames.length > 0);
134
+
135
+ const firstFrame = frames[0];
136
+ t.regex(firstFrame, /Where would you like to create your configuration\?/);
137
+ });
138
+
139
+ test('LocationStep renders without crashing', t => {
140
+ const {lastFrame} = render(
141
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
142
+ );
143
+
144
+ t.truthy(lastFrame());
145
+ });
146
+
147
+ test('LocationStep accepts projectDir with different paths', t => {
148
+ const testPaths = [
149
+ '/home/user/project',
150
+ '/var/www/app',
151
+ 'C:\\Users\\test\\project',
152
+ ];
153
+
154
+ for (const projectDir of testPaths) {
155
+ const {lastFrame} = render(
156
+ <LocationStep onComplete={() => {}} projectDir={projectDir} />,
157
+ );
158
+
159
+ t.truthy(lastFrame());
160
+ }
161
+ });
162
+
163
+ // ============================================================================
164
+ // Tests for LocationStep Narrow Terminal Mode
165
+ // ============================================================================
166
+
167
+ test('LocationStep renders in narrow mode', t => {
168
+ // Note: This test verifies the component renders, but we can't easily
169
+ // simulate terminal width changes in tests
170
+ const {lastFrame} = render(
171
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
172
+ );
173
+
174
+ t.truthy(lastFrame());
175
+ });
176
+
177
+ test('LocationStep shows config path information', t => {
178
+ const {lastFrame} = render(
179
+ <LocationStep onComplete={() => {}} projectDir="/test/project" />,
180
+ );
181
+
182
+ const output = lastFrame();
183
+ t.truthy(output);
184
+ // Should contain information about where configs will be created
185
+ t.regex(output!, /Current project directory|Global user config/);
186
+ });
@@ -0,0 +1,147 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ import { colors } from '@/config';
4
+ import { getConfigPath } from '@/config/paths';
5
+ import { useResponsiveTerminal } from '@/hooks/useTerminalWidth';
6
+ import { Box, Text, useInput } from 'ink';
7
+ import SelectInput from 'ink-select-input';
8
+ import { useState } from 'react';
9
+
10
+ export type ConfigLocation = 'project' | 'global';
11
+
12
+ interface LocationStepProps {
13
+ onComplete: (location: ConfigLocation, path: string) => void;
14
+ onBack?: () => void;
15
+ projectDir: string;
16
+ }
17
+
18
+ interface LocationOption {
19
+ label: string;
20
+ value: ConfigLocation;
21
+ }
22
+
23
+ export function LocationStep({
24
+ onComplete,
25
+ onBack,
26
+ projectDir,
27
+ }: LocationStepProps) {
28
+ const { isNarrow, truncatePath } = useResponsiveTerminal();
29
+ const projectPath = join(projectDir, 'coder.config.json');
30
+ const globalPath = join(getConfigPath(), 'coder.config.json');
31
+
32
+ const projectExists = existsSync(projectPath);
33
+ const globalExists = existsSync(globalPath);
34
+
35
+ const [mode, setMode] = useState<
36
+ 'select-location' | 'existing-config' | null
37
+ >(() => {
38
+ // If project config exists, show existing config menu
39
+ if (projectExists) {
40
+ return 'existing-config';
41
+ }
42
+ // If global exists but project doesn't, still show location selection
43
+ // but we'll note the global config exists
44
+ return 'select-location';
45
+ });
46
+
47
+ const existingPath = projectExists ? projectPath : globalPath;
48
+
49
+ const locationOptions: LocationOption[] = [
50
+ {
51
+ label: `Current project directory`,
52
+ value: 'project',
53
+ },
54
+ {
55
+ label: `Global user config`,
56
+ value: 'global',
57
+ },
58
+ ];
59
+
60
+ const existingConfigOptions = [
61
+ { label: 'Edit this configuration', value: 'edit' },
62
+ { label: 'Create new config in different location', value: 'new' },
63
+ ];
64
+
65
+ const handleLocationSelect = (item: LocationOption) => {
66
+ const path = item.value === 'project' ? projectPath : globalPath;
67
+ onComplete(item.value, path);
68
+ };
69
+
70
+ const handleExistingConfigSelect = (item: { value: string }) => {
71
+ if (item.value === 'edit') {
72
+ const location: ConfigLocation = projectExists ? 'project' : 'global';
73
+ onComplete(location, existingPath);
74
+ } else if (item.value === 'new') {
75
+ setMode('select-location');
76
+ } else {
77
+ // Cancel
78
+ onBack?.();
79
+ }
80
+ };
81
+
82
+ // Handle Shift+Tab to go back from select-location to existing-config
83
+ useInput((_input, key) => {
84
+ if (key.shift && key.tab) {
85
+ // If we're in select-location mode and came from existing-config, go back
86
+ if (mode === 'select-location' && (projectExists || globalExists)) {
87
+ setMode('existing-config');
88
+ } else {
89
+ // Otherwise, let the parent wizard handle it
90
+ onBack?.();
91
+ }
92
+ }
93
+ });
94
+
95
+ if (mode === 'existing-config') {
96
+ return (
97
+ <Box flexDirection="column">
98
+ <Box marginBottom={1} flexDirection="column">
99
+ <Text bold color={colors.primary}>
100
+ Configuration found at:{' '}
101
+ </Text>
102
+ <Text color={colors.secondary}>
103
+ {isNarrow ? truncatePath(existingPath, 40) : existingPath}
104
+ </Text>
105
+ </Box>
106
+ <SelectInput
107
+ items={existingConfigOptions}
108
+ onSelect={(item: { value: string }) => handleExistingConfigSelect(item)}
109
+ />
110
+ </Box>
111
+ );
112
+ }
113
+
114
+ return (
115
+ <Box flexDirection="column">
116
+ <Box marginBottom={1}>
117
+ <Text bold color={colors.primary}>
118
+ {isNarrow
119
+ ? 'Where to create config?'
120
+ : 'Where would you like to create your configuration?'}
121
+ </Text>
122
+ </Box>
123
+ {globalExists && !projectExists && (
124
+ <Box marginBottom={1} flexDirection="column">
125
+ <Text color={colors.warning}>
126
+ {isNarrow
127
+ ? 'Note: Global config exists'
128
+ : 'Note: Global config exists at'}
129
+ </Text>
130
+ {!isNarrow && <Text color={colors.secondary}>{globalPath}</Text>}
131
+ </Box>
132
+ )}
133
+ <SelectInput
134
+ items={locationOptions}
135
+ onSelect={(item: LocationOption) => handleLocationSelect(item)}
136
+ />
137
+ {!isNarrow && (
138
+ <Box marginTop={1}>
139
+ <Text color={colors.secondary}>
140
+ Tip: Project configs are useful for team settings. Global configs
141
+ work across all projects.
142
+ </Text>
143
+ </Box>
144
+ )}
145
+ </Box>
146
+ );
147
+ }