@nathapp/nax 0.18.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 (459) hide show
  1. package/.gitlab-ci.yml +96 -0
  2. package/BRIEF.md +140 -0
  3. package/CHANGELOG.md +60 -0
  4. package/CLAUDE.md +159 -0
  5. package/README.md +373 -0
  6. package/US-007-IMPLEMENTATION.md +139 -0
  7. package/bin/nax.ts +930 -0
  8. package/biome.json +14 -0
  9. package/bun.lock +168 -0
  10. package/bunfig.toml +11 -0
  11. package/docs/20260216-fix-plan-context-review.md +56 -0
  12. package/docs/20260216-relentless-vs-ngent-comparison.md +208 -0
  13. package/docs/20260216-v02-plan.md +136 -0
  14. package/docs/20260216-v02-review.md +685 -0
  15. package/docs/20260217-dogfood-findings.md +56 -0
  16. package/docs/20260217-p2-plus-plan.md +117 -0
  17. package/docs/20260217-partial-fixes-plan.md +62 -0
  18. package/docs/20260217-plan-analyze-spec.md +117 -0
  19. package/docs/20260217-post-impl-review.md +1137 -0
  20. package/docs/20260217-quick-wins-plan.md +66 -0
  21. package/docs/20260217-split-runner-plan.md +75 -0
  22. package/docs/20260217-v03-impl-plan.md +80 -0
  23. package/docs/20260217-v03-post-impl-review.md +589 -0
  24. package/docs/20260217-v04-impl-plan.md +86 -0
  25. package/docs/20260217-v05-post-impl-review.md +850 -0
  26. package/docs/20260217-v06-post-impl-review.md +817 -0
  27. package/docs/20260218-adr003-port-plan.md +151 -0
  28. package/docs/20260218-review-adr003-verification.md +175 -0
  29. package/docs/20260219-fix-plan-bug16-19.md +79 -0
  30. package/docs/20260219-fix-plan-bug20-22.md +114 -0
  31. package/docs/20260219-plan-llm-routing.md +116 -0
  32. package/docs/20260219-review-bug20-22-fixes.md +135 -0
  33. package/docs/20260219-routing-baseline-keyword.md +63 -0
  34. package/docs/20260220-plan-structured-logging-p1.md +80 -0
  35. package/docs/20260220-plan-structured-logging-p2.md +37 -0
  36. package/docs/20260220-review-llm-routing.md +180 -0
  37. package/docs/20260220-review-post-fix-llm-routing.md +70 -0
  38. package/docs/20260221-fix-plan-relevantfiles-split.md +101 -0
  39. package/docs/20260221-fix-plan-routing-mode.md +125 -0
  40. package/docs/20260221-review-v0.9-implementation.md +379 -0
  41. package/docs/20260222-fix-plan-v091-routing-isolation.md +197 -0
  42. package/docs/20260223-fix-plan-prompt-audit.md +62 -0
  43. package/docs/20260224-nax-roadmap-phases.md +189 -0
  44. package/docs/20260225-phase2-llm-service-layer.md +401 -0
  45. package/docs/20260225-review-v0.10.1.md +187 -0
  46. package/docs/20260303-v010-implementation-plan.md +165 -0
  47. package/docs/CLAUDE.md.bak +191 -0
  48. package/docs/ROADMAP.md +165 -0
  49. package/docs/SPEC-rectification.md +0 -0
  50. package/docs/SPEC.md +324 -0
  51. package/docs/US-001-plugin-loading-verification.md +152 -0
  52. package/docs/architecture-analysis.md +1076 -0
  53. package/docs/bugs/BUG-21-escalation-null-attempts.md +48 -0
  54. package/docs/bugs-from-dogfood-run-c.md +243 -0
  55. package/docs/code-review-20260228.md +612 -0
  56. package/docs/code-review-v0.15.0.md +629 -0
  57. package/docs/hook-lifecycle-test-plan.md +149 -0
  58. package/docs/releases/v0.11.0-and-earlier.md +20 -0
  59. package/docs/releases/v0.12.0.md +15 -0
  60. package/docs/releases/v0.13.0.md +14 -0
  61. package/docs/releases/v0.14.0.md +20 -0
  62. package/docs/releases/v0.14.1.md +36 -0
  63. package/docs/releases/v0.14.2.md +51 -0
  64. package/docs/releases/v0.14.3.md +174 -0
  65. package/docs/releases/v0.14.4.md +94 -0
  66. package/docs/releases/v0.15.0.md +502 -0
  67. package/docs/releases/v0.15.1.md +170 -0
  68. package/docs/releases/v0.15.3.md +193 -0
  69. package/docs/specs/status-file-v0.10.1.md +812 -0
  70. package/docs/v0.10-global-config.md +206 -0
  71. package/docs/v0.10-plugin-system.md +415 -0
  72. package/docs/v0.10-prompt-optimizer.md +234 -0
  73. package/docs/v0.3-spec.md +244 -0
  74. package/docs/v0.4-spec.md +140 -0
  75. package/docs/v0.5-spec.md +237 -0
  76. package/docs/v0.6-spec.md +371 -0
  77. package/docs/v0.7-spec.md +177 -0
  78. package/docs/v0.8-llm-routing.md +206 -0
  79. package/docs/v0.8-structured-logging.md +132 -0
  80. package/docs/v0.9.3-prompt-audit.md +112 -0
  81. package/examples/plugins/console-reporter/index.test.ts +207 -0
  82. package/examples/plugins/console-reporter/index.ts +110 -0
  83. package/nax/config.json +147 -0
  84. package/nax/features/bugfix-v0171/prd.json +52 -0
  85. package/nax/features/config-management/prd.json +108 -0
  86. package/nax/features/config-management/progress.txt +5 -0
  87. package/nax/features/diagnose/acceptance.test.ts +412 -0
  88. package/nax/features/diagnose/prd.json +41 -0
  89. package/nax/features/orchestration-fixes/prd.json +89 -0
  90. package/nax/features/orchestration-fixes/progress.txt +1 -0
  91. package/nax/features/plugin-integration/US-007-VERIFICATION.md +259 -0
  92. package/nax/features/plugin-integration/prd.json +208 -0
  93. package/nax/features/plugin-integration/progress.txt +5 -0
  94. package/nax/features/precheck/prd.json +205 -0
  95. package/nax/features/precheck/progress.txt +15 -0
  96. package/nax/features/structured-logging/prd.json +199 -0
  97. package/nax/features/unlock/prd.json +36 -0
  98. package/package.json +47 -0
  99. package/src/acceptance/fix-generator.ts +348 -0
  100. package/src/acceptance/generator.ts +282 -0
  101. package/src/acceptance/index.ts +30 -0
  102. package/src/acceptance/types.ts +79 -0
  103. package/src/agents/claude-decompose.ts +169 -0
  104. package/src/agents/claude-plan.ts +139 -0
  105. package/src/agents/claude.ts +324 -0
  106. package/src/agents/cost.ts +268 -0
  107. package/src/agents/index.ts +13 -0
  108. package/src/agents/registry.ts +48 -0
  109. package/src/agents/types-extended.ts +133 -0
  110. package/src/agents/types.ts +113 -0
  111. package/src/agents/validation.ts +69 -0
  112. package/src/analyze/classifier.ts +305 -0
  113. package/src/analyze/index.ts +16 -0
  114. package/src/analyze/scanner.ts +175 -0
  115. package/src/analyze/types.ts +51 -0
  116. package/src/cli/accept.ts +108 -0
  117. package/src/cli/analyze-parser.ts +284 -0
  118. package/src/cli/analyze.ts +207 -0
  119. package/src/cli/config.ts +561 -0
  120. package/src/cli/constitution.ts +109 -0
  121. package/src/cli/diagnose-analysis.ts +159 -0
  122. package/src/cli/diagnose-formatter.ts +87 -0
  123. package/src/cli/diagnose.ts +203 -0
  124. package/src/cli/generate.ts +127 -0
  125. package/src/cli/index.ts +37 -0
  126. package/src/cli/init.ts +188 -0
  127. package/src/cli/interact.ts +295 -0
  128. package/src/cli/plan.ts +198 -0
  129. package/src/cli/plugins.ts +111 -0
  130. package/src/cli/prompts.ts +295 -0
  131. package/src/cli/runs.ts +174 -0
  132. package/src/cli/status-cost.ts +151 -0
  133. package/src/cli/status-features.ts +338 -0
  134. package/src/cli/status.ts +13 -0
  135. package/src/commands/common.ts +171 -0
  136. package/src/commands/diagnose.ts +17 -0
  137. package/src/commands/index.ts +8 -0
  138. package/src/commands/logs.ts +384 -0
  139. package/src/commands/precheck.ts +86 -0
  140. package/src/commands/unlock.ts +96 -0
  141. package/src/config/defaults.ts +160 -0
  142. package/src/config/index.ts +22 -0
  143. package/src/config/loader.ts +121 -0
  144. package/src/config/merger.ts +147 -0
  145. package/src/config/path-security.ts +121 -0
  146. package/src/config/paths.ts +27 -0
  147. package/src/config/schema.ts +56 -0
  148. package/src/config/schemas.ts +286 -0
  149. package/src/config/types.ts +423 -0
  150. package/src/config/validate.ts +103 -0
  151. package/src/constitution/generator.ts +191 -0
  152. package/src/constitution/generators/aider.ts +41 -0
  153. package/src/constitution/generators/claude.ts +35 -0
  154. package/src/constitution/generators/cursor.ts +36 -0
  155. package/src/constitution/generators/opencode.ts +38 -0
  156. package/src/constitution/generators/types.ts +33 -0
  157. package/src/constitution/generators/windsurf.ts +36 -0
  158. package/src/constitution/index.ts +10 -0
  159. package/src/constitution/loader.ts +133 -0
  160. package/src/constitution/types.ts +31 -0
  161. package/src/context/auto-detect.ts +227 -0
  162. package/src/context/builder.ts +246 -0
  163. package/src/context/elements.ts +83 -0
  164. package/src/context/formatter.ts +107 -0
  165. package/src/context/generator.ts +129 -0
  166. package/src/context/generators/aider.ts +34 -0
  167. package/src/context/generators/claude.ts +28 -0
  168. package/src/context/generators/cursor.ts +28 -0
  169. package/src/context/generators/opencode.ts +30 -0
  170. package/src/context/generators/windsurf.ts +28 -0
  171. package/src/context/greenfield.ts +114 -0
  172. package/src/context/index.ts +33 -0
  173. package/src/context/injector.ts +279 -0
  174. package/src/context/test-scanner.ts +370 -0
  175. package/src/context/types.ts +98 -0
  176. package/src/errors.ts +67 -0
  177. package/src/execution/batching.ts +157 -0
  178. package/src/execution/crash-recovery.ts +373 -0
  179. package/src/execution/escalation/escalation.ts +44 -0
  180. package/src/execution/escalation/index.ts +13 -0
  181. package/src/execution/escalation/tier-escalation.ts +295 -0
  182. package/src/execution/escalation/tier-outcome.ts +158 -0
  183. package/src/execution/helpers.ts +38 -0
  184. package/src/execution/index.ts +45 -0
  185. package/src/execution/lifecycle/acceptance-loop.ts +272 -0
  186. package/src/execution/lifecycle/headless-formatter.ts +85 -0
  187. package/src/execution/lifecycle/index.ts +12 -0
  188. package/src/execution/lifecycle/parallel-lifecycle.ts +101 -0
  189. package/src/execution/lifecycle/precheck-runner.ts +140 -0
  190. package/src/execution/lifecycle/run-cleanup.ts +81 -0
  191. package/src/execution/lifecycle/run-completion.ts +129 -0
  192. package/src/execution/lifecycle/run-initialization.ts +141 -0
  193. package/src/execution/lifecycle/run-lifecycle.ts +312 -0
  194. package/src/execution/lifecycle/run-setup.ts +204 -0
  195. package/src/execution/lifecycle/story-hooks.ts +38 -0
  196. package/src/execution/lifecycle/story-size-prompts.ts +123 -0
  197. package/src/execution/lock.ts +115 -0
  198. package/src/execution/parallel-executor.ts +216 -0
  199. package/src/execution/parallel.ts +400 -0
  200. package/src/execution/pid-registry.ts +280 -0
  201. package/src/execution/pipeline-result-handler.ts +388 -0
  202. package/src/execution/post-verify-rectification.ts +188 -0
  203. package/src/execution/post-verify.ts +274 -0
  204. package/src/execution/progress.ts +25 -0
  205. package/src/execution/prompts.ts +127 -0
  206. package/src/execution/queue-handler.ts +109 -0
  207. package/src/execution/rectification.ts +13 -0
  208. package/src/execution/runner.ts +377 -0
  209. package/src/execution/sequential-executor.ts +388 -0
  210. package/src/execution/status-file.ts +264 -0
  211. package/src/execution/status-writer.ts +139 -0
  212. package/src/execution/story-context.ts +229 -0
  213. package/src/execution/test-output-parser.ts +14 -0
  214. package/src/execution/verification.ts +72 -0
  215. package/src/hooks/index.ts +2 -0
  216. package/src/hooks/runner.ts +286 -0
  217. package/src/hooks/types.ts +67 -0
  218. package/src/interaction/chain.ts +154 -0
  219. package/src/interaction/index.ts +60 -0
  220. package/src/interaction/init.ts +83 -0
  221. package/src/interaction/plugins/auto.ts +217 -0
  222. package/src/interaction/plugins/cli.ts +300 -0
  223. package/src/interaction/plugins/telegram.ts +384 -0
  224. package/src/interaction/plugins/webhook.ts +258 -0
  225. package/src/interaction/state.ts +171 -0
  226. package/src/interaction/triggers.ts +229 -0
  227. package/src/interaction/types.ts +163 -0
  228. package/src/logger/formatters.ts +84 -0
  229. package/src/logger/index.ts +16 -0
  230. package/src/logger/logger.ts +298 -0
  231. package/src/logger/types.ts +48 -0
  232. package/src/logging/formatter.ts +355 -0
  233. package/src/logging/index.ts +22 -0
  234. package/src/logging/types.ts +93 -0
  235. package/src/metrics/aggregator.ts +190 -0
  236. package/src/metrics/index.ts +14 -0
  237. package/src/metrics/tracker.ts +200 -0
  238. package/src/metrics/types.ts +109 -0
  239. package/src/optimizer/index.ts +62 -0
  240. package/src/optimizer/noop.optimizer.ts +24 -0
  241. package/src/optimizer/rule-based.optimizer.ts +248 -0
  242. package/src/optimizer/types.ts +53 -0
  243. package/src/pipeline/events.ts +130 -0
  244. package/src/pipeline/index.ts +19 -0
  245. package/src/pipeline/runner.ts +161 -0
  246. package/src/pipeline/stages/acceptance.ts +197 -0
  247. package/src/pipeline/stages/completion.ts +99 -0
  248. package/src/pipeline/stages/constitution.ts +63 -0
  249. package/src/pipeline/stages/context.ts +117 -0
  250. package/src/pipeline/stages/execution.ts +194 -0
  251. package/src/pipeline/stages/index.ts +62 -0
  252. package/src/pipeline/stages/optimizer.ts +74 -0
  253. package/src/pipeline/stages/prompt.ts +57 -0
  254. package/src/pipeline/stages/queue-check.ts +103 -0
  255. package/src/pipeline/stages/review.ts +181 -0
  256. package/src/pipeline/stages/routing.ts +81 -0
  257. package/src/pipeline/stages/verify.ts +100 -0
  258. package/src/pipeline/types.ts +167 -0
  259. package/src/plugins/index.ts +31 -0
  260. package/src/plugins/loader.ts +287 -0
  261. package/src/plugins/registry.ts +168 -0
  262. package/src/plugins/types.ts +327 -0
  263. package/src/plugins/validator.ts +352 -0
  264. package/src/prd/index.ts +172 -0
  265. package/src/prd/types.ts +202 -0
  266. package/src/precheck/checks-blockers.ts +391 -0
  267. package/src/precheck/checks-warnings.ts +142 -0
  268. package/src/precheck/checks.ts +30 -0
  269. package/src/precheck/index.ts +247 -0
  270. package/src/precheck/story-size-gate.ts +144 -0
  271. package/src/precheck/types.ts +31 -0
  272. package/src/queue/index.ts +2 -0
  273. package/src/queue/manager.ts +254 -0
  274. package/src/queue/types.ts +54 -0
  275. package/src/review/index.ts +8 -0
  276. package/src/review/runner.ts +172 -0
  277. package/src/review/types.ts +66 -0
  278. package/src/routing/builder.ts +81 -0
  279. package/src/routing/chain.ts +74 -0
  280. package/src/routing/index.ts +16 -0
  281. package/src/routing/loader.ts +58 -0
  282. package/src/routing/router.ts +303 -0
  283. package/src/routing/strategies/adaptive.ts +215 -0
  284. package/src/routing/strategies/index.ts +8 -0
  285. package/src/routing/strategies/keyword.ts +163 -0
  286. package/src/routing/strategies/llm-prompts.ts +209 -0
  287. package/src/routing/strategies/llm.ts +235 -0
  288. package/src/routing/strategies/manual.ts +50 -0
  289. package/src/routing/strategy.ts +99 -0
  290. package/src/tdd/cleanup.ts +111 -0
  291. package/src/tdd/index.ts +23 -0
  292. package/src/tdd/isolation.ts +123 -0
  293. package/src/tdd/orchestrator.ts +383 -0
  294. package/src/tdd/prompts.ts +270 -0
  295. package/src/tdd/rectification-gate.ts +183 -0
  296. package/src/tdd/session-runner.ts +179 -0
  297. package/src/tdd/types.ts +81 -0
  298. package/src/tdd/verdict.ts +271 -0
  299. package/src/tui/App.tsx +265 -0
  300. package/src/tui/components/AgentPanel.tsx +75 -0
  301. package/src/tui/components/CostOverlay.tsx +118 -0
  302. package/src/tui/components/HelpOverlay.tsx +107 -0
  303. package/src/tui/components/StatusBar.tsx +63 -0
  304. package/src/tui/components/StoriesPanel.tsx +177 -0
  305. package/src/tui/hooks/useKeyboard.ts +142 -0
  306. package/src/tui/hooks/useLayout.ts +137 -0
  307. package/src/tui/hooks/usePipelineEvents.ts +183 -0
  308. package/src/tui/hooks/usePty.ts +194 -0
  309. package/src/tui/index.tsx +38 -0
  310. package/src/tui/types.ts +76 -0
  311. package/src/utils/git.ts +83 -0
  312. package/src/utils/queue-writer.ts +54 -0
  313. package/src/verification/executor.ts +235 -0
  314. package/src/verification/gate.ts +207 -0
  315. package/src/verification/index.ts +12 -0
  316. package/src/verification/parser.ts +230 -0
  317. package/src/verification/rectification.ts +108 -0
  318. package/src/verification/types.ts +113 -0
  319. package/src/worktree/dispatcher.ts +65 -0
  320. package/src/worktree/index.ts +2 -0
  321. package/src/worktree/manager.ts +187 -0
  322. package/src/worktree/merge.ts +301 -0
  323. package/src/worktree/types.ts +4 -0
  324. package/test/TEST_COVERAGE_US001.md +217 -0
  325. package/test/TEST_COVERAGE_US003.md +84 -0
  326. package/test/TEST_COVERAGE_US005.md +86 -0
  327. package/test/US-002-orchestrator.test.ts +246 -0
  328. package/test/acceptance/cm-003-default-view.test.ts +194 -0
  329. package/test/execution/pid-registry.test.ts +240 -0
  330. package/test/execution/post-verify.test.ts +224 -0
  331. package/test/helpers/timeout.ts +42 -0
  332. package/test/integration/US-002-TEST-SUMMARY.md +107 -0
  333. package/test/integration/US-003-TEST-SUMMARY.md +149 -0
  334. package/test/integration/US-004-TEST-SUMMARY.md +106 -0
  335. package/test/integration/US-005-TEST-SUMMARY.md +138 -0
  336. package/test/integration/US-007-TEST-SUMMARY.md +100 -0
  337. package/test/integration/agent-validation.test.ts +439 -0
  338. package/test/integration/analyze-integration.test.ts +261 -0
  339. package/test/integration/analyze-scanner.test.ts +131 -0
  340. package/test/integration/cli-config-default-edge-cases.test.ts +222 -0
  341. package/test/integration/cli-config-default-view.test.ts +229 -0
  342. package/test/integration/cli-config-diff.test.ts +460 -0
  343. package/test/integration/cli-config.test.ts +736 -0
  344. package/test/integration/cli-diagnose.test.ts +592 -0
  345. package/test/integration/cli-logs.test.ts +314 -0
  346. package/test/integration/cli-plugins.test.ts +678 -0
  347. package/test/integration/cli-precheck.test.ts +371 -0
  348. package/test/integration/cli-run-headless.test.ts +173 -0
  349. package/test/integration/cli.test.ts +75 -0
  350. package/test/integration/config/merger.test.ts +465 -0
  351. package/test/integration/config/paths.test.ts +51 -0
  352. package/test/integration/config-loader.test.ts +265 -0
  353. package/test/integration/config.test.ts +444 -0
  354. package/test/integration/context-integration.test.ts +702 -0
  355. package/test/integration/context-provider-injection.test.ts +506 -0
  356. package/test/integration/context-verification-integration.test.ts +295 -0
  357. package/test/integration/e2e.test.ts +896 -0
  358. package/test/integration/execution.test.ts +625 -0
  359. package/test/integration/helpers.test.ts +295 -0
  360. package/test/integration/hooks.test.ts +361 -0
  361. package/test/integration/interaction-chain-pipeline.test.ts +464 -0
  362. package/test/integration/isolation.test.ts +143 -0
  363. package/test/integration/logger.test.ts +461 -0
  364. package/test/integration/parallel.test.ts +250 -0
  365. package/test/integration/path-security.test.ts +173 -0
  366. package/test/integration/pipeline-acceptance.test.ts +302 -0
  367. package/test/integration/pipeline-events.test.ts +475 -0
  368. package/test/integration/pipeline.test.ts +658 -0
  369. package/test/integration/plan.test.ts +157 -0
  370. package/test/integration/plugin-routing.test.ts +921 -0
  371. package/test/integration/plugins/config-integration.test.ts +172 -0
  372. package/test/integration/plugins/config-resolution.test.ts +522 -0
  373. package/test/integration/plugins/loader.test.ts +641 -0
  374. package/test/integration/plugins/registry.test.ts +746 -0
  375. package/test/integration/plugins/validator.test.ts +563 -0
  376. package/test/integration/prd-pause.test.ts +205 -0
  377. package/test/integration/prd-resolvers.test.ts +185 -0
  378. package/test/integration/precheck-integration.test.ts +468 -0
  379. package/test/integration/precheck.test.ts +805 -0
  380. package/test/integration/progress.test.ts +34 -0
  381. package/test/integration/rectification-flow.test.ts +512 -0
  382. package/test/integration/reporter-lifecycle.test.ts +860 -0
  383. package/test/integration/review-config-commands.test.ts +319 -0
  384. package/test/integration/review-config-schema.test.ts +116 -0
  385. package/test/integration/review-plugin-integration.test.ts +722 -0
  386. package/test/integration/review.test.ts +149 -0
  387. package/test/integration/routing-stage-bug-021.test.ts +274 -0
  388. package/test/integration/routing-stage-greenfield.test.ts +286 -0
  389. package/test/integration/runner-config-plugins.test.ts +461 -0
  390. package/test/integration/runner-fixes.test.ts +399 -0
  391. package/test/integration/runner-plugin-integration.test.ts +543 -0
  392. package/test/integration/runner.test.ts +1679 -0
  393. package/test/integration/s5-greenfield-fallback.test.ts +297 -0
  394. package/test/integration/status-file-integration.test.ts +325 -0
  395. package/test/integration/status-file.test.ts +379 -0
  396. package/test/integration/status-writer.test.ts +345 -0
  397. package/test/integration/story-id-in-events.test.ts +273 -0
  398. package/test/integration/tdd-cleanup.test.ts +246 -0
  399. package/test/integration/tdd-orchestrator.test.ts +1762 -0
  400. package/test/integration/test-scanner.test.ts +403 -0
  401. package/test/integration/verification-asset-check.test.ts +142 -0
  402. package/test/integration/verify-stage.test.ts +275 -0
  403. package/test/integration/worktree/manager.test.ts +218 -0
  404. package/test/integration/worktree/merge.test.ts +341 -0
  405. package/test/manual/logging-formatter-demo.ts +158 -0
  406. package/test/ui/tui-agent-panel.test.tsx +99 -0
  407. package/test/ui/tui-controls.test.ts +334 -0
  408. package/test/ui/tui-cost-and-pty.test.ts +189 -0
  409. package/test/ui/tui-layout.test.ts +378 -0
  410. package/test/ui/tui-pty-integration.test.tsx +159 -0
  411. package/test/ui/tui-stories.test.ts +332 -0
  412. package/test/unit/acceptance.test.ts +186 -0
  413. package/test/unit/agent-stderr-capture.test.ts +146 -0
  414. package/test/unit/analyze-classifier.test.ts +215 -0
  415. package/test/unit/analyze.test.ts +224 -0
  416. package/test/unit/auto-detect.test.ts +249 -0
  417. package/test/unit/cli-status.test.ts +417 -0
  418. package/test/unit/commands/common.test.ts +320 -0
  419. package/test/unit/commands/logs.test.ts +416 -0
  420. package/test/unit/commands/unlock.test.ts +319 -0
  421. package/test/unit/constitution-generators.test.ts +160 -0
  422. package/test/unit/constitution.test.ts +209 -0
  423. package/test/unit/context.test.ts +1722 -0
  424. package/test/unit/cost.test.ts +231 -0
  425. package/test/unit/crash-recovery.test.ts +308 -0
  426. package/test/unit/escalation.test.ts +126 -0
  427. package/test/unit/execution-logging-stderr.test.ts +156 -0
  428. package/test/unit/execution-stage.test.ts +122 -0
  429. package/test/unit/fix-generator.test.ts +275 -0
  430. package/test/unit/formatters.test.ts +469 -0
  431. package/test/unit/greenfield.test.ts +179 -0
  432. package/test/unit/helpers.test.ts +317 -0
  433. package/test/unit/interaction/human-review-trigger.test.ts +164 -0
  434. package/test/unit/interaction-network-failures.test.ts +389 -0
  435. package/test/unit/interaction-plugins.test.ts +164 -0
  436. package/test/unit/isolation.test.ts +134 -0
  437. package/test/unit/logging/formatter.test.ts +455 -0
  438. package/test/unit/merge.test.ts +268 -0
  439. package/test/unit/metrics.test.ts +276 -0
  440. package/test/unit/optimizer/noop.optimizer.test.ts +125 -0
  441. package/test/unit/optimizer/rule-based.optimizer.test.ts +358 -0
  442. package/test/unit/prd-auto-default.test.ts +290 -0
  443. package/test/unit/prd-failure-category.test.ts +176 -0
  444. package/test/unit/prd-get-next-story.test.ts +186 -0
  445. package/test/unit/precheck-checks.test.ts +840 -0
  446. package/test/unit/precheck-story-size-gate.test.ts +287 -0
  447. package/test/unit/precheck-types.test.ts +142 -0
  448. package/test/unit/prompts.test.ts +475 -0
  449. package/test/unit/queue.test.ts +237 -0
  450. package/test/unit/rectification.test.ts +284 -0
  451. package/test/unit/registry.test.ts +287 -0
  452. package/test/unit/routing.test.ts +937 -0
  453. package/test/unit/run-lifecycle.test.ts +140 -0
  454. package/test/unit/storyid-events.test.ts +224 -0
  455. package/test/unit/tdd-verdict.test.ts +492 -0
  456. package/test/unit/test-output-parser.test.ts +377 -0
  457. package/test/unit/verdict.test.ts +324 -0
  458. package/test/unit/worktree-manager.test.ts +158 -0
  459. package/tsconfig.json +27 -0
@@ -0,0 +1,265 @@
1
+ /**
2
+ * Config Loader Tests
3
+ *
4
+ * Tests backward compatibility mapping of batchMode to mode enum.
5
+ *
6
+ * NOTE: The backward compat feature (loader.ts line 94-107) has a known limitation:
7
+ * The check `!("mode" in llm)` always fails because DEFAULT_CONFIG has mode:"hybrid",
8
+ * which gets merged before the backward compat check runs. This means batchMode->mode
9
+ * mapping never actually happens unless the user provides an invalid mode value that
10
+ * fails Zod validation, or unless they avoid the defaults entirely (global config).
11
+ *
12
+ * These tests document the ACTUAL behavior, not the intended behavior.
13
+ */
14
+
15
+ import { afterEach, beforeEach, describe, expect, test } from "bun:test";
16
+ import { mkdirSync, rmSync, writeFileSync } from "node:fs";
17
+ import { existsSync, renameSync } from "node:fs";
18
+ import { tmpdir } from "node:os";
19
+ import { homedir } from "node:os";
20
+ import { join } from "node:path";
21
+ import { globalConfigPath, loadConfig } from "../../src/config/loader";
22
+
23
+ describe("Config Loader - Backward Compatibility", () => {
24
+ let tempDir: string;
25
+ let globalBackup: string | null = null;
26
+
27
+ beforeEach(() => {
28
+ // Create a temporary test directory
29
+ tempDir = join(tmpdir(), `nax-test-${Date.now()}`);
30
+ mkdirSync(join(tempDir, "nax"), { recursive: true });
31
+
32
+ // Backup existing global config if present
33
+ const globalPath = globalConfigPath();
34
+ if (existsSync(globalPath)) {
35
+ globalBackup = `${globalPath}.test-backup-${Date.now()}`;
36
+ renameSync(globalPath, globalBackup);
37
+ }
38
+ });
39
+
40
+ afterEach(() => {
41
+ // Clean up temp directory
42
+ if (tempDir) {
43
+ rmSync(tempDir, { recursive: true, force: true });
44
+ }
45
+
46
+ // Restore global config if we backed it up
47
+ if (globalBackup && existsSync(globalBackup)) {
48
+ const globalPath = globalConfigPath();
49
+ if (existsSync(globalPath)) {
50
+ rmSync(globalPath);
51
+ }
52
+ renameSync(globalBackup, globalPath);
53
+ globalBackup = null;
54
+ }
55
+ });
56
+
57
+ test("batchMode:true maps to mode:one-shot (backward compat)", async () => {
58
+ const configPath = join(tempDir, "nax", "config.json");
59
+ const testConfig = {
60
+ routing: {
61
+ strategy: "llm",
62
+ llm: {
63
+ batchMode: true,
64
+ // mode not specified - should map from batchMode before default merge
65
+ },
66
+ },
67
+ };
68
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
69
+
70
+ const config = await loadConfig(join(tempDir, "nax"));
71
+
72
+ // applyBatchModeCompat runs on raw projConf before deepMerge with defaults,
73
+ // so batchMode:true correctly maps to mode:"one-shot" overriding DEFAULT_CONFIG's "hybrid"
74
+ expect(config.routing.llm?.mode).toBe("one-shot");
75
+ expect(config.routing.llm?.batchMode).toBe(true);
76
+ });
77
+
78
+ test("explicit mode takes precedence over batchMode", async () => {
79
+ const configPath = join(tempDir, "nax", "config.json");
80
+ const testConfig = {
81
+ routing: {
82
+ strategy: "llm",
83
+ llm: {
84
+ batchMode: true,
85
+ mode: "per-story", // Explicit mode overrides batchMode
86
+ },
87
+ },
88
+ };
89
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
90
+
91
+ const config = await loadConfig(join(tempDir, "nax"));
92
+
93
+ // Explicit mode is used, batchMode is ignored
94
+ expect(config.routing.llm?.mode).toBe("per-story");
95
+ expect(config.routing.llm?.batchMode).toBe(true);
96
+ });
97
+
98
+ test("rejects invalid batchMode value", async () => {
99
+ const configPath = join(tempDir, "nax", "config.json");
100
+ const testConfig = {
101
+ routing: {
102
+ strategy: "llm",
103
+ llm: {
104
+ batchMode: "yes", // Invalid - must be boolean
105
+ },
106
+ },
107
+ };
108
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
109
+
110
+ // Should throw validation error because Zod expects boolean
111
+ await expect(loadConfig(join(tempDir, "nax"))).rejects.toThrow("Invalid configuration");
112
+ });
113
+
114
+ test("uses DEFAULT_CONFIG mode when user config has no routing.llm", async () => {
115
+ const configPath = join(tempDir, "nax", "config.json");
116
+ const testConfig = {
117
+ routing: {
118
+ strategy: "keyword", // Different strategy, no llm config
119
+ },
120
+ };
121
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
122
+
123
+ const config = await loadConfig(join(tempDir, "nax"));
124
+
125
+ // Should get "hybrid" from DEFAULT_CONFIG.routing.llm.mode
126
+ expect(config.routing.llm?.mode).toBe("hybrid");
127
+ });
128
+ });
129
+
130
+ describe("Config Loader - Plugin Configuration (US-007)", () => {
131
+ let tempDir: string;
132
+ let globalBackup: string | null = null;
133
+
134
+ beforeEach(() => {
135
+ // Create a temporary test directory
136
+ tempDir = join(tmpdir(), `nax-test-plugins-${Date.now()}`);
137
+ mkdirSync(join(tempDir, "nax"), { recursive: true });
138
+
139
+ // Backup existing global config if present
140
+ const globalPath = globalConfigPath();
141
+ if (existsSync(globalPath)) {
142
+ globalBackup = `${globalPath}.test-backup-${Date.now()}`;
143
+ renameSync(globalPath, globalBackup);
144
+ }
145
+ });
146
+
147
+ afterEach(() => {
148
+ // Clean up temp directory
149
+ if (tempDir) {
150
+ rmSync(tempDir, { recursive: true, force: true });
151
+ }
152
+
153
+ // Restore global config if we backed it up
154
+ if (globalBackup && existsSync(globalBackup)) {
155
+ const globalPath = globalConfigPath();
156
+ if (existsSync(globalPath)) {
157
+ rmSync(globalPath);
158
+ }
159
+ renameSync(globalBackup, globalPath);
160
+ globalBackup = null;
161
+ }
162
+ });
163
+
164
+ test("loads plugins[] from config.json", async () => {
165
+ const configPath = join(tempDir, "nax", "config.json");
166
+ const testConfig = {
167
+ plugins: [
168
+ {
169
+ module: "./custom-plugins/my-plugin.ts",
170
+ config: {
171
+ apiKey: "test-123",
172
+ enabled: true,
173
+ },
174
+ },
175
+ {
176
+ module: "npm-plugin-package",
177
+ config: {
178
+ timeout: 5000,
179
+ },
180
+ },
181
+ ],
182
+ };
183
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
184
+
185
+ const config = await loadConfig(join(tempDir, "nax"));
186
+
187
+ // Verify plugins array is loaded
188
+ expect(config.plugins).toBeDefined();
189
+ expect(config.plugins).toHaveLength(2);
190
+ expect(config.plugins?.[0].module).toBe("./custom-plugins/my-plugin.ts");
191
+ expect(config.plugins?.[0].config).toEqual({
192
+ apiKey: "test-123",
193
+ enabled: true,
194
+ });
195
+ expect(config.plugins?.[1].module).toBe("npm-plugin-package");
196
+ expect(config.plugins?.[1].config).toEqual({
197
+ timeout: 5000,
198
+ });
199
+ });
200
+
201
+ test("handles missing plugins[] array (defaults to undefined)", async () => {
202
+ const configPath = join(tempDir, "nax", "config.json");
203
+ const testConfig = {
204
+ routing: {
205
+ strategy: "keyword",
206
+ },
207
+ };
208
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
209
+
210
+ const config = await loadConfig(join(tempDir, "nax"));
211
+
212
+ // plugins[] is optional, should be undefined if not provided
213
+ expect(config.plugins).toBeUndefined();
214
+ });
215
+
216
+ test("merges plugins[] from global and project config", async () => {
217
+ // Create global config with plugins
218
+ const globalPath = globalConfigPath();
219
+ mkdirSync(join(homedir(), ".nax"), { recursive: true });
220
+ const globalConfig = {
221
+ plugins: [
222
+ {
223
+ module: "global-plugin",
224
+ config: { global: true },
225
+ },
226
+ ],
227
+ };
228
+ writeFileSync(globalPath, JSON.stringify(globalConfig, null, 2));
229
+
230
+ // Create project config with plugins
231
+ const projectPath = join(tempDir, "nax", "config.json");
232
+ const projectConfig = {
233
+ plugins: [
234
+ {
235
+ module: "project-plugin",
236
+ config: { project: true },
237
+ },
238
+ ],
239
+ };
240
+ writeFileSync(projectPath, JSON.stringify(projectConfig, null, 2));
241
+
242
+ const config = await loadConfig(join(tempDir, "nax"));
243
+
244
+ // Verify plugins are merged (project overrides global)
245
+ expect(config.plugins).toBeDefined();
246
+ expect(config.plugins).toHaveLength(1);
247
+ expect(config.plugins?.[0].module).toBe("project-plugin");
248
+ });
249
+
250
+ test("validates plugin config entries have required fields", async () => {
251
+ const configPath = join(tempDir, "nax", "config.json");
252
+ const testConfig = {
253
+ plugins: [
254
+ {
255
+ // Missing module field
256
+ config: { foo: "bar" },
257
+ },
258
+ ],
259
+ };
260
+ writeFileSync(configPath, JSON.stringify(testConfig, null, 2));
261
+
262
+ // Should throw validation error
263
+ await expect(loadConfig(join(tempDir, "nax"))).rejects.toThrow("Invalid configuration");
264
+ });
265
+ });
@@ -0,0 +1,444 @@
1
+ import { describe, expect, test } from "bun:test";
2
+ import { DEFAULT_CONFIG, NaxConfigSchema } from "../../src/config/schema";
3
+ import type { TddStrategy, TestStrategy } from "../../src/config/schema";
4
+
5
+ describe("Config Validation", () => {
6
+ test("accepts valid default config", () => {
7
+ const result = NaxConfigSchema.safeParse(DEFAULT_CONFIG);
8
+ expect(result.success).toBe(true);
9
+ });
10
+
11
+ test("rejects invalid version", () => {
12
+ const config = {
13
+ ...DEFAULT_CONFIG,
14
+ version: 2, // Invalid version
15
+ };
16
+ const result = NaxConfigSchema.safeParse(config);
17
+ expect(result.success).toBe(false);
18
+ if (!result.success) {
19
+ const errorMessages = result.error.issues.map((e) => e.message);
20
+ expect(errorMessages.some((msg) => msg.includes("Invalid version"))).toBe(true);
21
+ }
22
+ });
23
+
24
+ test("rejects maxIterations <= 0", () => {
25
+ const config = {
26
+ ...DEFAULT_CONFIG,
27
+ execution: {
28
+ ...DEFAULT_CONFIG.execution,
29
+ maxIterations: 0,
30
+ },
31
+ };
32
+ const result = NaxConfigSchema.safeParse(config);
33
+ expect(result.success).toBe(false);
34
+ if (!result.success) {
35
+ const errorMessages = result.error.issues.map((e) => e.message);
36
+ expect(errorMessages.some((e) => e.includes("maxIterations must be > 0"))).toBe(true);
37
+ }
38
+ });
39
+
40
+ test("rejects costLimit <= 0", () => {
41
+ const config = {
42
+ ...DEFAULT_CONFIG,
43
+ execution: {
44
+ ...DEFAULT_CONFIG.execution,
45
+ costLimit: -1,
46
+ },
47
+ };
48
+ const result = NaxConfigSchema.safeParse(config);
49
+ expect(result.success).toBe(false);
50
+ if (!result.success) {
51
+ const errorMessages = result.error.issues.map((e) => e.message);
52
+ expect(errorMessages.some((e) => e.includes("costLimit must be > 0"))).toBe(true);
53
+ }
54
+ });
55
+
56
+ test("rejects sessionTimeoutSeconds <= 0", () => {
57
+ const config = {
58
+ ...DEFAULT_CONFIG,
59
+ execution: {
60
+ ...DEFAULT_CONFIG.execution,
61
+ sessionTimeoutSeconds: 0,
62
+ },
63
+ };
64
+ const result = NaxConfigSchema.safeParse(config);
65
+ expect(result.success).toBe(false);
66
+ if (!result.success) {
67
+ const errorMessages = result.error.issues.map((e) => e.message);
68
+ expect(errorMessages.some((e) => e.includes("sessionTimeoutSeconds must be > 0"))).toBe(true);
69
+ }
70
+ });
71
+
72
+ test("rejects empty defaultAgent", () => {
73
+ const config = {
74
+ ...DEFAULT_CONFIG,
75
+ autoMode: {
76
+ ...DEFAULT_CONFIG.autoMode,
77
+ defaultAgent: "",
78
+ },
79
+ };
80
+ const result = NaxConfigSchema.safeParse(config);
81
+ expect(result.success).toBe(false);
82
+ if (!result.success) {
83
+ const errorMessages = result.error.issues.map((e) => e.message);
84
+ expect(errorMessages.some((msg) => msg.includes("defaultAgent must be non-empty"))).toBe(true);
85
+ }
86
+ });
87
+
88
+ test("rejects whitespace-only defaultAgent", () => {
89
+ const config = {
90
+ ...DEFAULT_CONFIG,
91
+ autoMode: {
92
+ ...DEFAULT_CONFIG.autoMode,
93
+ defaultAgent: " ",
94
+ },
95
+ };
96
+ const result = NaxConfigSchema.safeParse(config);
97
+ expect(result.success).toBe(false);
98
+ // Zod's min(1) validation will trim and reject whitespace
99
+ });
100
+
101
+ test("rejects empty tierOrder", () => {
102
+ const config = {
103
+ ...DEFAULT_CONFIG,
104
+ autoMode: {
105
+ ...DEFAULT_CONFIG.autoMode,
106
+ escalation: {
107
+ ...DEFAULT_CONFIG.autoMode.escalation,
108
+ tierOrder: [],
109
+ },
110
+ },
111
+ };
112
+ const result = NaxConfigSchema.safeParse(config);
113
+ expect(result.success).toBe(false);
114
+ if (!result.success) {
115
+ const errorMessages = result.error.issues.map((e) => e.message);
116
+ expect(errorMessages.some((e) => e.includes("tierOrder must have at least one tier"))).toBe(true);
117
+ }
118
+ });
119
+
120
+ test("rejects tierOrder with attempts out of range", () => {
121
+ const config = {
122
+ ...DEFAULT_CONFIG,
123
+ autoMode: {
124
+ ...DEFAULT_CONFIG.autoMode,
125
+ escalation: {
126
+ ...DEFAULT_CONFIG.autoMode.escalation,
127
+ tierOrder: [{ tier: "fast", attempts: 0 }],
128
+ },
129
+ },
130
+ };
131
+ const result = NaxConfigSchema.safeParse(config);
132
+ expect(result.success).toBe(false);
133
+ });
134
+
135
+ test("accepts custom tier names in tierOrder", () => {
136
+ const config = {
137
+ ...DEFAULT_CONFIG,
138
+ autoMode: {
139
+ ...DEFAULT_CONFIG.autoMode,
140
+ escalation: {
141
+ ...DEFAULT_CONFIG.autoMode.escalation,
142
+ tierOrder: [
143
+ { tier: "free", attempts: 10 },
144
+ { tier: "ultra", attempts: 1 },
145
+ ],
146
+ },
147
+ },
148
+ };
149
+ const result = NaxConfigSchema.safeParse(config);
150
+ expect(result.success).toBe(true);
151
+ });
152
+
153
+ test("collects multiple validation errors", () => {
154
+ const config = {
155
+ ...DEFAULT_CONFIG,
156
+ version: 99,
157
+ execution: {
158
+ ...DEFAULT_CONFIG.execution,
159
+ maxIterations: 0,
160
+ costLimit: -5,
161
+ },
162
+ autoMode: {
163
+ ...DEFAULT_CONFIG.autoMode,
164
+ defaultAgent: "",
165
+ },
166
+ };
167
+ const result = NaxConfigSchema.safeParse(config);
168
+ expect(result.success).toBe(false);
169
+ if (!result.success) {
170
+ expect(result.error.issues.length).toBeGreaterThanOrEqual(4);
171
+ }
172
+ });
173
+
174
+ test("accepts any non-empty string as complexityRouting tier", () => {
175
+ // Tiers are now extensible (z.string), validated against models at runtime
176
+ const config = {
177
+ ...DEFAULT_CONFIG,
178
+ autoMode: {
179
+ ...DEFAULT_CONFIG.autoMode,
180
+ complexityRouting: {
181
+ ...DEFAULT_CONFIG.autoMode.complexityRouting,
182
+ simple: "custom-tier",
183
+ },
184
+ },
185
+ };
186
+ const result = NaxConfigSchema.safeParse(config);
187
+ expect(result.success).toBe(true);
188
+ });
189
+
190
+ test("rejects empty string as complexityRouting tier", () => {
191
+ const config = {
192
+ ...DEFAULT_CONFIG,
193
+ autoMode: {
194
+ ...DEFAULT_CONFIG.autoMode,
195
+ complexityRouting: {
196
+ ...DEFAULT_CONFIG.autoMode.complexityRouting,
197
+ simple: "",
198
+ },
199
+ },
200
+ };
201
+ const result = NaxConfigSchema.safeParse(config);
202
+ expect(result.success).toBe(false);
203
+ });
204
+
205
+ test("accepts all valid tiers in complexityRouting", () => {
206
+ const config = {
207
+ ...DEFAULT_CONFIG,
208
+ autoMode: {
209
+ ...DEFAULT_CONFIG.autoMode,
210
+ complexityRouting: {
211
+ simple: "fast",
212
+ medium: "balanced",
213
+ complex: "powerful",
214
+ expert: "powerful",
215
+ },
216
+ },
217
+ };
218
+ const result = NaxConfigSchema.safeParse(config);
219
+ expect(result.success).toBe(true);
220
+ });
221
+
222
+ test("validates verificationTimeoutSeconds bounds", () => {
223
+ const tooLow = {
224
+ ...DEFAULT_CONFIG,
225
+ execution: { ...DEFAULT_CONFIG.execution, verificationTimeoutSeconds: 0 },
226
+ };
227
+ expect(NaxConfigSchema.safeParse(tooLow).success).toBe(false);
228
+
229
+ const tooHigh = {
230
+ ...DEFAULT_CONFIG,
231
+ execution: { ...DEFAULT_CONFIG.execution, verificationTimeoutSeconds: 7200 },
232
+ };
233
+ expect(NaxConfigSchema.safeParse(tooHigh).success).toBe(false);
234
+
235
+ const valid = {
236
+ ...DEFAULT_CONFIG,
237
+ execution: { ...DEFAULT_CONFIG.execution, verificationTimeoutSeconds: 120 },
238
+ };
239
+ expect(NaxConfigSchema.safeParse(valid).success).toBe(true);
240
+ });
241
+
242
+ test("validates quality config extensions", () => {
243
+ const config = {
244
+ ...DEFAULT_CONFIG,
245
+ quality: {
246
+ ...DEFAULT_CONFIG.quality,
247
+ forceExit: true,
248
+ detectOpenHandles: false,
249
+ detectOpenHandlesRetries: 3,
250
+ gracePeriodMs: 10000,
251
+ drainTimeoutMs: 5000,
252
+ shell: "/bin/bash",
253
+ stripEnvVars: ["CLAUDECODE", "CUSTOM_VAR"],
254
+ environmentalEscalationDivisor: 3,
255
+ },
256
+ };
257
+ const result = NaxConfigSchema.safeParse(config);
258
+ expect(result.success).toBe(true);
259
+ });
260
+ });
261
+
262
+ describe("LLM Routing Mode Config", () => {
263
+ test("accepts one-shot mode", () => {
264
+ const config = {
265
+ ...DEFAULT_CONFIG,
266
+ routing: {
267
+ ...DEFAULT_CONFIG.routing,
268
+ llm: {
269
+ mode: "one-shot" as const,
270
+ },
271
+ },
272
+ };
273
+ const result = NaxConfigSchema.safeParse(config);
274
+ expect(result.success).toBe(true);
275
+ if (result.success) {
276
+ expect(result.data.routing.llm?.mode).toBe("one-shot");
277
+ }
278
+ });
279
+
280
+ test("accepts per-story mode", () => {
281
+ const config = {
282
+ ...DEFAULT_CONFIG,
283
+ routing: {
284
+ ...DEFAULT_CONFIG.routing,
285
+ llm: {
286
+ mode: "per-story" as const,
287
+ },
288
+ },
289
+ };
290
+ const result = NaxConfigSchema.safeParse(config);
291
+ expect(result.success).toBe(true);
292
+ if (result.success) {
293
+ expect(result.data.routing.llm?.mode).toBe("per-story");
294
+ }
295
+ });
296
+
297
+ test("accepts hybrid mode", () => {
298
+ const config = {
299
+ ...DEFAULT_CONFIG,
300
+ routing: {
301
+ ...DEFAULT_CONFIG.routing,
302
+ llm: {
303
+ mode: "hybrid" as const,
304
+ },
305
+ },
306
+ };
307
+ const result = NaxConfigSchema.safeParse(config);
308
+ expect(result.success).toBe(true);
309
+ if (result.success) {
310
+ expect(result.data.routing.llm?.mode).toBe("hybrid");
311
+ }
312
+ });
313
+
314
+ test("rejects invalid mode value", () => {
315
+ const config = {
316
+ ...DEFAULT_CONFIG,
317
+ routing: {
318
+ ...DEFAULT_CONFIG.routing,
319
+ llm: {
320
+ mode: "ultra-batch",
321
+ },
322
+ },
323
+ };
324
+ const result = NaxConfigSchema.safeParse(config);
325
+ expect(result.success).toBe(false);
326
+ });
327
+
328
+ test("defaults to hybrid when mode not specified", () => {
329
+ const config = {
330
+ ...DEFAULT_CONFIG,
331
+ routing: {
332
+ ...DEFAULT_CONFIG.routing,
333
+ strategy: "llm" as const,
334
+ llm: {
335
+ cacheDecisions: true,
336
+ },
337
+ },
338
+ };
339
+ const result = NaxConfigSchema.safeParse(config);
340
+ expect(result.success).toBe(true);
341
+ // Default is applied by loader, schema allows undefined
342
+ expect(result.data.routing.llm?.mode).toBeUndefined();
343
+ });
344
+
345
+ test("accepts deprecated batchMode alongside mode", () => {
346
+ const config = {
347
+ ...DEFAULT_CONFIG,
348
+ routing: {
349
+ ...DEFAULT_CONFIG.routing,
350
+ llm: {
351
+ mode: "one-shot" as const,
352
+ batchMode: true,
353
+ },
354
+ },
355
+ };
356
+ const result = NaxConfigSchema.safeParse(config);
357
+ expect(result.success).toBe(true);
358
+ if (result.success) {
359
+ expect(result.data.routing.llm?.mode).toBe("one-shot");
360
+ expect(result.data.routing.llm?.batchMode).toBe(true);
361
+ }
362
+ });
363
+ });
364
+
365
+ describe("TDD Strategy Config", () => {
366
+ test("TestStrategy type includes three-session-tdd-lite", () => {
367
+ // Type-level check: ensure 'three-session-tdd-lite' is assignable to TestStrategy
368
+ const strategy: TestStrategy = "three-session-tdd-lite";
369
+ expect(strategy).toBe("three-session-tdd-lite");
370
+ });
371
+
372
+ test("TddStrategy type alias covers all four values", () => {
373
+ const strategies: TddStrategy[] = ["auto", "strict", "lite", "off"];
374
+ expect(strategies).toHaveLength(4);
375
+ });
376
+
377
+ test("default config has strategy: auto", () => {
378
+ expect(DEFAULT_CONFIG.tdd.strategy).toBe("auto");
379
+ });
380
+
381
+ test("default config parses successfully", () => {
382
+ const result = NaxConfigSchema.safeParse(DEFAULT_CONFIG);
383
+ expect(result.success).toBe(true);
384
+ if (result.success) {
385
+ expect(result.data.tdd.strategy).toBe("auto");
386
+ }
387
+ });
388
+
389
+ test("accepts strategy: strict", () => {
390
+ const config = {
391
+ ...DEFAULT_CONFIG,
392
+ tdd: { ...DEFAULT_CONFIG.tdd, strategy: "strict" as const },
393
+ };
394
+ const result = NaxConfigSchema.safeParse(config);
395
+ expect(result.success).toBe(true);
396
+ if (result.success) {
397
+ expect(result.data.tdd.strategy).toBe("strict");
398
+ }
399
+ });
400
+
401
+ test("accepts strategy: lite", () => {
402
+ const config = {
403
+ ...DEFAULT_CONFIG,
404
+ tdd: { ...DEFAULT_CONFIG.tdd, strategy: "lite" as const },
405
+ };
406
+ const result = NaxConfigSchema.safeParse(config);
407
+ expect(result.success).toBe(true);
408
+ if (result.success) {
409
+ expect(result.data.tdd.strategy).toBe("lite");
410
+ }
411
+ });
412
+
413
+ test("accepts strategy: off", () => {
414
+ const config = {
415
+ ...DEFAULT_CONFIG,
416
+ tdd: { ...DEFAULT_CONFIG.tdd, strategy: "off" as const },
417
+ };
418
+ const result = NaxConfigSchema.safeParse(config);
419
+ expect(result.success).toBe(true);
420
+ if (result.success) {
421
+ expect(result.data.tdd.strategy).toBe("off");
422
+ }
423
+ });
424
+
425
+ test("rejects invalid strategy value", () => {
426
+ const config = {
427
+ ...DEFAULT_CONFIG,
428
+ tdd: { ...DEFAULT_CONFIG.tdd, strategy: "invalid-strategy" },
429
+ };
430
+ const result = NaxConfigSchema.safeParse(config);
431
+ expect(result.success).toBe(false);
432
+ });
433
+
434
+ test("backward compat: config without strategy field defaults to auto", () => {
435
+ // Simulate a config file that was written before strategy was added (no strategy key)
436
+ const { strategy: _omitted, ...tddWithoutStrategy } = DEFAULT_CONFIG.tdd;
437
+ const config = { ...DEFAULT_CONFIG, tdd: tddWithoutStrategy };
438
+ const result = NaxConfigSchema.safeParse(config);
439
+ expect(result.success).toBe(true);
440
+ if (result.success) {
441
+ expect(result.data.tdd.strategy).toBe("auto");
442
+ }
443
+ });
444
+ });