@nathapp/nax 0.18.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (459) hide show
  1. package/.gitlab-ci.yml +96 -0
  2. package/BRIEF.md +140 -0
  3. package/CHANGELOG.md +60 -0
  4. package/CLAUDE.md +159 -0
  5. package/README.md +373 -0
  6. package/US-007-IMPLEMENTATION.md +139 -0
  7. package/bin/nax.ts +930 -0
  8. package/biome.json +14 -0
  9. package/bun.lock +168 -0
  10. package/bunfig.toml +11 -0
  11. package/docs/20260216-fix-plan-context-review.md +56 -0
  12. package/docs/20260216-relentless-vs-ngent-comparison.md +208 -0
  13. package/docs/20260216-v02-plan.md +136 -0
  14. package/docs/20260216-v02-review.md +685 -0
  15. package/docs/20260217-dogfood-findings.md +56 -0
  16. package/docs/20260217-p2-plus-plan.md +117 -0
  17. package/docs/20260217-partial-fixes-plan.md +62 -0
  18. package/docs/20260217-plan-analyze-spec.md +117 -0
  19. package/docs/20260217-post-impl-review.md +1137 -0
  20. package/docs/20260217-quick-wins-plan.md +66 -0
  21. package/docs/20260217-split-runner-plan.md +75 -0
  22. package/docs/20260217-v03-impl-plan.md +80 -0
  23. package/docs/20260217-v03-post-impl-review.md +589 -0
  24. package/docs/20260217-v04-impl-plan.md +86 -0
  25. package/docs/20260217-v05-post-impl-review.md +850 -0
  26. package/docs/20260217-v06-post-impl-review.md +817 -0
  27. package/docs/20260218-adr003-port-plan.md +151 -0
  28. package/docs/20260218-review-adr003-verification.md +175 -0
  29. package/docs/20260219-fix-plan-bug16-19.md +79 -0
  30. package/docs/20260219-fix-plan-bug20-22.md +114 -0
  31. package/docs/20260219-plan-llm-routing.md +116 -0
  32. package/docs/20260219-review-bug20-22-fixes.md +135 -0
  33. package/docs/20260219-routing-baseline-keyword.md +63 -0
  34. package/docs/20260220-plan-structured-logging-p1.md +80 -0
  35. package/docs/20260220-plan-structured-logging-p2.md +37 -0
  36. package/docs/20260220-review-llm-routing.md +180 -0
  37. package/docs/20260220-review-post-fix-llm-routing.md +70 -0
  38. package/docs/20260221-fix-plan-relevantfiles-split.md +101 -0
  39. package/docs/20260221-fix-plan-routing-mode.md +125 -0
  40. package/docs/20260221-review-v0.9-implementation.md +379 -0
  41. package/docs/20260222-fix-plan-v091-routing-isolation.md +197 -0
  42. package/docs/20260223-fix-plan-prompt-audit.md +62 -0
  43. package/docs/20260224-nax-roadmap-phases.md +189 -0
  44. package/docs/20260225-phase2-llm-service-layer.md +401 -0
  45. package/docs/20260225-review-v0.10.1.md +187 -0
  46. package/docs/20260303-v010-implementation-plan.md +165 -0
  47. package/docs/CLAUDE.md.bak +191 -0
  48. package/docs/ROADMAP.md +165 -0
  49. package/docs/SPEC-rectification.md +0 -0
  50. package/docs/SPEC.md +324 -0
  51. package/docs/US-001-plugin-loading-verification.md +152 -0
  52. package/docs/architecture-analysis.md +1076 -0
  53. package/docs/bugs/BUG-21-escalation-null-attempts.md +48 -0
  54. package/docs/bugs-from-dogfood-run-c.md +243 -0
  55. package/docs/code-review-20260228.md +612 -0
  56. package/docs/code-review-v0.15.0.md +629 -0
  57. package/docs/hook-lifecycle-test-plan.md +149 -0
  58. package/docs/releases/v0.11.0-and-earlier.md +20 -0
  59. package/docs/releases/v0.12.0.md +15 -0
  60. package/docs/releases/v0.13.0.md +14 -0
  61. package/docs/releases/v0.14.0.md +20 -0
  62. package/docs/releases/v0.14.1.md +36 -0
  63. package/docs/releases/v0.14.2.md +51 -0
  64. package/docs/releases/v0.14.3.md +174 -0
  65. package/docs/releases/v0.14.4.md +94 -0
  66. package/docs/releases/v0.15.0.md +502 -0
  67. package/docs/releases/v0.15.1.md +170 -0
  68. package/docs/releases/v0.15.3.md +193 -0
  69. package/docs/specs/status-file-v0.10.1.md +812 -0
  70. package/docs/v0.10-global-config.md +206 -0
  71. package/docs/v0.10-plugin-system.md +415 -0
  72. package/docs/v0.10-prompt-optimizer.md +234 -0
  73. package/docs/v0.3-spec.md +244 -0
  74. package/docs/v0.4-spec.md +140 -0
  75. package/docs/v0.5-spec.md +237 -0
  76. package/docs/v0.6-spec.md +371 -0
  77. package/docs/v0.7-spec.md +177 -0
  78. package/docs/v0.8-llm-routing.md +206 -0
  79. package/docs/v0.8-structured-logging.md +132 -0
  80. package/docs/v0.9.3-prompt-audit.md +112 -0
  81. package/examples/plugins/console-reporter/index.test.ts +207 -0
  82. package/examples/plugins/console-reporter/index.ts +110 -0
  83. package/nax/config.json +147 -0
  84. package/nax/features/bugfix-v0171/prd.json +52 -0
  85. package/nax/features/config-management/prd.json +108 -0
  86. package/nax/features/config-management/progress.txt +5 -0
  87. package/nax/features/diagnose/acceptance.test.ts +412 -0
  88. package/nax/features/diagnose/prd.json +41 -0
  89. package/nax/features/orchestration-fixes/prd.json +89 -0
  90. package/nax/features/orchestration-fixes/progress.txt +1 -0
  91. package/nax/features/plugin-integration/US-007-VERIFICATION.md +259 -0
  92. package/nax/features/plugin-integration/prd.json +208 -0
  93. package/nax/features/plugin-integration/progress.txt +5 -0
  94. package/nax/features/precheck/prd.json +205 -0
  95. package/nax/features/precheck/progress.txt +15 -0
  96. package/nax/features/structured-logging/prd.json +199 -0
  97. package/nax/features/unlock/prd.json +36 -0
  98. package/package.json +47 -0
  99. package/src/acceptance/fix-generator.ts +348 -0
  100. package/src/acceptance/generator.ts +282 -0
  101. package/src/acceptance/index.ts +30 -0
  102. package/src/acceptance/types.ts +79 -0
  103. package/src/agents/claude-decompose.ts +169 -0
  104. package/src/agents/claude-plan.ts +139 -0
  105. package/src/agents/claude.ts +324 -0
  106. package/src/agents/cost.ts +268 -0
  107. package/src/agents/index.ts +13 -0
  108. package/src/agents/registry.ts +48 -0
  109. package/src/agents/types-extended.ts +133 -0
  110. package/src/agents/types.ts +113 -0
  111. package/src/agents/validation.ts +69 -0
  112. package/src/analyze/classifier.ts +305 -0
  113. package/src/analyze/index.ts +16 -0
  114. package/src/analyze/scanner.ts +175 -0
  115. package/src/analyze/types.ts +51 -0
  116. package/src/cli/accept.ts +108 -0
  117. package/src/cli/analyze-parser.ts +284 -0
  118. package/src/cli/analyze.ts +207 -0
  119. package/src/cli/config.ts +561 -0
  120. package/src/cli/constitution.ts +109 -0
  121. package/src/cli/diagnose-analysis.ts +159 -0
  122. package/src/cli/diagnose-formatter.ts +87 -0
  123. package/src/cli/diagnose.ts +203 -0
  124. package/src/cli/generate.ts +127 -0
  125. package/src/cli/index.ts +37 -0
  126. package/src/cli/init.ts +188 -0
  127. package/src/cli/interact.ts +295 -0
  128. package/src/cli/plan.ts +198 -0
  129. package/src/cli/plugins.ts +111 -0
  130. package/src/cli/prompts.ts +295 -0
  131. package/src/cli/runs.ts +174 -0
  132. package/src/cli/status-cost.ts +151 -0
  133. package/src/cli/status-features.ts +338 -0
  134. package/src/cli/status.ts +13 -0
  135. package/src/commands/common.ts +171 -0
  136. package/src/commands/diagnose.ts +17 -0
  137. package/src/commands/index.ts +8 -0
  138. package/src/commands/logs.ts +384 -0
  139. package/src/commands/precheck.ts +86 -0
  140. package/src/commands/unlock.ts +96 -0
  141. package/src/config/defaults.ts +160 -0
  142. package/src/config/index.ts +22 -0
  143. package/src/config/loader.ts +121 -0
  144. package/src/config/merger.ts +147 -0
  145. package/src/config/path-security.ts +121 -0
  146. package/src/config/paths.ts +27 -0
  147. package/src/config/schema.ts +56 -0
  148. package/src/config/schemas.ts +286 -0
  149. package/src/config/types.ts +423 -0
  150. package/src/config/validate.ts +103 -0
  151. package/src/constitution/generator.ts +191 -0
  152. package/src/constitution/generators/aider.ts +41 -0
  153. package/src/constitution/generators/claude.ts +35 -0
  154. package/src/constitution/generators/cursor.ts +36 -0
  155. package/src/constitution/generators/opencode.ts +38 -0
  156. package/src/constitution/generators/types.ts +33 -0
  157. package/src/constitution/generators/windsurf.ts +36 -0
  158. package/src/constitution/index.ts +10 -0
  159. package/src/constitution/loader.ts +133 -0
  160. package/src/constitution/types.ts +31 -0
  161. package/src/context/auto-detect.ts +227 -0
  162. package/src/context/builder.ts +246 -0
  163. package/src/context/elements.ts +83 -0
  164. package/src/context/formatter.ts +107 -0
  165. package/src/context/generator.ts +129 -0
  166. package/src/context/generators/aider.ts +34 -0
  167. package/src/context/generators/claude.ts +28 -0
  168. package/src/context/generators/cursor.ts +28 -0
  169. package/src/context/generators/opencode.ts +30 -0
  170. package/src/context/generators/windsurf.ts +28 -0
  171. package/src/context/greenfield.ts +114 -0
  172. package/src/context/index.ts +33 -0
  173. package/src/context/injector.ts +279 -0
  174. package/src/context/test-scanner.ts +370 -0
  175. package/src/context/types.ts +98 -0
  176. package/src/errors.ts +67 -0
  177. package/src/execution/batching.ts +157 -0
  178. package/src/execution/crash-recovery.ts +373 -0
  179. package/src/execution/escalation/escalation.ts +44 -0
  180. package/src/execution/escalation/index.ts +13 -0
  181. package/src/execution/escalation/tier-escalation.ts +295 -0
  182. package/src/execution/escalation/tier-outcome.ts +158 -0
  183. package/src/execution/helpers.ts +38 -0
  184. package/src/execution/index.ts +45 -0
  185. package/src/execution/lifecycle/acceptance-loop.ts +272 -0
  186. package/src/execution/lifecycle/headless-formatter.ts +85 -0
  187. package/src/execution/lifecycle/index.ts +12 -0
  188. package/src/execution/lifecycle/parallel-lifecycle.ts +101 -0
  189. package/src/execution/lifecycle/precheck-runner.ts +140 -0
  190. package/src/execution/lifecycle/run-cleanup.ts +81 -0
  191. package/src/execution/lifecycle/run-completion.ts +129 -0
  192. package/src/execution/lifecycle/run-initialization.ts +141 -0
  193. package/src/execution/lifecycle/run-lifecycle.ts +312 -0
  194. package/src/execution/lifecycle/run-setup.ts +204 -0
  195. package/src/execution/lifecycle/story-hooks.ts +38 -0
  196. package/src/execution/lifecycle/story-size-prompts.ts +123 -0
  197. package/src/execution/lock.ts +115 -0
  198. package/src/execution/parallel-executor.ts +216 -0
  199. package/src/execution/parallel.ts +400 -0
  200. package/src/execution/pid-registry.ts +280 -0
  201. package/src/execution/pipeline-result-handler.ts +388 -0
  202. package/src/execution/post-verify-rectification.ts +188 -0
  203. package/src/execution/post-verify.ts +274 -0
  204. package/src/execution/progress.ts +25 -0
  205. package/src/execution/prompts.ts +127 -0
  206. package/src/execution/queue-handler.ts +109 -0
  207. package/src/execution/rectification.ts +13 -0
  208. package/src/execution/runner.ts +377 -0
  209. package/src/execution/sequential-executor.ts +388 -0
  210. package/src/execution/status-file.ts +264 -0
  211. package/src/execution/status-writer.ts +139 -0
  212. package/src/execution/story-context.ts +229 -0
  213. package/src/execution/test-output-parser.ts +14 -0
  214. package/src/execution/verification.ts +72 -0
  215. package/src/hooks/index.ts +2 -0
  216. package/src/hooks/runner.ts +286 -0
  217. package/src/hooks/types.ts +67 -0
  218. package/src/interaction/chain.ts +154 -0
  219. package/src/interaction/index.ts +60 -0
  220. package/src/interaction/init.ts +83 -0
  221. package/src/interaction/plugins/auto.ts +217 -0
  222. package/src/interaction/plugins/cli.ts +300 -0
  223. package/src/interaction/plugins/telegram.ts +384 -0
  224. package/src/interaction/plugins/webhook.ts +258 -0
  225. package/src/interaction/state.ts +171 -0
  226. package/src/interaction/triggers.ts +229 -0
  227. package/src/interaction/types.ts +163 -0
  228. package/src/logger/formatters.ts +84 -0
  229. package/src/logger/index.ts +16 -0
  230. package/src/logger/logger.ts +298 -0
  231. package/src/logger/types.ts +48 -0
  232. package/src/logging/formatter.ts +355 -0
  233. package/src/logging/index.ts +22 -0
  234. package/src/logging/types.ts +93 -0
  235. package/src/metrics/aggregator.ts +190 -0
  236. package/src/metrics/index.ts +14 -0
  237. package/src/metrics/tracker.ts +200 -0
  238. package/src/metrics/types.ts +109 -0
  239. package/src/optimizer/index.ts +62 -0
  240. package/src/optimizer/noop.optimizer.ts +24 -0
  241. package/src/optimizer/rule-based.optimizer.ts +248 -0
  242. package/src/optimizer/types.ts +53 -0
  243. package/src/pipeline/events.ts +130 -0
  244. package/src/pipeline/index.ts +19 -0
  245. package/src/pipeline/runner.ts +161 -0
  246. package/src/pipeline/stages/acceptance.ts +197 -0
  247. package/src/pipeline/stages/completion.ts +99 -0
  248. package/src/pipeline/stages/constitution.ts +63 -0
  249. package/src/pipeline/stages/context.ts +117 -0
  250. package/src/pipeline/stages/execution.ts +194 -0
  251. package/src/pipeline/stages/index.ts +62 -0
  252. package/src/pipeline/stages/optimizer.ts +74 -0
  253. package/src/pipeline/stages/prompt.ts +57 -0
  254. package/src/pipeline/stages/queue-check.ts +103 -0
  255. package/src/pipeline/stages/review.ts +181 -0
  256. package/src/pipeline/stages/routing.ts +81 -0
  257. package/src/pipeline/stages/verify.ts +100 -0
  258. package/src/pipeline/types.ts +167 -0
  259. package/src/plugins/index.ts +31 -0
  260. package/src/plugins/loader.ts +287 -0
  261. package/src/plugins/registry.ts +168 -0
  262. package/src/plugins/types.ts +327 -0
  263. package/src/plugins/validator.ts +352 -0
  264. package/src/prd/index.ts +172 -0
  265. package/src/prd/types.ts +202 -0
  266. package/src/precheck/checks-blockers.ts +391 -0
  267. package/src/precheck/checks-warnings.ts +142 -0
  268. package/src/precheck/checks.ts +30 -0
  269. package/src/precheck/index.ts +247 -0
  270. package/src/precheck/story-size-gate.ts +144 -0
  271. package/src/precheck/types.ts +31 -0
  272. package/src/queue/index.ts +2 -0
  273. package/src/queue/manager.ts +254 -0
  274. package/src/queue/types.ts +54 -0
  275. package/src/review/index.ts +8 -0
  276. package/src/review/runner.ts +172 -0
  277. package/src/review/types.ts +66 -0
  278. package/src/routing/builder.ts +81 -0
  279. package/src/routing/chain.ts +74 -0
  280. package/src/routing/index.ts +16 -0
  281. package/src/routing/loader.ts +58 -0
  282. package/src/routing/router.ts +303 -0
  283. package/src/routing/strategies/adaptive.ts +215 -0
  284. package/src/routing/strategies/index.ts +8 -0
  285. package/src/routing/strategies/keyword.ts +163 -0
  286. package/src/routing/strategies/llm-prompts.ts +209 -0
  287. package/src/routing/strategies/llm.ts +235 -0
  288. package/src/routing/strategies/manual.ts +50 -0
  289. package/src/routing/strategy.ts +99 -0
  290. package/src/tdd/cleanup.ts +111 -0
  291. package/src/tdd/index.ts +23 -0
  292. package/src/tdd/isolation.ts +123 -0
  293. package/src/tdd/orchestrator.ts +383 -0
  294. package/src/tdd/prompts.ts +270 -0
  295. package/src/tdd/rectification-gate.ts +183 -0
  296. package/src/tdd/session-runner.ts +179 -0
  297. package/src/tdd/types.ts +81 -0
  298. package/src/tdd/verdict.ts +271 -0
  299. package/src/tui/App.tsx +265 -0
  300. package/src/tui/components/AgentPanel.tsx +75 -0
  301. package/src/tui/components/CostOverlay.tsx +118 -0
  302. package/src/tui/components/HelpOverlay.tsx +107 -0
  303. package/src/tui/components/StatusBar.tsx +63 -0
  304. package/src/tui/components/StoriesPanel.tsx +177 -0
  305. package/src/tui/hooks/useKeyboard.ts +142 -0
  306. package/src/tui/hooks/useLayout.ts +137 -0
  307. package/src/tui/hooks/usePipelineEvents.ts +183 -0
  308. package/src/tui/hooks/usePty.ts +194 -0
  309. package/src/tui/index.tsx +38 -0
  310. package/src/tui/types.ts +76 -0
  311. package/src/utils/git.ts +83 -0
  312. package/src/utils/queue-writer.ts +54 -0
  313. package/src/verification/executor.ts +235 -0
  314. package/src/verification/gate.ts +207 -0
  315. package/src/verification/index.ts +12 -0
  316. package/src/verification/parser.ts +230 -0
  317. package/src/verification/rectification.ts +108 -0
  318. package/src/verification/types.ts +113 -0
  319. package/src/worktree/dispatcher.ts +65 -0
  320. package/src/worktree/index.ts +2 -0
  321. package/src/worktree/manager.ts +187 -0
  322. package/src/worktree/merge.ts +301 -0
  323. package/src/worktree/types.ts +4 -0
  324. package/test/TEST_COVERAGE_US001.md +217 -0
  325. package/test/TEST_COVERAGE_US003.md +84 -0
  326. package/test/TEST_COVERAGE_US005.md +86 -0
  327. package/test/US-002-orchestrator.test.ts +246 -0
  328. package/test/acceptance/cm-003-default-view.test.ts +194 -0
  329. package/test/execution/pid-registry.test.ts +240 -0
  330. package/test/execution/post-verify.test.ts +224 -0
  331. package/test/helpers/timeout.ts +42 -0
  332. package/test/integration/US-002-TEST-SUMMARY.md +107 -0
  333. package/test/integration/US-003-TEST-SUMMARY.md +149 -0
  334. package/test/integration/US-004-TEST-SUMMARY.md +106 -0
  335. package/test/integration/US-005-TEST-SUMMARY.md +138 -0
  336. package/test/integration/US-007-TEST-SUMMARY.md +100 -0
  337. package/test/integration/agent-validation.test.ts +439 -0
  338. package/test/integration/analyze-integration.test.ts +261 -0
  339. package/test/integration/analyze-scanner.test.ts +131 -0
  340. package/test/integration/cli-config-default-edge-cases.test.ts +222 -0
  341. package/test/integration/cli-config-default-view.test.ts +229 -0
  342. package/test/integration/cli-config-diff.test.ts +460 -0
  343. package/test/integration/cli-config.test.ts +736 -0
  344. package/test/integration/cli-diagnose.test.ts +592 -0
  345. package/test/integration/cli-logs.test.ts +314 -0
  346. package/test/integration/cli-plugins.test.ts +678 -0
  347. package/test/integration/cli-precheck.test.ts +371 -0
  348. package/test/integration/cli-run-headless.test.ts +173 -0
  349. package/test/integration/cli.test.ts +75 -0
  350. package/test/integration/config/merger.test.ts +465 -0
  351. package/test/integration/config/paths.test.ts +51 -0
  352. package/test/integration/config-loader.test.ts +265 -0
  353. package/test/integration/config.test.ts +444 -0
  354. package/test/integration/context-integration.test.ts +702 -0
  355. package/test/integration/context-provider-injection.test.ts +506 -0
  356. package/test/integration/context-verification-integration.test.ts +295 -0
  357. package/test/integration/e2e.test.ts +896 -0
  358. package/test/integration/execution.test.ts +625 -0
  359. package/test/integration/helpers.test.ts +295 -0
  360. package/test/integration/hooks.test.ts +361 -0
  361. package/test/integration/interaction-chain-pipeline.test.ts +464 -0
  362. package/test/integration/isolation.test.ts +143 -0
  363. package/test/integration/logger.test.ts +461 -0
  364. package/test/integration/parallel.test.ts +250 -0
  365. package/test/integration/path-security.test.ts +173 -0
  366. package/test/integration/pipeline-acceptance.test.ts +302 -0
  367. package/test/integration/pipeline-events.test.ts +475 -0
  368. package/test/integration/pipeline.test.ts +658 -0
  369. package/test/integration/plan.test.ts +157 -0
  370. package/test/integration/plugin-routing.test.ts +921 -0
  371. package/test/integration/plugins/config-integration.test.ts +172 -0
  372. package/test/integration/plugins/config-resolution.test.ts +522 -0
  373. package/test/integration/plugins/loader.test.ts +641 -0
  374. package/test/integration/plugins/registry.test.ts +746 -0
  375. package/test/integration/plugins/validator.test.ts +563 -0
  376. package/test/integration/prd-pause.test.ts +205 -0
  377. package/test/integration/prd-resolvers.test.ts +185 -0
  378. package/test/integration/precheck-integration.test.ts +468 -0
  379. package/test/integration/precheck.test.ts +805 -0
  380. package/test/integration/progress.test.ts +34 -0
  381. package/test/integration/rectification-flow.test.ts +512 -0
  382. package/test/integration/reporter-lifecycle.test.ts +860 -0
  383. package/test/integration/review-config-commands.test.ts +319 -0
  384. package/test/integration/review-config-schema.test.ts +116 -0
  385. package/test/integration/review-plugin-integration.test.ts +722 -0
  386. package/test/integration/review.test.ts +149 -0
  387. package/test/integration/routing-stage-bug-021.test.ts +274 -0
  388. package/test/integration/routing-stage-greenfield.test.ts +286 -0
  389. package/test/integration/runner-config-plugins.test.ts +461 -0
  390. package/test/integration/runner-fixes.test.ts +399 -0
  391. package/test/integration/runner-plugin-integration.test.ts +543 -0
  392. package/test/integration/runner.test.ts +1679 -0
  393. package/test/integration/s5-greenfield-fallback.test.ts +297 -0
  394. package/test/integration/status-file-integration.test.ts +325 -0
  395. package/test/integration/status-file.test.ts +379 -0
  396. package/test/integration/status-writer.test.ts +345 -0
  397. package/test/integration/story-id-in-events.test.ts +273 -0
  398. package/test/integration/tdd-cleanup.test.ts +246 -0
  399. package/test/integration/tdd-orchestrator.test.ts +1762 -0
  400. package/test/integration/test-scanner.test.ts +403 -0
  401. package/test/integration/verification-asset-check.test.ts +142 -0
  402. package/test/integration/verify-stage.test.ts +275 -0
  403. package/test/integration/worktree/manager.test.ts +218 -0
  404. package/test/integration/worktree/merge.test.ts +341 -0
  405. package/test/manual/logging-formatter-demo.ts +158 -0
  406. package/test/ui/tui-agent-panel.test.tsx +99 -0
  407. package/test/ui/tui-controls.test.ts +334 -0
  408. package/test/ui/tui-cost-and-pty.test.ts +189 -0
  409. package/test/ui/tui-layout.test.ts +378 -0
  410. package/test/ui/tui-pty-integration.test.tsx +159 -0
  411. package/test/ui/tui-stories.test.ts +332 -0
  412. package/test/unit/acceptance.test.ts +186 -0
  413. package/test/unit/agent-stderr-capture.test.ts +146 -0
  414. package/test/unit/analyze-classifier.test.ts +215 -0
  415. package/test/unit/analyze.test.ts +224 -0
  416. package/test/unit/auto-detect.test.ts +249 -0
  417. package/test/unit/cli-status.test.ts +417 -0
  418. package/test/unit/commands/common.test.ts +320 -0
  419. package/test/unit/commands/logs.test.ts +416 -0
  420. package/test/unit/commands/unlock.test.ts +319 -0
  421. package/test/unit/constitution-generators.test.ts +160 -0
  422. package/test/unit/constitution.test.ts +209 -0
  423. package/test/unit/context.test.ts +1722 -0
  424. package/test/unit/cost.test.ts +231 -0
  425. package/test/unit/crash-recovery.test.ts +308 -0
  426. package/test/unit/escalation.test.ts +126 -0
  427. package/test/unit/execution-logging-stderr.test.ts +156 -0
  428. package/test/unit/execution-stage.test.ts +122 -0
  429. package/test/unit/fix-generator.test.ts +275 -0
  430. package/test/unit/formatters.test.ts +469 -0
  431. package/test/unit/greenfield.test.ts +179 -0
  432. package/test/unit/helpers.test.ts +317 -0
  433. package/test/unit/interaction/human-review-trigger.test.ts +164 -0
  434. package/test/unit/interaction-network-failures.test.ts +389 -0
  435. package/test/unit/interaction-plugins.test.ts +164 -0
  436. package/test/unit/isolation.test.ts +134 -0
  437. package/test/unit/logging/formatter.test.ts +455 -0
  438. package/test/unit/merge.test.ts +268 -0
  439. package/test/unit/metrics.test.ts +276 -0
  440. package/test/unit/optimizer/noop.optimizer.test.ts +125 -0
  441. package/test/unit/optimizer/rule-based.optimizer.test.ts +358 -0
  442. package/test/unit/prd-auto-default.test.ts +290 -0
  443. package/test/unit/prd-failure-category.test.ts +176 -0
  444. package/test/unit/prd-get-next-story.test.ts +186 -0
  445. package/test/unit/precheck-checks.test.ts +840 -0
  446. package/test/unit/precheck-story-size-gate.test.ts +287 -0
  447. package/test/unit/precheck-types.test.ts +142 -0
  448. package/test/unit/prompts.test.ts +475 -0
  449. package/test/unit/queue.test.ts +237 -0
  450. package/test/unit/rectification.test.ts +284 -0
  451. package/test/unit/registry.test.ts +287 -0
  452. package/test/unit/routing.test.ts +937 -0
  453. package/test/unit/run-lifecycle.test.ts +140 -0
  454. package/test/unit/storyid-events.test.ts +224 -0
  455. package/test/unit/tdd-verdict.test.ts +492 -0
  456. package/test/unit/test-output-parser.test.ts +377 -0
  457. package/test/unit/verdict.test.ts +324 -0
  458. package/test/unit/worktree-manager.test.ts +158 -0
  459. package/tsconfig.json +27 -0
@@ -0,0 +1,139 @@
1
+ /**
2
+ * StatusWriter — Encapsulates status file state and write logic
3
+ *
4
+ * Extracted from runner.ts. Manages the _runStatus, _prd, _currentStory state
5
+ * that was previously tracked as closure variables, plus the BUG-2 consecutive
6
+ * write failure counter. Provides atomic status file writes via writeStatusFile.
7
+ */
8
+
9
+ import type { NaxConfig } from "../config";
10
+ import { getSafeLogger } from "../logger";
11
+ import type { PRD } from "../prd";
12
+ import { type RunStateSnapshot, buildStatusSnapshot, writeStatusFile } from "./status-file";
13
+
14
+ // ============================================================================
15
+ // StatusWriterContext — fixed run metadata set at construction
16
+ // ============================================================================
17
+
18
+ export interface StatusWriterContext {
19
+ /** Unique run identifier */
20
+ runId: string;
21
+ /** Feature name */
22
+ feature: string;
23
+ /** ISO 8601 run start timestamp */
24
+ startedAt: string;
25
+ /** Whether this is a dry run */
26
+ dryRun: boolean;
27
+ /** Run start time as ms epoch (for computing durationMs) */
28
+ startTimeMs: number;
29
+ /** Process ID for crash detection */
30
+ pid: number;
31
+ }
32
+
33
+ // ============================================================================
34
+ // StatusWriter
35
+ // ============================================================================
36
+
37
+ /**
38
+ * Manages status file state and write logic for the execution runner.
39
+ *
40
+ * Encapsulates all status-file-related mutable state that was previously
41
+ * tracked as closure variables in runner.ts:
42
+ * - _runStatus, _prd, _currentStory (set via setters before each write)
43
+ * - _consecutiveWriteFailures (BUG-2 failure counter, reset on success)
44
+ *
45
+ * Usage:
46
+ * const sw = new StatusWriter(statusFile, config, { runId, feature, ... });
47
+ * sw.setPrd(prd);
48
+ * sw.setRunStatus("running");
49
+ * sw.setCurrentStory(null);
50
+ * await sw.update(totalCost, iterations);
51
+ */
52
+ export class StatusWriter {
53
+ private readonly statusFile: string | undefined;
54
+ private readonly costLimit: number | null;
55
+ private readonly ctx: StatusWriterContext;
56
+
57
+ // Encapsulated mutable state (was closure vars in runner.ts)
58
+ private _runStatus: RunStateSnapshot["runStatus"] = "running";
59
+ private _prd: PRD | null = null;
60
+ private _currentStory: RunStateSnapshot["currentStory"] = null;
61
+ private _consecutiveWriteFailures = 0; // BUG-2: Track consecutive write failures
62
+
63
+ constructor(statusFile: string | undefined, config: NaxConfig, ctx: StatusWriterContext) {
64
+ this.statusFile = statusFile;
65
+ this.costLimit = config.execution.costLimit === Number.POSITIVE_INFINITY ? null : config.execution.costLimit;
66
+ this.ctx = ctx;
67
+ }
68
+
69
+ /** Update the current run status (running / completed / failed / stalled / crashed / precheck-failed) */
70
+ setRunStatus(status: RunStateSnapshot["runStatus"]): void {
71
+ this._runStatus = status;
72
+ }
73
+
74
+ /** Update the loaded PRD used for progress counting */
75
+ setPrd(prd: PRD): void {
76
+ this._prd = prd;
77
+ }
78
+
79
+ /** Update the currently-executing story info (null between stories) */
80
+ setCurrentStory(story: RunStateSnapshot["currentStory"]): void {
81
+ this._currentStory = story;
82
+ }
83
+
84
+ /**
85
+ * Build a RunStateSnapshot from current state + live runner values.
86
+ *
87
+ * Returns null if no PRD has been set yet (status write is a no-op).
88
+ */
89
+ getSnapshot(totalCost: number, iterations: number): RunStateSnapshot | null {
90
+ if (!this._prd) return null;
91
+ return {
92
+ runId: this.ctx.runId,
93
+ feature: this.ctx.feature,
94
+ startedAt: this.ctx.startedAt,
95
+ runStatus: this._runStatus,
96
+ dryRun: this.ctx.dryRun,
97
+ pid: this.ctx.pid,
98
+ prd: this._prd,
99
+ totalCost,
100
+ costLimit: this.costLimit,
101
+ iterations,
102
+ startTimeMs: this.ctx.startTimeMs,
103
+ currentStory: this._currentStory,
104
+ };
105
+ }
106
+
107
+ /**
108
+ * Write the current status to disk (atomic via .tmp + rename).
109
+ *
110
+ * No-ops if statusFile was not provided or _prd has not been set.
111
+ * On failure, logs a warning/error and increments the BUG-2 failure counter.
112
+ * Counter resets to 0 on next successful write.
113
+ *
114
+ * @param totalCost - Accumulated cost at this write point
115
+ * @param iterations - Loop iteration count at this write point
116
+ * @param overrides - Optional partial snapshot overrides (spread last)
117
+ */
118
+ async update(totalCost: number, iterations: number, overrides: Partial<RunStateSnapshot> = {}): Promise<void> {
119
+ if (!this.statusFile || !this._prd) return;
120
+ const safeLogger = getSafeLogger();
121
+ try {
122
+ const base = this.getSnapshot(totalCost, iterations);
123
+ if (!base) {
124
+ throw new Error("Failed to get snapshot");
125
+ }
126
+ const state: RunStateSnapshot = { ...base, ...overrides };
127
+ await writeStatusFile(this.statusFile, buildStatusSnapshot(state));
128
+ this._consecutiveWriteFailures = 0; // Reset counter on success
129
+ } catch (err) {
130
+ this._consecutiveWriteFailures++;
131
+ const logLevel = this._consecutiveWriteFailures >= 3 ? "error" : "warn";
132
+ safeLogger?.[logLevel]("status-file", "Failed to write status file (non-fatal)", {
133
+ path: this.statusFile,
134
+ error: (err as Error).message,
135
+ consecutiveFailures: this._consecutiveWriteFailures,
136
+ });
137
+ }
138
+ }
139
+ }
@@ -0,0 +1,229 @@
1
+ /**
2
+ * Story Context Building
3
+ *
4
+ * Extracted from helpers.ts: context building and story readiness functions.
5
+ */
6
+
7
+ import type { NaxConfig } from "../config";
8
+ import { buildContext, formatContextAsMarkdown } from "../context";
9
+ import type { BuiltContext, ContextBudget, StoryContext } from "../context";
10
+ import type { HookContext } from "../hooks";
11
+ import { getLogger } from "../logger";
12
+ import type { PRD, UserStory } from "../prd";
13
+
14
+ /** Safely get logger instance, returns null if not initialized */
15
+ function getSafeLogger() {
16
+ try {
17
+ return getLogger();
18
+ } catch {
19
+ return null;
20
+ }
21
+ }
22
+
23
+ /**
24
+ * Maximum tokens allowed in context budget for Claude agents.
25
+ *
26
+ * Claude has 200k token context window, but we reserve space for system prompt,
27
+ * agent instructions, conversation history, and output buffer. 100k token limit
28
+ * for story context is conservative but safe, leaving 100k tokens for agent
29
+ * reasoning and responses.
30
+ */
31
+ const CONTEXT_MAX_TOKENS = 100_000;
32
+
33
+ /**
34
+ * Tokens reserved for agent instructions and prompts.
35
+ *
36
+ * Agent needs space for task instructions, TDD prompts, hook output.
37
+ * 10k token reservation is safe upper bound for all instruction scenarios.
38
+ */
39
+ const CONTEXT_RESERVED_TOKENS = 10_000;
40
+
41
+ /** Result from executing a batch or single story */
42
+ export interface ExecutionResult {
43
+ success: boolean;
44
+ cost: number;
45
+ storiesProcessed: string[];
46
+ }
47
+
48
+ /**
49
+ * Build a hook context object
50
+ *
51
+ * @param feature - Feature name
52
+ * @param opts - Optional context fields to override
53
+ * @returns Hook context with event set to "on-start" (overridden by fireHook)
54
+ */
55
+ export function hookCtx(feature: string, opts?: Partial<Omit<HookContext, "event" | "feature">>): HookContext {
56
+ return {
57
+ event: "on-start", // overridden by fireHook
58
+ feature,
59
+ ...opts,
60
+ };
61
+ }
62
+
63
+ /**
64
+ * Maybe build context if enabled
65
+ *
66
+ * @param prd - PRD to build context from
67
+ * @param story - Current story being executed
68
+ * @param config - Ngent config
69
+ * @param useContext - Whether to build context
70
+ * @returns Context markdown or undefined if disabled/failed
71
+ */
72
+ export async function maybeGetContext(
73
+ prd: PRD,
74
+ story: UserStory,
75
+ config: NaxConfig,
76
+ useContext: boolean,
77
+ ): Promise<string | undefined> {
78
+ if (!useContext) {
79
+ return undefined;
80
+ }
81
+
82
+ const logger = getSafeLogger();
83
+ logger?.debug("context", "Building context...");
84
+ const contextMarkdown = await buildStoryContext(prd, story, config);
85
+ if (contextMarkdown) {
86
+ logger?.debug("context", "Context built successfully");
87
+ }
88
+ return contextMarkdown;
89
+ }
90
+
91
+ /**
92
+ * Build story context for context builder
93
+ *
94
+ * @param prd - PRD containing all stories
95
+ * @param story - Current story to build context for
96
+ * @param config - Ngent config
97
+ * @returns Context markdown or undefined if no context available
98
+ */
99
+ export async function buildStoryContext(prd: PRD, story: UserStory, _config: NaxConfig): Promise<string | undefined> {
100
+ try {
101
+ const storyContext: StoryContext = {
102
+ prd,
103
+ currentStoryId: story.id,
104
+ workdir: process.cwd(),
105
+ config: _config,
106
+ };
107
+
108
+ const budget: ContextBudget = {
109
+ maxTokens: CONTEXT_MAX_TOKENS,
110
+ reservedForInstructions: CONTEXT_RESERVED_TOKENS,
111
+ availableForContext: CONTEXT_MAX_TOKENS - CONTEXT_RESERVED_TOKENS,
112
+ };
113
+
114
+ const built = await buildContext(storyContext, budget);
115
+
116
+ if (built.elements.length === 0) {
117
+ return undefined;
118
+ }
119
+
120
+ return formatContextAsMarkdown(built);
121
+ } catch (error) {
122
+ const logger = getSafeLogger();
123
+ logger?.warn("context", "Context builder failed", {
124
+ error: (error as Error).message,
125
+ });
126
+ return undefined;
127
+ }
128
+ }
129
+
130
+ /**
131
+ * Build story context returning both markdown and element-level data.
132
+ * Used by `nax prompts` CLI for accurate frontmatter token counts.
133
+ */
134
+ export async function buildStoryContextFull(
135
+ prd: PRD,
136
+ story: UserStory,
137
+ config: NaxConfig,
138
+ ): Promise<{ markdown: string; builtContext: BuiltContext } | undefined> {
139
+ try {
140
+ const storyContext: StoryContext = {
141
+ prd,
142
+ currentStoryId: story.id,
143
+ workdir: process.cwd(),
144
+ config,
145
+ };
146
+
147
+ const budget: ContextBudget = {
148
+ maxTokens: CONTEXT_MAX_TOKENS,
149
+ reservedForInstructions: CONTEXT_RESERVED_TOKENS,
150
+ availableForContext: CONTEXT_MAX_TOKENS - CONTEXT_RESERVED_TOKENS,
151
+ };
152
+
153
+ const built = await buildContext(storyContext, budget);
154
+
155
+ if (built.elements.length === 0) {
156
+ return undefined;
157
+ }
158
+
159
+ return { markdown: formatContextAsMarkdown(built), builtContext: built };
160
+ } catch (error) {
161
+ const logger = getSafeLogger();
162
+ logger?.warn("context", "Context builder failed", {
163
+ error: (error as Error).message,
164
+ });
165
+ return undefined;
166
+ }
167
+ }
168
+
169
+ /**
170
+ * Get all stories that are ready to execute (pending, dependencies satisfied)
171
+ *
172
+ * @param prd - PRD containing all stories
173
+ * @returns Array of stories that can be executed now
174
+ */
175
+ export function getAllReadyStories(prd: PRD): UserStory[] {
176
+ const completedIds = new Set(prd.userStories.filter((s) => s.passes || s.status === "skipped").map((s) => s.id));
177
+
178
+ return prd.userStories.filter(
179
+ (s) =>
180
+ !s.passes &&
181
+ s.status !== "skipped" &&
182
+ s.status !== "failed" &&
183
+ s.status !== "paused" &&
184
+ s.status !== "blocked" &&
185
+ s.dependencies.every((dep) => completedIds.has(dep)),
186
+ );
187
+ }
188
+
189
+ /** Story counts for progress display */
190
+ export interface StoryCounts {
191
+ total: number;
192
+ passed: number;
193
+ failed: number;
194
+ pending: number;
195
+ }
196
+
197
+ /**
198
+ * Format a progress line with counts, cost, and ETA
199
+ *
200
+ * @param counts - Story counts (total, passed, failed, pending)
201
+ * @param totalCost - Total cost so far
202
+ * @param costLimit - Cost limit from config
203
+ * @param elapsedMs - Elapsed time in milliseconds
204
+ * @param totalStories - Total number of stories (for ETA calculation)
205
+ * @returns Formatted progress string
206
+ */
207
+ export function formatProgress(
208
+ counts: StoryCounts,
209
+ totalCost: number,
210
+ costLimit: number,
211
+ elapsedMs: number,
212
+ totalStories: number,
213
+ ): string {
214
+ const completedStories = counts.passed + counts.failed;
215
+ const remainingStories = totalStories - completedStories;
216
+
217
+ // Calculate ETA from average story duration
218
+ let etaText = "calculating...";
219
+ if (completedStories > 0 && remainingStories > 0) {
220
+ const avgDurationPerStory = elapsedMs / completedStories;
221
+ const etaMs = avgDurationPerStory * remainingStories;
222
+ const etaMinutes = Math.round(etaMs / 1000 / 60);
223
+ etaText = `~${etaMinutes} min remaining`;
224
+ } else if (remainingStories === 0) {
225
+ etaText = "complete";
226
+ }
227
+
228
+ return `Progress: ${completedStories}/${totalStories} stories | ${counts.passed} passed | ${counts.failed} failed | $${totalCost.toFixed(2)}/$${costLimit.toFixed(2)} | ${etaText}`;
229
+ }
@@ -0,0 +1,14 @@
1
+ /**
2
+ * Test Output Parser
3
+ *
4
+ * DEPRECATED: Use src/verification/parser.ts instead.
5
+ * This file is kept for backward compatibility only.
6
+ */
7
+
8
+ // Re-export from unified verification layer
9
+ export {
10
+ type TestFailure,
11
+ type TestSummary,
12
+ parseBunTestOutput,
13
+ formatFailureSummary,
14
+ } from "../verification/parser";
@@ -0,0 +1,72 @@
1
+ /**
2
+ * ADR-003: Robust Orchestration Feedback Loop - Verification Module
3
+ *
4
+ * DEPRECATED: Use src/verification/ unified layer instead.
5
+ * This file is kept for backward compatibility only.
6
+ *
7
+ * Implements Decisions 3-6:
8
+ * - Pre-Flight Asset Verification
9
+ * - Execution Guard (Verification Timeout)
10
+ * - Smart Exit-Code Analysis
11
+ * - Environment Normalization
12
+ */
13
+
14
+ // Re-export from unified verification layer
15
+ export {
16
+ type AssetVerificationResult,
17
+ type TestExecutionResult as TimeoutExecutionResult,
18
+ type TestOutputAnalysis,
19
+ type VerificationResult,
20
+ type VerificationStatus,
21
+ verifyAssets,
22
+ executeWithTimeout,
23
+ parseTestOutput,
24
+ getEnvironmentalEscalationThreshold,
25
+ normalizeEnvironment,
26
+ appendOpenHandlesFlag,
27
+ appendForceExitFlag,
28
+ buildTestCommand,
29
+ } from "../verification";
30
+
31
+ // Adapter function for backward compatibility
32
+ import { fullSuite } from "../verification";
33
+ import type { VerificationGateOptions } from "../verification/types";
34
+
35
+ /**
36
+ * Run complete verification flow with all ADR-003 safety checks.
37
+ *
38
+ * @deprecated Use fullSuite() from src/verification/gate.ts instead
39
+ */
40
+ export async function runVerification(options: {
41
+ workingDirectory: string;
42
+ expectedFiles?: string[];
43
+ command: string;
44
+ timeoutSeconds: number;
45
+ forceExit?: boolean;
46
+ detectOpenHandles?: boolean;
47
+ detectOpenHandlesRetries?: number;
48
+ timeoutRetryCount?: number;
49
+ gracePeriodMs?: number;
50
+ drainTimeoutMs?: number;
51
+ shell?: string;
52
+ stripEnvVars?: string[];
53
+ cwd?: string;
54
+ }) {
55
+ // Map old options to new VerificationGateOptions
56
+ const gateOptions: VerificationGateOptions = {
57
+ workdir: options.workingDirectory,
58
+ expectedFiles: options.expectedFiles,
59
+ command: options.command,
60
+ timeoutSeconds: options.timeoutSeconds,
61
+ forceExit: options.forceExit,
62
+ detectOpenHandles: options.detectOpenHandles,
63
+ detectOpenHandlesRetries: options.detectOpenHandlesRetries,
64
+ timeoutRetryCount: options.timeoutRetryCount,
65
+ gracePeriodMs: options.gracePeriodMs,
66
+ drainTimeoutMs: options.drainTimeoutMs,
67
+ shell: options.shell,
68
+ stripEnvVars: options.stripEnvVars,
69
+ };
70
+
71
+ return fullSuite(gateOptions);
72
+ }
@@ -0,0 +1,2 @@
1
+ export type { HookEvent, HookDef, HooksConfig, HookContext } from "./types";
2
+ export { loadHooksConfig, fireHook, type LoadedHooksConfig } from "./runner";