@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.
- package/.gitlab-ci.yml +96 -0
- package/BRIEF.md +140 -0
- package/CHANGELOG.md +60 -0
- package/CLAUDE.md +159 -0
- package/README.md +373 -0
- package/US-007-IMPLEMENTATION.md +139 -0
- package/bin/nax.ts +930 -0
- package/biome.json +14 -0
- package/bun.lock +168 -0
- package/bunfig.toml +11 -0
- package/docs/20260216-fix-plan-context-review.md +56 -0
- package/docs/20260216-relentless-vs-ngent-comparison.md +208 -0
- package/docs/20260216-v02-plan.md +136 -0
- package/docs/20260216-v02-review.md +685 -0
- package/docs/20260217-dogfood-findings.md +56 -0
- package/docs/20260217-p2-plus-plan.md +117 -0
- package/docs/20260217-partial-fixes-plan.md +62 -0
- package/docs/20260217-plan-analyze-spec.md +117 -0
- package/docs/20260217-post-impl-review.md +1137 -0
- package/docs/20260217-quick-wins-plan.md +66 -0
- package/docs/20260217-split-runner-plan.md +75 -0
- package/docs/20260217-v03-impl-plan.md +80 -0
- package/docs/20260217-v03-post-impl-review.md +589 -0
- package/docs/20260217-v04-impl-plan.md +86 -0
- package/docs/20260217-v05-post-impl-review.md +850 -0
- package/docs/20260217-v06-post-impl-review.md +817 -0
- package/docs/20260218-adr003-port-plan.md +151 -0
- package/docs/20260218-review-adr003-verification.md +175 -0
- package/docs/20260219-fix-plan-bug16-19.md +79 -0
- package/docs/20260219-fix-plan-bug20-22.md +114 -0
- package/docs/20260219-plan-llm-routing.md +116 -0
- package/docs/20260219-review-bug20-22-fixes.md +135 -0
- package/docs/20260219-routing-baseline-keyword.md +63 -0
- package/docs/20260220-plan-structured-logging-p1.md +80 -0
- package/docs/20260220-plan-structured-logging-p2.md +37 -0
- package/docs/20260220-review-llm-routing.md +180 -0
- package/docs/20260220-review-post-fix-llm-routing.md +70 -0
- package/docs/20260221-fix-plan-relevantfiles-split.md +101 -0
- package/docs/20260221-fix-plan-routing-mode.md +125 -0
- package/docs/20260221-review-v0.9-implementation.md +379 -0
- package/docs/20260222-fix-plan-v091-routing-isolation.md +197 -0
- package/docs/20260223-fix-plan-prompt-audit.md +62 -0
- package/docs/20260224-nax-roadmap-phases.md +189 -0
- package/docs/20260225-phase2-llm-service-layer.md +401 -0
- package/docs/20260225-review-v0.10.1.md +187 -0
- package/docs/20260303-v010-implementation-plan.md +165 -0
- package/docs/CLAUDE.md.bak +191 -0
- package/docs/ROADMAP.md +165 -0
- package/docs/SPEC-rectification.md +0 -0
- package/docs/SPEC.md +324 -0
- package/docs/US-001-plugin-loading-verification.md +152 -0
- package/docs/architecture-analysis.md +1076 -0
- package/docs/bugs/BUG-21-escalation-null-attempts.md +48 -0
- package/docs/bugs-from-dogfood-run-c.md +243 -0
- package/docs/code-review-20260228.md +612 -0
- package/docs/code-review-v0.15.0.md +629 -0
- package/docs/hook-lifecycle-test-plan.md +149 -0
- package/docs/releases/v0.11.0-and-earlier.md +20 -0
- package/docs/releases/v0.12.0.md +15 -0
- package/docs/releases/v0.13.0.md +14 -0
- package/docs/releases/v0.14.0.md +20 -0
- package/docs/releases/v0.14.1.md +36 -0
- package/docs/releases/v0.14.2.md +51 -0
- package/docs/releases/v0.14.3.md +174 -0
- package/docs/releases/v0.14.4.md +94 -0
- package/docs/releases/v0.15.0.md +502 -0
- package/docs/releases/v0.15.1.md +170 -0
- package/docs/releases/v0.15.3.md +193 -0
- package/docs/specs/status-file-v0.10.1.md +812 -0
- package/docs/v0.10-global-config.md +206 -0
- package/docs/v0.10-plugin-system.md +415 -0
- package/docs/v0.10-prompt-optimizer.md +234 -0
- package/docs/v0.3-spec.md +244 -0
- package/docs/v0.4-spec.md +140 -0
- package/docs/v0.5-spec.md +237 -0
- package/docs/v0.6-spec.md +371 -0
- package/docs/v0.7-spec.md +177 -0
- package/docs/v0.8-llm-routing.md +206 -0
- package/docs/v0.8-structured-logging.md +132 -0
- package/docs/v0.9.3-prompt-audit.md +112 -0
- package/examples/plugins/console-reporter/index.test.ts +207 -0
- package/examples/plugins/console-reporter/index.ts +110 -0
- package/nax/config.json +147 -0
- package/nax/features/bugfix-v0171/prd.json +52 -0
- package/nax/features/config-management/prd.json +108 -0
- package/nax/features/config-management/progress.txt +5 -0
- package/nax/features/diagnose/acceptance.test.ts +412 -0
- package/nax/features/diagnose/prd.json +41 -0
- package/nax/features/orchestration-fixes/prd.json +89 -0
- package/nax/features/orchestration-fixes/progress.txt +1 -0
- package/nax/features/plugin-integration/US-007-VERIFICATION.md +259 -0
- package/nax/features/plugin-integration/prd.json +208 -0
- package/nax/features/plugin-integration/progress.txt +5 -0
- package/nax/features/precheck/prd.json +205 -0
- package/nax/features/precheck/progress.txt +15 -0
- package/nax/features/structured-logging/prd.json +199 -0
- package/nax/features/unlock/prd.json +36 -0
- package/package.json +47 -0
- package/src/acceptance/fix-generator.ts +348 -0
- package/src/acceptance/generator.ts +282 -0
- package/src/acceptance/index.ts +30 -0
- package/src/acceptance/types.ts +79 -0
- package/src/agents/claude-decompose.ts +169 -0
- package/src/agents/claude-plan.ts +139 -0
- package/src/agents/claude.ts +324 -0
- package/src/agents/cost.ts +268 -0
- package/src/agents/index.ts +13 -0
- package/src/agents/registry.ts +48 -0
- package/src/agents/types-extended.ts +133 -0
- package/src/agents/types.ts +113 -0
- package/src/agents/validation.ts +69 -0
- package/src/analyze/classifier.ts +305 -0
- package/src/analyze/index.ts +16 -0
- package/src/analyze/scanner.ts +175 -0
- package/src/analyze/types.ts +51 -0
- package/src/cli/accept.ts +108 -0
- package/src/cli/analyze-parser.ts +284 -0
- package/src/cli/analyze.ts +207 -0
- package/src/cli/config.ts +561 -0
- package/src/cli/constitution.ts +109 -0
- package/src/cli/diagnose-analysis.ts +159 -0
- package/src/cli/diagnose-formatter.ts +87 -0
- package/src/cli/diagnose.ts +203 -0
- package/src/cli/generate.ts +127 -0
- package/src/cli/index.ts +37 -0
- package/src/cli/init.ts +188 -0
- package/src/cli/interact.ts +295 -0
- package/src/cli/plan.ts +198 -0
- package/src/cli/plugins.ts +111 -0
- package/src/cli/prompts.ts +295 -0
- package/src/cli/runs.ts +174 -0
- package/src/cli/status-cost.ts +151 -0
- package/src/cli/status-features.ts +338 -0
- package/src/cli/status.ts +13 -0
- package/src/commands/common.ts +171 -0
- package/src/commands/diagnose.ts +17 -0
- package/src/commands/index.ts +8 -0
- package/src/commands/logs.ts +384 -0
- package/src/commands/precheck.ts +86 -0
- package/src/commands/unlock.ts +96 -0
- package/src/config/defaults.ts +160 -0
- package/src/config/index.ts +22 -0
- package/src/config/loader.ts +121 -0
- package/src/config/merger.ts +147 -0
- package/src/config/path-security.ts +121 -0
- package/src/config/paths.ts +27 -0
- package/src/config/schema.ts +56 -0
- package/src/config/schemas.ts +286 -0
- package/src/config/types.ts +423 -0
- package/src/config/validate.ts +103 -0
- package/src/constitution/generator.ts +191 -0
- package/src/constitution/generators/aider.ts +41 -0
- package/src/constitution/generators/claude.ts +35 -0
- package/src/constitution/generators/cursor.ts +36 -0
- package/src/constitution/generators/opencode.ts +38 -0
- package/src/constitution/generators/types.ts +33 -0
- package/src/constitution/generators/windsurf.ts +36 -0
- package/src/constitution/index.ts +10 -0
- package/src/constitution/loader.ts +133 -0
- package/src/constitution/types.ts +31 -0
- package/src/context/auto-detect.ts +227 -0
- package/src/context/builder.ts +246 -0
- package/src/context/elements.ts +83 -0
- package/src/context/formatter.ts +107 -0
- package/src/context/generator.ts +129 -0
- package/src/context/generators/aider.ts +34 -0
- package/src/context/generators/claude.ts +28 -0
- package/src/context/generators/cursor.ts +28 -0
- package/src/context/generators/opencode.ts +30 -0
- package/src/context/generators/windsurf.ts +28 -0
- package/src/context/greenfield.ts +114 -0
- package/src/context/index.ts +33 -0
- package/src/context/injector.ts +279 -0
- package/src/context/test-scanner.ts +370 -0
- package/src/context/types.ts +98 -0
- package/src/errors.ts +67 -0
- package/src/execution/batching.ts +157 -0
- package/src/execution/crash-recovery.ts +373 -0
- package/src/execution/escalation/escalation.ts +44 -0
- package/src/execution/escalation/index.ts +13 -0
- package/src/execution/escalation/tier-escalation.ts +295 -0
- package/src/execution/escalation/tier-outcome.ts +158 -0
- package/src/execution/helpers.ts +38 -0
- package/src/execution/index.ts +45 -0
- package/src/execution/lifecycle/acceptance-loop.ts +272 -0
- package/src/execution/lifecycle/headless-formatter.ts +85 -0
- package/src/execution/lifecycle/index.ts +12 -0
- package/src/execution/lifecycle/parallel-lifecycle.ts +101 -0
- package/src/execution/lifecycle/precheck-runner.ts +140 -0
- package/src/execution/lifecycle/run-cleanup.ts +81 -0
- package/src/execution/lifecycle/run-completion.ts +129 -0
- package/src/execution/lifecycle/run-initialization.ts +141 -0
- package/src/execution/lifecycle/run-lifecycle.ts +312 -0
- package/src/execution/lifecycle/run-setup.ts +204 -0
- package/src/execution/lifecycle/story-hooks.ts +38 -0
- package/src/execution/lifecycle/story-size-prompts.ts +123 -0
- package/src/execution/lock.ts +115 -0
- package/src/execution/parallel-executor.ts +216 -0
- package/src/execution/parallel.ts +400 -0
- package/src/execution/pid-registry.ts +280 -0
- package/src/execution/pipeline-result-handler.ts +388 -0
- package/src/execution/post-verify-rectification.ts +188 -0
- package/src/execution/post-verify.ts +274 -0
- package/src/execution/progress.ts +25 -0
- package/src/execution/prompts.ts +127 -0
- package/src/execution/queue-handler.ts +109 -0
- package/src/execution/rectification.ts +13 -0
- package/src/execution/runner.ts +377 -0
- package/src/execution/sequential-executor.ts +388 -0
- package/src/execution/status-file.ts +264 -0
- package/src/execution/status-writer.ts +139 -0
- package/src/execution/story-context.ts +229 -0
- package/src/execution/test-output-parser.ts +14 -0
- package/src/execution/verification.ts +72 -0
- package/src/hooks/index.ts +2 -0
- package/src/hooks/runner.ts +286 -0
- package/src/hooks/types.ts +67 -0
- package/src/interaction/chain.ts +154 -0
- package/src/interaction/index.ts +60 -0
- package/src/interaction/init.ts +83 -0
- package/src/interaction/plugins/auto.ts +217 -0
- package/src/interaction/plugins/cli.ts +300 -0
- package/src/interaction/plugins/telegram.ts +384 -0
- package/src/interaction/plugins/webhook.ts +258 -0
- package/src/interaction/state.ts +171 -0
- package/src/interaction/triggers.ts +229 -0
- package/src/interaction/types.ts +163 -0
- package/src/logger/formatters.ts +84 -0
- package/src/logger/index.ts +16 -0
- package/src/logger/logger.ts +298 -0
- package/src/logger/types.ts +48 -0
- package/src/logging/formatter.ts +355 -0
- package/src/logging/index.ts +22 -0
- package/src/logging/types.ts +93 -0
- package/src/metrics/aggregator.ts +190 -0
- package/src/metrics/index.ts +14 -0
- package/src/metrics/tracker.ts +200 -0
- package/src/metrics/types.ts +109 -0
- package/src/optimizer/index.ts +62 -0
- package/src/optimizer/noop.optimizer.ts +24 -0
- package/src/optimizer/rule-based.optimizer.ts +248 -0
- package/src/optimizer/types.ts +53 -0
- package/src/pipeline/events.ts +130 -0
- package/src/pipeline/index.ts +19 -0
- package/src/pipeline/runner.ts +161 -0
- package/src/pipeline/stages/acceptance.ts +197 -0
- package/src/pipeline/stages/completion.ts +99 -0
- package/src/pipeline/stages/constitution.ts +63 -0
- package/src/pipeline/stages/context.ts +117 -0
- package/src/pipeline/stages/execution.ts +194 -0
- package/src/pipeline/stages/index.ts +62 -0
- package/src/pipeline/stages/optimizer.ts +74 -0
- package/src/pipeline/stages/prompt.ts +57 -0
- package/src/pipeline/stages/queue-check.ts +103 -0
- package/src/pipeline/stages/review.ts +181 -0
- package/src/pipeline/stages/routing.ts +81 -0
- package/src/pipeline/stages/verify.ts +100 -0
- package/src/pipeline/types.ts +167 -0
- package/src/plugins/index.ts +31 -0
- package/src/plugins/loader.ts +287 -0
- package/src/plugins/registry.ts +168 -0
- package/src/plugins/types.ts +327 -0
- package/src/plugins/validator.ts +352 -0
- package/src/prd/index.ts +172 -0
- package/src/prd/types.ts +202 -0
- package/src/precheck/checks-blockers.ts +391 -0
- package/src/precheck/checks-warnings.ts +142 -0
- package/src/precheck/checks.ts +30 -0
- package/src/precheck/index.ts +247 -0
- package/src/precheck/story-size-gate.ts +144 -0
- package/src/precheck/types.ts +31 -0
- package/src/queue/index.ts +2 -0
- package/src/queue/manager.ts +254 -0
- package/src/queue/types.ts +54 -0
- package/src/review/index.ts +8 -0
- package/src/review/runner.ts +172 -0
- package/src/review/types.ts +66 -0
- package/src/routing/builder.ts +81 -0
- package/src/routing/chain.ts +74 -0
- package/src/routing/index.ts +16 -0
- package/src/routing/loader.ts +58 -0
- package/src/routing/router.ts +303 -0
- package/src/routing/strategies/adaptive.ts +215 -0
- package/src/routing/strategies/index.ts +8 -0
- package/src/routing/strategies/keyword.ts +163 -0
- package/src/routing/strategies/llm-prompts.ts +209 -0
- package/src/routing/strategies/llm.ts +235 -0
- package/src/routing/strategies/manual.ts +50 -0
- package/src/routing/strategy.ts +99 -0
- package/src/tdd/cleanup.ts +111 -0
- package/src/tdd/index.ts +23 -0
- package/src/tdd/isolation.ts +123 -0
- package/src/tdd/orchestrator.ts +383 -0
- package/src/tdd/prompts.ts +270 -0
- package/src/tdd/rectification-gate.ts +183 -0
- package/src/tdd/session-runner.ts +179 -0
- package/src/tdd/types.ts +81 -0
- package/src/tdd/verdict.ts +271 -0
- package/src/tui/App.tsx +265 -0
- package/src/tui/components/AgentPanel.tsx +75 -0
- package/src/tui/components/CostOverlay.tsx +118 -0
- package/src/tui/components/HelpOverlay.tsx +107 -0
- package/src/tui/components/StatusBar.tsx +63 -0
- package/src/tui/components/StoriesPanel.tsx +177 -0
- package/src/tui/hooks/useKeyboard.ts +142 -0
- package/src/tui/hooks/useLayout.ts +137 -0
- package/src/tui/hooks/usePipelineEvents.ts +183 -0
- package/src/tui/hooks/usePty.ts +194 -0
- package/src/tui/index.tsx +38 -0
- package/src/tui/types.ts +76 -0
- package/src/utils/git.ts +83 -0
- package/src/utils/queue-writer.ts +54 -0
- package/src/verification/executor.ts +235 -0
- package/src/verification/gate.ts +207 -0
- package/src/verification/index.ts +12 -0
- package/src/verification/parser.ts +230 -0
- package/src/verification/rectification.ts +108 -0
- package/src/verification/types.ts +113 -0
- package/src/worktree/dispatcher.ts +65 -0
- package/src/worktree/index.ts +2 -0
- package/src/worktree/manager.ts +187 -0
- package/src/worktree/merge.ts +301 -0
- package/src/worktree/types.ts +4 -0
- package/test/TEST_COVERAGE_US001.md +217 -0
- package/test/TEST_COVERAGE_US003.md +84 -0
- package/test/TEST_COVERAGE_US005.md +86 -0
- package/test/US-002-orchestrator.test.ts +246 -0
- package/test/acceptance/cm-003-default-view.test.ts +194 -0
- package/test/execution/pid-registry.test.ts +240 -0
- package/test/execution/post-verify.test.ts +224 -0
- package/test/helpers/timeout.ts +42 -0
- package/test/integration/US-002-TEST-SUMMARY.md +107 -0
- package/test/integration/US-003-TEST-SUMMARY.md +149 -0
- package/test/integration/US-004-TEST-SUMMARY.md +106 -0
- package/test/integration/US-005-TEST-SUMMARY.md +138 -0
- package/test/integration/US-007-TEST-SUMMARY.md +100 -0
- package/test/integration/agent-validation.test.ts +439 -0
- package/test/integration/analyze-integration.test.ts +261 -0
- package/test/integration/analyze-scanner.test.ts +131 -0
- package/test/integration/cli-config-default-edge-cases.test.ts +222 -0
- package/test/integration/cli-config-default-view.test.ts +229 -0
- package/test/integration/cli-config-diff.test.ts +460 -0
- package/test/integration/cli-config.test.ts +736 -0
- package/test/integration/cli-diagnose.test.ts +592 -0
- package/test/integration/cli-logs.test.ts +314 -0
- package/test/integration/cli-plugins.test.ts +678 -0
- package/test/integration/cli-precheck.test.ts +371 -0
- package/test/integration/cli-run-headless.test.ts +173 -0
- package/test/integration/cli.test.ts +75 -0
- package/test/integration/config/merger.test.ts +465 -0
- package/test/integration/config/paths.test.ts +51 -0
- package/test/integration/config-loader.test.ts +265 -0
- package/test/integration/config.test.ts +444 -0
- package/test/integration/context-integration.test.ts +702 -0
- package/test/integration/context-provider-injection.test.ts +506 -0
- package/test/integration/context-verification-integration.test.ts +295 -0
- package/test/integration/e2e.test.ts +896 -0
- package/test/integration/execution.test.ts +625 -0
- package/test/integration/helpers.test.ts +295 -0
- package/test/integration/hooks.test.ts +361 -0
- package/test/integration/interaction-chain-pipeline.test.ts +464 -0
- package/test/integration/isolation.test.ts +143 -0
- package/test/integration/logger.test.ts +461 -0
- package/test/integration/parallel.test.ts +250 -0
- package/test/integration/path-security.test.ts +173 -0
- package/test/integration/pipeline-acceptance.test.ts +302 -0
- package/test/integration/pipeline-events.test.ts +475 -0
- package/test/integration/pipeline.test.ts +658 -0
- package/test/integration/plan.test.ts +157 -0
- package/test/integration/plugin-routing.test.ts +921 -0
- package/test/integration/plugins/config-integration.test.ts +172 -0
- package/test/integration/plugins/config-resolution.test.ts +522 -0
- package/test/integration/plugins/loader.test.ts +641 -0
- package/test/integration/plugins/registry.test.ts +746 -0
- package/test/integration/plugins/validator.test.ts +563 -0
- package/test/integration/prd-pause.test.ts +205 -0
- package/test/integration/prd-resolvers.test.ts +185 -0
- package/test/integration/precheck-integration.test.ts +468 -0
- package/test/integration/precheck.test.ts +805 -0
- package/test/integration/progress.test.ts +34 -0
- package/test/integration/rectification-flow.test.ts +512 -0
- package/test/integration/reporter-lifecycle.test.ts +860 -0
- package/test/integration/review-config-commands.test.ts +319 -0
- package/test/integration/review-config-schema.test.ts +116 -0
- package/test/integration/review-plugin-integration.test.ts +722 -0
- package/test/integration/review.test.ts +149 -0
- package/test/integration/routing-stage-bug-021.test.ts +274 -0
- package/test/integration/routing-stage-greenfield.test.ts +286 -0
- package/test/integration/runner-config-plugins.test.ts +461 -0
- package/test/integration/runner-fixes.test.ts +399 -0
- package/test/integration/runner-plugin-integration.test.ts +543 -0
- package/test/integration/runner.test.ts +1679 -0
- package/test/integration/s5-greenfield-fallback.test.ts +297 -0
- package/test/integration/status-file-integration.test.ts +325 -0
- package/test/integration/status-file.test.ts +379 -0
- package/test/integration/status-writer.test.ts +345 -0
- package/test/integration/story-id-in-events.test.ts +273 -0
- package/test/integration/tdd-cleanup.test.ts +246 -0
- package/test/integration/tdd-orchestrator.test.ts +1762 -0
- package/test/integration/test-scanner.test.ts +403 -0
- package/test/integration/verification-asset-check.test.ts +142 -0
- package/test/integration/verify-stage.test.ts +275 -0
- package/test/integration/worktree/manager.test.ts +218 -0
- package/test/integration/worktree/merge.test.ts +341 -0
- package/test/manual/logging-formatter-demo.ts +158 -0
- package/test/ui/tui-agent-panel.test.tsx +99 -0
- package/test/ui/tui-controls.test.ts +334 -0
- package/test/ui/tui-cost-and-pty.test.ts +189 -0
- package/test/ui/tui-layout.test.ts +378 -0
- package/test/ui/tui-pty-integration.test.tsx +159 -0
- package/test/ui/tui-stories.test.ts +332 -0
- package/test/unit/acceptance.test.ts +186 -0
- package/test/unit/agent-stderr-capture.test.ts +146 -0
- package/test/unit/analyze-classifier.test.ts +215 -0
- package/test/unit/analyze.test.ts +224 -0
- package/test/unit/auto-detect.test.ts +249 -0
- package/test/unit/cli-status.test.ts +417 -0
- package/test/unit/commands/common.test.ts +320 -0
- package/test/unit/commands/logs.test.ts +416 -0
- package/test/unit/commands/unlock.test.ts +319 -0
- package/test/unit/constitution-generators.test.ts +160 -0
- package/test/unit/constitution.test.ts +209 -0
- package/test/unit/context.test.ts +1722 -0
- package/test/unit/cost.test.ts +231 -0
- package/test/unit/crash-recovery.test.ts +308 -0
- package/test/unit/escalation.test.ts +126 -0
- package/test/unit/execution-logging-stderr.test.ts +156 -0
- package/test/unit/execution-stage.test.ts +122 -0
- package/test/unit/fix-generator.test.ts +275 -0
- package/test/unit/formatters.test.ts +469 -0
- package/test/unit/greenfield.test.ts +179 -0
- package/test/unit/helpers.test.ts +317 -0
- package/test/unit/interaction/human-review-trigger.test.ts +164 -0
- package/test/unit/interaction-network-failures.test.ts +389 -0
- package/test/unit/interaction-plugins.test.ts +164 -0
- package/test/unit/isolation.test.ts +134 -0
- package/test/unit/logging/formatter.test.ts +455 -0
- package/test/unit/merge.test.ts +268 -0
- package/test/unit/metrics.test.ts +276 -0
- package/test/unit/optimizer/noop.optimizer.test.ts +125 -0
- package/test/unit/optimizer/rule-based.optimizer.test.ts +358 -0
- package/test/unit/prd-auto-default.test.ts +290 -0
- package/test/unit/prd-failure-category.test.ts +176 -0
- package/test/unit/prd-get-next-story.test.ts +186 -0
- package/test/unit/precheck-checks.test.ts +840 -0
- package/test/unit/precheck-story-size-gate.test.ts +287 -0
- package/test/unit/precheck-types.test.ts +142 -0
- package/test/unit/prompts.test.ts +475 -0
- package/test/unit/queue.test.ts +237 -0
- package/test/unit/rectification.test.ts +284 -0
- package/test/unit/registry.test.ts +287 -0
- package/test/unit/routing.test.ts +937 -0
- package/test/unit/run-lifecycle.test.ts +140 -0
- package/test/unit/storyid-events.test.ts +224 -0
- package/test/unit/tdd-verdict.test.ts +492 -0
- package/test/unit/test-output-parser.test.ts +377 -0
- package/test/unit/verdict.test.ts +324 -0
- package/test/unit/worktree-manager.test.ts +158 -0
- 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
|
+
});
|