@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,171 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* State Persistence for Pause/Resume (v0.15.0 US-003)
|
|
3
|
+
*
|
|
4
|
+
* Serializes run state when pausing, loads state when resuming.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import * as path from "node:path";
|
|
8
|
+
import type { InteractionRequest, InteractionResponse } from "./types";
|
|
9
|
+
|
|
10
|
+
/** Serialized run state for pause/resume */
|
|
11
|
+
export interface RunState {
|
|
12
|
+
/** Feature name */
|
|
13
|
+
feature: string;
|
|
14
|
+
/** PRD path */
|
|
15
|
+
prdPath: string;
|
|
16
|
+
/** Current iteration number */
|
|
17
|
+
iteration: number;
|
|
18
|
+
/** Accumulated cost (USD) */
|
|
19
|
+
totalCost: number;
|
|
20
|
+
/** Stories completed */
|
|
21
|
+
storiesCompleted: number;
|
|
22
|
+
/** Pending interactions */
|
|
23
|
+
pendingInteractions: InteractionRequest[];
|
|
24
|
+
/** Completed interactions */
|
|
25
|
+
completedInteractions: Array<{
|
|
26
|
+
request: InteractionRequest;
|
|
27
|
+
response: InteractionResponse;
|
|
28
|
+
}>;
|
|
29
|
+
/** Pause timestamp */
|
|
30
|
+
pausedAt: number;
|
|
31
|
+
/** Pause reason */
|
|
32
|
+
pauseReason: string;
|
|
33
|
+
/** Current story ID (if paused mid-story) */
|
|
34
|
+
currentStoryId?: string;
|
|
35
|
+
/** Current tier */
|
|
36
|
+
currentTier?: string;
|
|
37
|
+
/** Current model */
|
|
38
|
+
currentModel?: string;
|
|
39
|
+
/** Arbitrary metadata */
|
|
40
|
+
metadata?: Record<string, unknown>;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Serialize run state to JSON file
|
|
45
|
+
*/
|
|
46
|
+
export async function serializeRunState(state: RunState, featureDir: string): Promise<string> {
|
|
47
|
+
const stateFile = path.join(featureDir, "run-state.json");
|
|
48
|
+
const json = JSON.stringify(state, null, 2);
|
|
49
|
+
await Bun.write(stateFile, json);
|
|
50
|
+
return stateFile;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Deserialize run state from JSON file
|
|
55
|
+
*/
|
|
56
|
+
export async function deserializeRunState(featureDir: string): Promise<RunState | null> {
|
|
57
|
+
const stateFile = path.join(featureDir, "run-state.json");
|
|
58
|
+
try {
|
|
59
|
+
const file = Bun.file(stateFile);
|
|
60
|
+
const exists = await file.exists();
|
|
61
|
+
if (!exists) {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
const json = await file.text();
|
|
65
|
+
const state = JSON.parse(json) as RunState;
|
|
66
|
+
return state;
|
|
67
|
+
} catch (err) {
|
|
68
|
+
// Corrupted or invalid state file
|
|
69
|
+
return null;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Delete run state file (after successful resume)
|
|
75
|
+
*/
|
|
76
|
+
export async function clearRunState(featureDir: string): Promise<void> {
|
|
77
|
+
const stateFile = path.join(featureDir, "run-state.json");
|
|
78
|
+
try {
|
|
79
|
+
await Bun.write(stateFile, ""); // truncate
|
|
80
|
+
// Note: Bun doesn't have fs.unlink, so we truncate instead
|
|
81
|
+
} catch {
|
|
82
|
+
// Ignore errors
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Save a pending interaction to the interactions directory
|
|
88
|
+
*/
|
|
89
|
+
export async function savePendingInteraction(request: InteractionRequest, featureDir: string): Promise<string> {
|
|
90
|
+
const interactionsDir = path.join(featureDir, "interactions");
|
|
91
|
+
// Ensure directory exists
|
|
92
|
+
await Bun.write(path.join(interactionsDir, ".gitkeep"), "");
|
|
93
|
+
|
|
94
|
+
const filename = `${request.id}.json`;
|
|
95
|
+
const filePath = path.join(interactionsDir, filename);
|
|
96
|
+
const json = JSON.stringify(request, null, 2);
|
|
97
|
+
await Bun.write(filePath, json);
|
|
98
|
+
return filePath;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Load a pending interaction from the interactions directory
|
|
103
|
+
*/
|
|
104
|
+
export async function loadPendingInteraction(
|
|
105
|
+
requestId: string,
|
|
106
|
+
featureDir: string,
|
|
107
|
+
): Promise<InteractionRequest | null> {
|
|
108
|
+
const interactionsDir = path.join(featureDir, "interactions");
|
|
109
|
+
const filename = `${requestId}.json`;
|
|
110
|
+
const filePath = path.join(interactionsDir, filename);
|
|
111
|
+
|
|
112
|
+
try {
|
|
113
|
+
const file = Bun.file(filePath);
|
|
114
|
+
const exists = await file.exists();
|
|
115
|
+
if (!exists) {
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
const json = await file.text();
|
|
119
|
+
const request = JSON.parse(json) as InteractionRequest;
|
|
120
|
+
return request;
|
|
121
|
+
} catch {
|
|
122
|
+
return null;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Delete a pending interaction file (after response received)
|
|
128
|
+
*/
|
|
129
|
+
export async function deletePendingInteraction(requestId: string, featureDir: string): Promise<void> {
|
|
130
|
+
const interactionsDir = path.join(featureDir, "interactions");
|
|
131
|
+
const filename = `${requestId}.json`;
|
|
132
|
+
const filePath = path.join(interactionsDir, filename);
|
|
133
|
+
|
|
134
|
+
try {
|
|
135
|
+
await Bun.write(filePath, ""); // truncate
|
|
136
|
+
} catch {
|
|
137
|
+
// Ignore errors
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* List all pending interaction IDs
|
|
143
|
+
*/
|
|
144
|
+
export async function listPendingInteractions(featureDir: string): Promise<string[]> {
|
|
145
|
+
const interactionsDir = path.join(featureDir, "interactions");
|
|
146
|
+
|
|
147
|
+
try {
|
|
148
|
+
const dir = Bun.file(interactionsDir);
|
|
149
|
+
const exists = await dir.exists();
|
|
150
|
+
if (!exists) {
|
|
151
|
+
return [];
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// Use Bun.spawn to list files
|
|
155
|
+
const proc = Bun.spawn(["ls", interactionsDir], {
|
|
156
|
+
stdout: "pipe",
|
|
157
|
+
stderr: "pipe",
|
|
158
|
+
});
|
|
159
|
+
const output = await new Response(proc.stdout).text();
|
|
160
|
+
await proc.exited;
|
|
161
|
+
|
|
162
|
+
const files = output
|
|
163
|
+
.split("\n")
|
|
164
|
+
.filter((f) => f.endsWith(".json") && f !== ".gitkeep")
|
|
165
|
+
.map((f) => f.replace(".json", ""));
|
|
166
|
+
|
|
167
|
+
return files;
|
|
168
|
+
} catch {
|
|
169
|
+
return [];
|
|
170
|
+
}
|
|
171
|
+
}
|
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Built-in Triggers Integration (v0.15.0 US-004)
|
|
3
|
+
*
|
|
4
|
+
* Wires 8 built-in triggers into the runner loop and hooks system.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import type { NaxConfig } from "../config";
|
|
8
|
+
import type { InteractionChain } from "./chain";
|
|
9
|
+
import type { InteractionFallback, InteractionRequest, InteractionResponse, TriggerName } from "./types";
|
|
10
|
+
import { TRIGGER_METADATA } from "./types";
|
|
11
|
+
|
|
12
|
+
/** Trigger context data for template substitution */
|
|
13
|
+
export interface TriggerContext {
|
|
14
|
+
featureName: string;
|
|
15
|
+
storyId?: string;
|
|
16
|
+
cost?: number;
|
|
17
|
+
limit?: number;
|
|
18
|
+
tier?: string;
|
|
19
|
+
model?: string;
|
|
20
|
+
iteration?: number;
|
|
21
|
+
reason?: string;
|
|
22
|
+
[key: string]: unknown;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Check if a trigger is enabled in config
|
|
27
|
+
*/
|
|
28
|
+
export function isTriggerEnabled(trigger: TriggerName, config: NaxConfig): boolean {
|
|
29
|
+
const triggerConfig = config.interaction?.triggers?.[trigger];
|
|
30
|
+
if (triggerConfig === undefined) return false;
|
|
31
|
+
if (typeof triggerConfig === "boolean") return triggerConfig;
|
|
32
|
+
return triggerConfig.enabled;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Get trigger configuration (fallback, timeout)
|
|
37
|
+
*/
|
|
38
|
+
function getTriggerConfig(trigger: TriggerName, config: NaxConfig): { fallback: InteractionFallback; timeout: number } {
|
|
39
|
+
const metadata = TRIGGER_METADATA[trigger];
|
|
40
|
+
const triggerConfig = config.interaction?.triggers?.[trigger];
|
|
41
|
+
const defaults = config.interaction?.defaults ?? {
|
|
42
|
+
timeout: 600000,
|
|
43
|
+
fallback: "escalate" as InteractionFallback,
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
let fallback: InteractionFallback = metadata.defaultFallback;
|
|
47
|
+
let timeout = defaults.timeout;
|
|
48
|
+
|
|
49
|
+
if (typeof triggerConfig === "object") {
|
|
50
|
+
if (triggerConfig.fallback) {
|
|
51
|
+
fallback = triggerConfig.fallback as InteractionFallback;
|
|
52
|
+
}
|
|
53
|
+
if (triggerConfig.timeout) {
|
|
54
|
+
timeout = triggerConfig.timeout;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return { fallback, timeout };
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Substitute {{variable}} placeholders in a template string
|
|
63
|
+
*/
|
|
64
|
+
function substituteTemplate(template: string, context: TriggerContext): string {
|
|
65
|
+
let result = template;
|
|
66
|
+
for (const [key, value] of Object.entries(context)) {
|
|
67
|
+
if (value !== undefined) {
|
|
68
|
+
result = result.replace(new RegExp(`\\{\\{${key}\\}\\}`, "g"), String(value));
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
return result;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Create an interaction request for a built-in trigger
|
|
76
|
+
*/
|
|
77
|
+
export function createTriggerRequest(
|
|
78
|
+
trigger: TriggerName,
|
|
79
|
+
context: TriggerContext,
|
|
80
|
+
config: NaxConfig,
|
|
81
|
+
): InteractionRequest {
|
|
82
|
+
const metadata = TRIGGER_METADATA[trigger];
|
|
83
|
+
const { fallback, timeout } = getTriggerConfig(trigger, config);
|
|
84
|
+
|
|
85
|
+
const summary = substituteTemplate(metadata.defaultSummary, context);
|
|
86
|
+
const id = `trigger-${trigger}-${Date.now()}`;
|
|
87
|
+
|
|
88
|
+
return {
|
|
89
|
+
id,
|
|
90
|
+
type: "confirm",
|
|
91
|
+
featureName: context.featureName,
|
|
92
|
+
storyId: context.storyId,
|
|
93
|
+
stage: "custom",
|
|
94
|
+
summary,
|
|
95
|
+
fallback,
|
|
96
|
+
timeout,
|
|
97
|
+
createdAt: Date.now(),
|
|
98
|
+
metadata: {
|
|
99
|
+
trigger,
|
|
100
|
+
safety: metadata.safety,
|
|
101
|
+
},
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Execute a trigger and return response
|
|
107
|
+
*/
|
|
108
|
+
export async function executeTrigger(
|
|
109
|
+
trigger: TriggerName,
|
|
110
|
+
context: TriggerContext,
|
|
111
|
+
config: NaxConfig,
|
|
112
|
+
chain: InteractionChain,
|
|
113
|
+
): Promise<InteractionResponse> {
|
|
114
|
+
const request = createTriggerRequest(trigger, context, config);
|
|
115
|
+
const response = await chain.prompt(request);
|
|
116
|
+
return response;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Check security-review trigger (abort on critical issues)
|
|
121
|
+
*/
|
|
122
|
+
export async function checkSecurityReview(
|
|
123
|
+
context: TriggerContext,
|
|
124
|
+
config: NaxConfig,
|
|
125
|
+
chain: InteractionChain,
|
|
126
|
+
): Promise<boolean> {
|
|
127
|
+
if (!isTriggerEnabled("security-review", config)) return true;
|
|
128
|
+
|
|
129
|
+
const response = await executeTrigger("security-review", context, config, chain);
|
|
130
|
+
return response.action !== "abort";
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Check cost-exceeded trigger (abort on limit exceeded)
|
|
135
|
+
*/
|
|
136
|
+
export async function checkCostExceeded(
|
|
137
|
+
context: TriggerContext,
|
|
138
|
+
config: NaxConfig,
|
|
139
|
+
chain: InteractionChain,
|
|
140
|
+
): Promise<boolean> {
|
|
141
|
+
if (!isTriggerEnabled("cost-exceeded", config)) return true;
|
|
142
|
+
|
|
143
|
+
const response = await executeTrigger("cost-exceeded", context, config, chain);
|
|
144
|
+
return response.action !== "abort";
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
/**
|
|
148
|
+
* Check merge-conflict trigger (abort on conflict)
|
|
149
|
+
*/
|
|
150
|
+
export async function checkMergeConflict(
|
|
151
|
+
context: TriggerContext,
|
|
152
|
+
config: NaxConfig,
|
|
153
|
+
chain: InteractionChain,
|
|
154
|
+
): Promise<boolean> {
|
|
155
|
+
if (!isTriggerEnabled("merge-conflict", config)) return true;
|
|
156
|
+
|
|
157
|
+
const response = await executeTrigger("merge-conflict", context, config, chain);
|
|
158
|
+
return response.action !== "abort";
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Check cost-warning trigger (escalate on approaching limit)
|
|
163
|
+
*/
|
|
164
|
+
export async function checkCostWarning(
|
|
165
|
+
context: TriggerContext,
|
|
166
|
+
config: NaxConfig,
|
|
167
|
+
chain: InteractionChain,
|
|
168
|
+
): Promise<"continue" | "escalate"> {
|
|
169
|
+
if (!isTriggerEnabled("cost-warning", config)) return "continue";
|
|
170
|
+
|
|
171
|
+
const response = await executeTrigger("cost-warning", context, config, chain);
|
|
172
|
+
return response.action === "approve" ? "escalate" : "continue";
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* Check max-retries trigger (skip story on max retries)
|
|
177
|
+
*/
|
|
178
|
+
export async function checkMaxRetries(
|
|
179
|
+
context: TriggerContext,
|
|
180
|
+
config: NaxConfig,
|
|
181
|
+
chain: InteractionChain,
|
|
182
|
+
): Promise<"continue" | "skip"> {
|
|
183
|
+
if (!isTriggerEnabled("max-retries", config)) return "continue";
|
|
184
|
+
|
|
185
|
+
const response = await executeTrigger("max-retries", context, config, chain);
|
|
186
|
+
return response.action === "skip" ? "skip" : "continue";
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
/**
|
|
190
|
+
* Check pre-merge trigger (escalate before merging)
|
|
191
|
+
*/
|
|
192
|
+
export async function checkPreMerge(
|
|
193
|
+
context: TriggerContext,
|
|
194
|
+
config: NaxConfig,
|
|
195
|
+
chain: InteractionChain,
|
|
196
|
+
): Promise<boolean> {
|
|
197
|
+
if (!isTriggerEnabled("pre-merge", config)) return true;
|
|
198
|
+
|
|
199
|
+
const response = await executeTrigger("pre-merge", context, config, chain);
|
|
200
|
+
return response.action === "approve";
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
/**
|
|
204
|
+
* Check story-ambiguity trigger (continue with best effort)
|
|
205
|
+
*/
|
|
206
|
+
export async function checkStoryAmbiguity(
|
|
207
|
+
context: TriggerContext,
|
|
208
|
+
config: NaxConfig,
|
|
209
|
+
chain: InteractionChain,
|
|
210
|
+
): Promise<boolean> {
|
|
211
|
+
if (!isTriggerEnabled("story-ambiguity", config)) return true;
|
|
212
|
+
|
|
213
|
+
const response = await executeTrigger("story-ambiguity", context, config, chain);
|
|
214
|
+
return response.action === "approve";
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
/**
|
|
218
|
+
* Check review-gate trigger (proceed with review)
|
|
219
|
+
*/
|
|
220
|
+
export async function checkReviewGate(
|
|
221
|
+
context: TriggerContext,
|
|
222
|
+
config: NaxConfig,
|
|
223
|
+
chain: InteractionChain,
|
|
224
|
+
): Promise<boolean> {
|
|
225
|
+
if (!isTriggerEnabled("review-gate", config)) return true;
|
|
226
|
+
|
|
227
|
+
const response = await executeTrigger("review-gate", context, config, chain);
|
|
228
|
+
return response.action === "approve";
|
|
229
|
+
}
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Interaction System Types (v0.15.0)
|
|
3
|
+
*
|
|
4
|
+
* Interactive pipeline core for user prompts, decision gates, and pause/resume.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
/** Interaction request types */
|
|
8
|
+
export type InteractionType = "confirm" | "choose" | "input" | "review" | "notify" | "webhook";
|
|
9
|
+
|
|
10
|
+
/** Pipeline stage where interaction occurs */
|
|
11
|
+
export type InteractionStage = "pre-flight" | "execution" | "review" | "merge" | "cost" | "custom";
|
|
12
|
+
|
|
13
|
+
/** Fallback behavior when interaction times out */
|
|
14
|
+
export type InteractionFallback = "continue" | "skip" | "escalate" | "abort";
|
|
15
|
+
|
|
16
|
+
/** Interaction request — sent to plugin */
|
|
17
|
+
export interface InteractionRequest {
|
|
18
|
+
/** Unique request ID (e.g., 'ix-US003-review-1') */
|
|
19
|
+
id: string;
|
|
20
|
+
/** Type of interaction */
|
|
21
|
+
type: InteractionType;
|
|
22
|
+
/** Feature name */
|
|
23
|
+
featureName: string;
|
|
24
|
+
/** Story ID (optional, for story-level interactions) */
|
|
25
|
+
storyId?: string;
|
|
26
|
+
/** Pipeline stage */
|
|
27
|
+
stage: InteractionStage;
|
|
28
|
+
/** Human-readable question/summary */
|
|
29
|
+
summary: string;
|
|
30
|
+
/** Longer context/details (optional) */
|
|
31
|
+
detail?: string;
|
|
32
|
+
/** Options for choose type */
|
|
33
|
+
options?: Array<{ key: string; label: string; description?: string }>;
|
|
34
|
+
/** Timeout in milliseconds (optional) */
|
|
35
|
+
timeout?: number;
|
|
36
|
+
/** Fallback behavior on timeout */
|
|
37
|
+
fallback: InteractionFallback;
|
|
38
|
+
/** Arbitrary metadata */
|
|
39
|
+
metadata?: Record<string, unknown>;
|
|
40
|
+
/** Creation timestamp */
|
|
41
|
+
createdAt: number;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/** Interaction response action */
|
|
45
|
+
export type InteractionAction = "approve" | "reject" | "choose" | "input" | "skip" | "abort";
|
|
46
|
+
|
|
47
|
+
/** Interaction response — returned from plugin */
|
|
48
|
+
export interface InteractionResponse {
|
|
49
|
+
/** Request ID this response is for */
|
|
50
|
+
requestId: string;
|
|
51
|
+
/** Action taken */
|
|
52
|
+
action: InteractionAction;
|
|
53
|
+
/** Value (for choose/input types) */
|
|
54
|
+
value?: string;
|
|
55
|
+
/** Who responded (user, system, timeout, etc.) */
|
|
56
|
+
respondedBy?: string;
|
|
57
|
+
/** Response timestamp */
|
|
58
|
+
respondedAt: number;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/** Interaction plugin interface */
|
|
62
|
+
export interface InteractionPlugin {
|
|
63
|
+
/** Plugin name */
|
|
64
|
+
name: string;
|
|
65
|
+
/** Send interaction request to user */
|
|
66
|
+
send(request: InteractionRequest): Promise<void>;
|
|
67
|
+
/** Receive interaction response from user (blocking) */
|
|
68
|
+
receive(requestId: string, timeout?: number): Promise<InteractionResponse>;
|
|
69
|
+
/** Cancel a pending interaction (optional) */
|
|
70
|
+
cancel?(requestId: string): Promise<void>;
|
|
71
|
+
/** Initialize plugin with config (optional) */
|
|
72
|
+
init?(config: Record<string, unknown>): Promise<void>;
|
|
73
|
+
/** Teardown plugin (optional) */
|
|
74
|
+
destroy?(): Promise<void>;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/** Built-in trigger names */
|
|
78
|
+
export type TriggerName =
|
|
79
|
+
| "security-review" // abort (red) — critical security issues found
|
|
80
|
+
| "cost-exceeded" // abort (red) — cost limit exceeded
|
|
81
|
+
| "merge-conflict" // abort (red) — merge conflict detected
|
|
82
|
+
| "cost-warning" // escalate (yellow) — approaching cost limit
|
|
83
|
+
| "max-retries" // skip (yellow) — max retries reached
|
|
84
|
+
| "pre-merge" // escalate (yellow) — before merging to main
|
|
85
|
+
| "human-review" // skip (yellow) — human review required on max retries / critical failure
|
|
86
|
+
| "story-ambiguity" // continue (green) — story requirements unclear
|
|
87
|
+
| "review-gate"; // continue (green) — code review checkpoint
|
|
88
|
+
|
|
89
|
+
/** Trigger configuration */
|
|
90
|
+
export interface TriggerConfig {
|
|
91
|
+
/** Whether trigger is enabled */
|
|
92
|
+
enabled: boolean;
|
|
93
|
+
/** Override default fallback behavior */
|
|
94
|
+
fallback?: InteractionFallback;
|
|
95
|
+
/** Override default timeout (ms) */
|
|
96
|
+
timeout?: number;
|
|
97
|
+
/** Custom summary template (supports {{variable}} syntax) */
|
|
98
|
+
summary?: string;
|
|
99
|
+
/** Custom detail template */
|
|
100
|
+
detail?: string;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
/** Safety tier for triggers */
|
|
104
|
+
export type TriggerSafety = "red" | "yellow" | "green";
|
|
105
|
+
|
|
106
|
+
/** Built-in trigger metadata */
|
|
107
|
+
export interface TriggerMetadata {
|
|
108
|
+
/** Default fallback behavior */
|
|
109
|
+
defaultFallback: InteractionFallback;
|
|
110
|
+
/** Safety tier */
|
|
111
|
+
safety: TriggerSafety;
|
|
112
|
+
/** Default summary */
|
|
113
|
+
defaultSummary: string;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/** Map of built-in triggers to their metadata */
|
|
117
|
+
export const TRIGGER_METADATA: Record<TriggerName, TriggerMetadata> = {
|
|
118
|
+
"security-review": {
|
|
119
|
+
defaultFallback: "abort",
|
|
120
|
+
safety: "red",
|
|
121
|
+
defaultSummary: "Security review failed — abort execution?",
|
|
122
|
+
},
|
|
123
|
+
"cost-exceeded": {
|
|
124
|
+
defaultFallback: "abort",
|
|
125
|
+
safety: "red",
|
|
126
|
+
defaultSummary: "Cost limit exceeded ({{cost}} USD) — abort execution?",
|
|
127
|
+
},
|
|
128
|
+
"merge-conflict": {
|
|
129
|
+
defaultFallback: "abort",
|
|
130
|
+
safety: "red",
|
|
131
|
+
defaultSummary: "Merge conflict detected in {{storyId}} — abort execution?",
|
|
132
|
+
},
|
|
133
|
+
"cost-warning": {
|
|
134
|
+
defaultFallback: "escalate",
|
|
135
|
+
safety: "yellow",
|
|
136
|
+
defaultSummary: "Cost warning: {{cost}} USD / {{limit}} USD — escalate to higher tier?",
|
|
137
|
+
},
|
|
138
|
+
"max-retries": {
|
|
139
|
+
defaultFallback: "skip",
|
|
140
|
+
safety: "yellow",
|
|
141
|
+
defaultSummary: "Max retries reached for {{storyId}} — skip story?",
|
|
142
|
+
},
|
|
143
|
+
"pre-merge": {
|
|
144
|
+
defaultFallback: "escalate",
|
|
145
|
+
safety: "yellow",
|
|
146
|
+
defaultSummary: "Pre-merge checkpoint for {{storyId}} — proceed with merge?",
|
|
147
|
+
},
|
|
148
|
+
"human-review": {
|
|
149
|
+
defaultFallback: "skip",
|
|
150
|
+
safety: "yellow",
|
|
151
|
+
defaultSummary: "Human review required for story {{storyId}} — skip and continue?",
|
|
152
|
+
},
|
|
153
|
+
"story-ambiguity": {
|
|
154
|
+
defaultFallback: "continue",
|
|
155
|
+
safety: "green",
|
|
156
|
+
defaultSummary: "Story {{storyId}} requirements unclear — continue with best effort?",
|
|
157
|
+
},
|
|
158
|
+
"review-gate": {
|
|
159
|
+
defaultFallback: "continue",
|
|
160
|
+
safety: "green",
|
|
161
|
+
defaultSummary: "Code review checkpoint for {{storyId}} — proceed?",
|
|
162
|
+
},
|
|
163
|
+
};
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import chalk from "chalk";
|
|
2
|
+
import type { LogEntry } from "./types.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Format log entry for human-readable console output with chalk colors
|
|
6
|
+
*
|
|
7
|
+
* @param entry - The log entry to format
|
|
8
|
+
* @returns Chalk-formatted console string
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* ```typescript
|
|
12
|
+
* const entry: LogEntry = {
|
|
13
|
+
* timestamp: "2026-02-20T10:30:00.123Z",
|
|
14
|
+
* level: "info",
|
|
15
|
+
* stage: "routing",
|
|
16
|
+
* storyId: "user-auth-001",
|
|
17
|
+
* message: "Classified as simple task",
|
|
18
|
+
* data: { complexity: "simple", model: "claude-sonnet-4-5" }
|
|
19
|
+
* };
|
|
20
|
+
* console.log(formatConsole(entry));
|
|
21
|
+
* // [10:30:00] [routing] [user-auth-001] Classified as simple task
|
|
22
|
+
* ```
|
|
23
|
+
*/
|
|
24
|
+
export function formatConsole(entry: LogEntry): string {
|
|
25
|
+
const timestamp = new Date(entry.timestamp).toLocaleTimeString("en-US", {
|
|
26
|
+
hour12: false,
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
// Level-specific color coding
|
|
30
|
+
let levelColor: (text: string) => string;
|
|
31
|
+
switch (entry.level) {
|
|
32
|
+
case "error":
|
|
33
|
+
levelColor = chalk.red;
|
|
34
|
+
break;
|
|
35
|
+
case "warn":
|
|
36
|
+
levelColor = chalk.yellow;
|
|
37
|
+
break;
|
|
38
|
+
case "info":
|
|
39
|
+
levelColor = chalk.blue;
|
|
40
|
+
break;
|
|
41
|
+
case "debug":
|
|
42
|
+
levelColor = chalk.gray;
|
|
43
|
+
break;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Build base message with timestamp, stage, and optional storyId
|
|
47
|
+
const parts = [chalk.gray(`[${timestamp}]`), levelColor(`[${entry.stage}]`)];
|
|
48
|
+
|
|
49
|
+
if (entry.storyId) {
|
|
50
|
+
parts.push(chalk.cyan(`[${entry.storyId}]`));
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
parts.push(entry.message);
|
|
54
|
+
|
|
55
|
+
// Append data if present (pretty-printed on next line)
|
|
56
|
+
let output = parts.join(" ");
|
|
57
|
+
if (entry.data && Object.keys(entry.data).length > 0) {
|
|
58
|
+
output += `\n${chalk.gray(JSON.stringify(entry.data, null, 2))}`;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
return output;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Format log entry as JSON Lines (JSONL) for machine-readable file output
|
|
66
|
+
*
|
|
67
|
+
* @param entry - The log entry to format
|
|
68
|
+
* @returns Single-line JSON string
|
|
69
|
+
*
|
|
70
|
+
* @example
|
|
71
|
+
* ```typescript
|
|
72
|
+
* const entry: LogEntry = {
|
|
73
|
+
* timestamp: "2026-02-20T10:30:00.123Z",
|
|
74
|
+
* level: "info",
|
|
75
|
+
* stage: "routing",
|
|
76
|
+
* message: "Task classified"
|
|
77
|
+
* };
|
|
78
|
+
* console.log(formatJsonl(entry));
|
|
79
|
+
* // {"timestamp":"2026-02-20T10:30:00.123Z","level":"info","stage":"routing","message":"Task classified"}
|
|
80
|
+
* ```
|
|
81
|
+
*/
|
|
82
|
+
export function formatJsonl(entry: LogEntry): string {
|
|
83
|
+
return JSON.stringify(entry);
|
|
84
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Structured logging module for nax
|
|
3
|
+
*
|
|
4
|
+
* Provides level-gated console output and JSONL file logging for all stages.
|
|
5
|
+
*
|
|
6
|
+
* @module logger
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
export { Logger, initLogger, getLogger, getSafeLogger, resetLogger } from "./logger.js";
|
|
10
|
+
export { formatConsole, formatJsonl } from "./formatters.js";
|
|
11
|
+
export type {
|
|
12
|
+
LogLevel,
|
|
13
|
+
LogEntry,
|
|
14
|
+
LoggerOptions,
|
|
15
|
+
StoryLogger,
|
|
16
|
+
} from "./types.js";
|