@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,228 +0,0 @@
1
- /**
2
- * Auto-detection for contextFiles when PRD omits them (BUG-006)
3
- *
4
- * Extracts keywords from story title, runs git grep to find matching source files,
5
- * excludes test/index/generated files, caps at maxFiles.
6
- */
7
-
8
- import { getLogger } from "../logger";
9
- import { errorMessage } from "../utils/errors";
10
-
11
- export interface AutoDetectOptions {
12
- /** Working directory for git grep */
13
- workdir: string;
14
- /** Story title to extract keywords from */
15
- storyTitle: string;
16
- /** Maximum files to return (default: 5) */
17
- maxFiles?: number;
18
- /** Enable import tracing (default: false, reserved for future use) */
19
- traceImports?: boolean;
20
- }
21
-
22
- /**
23
- * Extract keywords from story title for git grep search.
24
- *
25
- * Heuristics:
26
- * - Remove common words (the, a, an, and, or, to, from, for, with, etc.)
27
- * - Split on spaces/punctuation
28
- * - Keep words >= 3 chars
29
- * - Lowercase for case-insensitive matching
30
- *
31
- * @example
32
- * extractKeywords("BUG-006: Context auto-detection (contextFiles)")
33
- * // => ["bug", "006", "context", "auto", "detection", "contextfiles"]
34
- */
35
- export function extractKeywords(title: string): string[] {
36
- const stopWords = new Set([
37
- "the",
38
- "a",
39
- "an",
40
- "and",
41
- "or",
42
- "to",
43
- "from",
44
- "for",
45
- "with",
46
- "in",
47
- "on",
48
- "at",
49
- "by",
50
- "is",
51
- "are",
52
- "was",
53
- "were",
54
- "be",
55
- "been",
56
- "being",
57
- "have",
58
- "has",
59
- "had",
60
- "do",
61
- "does",
62
- "did",
63
- "will",
64
- "would",
65
- "should",
66
- "could",
67
- "may",
68
- "might",
69
- "can",
70
- "must",
71
- "of",
72
- "as",
73
- "if",
74
- "when",
75
- "bug",
76
- "fix",
77
- "add",
78
- "update",
79
- "remove",
80
- "implement",
81
- ]);
82
-
83
- // Split on non-alphanumeric, filter, lowercase
84
- const words = title
85
- .toLowerCase()
86
- .split(/[^a-z0-9]+/)
87
- .filter((w) => w.length >= 3 && !stopWords.has(w));
88
-
89
- // Deduplicate
90
- return Array.from(new Set(words));
91
- }
92
-
93
- /**
94
- * Auto-detect context files via git grep when PRD omits contextFiles.
95
- *
96
- * Algorithm:
97
- * 1. Extract keywords from story title
98
- * 2. Run git grep for each keyword in src/ directories
99
- * 3. Deduplicate files
100
- * 4. Exclude test/index/generated files
101
- * 5. Cap at maxFiles (default: 5)
102
- *
103
- * Returns empty array if:
104
- * - Not a git repository
105
- * - No keywords extracted
106
- * - git grep fails
107
- * - No matching files found
108
- *
109
- * @param options - Auto-detect configuration
110
- * @returns Array of relative file paths (sorted by relevance score)
111
- */
112
- export async function autoDetectContextFiles(options: AutoDetectOptions): Promise<string[]> {
113
- const { workdir, storyTitle, maxFiles = 5 } = options;
114
- const logger = getLogger();
115
-
116
- // Extract keywords
117
- const keywords = extractKeywords(storyTitle);
118
- if (keywords.length === 0) {
119
- logger.debug("auto-detect", "No keywords extracted from story title", { storyTitle });
120
- return [];
121
- }
122
-
123
- logger.debug("auto-detect", "Extracted keywords", { keywords, storyTitle });
124
-
125
- // Build git grep command
126
- // Use -i for case-insensitive, -l for filename-only, -I to skip binary files
127
- // Search in src/ directories only (exclude test, node_modules, etc.)
128
- const grepPattern = keywords.join("|"); // OR pattern
129
- const grepCommand = [
130
- "git",
131
- "grep",
132
- "-i", // case-insensitive
133
- "-l", // files-with-matches
134
- "-I", // skip binary files
135
- "-E", // extended regex
136
- "-e",
137
- grepPattern,
138
- "--", // separator
139
- "src/", // limit to src directory
140
- ];
141
-
142
- try {
143
- const proc = Bun.spawn(grepCommand, {
144
- cwd: workdir,
145
- stdout: "pipe",
146
- stderr: "pipe",
147
- });
148
-
149
- const stdout = await new Response(proc.stdout).text();
150
- const exitCode = await proc.exited;
151
-
152
- // git grep returns 1 when no matches found (not an error)
153
- if (exitCode !== 0 && exitCode !== 1) {
154
- const stderr = await new Response(proc.stderr).text();
155
- logger.warn("auto-detect", "git grep failed", { exitCode, stderr: stderr.trim() });
156
- return [];
157
- }
158
-
159
- if (!stdout.trim()) {
160
- logger.debug("auto-detect", "No files matched keywords", { keywords });
161
- return [];
162
- }
163
-
164
- // Parse file list
165
- const allFiles = stdout
166
- .trim()
167
- .split("\n")
168
- .map((line) => line.trim())
169
- .filter((line) => line.length > 0);
170
-
171
- // Filter out test files, index files, generated files
172
- const filtered = allFiles.filter((filePath) => {
173
- const lower = filePath.toLowerCase();
174
- // Exclude test files
175
- if (lower.includes(".test.") || lower.includes(".spec.") || lower.includes("/test/")) {
176
- return false;
177
- }
178
- // Exclude index files (barrel exports, low signal)
179
- if (lower.endsWith("/index.ts") || lower.endsWith("/index.js")) {
180
- return false;
181
- }
182
- // Exclude generated files
183
- if (lower.includes(".generated.") || lower.includes("/__generated__/")) {
184
- return false;
185
- }
186
- return true;
187
- });
188
-
189
- // Score and sort by relevance
190
- // Simple heuristic: count keyword matches in file path
191
- interface ScoredFile {
192
- path: string;
193
- score: number;
194
- }
195
-
196
- const scored: ScoredFile[] = filtered.map((filePath) => {
197
- const lowerPath = filePath.toLowerCase();
198
- const score = keywords.filter((kw) => lowerPath.includes(kw)).length;
199
- return { path: filePath, score };
200
- });
201
-
202
- // Sort by score descending, then alphabetically
203
- scored.sort((a, b) => {
204
- if (a.score !== b.score) {
205
- return b.score - a.score; // Higher score first
206
- }
207
- return a.path.localeCompare(b.path); // Alphabetical tiebreaker
208
- });
209
-
210
- // Cap at maxFiles
211
- const selected = scored.slice(0, maxFiles).map((s) => s.path);
212
-
213
- logger.info("auto-detect", "Auto-detected context files", {
214
- keywords,
215
- totalMatches: allFiles.length,
216
- afterFilter: filtered.length,
217
- selected: selected.length,
218
- files: selected,
219
- });
220
-
221
- return selected;
222
- } catch (error) {
223
- logger.warn("auto-detect", "Auto-detection failed", {
224
- error: errorMessage(error),
225
- });
226
- return [];
227
- }
228
- }
@@ -1,299 +0,0 @@
1
- /**
2
- * Context builder for story-scoped prompt optimization
3
- *
4
- * Extracts current story + dependency stories from PRD and builds context within token budget.
5
- */
6
-
7
- import path from "node:path";
8
- import type { NaxConfig } from "../config";
9
- import { getLogger } from "../logger";
10
- import { estimateTokens } from "../optimizer/types";
11
- import type { UserStory } from "../prd";
12
- import { countStories, getContextFiles } from "../prd";
13
- import { errorMessage } from "../utils/errors";
14
- import { autoDetectContextFiles } from "./auto-detect";
15
- import {
16
- createDependencyContext,
17
- createErrorContext,
18
- createFileContext,
19
- createPriorFailuresContext,
20
- createProgressContext,
21
- createStoryContext,
22
- createTestCoverageContext,
23
- } from "./elements";
24
- import { getParentOutputFiles } from "./parent-context";
25
- import { generateTestCoverageSummary } from "./test-scanner";
26
- import type { BuiltContext, ContextBudget, ContextElement, StoryContext } from "./types";
27
-
28
- // Dependency injection for testability
29
- export const _deps = {
30
- autoDetectContextFiles,
31
- };
32
-
33
- // Re-export for backward compatibility
34
- export {
35
- createStoryContext,
36
- createDependencyContext,
37
- createErrorContext,
38
- createProgressContext,
39
- createFileContext,
40
- createTestCoverageContext,
41
- createPriorFailuresContext,
42
- } from "./elements";
43
- export { formatContextAsMarkdown } from "./formatter";
44
-
45
- /** Sort context elements by priority (descending) and token count (ascending for same priority) */
46
- export function sortContextElements(elements: ContextElement[]): ContextElement[] {
47
- return [...elements].sort((a, b) => {
48
- if (a.priority !== b.priority) return b.priority - a.priority;
49
- return a.tokens - b.tokens;
50
- });
51
- }
52
-
53
- /** Generate progress summary */
54
- function generateProgressSummary(prd: StoryContext["prd"]): string {
55
- const counts = countStories(prd);
56
- const total = counts.total;
57
- const complete = counts.passed + counts.failed;
58
- if (counts.failed > 0) {
59
- return `Progress: ${complete}/${total} stories complete (${counts.passed} passed, ${counts.failed} failed)`;
60
- }
61
- return `Progress: ${complete}/${total} stories complete (${counts.passed} passed)`;
62
- }
63
-
64
- /** Generate human-readable summary of built context */
65
- function generateSummary(elements: ContextElement[], totalTokens: number, truncated: boolean): string {
66
- const counts: Record<string, number> = {
67
- story: 0,
68
- dependency: 0,
69
- error: 0,
70
- progress: 0,
71
- file: 0,
72
- "test-coverage": 0,
73
- };
74
- for (const element of elements) {
75
- counts[element.type]++;
76
- }
77
- const parts: string[] = [];
78
- if (counts.progress > 0) parts.push(`${counts.progress} progress`);
79
- if (counts.story > 0) parts.push(`${counts.story} story`);
80
- if (counts.dependency > 0) parts.push(`${counts.dependency} dependencies`);
81
- if (counts.error > 0) parts.push(`${counts.error} errors`);
82
- if (counts.file > 0) parts.push(`${counts.file} files`);
83
- if (counts["test-coverage"] > 0) parts.push("test coverage");
84
-
85
- const summary = `Context: ${parts.join(", ")} (${totalTokens} tokens)`;
86
- return truncated ? `${summary} [TRUNCATED]` : summary;
87
- }
88
-
89
- /** Build context from PRD + current story within token budget. */
90
- export async function buildContext(storyContext: StoryContext, budget: ContextBudget): Promise<BuiltContext> {
91
- const { prd, currentStoryId } = storyContext;
92
- const elements: ContextElement[] = [];
93
-
94
- const currentStory = prd.userStories.find((s) => s.id === currentStoryId);
95
- if (!currentStory) throw new Error(`Story ${currentStoryId} not found in PRD`);
96
-
97
- // Add progress summary (highest priority)
98
- elements.push(createProgressContext(generateProgressSummary(prd), 100));
99
-
100
- // Add prior failures (highest priority after progress, priority 95)
101
- if (
102
- currentStory.priorFailures &&
103
- Array.isArray(currentStory.priorFailures) &&
104
- currentStory.priorFailures.length > 0
105
- ) {
106
- elements.push(createPriorFailuresContext(currentStory.priorFailures, 95));
107
- }
108
-
109
- // Add prior errors (high priority)
110
- if (currentStory.priorErrors && Array.isArray(currentStory.priorErrors) && currentStory.priorErrors.length > 0) {
111
- for (const error of currentStory.priorErrors) {
112
- elements.push(createErrorContext(error, 90));
113
- }
114
- }
115
-
116
- // Add current story (high priority)
117
- elements.push(createStoryContext(currentStory, 80));
118
-
119
- // ENH-006: Inject planning analysis from prd.analysis (priority 88 — above story, below errors)
120
- if (prd.analysis) {
121
- const analysisContent = `The following analysis was performed during the planning phase. Use it to understand the codebase context before implementing:\n\n${prd.analysis}`;
122
- elements.push({
123
- type: "planning-analysis",
124
- label: "Planning Analysis",
125
- content: analysisContent,
126
- priority: 88,
127
- tokens: estimateTokens(analysisContent),
128
- });
129
- }
130
-
131
- // Add dependency stories (medium priority)
132
- addDependencyElements(elements, currentStory, prd);
133
-
134
- // Add test coverage summary (priority 85)
135
- await addTestCoverageElement(elements, storyContext, currentStory);
136
-
137
- // Add relevant source files (priority 60)
138
- await addFileElements(elements, storyContext, currentStory);
139
-
140
- // Select elements within budget
141
- const sorted = sortContextElements(elements);
142
- const selected: ContextElement[] = [];
143
- let totalTokens = 0;
144
- let truncated = false;
145
-
146
- for (const element of sorted) {
147
- if (totalTokens + element.tokens <= budget.availableForContext) {
148
- selected.push(element);
149
- totalTokens += element.tokens;
150
- } else {
151
- truncated = true;
152
- }
153
- }
154
-
155
- return { elements: selected, totalTokens, truncated, summary: generateSummary(selected, totalTokens, truncated) };
156
- }
157
-
158
- /** Add dependency story elements to the context. */
159
- function addDependencyElements(elements: ContextElement[], story: UserStory, prd: StoryContext["prd"]): void {
160
- if (!story.dependencies || story.dependencies.length === 0) return;
161
- for (const depId of story.dependencies) {
162
- const depStory = prd.userStories.find((s) => s.id === depId);
163
- if (depStory) {
164
- elements.push(createDependencyContext(depStory, 50));
165
- } else {
166
- const logger = getLogger();
167
- logger.warn("context", "Dependency story not found in PRD", { dependencyId: depId, referencedBy: story.id });
168
- }
169
- }
170
- }
171
-
172
- /** Add test coverage summary element. */
173
- async function addTestCoverageElement(
174
- elements: ContextElement[],
175
- storyContext: StoryContext,
176
- story: UserStory,
177
- ): Promise<void> {
178
- if (storyContext.config?.context?.testCoverage?.enabled === false || !storyContext.workdir) return;
179
- try {
180
- const tcConfig = storyContext.config?.context?.testCoverage;
181
- const contextFiles = getContextFiles(story);
182
- const scanResult = await generateTestCoverageSummary({
183
- workdir: storyContext.workdir,
184
- testDir: tcConfig?.testDir,
185
- testPattern: tcConfig?.testPattern,
186
- maxTokens: tcConfig?.maxTokens ?? 500,
187
- detail: tcConfig?.detail ?? "names-and-counts",
188
- contextFiles: contextFiles.length > 0 ? contextFiles : undefined,
189
- scopeToStory: tcConfig?.scopeToStory ?? true,
190
- });
191
- if (scanResult.summary) {
192
- elements.push(createTestCoverageContext(scanResult.summary, scanResult.tokens, 85));
193
- }
194
- } catch (error) {
195
- const logger = getLogger();
196
- logger.warn("context", "Test coverage scan failed", { error: (error as Error).message });
197
- }
198
- }
199
-
200
- /** Add relevant source file elements (auto-detected or from story config). */
201
- async function addFileElements(
202
- elements: ContextElement[],
203
- storyContext: StoryContext,
204
- story: UserStory,
205
- ): Promise<void> {
206
- const MAX_FILE_SIZE_BYTES = 10 * 1024;
207
- const MAX_FILES = 5;
208
-
209
- // Skip all file injection when fileInjection is 'disabled' or undefined (treat missing as disabled)
210
- const fileInjection = storyContext.config?.context?.fileInjection;
211
- if (fileInjection !== "keyword") return;
212
-
213
- let contextFiles = getContextFiles(story);
214
-
215
- // ENH-005: Inject parent output files for context chaining
216
- const parentFiles = getParentOutputFiles(story, storyContext.prd?.userStories ?? []);
217
- if (parentFiles.length > 0) {
218
- const logger = getLogger();
219
- logger.info("context", "Injecting parent output files for context chaining", {
220
- storyId: story.id,
221
- parentFiles,
222
- });
223
- // Merge with existing contextFiles (don't replace — parent files are supplementary)
224
- contextFiles = [...new Set([...contextFiles, ...parentFiles])];
225
- }
226
-
227
- // Auto-detect contextFiles if empty and enabled (BUG-006)
228
- if (
229
- contextFiles.length === 0 &&
230
- storyContext.config?.context?.autoDetect?.enabled !== false &&
231
- storyContext.workdir
232
- ) {
233
- const autoDetectConfig = storyContext.config?.context?.autoDetect;
234
- try {
235
- const detected = await _deps.autoDetectContextFiles({
236
- workdir: storyContext.workdir,
237
- storyTitle: story.title,
238
- maxFiles: autoDetectConfig?.maxFiles ?? 5,
239
- traceImports: autoDetectConfig?.traceImports ?? false,
240
- });
241
- if (detected.length > 0) {
242
- contextFiles = detected;
243
- const logger = getLogger();
244
- logger.info("context", "Auto-detected context files", { storyId: story.id, files: detected });
245
- }
246
- } catch (error) {
247
- const logger = getLogger();
248
- logger.warn("context", "Context auto-detection failed", {
249
- storyId: story.id,
250
- error: errorMessage(error),
251
- });
252
- }
253
- }
254
-
255
- if (contextFiles.length === 0) return;
256
- const filesToLoad = contextFiles.slice(0, MAX_FILES);
257
- const workdir = storyContext.workdir || process.cwd();
258
-
259
- for (const relativeFilePath of filesToLoad) {
260
- try {
261
- const absolutePath = path.resolve(workdir, relativeFilePath);
262
- const file = Bun.file(absolutePath);
263
- if (!(await file.exists())) {
264
- const logger = getLogger();
265
- logger.warn("context", "Relevant file not found", { filePath: relativeFilePath, storyId: story.id });
266
- continue;
267
- }
268
- if (file.size > MAX_FILE_SIZE_BYTES) {
269
- // FEAT-011: File too large to inline — pass path-only so agent can read it if needed
270
- const logger = getLogger();
271
- logger.warn("context", "File too large for inline — using path-only", {
272
- filePath: relativeFilePath,
273
- sizeKB: Math.round(file.size / 1024),
274
- maxKB: 10,
275
- storyId: story.id,
276
- });
277
- elements.push(
278
- createFileContext(
279
- relativeFilePath,
280
- `_File too large to inline (${Math.round(file.size / 1024)}KB). Path: \`${relativeFilePath}\` — read it directly if needed._`,
281
- 5,
282
- ),
283
- );
284
- continue;
285
- }
286
- const content = await file.text();
287
- const ext = path.extname(relativeFilePath).slice(1) || "txt";
288
- elements.push(
289
- createFileContext(relativeFilePath, `\`\`\`${ext}\n// File: ${relativeFilePath}\n${content}\n\`\`\``, 60),
290
- );
291
- } catch (error) {
292
- const logger = getLogger();
293
- logger.warn("context", "Error loading file", {
294
- filePath: relativeFilePath,
295
- error: errorMessage(error),
296
- });
297
- }
298
- }
299
- }
@@ -1,122 +0,0 @@
1
- /**
2
- * Context Element Factories
3
- *
4
- * Extracted from builder.ts: token estimation and context element creation.
5
- */
6
-
7
- import { getLogger } from "../logger";
8
- import { estimateTokens } from "../optimizer/types";
9
- import type { StructuredFailure, UserStory } from "../prd";
10
- import type { ContextElement } from "./types";
11
-
12
- /** Create context element from current story */
13
- export function createStoryContext(story: UserStory, priority: number): ContextElement {
14
- const content = formatStoryAsText(story);
15
- return { type: "story", storyId: story.id, content, priority, tokens: estimateTokens(content) };
16
- }
17
-
18
- /** Create context element from dependency story */
19
- export function createDependencyContext(story: UserStory, priority: number): ContextElement {
20
- const content = formatStoryAsText(story);
21
- return { type: "dependency", storyId: story.id, content, priority, tokens: estimateTokens(content) };
22
- }
23
-
24
- /** Create context element from error */
25
- export function createErrorContext(errorMessage: string, priority: number): ContextElement {
26
- return { type: "error", content: errorMessage, priority, tokens: estimateTokens(errorMessage) };
27
- }
28
-
29
- /** Create context element from progress summary */
30
- export function createProgressContext(progressText: string, priority: number): ContextElement {
31
- return { type: "progress", content: progressText, priority, tokens: estimateTokens(progressText) };
32
- }
33
-
34
- /** Create context element from file content */
35
- export function createFileContext(filePath: string, content: string, priority: number): ContextElement {
36
- return { type: "file", filePath, content, priority, tokens: estimateTokens(content) };
37
- }
38
-
39
- /** Create context element from test coverage summary */
40
- export function createTestCoverageContext(content: string, tokens: number, priority: number): ContextElement {
41
- return { type: "test-coverage", content, priority, tokens };
42
- }
43
-
44
- /** Create context element from prior failures */
45
- export function createPriorFailuresContext(failures: StructuredFailure[], priority: number): ContextElement {
46
- const content = formatPriorFailures(failures);
47
- return { type: "prior-failures", content, priority, tokens: estimateTokens(content) };
48
- }
49
-
50
- /** Format prior failures as markdown for agent context */
51
- export function formatPriorFailures(failures: StructuredFailure[]): string {
52
- if (!failures || failures.length === 0) {
53
- return "";
54
- }
55
-
56
- const parts: string[] = [];
57
- parts.push("## Prior Failures (Structured Context)\n");
58
-
59
- for (const failure of failures) {
60
- parts.push(`### Attempt ${failure.attempt} — ${failure.modelTier}`);
61
- parts.push(`**Stage:** ${failure.stage}`);
62
- parts.push(`**Summary:** ${failure.summary}`);
63
-
64
- if (failure.testFailures && failure.testFailures.length > 0) {
65
- parts.push("\n**Test Failures:**");
66
- for (const testFailure of failure.testFailures) {
67
- parts.push(`\n- **File:** \`${testFailure.file}\``);
68
- parts.push(` **Test:** ${testFailure.testName}`);
69
- parts.push(` **Error:** ${testFailure.error}`);
70
- if (testFailure.stackTrace && testFailure.stackTrace.length > 0) {
71
- parts.push(` **Stack:** ${testFailure.stackTrace[0]}`);
72
- }
73
- }
74
- }
75
-
76
- if (failure.reviewFindings && failure.reviewFindings.length > 0) {
77
- parts.push("\n**Review Findings (fix these issues):**");
78
- for (const finding of failure.reviewFindings) {
79
- const source = finding.source ? ` (${finding.source})` : "";
80
- parts.push(`\n- **[${finding.severity}]** \`${finding.file}:${finding.line}\`${source}`);
81
- parts.push(` **Rule:** ${finding.ruleId}`);
82
- parts.push(` **Issue:** ${finding.message}`);
83
- if (finding.url) {
84
- parts.push(` **Docs:** ${finding.url}`);
85
- }
86
- }
87
- }
88
- parts.push("");
89
- }
90
-
91
- return parts.join("\n");
92
- }
93
-
94
- /** Format story as text for context (defensive checks for malformed PRD data) */
95
- export function formatStoryAsText(story: UserStory): string {
96
- const parts: string[] = [];
97
- parts.push(`## ${story.id}: ${story.title}`);
98
- parts.push("");
99
- parts.push(`**Description:** ${story.description}`);
100
- parts.push("");
101
- parts.push("**Acceptance Criteria:**");
102
-
103
- if (story.acceptanceCriteria && Array.isArray(story.acceptanceCriteria)) {
104
- for (const ac of story.acceptanceCriteria) {
105
- parts.push(`- ${ac}`);
106
- }
107
- } else {
108
- const logger = getLogger();
109
- logger.warn("context", "Story has invalid acceptanceCriteria", {
110
- storyId: story.id,
111
- type: typeof story.acceptanceCriteria,
112
- });
113
- parts.push("- (No acceptance criteria defined)");
114
- }
115
-
116
- if (story.tags && story.tags.length > 0) {
117
- parts.push("");
118
- parts.push(`**Tags:** ${story.tags.join(", ")}`);
119
- }
120
-
121
- return parts.join("\n");
122
- }