@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,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
- };