@build-astron-co/nimbus 0.2.0 → 0.4.0

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 (469) hide show
  1. package/bin/nimbus +26 -10
  2. package/bin/nimbus.cmd +41 -0
  3. package/bin/nimbus.mjs +70 -0
  4. package/completions/nimbus.bash +38 -0
  5. package/completions/nimbus.fish +48 -0
  6. package/completions/nimbus.zsh +81 -0
  7. package/dist/src/agent/compaction-agent.js +215 -0
  8. package/dist/src/agent/context-manager.js +385 -0
  9. package/dist/src/agent/context.js +322 -0
  10. package/dist/src/agent/deploy-preview.js +395 -0
  11. package/dist/src/agent/expand-files.js +95 -0
  12. package/dist/src/agent/index.js +18 -0
  13. package/dist/src/agent/loop.js +1535 -0
  14. package/dist/src/agent/modes.js +347 -0
  15. package/dist/src/agent/permissions.js +396 -0
  16. package/dist/src/agent/subagents/base.js +67 -0
  17. package/dist/src/agent/subagents/cost.js +45 -0
  18. package/dist/src/agent/subagents/explore.js +36 -0
  19. package/dist/src/agent/subagents/general.js +41 -0
  20. package/dist/src/agent/subagents/index.js +88 -0
  21. package/dist/src/agent/subagents/infra.js +52 -0
  22. package/dist/src/agent/subagents/security.js +60 -0
  23. package/dist/src/agent/system-prompt.js +860 -0
  24. package/dist/src/app.js +152 -0
  25. package/dist/src/audit/activity-log.js +209 -0
  26. package/dist/src/audit/compliance-checker.js +419 -0
  27. package/dist/src/audit/cost-tracker.js +231 -0
  28. package/dist/src/audit/index.js +10 -0
  29. package/dist/src/audit/security-scanner.js +490 -0
  30. package/dist/src/auth/guard.js +64 -0
  31. package/dist/src/auth/index.js +19 -0
  32. package/dist/src/auth/keychain.js +79 -0
  33. package/dist/src/auth/oauth.js +389 -0
  34. package/dist/src/auth/providers.js +415 -0
  35. package/dist/src/auth/sso.js +87 -0
  36. package/dist/src/auth/store.js +424 -0
  37. package/dist/src/auth/types.js +5 -0
  38. package/dist/src/cli/index.js +8 -0
  39. package/dist/src/cli/init.js +1048 -0
  40. package/dist/src/cli/openapi-spec.js +346 -0
  41. package/dist/src/cli/run.js +505 -0
  42. package/dist/src/cli/serve-auth.js +56 -0
  43. package/dist/src/cli/serve.js +432 -0
  44. package/dist/src/cli/web.js +50 -0
  45. package/dist/src/cli.js +1574 -0
  46. package/dist/src/clients/core-engine-client.js +156 -0
  47. package/dist/src/clients/enterprise-client.js +246 -0
  48. package/dist/src/clients/generator-client.js +219 -0
  49. package/dist/src/clients/git-client.js +367 -0
  50. package/dist/src/clients/github-client.js +229 -0
  51. package/dist/src/clients/helm-client.js +299 -0
  52. package/dist/src/clients/index.js +18 -0
  53. package/dist/src/clients/k8s-client.js +270 -0
  54. package/dist/src/clients/llm-client.js +119 -0
  55. package/dist/src/clients/rest-client.js +104 -0
  56. package/dist/src/clients/service-discovery.js +35 -0
  57. package/dist/src/clients/terraform-client.js +302 -0
  58. package/dist/src/clients/tools-client.js +1227 -0
  59. package/dist/src/clients/ws-client.js +93 -0
  60. package/dist/src/commands/alias.js +91 -0
  61. package/dist/src/commands/analyze/index.js +313 -0
  62. package/dist/src/commands/apply/helm.js +375 -0
  63. package/dist/src/commands/apply/index.js +176 -0
  64. package/dist/src/commands/apply/k8s.js +350 -0
  65. package/dist/src/commands/apply/terraform.js +465 -0
  66. package/dist/src/commands/ask.js +137 -0
  67. package/dist/src/commands/audit/index.js +322 -0
  68. package/dist/src/commands/auth-cloud.js +345 -0
  69. package/dist/src/commands/auth-list.js +112 -0
  70. package/dist/src/commands/auth-profile.js +104 -0
  71. package/dist/src/commands/auth-refresh.js +161 -0
  72. package/dist/src/commands/auth-status.js +122 -0
  73. package/dist/src/commands/aws/ec2.js +402 -0
  74. package/dist/src/commands/aws/iam.js +304 -0
  75. package/dist/src/commands/aws/index.js +108 -0
  76. package/dist/src/commands/aws/lambda.js +317 -0
  77. package/dist/src/commands/aws/rds.js +345 -0
  78. package/dist/src/commands/aws/s3.js +346 -0
  79. package/dist/src/commands/aws/vpc.js +302 -0
  80. package/dist/src/commands/aws-discover.js +413 -0
  81. package/dist/src/commands/aws-terraform.js +618 -0
  82. package/dist/src/commands/azure/aks.js +305 -0
  83. package/dist/src/commands/azure/functions.js +200 -0
  84. package/dist/src/commands/azure/index.js +93 -0
  85. package/dist/src/commands/azure/storage.js +378 -0
  86. package/dist/src/commands/azure/vm.js +291 -0
  87. package/dist/src/commands/billing/index.js +224 -0
  88. package/dist/src/commands/chat.js +259 -0
  89. package/dist/src/commands/completions.js +255 -0
  90. package/dist/src/commands/config.js +291 -0
  91. package/dist/src/commands/cost/cloud-cost-estimator.js +211 -0
  92. package/dist/src/commands/cost/estimator.js +73 -0
  93. package/dist/src/commands/cost/index.js +625 -0
  94. package/dist/src/commands/cost/parsers/terraform.js +234 -0
  95. package/dist/src/commands/cost/parsers/types.js +4 -0
  96. package/dist/src/commands/cost/pricing/aws.js +501 -0
  97. package/dist/src/commands/cost/pricing/azure.js +462 -0
  98. package/dist/src/commands/cost/pricing/gcp.js +359 -0
  99. package/dist/src/commands/cost/pricing/index.js +24 -0
  100. package/dist/src/commands/demo.js +196 -0
  101. package/dist/src/commands/deploy.js +215 -0
  102. package/dist/src/commands/doctor.js +1291 -0
  103. package/dist/src/commands/drift/index.js +674 -0
  104. package/dist/src/commands/explain.js +235 -0
  105. package/dist/src/commands/export.js +120 -0
  106. package/dist/src/commands/feedback.js +319 -0
  107. package/dist/src/commands/fix.js +263 -0
  108. package/dist/src/commands/fs/index.js +338 -0
  109. package/dist/src/commands/gcp/compute.js +266 -0
  110. package/dist/src/commands/gcp/functions.js +221 -0
  111. package/dist/src/commands/gcp/gke.js +357 -0
  112. package/dist/src/commands/gcp/iam.js +295 -0
  113. package/dist/src/commands/gcp/index.js +105 -0
  114. package/dist/src/commands/gcp/storage.js +232 -0
  115. package/dist/src/commands/generate-helm.js +1026 -0
  116. package/dist/src/commands/generate-k8s.js +1263 -0
  117. package/dist/src/commands/generate-terraform.js +1058 -0
  118. package/dist/src/commands/gh/index.js +663 -0
  119. package/dist/src/commands/git/index.js +1208 -0
  120. package/dist/src/commands/helm/index.js +985 -0
  121. package/dist/src/commands/help.js +639 -0
  122. package/dist/src/commands/history.js +120 -0
  123. package/dist/src/commands/import.js +782 -0
  124. package/dist/src/commands/incident.js +144 -0
  125. package/dist/src/commands/index.js +109 -0
  126. package/dist/src/commands/init.js +955 -0
  127. package/dist/src/commands/k8s/index.js +979 -0
  128. package/dist/src/commands/login.js +588 -0
  129. package/dist/src/commands/logout.js +61 -0
  130. package/dist/src/commands/logs.js +160 -0
  131. package/dist/src/commands/onboarding.js +382 -0
  132. package/dist/src/commands/pipeline.js +153 -0
  133. package/dist/src/commands/plan/display.js +216 -0
  134. package/dist/src/commands/plan/index.js +525 -0
  135. package/dist/src/commands/plugin.js +325 -0
  136. package/dist/src/commands/preview.js +356 -0
  137. package/dist/src/commands/profile.js +297 -0
  138. package/dist/src/commands/questionnaire.js +1021 -0
  139. package/dist/src/commands/resume.js +35 -0
  140. package/dist/src/commands/rollback.js +259 -0
  141. package/dist/src/commands/rollout.js +74 -0
  142. package/dist/src/commands/runbook.js +307 -0
  143. package/dist/src/commands/schedule.js +202 -0
  144. package/dist/src/commands/status.js +213 -0
  145. package/dist/src/commands/team/index.js +309 -0
  146. package/dist/src/commands/team-context.js +200 -0
  147. package/dist/src/commands/template.js +204 -0
  148. package/dist/src/commands/tf/index.js +989 -0
  149. package/dist/src/commands/upgrade.js +515 -0
  150. package/dist/src/commands/usage/index.js +118 -0
  151. package/dist/src/commands/version.js +145 -0
  152. package/dist/src/commands/watch.js +127 -0
  153. package/dist/src/compat/index.js +2 -0
  154. package/dist/src/compat/runtime.js +10 -0
  155. package/dist/src/compat/sqlite.js +144 -0
  156. package/dist/src/config/index.js +6 -0
  157. package/dist/src/config/manager.js +469 -0
  158. package/dist/src/config/mode-store.js +57 -0
  159. package/dist/src/config/profiles.js +66 -0
  160. package/dist/src/config/safety-policy.js +251 -0
  161. package/dist/src/config/schema.js +107 -0
  162. package/dist/src/config/types.js +311 -0
  163. package/dist/src/config/workspace-state.js +38 -0
  164. package/dist/src/context/context-db.js +138 -0
  165. package/dist/src/demo/index.js +295 -0
  166. package/dist/src/demo/scenarios/full-journey.js +226 -0
  167. package/dist/src/demo/scenarios/getting-started.js +124 -0
  168. package/dist/src/demo/scenarios/helm-release.js +334 -0
  169. package/dist/src/demo/scenarios/k8s-deployment.js +190 -0
  170. package/dist/src/demo/scenarios/terraform-vpc.js +167 -0
  171. package/dist/src/demo/types.js +6 -0
  172. package/dist/src/engine/cost-estimator.js +334 -0
  173. package/dist/src/engine/diagram-generator.js +192 -0
  174. package/dist/src/engine/drift-detector.js +688 -0
  175. package/dist/src/engine/executor.js +832 -0
  176. package/dist/src/engine/index.js +39 -0
  177. package/dist/src/engine/orchestrator.js +436 -0
  178. package/dist/src/engine/planner.js +616 -0
  179. package/dist/src/engine/safety.js +609 -0
  180. package/dist/src/engine/verifier.js +664 -0
  181. package/dist/src/enterprise/audit.js +241 -0
  182. package/dist/src/enterprise/auth.js +189 -0
  183. package/dist/src/enterprise/billing.js +512 -0
  184. package/dist/src/enterprise/index.js +16 -0
  185. package/dist/src/enterprise/teams.js +315 -0
  186. package/dist/src/generator/best-practices.js +1375 -0
  187. package/dist/src/generator/helm.js +495 -0
  188. package/dist/src/generator/index.js +11 -0
  189. package/dist/src/generator/intent-parser.js +420 -0
  190. package/dist/src/generator/kubernetes.js +773 -0
  191. package/dist/src/generator/terraform.js +1472 -0
  192. package/dist/src/history/index.js +6 -0
  193. package/dist/src/history/manager.js +199 -0
  194. package/dist/src/history/types.js +6 -0
  195. package/dist/src/hooks/config.js +318 -0
  196. package/dist/src/hooks/engine.js +317 -0
  197. package/dist/src/hooks/index.js +2 -0
  198. package/dist/src/llm/auth-bridge.js +157 -0
  199. package/dist/src/llm/circuit-breaker.js +116 -0
  200. package/dist/src/llm/config-loader.js +172 -0
  201. package/dist/src/llm/cost-calculator.js +137 -0
  202. package/dist/src/llm/index.js +7 -0
  203. package/dist/src/llm/model-aliases.js +99 -0
  204. package/dist/src/llm/provider-registry.js +57 -0
  205. package/dist/src/llm/providers/anthropic.js +430 -0
  206. package/dist/src/llm/providers/bedrock.js +409 -0
  207. package/dist/src/llm/providers/google.js +344 -0
  208. package/dist/src/llm/providers/ollama.js +661 -0
  209. package/dist/src/llm/providers/openai-compatible.js +289 -0
  210. package/dist/src/llm/providers/openai.js +284 -0
  211. package/dist/src/llm/providers/openrouter.js +293 -0
  212. package/dist/src/llm/router.js +844 -0
  213. package/dist/src/llm/types.js +69 -0
  214. package/dist/src/lsp/client.js +239 -0
  215. package/dist/src/lsp/languages.js +95 -0
  216. package/dist/src/lsp/manager.js +243 -0
  217. package/dist/src/mcp/client.js +289 -0
  218. package/dist/src/mcp/index.js +5 -0
  219. package/dist/src/mcp/manager.js +113 -0
  220. package/dist/src/nimbus.js +212 -0
  221. package/dist/src/plugins/index.js +13 -0
  222. package/dist/src/plugins/loader.js +280 -0
  223. package/dist/src/plugins/manager.js +282 -0
  224. package/dist/src/plugins/types.js +23 -0
  225. package/dist/src/scanners/cicd-scanner.js +230 -0
  226. package/dist/src/scanners/cloud-scanner.js +415 -0
  227. package/dist/src/scanners/framework-scanner.js +430 -0
  228. package/dist/src/scanners/iac-scanner.js +350 -0
  229. package/dist/src/scanners/index.js +454 -0
  230. package/dist/src/scanners/language-scanner.js +258 -0
  231. package/dist/src/scanners/package-manager-scanner.js +252 -0
  232. package/dist/src/scanners/types.js +6 -0
  233. package/dist/src/sessions/manager.js +395 -0
  234. package/dist/src/sessions/types.js +4 -0
  235. package/dist/src/sharing/sync.js +238 -0
  236. package/dist/src/sharing/viewer.js +131 -0
  237. package/dist/src/snapshots/index.js +1 -0
  238. package/dist/src/snapshots/manager.js +432 -0
  239. package/dist/src/state/artifacts.js +94 -0
  240. package/dist/src/state/audit.js +73 -0
  241. package/dist/src/state/billing.js +126 -0
  242. package/dist/src/state/checkpoints.js +81 -0
  243. package/dist/src/state/config.js +58 -0
  244. package/dist/src/state/conversations.js +7 -0
  245. package/dist/src/state/credentials.js +96 -0
  246. package/dist/src/state/db.js +53 -0
  247. package/dist/src/state/index.js +23 -0
  248. package/dist/src/state/messages.js +76 -0
  249. package/dist/src/state/projects.js +92 -0
  250. package/dist/src/state/schema.js +233 -0
  251. package/dist/src/state/sessions.js +79 -0
  252. package/dist/src/state/teams.js +131 -0
  253. package/dist/src/telemetry.js +91 -0
  254. package/dist/src/tools/aws-ops.js +747 -0
  255. package/dist/src/tools/azure-ops.js +491 -0
  256. package/dist/src/tools/file-ops.js +451 -0
  257. package/dist/src/tools/gcp-ops.js +559 -0
  258. package/dist/src/tools/git-ops.js +557 -0
  259. package/dist/src/tools/github-ops.js +460 -0
  260. package/dist/src/tools/helm-ops.js +634 -0
  261. package/dist/src/tools/index.js +16 -0
  262. package/dist/src/tools/k8s-ops.js +579 -0
  263. package/dist/src/tools/schemas/converter.js +129 -0
  264. package/dist/src/tools/schemas/devops.js +3319 -0
  265. package/dist/src/tools/schemas/index.js +19 -0
  266. package/dist/src/tools/schemas/standard.js +966 -0
  267. package/dist/src/tools/schemas/types.js +409 -0
  268. package/dist/src/tools/spawn-exec.js +109 -0
  269. package/dist/src/tools/terraform-ops.js +627 -0
  270. package/dist/src/types/config.js +1 -0
  271. package/dist/src/types/drift.js +4 -0
  272. package/dist/src/types/enterprise.js +5 -0
  273. package/dist/src/types/index.js +14 -0
  274. package/dist/src/types/plan.js +1 -0
  275. package/dist/src/types/request.js +1 -0
  276. package/dist/src/types/response.js +1 -0
  277. package/dist/src/types/service.js +1 -0
  278. package/dist/src/ui/App.js +1672 -0
  279. package/dist/src/ui/DeployPreview.js +60 -0
  280. package/dist/src/ui/FileDiffModal.js +108 -0
  281. package/dist/src/ui/Header.js +46 -0
  282. package/dist/src/ui/HelpModal.js +9 -0
  283. package/dist/src/ui/InputBox.js +408 -0
  284. package/dist/src/ui/MessageList.js +795 -0
  285. package/dist/src/ui/PermissionPrompt.js +72 -0
  286. package/dist/src/ui/StatusBar.js +109 -0
  287. package/dist/src/ui/TerminalPane.js +31 -0
  288. package/dist/src/ui/ToolCallDisplay.js +303 -0
  289. package/dist/src/ui/TreePane.js +83 -0
  290. package/dist/src/ui/chat-ui.js +721 -0
  291. package/dist/src/ui/index.js +11 -0
  292. package/dist/src/ui/ink/index.js +1325 -0
  293. package/dist/src/ui/streaming.js +137 -0
  294. package/dist/src/ui/theme.js +78 -0
  295. package/dist/src/ui/types.js +7 -0
  296. package/dist/src/utils/analytics.js +61 -0
  297. package/dist/src/utils/cost-warning.js +25 -0
  298. package/dist/src/utils/env.js +42 -0
  299. package/dist/src/utils/errors.js +54 -0
  300. package/dist/src/utils/event-bus.js +22 -0
  301. package/dist/src/utils/index.js +16 -0
  302. package/dist/src/utils/logger.js +150 -0
  303. package/dist/src/utils/rate-limiter.js +90 -0
  304. package/dist/src/utils/service-auth.js +36 -0
  305. package/dist/src/utils/validation.js +39 -0
  306. package/dist/src/version.js +3 -0
  307. package/dist/src/watcher/index.js +192 -0
  308. package/dist/src/wizard/approval.js +275 -0
  309. package/dist/src/wizard/index.js +13 -0
  310. package/dist/src/wizard/prompts.js +273 -0
  311. package/dist/src/wizard/types.js +4 -0
  312. package/dist/src/wizard/ui.js +453 -0
  313. package/dist/src/wizard/wizard.js +227 -0
  314. package/package.json +31 -23
  315. package/src/__tests__/alias.test.ts +133 -0
  316. package/src/__tests__/app.test.ts +1 -1
  317. package/src/__tests__/audit.test.ts +1 -1
  318. package/src/__tests__/circuit-breaker.test.ts +1 -1
  319. package/src/__tests__/cli-run.test.ts +237 -1
  320. package/src/__tests__/compat-sqlite.test.ts +68 -0
  321. package/src/__tests__/context-manager.test.ts +131 -1
  322. package/src/__tests__/context.test.ts +1 -1
  323. package/src/__tests__/devops-terminal-gaps.test.ts +718 -0
  324. package/src/__tests__/doctor.test.ts +48 -0
  325. package/src/__tests__/enterprise.test.ts +1 -1
  326. package/src/__tests__/export.test.ts +236 -0
  327. package/src/__tests__/gap-11-18-20.test.ts +958 -0
  328. package/src/__tests__/generator.test.ts +1 -1
  329. package/src/__tests__/helm-streaming.test.ts +127 -0
  330. package/src/__tests__/hooks.test.ts +1 -1
  331. package/src/__tests__/incident.test.ts +179 -0
  332. package/src/__tests__/init.test.ts +55 -4
  333. package/src/__tests__/intent-parser.test.ts +1 -1
  334. package/src/__tests__/llm-router.test.ts +1 -1
  335. package/src/__tests__/logs.test.ts +107 -0
  336. package/src/__tests__/loop-errors.test.ts +244 -0
  337. package/src/__tests__/lsp.test.ts +1 -1
  338. package/src/__tests__/modes.test.ts +1 -1
  339. package/src/__tests__/perf-optimizations.test.ts +847 -0
  340. package/src/__tests__/permissions.test.ts +1 -1
  341. package/src/__tests__/pipeline.test.ts +50 -0
  342. package/src/__tests__/polish-phase3.test.ts +340 -0
  343. package/src/__tests__/profile.test.ts +237 -0
  344. package/src/__tests__/rollback.test.ts +83 -0
  345. package/src/__tests__/runbook.test.ts +219 -0
  346. package/src/__tests__/schedule.test.ts +206 -0
  347. package/src/__tests__/serve.test.ts +1 -1
  348. package/src/__tests__/sessions.test.ts +96 -1
  349. package/src/__tests__/sharing.test.ts +53 -1
  350. package/src/__tests__/snapshots.test.ts +1 -1
  351. package/src/__tests__/standalone-migration.test.ts +199 -0
  352. package/src/__tests__/state-db.test.ts +1 -1
  353. package/src/__tests__/status.test.ts +158 -0
  354. package/src/__tests__/stream-with-tools.test.ts +71 -25
  355. package/src/__tests__/subagents.test.ts +1 -1
  356. package/src/__tests__/system-prompt.test.ts +82 -3
  357. package/src/__tests__/terminal-gap-v2.test.ts +395 -0
  358. package/src/__tests__/terminal-parity.test.ts +393 -0
  359. package/src/__tests__/tf-apply.test.ts +187 -0
  360. package/src/__tests__/tool-converter.test.ts +1 -1
  361. package/src/__tests__/tool-schemas.test.ts +209 -4
  362. package/src/__tests__/tools.test.ts +4 -3
  363. package/src/__tests__/version-json.test.ts +184 -0
  364. package/src/__tests__/version.test.ts +1 -1
  365. package/src/__tests__/watch.test.ts +129 -0
  366. package/src/agent/compaction-agent.ts +40 -1
  367. package/src/agent/context-manager.ts +67 -3
  368. package/src/agent/deploy-preview.ts +62 -1
  369. package/src/agent/expand-files.ts +108 -0
  370. package/src/agent/loop.ts +1312 -31
  371. package/src/agent/permissions.ts +51 -4
  372. package/src/agent/system-prompt.ts +573 -19
  373. package/src/app.ts +58 -0
  374. package/src/audit/security-scanner.ts +45 -0
  375. package/src/auth/keychain.ts +82 -0
  376. package/src/auth/oauth.ts +15 -5
  377. package/src/cli/init.ts +378 -5
  378. package/src/cli/run.ts +407 -16
  379. package/src/cli/serve.ts +78 -1
  380. package/src/cli/web.ts +10 -6
  381. package/src/cli.ts +312 -1
  382. package/src/clients/service-discovery.ts +30 -25
  383. package/src/commands/alias.ts +100 -0
  384. package/src/commands/audit/index.ts +121 -2
  385. package/src/commands/auth-cloud.ts +113 -0
  386. package/src/commands/auth-refresh.ts +187 -0
  387. package/src/commands/aws-discover.ts +144 -251
  388. package/src/commands/aws-terraform.ts +68 -118
  389. package/src/commands/chat.ts +9 -3
  390. package/src/commands/completions.ts +268 -0
  391. package/src/commands/config.ts +26 -0
  392. package/src/commands/cost/index.ts +218 -2
  393. package/src/commands/deploy.ts +260 -0
  394. package/src/commands/doctor.ts +744 -152
  395. package/src/commands/drift/index.ts +371 -23
  396. package/src/commands/export.ts +146 -0
  397. package/src/commands/generate-k8s.ts +9 -61
  398. package/src/commands/generate-terraform.ts +191 -449
  399. package/src/commands/help.ts +212 -36
  400. package/src/commands/history.ts +8 -1
  401. package/src/commands/incident.ts +166 -0
  402. package/src/commands/init.ts +5 -0
  403. package/src/commands/login.ts +86 -1
  404. package/src/commands/logs.ts +167 -0
  405. package/src/commands/onboarding.ts +211 -34
  406. package/src/commands/pipeline.ts +186 -0
  407. package/src/commands/plugin.ts +398 -0
  408. package/src/commands/profile.ts +342 -0
  409. package/src/commands/questionnaire.ts +0 -98
  410. package/src/commands/resume.ts +26 -34
  411. package/src/commands/rollback.ts +315 -0
  412. package/src/commands/rollout.ts +88 -0
  413. package/src/commands/runbook.ts +346 -0
  414. package/src/commands/schedule.ts +236 -0
  415. package/src/commands/status.ts +252 -0
  416. package/src/commands/team-context.ts +220 -0
  417. package/src/commands/template.ts +58 -57
  418. package/src/commands/tf/index.ts +70 -11
  419. package/src/commands/upgrade.ts +57 -0
  420. package/src/commands/version.ts +54 -50
  421. package/src/commands/watch.ts +153 -0
  422. package/src/compat/runtime.ts +1 -1
  423. package/src/compat/sqlite.ts +75 -5
  424. package/src/config/mode-store.ts +62 -0
  425. package/src/config/profiles.ts +84 -0
  426. package/src/config/types.ts +83 -1
  427. package/src/config/workspace-state.ts +53 -0
  428. package/src/engine/cost-estimator.ts +52 -10
  429. package/src/engine/executor.ts +33 -2
  430. package/src/engine/planner.ts +68 -1
  431. package/src/generator/terraform.ts +8 -0
  432. package/src/history/manager.ts +2 -74
  433. package/src/hooks/engine.ts +5 -4
  434. package/src/llm/cost-calculator.ts +2 -2
  435. package/src/llm/providers/anthropic.ts +50 -21
  436. package/src/llm/router.ts +76 -7
  437. package/src/lsp/languages.ts +3 -0
  438. package/src/lsp/manager.ts +21 -5
  439. package/src/nimbus.ts +37 -18
  440. package/src/sessions/manager.ts +108 -1
  441. package/src/sharing/sync.ts +4 -0
  442. package/src/sharing/viewer.ts +66 -0
  443. package/src/tools/file-ops.ts +22 -0
  444. package/src/tools/schemas/devops.ts +3007 -117
  445. package/src/tools/schemas/standard.ts +5 -1
  446. package/src/tools/schemas/types.ts +31 -1
  447. package/src/tools/spawn-exec.ts +148 -0
  448. package/src/ui/App.tsx +1183 -66
  449. package/src/ui/DeployPreview.tsx +62 -57
  450. package/src/ui/FileDiffModal.tsx +162 -0
  451. package/src/ui/Header.tsx +87 -24
  452. package/src/ui/HelpModal.tsx +57 -0
  453. package/src/ui/InputBox.tsx +163 -10
  454. package/src/ui/MessageList.tsx +487 -40
  455. package/src/ui/PermissionPrompt.tsx +17 -5
  456. package/src/ui/StatusBar.tsx +122 -3
  457. package/src/ui/TerminalPane.tsx +84 -0
  458. package/src/ui/ToolCallDisplay.tsx +252 -18
  459. package/src/ui/TreePane.tsx +132 -0
  460. package/src/ui/chat-ui.ts +41 -44
  461. package/src/ui/ink/index.ts +771 -38
  462. package/src/ui/streaming.ts +1 -1
  463. package/src/ui/theme.ts +104 -0
  464. package/src/ui/types.ts +18 -0
  465. package/src/version.ts +1 -1
  466. package/src/watcher/index.ts +66 -15
  467. package/src/wizard/types.ts +1 -0
  468. package/src/wizard/ui.ts +1 -1
  469. package/tsconfig.json +2 -2
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * Tests for Session Sharing
3
3
  */
4
- import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
4
+ import { describe, test, expect, beforeEach, afterEach } from 'vitest';
5
5
  import {
6
6
  shareSession,
7
7
  getSharedSession,
@@ -39,7 +39,58 @@ const mockMessages = [
39
39
  // Inject test dependencies (no module-level mocks — avoids cross-file leaks)
40
40
  // ---------------------------------------------------------------------------
41
41
 
42
+ /** Minimal in-memory DB that mimics the SQLite API used by sharing/sync.ts */
43
+ function makeInMemoryDb() {
44
+ const rows = new Map<string, any>();
45
+
46
+ return {
47
+ run(sql: string, params: any[]) {
48
+ if (/INSERT INTO shares/.test(sql)) {
49
+ const [id, session_id, name, messages, model, mode, cost_usd, token_count, is_live, write_token, created_at, expires_at] = params;
50
+ rows.set(id, { id, session_id, name, messages, model, mode, cost_usd, token_count, is_live, write_token, created_at, expires_at });
51
+ return { changes: 1 };
52
+ } else if (/UPDATE shares SET messages/.test(sql)) {
53
+ const [messages, id] = params;
54
+ const row = rows.get(id);
55
+ if (row) { row.messages = messages; return { changes: 1 }; }
56
+ return { changes: 0 };
57
+ } else if (/DELETE FROM shares WHERE expires_at/.test(sql)) {
58
+ const [now] = params;
59
+ let count = 0;
60
+ for (const [id, row] of rows) {
61
+ if (row.expires_at <= now) { rows.delete(id); count++; }
62
+ }
63
+ return { changes: count };
64
+ } else if (/DELETE FROM shares WHERE id/.test(sql)) {
65
+ const [id] = params;
66
+ const existed = rows.has(id);
67
+ rows.delete(id);
68
+ return { changes: existed ? 1 : 0 };
69
+ }
70
+ return { changes: 0 };
71
+ },
72
+ query(sql: string) {
73
+ return {
74
+ get(...params: any[]) {
75
+ if (/WHERE id = \? AND expires_at/.test(sql)) {
76
+ const [id, now] = params;
77
+ const row = rows.get(id);
78
+ return row && row.expires_at > now ? row : undefined;
79
+ }
80
+ return undefined;
81
+ },
82
+ all(..._params: any[]) {
83
+ // listShares uses DELETE first then SELECT with ORDER BY (no WHERE params)
84
+ return [...rows.values()];
85
+ },
86
+ };
87
+ },
88
+ };
89
+ }
90
+
42
91
  beforeEach(() => {
92
+ const db = makeInMemoryDb();
93
+ _deps.getDb = () => db;
43
94
  _deps.getConversation = (id: string) =>
44
95
  id === 'test-session-id' ? { messages: mockMessages } : null;
45
96
  _deps.getSessionManager = () => ({
@@ -48,6 +99,7 @@ beforeEach(() => {
48
99
  });
49
100
 
50
101
  afterEach(() => {
102
+ _deps.getDb = undefined;
51
103
  _deps.getConversation = undefined;
52
104
  _deps.getSessionManager = undefined;
53
105
  });
@@ -9,7 +9,7 @@
9
9
  * git write-tree / read-tree / checkout-index workflow.
10
10
  */
11
11
 
12
- import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
12
+ import { describe, test, expect, beforeEach, afterEach } from 'vitest';
13
13
  import * as fs from 'node:fs';
14
14
  import * as path from 'node:path';
15
15
  import * as os from 'node:os';
@@ -0,0 +1,199 @@
1
+ /**
2
+ * Standalone Migration Tests
3
+ *
4
+ * Source-level checks verifying that microservice-dependent commands
5
+ * (RestClient, CoreEngineClient, localhost:300X) have been replaced
6
+ * with standalone CLI/SQLite/generator implementations.
7
+ *
8
+ * These are intentionally static — no runtime startup needed.
9
+ */
10
+
11
+ import { readFileSync } from 'node:fs';
12
+ import { join } from 'node:path';
13
+ import { describe, it, expect } from 'vitest';
14
+
15
+ const ROOT = join(__dirname, '..', '..');
16
+
17
+ function src(relativePath: string): string {
18
+ return readFileSync(join(ROOT, relativePath), 'utf8');
19
+ }
20
+
21
+ // ---------------------------------------------------------------------------
22
+ // C1 — generate-terraform: standalone
23
+ // ---------------------------------------------------------------------------
24
+ describe('C1 — generate-terraform standalone', () => {
25
+ it('has no new RestClient call', () => {
26
+ expect(src('src/commands/generate-terraform.ts')).not.toContain('new RestClient');
27
+ });
28
+ it('imports generateTerraformProject from generator', () => {
29
+ expect(src('src/commands/generate-terraform.ts')).toContain('generateTerraformProject');
30
+ });
31
+ it('has no localhost:300 reference', () => {
32
+ expect(src('src/commands/generate-terraform.ts')).not.toMatch(/localhost:300\d/);
33
+ });
34
+ });
35
+
36
+ // ---------------------------------------------------------------------------
37
+ // C2 — generate-k8s: standalone
38
+ // ---------------------------------------------------------------------------
39
+ describe('C2 — generate-k8s standalone', () => {
40
+ it('has no new RestClient call', () => {
41
+ expect(src('src/commands/generate-k8s.ts')).not.toContain('new RestClient');
42
+ });
43
+ it('uses generateManifestsLocally or kubernetes generator', () => {
44
+ expect(src('src/commands/generate-k8s.ts')).toMatch(/generateManifestsLocally|generateK8sManifests|K8sGeneratorConfig/);
45
+ });
46
+ it('has no localhost:300 reference', () => {
47
+ expect(src('src/commands/generate-k8s.ts')).not.toMatch(/localhost:300\d/);
48
+ });
49
+ });
50
+
51
+ // ---------------------------------------------------------------------------
52
+ // C3 — aws-discover: standalone
53
+ // ---------------------------------------------------------------------------
54
+ describe('C3 — aws-discover standalone', () => {
55
+ it('has no new RestClient call', () => {
56
+ expect(src('src/commands/aws-discover.ts')).not.toContain('new RestClient');
57
+ });
58
+ it('uses aws configure or execFile for profiles', () => {
59
+ expect(src('src/commands/aws-discover.ts')).toMatch(/configure|execFile|execFileSync|cliGetAwsProfiles/);
60
+ });
61
+ it('has no localhost:300 reference', () => {
62
+ expect(src('src/commands/aws-discover.ts')).not.toMatch(/localhost:300\d/);
63
+ });
64
+ });
65
+
66
+ // ---------------------------------------------------------------------------
67
+ // C4 — aws-terraform: standalone
68
+ // ---------------------------------------------------------------------------
69
+ describe('C4 — aws-terraform standalone', () => {
70
+ it('has no new RestClient call', () => {
71
+ expect(src('src/commands/aws-terraform.ts')).not.toContain('new RestClient');
72
+ });
73
+ it('imports generateTerraformProject from generator', () => {
74
+ expect(src('src/commands/aws-terraform.ts')).toContain('generateTerraformProject');
75
+ });
76
+ it('has no localhost:300 reference', () => {
77
+ expect(src('src/commands/aws-terraform.ts')).not.toMatch(/localhost:300\d/);
78
+ });
79
+ });
80
+
81
+ // ---------------------------------------------------------------------------
82
+ // C5 — resume: standalone
83
+ // ---------------------------------------------------------------------------
84
+ describe('C5 — resume command standalone', () => {
85
+ it('has no CoreEngineClient import', () => {
86
+ expect(src('src/commands/resume.ts')).not.toContain('CoreEngineClient');
87
+ });
88
+ it('references chatCommand or getDb for session lookup', () => {
89
+ expect(src('src/commands/resume.ts')).toMatch(/chatCommand|getDb|SessionManager/);
90
+ });
91
+ it('has no localhost:300 reference', () => {
92
+ expect(src('src/commands/resume.ts')).not.toMatch(/localhost:300\d/);
93
+ });
94
+ });
95
+
96
+ // ---------------------------------------------------------------------------
97
+ // C6 — version: DevOps CLI tools (not localhost services)
98
+ // ---------------------------------------------------------------------------
99
+ describe('C6 — version command DevOps tools', () => {
100
+ it('has no localhost:300 references in fetchComponentVersions', () => {
101
+ expect(src('src/commands/version.ts')).not.toMatch(/localhost:300\d/);
102
+ });
103
+ it('includes terraform version check', () => {
104
+ expect(src('src/commands/version.ts')).toContain('terraform');
105
+ });
106
+ it('includes kubectl version check', () => {
107
+ expect(src('src/commands/version.ts')).toContain('kubectl');
108
+ });
109
+ it('includes helm version check', () => {
110
+ expect(src('src/commands/version.ts')).toContain('helm');
111
+ });
112
+ it('shows [+] / [-] icon format', () => {
113
+ expect(src('src/commands/version.ts')).toMatch(/\[\+\]|\[-\]/);
114
+ });
115
+ });
116
+
117
+ // ---------------------------------------------------------------------------
118
+ // C7 — drift: no CoreEngineClient
119
+ // ---------------------------------------------------------------------------
120
+ describe('C7 — drift no CoreEngineClient', () => {
121
+ it('drift/index.ts has no CoreEngineClient import', () => {
122
+ const content = src('src/commands/drift/index.ts');
123
+ // Must not have an import statement (comments are fine)
124
+ expect(content).not.toMatch(/^import.*CoreEngineClient/m);
125
+ });
126
+ it('drift/index.ts uses execFileSync or execFile for terraform', () => {
127
+ expect(src('src/commands/drift/index.ts')).toMatch(/execFileSync|execFile|spawnExec/);
128
+ });
129
+ });
130
+
131
+ // ---------------------------------------------------------------------------
132
+ // C8 — dead client code deleted/stubbed
133
+ // ---------------------------------------------------------------------------
134
+ describe('C8 — dead client code', () => {
135
+ it('service-discovery.ts is either deleted or a stub without ghost URLs', () => {
136
+ try {
137
+ const content = src('src/clients/service-discovery.ts');
138
+ // If it exists, it must not contain the old ghost localhost service URLs
139
+ expect(content).not.toContain('localhost:3001');
140
+ expect(content).not.toContain('localhost:3003');
141
+ expect(content).not.toContain('localhost:3009');
142
+ } catch {
143
+ // File deleted — that's fine
144
+ }
145
+ });
146
+ });
147
+
148
+ // ---------------------------------------------------------------------------
149
+ // H1 — Emoji removed, replaced with ASCII icons
150
+ // ---------------------------------------------------------------------------
151
+ describe('H1 — emoji removed from UI', () => {
152
+ it('TreePane.tsx has no folder emoji', () => {
153
+ expect(src('src/ui/TreePane.tsx')).not.toContain('\u{1F4C1}'); // 📁
154
+ });
155
+ it('TreePane.tsx uses [/] for directories', () => {
156
+ expect(src('src/ui/TreePane.tsx')).toContain('[/]');
157
+ });
158
+ it('TerminalPane.tsx has no right-pointing triangle', () => {
159
+ expect(src('src/ui/TerminalPane.tsx')).not.toContain('\u25B6'); // ▶
160
+ });
161
+ it('TerminalPane.tsx uses [>] icon', () => {
162
+ expect(src('src/ui/TerminalPane.tsx')).toContain('[>]');
163
+ });
164
+ it('StatusBar.tsx has no Greek capital delta (Δ)', () => {
165
+ expect(src('src/ui/StatusBar.tsx')).not.toContain('\u0394'); // Δ
166
+ });
167
+ it('StatusBar.tsx uses delta: text', () => {
168
+ expect(src('src/ui/StatusBar.tsx')).toContain('delta:');
169
+ });
170
+ });
171
+
172
+ // ---------------------------------------------------------------------------
173
+ // M1 — Secret redaction in spawn-exec
174
+ // ---------------------------------------------------------------------------
175
+ describe('M1 — secret redaction in spawn-exec', () => {
176
+ it('spawn-exec.ts has SECRET_PATTERNS constant', () => {
177
+ expect(src('src/tools/spawn-exec.ts')).toContain('SECRET_PATTERNS');
178
+ });
179
+ it('spawn-exec.ts has redactSecrets function', () => {
180
+ expect(src('src/tools/spawn-exec.ts')).toContain('redactSecrets');
181
+ });
182
+ it('spawn-exec.ts redacts AKIA keys', () => {
183
+ expect(src('src/tools/spawn-exec.ts')).toContain('AKIA');
184
+ });
185
+ it('spawn-exec.ts redacts Bearer tokens', () => {
186
+ expect(src('src/tools/spawn-exec.ts')).toContain('Bearer');
187
+ });
188
+ });
189
+
190
+ // ---------------------------------------------------------------------------
191
+ // M2 — Doctor shows inline versions
192
+ // ---------------------------------------------------------------------------
193
+ describe('M2 — doctor inline versions', () => {
194
+ it('doctor.ts references version in available tools output', () => {
195
+ // The available list should include t.version or similar version string
196
+ const content = src('src/commands/doctor.ts');
197
+ expect(content).toMatch(/t\.version|tool\.version|\.version\b/);
198
+ });
199
+ });
@@ -10,7 +10,7 @@
10
10
  * is never touched, and tests are fully isolated and fast.
11
11
  */
12
12
 
13
- import { describe, it, expect, beforeEach } from 'bun:test';
13
+ import { describe, it, expect, beforeEach } from 'vitest';
14
14
  import type { Database } from '../compat/sqlite';
15
15
  import { getTestDb } from '../state/db';
16
16
  import {
@@ -0,0 +1,158 @@
1
+ /**
2
+ * Tests for the nimbus status command (G18).
3
+ *
4
+ * The statusCommand runs concurrent CLI checks. We test the module structure
5
+ * and basic argument parsing rather than the actual CLI calls to avoid
6
+ * platform dependencies.
7
+ */
8
+
9
+ import { describe, test, expect } from 'vitest';
10
+ import { statusCommand, type StatusOptions } from '../commands/status';
11
+
12
+ describe('statusCommand (G18)', () => {
13
+ test('exports statusCommand function', () => {
14
+ expect(typeof statusCommand).toBe('function');
15
+ });
16
+
17
+ test('StatusOptions accepts json flag', () => {
18
+ const opts: StatusOptions = { json: true };
19
+ expect(opts.json).toBe(true);
20
+ });
21
+
22
+ test('StatusOptions json defaults to undefined', () => {
23
+ const opts: StatusOptions = {};
24
+ expect(opts.json).toBeUndefined();
25
+ });
26
+
27
+ test('statusCommand resolves without throwing when CLIs are unavailable', async () => {
28
+ // Mock console.log to suppress output in test
29
+ const original = console.log;
30
+ const logs: string[] = [];
31
+ console.log = (msg: string) => logs.push(msg);
32
+
33
+ try {
34
+ // Should not throw even if kubectl/terraform/aws/gcloud are not installed
35
+ // (the command catches errors gracefully)
36
+ await expect(statusCommand({ json: true })).resolves.toBeUndefined();
37
+ } finally {
38
+ console.log = original;
39
+ }
40
+ });
41
+
42
+ test('statusCommand with json option outputs valid JSON', async () => {
43
+ const original = console.log;
44
+ let jsonOutput = '';
45
+ console.log = (msg: string) => { jsonOutput = msg; };
46
+
47
+ try {
48
+ await statusCommand({ json: true });
49
+ // Should produce valid JSON
50
+ const parsed = JSON.parse(jsonOutput);
51
+ expect(parsed).toBeTypeOf('object');
52
+ // Should have an errors array
53
+ expect(Array.isArray(parsed.errors)).toBe(true);
54
+ } finally {
55
+ console.log = original;
56
+ }
57
+ });
58
+ });
59
+
60
+ describe('statusCommand C2 enhancements', () => {
61
+ test('JSON output includes model field with default fallback', async () => {
62
+ const original = console.log;
63
+ let jsonOutput = '';
64
+ console.log = (msg: string) => { jsonOutput = msg; };
65
+
66
+ try {
67
+ await statusCommand({ json: true });
68
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
69
+ // model should be set (either from config.json or default 'claude-sonnet-4-6')
70
+ expect(typeof parsed.model).toBe('string');
71
+ expect((parsed.model as string).length).toBeGreaterThan(0);
72
+ } finally {
73
+ console.log = original;
74
+ }
75
+ });
76
+
77
+ test('JSON output includes provider field with default fallback', async () => {
78
+ const original = console.log;
79
+ let jsonOutput = '';
80
+ console.log = (msg: string) => { jsonOutput = msg; };
81
+
82
+ try {
83
+ await statusCommand({ json: true });
84
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
85
+ // provider should be set (either from config.json or default 'anthropic')
86
+ expect(typeof parsed.provider).toBe('string');
87
+ expect((parsed.provider as string).length).toBeGreaterThan(0);
88
+ } finally {
89
+ console.log = original;
90
+ }
91
+ });
92
+
93
+ test('JSON output includes nimbusMdFound field', async () => {
94
+ const original = console.log;
95
+ let jsonOutput = '';
96
+ console.log = (msg: string) => { jsonOutput = msg; };
97
+
98
+ try {
99
+ await statusCommand({ json: true });
100
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
101
+ // nimbusMdFound should be a boolean (true or undefined/false)
102
+ expect(
103
+ parsed.nimbusMdFound === undefined || typeof parsed.nimbusMdFound === 'boolean'
104
+ ).toBe(true);
105
+ } finally {
106
+ console.log = original;
107
+ }
108
+ });
109
+
110
+ test('JSON output includes sessionCount field', async () => {
111
+ const original = console.log;
112
+ let jsonOutput = '';
113
+ console.log = (msg: string) => { jsonOutput = msg; };
114
+
115
+ try {
116
+ await statusCommand({ json: true });
117
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
118
+ // sessionCount should be a number or 'N/A'
119
+ expect(
120
+ typeof parsed.sessionCount === 'number' || parsed.sessionCount === 'N/A'
121
+ ).toBe(true);
122
+ } finally {
123
+ console.log = original;
124
+ }
125
+ });
126
+
127
+ test('default model is claude-sonnet-4-6 when config.json is absent', async () => {
128
+ const original = console.log;
129
+ let jsonOutput = '';
130
+ console.log = (msg: string) => { jsonOutput = msg; };
131
+
132
+ // We can test this by verifying the model is never empty
133
+ try {
134
+ await statusCommand({ json: true });
135
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
136
+ // Model should be a non-empty string
137
+ expect(typeof parsed.model).toBe('string');
138
+ expect((parsed.model as string).length).toBeGreaterThan(0);
139
+ } finally {
140
+ console.log = original;
141
+ }
142
+ });
143
+
144
+ test('default provider is anthropic when config.json is absent', async () => {
145
+ const original = console.log;
146
+ let jsonOutput = '';
147
+ console.log = (msg: string) => { jsonOutput = msg; };
148
+
149
+ try {
150
+ await statusCommand({ json: true });
151
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
152
+ expect(typeof parsed.provider).toBe('string');
153
+ expect((parsed.provider as string).length).toBeGreaterThan(0);
154
+ } finally {
155
+ console.log = original;
156
+ }
157
+ });
158
+ });
@@ -14,7 +14,7 @@
14
14
  * All tests use mocks -- no real API calls are made.
15
15
  */
16
16
 
17
- import { describe, test, expect, mock, beforeEach } from 'bun:test';
17
+ import { describe, test, it, expect, vi, beforeEach } from 'vitest';
18
18
  import type { ToolCompletionRequest, StreamChunk } from '../llm/types';
19
19
 
20
20
  // ---------------------------------------------------------------------------
@@ -120,7 +120,7 @@ describe('OllamaProvider.streamWithTools', () => {
120
120
  'data: [DONE]\n\n',
121
121
  ];
122
122
 
123
- globalThis.fetch = mock(() =>
123
+ globalThis.fetch = vi.fn(() =>
124
124
  Promise.resolve(
125
125
  new Response(buildReadableStream(sseLines), {
126
126
  status: 200,
@@ -170,7 +170,7 @@ describe('OllamaProvider.streamWithTools', () => {
170
170
  'data: [DONE]\n\n',
171
171
  ];
172
172
 
173
- globalThis.fetch = mock(() =>
173
+ globalThis.fetch = vi.fn(() =>
174
174
  Promise.resolve(
175
175
  new Response(buildReadableStream(sseLines), {
176
176
  status: 200,
@@ -202,7 +202,7 @@ describe('OllamaProvider.streamWithTools', () => {
202
202
  test('fallback: when native streaming fails, falls back to completeWithTools', async () => {
203
203
  let _callCount = 0;
204
204
 
205
- globalThis.fetch = mock((url: string | URL | Request) => {
205
+ globalThis.fetch = vi.fn((url: string | URL | Request) => {
206
206
  _callCount++;
207
207
  const urlStr = typeof url === 'string' ? url : url instanceof URL ? url.toString() : url.url;
208
208
 
@@ -268,7 +268,7 @@ describe('OllamaProvider.streamWithTools', () => {
268
268
  'data: [DONE]\n\n',
269
269
  ];
270
270
 
271
- globalThis.fetch = mock(() =>
271
+ globalThis.fetch = vi.fn(() =>
272
272
  Promise.resolve(
273
273
  new Response(buildReadableStream(sseLines), {
274
274
  status: 200,
@@ -311,7 +311,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
311
311
  },
312
312
  ]);
313
313
 
314
- const mockCreate = mock(() => Promise.resolve(streamChunks));
314
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
315
315
 
316
316
  const { OpenRouterProvider } = await import('../llm/providers/openrouter');
317
317
  const provider = new OpenRouterProvider('test-api-key');
@@ -392,7 +392,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
392
392
  },
393
393
  ]);
394
394
 
395
- const mockCreate = mock(() => Promise.resolve(streamChunks));
395
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
396
396
 
397
397
  const { OpenRouterProvider } = await import('../llm/providers/openrouter');
398
398
  const provider = new OpenRouterProvider('test-api-key');
@@ -415,7 +415,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
415
415
  });
416
416
 
417
417
  test('fallback: when SDK stream creation throws, the generator yields nothing', async () => {
418
- const mockCreate = mock(() => Promise.reject(new Error('API unavailable')));
418
+ const mockCreate = vi.fn(() => Promise.reject(new Error('API unavailable')));
419
419
 
420
420
  const { OpenRouterProvider } = await import('../llm/providers/openrouter');
421
421
  const provider = new OpenRouterProvider('test-api-key');
@@ -462,7 +462,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
462
462
  },
463
463
  ]);
464
464
 
465
- const mockCreate = mock(() => Promise.resolve(streamChunks));
465
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
466
466
 
467
467
  const { OpenRouterProvider } = await import('../llm/providers/openrouter');
468
468
  const provider = new OpenRouterProvider('test-api-key');
@@ -495,10 +495,8 @@ describe('OpenRouterProvider.streamWithTools', () => {
495
495
  // ===========================================================================
496
496
 
497
497
  describe('OpenAICompatibleProvider.streamWithTools', () => {
498
- function createProvider() {
499
- // Dynamic import to avoid module-level side effects
500
- // eslint-disable-next-line @typescript-eslint/no-var-requires
501
- const { OpenAICompatibleProvider } = require('../llm/providers/openai-compatible');
498
+ async function createProvider() {
499
+ const { OpenAICompatibleProvider } = await import('../llm/providers/openai-compatible');
502
500
  return new OpenAICompatibleProvider({
503
501
  name: 'test-compat',
504
502
  apiKey: 'test-key',
@@ -517,9 +515,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
517
515
  },
518
516
  ]);
519
517
 
520
- const mockCreate = mock(() => Promise.resolve(streamChunks));
518
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
521
519
 
522
- const provider = createProvider();
520
+ const provider = await createProvider();
523
521
  (provider as any).client = {
524
522
  chat: { completions: { create: mockCreate } },
525
523
  };
@@ -578,9 +576,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
578
576
  },
579
577
  ]);
580
578
 
581
- const mockCreate = mock(() => Promise.resolve(streamChunks));
579
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
582
580
 
583
- const provider = createProvider();
581
+ const provider = await createProvider();
584
582
  (provider as any).client = {
585
583
  chat: { completions: { create: mockCreate } },
586
584
  };
@@ -600,9 +598,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
600
598
  });
601
599
 
602
600
  test('fallback: when SDK stream creation throws, the error propagates', async () => {
603
- const mockCreate = mock(() => Promise.reject(new Error('Provider down')));
601
+ const mockCreate = vi.fn(() => Promise.reject(new Error('Provider down')));
604
602
 
605
- const provider = createProvider();
603
+ const provider = await createProvider();
606
604
  (provider as any).client = {
607
605
  chat: { completions: { create: mockCreate } },
608
606
  };
@@ -655,9 +653,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
655
653
  },
656
654
  ]);
657
655
 
658
- const mockCreate = mock(() => Promise.resolve(streamChunks));
656
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
659
657
 
660
- const provider = createProvider();
658
+ const provider = await createProvider();
661
659
  (provider as any).client = {
662
660
  chat: { completions: { create: mockCreate } },
663
661
  };
@@ -686,9 +684,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
686
684
  },
687
685
  ]);
688
686
 
689
- const mockCreate = mock(() => Promise.resolve(streamChunks));
687
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
690
688
 
691
- const provider = createProvider();
689
+ const provider = await createProvider();
692
690
  (provider as any).client = {
693
691
  chat: { completions: { create: mockCreate } },
694
692
  };
@@ -709,9 +707,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
709
707
  { choices: [{ delta: { content: 'done' }, finish_reason: 'stop' }] },
710
708
  ]);
711
709
 
712
- const mockCreate = mock(() => Promise.resolve(streamChunks));
710
+ const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
713
711
 
714
- const provider = createProvider();
712
+ const provider = await createProvider();
715
713
  (provider as any).client = {
716
714
  chat: { completions: { create: mockCreate } },
717
715
  };
@@ -730,3 +728,51 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
730
728
  expect(createArg.max_tokens).toBe(1024);
731
729
  });
732
730
  });
731
+
732
+ // ---------------------------------------------------------------------------
733
+ // PERF-2c: Unbuffered streaming in LLMRouter fallback branch
734
+ // ---------------------------------------------------------------------------
735
+
736
+ describe('PERF-2c: LLMRouter unbuffered streaming', () => {
737
+ it('streamWithTools fallback loop no longer buffers chunks before yielding', async () => {
738
+ const { readFileSync } = await import('node:fs');
739
+ const { join } = await import('node:path');
740
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
741
+ // Extract only the streamWithTools section (between the streamWithTools guards)
742
+ const swStart = src.indexOf('Use native streaming-with-tools if providers support it');
743
+ const swEnd = src.indexOf('If all providers with streamWithTools failed', swStart);
744
+ const swSection = swStart > 0 && swEnd > swStart ? src.slice(swStart, swEnd) : '';
745
+ // The streamWithTools fallback section should NOT contain buffering
746
+ expect(swSection).not.toContain('bufferedChunks');
747
+ // It should contain yield chunk directly
748
+ expect(swSection).toContain('yield chunk;');
749
+ });
750
+
751
+ it('router.ts yields each chunk immediately inside the for-await loop', async () => {
752
+ const { readFileSync } = await import('node:fs');
753
+ const { join } = await import('node:path');
754
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
755
+ // The yield statement should appear inside the for-await (before circuitBreaker.recordSuccess)
756
+ expect(src).toContain('yield chunk;');
757
+ // circuitBreaker.recordSuccess comes after the loop ends (done chunk received)
758
+ expect(src).toContain('circuitBreaker.recordSuccess(p.name);');
759
+ });
760
+
761
+ it('circuitBreaker.recordSuccess is called after stream ends (not before yield)', async () => {
762
+ const { readFileSync } = await import('node:fs');
763
+ const { join } = await import('node:path');
764
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
765
+ // Find the fallback for-loop block that has yield chunk
766
+ const yieldIdx = src.indexOf('yield chunk;');
767
+ const recordSuccessIdx = src.indexOf('circuitBreaker.recordSuccess(p.name);');
768
+ // recordSuccess should appear AFTER the yield in the source
769
+ expect(recordSuccessIdx).toBeGreaterThan(yieldIdx);
770
+ });
771
+
772
+ it('stream error path still calls circuitBreaker.recordFailure', async () => {
773
+ const { readFileSync } = await import('node:fs');
774
+ const { join } = await import('node:path');
775
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
776
+ expect(src).toContain('circuitBreaker.recordFailure(p.name);');
777
+ });
778
+ });
@@ -5,7 +5,7 @@
5
5
  * the @agent mention parser.
6
6
  */
7
7
 
8
- import { describe, test, expect } from 'bun:test';
8
+ import { describe, test, expect } from 'vitest';
9
9
  import {
10
10
  createSubagent,
11
11
  parseAgentMention,