@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,216 @@
1
+ import test from 'ava';
2
+ import {normalizeIndentation} from './indentation-normalizer';
3
+
4
+ test('normalizeIndentation - empty array returns empty array', t => {
5
+ const result = normalizeIndentation([]);
6
+ t.deepEqual(result, []);
7
+ });
8
+
9
+ test('normalizeIndentation - single line with no indentation', t => {
10
+ const result = normalizeIndentation(['const x = 1;']);
11
+ t.deepEqual(result, ['const x = 1;']);
12
+ });
13
+
14
+ test('normalizeIndentation - single line with indentation becomes unindented', t => {
15
+ const result = normalizeIndentation([' const x = 1;']);
16
+ t.deepEqual(result, ['const x = 1;']);
17
+ });
18
+
19
+ test('normalizeIndentation - multiple lines with same indentation', t => {
20
+ const input = [' const x = 1;', ' const y = 2;', ' const z = 3;'];
21
+ const expected = ['const x = 1;', 'const y = 2;', 'const z = 3;'];
22
+ t.deepEqual(normalizeIndentation(input), expected);
23
+ });
24
+
25
+ test('normalizeIndentation - preserves relative indentation with spaces', t => {
26
+ const input = [
27
+ ' function test() {',
28
+ ' return 1;',
29
+ ' }',
30
+ ];
31
+ const expected = [
32
+ 'function test() {',
33
+ ' return 1;',
34
+ '}',
35
+ ];
36
+ t.deepEqual(normalizeIndentation(input), expected);
37
+ });
38
+
39
+ test('normalizeIndentation - preserves relative indentation with tabs', t => {
40
+ const input = [
41
+ '\t\tfunction test() {',
42
+ '\t\t\treturn 1;',
43
+ '\t\t}',
44
+ ];
45
+ const expected = [
46
+ 'function test() {',
47
+ ' return 1;', // Now uses 2 spaces instead of tab
48
+ '}',
49
+ ];
50
+ t.deepEqual(normalizeIndentation(input), expected);
51
+ });
52
+
53
+ test('normalizeIndentation - handles mixed indentation (tabs and spaces)', t => {
54
+ const input = [
55
+ '\t\tfunction test() {',
56
+ '\t\t return 1;', // 2 tabs + 2 spaces
57
+ '\t\t}',
58
+ ];
59
+ // Min indent is 2 tabs = 2 normalized units
60
+ // Line 2 has 2 tabs + 2 spaces = 2 + 1 = 3 normalized units
61
+ // Relative indent: 0, 1, 0
62
+ // Always uses 2 spaces for display
63
+ const expected = [
64
+ 'function test() {',
65
+ ' return 1;',
66
+ '}',
67
+ ];
68
+ t.deepEqual(normalizeIndentation(input), expected);
69
+ });
70
+
71
+ test('normalizeIndentation - empty lines are preserved', t => {
72
+ const input = [
73
+ ' function test() {',
74
+ '',
75
+ ' return 1;',
76
+ ' }',
77
+ ];
78
+ const expected = [
79
+ 'function test() {',
80
+ '',
81
+ ' return 1;',
82
+ '}',
83
+ ];
84
+ t.deepEqual(normalizeIndentation(input), expected);
85
+ });
86
+
87
+ test('normalizeIndentation - handles code with no minimum indentation', t => {
88
+ const input = [
89
+ 'function test() {',
90
+ ' return 1;',
91
+ '}',
92
+ ];
93
+ // Already at 0 indentation, should return as-is
94
+ t.deepEqual(normalizeIndentation(input), input);
95
+ });
96
+
97
+ test('normalizeIndentation - deeply nested code', t => {
98
+ const input = [
99
+ ' if (condition) {',
100
+ ' while (true) {',
101
+ ' doSomething();',
102
+ ' }',
103
+ ' }',
104
+ ];
105
+ const expected = [
106
+ 'if (condition) {',
107
+ ' while (true) {',
108
+ ' doSomething();',
109
+ ' }',
110
+ '}',
111
+ ];
112
+ t.deepEqual(normalizeIndentation(input), expected);
113
+ });
114
+
115
+ test('normalizeIndentation - code with varying indentation levels', t => {
116
+ const input = [
117
+ ' const items = [',
118
+ ' {id: 1},',
119
+ ' {id: 2},',
120
+ ' ];',
121
+ ];
122
+ const expected = [
123
+ 'const items = [',
124
+ ' {id: 1},',
125
+ ' {id: 2},',
126
+ '];',
127
+ ];
128
+ t.deepEqual(normalizeIndentation(input), expected);
129
+ });
130
+
131
+ test('normalizeIndentation - all empty lines returns input', t => {
132
+ const input = ['', '', ''];
133
+ t.deepEqual(normalizeIndentation(input), input);
134
+ });
135
+
136
+ test('normalizeIndentation - handles odd number of spaces', t => {
137
+ const input = [
138
+ ' const x = 1;', // 5 spaces = 2 normalized units
139
+ ' const y = 2;', // 7 spaces = 3 normalized units
140
+ ];
141
+ // Min indent is 2 units
142
+ // Relative: 0, 1
143
+ const expected = [
144
+ 'const x = 1;',
145
+ ' const y = 2;',
146
+ ];
147
+ t.deepEqual(normalizeIndentation(input), expected);
148
+ });
149
+
150
+ test('normalizeIndentation - always uses 2 spaces for display', t => {
151
+ const input = [
152
+ '\t\tconst x = 1;',
153
+ '\t\t\tconst y = 2;',
154
+ ];
155
+ const result = normalizeIndentation(input);
156
+ // Should use 2 spaces for relative indentation (not tabs)
157
+ t.deepEqual(result, [
158
+ 'const x = 1;',
159
+ ' const y = 2;',
160
+ ]);
161
+ });
162
+
163
+ test('normalizeIndentation - uses spaces when no tabs detected', t => {
164
+ const input = [
165
+ ' const x = 1;',
166
+ ' const y = 2;',
167
+ ];
168
+ const result = normalizeIndentation(input);
169
+ // Should use 2 spaces for relative indentation
170
+ t.deepEqual(result, [
171
+ 'const x = 1;',
172
+ ' const y = 2;',
173
+ ]);
174
+ });
175
+
176
+ test('normalizeIndentation - JSX example with deep nesting', t => {
177
+ const input = [
178
+ ' return (',
179
+ ' <div>',
180
+ ' {items.map(item => (',
181
+ ' <button onClick={() => handleClick(item.id)}>',
182
+ ' {item.name}',
183
+ ' </button>',
184
+ ' ))}',
185
+ ' </div>',
186
+ ' );',
187
+ ];
188
+ const expected = [
189
+ 'return (',
190
+ ' <div>',
191
+ ' {items.map(item => (',
192
+ ' <button onClick={() => handleClick(item.id)}>',
193
+ ' {item.name}',
194
+ ' </button>',
195
+ ' ))}',
196
+ ' </div>',
197
+ ');',
198
+ ];
199
+ t.deepEqual(normalizeIndentation(input), expected);
200
+ });
201
+
202
+ test('normalizeIndentation - handles whitespace-only lines as empty', t => {
203
+ const input = [
204
+ ' function test() {',
205
+ ' ',
206
+ ' return 1;',
207
+ ' }',
208
+ ];
209
+ const expected = [
210
+ 'function test() {',
211
+ ' ', // Whitespace-only line preserved as-is
212
+ ' return 1;',
213
+ '}',
214
+ ];
215
+ t.deepEqual(normalizeIndentation(input), expected);
216
+ });
@@ -0,0 +1,76 @@
1
+ /**
2
+ * Detects the minimum indentation level in a set of lines and returns
3
+ * normalized lines with relative indentation.
4
+ *
5
+ * Example:
6
+ * Input: [" const x = 1;", " const y = 2;"]
7
+ * Output: ["const x = 1;", "\tconst y = 2;"]
8
+ */
9
+ export function normalizeIndentation(lines: string[]): string[] {
10
+ if (lines.length === 0) {
11
+ return lines;
12
+ }
13
+
14
+ // Find minimum indentation level (excluding empty lines)
15
+ let minIndent = Number.POSITIVE_INFINITY;
16
+
17
+ for (const line of lines) {
18
+ if (line.trim().length === 0) {
19
+ continue; // Skip empty lines
20
+ }
21
+
22
+ // Count leading whitespace
23
+ const match = line.match(/^(\s+)/);
24
+ if (match) {
25
+ const whitespace = match[1];
26
+
27
+ // Convert to a normalized count (treating tab as 1 unit, 2 spaces as 1 unit)
28
+ const tabCount = (whitespace.match(/\t/g) || []).length;
29
+ const spaceCount = (whitespace.match(/ /g) || []).length;
30
+ const normalizedIndent = tabCount + Math.floor(spaceCount / 2);
31
+
32
+ minIndent = Math.min(minIndent, normalizedIndent);
33
+ } else {
34
+ // Line has no leading whitespace
35
+ minIndent = 0;
36
+ break;
37
+ }
38
+ }
39
+
40
+ // If all lines are empty, return as-is
41
+ if (minIndent === Number.POSITIVE_INFINITY) {
42
+ return lines;
43
+ }
44
+
45
+ // No normalization needed if already at 0 indentation
46
+ if (minIndent === 0) {
47
+ return lines;
48
+ }
49
+
50
+ // Always use 2 spaces for display (compact and terminal-friendly)
51
+ const indentChar = ' ';
52
+
53
+ // Normalize each line
54
+ return lines.map(line => {
55
+ if (line.trim().length === 0) {
56
+ return line; // Keep empty lines as-is
57
+ }
58
+
59
+ // Calculate current indentation level
60
+ const match = line.match(/^(\s+)/);
61
+ if (!match) {
62
+ return line; // No indentation to normalize
63
+ }
64
+
65
+ const whitespace = match[1];
66
+ const tabCount = (whitespace.match(/\t/g) || []).length;
67
+ const spaceCount = (whitespace.match(/ /g) || []).length;
68
+ const currentIndent = tabCount + Math.floor(spaceCount / 2);
69
+
70
+ // Calculate relative indentation
71
+ const relativeIndent = Math.max(0, currentIndent - minIndent);
72
+
73
+ // Return line with normalized indentation
74
+ return indentChar.repeat(relativeIndent) + line.trimStart();
75
+ });
76
+ }
@@ -0,0 +1,178 @@
1
+ import test from 'ava';
2
+ import {
3
+ detectFromPath,
4
+ detectInstallationMethod,
5
+ } from './installation-detector';
6
+
7
+ console.log(`\ninstallation-detector.spec.ts`);
8
+
9
+ test.beforeEach(() => {
10
+ // Clean up environment variables before each test
11
+ delete process.env.CODER_INSTALL_METHOD;
12
+ delete process.env.npm_config_prefix;
13
+ delete process.env.npm_config_global;
14
+ delete process.env.PNPM_HOME;
15
+ delete process.env.npm_execpath;
16
+ delete process.env.HOMEBREW_PREFIX;
17
+ delete process.env.HOMEBREW_CELLAR;
18
+ });
19
+
20
+ // Environment override tests
21
+ test('detectInstallationMethod: respects env override for npm', t => {
22
+ process.env.CODER_INSTALL_METHOD = 'npm';
23
+ t.is(detectInstallationMethod(), 'npm');
24
+ });
25
+
26
+ test('detectInstallationMethod: respects env override for homebrew', t => {
27
+ process.env.CODER_INSTALL_METHOD = 'homebrew';
28
+ t.is(detectInstallationMethod(), 'homebrew');
29
+ });
30
+
31
+ test('detectInstallationMethod: respects env override for nix', t => {
32
+ process.env.CODER_INSTALL_METHOD = 'nix';
33
+ t.is(detectInstallationMethod(), 'nix');
34
+ });
35
+
36
+ test('detectInstallationMethod: respects env override for unknown', t => {
37
+ process.env.CODER_INSTALL_METHOD = 'unknown';
38
+ t.is(detectInstallationMethod(), 'unknown');
39
+ });
40
+
41
+ // Homebrew environment variable detection
42
+ test('detectInstallationMethod: detects homebrew via HOMEBREW_PREFIX', t => {
43
+ process.env.HOMEBREW_PREFIX = '/opt/homebrew';
44
+ t.is(detectInstallationMethod(), 'homebrew');
45
+ });
46
+
47
+ test('detectInstallationMethod: detects homebrew via HOMEBREW_CELLAR', t => {
48
+ process.env.HOMEBREW_CELLAR = '/opt/homebrew/Cellar';
49
+ t.is(detectInstallationMethod(), 'homebrew');
50
+ });
51
+
52
+ // NPM environment variable detection
53
+ test('detectInstallationMethod: detects npm via npm_config_prefix', t => {
54
+ process.env.npm_config_prefix = '/usr/local';
55
+ t.is(detectInstallationMethod(), 'npm');
56
+ });
57
+
58
+ test('detectInstallationMethod: detects npm via npm_config_global', t => {
59
+ process.env.npm_config_global = 'true';
60
+ t.is(detectInstallationMethod(), 'npm');
61
+ });
62
+
63
+ test('detectInstallationMethod: detects npm via PNPM_HOME', t => {
64
+ process.env.PNPM_HOME = '/home/user/.local/share/pnpm';
65
+ t.is(detectInstallationMethod(), 'npm');
66
+ });
67
+
68
+ test('detectInstallationMethod: detects npm via npm_execpath', t => {
69
+ process.env.npm_execpath = '/usr/local/lib/node_modules/npm/bin/npm-cli.js';
70
+ t.is(detectInstallationMethod(), 'npm');
71
+ });
72
+
73
+ // Path Detection Tests
74
+ // These tests verify the detectFromPath function correctly identifies installation methods from paths
75
+
76
+ test('detectFromPath: detects nix from /nix/store path', t => {
77
+ const path =
78
+ '/nix/store/abc123-coder-1.0.0/lib/node_modules/@nanocollective/coder/dist';
79
+ t.is(detectFromPath(path), 'nix');
80
+ });
81
+
82
+ test('detectFromPath: detects homebrew from Cellar path (macOS Intel)', t => {
83
+ const path =
84
+ '/usr/local/Cellar/coder/1.0.0/libexec/lib/node_modules/@nanocollective/coder/dist';
85
+ t.is(detectFromPath(path), 'homebrew');
86
+ });
87
+
88
+ test('detectFromPath: detects homebrew from Cellar path (macOS ARM)', t => {
89
+ const path =
90
+ '/opt/homebrew/Cellar/coder/1.0.0/libexec/lib/node_modules/@nanocollective/coder/dist';
91
+ t.is(detectFromPath(path), 'homebrew');
92
+ });
93
+
94
+ test('detectFromPath: detects homebrew from generic homebrew path', t => {
95
+ const path = '/opt/homebrew/lib/node_modules/@nanocollective/coder/dist';
96
+ t.is(detectFromPath(path), 'homebrew');
97
+ });
98
+
99
+ test('detectFromPath: detects homebrew from Linux homebrew path', t => {
100
+ const path =
101
+ '/home/linuxbrew/.linuxbrew/Cellar/coder/1.0.0/lib/node_modules/@nanocollective/coder/dist';
102
+ t.is(detectFromPath(path), 'homebrew');
103
+ });
104
+
105
+ test('detectFromPath: detects npm from node_modules path (global)', t => {
106
+ const path = '/usr/local/lib/node_modules/@nanocollective/coder/dist';
107
+ t.is(detectFromPath(path), 'npm');
108
+ });
109
+
110
+ test('detectFromPath: detects npm from node_modules path (local)', t => {
111
+ const path = '/home/user/project/node_modules/@nanocollective/coder/dist';
112
+ t.is(detectFromPath(path), 'npm');
113
+ });
114
+
115
+ test('detectFromPath: detects npm from pnpm store path', t => {
116
+ const path =
117
+ '/home/user/.pnpm-store/.pnpm/@nanocollective+coder@1.0.0/node_modules/@nanocollective/coder/dist';
118
+ t.is(detectFromPath(path), 'npm');
119
+ });
120
+
121
+ test('detectFromPath: detects npm from .bin directory', t => {
122
+ const path = '/usr/local/lib/node_modules/.bin/coder';
123
+ t.is(detectFromPath(path), 'npm');
124
+ });
125
+
126
+ test('detectFromPath: returns null for unrecognized paths', t => {
127
+ const path = '/home/user/Downloads/coder/dist';
128
+ t.is(detectFromPath(path), null);
129
+ });
130
+
131
+ test('detectFromPath: detects npm from Windows AppData path', t => {
132
+ // Windows npm global installations typically go to AppData
133
+ const path =
134
+ 'C:\\Users\\user\\AppData\\Roaming\\npm\\node_modules\\@nanocollective\\coder\\dist';
135
+ t.is(detectFromPath(path), 'npm');
136
+ });
137
+
138
+ test('detectFromPath: nix takes precedence over node_modules', t => {
139
+ // Edge case: nix store might contain "node_modules" in the path
140
+ const path =
141
+ '/nix/store/abc123-coder-1.0.0/lib/node_modules/@nanocollective/coder/dist';
142
+ t.is(detectFromPath(path), 'nix');
143
+ });
144
+
145
+ test('detectFromPath: homebrew takes precedence over node_modules', t => {
146
+ // Edge case: homebrew path contains node_modules
147
+ const path =
148
+ '/opt/homebrew/Cellar/coder/1.0.0/lib/node_modules/@nanocollective/coder/dist';
149
+ t.is(detectFromPath(path), 'homebrew');
150
+ });
151
+
152
+ // Edge case tests
153
+ test('detectInstallationMethod: env vars take precedence over path', t => {
154
+ // Even if running from a homebrew path, HOMEBREW_PREFIX should be checked first
155
+ process.env.HOMEBREW_PREFIX = '/opt/homebrew';
156
+ t.is(detectInstallationMethod(), 'homebrew');
157
+ });
158
+
159
+ test('detectInstallationMethod: ignores invalid env override and continues detection', t => {
160
+ // Set an invalid value
161
+ process.env.CODER_INSTALL_METHOD = 'invalid-method';
162
+
163
+ // Should fall back to normal detection (ignoring invalid env var)
164
+ // and return a valid installation method
165
+ const result = detectInstallationMethod();
166
+
167
+ // Should still return a valid installation method after warning
168
+ // (will be detected from the actual running environment - npm in this case since we're in node_modules)
169
+ t.true(
170
+ ['npm', 'homebrew', 'nix', 'unknown'].includes(result),
171
+ `Should return valid installation method when invalid env var is set, got: ${result}`,
172
+ );
173
+
174
+ // Note: A warning is logged when an invalid env var is provided (visible in test output)
175
+ // but we don't assert on logging behavior here due to the complexity of mocking
176
+ // the structured logging system in tests. The warning is verified manually by inspecting
177
+ // test output which shows: "Invalid CODER_INSTALL_METHOD: invalid-method"
178
+ });
@@ -0,0 +1,153 @@
1
+ import {existsSync} from 'fs';
2
+ import {dirname, join, sep} from 'path';
3
+ import {fileURLToPath} from 'url';
4
+ import {logWarning} from './message-queue';
5
+
6
+ export type InstallationMethod = 'npm' | 'homebrew' | 'nix' | 'unknown';
7
+
8
+ // Define a safe process wrapper to avoid using `any` while keeping compatibility
9
+ type MaybeProcess = {
10
+ env?: {[key: string]: string | undefined};
11
+ argv?: string[];
12
+ };
13
+
14
+ const safeProcess: MaybeProcess =
15
+ typeof process !== 'undefined' ? (process as unknown as MaybeProcess) : {};
16
+
17
+ /**
18
+ * Detects installation method from a given module path.
19
+ * Exported for testing purposes.
20
+ * @param modulePath The path to check
21
+ * @returns The detected installation method or null if not detected from path
22
+ */
23
+ export function detectFromPath(modulePath: string): InstallationMethod | null {
24
+ // Strategy 1: Check for Nix installation (most specific)
25
+ // Nix store has `/nix/store/` path with store hashes - this is very reliable
26
+ if (modulePath.includes('/nix/store/')) {
27
+ return 'nix';
28
+ }
29
+
30
+ // Strategy 2: Check for Homebrew installation via path
31
+ // Homebrew puts packages under Cellar directory in standard locations
32
+ // Common paths: /opt/homebrew, /usr/local, /home/linuxbrew/.linuxbrew
33
+ if (
34
+ modulePath.includes(`${sep}Cellar${sep}`) ||
35
+ modulePath.includes(`${sep}homebrew${sep}`)
36
+ ) {
37
+ return 'homebrew';
38
+ }
39
+
40
+ // Strategy 3: Check for npm/pnpm/yarn installation using multiple signals
41
+ if (isNpmBasedInstallation(modulePath)) {
42
+ return 'npm';
43
+ }
44
+
45
+ return null;
46
+ }
47
+
48
+ /**
49
+ * Detects how Coder was installed by using multiple detection strategies.
50
+ * Uses a combination of path inspection, environment variables, and file system markers.
51
+ * An environment variable `CODER_INSTALL_METHOD` can be used to override detection for testing.
52
+ * @returns {InstallationMethod} The detected installation method.
53
+ */
54
+ export function detectInstallationMethod(): InstallationMethod {
55
+ // Env var override has highest priority for testing / debugging
56
+ const envOverride = safeProcess.env?.CODER_INSTALL_METHOD;
57
+ if (envOverride) {
58
+ const validMethods: InstallationMethod[] = [
59
+ 'npm',
60
+ 'homebrew',
61
+ 'nix',
62
+ 'unknown',
63
+ ];
64
+ if (validMethods.includes(envOverride as InstallationMethod)) {
65
+ return envOverride as InstallationMethod;
66
+ }
67
+ // Warn about invalid value but continue with normal detection
68
+ logWarning(
69
+ `Invalid CODER_INSTALL_METHOD: "${envOverride}". Valid values: ${validMethods.join(
70
+ ', ',
71
+ )}`,
72
+ );
73
+ }
74
+
75
+ // Strategy 1: Check environment variables (most reliable after override)
76
+ if (safeProcess.env?.HOMEBREW_PREFIX || safeProcess.env?.HOMEBREW_CELLAR) {
77
+ return 'homebrew';
78
+ }
79
+
80
+ // Use the module path of this file (compiled output in `dist`) to determine how it was installed.
81
+ const modulePath = dirname(fileURLToPath(import.meta.url));
82
+
83
+ // Strategy 2: Check path-based detection
84
+ const pathResult = detectFromPath(modulePath);
85
+ if (pathResult) {
86
+ return pathResult;
87
+ }
88
+
89
+ return 'unknown';
90
+ }
91
+
92
+ /**
93
+ * Checks if this is an npm-based installation (npm, pnpm, or yarn).
94
+ * Uses multiple detection strategies for robustness across different package managers.
95
+ */
96
+ function isNpmBasedInstallation(modulePath: string): boolean {
97
+ // Check 1: Environment variables set by package managers
98
+ if (
99
+ safeProcess.env?.npm_config_prefix ||
100
+ safeProcess.env?.npm_config_global ||
101
+ safeProcess.env?.PNPM_HOME ||
102
+ safeProcess.env?.npm_execpath
103
+ ) {
104
+ return true;
105
+ }
106
+
107
+ // Check 2: Standard node_modules path (npm, yarn v1)
108
+ if (modulePath.includes('node_modules')) {
109
+ return true;
110
+ }
111
+
112
+ // Check 3: pnpm store structure (.pnpm directory)
113
+ if (modulePath.includes(`.pnpm${sep}`)) {
114
+ return true;
115
+ }
116
+
117
+ // Check 4: Look for .bin directory in parent paths (all package managers use this)
118
+ // This handles symlinked executables
119
+ const binDirPattern = `${sep}.bin${sep}`;
120
+ if (modulePath.includes(binDirPattern)) {
121
+ return true;
122
+ }
123
+
124
+ // Check 5: Look for package.json in expected locations relative to the module
125
+ // For global installs, package.json should be in parent directories
126
+ // This handles edge cases like custom install locations
127
+ return hasPackageJsonMarker(modulePath);
128
+ }
129
+
130
+ /**
131
+ * Walks up the directory tree looking for package.json as a marker of npm installation.
132
+ * Only checks a few levels to avoid excessive file system operations.
133
+ */
134
+ function hasPackageJsonMarker(startPath: string): boolean {
135
+ let currentPath = startPath;
136
+ const maxLevelsToCheck = 4; // Limit to prevent excessive traversal
137
+
138
+ for (let i = 0; i < maxLevelsToCheck; i++) {
139
+ const packageJsonPath = join(currentPath, 'package.json'); // nosemgrep
140
+ if (existsSync(packageJsonPath)) {
141
+ return true;
142
+ }
143
+
144
+ const parentPath = dirname(currentPath);
145
+ // Stop if we've reached the root
146
+ if (parentPath === currentPath) {
147
+ break;
148
+ }
149
+ currentPath = parentPath;
150
+ }
151
+
152
+ return false;
153
+ }