@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,238 @@
1
+ /**
2
+ * Usage data storage
3
+ * Persists usage statistics to the app data directory
4
+ */
5
+
6
+ import * as fs from 'node:fs';
7
+ import * as path from 'node:path';
8
+ import {getAppDataPath, getConfigPath} from '@/config/paths';
9
+ import {MAX_DAILY_AGGREGATES, MAX_USAGE_SESSIONS} from '@/constants';
10
+ import {logInfo, logWarning} from '@/utils/message-queue';
11
+ import type {DailyAggregate, SessionUsage, UsageData} from '../types/usage';
12
+
13
+ const USAGE_FILE_NAME = 'usage.json';
14
+
15
+ function getLegacyUsageFilePath(): string {
16
+ // Legacy location: config directory (pre-app-data change)
17
+ try {
18
+ const configDir = getConfigPath();
19
+ return path.join(configDir, USAGE_FILE_NAME);
20
+ } catch {
21
+ return '';
22
+ }
23
+ }
24
+
25
+ function getUsageFilePath(): string {
26
+ const appDataDir = getAppDataPath();
27
+ const newPath = path.join(appDataDir, USAGE_FILE_NAME);
28
+
29
+ // If new path already exists, use it
30
+ if (fs.existsSync(newPath)) {
31
+ return newPath;
32
+ }
33
+
34
+ // Attempt one-time lazy migration from legacy location
35
+ const legacyPath = getLegacyUsageFilePath();
36
+ if (legacyPath && fs.existsSync(legacyPath)) {
37
+ try {
38
+ if (!fs.existsSync(appDataDir)) {
39
+ fs.mkdirSync(appDataDir, {recursive: true});
40
+ }
41
+
42
+ try {
43
+ fs.renameSync(legacyPath, newPath);
44
+ logInfo(`Migrated usage data to new location: ${newPath}`);
45
+ } catch (renameError) {
46
+ // Fallback if rename/move fails: copy then best-effort delete
47
+ logWarning(
48
+ `Could not move usage file (${
49
+ renameError instanceof Error ? renameError.message : 'unknown error'
50
+ }), copying instead...`,
51
+ );
52
+ fs.copyFileSync(legacyPath, newPath);
53
+ try {
54
+ fs.unlinkSync(legacyPath);
55
+ logInfo(`Successfully migrated usage data to: ${newPath}`);
56
+ } catch {
57
+ logWarning(
58
+ `Migrated usage data to new location, but could not remove old file at ${legacyPath}. You may want to manually delete it.`,
59
+ );
60
+ }
61
+ }
62
+
63
+ return newPath;
64
+ } catch (error) {
65
+ // On any failure, fall through to using newPath without migration
66
+ logWarning(
67
+ `Failed to migrate usage data from ${legacyPath}: ${
68
+ error instanceof Error ? error.message : 'unknown error'
69
+ }. Old data remains at legacy location.`,
70
+ );
71
+ }
72
+ }
73
+
74
+ return newPath;
75
+ }
76
+
77
+ function ensureAppDataDir(): void {
78
+ const appDataDir = getAppDataPath();
79
+ if (!fs.existsSync(appDataDir)) {
80
+ fs.mkdirSync(appDataDir, {recursive: true});
81
+ }
82
+ }
83
+
84
+ function createEmptyUsageData(): UsageData {
85
+ return {
86
+ sessions: [],
87
+ dailyAggregates: [],
88
+ totalLifetime: 0,
89
+ lastUpdated: Date.now(),
90
+ };
91
+ }
92
+
93
+ export function readUsageData(): UsageData {
94
+ try {
95
+ const filePath = getUsageFilePath();
96
+
97
+ if (!fs.existsSync(filePath)) {
98
+ return createEmptyUsageData();
99
+ }
100
+
101
+ const content = fs.readFileSync(filePath, 'utf-8');
102
+ const data = JSON.parse(content) as UsageData;
103
+
104
+ return data;
105
+ } catch (error) {
106
+ logWarning('Failed to read usage data:', true, {
107
+ context: {error},
108
+ });
109
+ return createEmptyUsageData();
110
+ }
111
+ }
112
+
113
+ export function writeUsageData(data: UsageData): void {
114
+ try {
115
+ ensureAppDataDir();
116
+
117
+ data.lastUpdated = Date.now();
118
+
119
+ const filePath = getUsageFilePath();
120
+ fs.writeFileSync(filePath, JSON.stringify(data, null, 2), 'utf-8');
121
+ } catch (error) {
122
+ logWarning('Failed to write usage data:', true, {
123
+ context: {error},
124
+ });
125
+ }
126
+ }
127
+
128
+ export function addSession(session: SessionUsage): void {
129
+ const data = readUsageData();
130
+
131
+ // Add session to the beginning (most recent first)
132
+ data.sessions.unshift(session);
133
+
134
+ // Keep only last MAX_USAGE_SESSIONS
135
+ if (data.sessions.length > MAX_USAGE_SESSIONS) {
136
+ data.sessions = data.sessions.slice(0, MAX_USAGE_SESSIONS);
137
+ }
138
+
139
+ // Update lifetime total
140
+ data.totalLifetime += session.tokens.total;
141
+
142
+ // Update daily aggregate
143
+ updateDailyAggregate(data, session);
144
+
145
+ writeUsageData(data);
146
+ }
147
+
148
+ function updateDailyAggregate(data: UsageData, session: SessionUsage): void {
149
+ const dateParts = new Date(session.timestamp).toISOString().split('T');
150
+ const dateStr = dateParts[0] || new Date().toISOString().split('T')[0] || '';
151
+
152
+ // Find or create daily aggregate
153
+ let dailyAggregate = data.dailyAggregates.find(agg => agg.date === dateStr);
154
+
155
+ if (!dailyAggregate) {
156
+ dailyAggregate = {
157
+ date: dateStr,
158
+ sessions: 0,
159
+ totalTokens: 0,
160
+ providers: {},
161
+ models: {},
162
+ };
163
+ data.dailyAggregates.push(dailyAggregate);
164
+ }
165
+
166
+ // Update aggregate
167
+ dailyAggregate.sessions += 1;
168
+ dailyAggregate.totalTokens += session.tokens.total;
169
+
170
+ // Update provider stats
171
+ dailyAggregate.providers[session.provider] =
172
+ (dailyAggregate.providers[session.provider] || 0) + session.tokens.total;
173
+
174
+ // Update model stats
175
+ dailyAggregate.models[session.model] =
176
+ (dailyAggregate.models[session.model] || 0) + session.tokens.total;
177
+
178
+ // Sort by date (newest first) and keep only last MAX_DAILY_AGGREGATES
179
+ data.dailyAggregates.sort((a, b) => b.date.localeCompare(a.date));
180
+ if (data.dailyAggregates.length > MAX_DAILY_AGGREGATES) {
181
+ data.dailyAggregates = data.dailyAggregates.slice(0, MAX_DAILY_AGGREGATES);
182
+ }
183
+ }
184
+
185
+ export function getTodayAggregate(): DailyAggregate | null {
186
+ const data = readUsageData();
187
+ const todayParts = new Date().toISOString().split('T');
188
+ const today = todayParts[0] || '';
189
+
190
+ return data.dailyAggregates.find(agg => agg.date === today) || null;
191
+ }
192
+
193
+ export function getLastNDaysAggregate(days: number): {
194
+ totalTokens: number;
195
+ totalSessions: number;
196
+ avgTokensPerDay: number;
197
+ } {
198
+ const data = readUsageData();
199
+ const cutoffDate = new Date();
200
+ cutoffDate.setDate(cutoffDate.getDate() - days);
201
+ const cutoffParts = cutoffDate.toISOString().split('T');
202
+ const cutoffStr = cutoffParts[0] || '';
203
+
204
+ const relevantAggregates = data.dailyAggregates.filter(
205
+ agg => agg.date >= cutoffStr,
206
+ );
207
+
208
+ const totalTokens = relevantAggregates.reduce(
209
+ (sum, agg) => sum + agg.totalTokens,
210
+ 0,
211
+ );
212
+ const totalSessions = relevantAggregates.reduce(
213
+ (sum, agg) => sum + agg.sessions,
214
+ 0,
215
+ );
216
+
217
+ return {
218
+ totalTokens,
219
+ totalSessions,
220
+ avgTokensPerDay: Math.round(totalTokens / (days || 1)),
221
+ };
222
+ }
223
+
224
+ /**
225
+ * Clear all usage data
226
+ */
227
+ export function clearUsageData(): void {
228
+ try {
229
+ const filePath = getUsageFilePath();
230
+ if (fs.existsSync(filePath)) {
231
+ fs.unlinkSync(filePath);
232
+ }
233
+ } catch (error) {
234
+ logWarning('Failed to clear usage data:', true, {
235
+ context: {error},
236
+ });
237
+ }
238
+ }
@@ -0,0 +1,456 @@
1
+ import * as fs from 'node:fs';
2
+ import * as os from 'node:os';
3
+ import * as path from 'node:path';
4
+ import type {Message} from '@/types/core.js';
5
+ import type {Tokenizer} from '@/types/tokenization.js';
6
+ import test from 'ava';
7
+ import {clearUsageData, readUsageData} from './storage.js';
8
+ import {
9
+ SessionTracker,
10
+ clearCurrentSession,
11
+ getCurrentSession,
12
+ initializeSession,
13
+ } from './tracker.js';
14
+
15
+ console.log('\ntracker.spec.ts');
16
+
17
+ // ============================================================================
18
+ // Mock Tokenizer
19
+ // ============================================================================
20
+
21
+ class MockTokenizer implements Tokenizer {
22
+ getName(): string {
23
+ return 'mock-tokenizer';
24
+ }
25
+
26
+ countTokens(message: Message): number {
27
+ // Simple mock: 1 token per 4 characters
28
+ return Math.ceil(message.content.length / 4);
29
+ }
30
+
31
+ encode(text: string): number {
32
+ // Return token count (1 token per 4 characters)
33
+ return Math.ceil(text.length / 4);
34
+ }
35
+ }
36
+
37
+ // ============================================================================
38
+ // Test Setup
39
+ // ============================================================================
40
+
41
+ function createTestDir(): string {
42
+ const testDir = path.join(
43
+ os.tmpdir(),
44
+ `coder-test-${Date.now()}-${Math.random().toString(36).substring(7)}`,
45
+ );
46
+ fs.mkdirSync(testDir, {recursive: true});
47
+ return testDir;
48
+ }
49
+
50
+ let originalEnv: NodeJS.ProcessEnv;
51
+
52
+ test.before(() => {
53
+ originalEnv = {...process.env};
54
+ });
55
+
56
+ test.beforeEach(() => {
57
+ const testDir = createTestDir();
58
+ process.env.XDG_CONFIG_HOME = testDir;
59
+ clearCurrentSession(); // Clear any lingering session from previous tests
60
+ });
61
+
62
+ test.afterEach(() => {
63
+ clearCurrentSession();
64
+ clearUsageData();
65
+ try {
66
+ if (process.env.XDG_CONFIG_HOME) {
67
+ fs.rmSync(process.env.XDG_CONFIG_HOME, {recursive: true, force: true});
68
+ }
69
+ } catch (error) {
70
+ // Ignore cleanup errors
71
+ }
72
+ });
73
+
74
+ test.after(() => {
75
+ process.env = originalEnv;
76
+ });
77
+
78
+ // Helper to create messages
79
+ function createMockMessages(): Message[] {
80
+ return [
81
+ {role: 'system', content: 'You are a helpful assistant.'},
82
+ {role: 'user', content: 'Hello'},
83
+ {role: 'assistant', content: 'Hi there!'},
84
+ ];
85
+ }
86
+
87
+ // ============================================================================
88
+ // SessionTracker Constructor Tests
89
+ // ============================================================================
90
+
91
+ test('SessionTracker initializes with provider and model', t => {
92
+ const tracker = new SessionTracker('openai', 'gpt-4');
93
+ const info = tracker.getSessionInfo();
94
+
95
+ t.is(info.provider, 'openai');
96
+ t.is(info.model, 'gpt-4');
97
+ t.truthy(info.id);
98
+ t.truthy(info.startTime);
99
+ });
100
+
101
+ test('SessionTracker generates unique session IDs', t => {
102
+ const tracker1 = new SessionTracker('openai', 'gpt-4');
103
+ const tracker2 = new SessionTracker('anthropic', 'claude');
104
+
105
+ const info1 = tracker1.getSessionInfo();
106
+ const info2 = tracker2.getSessionInfo();
107
+
108
+ t.not(info1.id, info2.id);
109
+ });
110
+
111
+ test('SessionTracker records start time', t => {
112
+ const before = Date.now();
113
+ const tracker = new SessionTracker('openai', 'gpt-4');
114
+ const after = Date.now();
115
+
116
+ const info = tracker.getSessionInfo();
117
+
118
+ t.true(info.startTime >= before);
119
+ t.true(info.startTime <= after);
120
+ });
121
+
122
+ // ============================================================================
123
+ // getCurrentStats Tests
124
+ // ============================================================================
125
+
126
+ test('getCurrentStats returns stats for empty messages', async t => {
127
+ const tracker = new SessionTracker('openai', 'gpt-4');
128
+ const tokenizer = new MockTokenizer();
129
+ const messages: Message[] = [];
130
+
131
+ const stats = await tracker.getCurrentStats(messages, tokenizer);
132
+
133
+ t.is(stats.provider, 'openai');
134
+ t.is(stats.model, 'gpt-4');
135
+ t.is(stats.messageCount, 0);
136
+ t.is(stats.tokens.total, 0);
137
+ t.truthy(stats.startTime);
138
+ });
139
+
140
+ test('getCurrentStats calculates token breakdown', async t => {
141
+ const tracker = new SessionTracker('openai', 'gpt-4');
142
+ const tokenizer = new MockTokenizer();
143
+ const messages = createMockMessages();
144
+
145
+ const stats = await tracker.getCurrentStats(messages, tokenizer);
146
+
147
+ t.is(stats.messageCount, 3);
148
+ t.true(stats.tokens.total > 0);
149
+ t.true(stats.tokens.system > 0);
150
+ t.true(stats.tokens.userMessages > 0);
151
+ t.true(stats.tokens.assistantMessages > 0);
152
+ });
153
+
154
+ test('getCurrentStats includes context limit and percent used', async t => {
155
+ const tracker = new SessionTracker('openai', 'gpt-4');
156
+ const tokenizer = new MockTokenizer();
157
+ const messages = createMockMessages();
158
+
159
+ const stats = await tracker.getCurrentStats(messages, tokenizer);
160
+
161
+ // Context limit may be null if model not found
162
+ if (stats.contextLimit) {
163
+ t.true(stats.contextLimit > 0);
164
+ t.true(stats.percentUsed >= 0);
165
+ t.true(stats.percentUsed <= 100 || stats.percentUsed > 100); // Allow over 100%
166
+ } else {
167
+ t.is(stats.percentUsed, 0);
168
+ }
169
+ });
170
+
171
+ test('getCurrentStats calculates percent used correctly', async t => {
172
+ const tracker = new SessionTracker('openai', 'gpt-4');
173
+ const tokenizer = new MockTokenizer();
174
+ // Create a message with known token count
175
+ const messages: Message[] = [
176
+ {role: 'user', content: 'a'.repeat(400)}, // 400 chars / 4 = 100 tokens
177
+ ];
178
+
179
+ const stats = await tracker.getCurrentStats(messages, tokenizer);
180
+
181
+ t.is(stats.tokens.total, 100);
182
+
183
+ if (stats.contextLimit) {
184
+ const expectedPercent = (100 / stats.contextLimit) * 100;
185
+ t.is(stats.percentUsed, expectedPercent);
186
+ }
187
+ });
188
+
189
+ // ============================================================================
190
+ // saveSession Tests
191
+ // ============================================================================
192
+
193
+ test('saveSession persists session to storage', t => {
194
+ const tracker = new SessionTracker('openai', 'gpt-4');
195
+ const tokenizer = new MockTokenizer();
196
+ const messages = createMockMessages();
197
+
198
+ tracker.saveSession(messages, tokenizer);
199
+
200
+ const data = readUsageData();
201
+ t.is(data.sessions.length, 1);
202
+
203
+ const session = data.sessions[0]!;
204
+ t.is(session.provider, 'openai');
205
+ t.is(session.model, 'gpt-4');
206
+ t.is(session.messageCount, 3);
207
+ t.true(session.tokens.total > 0);
208
+ t.true(session.duration !== undefined);
209
+ t.true(session.duration! >= 0);
210
+ });
211
+
212
+ test('saveSession includes duration', t => {
213
+ const tracker = new SessionTracker('openai', 'gpt-4');
214
+ const tokenizer = new MockTokenizer();
215
+ const messages = createMockMessages();
216
+
217
+ // Wait a bit to ensure duration > 0
218
+ const start = Date.now();
219
+ while (Date.now() - start < 10) {
220
+ // Small delay
221
+ }
222
+
223
+ tracker.saveSession(messages, tokenizer);
224
+
225
+ const data = readUsageData();
226
+ const session = data.sessions[0]!;
227
+
228
+ t.true(session.duration! >= 10);
229
+ });
230
+
231
+ test('saveSession saves correct token breakdown', t => {
232
+ const tracker = new SessionTracker('openai', 'gpt-4');
233
+ const tokenizer = new MockTokenizer();
234
+ const messages: Message[] = [
235
+ {role: 'system', content: 'abcd'}, // 4 chars = 1 token
236
+ {role: 'user', content: 'abcdefgh'}, // 8 chars = 2 tokens
237
+ {role: 'assistant', content: 'abcdefghijkl'}, // 12 chars = 3 tokens
238
+ ];
239
+
240
+ tracker.saveSession(messages, tokenizer);
241
+
242
+ const data = readUsageData();
243
+ const session = data.sessions[0]!;
244
+
245
+ t.is(session.tokens.system, 1);
246
+ t.is(session.tokens.userMessages, 2);
247
+ t.is(session.tokens.assistantMessages, 3);
248
+ t.is(session.tokens.total, 6);
249
+ });
250
+
251
+ // ============================================================================
252
+ // updateProviderModel Tests
253
+ // ============================================================================
254
+
255
+ test('updateProviderModel changes provider and model', t => {
256
+ const tracker = new SessionTracker('openai', 'gpt-4');
257
+
258
+ tracker.updateProviderModel('anthropic', 'claude-3');
259
+
260
+ const info = tracker.getSessionInfo();
261
+ t.is(info.provider, 'anthropic');
262
+ t.is(info.model, 'claude-3');
263
+ });
264
+
265
+ test('updateProviderModel preserves session ID and start time', t => {
266
+ const tracker = new SessionTracker('openai', 'gpt-4');
267
+ const infoBefore = tracker.getSessionInfo();
268
+
269
+ tracker.updateProviderModel('anthropic', 'claude');
270
+ const infoAfter = tracker.getSessionInfo();
271
+
272
+ t.is(infoAfter.id, infoBefore.id);
273
+ t.is(infoAfter.startTime, infoBefore.startTime);
274
+ });
275
+
276
+ test('updateProviderModel affects saved session', t => {
277
+ const tracker = new SessionTracker('openai', 'gpt-4');
278
+ const tokenizer = new MockTokenizer();
279
+ const messages = createMockMessages();
280
+
281
+ tracker.updateProviderModel('anthropic', 'claude-3-opus');
282
+ tracker.saveSession(messages, tokenizer);
283
+
284
+ const data = readUsageData();
285
+ const session = data.sessions[0]!;
286
+
287
+ t.is(session.provider, 'anthropic');
288
+ t.is(session.model, 'claude-3-opus');
289
+ });
290
+
291
+ // ============================================================================
292
+ // getSessionInfo Tests
293
+ // ============================================================================
294
+
295
+ test('getSessionInfo returns current session details', t => {
296
+ const tracker = new SessionTracker('openai', 'gpt-4');
297
+ const info = tracker.getSessionInfo();
298
+
299
+ t.truthy(info.id);
300
+ t.is(info.provider, 'openai');
301
+ t.is(info.model, 'gpt-4');
302
+ t.truthy(info.startTime);
303
+ });
304
+
305
+ // ============================================================================
306
+ // Module-level Functions Tests
307
+ // ============================================================================
308
+
309
+ test('initializeSession creates new session tracker', t => {
310
+ initializeSession('openai', 'gpt-4');
311
+
312
+ const session = getCurrentSession();
313
+ t.truthy(session);
314
+
315
+ const info = session!.getSessionInfo();
316
+ t.is(info.provider, 'openai');
317
+ t.is(info.model, 'gpt-4');
318
+ });
319
+
320
+ test('getCurrentSession returns null initially', t => {
321
+ const session = getCurrentSession();
322
+ t.is(session, null);
323
+ });
324
+
325
+ test('getCurrentSession returns initialized session', t => {
326
+ initializeSession('openai', 'gpt-4');
327
+
328
+ const session = getCurrentSession();
329
+ t.truthy(session);
330
+ t.true(session instanceof SessionTracker);
331
+ });
332
+
333
+ test('clearCurrentSession removes current session', t => {
334
+ initializeSession('openai', 'gpt-4');
335
+
336
+ let session = getCurrentSession();
337
+ t.truthy(session);
338
+
339
+ clearCurrentSession();
340
+
341
+ session = getCurrentSession();
342
+ t.is(session, null);
343
+ });
344
+
345
+ test('clearCurrentSession is idempotent', t => {
346
+ clearCurrentSession();
347
+ clearCurrentSession();
348
+
349
+ const session = getCurrentSession();
350
+ t.is(session, null);
351
+ });
352
+
353
+ test('initializeSession replaces existing session', t => {
354
+ initializeSession('openai', 'gpt-4');
355
+ const session1 = getCurrentSession();
356
+ const info1 = session1!.getSessionInfo();
357
+
358
+ initializeSession('anthropic', 'claude-3-opus');
359
+ const session2 = getCurrentSession();
360
+ const info2 = session2!.getSessionInfo();
361
+
362
+ t.not(info1.id, info2.id);
363
+ t.is(info2.provider, 'anthropic');
364
+ t.is(info2.model, 'claude-3-opus');
365
+ });
366
+
367
+ // ============================================================================
368
+ // Integration Tests
369
+ // ============================================================================
370
+
371
+ test('complete session tracking flow', async t => {
372
+ // Initialize session
373
+ initializeSession('openai', 'gpt-4');
374
+
375
+ const tracker = getCurrentSession();
376
+ t.truthy(tracker);
377
+
378
+ // Get current stats
379
+ const tokenizer = new MockTokenizer();
380
+ const messages = createMockMessages();
381
+ const stats = await tracker!.getCurrentStats(messages, tokenizer);
382
+
383
+ t.is(stats.provider, 'openai');
384
+ t.is(stats.model, 'gpt-4');
385
+ t.is(stats.messageCount, 3);
386
+ t.true(stats.tokens.total > 0);
387
+
388
+ // Save session
389
+ tracker!.saveSession(messages, tokenizer);
390
+
391
+ // Verify session was saved
392
+ const data = readUsageData();
393
+ t.is(data.sessions.length, 1);
394
+
395
+ const savedSession = data.sessions[0]!;
396
+ t.is(savedSession.provider, 'openai');
397
+ t.is(savedSession.model, 'gpt-4');
398
+ t.is(savedSession.messageCount, 3);
399
+
400
+ // Clear session
401
+ clearCurrentSession();
402
+ t.is(getCurrentSession(), null);
403
+ });
404
+
405
+ test('session tracking across provider/model changes', async t => {
406
+ // Initialize with first provider/model
407
+ initializeSession('openai', 'gpt-4');
408
+ const tracker = getCurrentSession()!;
409
+
410
+ // Change provider/model
411
+ tracker.updateProviderModel('anthropic', 'claude-3-opus');
412
+
413
+ // Get stats
414
+ const tokenizer = new MockTokenizer();
415
+ const messages = createMockMessages();
416
+ const stats = await tracker.getCurrentStats(messages, tokenizer);
417
+
418
+ t.is(stats.provider, 'anthropic');
419
+ t.is(stats.model, 'claude-3-opus');
420
+
421
+ // Save session
422
+ tracker.saveSession(messages, tokenizer);
423
+
424
+ // Verify correct provider/model was saved
425
+ const data = readUsageData();
426
+ const session = data.sessions[0]!;
427
+
428
+ t.is(session.provider, 'anthropic');
429
+ t.is(session.model, 'claude-3-opus');
430
+ });
431
+
432
+ test('multiple sessions tracked correctly', t => {
433
+ const tokenizer = new MockTokenizer();
434
+ const messages = createMockMessages();
435
+
436
+ // Session 1
437
+ initializeSession('openai', 'gpt-4');
438
+ getCurrentSession()!.saveSession(messages, tokenizer);
439
+
440
+ // Session 2
441
+ initializeSession('anthropic', 'claude');
442
+ getCurrentSession()!.saveSession(messages, tokenizer);
443
+
444
+ // Session 3
445
+ initializeSession('openai', 'gpt-3.5');
446
+ getCurrentSession()!.saveSession(messages, tokenizer);
447
+
448
+ // Verify all sessions saved
449
+ const data = readUsageData();
450
+ t.is(data.sessions.length, 3);
451
+
452
+ // Verify they're in reverse order (most recent first)
453
+ t.is(data.sessions[0]!.model, 'gpt-3.5');
454
+ t.is(data.sessions[1]!.model, 'claude');
455
+ t.is(data.sessions[2]!.model, 'gpt-4');
456
+ });