@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.
- package/CHANGELOG.md +30 -0
- package/README.md +177 -104
- package/dist/nax.js +417 -213
- package/package.json +1 -3
- package/bin/nax.ts +0 -1195
- package/src/acceptance/fix-generator.ts +0 -322
- package/src/acceptance/generator.ts +0 -415
- package/src/acceptance/index.ts +0 -42
- package/src/acceptance/refinement.ts +0 -224
- package/src/acceptance/templates/cli.ts +0 -47
- package/src/acceptance/templates/component.ts +0 -78
- package/src/acceptance/templates/e2e.ts +0 -43
- package/src/acceptance/templates/index.ts +0 -21
- package/src/acceptance/templates/snapshot.ts +0 -50
- package/src/acceptance/templates/unit.ts +0 -48
- package/src/acceptance/types.ts +0 -138
- package/src/agents/acp/adapter.ts +0 -888
- package/src/agents/acp/cost.ts +0 -9
- package/src/agents/acp/index.ts +0 -7
- package/src/agents/acp/interaction-bridge.ts +0 -126
- package/src/agents/acp/parser.ts +0 -119
- package/src/agents/acp/spawn-client.ts +0 -373
- package/src/agents/acp/types.ts +0 -22
- package/src/agents/aider/adapter.ts +0 -135
- package/src/agents/claude/adapter.ts +0 -258
- package/src/agents/claude/complete.ts +0 -80
- package/src/agents/claude/cost.ts +0 -16
- package/src/agents/claude/execution.ts +0 -215
- package/src/agents/claude/index.ts +0 -3
- package/src/agents/claude/interactive.ts +0 -77
- package/src/agents/claude/plan.ts +0 -179
- package/src/agents/codex/adapter.ts +0 -153
- package/src/agents/cost/calculate.ts +0 -154
- package/src/agents/cost/index.ts +0 -10
- package/src/agents/cost/parse.ts +0 -97
- package/src/agents/cost/pricing.ts +0 -59
- package/src/agents/cost/types.ts +0 -45
- package/src/agents/gemini/adapter.ts +0 -177
- package/src/agents/index.ts +0 -18
- package/src/agents/opencode/adapter.ts +0 -106
- package/src/agents/registry.ts +0 -136
- package/src/agents/shared/decompose.ts +0 -154
- package/src/agents/shared/model-resolution.ts +0 -43
- package/src/agents/shared/types-extended.ts +0 -164
- package/src/agents/shared/validation.ts +0 -69
- package/src/agents/shared/version-detection.ts +0 -109
- package/src/agents/types.ts +0 -205
- package/src/analyze/classifier.ts +0 -282
- package/src/analyze/index.ts +0 -16
- package/src/analyze/scanner.ts +0 -171
- package/src/analyze/types.ts +0 -51
- package/src/cli/accept.ts +0 -108
- package/src/cli/agents.ts +0 -87
- package/src/cli/analyze-parser.ts +0 -291
- package/src/cli/analyze.ts +0 -352
- package/src/cli/config-descriptions.ts +0 -219
- package/src/cli/config-diff.ts +0 -103
- package/src/cli/config-display.ts +0 -285
- package/src/cli/config-get.ts +0 -55
- package/src/cli/config.ts +0 -14
- package/src/cli/constitution.ts +0 -17
- package/src/cli/diagnose-analysis.ts +0 -159
- package/src/cli/diagnose-formatter.ts +0 -87
- package/src/cli/diagnose.ts +0 -203
- package/src/cli/generate.ts +0 -250
- package/src/cli/index.ts +0 -42
- package/src/cli/init-context.ts +0 -405
- package/src/cli/init-detect.ts +0 -303
- package/src/cli/init.ts +0 -296
- package/src/cli/interact.ts +0 -295
- package/src/cli/plan.ts +0 -509
- package/src/cli/plugins.ts +0 -122
- package/src/cli/prompts-export.ts +0 -58
- package/src/cli/prompts-init.ts +0 -200
- package/src/cli/prompts-main.ts +0 -183
- package/src/cli/prompts-shared.ts +0 -70
- package/src/cli/prompts-tdd.ts +0 -88
- package/src/cli/prompts.ts +0 -17
- package/src/cli/runs.ts +0 -174
- package/src/cli/status-cost.ts +0 -151
- package/src/cli/status-features.ts +0 -405
- package/src/cli/status.ts +0 -13
- package/src/commands/common.ts +0 -171
- package/src/commands/diagnose.ts +0 -17
- package/src/commands/index.ts +0 -9
- package/src/commands/logs-formatter.ts +0 -201
- package/src/commands/logs-reader.ts +0 -171
- package/src/commands/logs.ts +0 -103
- package/src/commands/precheck.ts +0 -86
- package/src/commands/runs.ts +0 -220
- package/src/commands/unlock.ts +0 -96
- package/src/config/defaults.ts +0 -218
- package/src/config/index.ts +0 -22
- package/src/config/loader.ts +0 -143
- package/src/config/merge.ts +0 -106
- package/src/config/merger.ts +0 -147
- package/src/config/path-security.ts +0 -121
- package/src/config/paths.ts +0 -27
- package/src/config/permissions.ts +0 -63
- package/src/config/runtime-types.ts +0 -522
- package/src/config/schema-types.ts +0 -53
- package/src/config/schema.ts +0 -60
- package/src/config/schemas.ts +0 -426
- package/src/config/test-strategy.ts +0 -71
- package/src/config/types.ts +0 -57
- package/src/config/validate.ts +0 -103
- package/src/constitution/generator.ts +0 -158
- package/src/constitution/generators/aider.ts +0 -41
- package/src/constitution/generators/claude.ts +0 -35
- package/src/constitution/generators/cursor.ts +0 -36
- package/src/constitution/generators/opencode.ts +0 -38
- package/src/constitution/generators/types.ts +0 -33
- package/src/constitution/generators/windsurf.ts +0 -36
- package/src/constitution/index.ts +0 -11
- package/src/constitution/loader.ts +0 -121
- package/src/constitution/types.ts +0 -31
- package/src/context/auto-detect.ts +0 -228
- package/src/context/builder.ts +0 -299
- package/src/context/elements.ts +0 -122
- package/src/context/formatter.ts +0 -107
- package/src/context/generator.ts +0 -343
- package/src/context/generators/aider.ts +0 -34
- package/src/context/generators/claude.ts +0 -28
- package/src/context/generators/codex.ts +0 -28
- package/src/context/generators/cursor.ts +0 -28
- package/src/context/generators/gemini.ts +0 -28
- package/src/context/generators/opencode.ts +0 -30
- package/src/context/generators/windsurf.ts +0 -28
- package/src/context/greenfield.ts +0 -114
- package/src/context/index.ts +0 -34
- package/src/context/injector.ts +0 -279
- package/src/context/parent-context.ts +0 -39
- package/src/context/test-scanner.ts +0 -370
- package/src/context/types.ts +0 -98
- package/src/decompose/apply.ts +0 -50
- package/src/decompose/builder.ts +0 -181
- package/src/decompose/index.ts +0 -8
- package/src/decompose/sections/codebase.ts +0 -26
- package/src/decompose/sections/constraints.ts +0 -32
- package/src/decompose/sections/index.ts +0 -4
- package/src/decompose/sections/sibling-stories.ts +0 -25
- package/src/decompose/sections/target-story.ts +0 -31
- package/src/decompose/types.ts +0 -55
- package/src/decompose/validators/complexity.ts +0 -45
- package/src/decompose/validators/coverage.ts +0 -134
- package/src/decompose/validators/dependency.ts +0 -91
- package/src/decompose/validators/index.ts +0 -35
- package/src/decompose/validators/overlap.ts +0 -128
- package/src/errors.ts +0 -67
- package/src/execution/batching.ts +0 -157
- package/src/execution/crash-heartbeat.ts +0 -77
- package/src/execution/crash-recovery.ts +0 -79
- package/src/execution/crash-signals.ts +0 -165
- package/src/execution/crash-writer.ts +0 -154
- package/src/execution/deferred-review.ts +0 -105
- package/src/execution/dry-run.ts +0 -81
- package/src/execution/escalation/escalation.ts +0 -46
- package/src/execution/escalation/index.ts +0 -13
- package/src/execution/escalation/tier-escalation.ts +0 -346
- package/src/execution/escalation/tier-outcome.ts +0 -143
- package/src/execution/executor-types.ts +0 -73
- package/src/execution/helpers.ts +0 -38
- package/src/execution/index.ts +0 -27
- package/src/execution/iteration-runner.ts +0 -160
- package/src/execution/lifecycle/acceptance-loop.ts +0 -309
- package/src/execution/lifecycle/headless-formatter.ts +0 -83
- package/src/execution/lifecycle/index.ts +0 -11
- package/src/execution/lifecycle/parallel-lifecycle.ts +0 -101
- package/src/execution/lifecycle/precheck-runner.ts +0 -140
- package/src/execution/lifecycle/run-cleanup.ts +0 -81
- package/src/execution/lifecycle/run-completion.ts +0 -247
- package/src/execution/lifecycle/run-initialization.ts +0 -187
- package/src/execution/lifecycle/run-regression.ts +0 -305
- package/src/execution/lifecycle/run-setup.ts +0 -240
- package/src/execution/lifecycle/story-size-prompts.ts +0 -123
- package/src/execution/lock.ts +0 -129
- package/src/execution/parallel-coordinator.ts +0 -281
- package/src/execution/parallel-executor-rectification-pass.ts +0 -117
- package/src/execution/parallel-executor-rectify.ts +0 -136
- package/src/execution/parallel-executor.ts +0 -330
- package/src/execution/parallel-worker.ts +0 -149
- package/src/execution/parallel.ts +0 -13
- package/src/execution/pid-registry.ts +0 -275
- package/src/execution/pipeline-result-handler.ts +0 -221
- package/src/execution/progress.ts +0 -27
- package/src/execution/queue-handler.ts +0 -109
- package/src/execution/runner-completion.ts +0 -171
- package/src/execution/runner-execution.ts +0 -243
- package/src/execution/runner-setup.ts +0 -86
- package/src/execution/runner.ts +0 -265
- package/src/execution/sequential-executor.ts +0 -219
- package/src/execution/status-file.ts +0 -264
- package/src/execution/status-writer.ts +0 -181
- package/src/execution/story-context.ts +0 -266
- package/src/execution/story-selector.ts +0 -76
- package/src/execution/test-output-parser.ts +0 -14
- package/src/execution/timeout-handler.ts +0 -100
- package/src/hooks/index.ts +0 -2
- package/src/hooks/runner.ts +0 -280
- package/src/hooks/types.ts +0 -79
- package/src/interaction/chain.ts +0 -170
- package/src/interaction/index.ts +0 -61
- package/src/interaction/init.ts +0 -84
- package/src/interaction/plugins/auto.ts +0 -243
- package/src/interaction/plugins/cli.ts +0 -300
- package/src/interaction/plugins/telegram.ts +0 -384
- package/src/interaction/plugins/webhook.ts +0 -286
- package/src/interaction/state.ts +0 -171
- package/src/interaction/triggers.ts +0 -250
- package/src/interaction/types.ts +0 -170
- package/src/logger/formatters.ts +0 -84
- package/src/logger/index.ts +0 -16
- package/src/logger/logger.ts +0 -296
- package/src/logger/types.ts +0 -48
- package/src/logging/formatter.ts +0 -355
- package/src/logging/index.ts +0 -22
- package/src/logging/types.ts +0 -93
- package/src/metrics/aggregator.ts +0 -191
- package/src/metrics/index.ts +0 -14
- package/src/metrics/tracker.ts +0 -200
- package/src/metrics/types.ts +0 -115
- package/src/optimizer/index.ts +0 -63
- package/src/optimizer/noop.optimizer.ts +0 -24
- package/src/optimizer/rule-based.optimizer.ts +0 -248
- package/src/optimizer/types.ts +0 -53
- package/src/pipeline/event-bus.ts +0 -297
- package/src/pipeline/events.ts +0 -130
- package/src/pipeline/index.ts +0 -19
- package/src/pipeline/runner.ts +0 -149
- package/src/pipeline/stages/acceptance-setup.ts +0 -144
- package/src/pipeline/stages/acceptance.ts +0 -215
- package/src/pipeline/stages/autofix.ts +0 -262
- package/src/pipeline/stages/completion.ts +0 -110
- package/src/pipeline/stages/constitution.ts +0 -63
- package/src/pipeline/stages/context.ts +0 -122
- package/src/pipeline/stages/execution.ts +0 -359
- package/src/pipeline/stages/index.ts +0 -86
- package/src/pipeline/stages/optimizer.ts +0 -74
- package/src/pipeline/stages/prompt.ts +0 -79
- package/src/pipeline/stages/queue-check.ts +0 -103
- package/src/pipeline/stages/rectify.ts +0 -101
- package/src/pipeline/stages/regression.ts +0 -99
- package/src/pipeline/stages/review.ts +0 -94
- package/src/pipeline/stages/routing.ts +0 -276
- package/src/pipeline/stages/verify.ts +0 -286
- package/src/pipeline/subscribers/events-writer.ts +0 -135
- package/src/pipeline/subscribers/hooks.ts +0 -179
- package/src/pipeline/subscribers/interaction.ts +0 -103
- package/src/pipeline/subscribers/registry.ts +0 -73
- package/src/pipeline/subscribers/reporters.ts +0 -174
- package/src/pipeline/types.ts +0 -220
- package/src/plugins/extensions.ts +0 -225
- package/src/plugins/index.ts +0 -33
- package/src/plugins/loader.ts +0 -352
- package/src/plugins/plugin-logger.ts +0 -41
- package/src/plugins/registry.ts +0 -168
- package/src/plugins/types.ts +0 -206
- package/src/plugins/validator.ts +0 -352
- package/src/prd/index.ts +0 -220
- package/src/prd/schema.ts +0 -268
- package/src/prd/types.ts +0 -273
- package/src/prd/validate.ts +0 -41
- package/src/precheck/checks-agents.ts +0 -63
- package/src/precheck/checks-blockers.ts +0 -23
- package/src/precheck/checks-cli.ts +0 -68
- package/src/precheck/checks-config.ts +0 -102
- package/src/precheck/checks-git.ts +0 -117
- package/src/precheck/checks-system.ts +0 -101
- package/src/precheck/checks-warnings.ts +0 -221
- package/src/precheck/checks.ts +0 -36
- package/src/precheck/index.ts +0 -374
- package/src/precheck/story-size-gate.ts +0 -144
- package/src/precheck/types.ts +0 -31
- package/src/prompts/builder.ts +0 -166
- package/src/prompts/index.ts +0 -2
- package/src/prompts/loader.ts +0 -43
- package/src/prompts/sections/conventions.ts +0 -19
- package/src/prompts/sections/hermetic.ts +0 -41
- package/src/prompts/sections/index.ts +0 -12
- package/src/prompts/sections/isolation.ts +0 -70
- package/src/prompts/sections/role-task.ts +0 -182
- package/src/prompts/sections/story.ts +0 -55
- package/src/prompts/sections/verdict.ts +0 -70
- package/src/prompts/types.ts +0 -21
- package/src/queue/index.ts +0 -2
- package/src/queue/manager.ts +0 -254
- package/src/queue/types.ts +0 -54
- package/src/review/index.ts +0 -8
- package/src/review/orchestrator.ts +0 -154
- package/src/review/runner.ts +0 -303
- package/src/review/types.ts +0 -70
- package/src/routing/batch-route.ts +0 -35
- package/src/routing/builder.ts +0 -81
- package/src/routing/chain.ts +0 -75
- package/src/routing/content-hash.ts +0 -25
- package/src/routing/index.ts +0 -20
- package/src/routing/loader.ts +0 -62
- package/src/routing/router.ts +0 -305
- package/src/routing/strategies/adaptive.ts +0 -215
- package/src/routing/strategies/index.ts +0 -8
- package/src/routing/strategies/keyword.ts +0 -180
- package/src/routing/strategies/llm-prompts.ts +0 -224
- package/src/routing/strategies/llm.ts +0 -320
- package/src/routing/strategies/manual.ts +0 -50
- package/src/routing/strategy.ts +0 -102
- package/src/tdd/cleanup.ts +0 -120
- package/src/tdd/index.ts +0 -22
- package/src/tdd/isolation.ts +0 -117
- package/src/tdd/orchestrator.ts +0 -406
- package/src/tdd/prompts.ts +0 -40
- package/src/tdd/rectification-gate.ts +0 -274
- package/src/tdd/session-runner.ts +0 -263
- package/src/tdd/types.ts +0 -84
- package/src/tdd/verdict-reader.ts +0 -266
- package/src/tdd/verdict.ts +0 -152
- package/src/tui/App.tsx +0 -265
- package/src/tui/components/AgentPanel.tsx +0 -75
- package/src/tui/components/CostOverlay.tsx +0 -118
- package/src/tui/components/HelpOverlay.tsx +0 -107
- package/src/tui/components/StatusBar.tsx +0 -63
- package/src/tui/components/StoriesPanel.tsx +0 -177
- package/src/tui/hooks/useKeyboard.ts +0 -142
- package/src/tui/hooks/useLayout.ts +0 -137
- package/src/tui/hooks/usePipelineEvents.ts +0 -183
- package/src/tui/hooks/usePty.ts +0 -189
- package/src/tui/index.tsx +0 -38
- package/src/tui/types.ts +0 -76
- package/src/utils/errors.ts +0 -12
- package/src/utils/git.ts +0 -245
- package/src/utils/json-file.ts +0 -72
- package/src/utils/log-test-output.ts +0 -25
- package/src/utils/path-security.ts +0 -73
- package/src/utils/queue-writer.ts +0 -54
- package/src/verification/crash-detector.ts +0 -34
- package/src/verification/executor.ts +0 -250
- package/src/verification/index.ts +0 -12
- package/src/verification/orchestrator-types.ts +0 -154
- package/src/verification/orchestrator.ts +0 -76
- package/src/verification/parser.ts +0 -220
- package/src/verification/rectification-loop.ts +0 -172
- package/src/verification/rectification.ts +0 -108
- package/src/verification/runners.ts +0 -129
- package/src/verification/smart-runner.ts +0 -307
- package/src/verification/strategies/acceptance.ts +0 -136
- package/src/verification/strategies/regression.ts +0 -90
- package/src/verification/strategies/scoped.ts +0 -154
- package/src/verification/types.ts +0 -117
- package/src/version.ts +0 -40
- package/src/worktree/dispatcher.ts +0 -6
- package/src/worktree/index.ts +0 -2
- package/src/worktree/manager.ts +0 -193
- package/src/worktree/merge.ts +0 -302
- 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
|
-
}
|
package/src/interaction/state.ts
DELETED
|
@@ -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
|
-
}
|