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