@nathapp/nax 0.50.3 → 0.51.2

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 (353) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/README.md +177 -104
  3. package/dist/nax.js +417 -213
  4. package/package.json +1 -3
  5. package/bin/nax.ts +0 -1195
  6. package/src/acceptance/fix-generator.ts +0 -322
  7. package/src/acceptance/generator.ts +0 -415
  8. package/src/acceptance/index.ts +0 -42
  9. package/src/acceptance/refinement.ts +0 -224
  10. package/src/acceptance/templates/cli.ts +0 -47
  11. package/src/acceptance/templates/component.ts +0 -78
  12. package/src/acceptance/templates/e2e.ts +0 -43
  13. package/src/acceptance/templates/index.ts +0 -21
  14. package/src/acceptance/templates/snapshot.ts +0 -50
  15. package/src/acceptance/templates/unit.ts +0 -48
  16. package/src/acceptance/types.ts +0 -138
  17. package/src/agents/acp/adapter.ts +0 -888
  18. package/src/agents/acp/cost.ts +0 -9
  19. package/src/agents/acp/index.ts +0 -7
  20. package/src/agents/acp/interaction-bridge.ts +0 -126
  21. package/src/agents/acp/parser.ts +0 -119
  22. package/src/agents/acp/spawn-client.ts +0 -373
  23. package/src/agents/acp/types.ts +0 -22
  24. package/src/agents/aider/adapter.ts +0 -135
  25. package/src/agents/claude/adapter.ts +0 -258
  26. package/src/agents/claude/complete.ts +0 -80
  27. package/src/agents/claude/cost.ts +0 -16
  28. package/src/agents/claude/execution.ts +0 -215
  29. package/src/agents/claude/index.ts +0 -3
  30. package/src/agents/claude/interactive.ts +0 -77
  31. package/src/agents/claude/plan.ts +0 -179
  32. package/src/agents/codex/adapter.ts +0 -153
  33. package/src/agents/cost/calculate.ts +0 -154
  34. package/src/agents/cost/index.ts +0 -10
  35. package/src/agents/cost/parse.ts +0 -97
  36. package/src/agents/cost/pricing.ts +0 -59
  37. package/src/agents/cost/types.ts +0 -45
  38. package/src/agents/gemini/adapter.ts +0 -177
  39. package/src/agents/index.ts +0 -18
  40. package/src/agents/opencode/adapter.ts +0 -106
  41. package/src/agents/registry.ts +0 -136
  42. package/src/agents/shared/decompose.ts +0 -154
  43. package/src/agents/shared/model-resolution.ts +0 -43
  44. package/src/agents/shared/types-extended.ts +0 -164
  45. package/src/agents/shared/validation.ts +0 -69
  46. package/src/agents/shared/version-detection.ts +0 -109
  47. package/src/agents/types.ts +0 -205
  48. package/src/analyze/classifier.ts +0 -282
  49. package/src/analyze/index.ts +0 -16
  50. package/src/analyze/scanner.ts +0 -171
  51. package/src/analyze/types.ts +0 -51
  52. package/src/cli/accept.ts +0 -108
  53. package/src/cli/agents.ts +0 -87
  54. package/src/cli/analyze-parser.ts +0 -291
  55. package/src/cli/analyze.ts +0 -352
  56. package/src/cli/config-descriptions.ts +0 -219
  57. package/src/cli/config-diff.ts +0 -103
  58. package/src/cli/config-display.ts +0 -285
  59. package/src/cli/config-get.ts +0 -55
  60. package/src/cli/config.ts +0 -14
  61. package/src/cli/constitution.ts +0 -17
  62. package/src/cli/diagnose-analysis.ts +0 -159
  63. package/src/cli/diagnose-formatter.ts +0 -87
  64. package/src/cli/diagnose.ts +0 -203
  65. package/src/cli/generate.ts +0 -250
  66. package/src/cli/index.ts +0 -42
  67. package/src/cli/init-context.ts +0 -405
  68. package/src/cli/init-detect.ts +0 -303
  69. package/src/cli/init.ts +0 -296
  70. package/src/cli/interact.ts +0 -295
  71. package/src/cli/plan.ts +0 -509
  72. package/src/cli/plugins.ts +0 -122
  73. package/src/cli/prompts-export.ts +0 -58
  74. package/src/cli/prompts-init.ts +0 -200
  75. package/src/cli/prompts-main.ts +0 -183
  76. package/src/cli/prompts-shared.ts +0 -70
  77. package/src/cli/prompts-tdd.ts +0 -88
  78. package/src/cli/prompts.ts +0 -17
  79. package/src/cli/runs.ts +0 -174
  80. package/src/cli/status-cost.ts +0 -151
  81. package/src/cli/status-features.ts +0 -405
  82. package/src/cli/status.ts +0 -13
  83. package/src/commands/common.ts +0 -171
  84. package/src/commands/diagnose.ts +0 -17
  85. package/src/commands/index.ts +0 -9
  86. package/src/commands/logs-formatter.ts +0 -201
  87. package/src/commands/logs-reader.ts +0 -171
  88. package/src/commands/logs.ts +0 -103
  89. package/src/commands/precheck.ts +0 -86
  90. package/src/commands/runs.ts +0 -220
  91. package/src/commands/unlock.ts +0 -96
  92. package/src/config/defaults.ts +0 -218
  93. package/src/config/index.ts +0 -22
  94. package/src/config/loader.ts +0 -143
  95. package/src/config/merge.ts +0 -106
  96. package/src/config/merger.ts +0 -147
  97. package/src/config/path-security.ts +0 -121
  98. package/src/config/paths.ts +0 -27
  99. package/src/config/permissions.ts +0 -63
  100. package/src/config/runtime-types.ts +0 -522
  101. package/src/config/schema-types.ts +0 -53
  102. package/src/config/schema.ts +0 -60
  103. package/src/config/schemas.ts +0 -426
  104. package/src/config/test-strategy.ts +0 -71
  105. package/src/config/types.ts +0 -57
  106. package/src/config/validate.ts +0 -103
  107. package/src/constitution/generator.ts +0 -158
  108. package/src/constitution/generators/aider.ts +0 -41
  109. package/src/constitution/generators/claude.ts +0 -35
  110. package/src/constitution/generators/cursor.ts +0 -36
  111. package/src/constitution/generators/opencode.ts +0 -38
  112. package/src/constitution/generators/types.ts +0 -33
  113. package/src/constitution/generators/windsurf.ts +0 -36
  114. package/src/constitution/index.ts +0 -11
  115. package/src/constitution/loader.ts +0 -121
  116. package/src/constitution/types.ts +0 -31
  117. package/src/context/auto-detect.ts +0 -228
  118. package/src/context/builder.ts +0 -299
  119. package/src/context/elements.ts +0 -122
  120. package/src/context/formatter.ts +0 -107
  121. package/src/context/generator.ts +0 -343
  122. package/src/context/generators/aider.ts +0 -34
  123. package/src/context/generators/claude.ts +0 -28
  124. package/src/context/generators/codex.ts +0 -28
  125. package/src/context/generators/cursor.ts +0 -28
  126. package/src/context/generators/gemini.ts +0 -28
  127. package/src/context/generators/opencode.ts +0 -30
  128. package/src/context/generators/windsurf.ts +0 -28
  129. package/src/context/greenfield.ts +0 -114
  130. package/src/context/index.ts +0 -34
  131. package/src/context/injector.ts +0 -279
  132. package/src/context/parent-context.ts +0 -39
  133. package/src/context/test-scanner.ts +0 -370
  134. package/src/context/types.ts +0 -98
  135. package/src/decompose/apply.ts +0 -50
  136. package/src/decompose/builder.ts +0 -181
  137. package/src/decompose/index.ts +0 -8
  138. package/src/decompose/sections/codebase.ts +0 -26
  139. package/src/decompose/sections/constraints.ts +0 -32
  140. package/src/decompose/sections/index.ts +0 -4
  141. package/src/decompose/sections/sibling-stories.ts +0 -25
  142. package/src/decompose/sections/target-story.ts +0 -31
  143. package/src/decompose/types.ts +0 -55
  144. package/src/decompose/validators/complexity.ts +0 -45
  145. package/src/decompose/validators/coverage.ts +0 -134
  146. package/src/decompose/validators/dependency.ts +0 -91
  147. package/src/decompose/validators/index.ts +0 -35
  148. package/src/decompose/validators/overlap.ts +0 -128
  149. package/src/errors.ts +0 -67
  150. package/src/execution/batching.ts +0 -157
  151. package/src/execution/crash-heartbeat.ts +0 -77
  152. package/src/execution/crash-recovery.ts +0 -79
  153. package/src/execution/crash-signals.ts +0 -165
  154. package/src/execution/crash-writer.ts +0 -154
  155. package/src/execution/deferred-review.ts +0 -105
  156. package/src/execution/dry-run.ts +0 -81
  157. package/src/execution/escalation/escalation.ts +0 -46
  158. package/src/execution/escalation/index.ts +0 -13
  159. package/src/execution/escalation/tier-escalation.ts +0 -346
  160. package/src/execution/escalation/tier-outcome.ts +0 -143
  161. package/src/execution/executor-types.ts +0 -73
  162. package/src/execution/helpers.ts +0 -38
  163. package/src/execution/index.ts +0 -27
  164. package/src/execution/iteration-runner.ts +0 -160
  165. package/src/execution/lifecycle/acceptance-loop.ts +0 -309
  166. package/src/execution/lifecycle/headless-formatter.ts +0 -83
  167. package/src/execution/lifecycle/index.ts +0 -11
  168. package/src/execution/lifecycle/parallel-lifecycle.ts +0 -101
  169. package/src/execution/lifecycle/precheck-runner.ts +0 -140
  170. package/src/execution/lifecycle/run-cleanup.ts +0 -81
  171. package/src/execution/lifecycle/run-completion.ts +0 -247
  172. package/src/execution/lifecycle/run-initialization.ts +0 -187
  173. package/src/execution/lifecycle/run-regression.ts +0 -305
  174. package/src/execution/lifecycle/run-setup.ts +0 -240
  175. package/src/execution/lifecycle/story-size-prompts.ts +0 -123
  176. package/src/execution/lock.ts +0 -129
  177. package/src/execution/parallel-coordinator.ts +0 -281
  178. package/src/execution/parallel-executor-rectification-pass.ts +0 -117
  179. package/src/execution/parallel-executor-rectify.ts +0 -136
  180. package/src/execution/parallel-executor.ts +0 -330
  181. package/src/execution/parallel-worker.ts +0 -149
  182. package/src/execution/parallel.ts +0 -13
  183. package/src/execution/pid-registry.ts +0 -275
  184. package/src/execution/pipeline-result-handler.ts +0 -221
  185. package/src/execution/progress.ts +0 -27
  186. package/src/execution/queue-handler.ts +0 -109
  187. package/src/execution/runner-completion.ts +0 -171
  188. package/src/execution/runner-execution.ts +0 -243
  189. package/src/execution/runner-setup.ts +0 -86
  190. package/src/execution/runner.ts +0 -265
  191. package/src/execution/sequential-executor.ts +0 -219
  192. package/src/execution/status-file.ts +0 -264
  193. package/src/execution/status-writer.ts +0 -181
  194. package/src/execution/story-context.ts +0 -266
  195. package/src/execution/story-selector.ts +0 -76
  196. package/src/execution/test-output-parser.ts +0 -14
  197. package/src/execution/timeout-handler.ts +0 -100
  198. package/src/hooks/index.ts +0 -2
  199. package/src/hooks/runner.ts +0 -280
  200. package/src/hooks/types.ts +0 -79
  201. package/src/interaction/chain.ts +0 -170
  202. package/src/interaction/index.ts +0 -61
  203. package/src/interaction/init.ts +0 -84
  204. package/src/interaction/plugins/auto.ts +0 -243
  205. package/src/interaction/plugins/cli.ts +0 -300
  206. package/src/interaction/plugins/telegram.ts +0 -384
  207. package/src/interaction/plugins/webhook.ts +0 -286
  208. package/src/interaction/state.ts +0 -171
  209. package/src/interaction/triggers.ts +0 -250
  210. package/src/interaction/types.ts +0 -170
  211. package/src/logger/formatters.ts +0 -84
  212. package/src/logger/index.ts +0 -16
  213. package/src/logger/logger.ts +0 -296
  214. package/src/logger/types.ts +0 -48
  215. package/src/logging/formatter.ts +0 -355
  216. package/src/logging/index.ts +0 -22
  217. package/src/logging/types.ts +0 -93
  218. package/src/metrics/aggregator.ts +0 -191
  219. package/src/metrics/index.ts +0 -14
  220. package/src/metrics/tracker.ts +0 -200
  221. package/src/metrics/types.ts +0 -115
  222. package/src/optimizer/index.ts +0 -63
  223. package/src/optimizer/noop.optimizer.ts +0 -24
  224. package/src/optimizer/rule-based.optimizer.ts +0 -248
  225. package/src/optimizer/types.ts +0 -53
  226. package/src/pipeline/event-bus.ts +0 -297
  227. package/src/pipeline/events.ts +0 -130
  228. package/src/pipeline/index.ts +0 -19
  229. package/src/pipeline/runner.ts +0 -149
  230. package/src/pipeline/stages/acceptance-setup.ts +0 -144
  231. package/src/pipeline/stages/acceptance.ts +0 -215
  232. package/src/pipeline/stages/autofix.ts +0 -262
  233. package/src/pipeline/stages/completion.ts +0 -110
  234. package/src/pipeline/stages/constitution.ts +0 -63
  235. package/src/pipeline/stages/context.ts +0 -122
  236. package/src/pipeline/stages/execution.ts +0 -359
  237. package/src/pipeline/stages/index.ts +0 -86
  238. package/src/pipeline/stages/optimizer.ts +0 -74
  239. package/src/pipeline/stages/prompt.ts +0 -79
  240. package/src/pipeline/stages/queue-check.ts +0 -103
  241. package/src/pipeline/stages/rectify.ts +0 -101
  242. package/src/pipeline/stages/regression.ts +0 -99
  243. package/src/pipeline/stages/review.ts +0 -94
  244. package/src/pipeline/stages/routing.ts +0 -276
  245. package/src/pipeline/stages/verify.ts +0 -286
  246. package/src/pipeline/subscribers/events-writer.ts +0 -135
  247. package/src/pipeline/subscribers/hooks.ts +0 -179
  248. package/src/pipeline/subscribers/interaction.ts +0 -103
  249. package/src/pipeline/subscribers/registry.ts +0 -73
  250. package/src/pipeline/subscribers/reporters.ts +0 -174
  251. package/src/pipeline/types.ts +0 -220
  252. package/src/plugins/extensions.ts +0 -225
  253. package/src/plugins/index.ts +0 -33
  254. package/src/plugins/loader.ts +0 -352
  255. package/src/plugins/plugin-logger.ts +0 -41
  256. package/src/plugins/registry.ts +0 -168
  257. package/src/plugins/types.ts +0 -206
  258. package/src/plugins/validator.ts +0 -352
  259. package/src/prd/index.ts +0 -220
  260. package/src/prd/schema.ts +0 -268
  261. package/src/prd/types.ts +0 -273
  262. package/src/prd/validate.ts +0 -41
  263. package/src/precheck/checks-agents.ts +0 -63
  264. package/src/precheck/checks-blockers.ts +0 -23
  265. package/src/precheck/checks-cli.ts +0 -68
  266. package/src/precheck/checks-config.ts +0 -102
  267. package/src/precheck/checks-git.ts +0 -117
  268. package/src/precheck/checks-system.ts +0 -101
  269. package/src/precheck/checks-warnings.ts +0 -221
  270. package/src/precheck/checks.ts +0 -36
  271. package/src/precheck/index.ts +0 -374
  272. package/src/precheck/story-size-gate.ts +0 -144
  273. package/src/precheck/types.ts +0 -31
  274. package/src/prompts/builder.ts +0 -166
  275. package/src/prompts/index.ts +0 -2
  276. package/src/prompts/loader.ts +0 -43
  277. package/src/prompts/sections/conventions.ts +0 -19
  278. package/src/prompts/sections/hermetic.ts +0 -41
  279. package/src/prompts/sections/index.ts +0 -12
  280. package/src/prompts/sections/isolation.ts +0 -70
  281. package/src/prompts/sections/role-task.ts +0 -182
  282. package/src/prompts/sections/story.ts +0 -55
  283. package/src/prompts/sections/verdict.ts +0 -70
  284. package/src/prompts/types.ts +0 -21
  285. package/src/queue/index.ts +0 -2
  286. package/src/queue/manager.ts +0 -254
  287. package/src/queue/types.ts +0 -54
  288. package/src/review/index.ts +0 -8
  289. package/src/review/orchestrator.ts +0 -154
  290. package/src/review/runner.ts +0 -303
  291. package/src/review/types.ts +0 -70
  292. package/src/routing/batch-route.ts +0 -35
  293. package/src/routing/builder.ts +0 -81
  294. package/src/routing/chain.ts +0 -75
  295. package/src/routing/content-hash.ts +0 -25
  296. package/src/routing/index.ts +0 -20
  297. package/src/routing/loader.ts +0 -62
  298. package/src/routing/router.ts +0 -305
  299. package/src/routing/strategies/adaptive.ts +0 -215
  300. package/src/routing/strategies/index.ts +0 -8
  301. package/src/routing/strategies/keyword.ts +0 -180
  302. package/src/routing/strategies/llm-prompts.ts +0 -224
  303. package/src/routing/strategies/llm.ts +0 -320
  304. package/src/routing/strategies/manual.ts +0 -50
  305. package/src/routing/strategy.ts +0 -102
  306. package/src/tdd/cleanup.ts +0 -120
  307. package/src/tdd/index.ts +0 -22
  308. package/src/tdd/isolation.ts +0 -117
  309. package/src/tdd/orchestrator.ts +0 -406
  310. package/src/tdd/prompts.ts +0 -40
  311. package/src/tdd/rectification-gate.ts +0 -274
  312. package/src/tdd/session-runner.ts +0 -263
  313. package/src/tdd/types.ts +0 -84
  314. package/src/tdd/verdict-reader.ts +0 -266
  315. package/src/tdd/verdict.ts +0 -152
  316. package/src/tui/App.tsx +0 -265
  317. package/src/tui/components/AgentPanel.tsx +0 -75
  318. package/src/tui/components/CostOverlay.tsx +0 -118
  319. package/src/tui/components/HelpOverlay.tsx +0 -107
  320. package/src/tui/components/StatusBar.tsx +0 -63
  321. package/src/tui/components/StoriesPanel.tsx +0 -177
  322. package/src/tui/hooks/useKeyboard.ts +0 -142
  323. package/src/tui/hooks/useLayout.ts +0 -137
  324. package/src/tui/hooks/usePipelineEvents.ts +0 -183
  325. package/src/tui/hooks/usePty.ts +0 -189
  326. package/src/tui/index.tsx +0 -38
  327. package/src/tui/types.ts +0 -76
  328. package/src/utils/errors.ts +0 -12
  329. package/src/utils/git.ts +0 -245
  330. package/src/utils/json-file.ts +0 -72
  331. package/src/utils/log-test-output.ts +0 -25
  332. package/src/utils/path-security.ts +0 -73
  333. package/src/utils/queue-writer.ts +0 -54
  334. package/src/verification/crash-detector.ts +0 -34
  335. package/src/verification/executor.ts +0 -250
  336. package/src/verification/index.ts +0 -12
  337. package/src/verification/orchestrator-types.ts +0 -154
  338. package/src/verification/orchestrator.ts +0 -76
  339. package/src/verification/parser.ts +0 -220
  340. package/src/verification/rectification-loop.ts +0 -172
  341. package/src/verification/rectification.ts +0 -108
  342. package/src/verification/runners.ts +0 -129
  343. package/src/verification/smart-runner.ts +0 -307
  344. package/src/verification/strategies/acceptance.ts +0 -136
  345. package/src/verification/strategies/regression.ts +0 -90
  346. package/src/verification/strategies/scoped.ts +0 -154
  347. package/src/verification/types.ts +0 -117
  348. package/src/version.ts +0 -40
  349. package/src/worktree/dispatcher.ts +0 -6
  350. package/src/worktree/index.ts +0 -2
  351. package/src/worktree/manager.ts +0 -193
  352. package/src/worktree/merge.ts +0 -302
  353. package/src/worktree/types.ts +0 -4
@@ -1,309 +0,0 @@
1
- /**
2
- * Acceptance Retry Loop
3
- *
4
- * Handles the acceptance testing retry loop after main execution completes:
5
- * 1. Runs acceptance validation
6
- * 2. Generates fix stories for failed acceptance criteria
7
- * 3. Executes fix stories through pipeline
8
- * 4. Retries until max retries or all tests pass
9
- */
10
-
11
- import path from "node:path";
12
- import { type FixStory, convertFixStoryToUserStory, generateFixStories } from "../../acceptance";
13
- import type { NaxConfig } from "../../config";
14
- import { resolveModel } from "../../config/schema";
15
- import { type LoadedHooksConfig, fireHook } from "../../hooks";
16
- import { getSafeLogger } from "../../logger";
17
- import type { StoryMetrics } from "../../metrics";
18
- import type { PipelineEventEmitter } from "../../pipeline/events";
19
- import { runPipeline } from "../../pipeline/runner";
20
- import { defaultPipeline } from "../../pipeline/stages";
21
- import type { AgentGetFn } from "../../pipeline/types";
22
- import type { PipelineContext, RoutingResult } from "../../pipeline/types";
23
- import type { PluginRegistry } from "../../plugins";
24
- import { loadPRD, savePRD } from "../../prd";
25
- import type { PRD, UserStory } from "../../prd/types";
26
- import { routeTask } from "../../routing";
27
- import { hookCtx } from "../helpers";
28
- import type { StatusWriter } from "../status-writer";
29
-
30
- export interface AcceptanceLoopContext {
31
- config: NaxConfig;
32
- prd: PRD;
33
- prdPath: string;
34
- workdir: string;
35
- featureDir?: string;
36
- hooks: LoadedHooksConfig;
37
- feature: string;
38
- totalCost: number;
39
- iterations: number;
40
- storiesCompleted: number;
41
- allStoryMetrics: StoryMetrics[];
42
- pluginRegistry: PluginRegistry;
43
- eventEmitter?: PipelineEventEmitter;
44
- statusWriter: StatusWriter;
45
- /** Protocol-aware agent resolver — passed from registry at run start */
46
- agentGetFn?: AgentGetFn;
47
- }
48
-
49
- export interface AcceptanceLoopResult {
50
- success: boolean;
51
- prd: PRD;
52
- totalCost: number;
53
- iterations: number;
54
- storiesCompleted: number;
55
- prdDirty: boolean;
56
- }
57
-
58
- export function isStubTestFile(content: string): boolean {
59
- // Detect skeleton stubs: expect(true).toBe(false) or expect(true).toBe(true) in test bodies
60
- return /expect\s*\(\s*true\s*\)\s*\.\s*toBe\s*\(\s*(?:false|true)\s*\)/.test(content);
61
- }
62
-
63
- /** Load spec.md content for AC text */
64
- async function loadSpecContent(featureDir?: string): Promise<string> {
65
- if (!featureDir) return "";
66
- const specPath = path.join(featureDir, "spec.md");
67
- const specFile = Bun.file(specPath);
68
- return (await specFile.exists()) ? await specFile.text() : "";
69
- }
70
-
71
- /** Build result object for loop exit */
72
- function buildResult(
73
- success: boolean,
74
- prd: PRD,
75
- totalCost: number,
76
- iterations: number,
77
- storiesCompleted: number,
78
- prdDirty: boolean,
79
- ): AcceptanceLoopResult {
80
- return { success, prd, totalCost, iterations, storiesCompleted, prdDirty };
81
- }
82
-
83
- /** Generate and add fix stories to PRD */
84
- async function generateAndAddFixStories(
85
- ctx: AcceptanceLoopContext,
86
- failures: { failedACs: string[]; testOutput: string },
87
- prd: PRD,
88
- ): Promise<FixStory[] | null> {
89
- const logger = getSafeLogger();
90
- const { getAgent } = await import("../../agents");
91
- const agent = (ctx.agentGetFn ?? getAgent)(ctx.config.autoMode.defaultAgent);
92
- if (!agent) {
93
- logger?.error("acceptance", "Agent not found, cannot generate fix stories");
94
- return null;
95
- }
96
- const modelDef = resolveModel(ctx.config.models[ctx.config.analyze.model]);
97
- const fixStories = await generateFixStories(agent, {
98
- failedACs: failures.failedACs,
99
- testOutput: failures.testOutput,
100
- prd,
101
- specContent: await loadSpecContent(ctx.featureDir),
102
- workdir: ctx.workdir,
103
- modelDef,
104
- config: ctx.config,
105
- });
106
- if (fixStories.length === 0) {
107
- logger?.error("acceptance", "Failed to generate fix stories");
108
- return null;
109
- }
110
- logger?.info("acceptance", `Generated ${fixStories.length} fix stories`);
111
- for (const fixStory of fixStories) {
112
- const userStory = convertFixStoryToUserStory(fixStory);
113
- prd.userStories.push(userStory);
114
- logger?.debug("acceptance", `Fix story added: ${userStory.id}: ${userStory.title}`);
115
- }
116
- return fixStories;
117
- }
118
-
119
- /** Execute a single fix story through the pipeline */
120
- async function executeFixStory(
121
- ctx: AcceptanceLoopContext,
122
- story: UserStory,
123
- prd: PRD,
124
- iterations: number,
125
- ): Promise<{ success: boolean; cost: number; metrics?: StoryMetrics[] }> {
126
- const logger = getSafeLogger();
127
- const routing = routeTask(story.title, story.description, story.acceptanceCriteria, story.tags, ctx.config);
128
- logger?.info("acceptance", `Starting fix story: ${story.id}`, { storyId: story.id, storyTitle: story.title });
129
- await fireHook(
130
- ctx.hooks,
131
- "on-story-start",
132
- hookCtx(ctx.feature, {
133
- storyId: story.id,
134
- model: routing.modelTier,
135
- agent: ctx.config.autoMode.defaultAgent,
136
- iteration: iterations,
137
- }),
138
- ctx.workdir,
139
- );
140
- const fixContext: PipelineContext = {
141
- config: ctx.config,
142
- effectiveConfig: ctx.config,
143
- prd,
144
- story,
145
- stories: [story],
146
- routing: routing as RoutingResult,
147
- workdir: ctx.workdir,
148
- featureDir: ctx.featureDir,
149
- hooks: ctx.hooks,
150
- plugins: ctx.pluginRegistry,
151
- storyStartTime: new Date().toISOString(),
152
- agentGetFn: ctx.agentGetFn,
153
- };
154
- const result = await runPipeline(defaultPipeline, fixContext, ctx.eventEmitter);
155
- logger?.info("acceptance", `Fix story ${story.id} ${result.success ? "passed" : "failed"}`);
156
- return {
157
- success: result.success,
158
- cost: result.context.agentResult?.estimatedCost || 0,
159
- metrics: result.context.storyMetrics,
160
- };
161
- }
162
-
163
- /**
164
- * Run the acceptance retry loop
165
- *
166
- * Executes acceptance tests and handles retry logic with fix story generation.
167
- */
168
- export async function runAcceptanceLoop(ctx: AcceptanceLoopContext): Promise<AcceptanceLoopResult> {
169
- const logger = getSafeLogger();
170
- const maxRetries = ctx.config.acceptance.maxRetries;
171
-
172
- let acceptanceRetries = 0;
173
- let prd = ctx.prd;
174
- let totalCost = ctx.totalCost;
175
- let iterations = ctx.iterations;
176
- let storiesCompleted = ctx.storiesCompleted;
177
- let prdDirty = false;
178
-
179
- logger?.info("acceptance", "All stories complete, running acceptance validation");
180
-
181
- while (acceptanceRetries < maxRetries) {
182
- // Run acceptance validation
183
- const firstStory = prd.userStories[0];
184
- const acceptanceContext: PipelineContext = {
185
- config: ctx.config,
186
- effectiveConfig: ctx.config,
187
- prd,
188
- story: firstStory,
189
- stories: [firstStory],
190
- routing: {
191
- complexity: "simple",
192
- modelTier: "balanced",
193
- testStrategy: "test-after",
194
- reasoning: "Acceptance validation",
195
- },
196
- workdir: ctx.workdir,
197
- featureDir: ctx.featureDir,
198
- hooks: ctx.hooks,
199
- plugins: ctx.pluginRegistry,
200
- agentGetFn: ctx.agentGetFn,
201
- };
202
-
203
- const { acceptanceStage } = await import("../../pipeline/stages/acceptance");
204
- const acceptanceResult = await acceptanceStage.execute(acceptanceContext);
205
-
206
- if (acceptanceResult.action === "continue") {
207
- logger?.info("acceptance", "Acceptance validation passed!");
208
- return buildResult(true, prd, totalCost, iterations, storiesCompleted, prdDirty);
209
- }
210
-
211
- if (acceptanceResult.action !== "fail") {
212
- logger?.warn("acceptance", `Unexpected acceptance result: ${acceptanceResult.action}`);
213
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
214
- }
215
-
216
- // Handle acceptance test failures
217
- const failures = acceptanceContext.acceptanceFailures;
218
- if (!failures || failures.failedACs.length === 0) {
219
- logger?.error("acceptance", "Acceptance tests failed but no specific failures detected");
220
- logger?.warn("acceptance", "Manual intervention required");
221
- await fireHook(
222
- ctx.hooks,
223
- "on-pause",
224
- hookCtx(ctx.feature, { reason: "Acceptance tests failed (no failures detected)", cost: totalCost }),
225
- ctx.workdir,
226
- );
227
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
228
- }
229
-
230
- acceptanceRetries++;
231
- logger?.warn("acceptance", `Acceptance retry ${acceptanceRetries}/${maxRetries}`, {
232
- failedACs: failures.failedACs,
233
- });
234
-
235
- if (acceptanceRetries >= maxRetries) {
236
- logger?.error("acceptance", "Max acceptance retries reached");
237
- logger?.warn("acceptance", "Manual intervention required");
238
- logger?.debug("acceptance", 'Run: nax accept --override AC-N "reason" to skip specific ACs');
239
- await fireHook(
240
- ctx.hooks,
241
- "on-pause",
242
- hookCtx(ctx.feature, {
243
- reason: `Acceptance validation failed after ${maxRetries} retries: ${failures.failedACs.join(", ")}`,
244
- cost: totalCost,
245
- }),
246
- ctx.workdir,
247
- );
248
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
249
- }
250
-
251
- // Check for stub test file before generating fix stories
252
- if (ctx.featureDir) {
253
- const testPath = path.join(ctx.featureDir, "acceptance.test.ts");
254
- const testFile = Bun.file(testPath);
255
- if (await testFile.exists()) {
256
- const testContent = await testFile.text();
257
- if (isStubTestFile(testContent)) {
258
- logger?.warn("acceptance", "Stub tests detected — re-generating acceptance tests");
259
- const { unlink } = await import("node:fs/promises");
260
- await unlink(testPath);
261
- const { acceptanceSetupStage } = await import("../../pipeline/stages/acceptance-setup");
262
- await acceptanceSetupStage.execute(acceptanceContext);
263
- const newContent = await Bun.file(testPath).text();
264
- if (isStubTestFile(newContent)) {
265
- logger?.error(
266
- "acceptance",
267
- "Acceptance test generation failed after retry — manual implementation required",
268
- );
269
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
270
- }
271
- }
272
- }
273
- }
274
-
275
- // Generate and add fix stories
276
- logger?.info("acceptance", "Generating fix stories...");
277
- const fixStories = await generateAndAddFixStories(ctx, failures, prd);
278
- if (!fixStories) {
279
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
280
- }
281
-
282
- await savePRD(prd, ctx.prdPath);
283
- prdDirty = true;
284
-
285
- // Execute fix stories
286
- logger?.info("acceptance", "Running fix stories...");
287
- for (const fixStory of fixStories) {
288
- const userStory = prd.userStories.find((s) => s.id === fixStory.id);
289
- if (!userStory || userStory.status !== "pending") continue;
290
-
291
- iterations++;
292
- const result = await executeFixStory(ctx, userStory, prd, iterations);
293
- prd = await loadPRD(ctx.prdPath); // Reload to get updated PRD
294
-
295
- if (result.success) {
296
- storiesCompleted++;
297
- totalCost += result.cost;
298
- if (result.metrics) ctx.allStoryMetrics.push(...result.metrics);
299
- }
300
-
301
- await savePRD(prd, ctx.prdPath);
302
- prdDirty = true;
303
- }
304
-
305
- logger?.info("acceptance", "Re-running acceptance tests...");
306
- }
307
-
308
- return buildResult(false, prd, totalCost, iterations, storiesCompleted, prdDirty);
309
- }
@@ -1,83 +0,0 @@
1
- /**
2
- * Headless Output Formatting
3
- *
4
- * Handles console output formatting for headless (non-TUI) mode.
5
- * Extracts run header and footer formatting logic from runner.ts.
6
- */
7
-
8
- import chalk from "chalk";
9
- import type { RunSummary } from "../../logging";
10
- import { formatRunSummary } from "../../logging";
11
- import { NAX_VERSION } from "../../version";
12
-
13
- export interface RunHeaderOptions {
14
- feature: string;
15
- totalStories: number;
16
- pendingStories: number;
17
- workdir: string;
18
- formatterMode: "quiet" | "normal" | "verbose" | "json";
19
- }
20
-
21
- export interface RunFooterOptions {
22
- finalCounts: {
23
- total: number;
24
- passed: number;
25
- failed: number;
26
- skipped: number;
27
- };
28
- durationMs: number;
29
- totalCost: number;
30
- startedAt: string;
31
- completedAt: string;
32
- formatterMode: "quiet" | "normal" | "verbose" | "json";
33
- }
34
-
35
- /**
36
- * Output run header in headless mode
37
- */
38
- export async function outputRunHeader(options: RunHeaderOptions): Promise<void> {
39
- const { feature, totalStories, pendingStories, workdir, formatterMode } = options;
40
-
41
- if (formatterMode === "json") {
42
- return;
43
- }
44
-
45
- console.log("");
46
- console.log(chalk.bold(chalk.blue("═".repeat(60))));
47
- console.log(chalk.bold(chalk.blue(` ▶ NAX v${NAX_VERSION} — RUN STARTED`)));
48
- console.log(chalk.blue("═".repeat(60)));
49
- console.log(` ${chalk.gray("Feature:")} ${chalk.cyan(feature)}`);
50
- console.log(` ${chalk.gray("Stories:")} ${chalk.cyan(`${totalStories} total, ${pendingStories} pending`)}`);
51
- console.log(` ${chalk.gray("Path:")} ${chalk.dim(workdir)}`);
52
- console.log(chalk.blue("═".repeat(60)));
53
- console.log("");
54
- }
55
-
56
- /**
57
- * Output run footer in headless mode
58
- */
59
- export function outputRunFooter(options: RunFooterOptions): void {
60
- const { finalCounts, durationMs, totalCost, startedAt, completedAt, formatterMode } = options;
61
-
62
- if (formatterMode === "json") {
63
- return;
64
- }
65
-
66
- const runSummary: RunSummary = {
67
- total: finalCounts.total,
68
- passed: finalCounts.passed,
69
- failed: finalCounts.failed,
70
- skipped: finalCounts.skipped,
71
- durationMs,
72
- totalCost,
73
- startedAt,
74
- completedAt,
75
- };
76
-
77
- const summaryOutput = formatRunSummary(runSummary, {
78
- mode: formatterMode,
79
- useColor: true,
80
- });
81
-
82
- console.log(summaryOutput);
83
- }
@@ -1,11 +0,0 @@
1
- /**
2
- * Lifecycle module exports
3
- */
4
-
5
- export { runAcceptanceLoop, type AcceptanceLoopContext, type AcceptanceLoopResult } from "./acceptance-loop";
6
- export { outputRunHeader, outputRunFooter, type RunHeaderOptions, type RunFooterOptions } from "./headless-formatter";
7
- export { handleParallelCompletion, type ParallelCompletionOptions } from "./parallel-lifecycle";
8
- export { handleRunCompletion, type RunCompletionOptions, type RunCompletionResult } from "./run-completion";
9
- export { cleanupRun, type RunCleanupOptions } from "./run-cleanup";
10
- export { setupRun, type RunSetupOptions, type RunSetupResult } from "./run-setup";
11
- export { runDeferredRegression, type DeferredRegressionOptions, type DeferredRegressionResult } from "./run-regression";
@@ -1,101 +0,0 @@
1
- /**
2
- * Parallel Execution Lifecycle
3
- *
4
- * Handles metrics saving and reporter notifications for parallel execution completion.
5
- * Extracted from runner.ts to reduce its size.
6
- */
7
-
8
- import { getSafeLogger } from "../../logger";
9
- import type { StoryMetrics } from "../../metrics";
10
- import { saveRunMetrics } from "../../metrics";
11
- import type { PluginRegistry } from "../../plugins/registry";
12
- import { countStories } from "../../prd";
13
- import type { PRD } from "../../prd";
14
-
15
- export interface ParallelCompletionOptions {
16
- runId: string;
17
- feature: string;
18
- startedAt: string;
19
- completedAt: string;
20
- prd: PRD;
21
- allStoryMetrics: StoryMetrics[];
22
- totalCost: number;
23
- storiesCompleted: number;
24
- durationMs: number;
25
- workdir: string;
26
- pluginRegistry: PluginRegistry;
27
- }
28
-
29
- /**
30
- * Handle parallel execution completion:
31
- * - Save run metrics
32
- * - Emit reporter events
33
- */
34
- export async function handleParallelCompletion(options: ParallelCompletionOptions): Promise<void> {
35
- const logger = getSafeLogger();
36
- const {
37
- runId,
38
- feature,
39
- startedAt,
40
- completedAt,
41
- prd,
42
- allStoryMetrics,
43
- totalCost,
44
- storiesCompleted,
45
- durationMs,
46
- workdir,
47
- pluginRegistry,
48
- } = options;
49
-
50
- // Save run metrics
51
- const runMetrics = {
52
- runId,
53
- feature,
54
- startedAt,
55
- completedAt,
56
- totalCost,
57
- totalStories: allStoryMetrics.length,
58
- storiesCompleted,
59
- storiesFailed: countStories(prd).failed,
60
- totalDurationMs: durationMs,
61
- stories: allStoryMetrics,
62
- };
63
-
64
- await saveRunMetrics(workdir, runMetrics);
65
-
66
- // Log run completion
67
- const finalCounts = countStories(prd);
68
- logger?.info("run.complete", "Feature execution completed", {
69
- runId,
70
- feature,
71
- success: true,
72
- totalStories: finalCounts.total,
73
- storiesCompleted,
74
- storiesFailed: finalCounts.failed,
75
- storiesPending: finalCounts.pending,
76
- totalCost,
77
- durationMs,
78
- });
79
-
80
- // Emit onRunEnd to reporters
81
- const reporters = pluginRegistry.getReporters();
82
- for (const reporter of reporters) {
83
- if (reporter.onRunEnd) {
84
- try {
85
- await reporter.onRunEnd({
86
- runId,
87
- totalDurationMs: durationMs,
88
- totalCost,
89
- storySummary: {
90
- completed: storiesCompleted,
91
- failed: finalCounts.failed,
92
- skipped: finalCounts.skipped,
93
- paused: finalCounts.paused,
94
- },
95
- });
96
- } catch (error) {
97
- logger?.warn("plugins", `Reporter '${reporter.name}' onRunEnd failed`, { error });
98
- }
99
- }
100
- }
101
- }
@@ -1,140 +0,0 @@
1
- /**
2
- * Precheck Runner
3
- *
4
- * Handles precheck validation execution before the main run starts.
5
- * Validates project state, dependencies, and configuration.
6
- * Blocks execution on Tier 1 failures, warns on Tier 2 failures.
7
- */
8
-
9
- import { appendFileSync, mkdirSync } from "node:fs";
10
- import path from "node:path";
11
- import chalk from "chalk";
12
- import type { NaxConfig } from "../../config";
13
- import type { InteractionChain } from "../../interaction/chain";
14
- import { getSafeLogger } from "../../logger";
15
- import type { PRD } from "../../prd/types";
16
- import type { StatusWriter } from "../status-writer";
17
-
18
- export interface PrecheckContext {
19
- config: NaxConfig;
20
- prd: PRD;
21
- workdir: string;
22
- logFilePath?: string;
23
- statusWriter: StatusWriter;
24
- headless: boolean;
25
- formatterMode: "quiet" | "normal" | "verbose" | "json";
26
- interactionChain?: InteractionChain | null;
27
- featureName?: string;
28
- }
29
-
30
- /**
31
- * Run precheck validations before execution starts
32
- *
33
- * @throws Error if precheck blockers are found (Tier 1 failures)
34
- */
35
- export async function runPrecheckValidation(ctx: PrecheckContext): Promise<void> {
36
- const logger = getSafeLogger();
37
-
38
- // Allow tests to skip precheck entirely (git-repo-exists, working-tree-clean, etc. fail in temp dirs)
39
- if (process.env.NAX_SKIP_PRECHECK === "1") {
40
- logger?.info("precheck", "Skipping precheck validations (NAX_SKIP_PRECHECK=1)");
41
- return;
42
- }
43
-
44
- logger?.info("precheck", "Running precheck validations...");
45
-
46
- const { runPrecheck } = await import("../../precheck");
47
- const precheckResult = await runPrecheck(ctx.config, ctx.prd, {
48
- workdir: ctx.workdir,
49
- format: "human",
50
- });
51
-
52
- // Log precheck results to JSONL
53
- if (ctx.logFilePath) {
54
- // Ensure directory exists
55
- mkdirSync(path.dirname(ctx.logFilePath), { recursive: true });
56
-
57
- const precheckLog = {
58
- type: "precheck",
59
- timestamp: new Date().toISOString(),
60
- passed: precheckResult.output.passed,
61
- blockers: precheckResult.output.blockers.map((b) => ({ name: b.name, message: b.message })),
62
- warnings: precheckResult.output.warnings.map((w) => ({ name: w.name, message: w.message })),
63
- summary: precheckResult.output.summary,
64
- };
65
- require("node:fs").appendFileSync(ctx.logFilePath, `${JSON.stringify(precheckLog)}\n`);
66
- }
67
-
68
- // Handle blockers (Tier 1 failures)
69
- if (!precheckResult.output.passed) {
70
- logger?.error("precheck", "Precheck failed - execution blocked", {
71
- blockers: precheckResult.output.blockers.length,
72
- failedChecks: precheckResult.output.blockers.map((b) => b.name),
73
- });
74
-
75
- // Update status file with precheck-failed status
76
- ctx.statusWriter.setRunStatus("precheck-failed");
77
- ctx.statusWriter.setCurrentStory(null);
78
- await ctx.statusWriter.update(0, 0);
79
-
80
- // Log detailed error message to console
81
- console.error("");
82
- console.error(chalk.red("❌ PRECHECK FAILED"));
83
- console.error(chalk.red("─".repeat(60)));
84
- for (const blocker of precheckResult.output.blockers) {
85
- console.error(chalk.red(`✗ ${blocker.name}: ${blocker.message}`));
86
- }
87
- console.error(chalk.red("─".repeat(60)));
88
- console.error(chalk.yellow("\nRun 'nax precheck' for detailed information"));
89
- console.error(chalk.dim("Use --skip-precheck to bypass (not recommended)\n"));
90
-
91
- throw new Error(`Precheck failed: ${precheckResult.output.blockers.map((b) => b.name).join(", ")}`);
92
- }
93
-
94
- // Handle warnings (Tier 2 failures) - log but continue
95
- if (precheckResult.output.warnings.length > 0) {
96
- logger?.warn("precheck", "Precheck passed with warnings", {
97
- warnings: precheckResult.output.warnings.length,
98
- issues: precheckResult.output.warnings.map((w) => w.name),
99
- });
100
-
101
- if (ctx.headless && ctx.formatterMode !== "json") {
102
- console.log(chalk.yellow("\n⚠️ Precheck warnings:"));
103
- for (const warning of precheckResult.output.warnings) {
104
- console.log(chalk.yellow(` ⚠ ${warning.name}: ${warning.message}`));
105
- }
106
- console.log("");
107
- }
108
- } else {
109
- logger?.info("precheck", "All precheck validations passed");
110
- }
111
-
112
- // Story size gate interaction (v0.16.0) - prompt for flagged stories if interaction chain exists
113
- if (precheckResult.flaggedStories && precheckResult.flaggedStories.length > 0) {
114
- if (ctx.interactionChain && ctx.featureName) {
115
- logger?.info("precheck", "Story size gate: prompting user for flagged stories", {
116
- count: precheckResult.flaggedStories.length,
117
- });
118
-
119
- const { promptForFlaggedStories } = await import("./story-size-prompts");
120
- const summary = await promptForFlaggedStories(
121
- precheckResult.flaggedStories,
122
- ctx.prd,
123
- ctx.interactionChain,
124
- ctx.featureName,
125
- );
126
-
127
- logger?.info("precheck", "Story size gate prompts complete", {
128
- approved: summary.approved.length,
129
- skipped: summary.skipped.length,
130
- aborted: summary.aborted,
131
- });
132
-
133
- // PRD has been mutated with skipped stories - no need to return anything
134
- } else {
135
- logger?.warn("precheck", "Story size gate: interaction chain not available, skipping prompts", {
136
- flaggedCount: precheckResult.flaggedStories.length,
137
- });
138
- }
139
- }
140
- }