@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,107 +0,0 @@
1
- /**
2
- * Context Markdown Formatter
3
- *
4
- * Extracted from builder.ts: formats built context as markdown for agent consumption.
5
- */
6
-
7
- import type { BuiltContext, ContextElement } from "./types";
8
-
9
- /**
10
- * Format built context as markdown for agent consumption.
11
- *
12
- * Generates markdown with sections for progress, prior errors,
13
- * test coverage, current story, dependency stories, and relevant files.
14
- */
15
- export function formatContextAsMarkdown(built: BuiltContext): string {
16
- const sections: string[] = [];
17
-
18
- sections.push("# Story Context\n");
19
- sections.push(`${built.summary}\n`);
20
-
21
- // Group by type
22
- const byType = new Map<string, ContextElement[]>();
23
- for (const element of built.elements) {
24
- const existing = byType.get(element.type) || [];
25
- existing.push(element);
26
- byType.set(element.type, existing);
27
- }
28
-
29
- renderSection(sections, byType, "progress", "## Progress\n", renderSimple);
30
- renderErrorSection(sections, byType);
31
- renderSection(sections, byType, "test-coverage", "", renderSimple);
32
- renderSection(sections, byType, "story", "## Current Story\n", renderSimple);
33
- renderSection(sections, byType, "dependency", "## Dependency Stories\n", renderSimple);
34
- renderSection(sections, byType, "file", "## Relevant Source Files\n", renderSimple);
35
-
36
- return sections.join("\n");
37
- }
38
-
39
- /** Render a simple section with elements as-is. */
40
- function renderSimple(sections: string[], elements: ContextElement[]): void {
41
- for (const element of elements) {
42
- sections.push(element.content);
43
- sections.push("\n");
44
- }
45
- }
46
-
47
- /** Render a typed section if elements exist. */
48
- function renderSection(
49
- sections: string[],
50
- byType: Map<string, ContextElement[]>,
51
- type: string,
52
- header: string,
53
- renderer: (sections: string[], elements: ContextElement[]) => void,
54
- ): void {
55
- const elements = byType.get(type);
56
- if (!elements || elements.length === 0) return;
57
- if (header) sections.push(header);
58
- renderer(sections, elements);
59
- }
60
-
61
- /** Render error section with special handling for ASSET_CHECK errors. */
62
- function renderErrorSection(sections: string[], byType: Map<string, ContextElement[]>): void {
63
- const errorElements = byType.get("error");
64
- if (!errorElements || errorElements.length === 0) return;
65
-
66
- const assetCheckErrors: ContextElement[] = [];
67
- const otherErrors: ContextElement[] = [];
68
-
69
- for (const element of errorElements) {
70
- if (element.content.startsWith("ASSET_CHECK_FAILED:")) {
71
- assetCheckErrors.push(element);
72
- } else {
73
- otherErrors.push(element);
74
- }
75
- }
76
-
77
- if (assetCheckErrors.length > 0) {
78
- sections.push("## ⚠️ MANDATORY: Missing Files from Previous Attempts\n");
79
- sections.push("**CRITICAL:** Previous attempts failed because these files were not created.\n");
80
- sections.push("You MUST create these exact files. Do NOT use alternative filenames.\n\n");
81
-
82
- for (const element of assetCheckErrors) {
83
- const match = element.content.match(/Missing files: \[([^\]]+)\]/);
84
- if (match) {
85
- const fileList = match[1].split(",").map((f) => f.trim());
86
- sections.push("**Required files:**\n");
87
- for (const file of fileList) {
88
- sections.push(`- \`${file}\``);
89
- }
90
- sections.push("\n");
91
- } else {
92
- sections.push("```");
93
- sections.push(element.content);
94
- sections.push("```\n");
95
- }
96
- }
97
- }
98
-
99
- if (otherErrors.length > 0) {
100
- sections.push("## Prior Errors\n");
101
- for (const element of otherErrors) {
102
- sections.push("```");
103
- sections.push(element.content);
104
- sections.push("```\n");
105
- }
106
- }
107
- }
@@ -1,343 +0,0 @@
1
- /**
2
- * Context Generator Orchestrator (v0.16.1)
3
- *
4
- * Generates agent-specific config files from nax/context.md + auto-injected metadata.
5
- * Replaces the old constitution generator.
6
- */
7
-
8
- import { existsSync, readFileSync } from "node:fs";
9
- import { join } from "node:path";
10
- import type { NaxConfig } from "../config";
11
- import { validateFilePath } from "../config/path-security";
12
- import { aiderGenerator } from "./generators/aider";
13
- import { claudeGenerator } from "./generators/claude";
14
- import { codexGenerator } from "./generators/codex";
15
- import { cursorGenerator } from "./generators/cursor";
16
- import { geminiGenerator } from "./generators/gemini";
17
- import { opencodeGenerator } from "./generators/opencode";
18
- import { windsurfGenerator } from "./generators/windsurf";
19
- import { buildProjectMetadata } from "./injector";
20
- import type { AgentContextGenerator, AgentType, ContextContent, GeneratorMap } from "./types";
21
-
22
- /** Generator registry */
23
- const GENERATORS: GeneratorMap = {
24
- claude: claudeGenerator,
25
- codex: codexGenerator,
26
- opencode: opencodeGenerator,
27
- cursor: cursorGenerator,
28
- windsurf: windsurfGenerator,
29
- aider: aiderGenerator,
30
- gemini: geminiGenerator,
31
- };
32
-
33
- /** Generation result for a single agent */
34
- export interface GenerationResult {
35
- agent: AgentType;
36
- outputFile: string;
37
- content: string;
38
- written: boolean;
39
- error?: string;
40
- }
41
-
42
- /** Generate options */
43
- export interface GenerateOptions {
44
- /** Path to nax/context.md (default: <workdir>/nax/context.md) */
45
- contextPath: string;
46
- /** Output directory (default: project root) */
47
- outputDir: string;
48
- /** Working directory for metadata injection */
49
- workdir: string;
50
- /** Dry run mode */
51
- dryRun?: boolean;
52
- /** Specific agent (default: all) */
53
- agent?: AgentType;
54
- /** Auto-inject project metadata (default: true) */
55
- autoInject?: boolean;
56
- }
57
-
58
- /**
59
- * Load context content and optionally inject project metadata.
60
- */
61
- async function loadContextContent(options: GenerateOptions, config: NaxConfig): Promise<ContextContent> {
62
- if (!existsSync(options.contextPath)) {
63
- throw new Error(`Context file not found: ${options.contextPath}`);
64
- }
65
-
66
- const file = Bun.file(options.contextPath);
67
- const markdown = await file.text();
68
-
69
- const autoInject = options.autoInject ?? true;
70
- const metadata = autoInject ? await buildProjectMetadata(options.workdir, config) : undefined;
71
-
72
- return { markdown, metadata };
73
- }
74
-
75
- /**
76
- * Generate config for a specific agent.
77
- */
78
- async function generateFor(agent: AgentType, options: GenerateOptions, config: NaxConfig): Promise<GenerationResult> {
79
- const generator = GENERATORS[agent];
80
- if (!generator) {
81
- return { agent, outputFile: "", content: "", written: false, error: `Unknown agent: ${agent}` };
82
- }
83
-
84
- try {
85
- const context = await loadContextContent(options, config);
86
- const content = generator.generate(context);
87
- const outputPath = join(options.outputDir, generator.outputFile);
88
-
89
- validateFilePath(outputPath, options.outputDir);
90
-
91
- if (!options.dryRun) {
92
- await Bun.write(outputPath, content);
93
- }
94
-
95
- return { agent, outputFile: generator.outputFile, content, written: !options.dryRun };
96
- } catch (err) {
97
- const error = err instanceof Error ? err.message : String(err);
98
- return { agent, outputFile: generator.outputFile, content: "", written: false, error };
99
- }
100
- }
101
-
102
- /**
103
- * Generate configs for all agents (or a filtered subset).
104
- *
105
- * @param agentFilter - Optional list of agent names to generate. When provided,
106
- * only those agents are written to disk. When omitted, all agents are generated.
107
- */
108
- async function generateAll(
109
- options: GenerateOptions,
110
- config: NaxConfig,
111
- agentFilter?: AgentType[],
112
- ): Promise<GenerationResult[]> {
113
- // Load context once and share across generators
114
- const context = await loadContextContent(options, config);
115
-
116
- const results: GenerationResult[] = [];
117
-
118
- const entries = (Object.entries(GENERATORS) as [AgentType, AgentContextGenerator][]).filter(
119
- ([agentKey]) => !agentFilter || agentFilter.length === 0 || agentFilter.includes(agentKey),
120
- );
121
-
122
- for (const [agentKey, generator] of entries) {
123
- try {
124
- const content = generator.generate(context);
125
- const outputPath = join(options.outputDir, generator.outputFile);
126
-
127
- validateFilePath(outputPath, options.outputDir);
128
-
129
- if (!options.dryRun) {
130
- await Bun.write(outputPath, content);
131
- }
132
-
133
- results.push({ agent: agentKey, outputFile: generator.outputFile, content, written: !options.dryRun });
134
- } catch (err) {
135
- const error = err instanceof Error ? err.message : String(err);
136
- results.push({ agent: agentKey, outputFile: generator.outputFile, content: "", written: false, error });
137
- }
138
- }
139
-
140
- return results;
141
- }
142
-
143
- /** Result from generateForPackage */
144
- export interface PackageGenerationResult {
145
- packageDir: string;
146
- outputFile: string;
147
- content: string;
148
- written: boolean;
149
- error?: string;
150
- }
151
-
152
- /**
153
- * Discover packages that have nax/context.md files.
154
- *
155
- * Scans up to 2 levels deep (one-level and two-level patterns).
156
- *
157
- * @param repoRoot - Absolute repo root to scan
158
- * @returns Array of package directory paths (absolute)
159
- */
160
- export async function discoverPackages(repoRoot: string): Promise<string[]> {
161
- const packages: string[] = [];
162
- const seen = new Set<string>();
163
-
164
- for (const pattern of ["*/nax/context.md", "*/*/nax/context.md"]) {
165
- const glob = new Bun.Glob(pattern);
166
- for await (const match of glob.scan(repoRoot)) {
167
- // match is e.g. "packages/api/nax/context.md" — strip trailing /nax/context.md
168
- const pkgRelative = match.replace(/\/nax\/context\.md$/, "");
169
- const pkgAbsolute = join(repoRoot, pkgRelative);
170
- if (!seen.has(pkgAbsolute)) {
171
- seen.add(pkgAbsolute);
172
- packages.push(pkgAbsolute);
173
- }
174
- }
175
- }
176
-
177
- return packages;
178
- }
179
-
180
- /**
181
- * Discover packages from workspace manifests (turbo.json, package.json workspaces,
182
- * pnpm-workspace.yaml). Used as fallback when no nax/context.md files exist yet.
183
- *
184
- * Returns relative paths (e.g. "packages/api") sorted alphabetically.
185
- */
186
- export async function discoverWorkspacePackages(repoRoot: string): Promise<string[]> {
187
- // 1. Prefer packages that already have nax/context.md
188
- const existing = await discoverPackages(repoRoot);
189
- if (existing.length > 0) {
190
- return existing.map((p) => p.replace(`${repoRoot}/`, ""));
191
- }
192
-
193
- const seen = new Set<string>();
194
- const results: string[] = [];
195
-
196
- async function resolveGlobs(patterns: string[]): Promise<void> {
197
- for (const pattern of patterns) {
198
- if (pattern.startsWith("!")) continue; // skip negations
199
- // Convert workspace pattern to package.json glob so Bun can scan files
200
- // "packages/*" → "packages/*/package.json"
201
- // "packages/**" → "packages/**/package.json"
202
- const base = pattern.replace(/\/+$/, ""); // strip trailing slashes
203
- const pkgPattern = base.endsWith("*") ? `${base}/package.json` : `${base}/*/package.json`;
204
-
205
- const g = new Bun.Glob(pkgPattern);
206
- for await (const match of g.scan(repoRoot)) {
207
- const rel = match.replace(/\/package\.json$/, "");
208
- if (!seen.has(rel)) {
209
- seen.add(rel);
210
- results.push(rel);
211
- }
212
- }
213
- }
214
- }
215
-
216
- // 2. turbo v2+: turbo.json top-level "packages" array
217
- const turboPath = join(repoRoot, "turbo.json");
218
- if (existsSync(turboPath)) {
219
- try {
220
- const turbo = JSON.parse(readFileSync(turboPath, "utf-8")) as Record<string, unknown>;
221
- if (Array.isArray(turbo.packages)) {
222
- await resolveGlobs(turbo.packages as string[]);
223
- }
224
- } catch {
225
- // malformed turbo.json — skip
226
- }
227
- }
228
-
229
- // 3. root package.json "workspaces" (npm/yarn/bun/turbo v1)
230
- const pkgPath = join(repoRoot, "package.json");
231
- if (existsSync(pkgPath)) {
232
- try {
233
- const pkg = JSON.parse(readFileSync(pkgPath, "utf-8")) as Record<string, unknown>;
234
- const ws = pkg.workspaces;
235
- const patterns: string[] = Array.isArray(ws)
236
- ? (ws as string[])
237
- : Array.isArray((ws as Record<string, unknown>)?.packages)
238
- ? ((ws as Record<string, unknown>).packages as string[])
239
- : [];
240
- if (patterns.length > 0) await resolveGlobs(patterns);
241
- } catch {
242
- // malformed package.json — skip
243
- }
244
- }
245
-
246
- // 4. pnpm-workspace.yaml
247
- const pnpmPath = join(repoRoot, "pnpm-workspace.yaml");
248
- if (existsSync(pnpmPath)) {
249
- try {
250
- const raw = readFileSync(pnpmPath, "utf-8");
251
- // Simple YAML parse for "packages:\n - 'packages/*'" without full YAML dep
252
- const lines = raw.split("\n");
253
- let inPackages = false;
254
- const patterns: string[] = [];
255
- for (const line of lines) {
256
- if (/^packages\s*:/.test(line)) {
257
- inPackages = true;
258
- continue;
259
- }
260
- if (inPackages && /^\s+-\s+/.test(line)) {
261
- patterns.push(line.replace(/^\s+-\s+['"]?/, "").replace(/['"]?\s*$/, ""));
262
- } else if (inPackages && !/^\s/.test(line)) {
263
- break;
264
- }
265
- }
266
- if (patterns.length > 0) await resolveGlobs(patterns);
267
- } catch {
268
- // malformed yaml — skip
269
- }
270
- }
271
-
272
- return results.sort();
273
- }
274
-
275
- /**
276
- * Generate agent config file(s) for a specific package.
277
- *
278
- * Reads `<packageDir>/nax/context.md` and writes agent files (e.g. CLAUDE.md,
279
- * AGENTS.md) into the package directory. Respects `config.generate.agents` — when
280
- * set, only generates for those agents; defaults to `["claude"]` when unset.
281
- *
282
- * Per-package files contain only package-specific content — Claude Code's native
283
- * directory hierarchy merges root CLAUDE.md + package CLAUDE.md at runtime.
284
- *
285
- * Returns one result per generated agent.
286
- */
287
- export async function generateForPackage(
288
- packageDir: string,
289
- config: NaxConfig,
290
- dryRun = false,
291
- ): Promise<PackageGenerationResult[]> {
292
- const contextPath = join(packageDir, "nax", "context.md");
293
-
294
- if (!existsSync(contextPath)) {
295
- return [
296
- {
297
- packageDir,
298
- outputFile: "CLAUDE.md",
299
- content: "",
300
- written: false,
301
- error: `context.md not found: ${contextPath}`,
302
- },
303
- ];
304
- }
305
-
306
- // Respect config.generate.agents; default to ["claude"] when unset
307
- const agentsToGenerate: AgentType[] =
308
- config?.generate?.agents && config.generate.agents.length > 0
309
- ? (config.generate.agents as AgentType[])
310
- : ["claude"];
311
-
312
- const options: GenerateOptions = {
313
- contextPath,
314
- outputDir: packageDir,
315
- workdir: packageDir,
316
- dryRun,
317
- autoInject: true,
318
- };
319
-
320
- const results: PackageGenerationResult[] = [];
321
-
322
- for (const agent of agentsToGenerate) {
323
- try {
324
- const result = await generateFor(agent, options, config);
325
- results.push({
326
- packageDir,
327
- outputFile: result.outputFile,
328
- content: result.content,
329
- written: result.written,
330
- error: result.error,
331
- });
332
- } catch (err) {
333
- const error = err instanceof Error ? err.message : String(err);
334
- const fallbackFile = GENERATORS[agent]?.outputFile ?? `${agent}.md`;
335
- results.push({ packageDir, outputFile: fallbackFile, content: "", written: false, error });
336
- }
337
- }
338
-
339
- return results;
340
- }
341
-
342
- export { generateFor, generateAll };
343
- export type { AgentType };
@@ -1,34 +0,0 @@
1
- /**
2
- * Aider Config Generator (v0.16.1)
3
- *
4
- * Generates .aider.conf.yml from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateAiderConfig(context: ContextContent): string {
11
- const header = `# Aider Configuration
12
- # Auto-generated from nax/context.md — run \`nax generate\` to regenerate.
13
- # DO NOT EDIT MANUALLY
14
-
15
- # Project instructions
16
- instructions: |
17
- `;
18
-
19
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
20
- const combined = metaSection + context.markdown;
21
-
22
- const indented = combined
23
- .split("\n")
24
- .map((line) => ` ${line}`)
25
- .join("\n");
26
-
27
- return `${header}${indented}\n`;
28
- }
29
-
30
- export const aiderGenerator: AgentContextGenerator = {
31
- name: "aider",
32
- outputFile: ".aider.conf.yml",
33
- generate: generateAiderConfig,
34
- };
@@ -1,28 +0,0 @@
1
- /**
2
- * Claude Code Config Generator (v0.16.1)
3
- *
4
- * Generates CLAUDE.md from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateClaudeConfig(context: ContextContent): string {
11
- const header = `# Project Context
12
-
13
- This file is auto-generated from \`nax/context.md\`.
14
- DO NOT EDIT MANUALLY — run \`nax generate\` to regenerate.
15
-
16
- ---
17
-
18
- `;
19
-
20
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
21
- return header + metaSection + context.markdown;
22
- }
23
-
24
- export const claudeGenerator: AgentContextGenerator = {
25
- name: "claude",
26
- outputFile: "CLAUDE.md",
27
- generate: generateClaudeConfig,
28
- };
@@ -1,28 +0,0 @@
1
- /**
2
- * Codex Config Generator (v0.16.1)
3
- *
4
- * Generates codex.md from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateCodexConfig(context: ContextContent): string {
11
- const header = `# Codex Instructions
12
-
13
- This file is auto-generated from \`nax/context.md\`.
14
- DO NOT EDIT MANUALLY — run \`nax generate\` to regenerate.
15
-
16
- ---
17
-
18
- `;
19
-
20
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
21
- return header + metaSection + context.markdown;
22
- }
23
-
24
- export const codexGenerator: AgentContextGenerator = {
25
- name: "codex",
26
- outputFile: "codex.md",
27
- generate: generateCodexConfig,
28
- };
@@ -1,28 +0,0 @@
1
- /**
2
- * Cursor Rules Generator (v0.16.1)
3
- *
4
- * Generates .cursorrules from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateCursorRules(context: ContextContent): string {
11
- const header = `# Project Rules
12
-
13
- Auto-generated from nax/context.md — run \`nax generate\` to regenerate.
14
- DO NOT EDIT MANUALLY
15
-
16
- ---
17
-
18
- `;
19
-
20
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
21
- return header + metaSection + context.markdown;
22
- }
23
-
24
- export const cursorGenerator: AgentContextGenerator = {
25
- name: "cursor",
26
- outputFile: ".cursorrules",
27
- generate: generateCursorRules,
28
- };
@@ -1,28 +0,0 @@
1
- /**
2
- * Gemini CLI Config Generator (v0.16.1)
3
- *
4
- * Generates GEMINI.md from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateGeminiConfig(context: ContextContent): string {
11
- const header = `# Gemini CLI Context
12
-
13
- This file is auto-generated from \`nax/context.md\`.
14
- DO NOT EDIT MANUALLY — run \`nax generate\` to regenerate.
15
-
16
- ---
17
-
18
- `;
19
-
20
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
21
- return header + metaSection + context.markdown;
22
- }
23
-
24
- export const geminiGenerator: AgentContextGenerator = {
25
- name: "gemini",
26
- outputFile: "GEMINI.md",
27
- generate: generateGeminiConfig,
28
- };
@@ -1,30 +0,0 @@
1
- /**
2
- * OpenCode Config Generator (v0.16.1)
3
- *
4
- * Generates AGENTS.md from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateOpencodeConfig(context: ContextContent): string {
11
- const header = `# Agent Instructions
12
-
13
- This file is auto-generated from \`nax/context.md\`.
14
- DO NOT EDIT MANUALLY — run \`nax generate\` to regenerate.
15
-
16
- These instructions apply to all AI coding agents in this project.
17
-
18
- ---
19
-
20
- `;
21
-
22
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
23
- return header + metaSection + context.markdown;
24
- }
25
-
26
- export const opencodeGenerator: AgentContextGenerator = {
27
- name: "opencode",
28
- outputFile: "AGENTS.md",
29
- generate: generateOpencodeConfig,
30
- };
@@ -1,28 +0,0 @@
1
- /**
2
- * Windsurf Rules Generator (v0.16.1)
3
- *
4
- * Generates .windsurfrules from nax/context.md + auto-injected metadata.
5
- */
6
-
7
- import { formatMetadataSection } from "../injector";
8
- import type { AgentContextGenerator, ContextContent } from "../types";
9
-
10
- function generateWindsurfRules(context: ContextContent): string {
11
- const header = `# Windsurf Project Rules
12
-
13
- Auto-generated from nax/context.md — run \`nax generate\` to regenerate.
14
- DO NOT EDIT MANUALLY
15
-
16
- ---
17
-
18
- `;
19
-
20
- const metaSection = context.metadata ? formatMetadataSection(context.metadata) : "";
21
- return header + metaSection + context.markdown;
22
- }
23
-
24
- export const windsurfGenerator: AgentContextGenerator = {
25
- name: "windsurf",
26
- outputFile: ".windsurfrules",
27
- generate: generateWindsurfRules,
28
- };