@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,206 +0,0 @@
1
- /**
2
- * Plugin System Types — Hub file
3
- *
4
- * Defines the plugin interface and extension point types for the nax plugin system.
5
- * Plugins export a NaxPlugin object with extension implementations.
6
- */
7
-
8
- import type { AgentAdapter } from "../agents/types";
9
- import type { IPromptOptimizer } from "../optimizer/types";
10
- import type { RoutingStrategy } from "../routing/strategy";
11
- import type { IContextProvider, IReporter, IReviewPlugin } from "./extensions";
12
-
13
- // Re-export extension types
14
- export type {
15
- ContextProviderResult,
16
- IContextProvider,
17
- IReporter,
18
- IReviewPlugin,
19
- ReviewCheckResult,
20
- ReviewFinding,
21
- RunEndEvent,
22
- RunStartEvent,
23
- StoryCompleteEvent,
24
- } from "./extensions";
25
-
26
- /**
27
- * Extension point types that plugins can provide.
28
- */
29
- export type PluginType = "optimizer" | "router" | "agent" | "reviewer" | "context-provider" | "reporter";
30
-
31
- /**
32
- * A nax plugin module.
33
- *
34
- * Plugins export a single NaxPlugin object (default or named export).
35
- * Each plugin declares which extension points it provides and supplies
36
- * the corresponding implementations.
37
- *
38
- * @example
39
- * ```ts
40
- * const myPlugin: NaxPlugin = {
41
- * name: "my-security-scanner",
42
- * version: "1.0.0",
43
- * provides: ["reviewer"],
44
- * async setup(config) {
45
- * // Initialize plugin
46
- * },
47
- * async teardown() {
48
- * // Cleanup
49
- * },
50
- * extensions: {
51
- * reviewer: {
52
- * name: "security-scan",
53
- * description: "Scans for security vulnerabilities",
54
- * async check(workdir, changedFiles) {
55
- * // Perform check
56
- * }
57
- * }
58
- * }
59
- * };
60
- * ```
61
- */
62
- export interface NaxPlugin {
63
- /** Unique plugin name (e.g., "jira-context", "llmlingua-optimizer") */
64
- name: string;
65
-
66
- /** Plugin version (semver) */
67
- version: string;
68
-
69
- /** Which extension points this plugin provides */
70
- provides: PluginType[];
71
-
72
- /**
73
- * Called once when plugin is loaded. Use for initialization,
74
- * validating config, establishing connections, etc.
75
- *
76
- * @param config - Plugin-specific config from nax config.json
77
- * @param logger - Write-only logger scoped to this plugin (stage auto-prefixed as `plugin:<name>`)
78
- */
79
- setup?(config: Record<string, unknown>, logger: PluginLogger): Promise<void>;
80
-
81
- /**
82
- * Called when the nax run ends (success or failure).
83
- * Use for cleanup, closing connections, flushing buffers.
84
- */
85
- teardown?(): Promise<void>;
86
-
87
- /**
88
- * Extension implementations. Only the types listed in `provides`
89
- * are required; others are ignored.
90
- */
91
- extensions: PluginExtensions;
92
- }
93
-
94
- /**
95
- * Extension implementations provided by a plugin.
96
- * Only extensions matching the plugin's `provides` array are required.
97
- */
98
- export interface PluginExtensions {
99
- /** Custom prompt optimizer */
100
- optimizer?: IPromptOptimizer;
101
-
102
- /** Custom routing strategy (inserted into the strategy chain) */
103
- router?: RoutingStrategy;
104
-
105
- /** Custom agent adapter (e.g., Codex, Gemini, Aider) */
106
- agent?: AgentAdapter;
107
-
108
- /** Custom review check (runs alongside built-in typecheck/lint/test) */
109
- reviewer?: IReviewPlugin;
110
-
111
- /** Custom context provider (injects external context into prompts) */
112
- contextProvider?: IContextProvider;
113
-
114
- /** Custom reporter (receives run events for dashboards, CI, etc.) */
115
- reporter?: IReporter;
116
- }
117
-
118
- // ============================================================================
119
- // Optimizer Extension
120
- // ============================================================================
121
-
122
- /**
123
- * Re-export optimizer types from the optimizer module.
124
- * Plugin optimizers use the same interface as built-in optimizers.
125
- */
126
- export type {
127
- IPromptOptimizer,
128
- PromptOptimizerInput,
129
- PromptOptimizerResult,
130
- } from "../optimizer/types";
131
-
132
- // ============================================================================
133
- // Plugin Logger
134
- // ============================================================================
135
-
136
- /**
137
- * Write-only, level-gated logger provided to plugins via setup().
138
- *
139
- * All log entries are auto-prefixed with `plugin:<name>` as the stage,
140
- * so plugins cannot impersonate core nax stages. The interface is
141
- * intentionally minimal — plugins only need to emit messages, not
142
- * configure log levels or access log files.
143
- *
144
- * @example
145
- * ```ts
146
- * let log: PluginLogger;
147
- *
148
- * const myPlugin: NaxPlugin = {
149
- * name: "my-plugin",
150
- * version: "1.0.0",
151
- * provides: ["reviewer"],
152
- * async setup(config, logger) {
153
- * log = logger;
154
- * log.info("Initialized with config", { keys: Object.keys(config) });
155
- * },
156
- * extensions: {
157
- * reviewer: {
158
- * name: "my-check",
159
- * description: "Custom check",
160
- * async check(workdir, changedFiles) {
161
- * log.debug("Scanning files", { count: changedFiles.length });
162
- * // ...
163
- * }
164
- * }
165
- * }
166
- * };
167
- * ```
168
- */
169
- export interface PluginLogger {
170
- /** Log an error message */
171
- error(message: string, data?: Record<string, unknown>): void;
172
- /** Log a warning message */
173
- warn(message: string, data?: Record<string, unknown>): void;
174
- /** Log an informational message */
175
- info(message: string, data?: Record<string, unknown>): void;
176
- /** Log a debug message */
177
- debug(message: string, data?: Record<string, unknown>): void;
178
- }
179
-
180
- // ============================================================================
181
- // Plugin Config
182
- // ============================================================================
183
-
184
- /**
185
- * Plugin configuration entry from nax config.json.
186
- *
187
- * @example
188
- * ```json
189
- * {
190
- * "plugins": [
191
- * {
192
- * "module": "./nax/plugins/my-plugin",
193
- * "config": { "apiKey": "secret" }
194
- * }
195
- * ]
196
- * }
197
- * ```
198
- */
199
- export interface PluginConfigEntry {
200
- /** Module path or npm package name */
201
- module: string;
202
- /** Plugin-specific configuration */
203
- config?: Record<string, unknown>;
204
- /** Whether this plugin is enabled (default: true) */
205
- enabled?: boolean;
206
- }
@@ -1,352 +0,0 @@
1
- /**
2
- * Plugin Validator
3
- *
4
- * Runtime type checking for plugin modules.
5
- * Validates plugin shape and ensures all required extensions are present.
6
- */
7
-
8
- import { getLogger } from "../logger";
9
- import type { NaxPlugin, PluginType } from "./types";
10
-
11
- /**
12
- * Safely get logger instance, returns null if not initialized
13
- */
14
- function getSafeLogger() {
15
- try {
16
- return getLogger();
17
- } catch {
18
- return null;
19
- }
20
- }
21
-
22
- const VALID_PLUGIN_TYPES: readonly PluginType[] = [
23
- "optimizer",
24
- "router",
25
- "agent",
26
- "reviewer",
27
- "context-provider",
28
- "reporter",
29
- ] as const;
30
-
31
- /**
32
- * Validate a plugin module at runtime.
33
- *
34
- * Returns the plugin if valid, null if invalid (with warning logged).
35
- *
36
- * @param module - The module to validate (can be any type)
37
- * @returns The validated plugin or null
38
- */
39
- export function validatePlugin(module: unknown): NaxPlugin | null {
40
- // Must be an object
41
- if (typeof module !== "object" || module === null) {
42
- getSafeLogger()?.warn("plugins", "Plugin validation failed: module is not an object");
43
- return null;
44
- }
45
-
46
- const plugin = module as Record<string, unknown>;
47
-
48
- // Validate name
49
- if (typeof plugin.name !== "string") {
50
- getSafeLogger()?.warn("plugins", "Plugin validation failed: missing or invalid 'name' (must be string)");
51
- return null;
52
- }
53
-
54
- // Validate version
55
- if (typeof plugin.version !== "string") {
56
- getSafeLogger()?.warn(
57
- "plugins",
58
- `Plugin '${plugin.name}' validation failed: missing or invalid 'version' (must be string)`,
59
- );
60
- return null;
61
- }
62
-
63
- // Validate provides
64
- if (!Array.isArray(plugin.provides)) {
65
- getSafeLogger()?.warn("plugins", `Plugin '${plugin.name}' validation failed: 'provides' must be an array`);
66
- return null;
67
- }
68
-
69
- if (plugin.provides.length === 0) {
70
- getSafeLogger()?.warn("plugins", `Plugin '${plugin.name}' validation failed: 'provides' must not be empty`);
71
- return null;
72
- }
73
-
74
- for (const type of plugin.provides) {
75
- if (!VALID_PLUGIN_TYPES.includes(type as PluginType)) {
76
- getSafeLogger()?.warn(
77
- "plugins",
78
- `Plugin '${plugin.name}' validation failed: invalid plugin type '${type}' in 'provides'`,
79
- );
80
- return null;
81
- }
82
- }
83
-
84
- // Validate setup (optional)
85
- if ("setup" in plugin && typeof plugin.setup !== "function") {
86
- getSafeLogger()?.warn("plugins", `Plugin '${plugin.name}' validation failed: 'setup' must be a function`);
87
- return null;
88
- }
89
-
90
- // Validate teardown (optional)
91
- if ("teardown" in plugin && typeof plugin.teardown !== "function") {
92
- getSafeLogger()?.warn("plugins", `Plugin '${plugin.name}' validation failed: 'teardown' must be a function`);
93
- return null;
94
- }
95
-
96
- // Validate extensions
97
- if (typeof plugin.extensions !== "object" || plugin.extensions === null) {
98
- getSafeLogger()?.warn("plugins", `Plugin '${plugin.name}' validation failed: 'extensions' must be an object`);
99
- return null;
100
- }
101
-
102
- const extensions = plugin.extensions as Record<string, unknown>;
103
-
104
- // Validate each extension type in provides
105
- for (const type of plugin.provides) {
106
- const isValid = validateExtension(plugin.name as string, type as PluginType, extensions);
107
- if (!isValid) {
108
- return null;
109
- }
110
- }
111
-
112
- return plugin as unknown as NaxPlugin;
113
- }
114
-
115
- /**
116
- * Validate a specific extension type.
117
- *
118
- * @param pluginName - Plugin name (for error messages)
119
- * @param type - Extension type to validate
120
- * @param extensions - Extensions object
121
- * @returns Whether the extension is valid
122
- */
123
- function validateExtension(pluginName: string, type: PluginType, extensions: Record<string, unknown>): boolean {
124
- switch (type) {
125
- case "optimizer":
126
- return validateOptimizer(pluginName, extensions.optimizer);
127
- case "router":
128
- return validateRouter(pluginName, extensions.router);
129
- case "agent":
130
- return validateAgent(pluginName, extensions.agent);
131
- case "reviewer":
132
- return validateReviewer(pluginName, extensions.reviewer);
133
- case "context-provider":
134
- return validateContextProvider(pluginName, extensions.contextProvider);
135
- case "reporter":
136
- return validateReporter(pluginName, extensions.reporter);
137
- default:
138
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: unknown extension type '${type}'`);
139
- return false;
140
- }
141
- }
142
-
143
- /**
144
- * Validate optimizer extension.
145
- */
146
- function validateOptimizer(pluginName: string, optimizer: unknown): boolean {
147
- if (typeof optimizer !== "object" || optimizer === null) {
148
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: optimizer extension must be an object`);
149
- return false;
150
- }
151
-
152
- const opt = optimizer as Record<string, unknown>;
153
-
154
- if (typeof opt.name !== "string") {
155
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: optimizer.name must be a string`);
156
- return false;
157
- }
158
-
159
- if (typeof opt.optimize !== "function") {
160
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: optimizer.optimize must be a function`);
161
- return false;
162
- }
163
-
164
- return true;
165
- }
166
-
167
- /**
168
- * Validate router extension.
169
- */
170
- function validateRouter(pluginName: string, router: unknown): boolean {
171
- if (typeof router !== "object" || router === null) {
172
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: router extension must be an object`);
173
- return false;
174
- }
175
-
176
- const rtr = router as Record<string, unknown>;
177
-
178
- if (typeof rtr.name !== "string") {
179
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: router.name must be a string`);
180
- return false;
181
- }
182
-
183
- if (typeof rtr.route !== "function") {
184
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: router.route must be a function`);
185
- return false;
186
- }
187
-
188
- return true;
189
- }
190
-
191
- /**
192
- * Validate agent extension.
193
- */
194
- function validateAgent(pluginName: string, agent: unknown): boolean {
195
- if (typeof agent !== "object" || agent === null) {
196
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: agent extension must be an object`);
197
- return false;
198
- }
199
-
200
- const agt = agent as Record<string, unknown>;
201
-
202
- const requiredFields = [
203
- { name: "name", type: "string" },
204
- { name: "displayName", type: "string" },
205
- { name: "binary", type: "string" },
206
- { name: "capabilities", type: "object" },
207
- { name: "isInstalled", type: "function" },
208
- { name: "run", type: "function" },
209
- { name: "buildCommand", type: "function" },
210
- { name: "plan", type: "function" },
211
- { name: "decompose", type: "function" },
212
- ];
213
-
214
- for (const field of requiredFields) {
215
- if (field.type === "object") {
216
- if (typeof agt[field.name] !== "object" || agt[field.name] === null) {
217
- getSafeLogger()?.warn(
218
- "plugins",
219
- `Plugin '${pluginName}' validation failed: agent.${field.name} must be an object`,
220
- );
221
- return false;
222
- }
223
- } else {
224
- // Validate field.type is a valid typeof result before comparison
225
- const expectedType = field.type as string;
226
-
227
- // Use explicit type checks instead of dynamic typeof comparison
228
- let isValid = false;
229
- if (expectedType === "string") {
230
- isValid = typeof agt[field.name] === "string";
231
- } else if (expectedType === "number") {
232
- isValid = typeof agt[field.name] === "number";
233
- } else if (expectedType === "boolean") {
234
- isValid = typeof agt[field.name] === "boolean";
235
- } else if (expectedType === "symbol") {
236
- isValid = typeof agt[field.name] === "symbol";
237
- } else if (expectedType === "undefined") {
238
- isValid = typeof agt[field.name] === "undefined";
239
- } else if (expectedType === "function") {
240
- isValid = typeof agt[field.name] === "function";
241
- } else if (expectedType === "bigint") {
242
- isValid = typeof agt[field.name] === "bigint";
243
- } else {
244
- getSafeLogger()?.warn(
245
- "plugins",
246
- `Plugin '${pluginName}' validation failed: invalid type constraint '${expectedType}'`,
247
- );
248
- return false;
249
- }
250
-
251
- if (!isValid) {
252
- getSafeLogger()?.warn(
253
- "plugins",
254
- `Plugin '${pluginName}' validation failed: agent.${field.name} must be a ${expectedType}`,
255
- );
256
- return false;
257
- }
258
- }
259
- }
260
-
261
- return true;
262
- }
263
-
264
- /**
265
- * Validate reviewer extension.
266
- */
267
- function validateReviewer(pluginName: string, reviewer: unknown): boolean {
268
- if (typeof reviewer !== "object" || reviewer === null) {
269
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reviewer extension must be an object`);
270
- return false;
271
- }
272
-
273
- const rev = reviewer as Record<string, unknown>;
274
-
275
- if (typeof rev.name !== "string") {
276
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reviewer.name must be a string`);
277
- return false;
278
- }
279
-
280
- if (typeof rev.description !== "string") {
281
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reviewer.description must be a string`);
282
- return false;
283
- }
284
-
285
- if (typeof rev.check !== "function") {
286
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reviewer.check must be a function`);
287
- return false;
288
- }
289
-
290
- return true;
291
- }
292
-
293
- /**
294
- * Validate context-provider extension.
295
- */
296
- function validateContextProvider(pluginName: string, provider: unknown): boolean {
297
- if (typeof provider !== "object" || provider === null) {
298
- getSafeLogger()?.warn(
299
- "plugins",
300
- `Plugin '${pluginName}' validation failed: contextProvider extension must be an object`,
301
- );
302
- return false;
303
- }
304
-
305
- const prov = provider as Record<string, unknown>;
306
-
307
- if (typeof prov.name !== "string") {
308
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: contextProvider.name must be a string`);
309
- return false;
310
- }
311
-
312
- if (typeof prov.getContext !== "function") {
313
- getSafeLogger()?.warn(
314
- "plugins",
315
- `Plugin '${pluginName}' validation failed: contextProvider.getContext must be a function`,
316
- );
317
- return false;
318
- }
319
-
320
- return true;
321
- }
322
-
323
- /**
324
- * Validate reporter extension.
325
- */
326
- function validateReporter(pluginName: string, reporter: unknown): boolean {
327
- if (typeof reporter !== "object" || reporter === null) {
328
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reporter extension must be an object`);
329
- return false;
330
- }
331
-
332
- const rep = reporter as Record<string, unknown>;
333
-
334
- if (typeof rep.name !== "string") {
335
- getSafeLogger()?.warn("plugins", `Plugin '${pluginName}' validation failed: reporter.name must be a string`);
336
- return false;
337
- }
338
-
339
- // At least one event handler is optional, but all must be functions if present
340
- const eventHandlers = ["onRunStart", "onStoryComplete", "onRunEnd"];
341
- for (const handler of eventHandlers) {
342
- if (handler in rep && typeof rep[handler] !== "function") {
343
- getSafeLogger()?.warn(
344
- "plugins",
345
- `Plugin '${pluginName}' validation failed: reporter.${handler} must be a function`,
346
- );
347
- return false;
348
- }
349
- }
350
-
351
- return true;
352
- }