@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,114 +0,0 @@
1
- /**
2
- * Greenfield Detection
3
- *
4
- * Detects whether a story is "greenfield" (no existing test files in workdir).
5
- * Greenfield stories skip TDD and use test-after strategy to prevent test-writer
6
- * from producing empty test files (BUG-010).
7
- */
8
-
9
- import { readdir } from "node:fs/promises";
10
- import { join } from "node:path";
11
- import type { UserStory } from "../prd/types";
12
-
13
- /**
14
- * Recursively scan directory for test files.
15
- * Ignores node_modules, dist, build, .next directories.
16
- * Throws error if root directory is unreadable.
17
- */
18
- async function scanForTestFiles(dir: string, testPattern: RegExp, isRootCall = true): Promise<string[]> {
19
- const results: string[] = [];
20
- const ignoreDirs = new Set(["node_modules", "dist", "build", ".next", ".git"]);
21
-
22
- try {
23
- const entries = await readdir(dir, { withFileTypes: true });
24
-
25
- for (const entry of entries) {
26
- const fullPath = join(dir, entry.name);
27
-
28
- if (entry.isDirectory()) {
29
- // Skip ignored directories
30
- if (ignoreDirs.has(entry.name)) continue;
31
-
32
- // Recursively scan subdirectories (not root call)
33
- const subResults = await scanForTestFiles(fullPath, testPattern, false);
34
- results.push(...subResults);
35
- } else if (entry.isFile()) {
36
- // Check if file matches test pattern
37
- if (testPattern.test(entry.name)) {
38
- results.push(fullPath);
39
- }
40
- }
41
- }
42
- } catch (error) {
43
- // If this is the root call and we can't read it, propagate the error
44
- if (isRootCall) {
45
- throw error;
46
- }
47
- // Otherwise, ignore errors from unreadable subdirectories
48
- }
49
-
50
- return results;
51
- }
52
-
53
- /**
54
- * Convert simple glob pattern to regex.
55
- * Supports:
56
- * - ** (any directory depth)
57
- * - * (any characters except /)
58
- * - {a,b,c} (alternatives)
59
- */
60
- function globToRegex(pattern: string): RegExp {
61
- // Extract filename pattern from glob (everything after last /)
62
- const parts = pattern.split("/");
63
- const filePattern = parts[parts.length - 1];
64
-
65
- // Convert glob syntax to regex
66
- const regexStr = filePattern
67
- .replace(/\./g, "\\.") // Escape dots
68
- .replace(/\*/g, "[^/]*") // * = any chars except /
69
- .replace(/\{([^}]+)\}/g, (_, group) => `(${group.replace(/,/g, "|")})`) // {a,b} = (a|b)
70
- .replace(/\\\.\\\*/g, "\\.[^/]*"); // Fix escaped .* back to .\*
71
-
72
- return new RegExp(`${regexStr}$`); // nosemgrep: detect-non-literal-regexp — pattern from internal .gitignore, not user input
73
- }
74
-
75
- /**
76
- * Detect if a story is greenfield based on test file presence in workdir.
77
- *
78
- * A story is greenfield if:
79
- * - No test files exist matching the test pattern in the working directory
80
- *
81
- * This prevents the TDD test-writer from struggling to create tests when there are
82
- * no existing test examples to follow.
83
- *
84
- * @param story - User story to check
85
- * @param workdir - Working directory to scan for test files
86
- * @param testPattern - Glob pattern for test files (default: "**\/*.{test,spec}.{ts,js,tsx,jsx}")
87
- * @returns true if no test files exist (greenfield), false otherwise
88
- *
89
- * @example
90
- * ```ts
91
- * // Empty project with no tests
92
- * const isGreenfield = await isGreenfieldStory(story, "/path/to/project");
93
- * // => true
94
- *
95
- * // Project with existing test files
96
- * const isGreenfield = await isGreenfieldStory(story, "/path/to/project");
97
- * // => false
98
- * ```
99
- */
100
- export async function isGreenfieldStory(
101
- story: UserStory,
102
- workdir: string,
103
- testPattern = "**/*.{test,spec}.{ts,js,tsx,jsx}",
104
- ): Promise<boolean> {
105
- try {
106
- const regex = globToRegex(testPattern);
107
- const testFiles = await scanForTestFiles(workdir, regex);
108
- return testFiles.length === 0;
109
- } catch (error) {
110
- // If scan fails completely (e.g., workdir doesn't exist), assume not greenfield (safe fallback)
111
- // This prevents skipping TDD when we can't determine the actual state
112
- return false;
113
- }
114
- }
@@ -1,34 +0,0 @@
1
- /**
2
- * Context builder module for story-scoped prompt optimization
3
- */
4
-
5
- export type { ContextElement, ContextBudget, StoryContext, BuiltContext } from "./types";
6
-
7
- export {
8
- createStoryContext,
9
- createDependencyContext,
10
- createErrorContext,
11
- createProgressContext,
12
- createFileContext,
13
- sortContextElements,
14
- buildContext,
15
- createTestCoverageContext,
16
- formatContextAsMarkdown,
17
- } from "./builder";
18
-
19
- export { estimateTokens } from "../optimizer/types";
20
-
21
- export {
22
- generateTestCoverageSummary,
23
- scanTestFiles,
24
- extractTestStructure,
25
- formatTestSummary,
26
- truncateToTokenBudget,
27
- type TestScanOptions,
28
- type TestScanResult,
29
- type TestFileInfo,
30
- type DescribeBlock,
31
- type TestSummaryDetail,
32
- } from "./test-scanner";
33
-
34
- export { autoDetectContextFiles, extractKeywords, type AutoDetectOptions } from "./auto-detect";
@@ -1,279 +0,0 @@
1
- /**
2
- * Project Metadata Auto-Injector (v0.16.1)
3
- *
4
- * Detects project language/stack and injects metadata into agent configs.
5
- * Supports: Node.js/Bun (package.json), Go (go.mod), Rust (Cargo.toml),
6
- * Python (pyproject.toml / requirements.txt), PHP (composer.json),
7
- * Ruby (Gemfile), Java/Kotlin (pom.xml / build.gradle).
8
- */
9
-
10
- import { existsSync } from "node:fs";
11
- import { join } from "node:path";
12
- import type { NaxConfig } from "../config";
13
- import type { ProjectMetadata } from "./types";
14
-
15
- /** Notable Node.js dependency keywords */
16
- const NOTABLE_NODE_DEPS = [
17
- "@nestjs",
18
- "express",
19
- "fastify",
20
- "koa",
21
- "hono",
22
- "next",
23
- "nuxt",
24
- "react",
25
- "vue",
26
- "svelte",
27
- "solid",
28
- "prisma",
29
- "typeorm",
30
- "mongoose",
31
- "drizzle",
32
- "sequelize",
33
- "jest",
34
- "vitest",
35
- "mocha",
36
- "bun",
37
- "zod",
38
- "typescript",
39
- "graphql",
40
- "trpc",
41
- "bull",
42
- "ioredis",
43
- ];
44
-
45
- // ─── Language detectors ──────────────────────────────────────────────────────
46
-
47
- /** Node.js / Bun: read package.json */
48
- async function detectNode(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
49
- const pkgPath = join(workdir, "package.json");
50
- if (!existsSync(pkgPath)) return null;
51
-
52
- try {
53
- const file = Bun.file(pkgPath);
54
- const pkg = await file.json();
55
- const allDeps = { ...(pkg.dependencies ?? {}), ...(pkg.devDependencies ?? {}) };
56
- const notable = [
57
- ...new Set(
58
- Object.keys(allDeps).filter((dep) =>
59
- NOTABLE_NODE_DEPS.some((kw) => dep === kw || dep.startsWith(`${kw}/`) || dep.includes(kw)),
60
- ),
61
- ),
62
- ].slice(0, 10);
63
- const lang = pkg.devDependencies?.typescript || pkg.dependencies?.typescript ? "TypeScript" : "JavaScript";
64
- return { name: pkg.name, lang, dependencies: notable };
65
- } catch {
66
- return null;
67
- }
68
- }
69
-
70
- /** Go: read go.mod for module name + direct dependencies */
71
- async function detectGo(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
72
- const goMod = join(workdir, "go.mod");
73
- if (!existsSync(goMod)) return null;
74
-
75
- try {
76
- const content = await Bun.file(goMod).text();
77
- const moduleMatch = content.match(/^module\s+(\S+)/m);
78
- const name = moduleMatch?.[1];
79
-
80
- // Extract require block entries (direct deps, not indirect)
81
- const requires: string[] = [];
82
- const requireBlock = content.match(/require\s*\(([^)]+)\)/s)?.[1] ?? "";
83
- for (const line of requireBlock.split("\n")) {
84
- const trimmed = line.trim();
85
- if (trimmed && !trimmed.startsWith("//") && !trimmed.includes("// indirect")) {
86
- const dep = trimmed.split(/\s+/)[0];
87
- if (dep) requires.push(dep.split("/").slice(-1)[0]); // last segment only
88
- }
89
- }
90
-
91
- return { name, lang: "Go", dependencies: requires.slice(0, 10) };
92
- } catch {
93
- return null;
94
- }
95
- }
96
-
97
- /** Rust: read Cargo.toml for package name + dependencies */
98
- async function detectRust(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
99
- const cargoPath = join(workdir, "Cargo.toml");
100
- if (!existsSync(cargoPath)) return null;
101
-
102
- try {
103
- const content = await Bun.file(cargoPath).text();
104
- const nameMatch = content.match(/^\[package\][^[]*name\s*=\s*"([^"]+)"/ms);
105
- const name = nameMatch?.[1];
106
-
107
- // Extract [dependencies] section keys
108
- const depsSection = content.match(/^\[dependencies\]([^[]*)/ms)?.[1] ?? "";
109
- const deps = depsSection
110
- .split("\n")
111
- .map((l) => l.split("=")[0].trim())
112
- .filter((l) => l && !l.startsWith("#"))
113
- .slice(0, 10);
114
-
115
- return { name, lang: "Rust", dependencies: deps };
116
- } catch {
117
- return null;
118
- }
119
- }
120
-
121
- /** Python: read pyproject.toml or requirements.txt */
122
- async function detectPython(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
123
- const pyproject = join(workdir, "pyproject.toml");
124
- const requirements = join(workdir, "requirements.txt");
125
-
126
- if (!existsSync(pyproject) && !existsSync(requirements)) return null;
127
-
128
- try {
129
- if (existsSync(pyproject)) {
130
- const content = await Bun.file(pyproject).text();
131
- const nameMatch = content.match(/^\s*name\s*=\s*"([^"]+)"/m);
132
- const depsSection = content.match(/^\[project\][^[]*dependencies\s*=\s*\[([^\]]*)\]/ms)?.[1] ?? "";
133
- const deps = depsSection
134
- .split(",")
135
- .map((d) => d.trim().replace(/["'\s>=<!^~].*/g, ""))
136
- .filter(Boolean)
137
- .slice(0, 10);
138
- return { name: nameMatch?.[1], lang: "Python", dependencies: deps };
139
- }
140
-
141
- // Fallback: requirements.txt
142
- const lines = (await Bun.file(requirements).text())
143
- .split("\n")
144
- .map((l) => l.split(/[>=<!]/)[0].trim())
145
- .filter((l) => l && !l.startsWith("#"))
146
- .slice(0, 10);
147
- return { lang: "Python", dependencies: lines };
148
- } catch {
149
- return null;
150
- }
151
- }
152
-
153
- /** PHP: read composer.json */
154
- async function detectPhp(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
155
- const composerPath = join(workdir, "composer.json");
156
- if (!existsSync(composerPath)) return null;
157
-
158
- try {
159
- const file = Bun.file(composerPath);
160
- const composer = await file.json();
161
- const deps = Object.keys({ ...(composer.require ?? {}), ...(composer["require-dev"] ?? {}) })
162
- .filter((d) => d !== "php")
163
- .map((d) => d.split("/").pop() ?? d)
164
- .slice(0, 10);
165
- return { name: composer.name, lang: "PHP", dependencies: deps };
166
- } catch {
167
- return null;
168
- }
169
- }
170
-
171
- /** Ruby: read Gemfile */
172
- async function detectRuby(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
173
- const gemfile = join(workdir, "Gemfile");
174
- if (!existsSync(gemfile)) return null;
175
-
176
- try {
177
- const content = await Bun.file(gemfile).text();
178
- const gems = [...content.matchAll(/^\s*gem\s+['"]([^'"]+)['"]/gm)].map((m) => m[1]).slice(0, 10);
179
- return { lang: "Ruby", dependencies: gems };
180
- } catch {
181
- return null;
182
- }
183
- }
184
-
185
- /** Java/Kotlin: detect from pom.xml or build.gradle */
186
- async function detectJvm(workdir: string): Promise<{ name?: string; lang: string; dependencies: string[] } | null> {
187
- const pom = join(workdir, "pom.xml");
188
- const gradle = join(workdir, "build.gradle");
189
- const gradleKts = join(workdir, "build.gradle.kts");
190
-
191
- if (!existsSync(pom) && !existsSync(gradle) && !existsSync(gradleKts)) return null;
192
-
193
- try {
194
- if (existsSync(pom)) {
195
- const content = await Bun.file(pom).text();
196
- const nameMatch = content.match(/<artifactId>([^<]+)<\/artifactId>/);
197
- const deps = [...content.matchAll(/<artifactId>([^<]+)<\/artifactId>/g)]
198
- .map((m) => m[1])
199
- .filter((d) => d !== nameMatch?.[1])
200
- .slice(0, 10);
201
- const lang = existsSync(join(workdir, "src/main/kotlin")) ? "Kotlin" : "Java";
202
- return { name: nameMatch?.[1], lang, dependencies: deps };
203
- }
204
-
205
- const gradleFile = existsSync(gradleKts) ? gradleKts : gradle;
206
- const content = await Bun.file(gradleFile).text();
207
- const lang = gradleFile.endsWith(".kts") ? "Kotlin" : "Java";
208
- const deps = [...content.matchAll(/implementation[^'"]*['"]([^:'"]+:[^:'"]+)[^'"]*['"]/g)]
209
- .map((m) => m[1].split(":").pop() ?? m[1])
210
- .slice(0, 10);
211
- return { lang, dependencies: deps };
212
- } catch {
213
- return null;
214
- }
215
- }
216
-
217
- // ─── Main ────────────────────────────────────────────────────────────────────
218
-
219
- /**
220
- * Detect project language and build metadata.
221
- * Runs all detectors; first match wins (Node checked last to avoid false positives in polyglot repos).
222
- */
223
- export async function buildProjectMetadata(workdir: string, config: NaxConfig): Promise<ProjectMetadata> {
224
- // Priority: Go > Rust > Python > PHP > Ruby > JVM > Node
225
- const detected =
226
- (await detectGo(workdir)) ??
227
- (await detectRust(workdir)) ??
228
- (await detectPython(workdir)) ??
229
- (await detectPhp(workdir)) ??
230
- (await detectRuby(workdir)) ??
231
- (await detectJvm(workdir)) ??
232
- (await detectNode(workdir));
233
-
234
- return {
235
- name: detected?.name,
236
- language: detected?.lang,
237
- dependencies: detected?.dependencies ?? [],
238
- testCommand: config.execution?.testCommand ?? undefined,
239
- lintCommand: config.execution?.lintCommand ?? undefined,
240
- typecheckCommand: config.execution?.typecheckCommand ?? undefined,
241
- };
242
- }
243
-
244
- /**
245
- * Format metadata as a markdown section for injection into agent configs.
246
- */
247
- export function formatMetadataSection(metadata: ProjectMetadata): string {
248
- const lines: string[] = ["## Project Metadata", "", "> Auto-injected by `nax generate`", ""];
249
-
250
- if (metadata.name) {
251
- lines.push(`**Project:** \`${metadata.name}\``);
252
- lines.push("");
253
- }
254
-
255
- if (metadata.language) {
256
- lines.push(`**Language:** ${metadata.language}`);
257
- lines.push("");
258
- }
259
-
260
- if (metadata.dependencies.length > 0) {
261
- lines.push(`**Key dependencies:** ${metadata.dependencies.join(", ")}`);
262
- lines.push("");
263
- }
264
-
265
- const commands: string[] = [];
266
- if (metadata.testCommand) commands.push(`test: \`${metadata.testCommand}\``);
267
- if (metadata.lintCommand) commands.push(`lint: \`${metadata.lintCommand}\``);
268
- if (metadata.typecheckCommand) commands.push(`typecheck: \`${metadata.typecheckCommand}\``);
269
-
270
- if (commands.length > 0) {
271
- lines.push(`**Commands:** ${commands.join(" | ")}`);
272
- lines.push("");
273
- }
274
-
275
- lines.push("---");
276
- lines.push("");
277
-
278
- return lines.join("\n");
279
- }
@@ -1,39 +0,0 @@
1
- /**
2
- * Parent output file resolution for context chaining (ENH-005).
3
- *
4
- * When a story has dependencies, its parent stories' outputFiles are injected
5
- * as additional contextFiles so agents have targeted context from prior work.
6
- */
7
-
8
- import type { UserStory } from "../prd/types";
9
-
10
- const MAX_PARENT_FILES = 10;
11
-
12
- const NOISE_PATTERNS = [
13
- /\.test\.(ts|js|tsx|jsx)$/,
14
- /\.spec\.(ts|js|tsx|jsx)$/,
15
- /package-lock\.json$/,
16
- /bun\.lockb?$/,
17
- /\.gitignore$/,
18
- /^nax\//,
19
- ];
20
-
21
- /**
22
- * Get output files from direct parent stories (dependencies[]).
23
- * Only direct parents — no transitive resolution (keep simple, extend later).
24
- * Returns deduped list, filtered of noise, capped at MAX_PARENT_FILES.
25
- */
26
- export function getParentOutputFiles(story: UserStory, allStories: UserStory[]): string[] {
27
- if (!story.dependencies || story.dependencies.length === 0) return [];
28
-
29
- const parentFiles: string[] = [];
30
- for (const depId of story.dependencies) {
31
- const parent = allStories.find((s) => s.id === depId);
32
- if (parent?.outputFiles) {
33
- parentFiles.push(...parent.outputFiles);
34
- }
35
- }
36
-
37
- const unique = [...new Set(parentFiles)];
38
- return unique.filter((f) => !NOISE_PATTERNS.some((p) => p.test(f))).slice(0, MAX_PARENT_FILES);
39
- }