@nathapp/nax 0.50.2 → 0.51.0

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 (352) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/dist/nax.js +579 -373
  3. package/package.json +1 -3
  4. package/bin/nax.ts +0 -1195
  5. package/src/acceptance/fix-generator.ts +0 -322
  6. package/src/acceptance/generator.ts +0 -423
  7. package/src/acceptance/index.ts +0 -42
  8. package/src/acceptance/refinement.ts +0 -224
  9. package/src/acceptance/templates/cli.ts +0 -47
  10. package/src/acceptance/templates/component.ts +0 -78
  11. package/src/acceptance/templates/e2e.ts +0 -43
  12. package/src/acceptance/templates/index.ts +0 -21
  13. package/src/acceptance/templates/snapshot.ts +0 -50
  14. package/src/acceptance/templates/unit.ts +0 -48
  15. package/src/acceptance/types.ts +0 -135
  16. package/src/agents/acp/adapter.ts +0 -888
  17. package/src/agents/acp/cost.ts +0 -9
  18. package/src/agents/acp/index.ts +0 -7
  19. package/src/agents/acp/interaction-bridge.ts +0 -126
  20. package/src/agents/acp/parser.ts +0 -119
  21. package/src/agents/acp/spawn-client.ts +0 -373
  22. package/src/agents/acp/types.ts +0 -22
  23. package/src/agents/aider/adapter.ts +0 -135
  24. package/src/agents/claude/adapter.ts +0 -258
  25. package/src/agents/claude/complete.ts +0 -80
  26. package/src/agents/claude/cost.ts +0 -16
  27. package/src/agents/claude/execution.ts +0 -215
  28. package/src/agents/claude/index.ts +0 -3
  29. package/src/agents/claude/interactive.ts +0 -77
  30. package/src/agents/claude/plan.ts +0 -179
  31. package/src/agents/codex/adapter.ts +0 -153
  32. package/src/agents/cost/calculate.ts +0 -154
  33. package/src/agents/cost/index.ts +0 -10
  34. package/src/agents/cost/parse.ts +0 -97
  35. package/src/agents/cost/pricing.ts +0 -59
  36. package/src/agents/cost/types.ts +0 -45
  37. package/src/agents/gemini/adapter.ts +0 -177
  38. package/src/agents/index.ts +0 -18
  39. package/src/agents/opencode/adapter.ts +0 -106
  40. package/src/agents/registry.ts +0 -136
  41. package/src/agents/shared/decompose.ts +0 -154
  42. package/src/agents/shared/model-resolution.ts +0 -43
  43. package/src/agents/shared/types-extended.ts +0 -164
  44. package/src/agents/shared/validation.ts +0 -69
  45. package/src/agents/shared/version-detection.ts +0 -109
  46. package/src/agents/types.ts +0 -205
  47. package/src/analyze/classifier.ts +0 -282
  48. package/src/analyze/index.ts +0 -16
  49. package/src/analyze/scanner.ts +0 -171
  50. package/src/analyze/types.ts +0 -51
  51. package/src/cli/accept.ts +0 -108
  52. package/src/cli/agents.ts +0 -87
  53. package/src/cli/analyze-parser.ts +0 -291
  54. package/src/cli/analyze.ts +0 -352
  55. package/src/cli/config-descriptions.ts +0 -218
  56. package/src/cli/config-diff.ts +0 -103
  57. package/src/cli/config-display.ts +0 -285
  58. package/src/cli/config-get.ts +0 -55
  59. package/src/cli/config.ts +0 -14
  60. package/src/cli/constitution.ts +0 -17
  61. package/src/cli/diagnose-analysis.ts +0 -159
  62. package/src/cli/diagnose-formatter.ts +0 -87
  63. package/src/cli/diagnose.ts +0 -203
  64. package/src/cli/generate.ts +0 -250
  65. package/src/cli/index.ts +0 -42
  66. package/src/cli/init-context.ts +0 -405
  67. package/src/cli/init-detect.ts +0 -303
  68. package/src/cli/init.ts +0 -296
  69. package/src/cli/interact.ts +0 -295
  70. package/src/cli/plan.ts +0 -509
  71. package/src/cli/plugins.ts +0 -122
  72. package/src/cli/prompts-export.ts +0 -58
  73. package/src/cli/prompts-init.ts +0 -200
  74. package/src/cli/prompts-main.ts +0 -183
  75. package/src/cli/prompts-shared.ts +0 -70
  76. package/src/cli/prompts-tdd.ts +0 -88
  77. package/src/cli/prompts.ts +0 -17
  78. package/src/cli/runs.ts +0 -174
  79. package/src/cli/status-cost.ts +0 -151
  80. package/src/cli/status-features.ts +0 -405
  81. package/src/cli/status.ts +0 -13
  82. package/src/commands/common.ts +0 -171
  83. package/src/commands/diagnose.ts +0 -17
  84. package/src/commands/index.ts +0 -9
  85. package/src/commands/logs-formatter.ts +0 -201
  86. package/src/commands/logs-reader.ts +0 -171
  87. package/src/commands/logs.ts +0 -103
  88. package/src/commands/precheck.ts +0 -86
  89. package/src/commands/runs.ts +0 -220
  90. package/src/commands/unlock.ts +0 -96
  91. package/src/config/defaults.ts +0 -217
  92. package/src/config/index.ts +0 -22
  93. package/src/config/loader.ts +0 -143
  94. package/src/config/merge.ts +0 -106
  95. package/src/config/merger.ts +0 -147
  96. package/src/config/path-security.ts +0 -121
  97. package/src/config/paths.ts +0 -27
  98. package/src/config/permissions.ts +0 -63
  99. package/src/config/runtime-types.ts +0 -520
  100. package/src/config/schema-types.ts +0 -53
  101. package/src/config/schema.ts +0 -60
  102. package/src/config/schemas.ts +0 -425
  103. package/src/config/test-strategy.ts +0 -71
  104. package/src/config/types.ts +0 -57
  105. package/src/config/validate.ts +0 -103
  106. package/src/constitution/generator.ts +0 -158
  107. package/src/constitution/generators/aider.ts +0 -41
  108. package/src/constitution/generators/claude.ts +0 -35
  109. package/src/constitution/generators/cursor.ts +0 -36
  110. package/src/constitution/generators/opencode.ts +0 -38
  111. package/src/constitution/generators/types.ts +0 -33
  112. package/src/constitution/generators/windsurf.ts +0 -36
  113. package/src/constitution/index.ts +0 -11
  114. package/src/constitution/loader.ts +0 -121
  115. package/src/constitution/types.ts +0 -31
  116. package/src/context/auto-detect.ts +0 -228
  117. package/src/context/builder.ts +0 -299
  118. package/src/context/elements.ts +0 -122
  119. package/src/context/formatter.ts +0 -107
  120. package/src/context/generator.ts +0 -343
  121. package/src/context/generators/aider.ts +0 -34
  122. package/src/context/generators/claude.ts +0 -28
  123. package/src/context/generators/codex.ts +0 -28
  124. package/src/context/generators/cursor.ts +0 -28
  125. package/src/context/generators/gemini.ts +0 -28
  126. package/src/context/generators/opencode.ts +0 -30
  127. package/src/context/generators/windsurf.ts +0 -28
  128. package/src/context/greenfield.ts +0 -114
  129. package/src/context/index.ts +0 -34
  130. package/src/context/injector.ts +0 -279
  131. package/src/context/parent-context.ts +0 -39
  132. package/src/context/test-scanner.ts +0 -370
  133. package/src/context/types.ts +0 -98
  134. package/src/decompose/apply.ts +0 -50
  135. package/src/decompose/builder.ts +0 -181
  136. package/src/decompose/index.ts +0 -8
  137. package/src/decompose/sections/codebase.ts +0 -26
  138. package/src/decompose/sections/constraints.ts +0 -32
  139. package/src/decompose/sections/index.ts +0 -4
  140. package/src/decompose/sections/sibling-stories.ts +0 -25
  141. package/src/decompose/sections/target-story.ts +0 -31
  142. package/src/decompose/types.ts +0 -55
  143. package/src/decompose/validators/complexity.ts +0 -45
  144. package/src/decompose/validators/coverage.ts +0 -134
  145. package/src/decompose/validators/dependency.ts +0 -91
  146. package/src/decompose/validators/index.ts +0 -35
  147. package/src/decompose/validators/overlap.ts +0 -128
  148. package/src/errors.ts +0 -67
  149. package/src/execution/batching.ts +0 -157
  150. package/src/execution/crash-heartbeat.ts +0 -77
  151. package/src/execution/crash-recovery.ts +0 -79
  152. package/src/execution/crash-signals.ts +0 -165
  153. package/src/execution/crash-writer.ts +0 -154
  154. package/src/execution/deferred-review.ts +0 -105
  155. package/src/execution/dry-run.ts +0 -81
  156. package/src/execution/escalation/escalation.ts +0 -46
  157. package/src/execution/escalation/index.ts +0 -13
  158. package/src/execution/escalation/tier-escalation.ts +0 -346
  159. package/src/execution/escalation/tier-outcome.ts +0 -143
  160. package/src/execution/executor-types.ts +0 -73
  161. package/src/execution/helpers.ts +0 -38
  162. package/src/execution/index.ts +0 -27
  163. package/src/execution/iteration-runner.ts +0 -160
  164. package/src/execution/lifecycle/acceptance-loop.ts +0 -280
  165. package/src/execution/lifecycle/headless-formatter.ts +0 -83
  166. package/src/execution/lifecycle/index.ts +0 -11
  167. package/src/execution/lifecycle/parallel-lifecycle.ts +0 -101
  168. package/src/execution/lifecycle/precheck-runner.ts +0 -140
  169. package/src/execution/lifecycle/run-cleanup.ts +0 -81
  170. package/src/execution/lifecycle/run-completion.ts +0 -247
  171. package/src/execution/lifecycle/run-initialization.ts +0 -187
  172. package/src/execution/lifecycle/run-regression.ts +0 -305
  173. package/src/execution/lifecycle/run-setup.ts +0 -240
  174. package/src/execution/lifecycle/story-size-prompts.ts +0 -123
  175. package/src/execution/lock.ts +0 -129
  176. package/src/execution/parallel-coordinator.ts +0 -281
  177. package/src/execution/parallel-executor-rectification-pass.ts +0 -117
  178. package/src/execution/parallel-executor-rectify.ts +0 -136
  179. package/src/execution/parallel-executor.ts +0 -330
  180. package/src/execution/parallel-worker.ts +0 -149
  181. package/src/execution/parallel.ts +0 -13
  182. package/src/execution/pid-registry.ts +0 -275
  183. package/src/execution/pipeline-result-handler.ts +0 -221
  184. package/src/execution/progress.ts +0 -27
  185. package/src/execution/queue-handler.ts +0 -109
  186. package/src/execution/runner-completion.ts +0 -171
  187. package/src/execution/runner-execution.ts +0 -243
  188. package/src/execution/runner-setup.ts +0 -86
  189. package/src/execution/runner.ts +0 -265
  190. package/src/execution/sequential-executor.ts +0 -219
  191. package/src/execution/status-file.ts +0 -264
  192. package/src/execution/status-writer.ts +0 -181
  193. package/src/execution/story-context.ts +0 -266
  194. package/src/execution/story-selector.ts +0 -76
  195. package/src/execution/test-output-parser.ts +0 -14
  196. package/src/execution/timeout-handler.ts +0 -100
  197. package/src/hooks/index.ts +0 -2
  198. package/src/hooks/runner.ts +0 -280
  199. package/src/hooks/types.ts +0 -79
  200. package/src/interaction/chain.ts +0 -170
  201. package/src/interaction/index.ts +0 -61
  202. package/src/interaction/init.ts +0 -84
  203. package/src/interaction/plugins/auto.ts +0 -243
  204. package/src/interaction/plugins/cli.ts +0 -300
  205. package/src/interaction/plugins/telegram.ts +0 -384
  206. package/src/interaction/plugins/webhook.ts +0 -286
  207. package/src/interaction/state.ts +0 -171
  208. package/src/interaction/triggers.ts +0 -250
  209. package/src/interaction/types.ts +0 -170
  210. package/src/logger/formatters.ts +0 -84
  211. package/src/logger/index.ts +0 -16
  212. package/src/logger/logger.ts +0 -296
  213. package/src/logger/types.ts +0 -48
  214. package/src/logging/formatter.ts +0 -355
  215. package/src/logging/index.ts +0 -22
  216. package/src/logging/types.ts +0 -93
  217. package/src/metrics/aggregator.ts +0 -191
  218. package/src/metrics/index.ts +0 -14
  219. package/src/metrics/tracker.ts +0 -200
  220. package/src/metrics/types.ts +0 -115
  221. package/src/optimizer/index.ts +0 -63
  222. package/src/optimizer/noop.optimizer.ts +0 -24
  223. package/src/optimizer/rule-based.optimizer.ts +0 -248
  224. package/src/optimizer/types.ts +0 -53
  225. package/src/pipeline/event-bus.ts +0 -297
  226. package/src/pipeline/events.ts +0 -130
  227. package/src/pipeline/index.ts +0 -19
  228. package/src/pipeline/runner.ts +0 -149
  229. package/src/pipeline/stages/acceptance-setup.ts +0 -140
  230. package/src/pipeline/stages/acceptance.ts +0 -215
  231. package/src/pipeline/stages/autofix.ts +0 -262
  232. package/src/pipeline/stages/completion.ts +0 -110
  233. package/src/pipeline/stages/constitution.ts +0 -63
  234. package/src/pipeline/stages/context.ts +0 -122
  235. package/src/pipeline/stages/execution.ts +0 -359
  236. package/src/pipeline/stages/index.ts +0 -86
  237. package/src/pipeline/stages/optimizer.ts +0 -74
  238. package/src/pipeline/stages/prompt.ts +0 -79
  239. package/src/pipeline/stages/queue-check.ts +0 -103
  240. package/src/pipeline/stages/rectify.ts +0 -101
  241. package/src/pipeline/stages/regression.ts +0 -99
  242. package/src/pipeline/stages/review.ts +0 -94
  243. package/src/pipeline/stages/routing.ts +0 -276
  244. package/src/pipeline/stages/verify.ts +0 -286
  245. package/src/pipeline/subscribers/events-writer.ts +0 -135
  246. package/src/pipeline/subscribers/hooks.ts +0 -179
  247. package/src/pipeline/subscribers/interaction.ts +0 -103
  248. package/src/pipeline/subscribers/registry.ts +0 -73
  249. package/src/pipeline/subscribers/reporters.ts +0 -174
  250. package/src/pipeline/types.ts +0 -220
  251. package/src/plugins/extensions.ts +0 -225
  252. package/src/plugins/index.ts +0 -33
  253. package/src/plugins/loader.ts +0 -352
  254. package/src/plugins/plugin-logger.ts +0 -41
  255. package/src/plugins/registry.ts +0 -168
  256. package/src/plugins/types.ts +0 -206
  257. package/src/plugins/validator.ts +0 -352
  258. package/src/prd/index.ts +0 -220
  259. package/src/prd/schema.ts +0 -268
  260. package/src/prd/types.ts +0 -273
  261. package/src/prd/validate.ts +0 -41
  262. package/src/precheck/checks-agents.ts +0 -63
  263. package/src/precheck/checks-blockers.ts +0 -23
  264. package/src/precheck/checks-cli.ts +0 -68
  265. package/src/precheck/checks-config.ts +0 -102
  266. package/src/precheck/checks-git.ts +0 -117
  267. package/src/precheck/checks-system.ts +0 -101
  268. package/src/precheck/checks-warnings.ts +0 -221
  269. package/src/precheck/checks.ts +0 -36
  270. package/src/precheck/index.ts +0 -374
  271. package/src/precheck/story-size-gate.ts +0 -144
  272. package/src/precheck/types.ts +0 -31
  273. package/src/prompts/builder.ts +0 -166
  274. package/src/prompts/index.ts +0 -2
  275. package/src/prompts/loader.ts +0 -43
  276. package/src/prompts/sections/conventions.ts +0 -19
  277. package/src/prompts/sections/hermetic.ts +0 -41
  278. package/src/prompts/sections/index.ts +0 -12
  279. package/src/prompts/sections/isolation.ts +0 -70
  280. package/src/prompts/sections/role-task.ts +0 -182
  281. package/src/prompts/sections/story.ts +0 -55
  282. package/src/prompts/sections/verdict.ts +0 -70
  283. package/src/prompts/types.ts +0 -21
  284. package/src/queue/index.ts +0 -2
  285. package/src/queue/manager.ts +0 -254
  286. package/src/queue/types.ts +0 -54
  287. package/src/review/index.ts +0 -8
  288. package/src/review/orchestrator.ts +0 -154
  289. package/src/review/runner.ts +0 -303
  290. package/src/review/types.ts +0 -70
  291. package/src/routing/batch-route.ts +0 -35
  292. package/src/routing/builder.ts +0 -81
  293. package/src/routing/chain.ts +0 -75
  294. package/src/routing/content-hash.ts +0 -25
  295. package/src/routing/index.ts +0 -20
  296. package/src/routing/loader.ts +0 -62
  297. package/src/routing/router.ts +0 -305
  298. package/src/routing/strategies/adaptive.ts +0 -215
  299. package/src/routing/strategies/index.ts +0 -8
  300. package/src/routing/strategies/keyword.ts +0 -180
  301. package/src/routing/strategies/llm-prompts.ts +0 -224
  302. package/src/routing/strategies/llm.ts +0 -320
  303. package/src/routing/strategies/manual.ts +0 -50
  304. package/src/routing/strategy.ts +0 -102
  305. package/src/tdd/cleanup.ts +0 -120
  306. package/src/tdd/index.ts +0 -22
  307. package/src/tdd/isolation.ts +0 -117
  308. package/src/tdd/orchestrator.ts +0 -406
  309. package/src/tdd/prompts.ts +0 -40
  310. package/src/tdd/rectification-gate.ts +0 -274
  311. package/src/tdd/session-runner.ts +0 -263
  312. package/src/tdd/types.ts +0 -84
  313. package/src/tdd/verdict-reader.ts +0 -266
  314. package/src/tdd/verdict.ts +0 -152
  315. package/src/tui/App.tsx +0 -265
  316. package/src/tui/components/AgentPanel.tsx +0 -75
  317. package/src/tui/components/CostOverlay.tsx +0 -118
  318. package/src/tui/components/HelpOverlay.tsx +0 -107
  319. package/src/tui/components/StatusBar.tsx +0 -63
  320. package/src/tui/components/StoriesPanel.tsx +0 -177
  321. package/src/tui/hooks/useKeyboard.ts +0 -142
  322. package/src/tui/hooks/useLayout.ts +0 -137
  323. package/src/tui/hooks/usePipelineEvents.ts +0 -183
  324. package/src/tui/hooks/usePty.ts +0 -189
  325. package/src/tui/index.tsx +0 -38
  326. package/src/tui/types.ts +0 -76
  327. package/src/utils/errors.ts +0 -12
  328. package/src/utils/git.ts +0 -245
  329. package/src/utils/json-file.ts +0 -72
  330. package/src/utils/log-test-output.ts +0 -25
  331. package/src/utils/path-security.ts +0 -73
  332. package/src/utils/queue-writer.ts +0 -54
  333. package/src/verification/crash-detector.ts +0 -34
  334. package/src/verification/executor.ts +0 -250
  335. package/src/verification/index.ts +0 -12
  336. package/src/verification/orchestrator-types.ts +0 -154
  337. package/src/verification/orchestrator.ts +0 -76
  338. package/src/verification/parser.ts +0 -220
  339. package/src/verification/rectification-loop.ts +0 -172
  340. package/src/verification/rectification.ts +0 -108
  341. package/src/verification/runners.ts +0 -129
  342. package/src/verification/smart-runner.ts +0 -307
  343. package/src/verification/strategies/acceptance.ts +0 -136
  344. package/src/verification/strategies/regression.ts +0 -90
  345. package/src/verification/strategies/scoped.ts +0 -154
  346. package/src/verification/types.ts +0 -117
  347. package/src/version.ts +0 -40
  348. package/src/worktree/dispatcher.ts +0 -6
  349. package/src/worktree/index.ts +0 -2
  350. package/src/worktree/manager.ts +0 -193
  351. package/src/worktree/merge.ts +0 -302
  352. 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
- }