@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,286 +0,0 @@
1
- /**
2
- * Webhook Interaction Plugin (v0.15.0 US-007)
3
- *
4
- * Send interaction requests via HTTP POST to configured URL.
5
- * Start local HTTP server to receive callbacks with HMAC verification.
6
- */
7
-
8
- import { createHmac, timingSafeEqual } from "node:crypto";
9
- import type { Server } from "node:http";
10
- import { z } from "zod";
11
- import type { InteractionPlugin, InteractionRequest, InteractionResponse } from "../types";
12
-
13
- /**
14
- * Injectable sleep — kept for backward compat with existing tests that override it.
15
- * No longer used internally by receive() (replaced by event-driven delivery).
16
- * @internal
17
- */
18
- export const _webhookPluginDeps = {
19
- sleep: (ms: number): Promise<void> => Bun.sleep(ms),
20
- };
21
-
22
- /** Webhook plugin configuration */
23
- interface WebhookConfig {
24
- /** Webhook URL to POST requests to */
25
- url?: string;
26
- /** Local callback port (default: 8765) */
27
- callbackPort?: number;
28
- /** HMAC secret for signature verification */
29
- secret?: string;
30
- /** Maximum payload size in bytes (default: 1MB) */
31
- maxPayloadBytes?: number;
32
- }
33
-
34
- /** Zod schema for validating webhook plugin config */
35
- const WebhookConfigSchema = z.object({
36
- url: z.string().url().optional(),
37
- callbackPort: z.number().int().min(1024).max(65535).optional(),
38
- secret: z.string().optional(),
39
- maxPayloadBytes: z.number().int().positive().optional(),
40
- });
41
-
42
- /** Zod schema for validating webhook callback payloads */
43
- const InteractionResponseSchema = z.object({
44
- requestId: z.string(),
45
- action: z.enum(["approve", "reject", "choose", "input", "skip", "abort"]),
46
- value: z.string().optional(),
47
- respondedBy: z.string().optional(),
48
- respondedAt: z.number(),
49
- });
50
-
51
- /**
52
- * Webhook plugin for HTTP-based interaction
53
- */
54
- export class WebhookInteractionPlugin implements InteractionPlugin {
55
- name = "webhook";
56
- private config: WebhookConfig = {};
57
- private server: Server | null = null;
58
- private serverStartPromise: Promise<void> | null = null;
59
- /** Legacy map for responses that arrive before receive() is called */
60
- private pendingResponses = new Map<string, InteractionResponse>();
61
- /** Event-driven callbacks: requestId → resolve fn (set by receive(), called by handleRequest) */
62
- private receiveCallbacks = new Map<string, (response: InteractionResponse) => void>();
63
-
64
- async init(config: Record<string, unknown>): Promise<void> {
65
- const cfg = WebhookConfigSchema.parse(config);
66
- this.config = {
67
- url: cfg.url,
68
- callbackPort: cfg.callbackPort ?? 8765,
69
- secret: cfg.secret,
70
- maxPayloadBytes: cfg.maxPayloadBytes ?? 1024 * 1024, // 1MB default
71
- };
72
- if (!this.config.url) {
73
- throw new Error("Webhook plugin requires 'url' config");
74
- }
75
- }
76
-
77
- async destroy(): Promise<void> {
78
- if (this.server) {
79
- await this.stopServer();
80
- }
81
- }
82
-
83
- async send(request: InteractionRequest): Promise<void> {
84
- if (!this.config.url) {
85
- throw new Error("Webhook plugin not initialized");
86
- }
87
-
88
- const payload = {
89
- ...request,
90
- callbackUrl: `http://localhost:${this.config.callbackPort}/nax/interact/${request.id}`,
91
- };
92
-
93
- const signature = this.config.secret ? this.sign(JSON.stringify(payload)) : undefined;
94
-
95
- const headers: Record<string, string> = {
96
- "Content-Type": "application/json",
97
- };
98
- if (signature) {
99
- headers["X-Nax-Signature"] = signature;
100
- }
101
-
102
- try {
103
- const response = await fetch(this.config.url, {
104
- method: "POST",
105
- headers,
106
- body: JSON.stringify(payload),
107
- });
108
-
109
- if (!response.ok) {
110
- const errorBody = await response.text().catch(() => "");
111
- throw new Error(`Webhook POST failed (${response.status}): ${errorBody || response.statusText}`);
112
- }
113
- } catch (err) {
114
- const msg = err instanceof Error ? err.message : String(err);
115
- throw new Error(`Failed to send webhook request: ${msg}`);
116
- }
117
- }
118
-
119
- async receive(requestId: string, timeout = 60000): Promise<InteractionResponse> {
120
- // Start HTTP server to receive callback
121
- await this.startServer();
122
-
123
- // Check if a response already arrived before receive() was called
124
- const early = this.pendingResponses.get(requestId);
125
- if (early) {
126
- this.pendingResponses.delete(requestId);
127
- return early;
128
- }
129
-
130
- // Event-driven: resolve immediately when handleRequest delivers the response
131
- return new Promise<InteractionResponse>((resolve) => {
132
- const timer = setTimeout(() => {
133
- this.receiveCallbacks.delete(requestId);
134
- resolve({
135
- requestId,
136
- action: "skip",
137
- respondedBy: "timeout",
138
- respondedAt: Date.now(),
139
- });
140
- }, timeout);
141
-
142
- this.receiveCallbacks.set(requestId, (response) => {
143
- clearTimeout(timer);
144
- this.receiveCallbacks.delete(requestId);
145
- resolve(response);
146
- });
147
- });
148
- }
149
-
150
- async cancel(requestId: string): Promise<void> {
151
- this.pendingResponses.delete(requestId);
152
- this.receiveCallbacks.delete(requestId);
153
- }
154
-
155
- /**
156
- * Deliver a response to a waiting receive() callback, or store for later pickup.
157
- */
158
- private deliverResponse(requestId: string, response: InteractionResponse): void {
159
- const cb = this.receiveCallbacks.get(requestId);
160
- if (cb) {
161
- cb(response);
162
- } else {
163
- // receive() hasn't been called yet — store for early-pickup path
164
- this.pendingResponses.set(requestId, response);
165
- }
166
- }
167
-
168
- /**
169
- * Start HTTP server for callbacks (with mutex to prevent race conditions)
170
- */
171
- private async startServer(): Promise<void> {
172
- if (this.server) return; // Already running
173
- if (this.serverStartPromise) {
174
- await this.serverStartPromise;
175
- return;
176
- }
177
- this.serverStartPromise = (async () => {
178
- const port = this.config.callbackPort ?? 8765;
179
- this.server = Bun.serve({
180
- port,
181
- fetch: (req) => this.handleRequest(req),
182
- }) as unknown as Server;
183
- })();
184
- await this.serverStartPromise;
185
- this.serverStartPromise = null;
186
- }
187
-
188
- /**
189
- * Stop HTTP server
190
- */
191
- private async stopServer(): Promise<void> {
192
- if (!this.server) return;
193
-
194
- // Bun.serve returns a server with stop() method
195
- const bunServer = this.server as unknown as { stop: () => void };
196
- bunServer.stop();
197
- this.server = null;
198
- this.serverStartPromise = null;
199
- }
200
-
201
- /**
202
- * Handle HTTP request
203
- */
204
- private async handleRequest(req: Request): Promise<Response> {
205
- const url = new URL(req.url);
206
-
207
- // Only accept POST to /nax/interact/:requestId
208
- if (req.method !== "POST" || !url.pathname.startsWith("/nax/interact/")) {
209
- return new Response("Not Found", { status: 404 });
210
- }
211
-
212
- const requestId = url.pathname.split("/").pop();
213
- if (!requestId) {
214
- return new Response("Bad Request", { status: 400 });
215
- }
216
-
217
- // Check content length before reading body
218
- const contentLength = req.headers.get("Content-Length");
219
- const maxBytes = this.config.maxPayloadBytes ?? 1024 * 1024;
220
- if (contentLength && Number.parseInt(contentLength, 10) > maxBytes) {
221
- return new Response("Payload Too Large", { status: 413 });
222
- }
223
-
224
- // Verify signature if secret is configured
225
- if (this.config.secret) {
226
- const signature = req.headers.get("X-Nax-Signature");
227
- const body = await req.text();
228
-
229
- // Check actual body size (in case Content-Length was missing)
230
- if (body.length > maxBytes) {
231
- return new Response("Payload Too Large", { status: 413 });
232
- }
233
-
234
- if (!signature || !this.verify(body, signature)) {
235
- return new Response("Unauthorized", { status: 401 });
236
- }
237
-
238
- // Parse and validate verified body
239
- try {
240
- const parsed = JSON.parse(body);
241
- const response = InteractionResponseSchema.parse(parsed);
242
- this.deliverResponse(requestId, response);
243
- } catch {
244
- // Sanitize error - do not leak parse/validation details
245
- return new Response("Bad Request: Invalid response format", { status: 400 });
246
- }
247
- } else {
248
- // No signature verification - still validate structure
249
- try {
250
- const parsed = await req.json();
251
- const response = InteractionResponseSchema.parse(parsed);
252
- this.deliverResponse(requestId, response);
253
- } catch {
254
- // Sanitize error - do not leak parse/validation details
255
- return new Response("Bad Request: Invalid response format", { status: 400 });
256
- }
257
- }
258
-
259
- return new Response("OK", { status: 200 });
260
- }
261
-
262
- /**
263
- * Sign payload with HMAC-SHA256
264
- */
265
- private sign(payload: string): string {
266
- if (!this.config.secret) return "";
267
- const hmac = createHmac("sha256", this.config.secret);
268
- hmac.update(payload);
269
- return hmac.digest("hex");
270
- }
271
-
272
- /**
273
- * Verify HMAC signature
274
- */
275
- private verify(payload: string, signature: string): boolean {
276
- if (!this.config.secret) return false;
277
- const expected = this.sign(payload);
278
- if (expected.length !== signature.length) return false;
279
-
280
- try {
281
- return timingSafeEqual(Buffer.from(expected), Buffer.from(signature));
282
- } catch {
283
- return false;
284
- }
285
- }
286
- }
@@ -1,171 +0,0 @@
1
- /**
2
- * State Persistence for Pause/Resume (v0.15.0 US-003)
3
- *
4
- * Serializes run state when pausing, loads state when resuming.
5
- */
6
-
7
- import * as path from "node:path";
8
- import type { InteractionRequest, InteractionResponse } from "./types";
9
-
10
- /** Serialized run state for pause/resume */
11
- export interface RunState {
12
- /** Feature name */
13
- feature: string;
14
- /** PRD path */
15
- prdPath: string;
16
- /** Current iteration number */
17
- iteration: number;
18
- /** Accumulated cost (USD) */
19
- totalCost: number;
20
- /** Stories completed */
21
- storiesCompleted: number;
22
- /** Pending interactions */
23
- pendingInteractions: InteractionRequest[];
24
- /** Completed interactions */
25
- completedInteractions: Array<{
26
- request: InteractionRequest;
27
- response: InteractionResponse;
28
- }>;
29
- /** Pause timestamp */
30
- pausedAt: number;
31
- /** Pause reason */
32
- pauseReason: string;
33
- /** Current story ID (if paused mid-story) */
34
- currentStoryId?: string;
35
- /** Current tier */
36
- currentTier?: string;
37
- /** Current model */
38
- currentModel?: string;
39
- /** Arbitrary metadata */
40
- metadata?: Record<string, unknown>;
41
- }
42
-
43
- /**
44
- * Serialize run state to JSON file
45
- */
46
- export async function serializeRunState(state: RunState, featureDir: string): Promise<string> {
47
- const stateFile = path.join(featureDir, "run-state.json");
48
- const json = JSON.stringify(state, null, 2);
49
- await Bun.write(stateFile, json);
50
- return stateFile;
51
- }
52
-
53
- /**
54
- * Deserialize run state from JSON file
55
- */
56
- export async function deserializeRunState(featureDir: string): Promise<RunState | null> {
57
- const stateFile = path.join(featureDir, "run-state.json");
58
- try {
59
- const file = Bun.file(stateFile);
60
- const exists = await file.exists();
61
- if (!exists) {
62
- return null;
63
- }
64
- const json = await file.text();
65
- const state = JSON.parse(json) as RunState;
66
- return state;
67
- } catch (err) {
68
- // Corrupted or invalid state file
69
- return null;
70
- }
71
- }
72
-
73
- /**
74
- * Delete run state file (after successful resume)
75
- */
76
- export async function clearRunState(featureDir: string): Promise<void> {
77
- const stateFile = path.join(featureDir, "run-state.json");
78
- try {
79
- await Bun.write(stateFile, ""); // truncate
80
- // Note: Bun doesn't have fs.unlink, so we truncate instead
81
- } catch {
82
- // Ignore errors
83
- }
84
- }
85
-
86
- /**
87
- * Save a pending interaction to the interactions directory
88
- */
89
- export async function savePendingInteraction(request: InteractionRequest, featureDir: string): Promise<string> {
90
- const interactionsDir = path.join(featureDir, "interactions");
91
- // Ensure directory exists
92
- await Bun.write(path.join(interactionsDir, ".gitkeep"), "");
93
-
94
- const filename = `${request.id}.json`;
95
- const filePath = path.join(interactionsDir, filename);
96
- const json = JSON.stringify(request, null, 2);
97
- await Bun.write(filePath, json);
98
- return filePath;
99
- }
100
-
101
- /**
102
- * Load a pending interaction from the interactions directory
103
- */
104
- export async function loadPendingInteraction(
105
- requestId: string,
106
- featureDir: string,
107
- ): Promise<InteractionRequest | null> {
108
- const interactionsDir = path.join(featureDir, "interactions");
109
- const filename = `${requestId}.json`;
110
- const filePath = path.join(interactionsDir, filename);
111
-
112
- try {
113
- const file = Bun.file(filePath);
114
- const exists = await file.exists();
115
- if (!exists) {
116
- return null;
117
- }
118
- const json = await file.text();
119
- const request = JSON.parse(json) as InteractionRequest;
120
- return request;
121
- } catch {
122
- return null;
123
- }
124
- }
125
-
126
- /**
127
- * Delete a pending interaction file (after response received)
128
- */
129
- export async function deletePendingInteraction(requestId: string, featureDir: string): Promise<void> {
130
- const interactionsDir = path.join(featureDir, "interactions");
131
- const filename = `${requestId}.json`;
132
- const filePath = path.join(interactionsDir, filename);
133
-
134
- try {
135
- await Bun.write(filePath, ""); // truncate
136
- } catch {
137
- // Ignore errors
138
- }
139
- }
140
-
141
- /**
142
- * List all pending interaction IDs
143
- */
144
- export async function listPendingInteractions(featureDir: string): Promise<string[]> {
145
- const interactionsDir = path.join(featureDir, "interactions");
146
-
147
- try {
148
- const dir = Bun.file(interactionsDir);
149
- const exists = await dir.exists();
150
- if (!exists) {
151
- return [];
152
- }
153
-
154
- // Use Bun.spawn to list files
155
- const proc = Bun.spawn(["ls", interactionsDir], {
156
- stdout: "pipe",
157
- stderr: "pipe",
158
- });
159
- const output = await new Response(proc.stdout).text();
160
- await proc.exited;
161
-
162
- const files = output
163
- .split("\n")
164
- .filter((f) => f.endsWith(".json") && f !== ".gitkeep")
165
- .map((f) => f.replace(".json", ""));
166
-
167
- return files;
168
- } catch {
169
- return [];
170
- }
171
- }
@@ -1,250 +0,0 @@
1
- /**
2
- * Built-in Triggers Integration (v0.15.0 US-004)
3
- *
4
- * Wires 8 built-in triggers into the runner loop and hooks system.
5
- */
6
-
7
- import type { NaxConfig } from "../config";
8
- import type { InteractionChain } from "./chain";
9
- import type { InteractionFallback, InteractionRequest, InteractionResponse, TriggerName } from "./types";
10
- import { TRIGGER_METADATA } from "./types";
11
-
12
- /** Trigger context data for template substitution */
13
- export interface TriggerContext {
14
- featureName: string;
15
- storyId?: string;
16
- cost?: number;
17
- limit?: number;
18
- tier?: string;
19
- model?: string;
20
- iteration?: number;
21
- reason?: string;
22
- [key: string]: unknown;
23
- }
24
-
25
- /**
26
- * Check if a trigger is enabled in config
27
- */
28
- export function isTriggerEnabled(trigger: TriggerName, config: NaxConfig): boolean {
29
- const triggerConfig = config.interaction?.triggers?.[trigger];
30
- if (triggerConfig === undefined) return false;
31
- if (typeof triggerConfig === "boolean") return triggerConfig;
32
- return triggerConfig.enabled;
33
- }
34
-
35
- /**
36
- * Get trigger configuration (fallback, timeout)
37
- */
38
- function getTriggerConfig(trigger: TriggerName, config: NaxConfig): { fallback: InteractionFallback; timeout: number } {
39
- const metadata = TRIGGER_METADATA[trigger];
40
- const triggerConfig = config.interaction?.triggers?.[trigger];
41
- const defaults = config.interaction?.defaults ?? {
42
- timeout: 600000,
43
- fallback: "escalate" as InteractionFallback,
44
- };
45
-
46
- let fallback: InteractionFallback = metadata.defaultFallback;
47
- let timeout = defaults.timeout;
48
-
49
- if (typeof triggerConfig === "object") {
50
- if (triggerConfig.fallback) {
51
- fallback = triggerConfig.fallback as InteractionFallback;
52
- }
53
- if (triggerConfig.timeout) {
54
- timeout = triggerConfig.timeout;
55
- }
56
- }
57
-
58
- return { fallback, timeout };
59
- }
60
-
61
- /**
62
- * Substitute {{variable}} placeholders in a template string
63
- */
64
- function substituteTemplate(template: string, context: TriggerContext): string {
65
- let result = template;
66
- for (const [key, value] of Object.entries(context)) {
67
- if (value !== undefined) {
68
- result = result.replace(new RegExp(`\\{\\{${key}\\}\\}`, "g"), String(value));
69
- }
70
- }
71
- return result;
72
- }
73
-
74
- /**
75
- * Create an interaction request for a built-in trigger
76
- */
77
- export function createTriggerRequest(
78
- trigger: TriggerName,
79
- context: TriggerContext,
80
- config: NaxConfig,
81
- ): InteractionRequest {
82
- const metadata = TRIGGER_METADATA[trigger];
83
- const { fallback, timeout } = getTriggerConfig(trigger, config);
84
-
85
- const summary = substituteTemplate(metadata.defaultSummary, context);
86
- const id = `trigger-${trigger}-${Date.now()}`;
87
-
88
- return {
89
- id,
90
- type: "confirm",
91
- featureName: context.featureName,
92
- storyId: context.storyId,
93
- stage: "custom",
94
- summary,
95
- fallback,
96
- timeout,
97
- createdAt: Date.now(),
98
- metadata: {
99
- trigger,
100
- safety: metadata.safety,
101
- },
102
- };
103
- }
104
-
105
- /**
106
- * Execute a trigger and return response
107
- */
108
- export async function executeTrigger(
109
- trigger: TriggerName,
110
- context: TriggerContext,
111
- config: NaxConfig,
112
- chain: InteractionChain,
113
- ): Promise<InteractionResponse> {
114
- const request = createTriggerRequest(trigger, context, config);
115
- const response = await chain.prompt(request);
116
- return response;
117
- }
118
-
119
- /**
120
- * Check security-review trigger (abort on critical issues)
121
- */
122
- export async function checkSecurityReview(
123
- context: TriggerContext,
124
- config: NaxConfig,
125
- chain: InteractionChain,
126
- ): Promise<boolean> {
127
- if (!isTriggerEnabled("security-review", config)) return true;
128
-
129
- const response = await executeTrigger("security-review", context, config, chain);
130
- return response.action !== "abort";
131
- }
132
-
133
- /**
134
- * Check cost-exceeded trigger (abort on limit exceeded)
135
- */
136
- export async function checkCostExceeded(
137
- context: TriggerContext,
138
- config: NaxConfig,
139
- chain: InteractionChain,
140
- ): Promise<boolean> {
141
- if (!isTriggerEnabled("cost-exceeded", config)) return true;
142
-
143
- const response = await executeTrigger("cost-exceeded", context, config, chain);
144
- return response.action !== "abort";
145
- }
146
-
147
- /**
148
- * Check merge-conflict trigger (abort on conflict)
149
- */
150
- export async function checkMergeConflict(
151
- context: TriggerContext,
152
- config: NaxConfig,
153
- chain: InteractionChain,
154
- ): Promise<boolean> {
155
- if (!isTriggerEnabled("merge-conflict", config)) return true;
156
-
157
- const response = await executeTrigger("merge-conflict", context, config, chain);
158
- return response.action !== "abort";
159
- }
160
-
161
- /**
162
- * Check cost-warning trigger (escalate on approaching limit)
163
- */
164
- export async function checkCostWarning(
165
- context: TriggerContext,
166
- config: NaxConfig,
167
- chain: InteractionChain,
168
- ): Promise<"continue" | "escalate"> {
169
- if (!isTriggerEnabled("cost-warning", config)) return "continue";
170
-
171
- const response = await executeTrigger("cost-warning", context, config, chain);
172
- return response.action === "approve" ? "escalate" : "continue";
173
- }
174
-
175
- /**
176
- * Check max-retries trigger (skip story on max retries)
177
- */
178
- export async function checkMaxRetries(
179
- context: TriggerContext,
180
- config: NaxConfig,
181
- chain: InteractionChain,
182
- ): Promise<"continue" | "skip"> {
183
- if (!isTriggerEnabled("max-retries", config)) return "continue";
184
-
185
- const response = await executeTrigger("max-retries", context, config, chain);
186
- return response.action === "skip" ? "skip" : "continue";
187
- }
188
-
189
- /**
190
- * Check pre-merge trigger (escalate before merging)
191
- */
192
- export async function checkPreMerge(
193
- context: TriggerContext,
194
- config: NaxConfig,
195
- chain: InteractionChain,
196
- ): Promise<boolean> {
197
- if (!isTriggerEnabled("pre-merge", config)) return true;
198
-
199
- const response = await executeTrigger("pre-merge", context, config, chain);
200
- return response.action === "approve";
201
- }
202
-
203
- /**
204
- * Check story-ambiguity trigger (continue with best effort)
205
- */
206
- export async function checkStoryAmbiguity(
207
- context: TriggerContext,
208
- config: NaxConfig,
209
- chain: InteractionChain,
210
- ): Promise<boolean> {
211
- if (!isTriggerEnabled("story-ambiguity", config)) return true;
212
-
213
- const response = await executeTrigger("story-ambiguity", context, config, chain);
214
- return response.action === "approve";
215
- }
216
-
217
- /**
218
- * Check review-gate trigger (proceed with review)
219
- */
220
- export async function checkReviewGate(
221
- context: TriggerContext,
222
- config: NaxConfig,
223
- chain: InteractionChain,
224
- ): Promise<boolean> {
225
- if (!isTriggerEnabled("review-gate", config)) return true;
226
-
227
- const response = await executeTrigger("review-gate", context, config, chain);
228
- return response.action === "approve";
229
- }
230
-
231
- /**
232
- * Check story-oversized trigger (decompose, skip, or continue)
233
- */
234
- export async function checkStoryOversized(
235
- context: TriggerContext,
236
- config: NaxConfig,
237
- chain: InteractionChain,
238
- ): Promise<"decompose" | "skip" | "continue"> {
239
- if (!isTriggerEnabled("story-oversized", config)) return "continue";
240
-
241
- try {
242
- const response = await executeTrigger("story-oversized", context, config, chain);
243
- if (response.action === "approve") return "decompose";
244
- if (response.action === "skip") return "skip";
245
- return "continue";
246
- } catch {
247
- // No plugin registered or all plugins failed — apply default fallback
248
- return "continue";
249
- }
250
- }