@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,370 +0,0 @@
1
- /**
2
- * Test File Scanner (v0.7)
3
- *
4
- * Scans test directories and extracts describe/test block names
5
- * to generate a coverage summary for prompt injection.
6
- * Prevents test duplication across isolated story sessions.
7
- */
8
-
9
- import path from "node:path";
10
- import { Glob } from "bun";
11
- import { getLogger } from "../logger";
12
- import { estimateTokens } from "../optimizer/types";
13
-
14
- // ============================================================================
15
- // Types
16
- // ============================================================================
17
-
18
- /** Detail level for test summary */
19
- export type TestSummaryDetail = "names-only" | "names-and-counts" | "describe-blocks";
20
-
21
- /** Options for scanning test files */
22
- export interface TestScanOptions {
23
- /** Working directory (base for testDir) */
24
- workdir: string;
25
- /** Test directory relative to workdir (default: auto-detect) */
26
- testDir?: string;
27
- /** Glob pattern for test files (default: "**\/*.test.{ts,js,tsx,jsx}") */
28
- testPattern?: string;
29
- /** Max tokens for the summary (default: 500) */
30
- maxTokens?: number;
31
- /** Summary detail level (default: "names-and-counts") */
32
- detail?: TestSummaryDetail;
33
- /** Context files to scope test coverage to (default: undefined = scan all) */
34
- contextFiles?: string[];
35
- /** Enable scoping to context files (default: true) */
36
- scopeToStory?: boolean;
37
- }
38
-
39
- /** A single describe block extracted from a test file */
40
- export interface DescribeBlock {
41
- name: string;
42
- tests: string[];
43
- }
44
-
45
- /** Parsed test file info */
46
- export interface TestFileInfo {
47
- /** Relative path from workdir */
48
- relativePath: string;
49
- /** Total test count (it/test calls) */
50
- testCount: number;
51
- /** Top-level describe blocks with their test names */
52
- describes: DescribeBlock[];
53
- }
54
-
55
- /** Scan result */
56
- export interface TestScanResult {
57
- files: TestFileInfo[];
58
- totalTests: number;
59
- summary: string;
60
- tokens: number;
61
- }
62
-
63
- // ============================================================================
64
- // Regex Extraction
65
- // ============================================================================
66
-
67
- /**
68
- * Extract describe and test block names from test file source.
69
- *
70
- * Uses regex to find:
71
- * - `describe("name", ...)` / `describe('name', ...)`
72
- * - `test("name", ...)` / `it("name", ...)`
73
- *
74
- * Only extracts top-level describes (not nested). All test/it calls
75
- * are associated with the most recent describe block.
76
- */
77
- export function extractTestStructure(source: string): { describes: DescribeBlock[]; testCount: number } {
78
- const describes: DescribeBlock[] = [];
79
- let currentDescribe: DescribeBlock | null = null;
80
- let testCount = 0;
81
-
82
- // Match describe/test/it calls with string arguments (single or double quotes, backticks)
83
- // Match describe/test/it calls anywhere (not just line-start) to handle single-line files
84
- const linePattern = /(?:^|\s|;|\{)(describe|test|it)\s*\(\s*(['"`])(.*?)\2/gm;
85
-
86
- let match: RegExpExecArray | null = linePattern.exec(source);
87
- while (match !== null) {
88
- const keyword = match[1];
89
- const name = match[3];
90
-
91
- if (keyword === "describe") {
92
- currentDescribe = { name, tests: [] };
93
- describes.push(currentDescribe);
94
- } else {
95
- // test or it
96
- testCount++;
97
- if (currentDescribe) {
98
- currentDescribe.tests.push(name);
99
- } else {
100
- // Top-level test without describe
101
- if (describes.length === 0 || describes[describes.length - 1].name !== "(top-level)") {
102
- describes.push({ name: "(top-level)", tests: [] });
103
- }
104
- describes[describes.length - 1].tests.push(name);
105
- }
106
- }
107
- match = linePattern.exec(source);
108
- }
109
-
110
- return { describes, testCount };
111
- }
112
-
113
- // ============================================================================
114
- // File Scanning
115
- // ============================================================================
116
-
117
- /** Common test directory names to auto-detect */
118
- const COMMON_TEST_DIRS = ["test", "tests", "__tests__", "src/__tests__", "spec"];
119
-
120
- /**
121
- * Derive test file patterns from source file paths.
122
- *
123
- * Maps source files to their likely test file counterparts:
124
- * - src/foo.ts → test/foo.test.ts, test/foo.spec.ts
125
- * - src/bar/baz.service.ts → test/bar/baz.service.test.ts, test/baz.service.test.ts
126
- *
127
- * @param contextFiles - Array of source file paths (relative to workdir)
128
- * @returns Array of test file path patterns (basename patterns for matching)
129
- */
130
- export function deriveTestPatterns(contextFiles: string[]): string[] {
131
- const patterns = new Set<string>();
132
-
133
- for (const filePath of contextFiles) {
134
- const basename = path.basename(filePath);
135
- const basenameNoExt = basename.replace(/\.(ts|js|tsx|jsx)$/, "");
136
-
137
- // Pattern 1: exact basename match with .test/.spec extension
138
- // e.g., foo.ts → foo.test.ts, foo.spec.ts
139
- patterns.add(`${basenameNoExt}.test.ts`);
140
- patterns.add(`${basenameNoExt}.test.js`);
141
- patterns.add(`${basenameNoExt}.test.tsx`);
142
- patterns.add(`${basenameNoExt}.test.jsx`);
143
- patterns.add(`${basenameNoExt}.spec.ts`);
144
- patterns.add(`${basenameNoExt}.spec.js`);
145
- patterns.add(`${basenameNoExt}.spec.tsx`);
146
- patterns.add(`${basenameNoExt}.spec.jsx`);
147
-
148
- // Pattern 2: if basename contains .service/.controller/etc, also match without it
149
- // e.g., foo.service.ts → foo.test.ts
150
- const simpleBasename = basenameNoExt.replace(
151
- /\.(service|controller|resolver|module|guard|middleware|util|helper)$/,
152
- "",
153
- );
154
- if (simpleBasename !== basenameNoExt) {
155
- patterns.add(`${simpleBasename}.test.ts`);
156
- patterns.add(`${simpleBasename}.test.js`);
157
- patterns.add(`${simpleBasename}.spec.ts`);
158
- patterns.add(`${simpleBasename}.spec.js`);
159
- }
160
- }
161
-
162
- return Array.from(patterns);
163
- }
164
-
165
- /**
166
- * Auto-detect test directory by checking common locations.
167
- */
168
- async function detectTestDir(workdir: string): Promise<string | null> {
169
- for (const dir of COMMON_TEST_DIRS) {
170
- const fullPath = path.join(workdir, dir);
171
- const file = Bun.file(path.join(fullPath, ".")); // Check directory exists
172
- try {
173
- const dirStat = await Bun.file(fullPath).exists();
174
- // Bun.file().exists() returns false for directories, use different check
175
- const proc = Bun.spawn(["test", "-d", fullPath], { stdout: "pipe", stderr: "pipe" });
176
- const exitCode = await proc.exited;
177
- if (exitCode === 0) return dir;
178
- } catch {}
179
- }
180
- return null;
181
- }
182
-
183
- /**
184
- * Scan test files and extract structure.
185
- *
186
- * @param options - Scan options
187
- * @returns Array of parsed test file info
188
- */
189
- export async function scanTestFiles(options: TestScanOptions): Promise<TestFileInfo[]> {
190
- const { workdir, testPattern = "**/*.test.{ts,js,tsx,jsx}", contextFiles, scopeToStory = true } = options;
191
- let testDir = options.testDir;
192
-
193
- // Auto-detect test directory if not specified
194
- if (!testDir) {
195
- testDir = (await detectTestDir(workdir)) || "test";
196
- }
197
-
198
- const scanDir = path.join(workdir, testDir);
199
-
200
- // Check directory exists
201
- const dirCheck = Bun.spawn(["test", "-d", scanDir], { stdout: "pipe", stderr: "pipe" });
202
- if ((await dirCheck.exited) !== 0) {
203
- return [];
204
- }
205
-
206
- // Derive test patterns from context files if scoping is enabled
207
- let allowedBasenames: Set<string> | null = null;
208
- if (scopeToStory && contextFiles && contextFiles.length > 0) {
209
- const patterns = deriveTestPatterns(contextFiles);
210
- allowedBasenames = new Set(patterns);
211
- }
212
-
213
- const glob = new Glob(testPattern);
214
- const files: TestFileInfo[] = [];
215
-
216
- for await (const filePath of glob.scan({ cwd: scanDir, absolute: false })) {
217
- // Filter by derived patterns if scoping is enabled
218
- if (allowedBasenames !== null) {
219
- const basename = path.basename(filePath);
220
- if (!allowedBasenames.has(basename)) {
221
- continue; // Skip test files not matching context files
222
- }
223
- }
224
-
225
- const fullPath = path.join(scanDir, filePath);
226
- try {
227
- const source = await Bun.file(fullPath).text();
228
- const { describes, testCount } = extractTestStructure(source);
229
-
230
- if (testCount > 0 || describes.length > 0) {
231
- files.push({
232
- relativePath: path.join(testDir, filePath),
233
- testCount,
234
- describes,
235
- });
236
- }
237
- } catch {}
238
- }
239
-
240
- // Sort by path for stable output
241
- files.sort((a, b) => a.relativePath.localeCompare(b.relativePath));
242
-
243
- return files;
244
- }
245
-
246
- // ============================================================================
247
- // Summary Formatting
248
- // ============================================================================
249
-
250
- /**
251
- * Format test files as a markdown summary at the specified detail level.
252
- */
253
- export function formatTestSummary(files: TestFileInfo[], detail: TestSummaryDetail): string {
254
- if (files.length === 0) {
255
- return "";
256
- }
257
-
258
- const lines: string[] = [];
259
- const totalTests = files.reduce((sum, f) => sum + f.testCount, 0);
260
-
261
- lines.push(`## Existing Test Coverage (${totalTests} tests across ${files.length} files)`);
262
- lines.push("");
263
- lines.push("The following tests already exist. DO NOT duplicate this coverage.");
264
- lines.push("Focus only on testing NEW behavior introduced by this story.");
265
- lines.push("");
266
-
267
- for (const file of files) {
268
- switch (detail) {
269
- case "names-only":
270
- lines.push(`- **${file.relativePath}** (${file.testCount} tests)`);
271
- break;
272
-
273
- case "names-and-counts":
274
- lines.push(`### ${file.relativePath} (${file.testCount} tests)`);
275
- for (const desc of file.describes) {
276
- lines.push(`- ${desc.name} (${desc.tests.length} tests)`);
277
- }
278
- lines.push("");
279
- break;
280
-
281
- case "describe-blocks":
282
- lines.push(`### ${file.relativePath} (${file.testCount} tests)`);
283
- for (const desc of file.describes) {
284
- lines.push(`- **${desc.name}** (${desc.tests.length} tests)`);
285
- for (const test of desc.tests) {
286
- lines.push(` - ${test}`);
287
- }
288
- }
289
- lines.push("");
290
- break;
291
- }
292
- }
293
-
294
- return lines.join("\n");
295
- }
296
-
297
- /**
298
- * Truncate summary to fit within token budget.
299
- *
300
- * Strategy: progressively reduce detail level, then truncate files.
301
- */
302
- export function truncateToTokenBudget(
303
- files: TestFileInfo[],
304
- maxTokens: number,
305
- preferredDetail: TestSummaryDetail,
306
- ): { summary: string; detail: TestSummaryDetail; truncated: boolean } {
307
- // Try preferred detail level first
308
- const detailLevels: TestSummaryDetail[] = ["describe-blocks", "names-and-counts", "names-only"];
309
- const startIndex = detailLevels.indexOf(preferredDetail);
310
-
311
- for (let i = startIndex; i < detailLevels.length; i++) {
312
- const detail = detailLevels[i];
313
- const summary = formatTestSummary(files, detail);
314
- const tokens = estimateTokens(summary);
315
-
316
- if (tokens <= maxTokens) {
317
- return { summary, detail, truncated: i !== startIndex };
318
- }
319
- }
320
-
321
- // Even names-only exceeds budget — truncate files
322
- let truncatedFiles = [...files];
323
- while (truncatedFiles.length > 1) {
324
- truncatedFiles = truncatedFiles.slice(0, truncatedFiles.length - 1);
325
- const summary = `${formatTestSummary(truncatedFiles, "names-only")}\n... and ${files.length - truncatedFiles.length} more test files`;
326
- if (estimateTokens(summary) <= maxTokens) {
327
- return { summary, detail: "names-only", truncated: true };
328
- }
329
- }
330
-
331
- // Last resort: just file count
332
- const fallback = `## Existing Test Coverage\n\n${files.length} test files with ${files.reduce((s, f) => s + f.testCount, 0)} total tests exist. Review test/ directory before adding new tests.`;
333
- return { summary: fallback, detail: "names-only", truncated: true };
334
- }
335
-
336
- // ============================================================================
337
- // Main Entry Point
338
- // ============================================================================
339
-
340
- /**
341
- * Scan test files and generate a token-budgeted summary.
342
- *
343
- * @param options - Scan and formatting options
344
- * @returns Scan result with summary, or empty result if no tests found
345
- */
346
- export async function generateTestCoverageSummary(options: TestScanOptions): Promise<TestScanResult> {
347
- const { maxTokens = 500, detail = "names-and-counts", contextFiles, scopeToStory = true } = options;
348
-
349
- // Log warning if scoping is enabled but no context files provided
350
- if (scopeToStory && (!contextFiles || contextFiles.length === 0)) {
351
- try {
352
- const logger = getLogger();
353
- logger.warn("context", "scopeToStory=true but no contextFiles provided — falling back to full scan");
354
- } catch {
355
- // Logger not initialized (e.g., in tests) — silently skip
356
- }
357
- }
358
-
359
- const files = await scanTestFiles(options);
360
-
361
- if (files.length === 0) {
362
- return { files: [], totalTests: 0, summary: "", tokens: 0 };
363
- }
364
-
365
- const totalTests = files.reduce((sum, f) => sum + f.testCount, 0);
366
- const { summary } = truncateToTokenBudget(files, maxTokens, detail);
367
- const tokens = estimateTokens(summary);
368
-
369
- return { files, totalTests, summary, tokens };
370
- }
@@ -1,98 +0,0 @@
1
- /**
2
- * Context Generator Types (v0.16.1)
3
- *
4
- * Types for generating agent config files from nax/context.md.
5
- * Replaces ConstitutionContent from the old constitution generator.
6
- */
7
-
8
- /** Auto-injected project metadata */
9
- export interface ProjectMetadata {
10
- /** Project name from manifest file */
11
- name?: string;
12
- /** Detected language/runtime (e.g. "TypeScript", "Go", "Rust", "Python") */
13
- language?: string;
14
- /** Key dependencies (framework, ORM, test runner, etc.) */
15
- dependencies: string[];
16
- /** Test command from nax config */
17
- testCommand?: string;
18
- /** Lint command from nax config */
19
- lintCommand?: string;
20
- /** Typecheck command from nax config */
21
- typecheckCommand?: string;
22
- }
23
-
24
- /** Context content passed to generators */
25
- export interface ContextContent {
26
- /** Raw markdown from nax/context.md */
27
- markdown: string;
28
- /** Auto-injected project metadata (if enabled) */
29
- metadata?: ProjectMetadata;
30
- }
31
-
32
- /** Agent config generator interface */
33
- export interface AgentContextGenerator {
34
- /** Generator name (e.g., 'claude', 'opencode', 'cursor') */
35
- name: string;
36
- /** Output filename (e.g., 'CLAUDE.md', '.cursorrules') */
37
- outputFile: string;
38
- /** Generate agent-specific config file content from context */
39
- generate(context: ContextContent): string;
40
- }
41
-
42
- /** All available generator types */
43
- export type AgentType = "claude" | "codex" | "opencode" | "cursor" | "windsurf" | "aider" | "gemini";
44
-
45
- /** Generator registry map */
46
- export type GeneratorMap = Record<AgentType, AgentContextGenerator>;
47
-
48
- /** A single context element (file content, error, story summary, etc.) */
49
- export interface ContextElement {
50
- /** Element type identifier */
51
- type: string;
52
- /** Content text */
53
- content: string;
54
- /** Estimated token count */
55
- tokens: number;
56
- /** Priority (higher = selected first when budgeting) */
57
- priority: number;
58
- /** Story ID (for story/dependency elements) */
59
- storyId?: string;
60
- /** File path (for file elements) */
61
- filePath?: string;
62
- /** Human-readable label (optional) */
63
- label?: string;
64
- }
65
-
66
- /** Token budget for context building */
67
- export interface ContextBudget {
68
- /** Total token limit */
69
- maxTokens: number;
70
- /** Tokens reserved for instructions/system prompt */
71
- reservedForInstructions: number;
72
- /** Tokens available for context elements */
73
- availableForContext: number;
74
- }
75
-
76
- /** Input to the context builder */
77
- export interface StoryContext {
78
- /** PRD containing all stories */
79
- prd: import("../prd/types").PRD;
80
- /** ID of the current story being worked on */
81
- currentStoryId: string;
82
- /** Working directory for file scanning */
83
- workdir?: string;
84
- /** nax config (for context settings) */
85
- config?: import("../config").NaxConfig;
86
- }
87
-
88
- /** Output of the context builder */
89
- export interface BuiltContext {
90
- /** Selected context elements (within budget) */
91
- elements: ContextElement[];
92
- /** Total tokens used */
93
- totalTokens: number;
94
- /** Whether some elements were truncated due to budget */
95
- truncated: boolean;
96
- /** Human-readable summary */
97
- summary: string;
98
- }
@@ -1,50 +0,0 @@
1
- /**
2
- * PRD Mutation — Apply Decomposition (SD-003)
3
- *
4
- * Marks the original story as 'decomposed', inserts substories after the
5
- * original with status 'pending' and parentStoryId.
6
- */
7
-
8
- import type { PRD, UserStory } from "../prd";
9
- import type { DecomposeResult } from "./types";
10
-
11
- /**
12
- * Apply a decomposition result to a PRD:
13
- * - Marks the original story as 'decomposed'
14
- * - Inserts substories after the original with status 'pending' and parentStoryId
15
- */
16
- export function applyDecomposition(prd: PRD, result: DecomposeResult): void {
17
- const { subStories } = result;
18
- if (subStories.length === 0) return;
19
-
20
- const parentStoryId = subStories[0].parentStoryId;
21
- const originalIndex = prd.userStories.findIndex((s) => s.id === parentStoryId);
22
- if (originalIndex === -1) return;
23
-
24
- const parentStory = prd.userStories[originalIndex];
25
-
26
- // Mark original story as decomposed
27
- parentStory.status = "decomposed";
28
-
29
- // Convert substories to UserStory format with parentStoryId attached
30
- // ENH-008: Inherit workdir from parent so sub-stories run in the same package scope
31
- const newStories = subStories.map(
32
- (sub): UserStory => ({
33
- id: sub.id,
34
- title: sub.title,
35
- description: sub.description,
36
- acceptanceCriteria: sub.acceptanceCriteria,
37
- tags: sub.tags,
38
- dependencies: sub.dependencies,
39
- status: "pending",
40
- passes: false,
41
- escalations: [],
42
- attempts: 0,
43
- parentStoryId: sub.parentStoryId,
44
- ...(parentStory.workdir !== undefined && { workdir: parentStory.workdir }),
45
- }),
46
- );
47
-
48
- // Insert substories immediately after the original story
49
- prd.userStories.splice(originalIndex + 1, 0, ...newStories);
50
- }
@@ -1,181 +0,0 @@
1
- /**
2
- * DecomposeBuilder — fluent API for composing story decomposition prompts.
3
- *
4
- * Usage:
5
- * const result = await DecomposeBuilder.for(story)
6
- * .prd(prd)
7
- * .codebase(scan)
8
- * .config(cfg)
9
- * .decompose(adapter);
10
- */
11
-
12
- import type { CodebaseScan } from "../analyze/types";
13
- import type { PRD, UserStory } from "../prd";
14
- import { buildCodebaseSection } from "./sections/codebase";
15
- import { buildConstraintsSection } from "./sections/constraints";
16
- import { buildSiblingStoriesSection } from "./sections/sibling-stories";
17
- import { buildTargetStorySection } from "./sections/target-story";
18
- import type { DecomposeAdapter, DecomposeConfig, DecomposeResult, SubStory } from "./types";
19
- import { runAllValidators } from "./validators/index";
20
-
21
- export const SECTION_SEP = "\n\n---\n\n";
22
-
23
- export class DecomposeBuilder {
24
- private _story: UserStory;
25
- private _prd: PRD | undefined;
26
- private _scan: CodebaseScan | undefined;
27
- private _cfg: DecomposeConfig | undefined;
28
-
29
- private constructor(story: UserStory) {
30
- this._story = story;
31
- }
32
-
33
- static for(story: UserStory): DecomposeBuilder {
34
- return new DecomposeBuilder(story);
35
- }
36
-
37
- prd(prd: PRD): this {
38
- this._prd = prd;
39
- return this;
40
- }
41
-
42
- codebase(scan: CodebaseScan): this {
43
- this._scan = scan;
44
- return this;
45
- }
46
-
47
- config(cfg: DecomposeConfig): this {
48
- this._cfg = cfg;
49
- return this;
50
- }
51
-
52
- buildPrompt(errorFeedback?: string): string {
53
- const sections: string[] = [];
54
-
55
- sections.push(buildTargetStorySection(this._story));
56
-
57
- if (this._prd) {
58
- sections.push(buildSiblingStoriesSection(this._story, this._prd));
59
- }
60
-
61
- if (this._scan) {
62
- sections.push(buildCodebaseSection(this._scan));
63
- }
64
-
65
- if (this._cfg) {
66
- sections.push(buildConstraintsSection(this._cfg));
67
- }
68
-
69
- if (errorFeedback) {
70
- sections.push(
71
- `## Validation Errors from Previous Attempt\n\nFix the following errors and try again:\n\n${errorFeedback}`,
72
- );
73
- }
74
-
75
- return sections.join(SECTION_SEP);
76
- }
77
-
78
- async decompose(adapter: DecomposeAdapter): Promise<DecomposeResult> {
79
- const cfg = this._cfg;
80
- const maxRetries = cfg?.maxRetries ?? 0;
81
- const existingStories = this._prd ? this._prd.userStories.filter((s) => s.id !== this._story.id) : [];
82
-
83
- let lastResult: DecomposeResult | undefined;
84
- let errorFeedback: string | undefined;
85
-
86
- for (let attempt = 0; attempt <= maxRetries; attempt++) {
87
- const prompt = this.buildPrompt(errorFeedback);
88
- const raw = await adapter.decompose(prompt);
89
- const parsed = parseSubStories(raw);
90
-
91
- if (!parsed.validation.valid) {
92
- lastResult = parsed;
93
- errorFeedback = parsed.validation.errors.join("\n");
94
- continue;
95
- }
96
-
97
- // Run post-parse validators
98
- const config: DecomposeConfig = cfg ?? { maxSubStories: 5, maxComplexity: "medium" };
99
- const validation = runAllValidators(this._story, parsed.subStories, existingStories, config);
100
-
101
- if (!validation.valid) {
102
- lastResult = { subStories: parsed.subStories, validation };
103
- errorFeedback = validation.errors.join("\n");
104
- continue;
105
- }
106
-
107
- return { subStories: parsed.subStories, validation };
108
- }
109
-
110
- return (
111
- lastResult ?? {
112
- subStories: [],
113
- validation: { valid: false, errors: ["Decomposition failed after all retries"], warnings: [] },
114
- }
115
- );
116
- }
117
- }
118
-
119
- function parseSubStories(output: string): DecomposeResult {
120
- // Extract JSON array (handles optional markdown code fences)
121
- const fenceMatch = output.match(/```(?:json)?\s*(\[[\s\S]*?\])\s*```/);
122
- let jsonText = fenceMatch ? fenceMatch[1] : output;
123
-
124
- if (!fenceMatch) {
125
- const arrayMatch = output.match(/\[[\s\S]*\]/);
126
- if (arrayMatch) {
127
- jsonText = arrayMatch[0];
128
- }
129
- }
130
-
131
- let parsed: unknown;
132
- try {
133
- parsed = JSON.parse(jsonText.trim());
134
- } catch (err) {
135
- return {
136
- subStories: [],
137
- validation: { valid: false, errors: [`Failed to parse JSON: ${(err as Error).message}`], warnings: [] },
138
- };
139
- }
140
-
141
- if (!Array.isArray(parsed)) {
142
- return {
143
- subStories: [],
144
- validation: { valid: false, errors: ["Output is not a JSON array"], warnings: [] },
145
- };
146
- }
147
-
148
- const errors: string[] = [];
149
- const subStories: SubStory[] = [];
150
-
151
- for (const [index, item] of parsed.entries()) {
152
- if (typeof item !== "object" || item === null) {
153
- errors.push(`Item at index ${index} is not an object`);
154
- continue;
155
- }
156
- const r = item as Record<string, unknown>;
157
- subStories.push({
158
- id: String(r.id ?? ""),
159
- parentStoryId: String(r.parentStoryId ?? ""),
160
- title: String(r.title ?? ""),
161
- description: String(r.description ?? ""),
162
- acceptanceCriteria: Array.isArray(r.acceptanceCriteria) ? (r.acceptanceCriteria as string[]) : [],
163
- tags: Array.isArray(r.tags) ? (r.tags as string[]) : [],
164
- dependencies: Array.isArray(r.dependencies) ? (r.dependencies as string[]) : [],
165
- complexity: normalizeComplexity(r.complexity),
166
- nonOverlapJustification: String(r.nonOverlapJustification ?? ""),
167
- });
168
- }
169
-
170
- return {
171
- subStories,
172
- validation: { valid: errors.length === 0, errors, warnings: [] },
173
- };
174
- }
175
-
176
- function normalizeComplexity(value: unknown): "simple" | "medium" | "complex" | "expert" {
177
- if (value === "simple" || value === "medium" || value === "complex" || value === "expert") {
178
- return value;
179
- }
180
- return "medium";
181
- }