@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,264 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Status File — Machine-readable run state for external tooling
|
|
3
|
-
*
|
|
4
|
-
* Writes a JSON status file that external tools (CI/CD, orchestrators,
|
|
5
|
-
* dashboards) can poll to monitor nax runs without parsing logs.
|
|
6
|
-
*
|
|
7
|
-
* Atomic writes: write to <path>.tmp then rename to <path>
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import { rename, unlink } from "node:fs/promises";
|
|
11
|
-
import { resolve } from "node:path";
|
|
12
|
-
import type { NaxConfig } from "../config";
|
|
13
|
-
import type { PRD, StoryStatus, UserStory } from "../prd";
|
|
14
|
-
|
|
15
|
-
// ============================================================================
|
|
16
|
-
// NaxStatusFile Interface
|
|
17
|
-
// ============================================================================
|
|
18
|
-
|
|
19
|
-
/** Machine-readable status file written during nax runs */
|
|
20
|
-
export interface NaxStatusFile {
|
|
21
|
-
/** Schema version for forward compatibility */
|
|
22
|
-
version: 1;
|
|
23
|
-
|
|
24
|
-
/** Run metadata */
|
|
25
|
-
run: {
|
|
26
|
-
/** Run ID (e.g. "run-2026-02-25T10-00-00-000Z") */
|
|
27
|
-
id: string;
|
|
28
|
-
/** Feature name */
|
|
29
|
-
feature: string;
|
|
30
|
-
/** ISO 8601 start timestamp */
|
|
31
|
-
startedAt: string;
|
|
32
|
-
/** Current run status */
|
|
33
|
-
status: "running" | "completed" | "failed" | "stalled" | "crashed" | "precheck-failed";
|
|
34
|
-
/** Whether this is a dry run */
|
|
35
|
-
dryRun: boolean;
|
|
36
|
-
/** Process ID for crash detection */
|
|
37
|
-
pid: number;
|
|
38
|
-
/** ISO 8601 crash timestamp (present when status is "crashed") */
|
|
39
|
-
crashedAt?: string;
|
|
40
|
-
/** Signal or exception that caused crash (present when status is "crashed") */
|
|
41
|
-
crashSignal?: string;
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
/** Aggregate progress counts */
|
|
45
|
-
progress: {
|
|
46
|
-
/** Total stories in PRD */
|
|
47
|
-
total: number;
|
|
48
|
-
/** Stories that passed */
|
|
49
|
-
passed: number;
|
|
50
|
-
/** Stories that failed */
|
|
51
|
-
failed: number;
|
|
52
|
-
/** Stories that are paused */
|
|
53
|
-
paused: number;
|
|
54
|
-
/** Stories that are blocked */
|
|
55
|
-
blocked: number;
|
|
56
|
-
/** Stories not yet processed (total - passed - failed - paused - blocked) */
|
|
57
|
-
pending: number;
|
|
58
|
-
};
|
|
59
|
-
|
|
60
|
-
/** Cost tracking */
|
|
61
|
-
cost: {
|
|
62
|
-
/** Accumulated cost in USD */
|
|
63
|
-
spent: number;
|
|
64
|
-
/** Cost limit from config (null if not set) */
|
|
65
|
-
limit: number | null;
|
|
66
|
-
};
|
|
67
|
-
|
|
68
|
-
/** Current story being processed (null if between stories or at run boundaries) */
|
|
69
|
-
current: {
|
|
70
|
-
/** Story ID */
|
|
71
|
-
storyId: string;
|
|
72
|
-
/** Story title */
|
|
73
|
-
title: string;
|
|
74
|
-
/** Complexity level */
|
|
75
|
-
complexity: string;
|
|
76
|
-
/** TDD strategy */
|
|
77
|
-
tddStrategy: string;
|
|
78
|
-
/** Resolved model name */
|
|
79
|
-
model: string;
|
|
80
|
-
/** Current attempt number (1-based) */
|
|
81
|
-
attempt: number;
|
|
82
|
-
/** Current phase */
|
|
83
|
-
phase: string;
|
|
84
|
-
} | null;
|
|
85
|
-
|
|
86
|
-
/** Number of loop iterations completed */
|
|
87
|
-
iterations: number;
|
|
88
|
-
|
|
89
|
-
/** ISO 8601 last-updated timestamp */
|
|
90
|
-
updatedAt: string;
|
|
91
|
-
|
|
92
|
-
/** Elapsed duration in milliseconds */
|
|
93
|
-
durationMs: number;
|
|
94
|
-
|
|
95
|
-
/** ISO 8601 last heartbeat timestamp (updated every 60s during execution) */
|
|
96
|
-
lastHeartbeat?: string;
|
|
97
|
-
|
|
98
|
-
/** Parallel execution info (present when --parallel is used) */
|
|
99
|
-
parallel?: {
|
|
100
|
-
/** Whether parallel mode is enabled */
|
|
101
|
-
enabled: boolean;
|
|
102
|
-
/** Max concurrent sessions */
|
|
103
|
-
maxConcurrency: number;
|
|
104
|
-
/** Currently executing stories in parallel */
|
|
105
|
-
activeStories: Array<{
|
|
106
|
-
storyId: string;
|
|
107
|
-
worktreePath: string;
|
|
108
|
-
}>;
|
|
109
|
-
};
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
// ============================================================================
|
|
113
|
-
// Progress Counting
|
|
114
|
-
// ============================================================================
|
|
115
|
-
|
|
116
|
-
/**
|
|
117
|
-
* Derive progress counts from PRD story statuses.
|
|
118
|
-
*
|
|
119
|
-
* Counts each story by its current status. `pending` is computed as
|
|
120
|
-
* everything not in the four explicit terminal/waiting states.
|
|
121
|
-
*/
|
|
122
|
-
export function countProgress(prd: PRD): NaxStatusFile["progress"] {
|
|
123
|
-
const stories = prd.userStories;
|
|
124
|
-
const passed = stories.filter((s) => s.status === "passed").length;
|
|
125
|
-
const failed = stories.filter((s) => s.status === "failed").length;
|
|
126
|
-
const paused = stories.filter((s) => s.status === "paused").length;
|
|
127
|
-
const blocked = stories.filter((s) => s.status === "blocked").length;
|
|
128
|
-
const total = stories.length;
|
|
129
|
-
const pending = total - passed - failed - paused - blocked;
|
|
130
|
-
|
|
131
|
-
return { total, passed, failed, paused, blocked, pending };
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
// ============================================================================
|
|
135
|
-
// Run State (for buildStatusSnapshot)
|
|
136
|
-
// ============================================================================
|
|
137
|
-
|
|
138
|
-
/**
|
|
139
|
-
* Snapshot of current run state used to build NaxStatusFile.
|
|
140
|
-
*
|
|
141
|
-
* This is a value-only snapshot — callers pass in what they have at the
|
|
142
|
-
* current write point. The runner constructs this inline from local variables.
|
|
143
|
-
*/
|
|
144
|
-
export interface RunStateSnapshot {
|
|
145
|
-
/** Unique run identifier */
|
|
146
|
-
runId: string;
|
|
147
|
-
/** Feature name */
|
|
148
|
-
feature: string;
|
|
149
|
-
/** ISO 8601 start timestamp */
|
|
150
|
-
startedAt: string;
|
|
151
|
-
/** Current run status */
|
|
152
|
-
runStatus: NaxStatusFile["run"]["status"];
|
|
153
|
-
/** Whether this is a dry run */
|
|
154
|
-
dryRun: boolean;
|
|
155
|
-
/** Process ID for crash detection */
|
|
156
|
-
pid: number;
|
|
157
|
-
/** Loaded PRD (for progress counting) */
|
|
158
|
-
prd: PRD;
|
|
159
|
-
/** Accumulated cost in USD */
|
|
160
|
-
totalCost: number;
|
|
161
|
-
/** Cost limit from config (or null) */
|
|
162
|
-
costLimit: number | null;
|
|
163
|
-
/** Currently-executing story info (null between stories) */
|
|
164
|
-
currentStory: {
|
|
165
|
-
storyId: string;
|
|
166
|
-
title: string;
|
|
167
|
-
complexity: string;
|
|
168
|
-
tddStrategy: string;
|
|
169
|
-
model: string;
|
|
170
|
-
attempt: number;
|
|
171
|
-
phase: string;
|
|
172
|
-
} | null;
|
|
173
|
-
/** Number of loop iterations */
|
|
174
|
-
iterations: number;
|
|
175
|
-
/** Run start time as ms epoch (for computing durationMs) */
|
|
176
|
-
startTimeMs: number;
|
|
177
|
-
/** Parallel execution info (optional) */
|
|
178
|
-
parallel?: NaxStatusFile["parallel"];
|
|
179
|
-
/** ISO 8601 crash timestamp (present when status is "crashed") */
|
|
180
|
-
crashedAt?: string;
|
|
181
|
-
/** Signal or exception that caused crash (present when status is "crashed") */
|
|
182
|
-
crashSignal?: string;
|
|
183
|
-
/** ISO 8601 last heartbeat timestamp (updated every 60s during execution) */
|
|
184
|
-
lastHeartbeat?: string;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
// ============================================================================
|
|
188
|
-
// buildStatusSnapshot
|
|
189
|
-
// ============================================================================
|
|
190
|
-
|
|
191
|
-
/**
|
|
192
|
-
* Build a NaxStatusFile object from current run state.
|
|
193
|
-
*
|
|
194
|
-
* Derives progress from PRD story statuses. Sets updatedAt and durationMs
|
|
195
|
-
* from the current time. Does not write to disk — call writeStatusFile() for that.
|
|
196
|
-
*/
|
|
197
|
-
export function buildStatusSnapshot(state: RunStateSnapshot): NaxStatusFile {
|
|
198
|
-
const now = Date.now();
|
|
199
|
-
const snapshot: NaxStatusFile = {
|
|
200
|
-
version: 1,
|
|
201
|
-
run: {
|
|
202
|
-
id: state.runId,
|
|
203
|
-
feature: state.feature,
|
|
204
|
-
startedAt: state.startedAt,
|
|
205
|
-
status: state.runStatus,
|
|
206
|
-
dryRun: state.dryRun,
|
|
207
|
-
pid: state.pid,
|
|
208
|
-
...(state.crashedAt && { crashedAt: state.crashedAt }),
|
|
209
|
-
...(state.crashSignal && { crashSignal: state.crashSignal }),
|
|
210
|
-
},
|
|
211
|
-
progress: countProgress(state.prd),
|
|
212
|
-
cost: {
|
|
213
|
-
spent: state.totalCost,
|
|
214
|
-
limit: state.costLimit,
|
|
215
|
-
},
|
|
216
|
-
current: state.currentStory,
|
|
217
|
-
iterations: state.iterations,
|
|
218
|
-
updatedAt: new Date(now).toISOString(),
|
|
219
|
-
durationMs: now - state.startTimeMs,
|
|
220
|
-
...(state.lastHeartbeat && { lastHeartbeat: state.lastHeartbeat }),
|
|
221
|
-
};
|
|
222
|
-
|
|
223
|
-
if (state.parallel) {
|
|
224
|
-
snapshot.parallel = state.parallel;
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
return snapshot;
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
// ============================================================================
|
|
231
|
-
// Atomic Writer
|
|
232
|
-
// ============================================================================
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* Atomically write a NaxStatusFile to disk.
|
|
236
|
-
*
|
|
237
|
-
* Writes to `<path>.tmp` first, then renames to `<path>` to prevent
|
|
238
|
-
* consumers from reading partial JSON during the write.
|
|
239
|
-
*
|
|
240
|
-
* @param filePath - Destination path for the status file
|
|
241
|
-
* @param status - Status file content to write
|
|
242
|
-
* @throws Error if path traversal is detected
|
|
243
|
-
*/
|
|
244
|
-
export async function writeStatusFile(filePath: string, status: NaxStatusFile): Promise<void> {
|
|
245
|
-
// SEC-1: Validate path to prevent path traversal attacks
|
|
246
|
-
const resolvedPath = resolve(filePath);
|
|
247
|
-
|
|
248
|
-
// Check for path traversal patterns in the original path
|
|
249
|
-
if (filePath.includes("../") || filePath.includes("..\\")) {
|
|
250
|
-
throw new Error("Invalid status file path: path traversal detected");
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
const tmpPath = `${resolvedPath}.tmp`;
|
|
254
|
-
|
|
255
|
-
// MEM-1: Clean up stale .tmp file if it exists
|
|
256
|
-
try {
|
|
257
|
-
await unlink(tmpPath);
|
|
258
|
-
} catch {
|
|
259
|
-
// Ignore error if file doesn't exist
|
|
260
|
-
}
|
|
261
|
-
|
|
262
|
-
await Bun.write(tmpPath, JSON.stringify(status, null, 2));
|
|
263
|
-
await rename(tmpPath, resolvedPath);
|
|
264
|
-
}
|
|
@@ -1,181 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* StatusWriter — Encapsulates status file state and write logic
|
|
3
|
-
*
|
|
4
|
-
* Extracted from runner.ts. Manages the _runStatus, _prd, _currentStory state
|
|
5
|
-
* that was previously tracked as closure variables, plus the BUG-2 consecutive
|
|
6
|
-
* write failure counter. Provides atomic status file writes via writeStatusFile.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import { join } from "node:path";
|
|
10
|
-
import type { NaxConfig } from "../config";
|
|
11
|
-
import { getSafeLogger } from "../logger";
|
|
12
|
-
import type { PRD } from "../prd";
|
|
13
|
-
import { type RunStateSnapshot, buildStatusSnapshot, writeStatusFile } from "./status-file";
|
|
14
|
-
|
|
15
|
-
// ============================================================================
|
|
16
|
-
// StatusWriterContext — fixed run metadata set at construction
|
|
17
|
-
// ============================================================================
|
|
18
|
-
|
|
19
|
-
export interface StatusWriterContext {
|
|
20
|
-
/** Unique run identifier */
|
|
21
|
-
runId: string;
|
|
22
|
-
/** Feature name */
|
|
23
|
-
feature: string;
|
|
24
|
-
/** ISO 8601 run start timestamp */
|
|
25
|
-
startedAt: string;
|
|
26
|
-
/** Whether this is a dry run */
|
|
27
|
-
dryRun: boolean;
|
|
28
|
-
/** Run start time as ms epoch (for computing durationMs) */
|
|
29
|
-
startTimeMs: number;
|
|
30
|
-
/** Process ID for crash detection */
|
|
31
|
-
pid: number;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
// ============================================================================
|
|
35
|
-
// StatusWriter
|
|
36
|
-
// ============================================================================
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Manages status file state and write logic for the execution runner.
|
|
40
|
-
*
|
|
41
|
-
* Encapsulates all status-file-related mutable state that was previously
|
|
42
|
-
* tracked as closure variables in runner.ts:
|
|
43
|
-
* - _runStatus, _prd, _currentStory (set via setters before each write)
|
|
44
|
-
* - _consecutiveWriteFailures (BUG-2 failure counter, reset on success)
|
|
45
|
-
*
|
|
46
|
-
* Usage:
|
|
47
|
-
* const sw = new StatusWriter(statusFile, config, { runId, feature, ... });
|
|
48
|
-
* sw.setPrd(prd);
|
|
49
|
-
* sw.setRunStatus("running");
|
|
50
|
-
* sw.setCurrentStory(null);
|
|
51
|
-
* await sw.update(totalCost, iterations);
|
|
52
|
-
*/
|
|
53
|
-
export class StatusWriter {
|
|
54
|
-
private readonly statusFile: string;
|
|
55
|
-
private readonly costLimit: number | null;
|
|
56
|
-
private readonly ctx: StatusWriterContext;
|
|
57
|
-
|
|
58
|
-
// Encapsulated mutable state (was closure vars in runner.ts)
|
|
59
|
-
private _runStatus: RunStateSnapshot["runStatus"] = "running";
|
|
60
|
-
private _prd: PRD | null = null;
|
|
61
|
-
private _currentStory: RunStateSnapshot["currentStory"] = null;
|
|
62
|
-
private _consecutiveWriteFailures = 0; // BUG-2: Track consecutive write failures
|
|
63
|
-
|
|
64
|
-
constructor(statusFile: string, config: NaxConfig, ctx: StatusWriterContext) {
|
|
65
|
-
this.statusFile = statusFile;
|
|
66
|
-
this.costLimit = config.execution.costLimit === Number.POSITIVE_INFINITY ? null : config.execution.costLimit;
|
|
67
|
-
this.ctx = ctx;
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
/** Update the current run status (running / completed / failed / stalled / crashed / precheck-failed) */
|
|
71
|
-
setRunStatus(status: RunStateSnapshot["runStatus"]): void {
|
|
72
|
-
this._runStatus = status;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
/** Update the loaded PRD used for progress counting */
|
|
76
|
-
setPrd(prd: PRD): void {
|
|
77
|
-
this._prd = prd;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
/** Update the currently-executing story info (null between stories) */
|
|
81
|
-
setCurrentStory(story: RunStateSnapshot["currentStory"]): void {
|
|
82
|
-
this._currentStory = story;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
/**
|
|
86
|
-
* Build a RunStateSnapshot from current state + live runner values.
|
|
87
|
-
*
|
|
88
|
-
* Returns null if no PRD has been set yet (status write is a no-op).
|
|
89
|
-
*/
|
|
90
|
-
getSnapshot(totalCost: number, iterations: number): RunStateSnapshot | null {
|
|
91
|
-
if (!this._prd) return null;
|
|
92
|
-
return {
|
|
93
|
-
runId: this.ctx.runId,
|
|
94
|
-
feature: this.ctx.feature,
|
|
95
|
-
startedAt: this.ctx.startedAt,
|
|
96
|
-
runStatus: this._runStatus,
|
|
97
|
-
dryRun: this.ctx.dryRun,
|
|
98
|
-
pid: this.ctx.pid,
|
|
99
|
-
prd: this._prd,
|
|
100
|
-
totalCost,
|
|
101
|
-
costLimit: this.costLimit,
|
|
102
|
-
iterations,
|
|
103
|
-
startTimeMs: this.ctx.startTimeMs,
|
|
104
|
-
currentStory: this._currentStory,
|
|
105
|
-
};
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
/**
|
|
109
|
-
* Write the current status to disk (atomic via .tmp + rename).
|
|
110
|
-
*
|
|
111
|
-
* No-ops if _prd has not been set.
|
|
112
|
-
* On failure, logs a warning/error and increments the BUG-2 failure counter.
|
|
113
|
-
* Counter resets to 0 on next successful write.
|
|
114
|
-
*
|
|
115
|
-
* @param totalCost - Accumulated cost at this write point
|
|
116
|
-
* @param iterations - Loop iteration count at this write point
|
|
117
|
-
* @param overrides - Optional partial snapshot overrides (spread last)
|
|
118
|
-
*/
|
|
119
|
-
async update(totalCost: number, iterations: number, overrides: Partial<RunStateSnapshot> = {}): Promise<void> {
|
|
120
|
-
if (!this._prd) return;
|
|
121
|
-
const safeLogger = getSafeLogger();
|
|
122
|
-
try {
|
|
123
|
-
const base = this.getSnapshot(totalCost, iterations);
|
|
124
|
-
if (!base) {
|
|
125
|
-
throw new Error("Failed to get snapshot");
|
|
126
|
-
}
|
|
127
|
-
const state: RunStateSnapshot = { ...base, ...overrides };
|
|
128
|
-
await writeStatusFile(this.statusFile, buildStatusSnapshot(state));
|
|
129
|
-
this._consecutiveWriteFailures = 0; // Reset counter on success
|
|
130
|
-
} catch (err) {
|
|
131
|
-
this._consecutiveWriteFailures++;
|
|
132
|
-
const logLevel = this._consecutiveWriteFailures >= 3 ? "error" : "warn";
|
|
133
|
-
safeLogger?.[logLevel]("status-file", "Failed to write status file (non-fatal)", {
|
|
134
|
-
path: this.statusFile,
|
|
135
|
-
error: (err as Error).message,
|
|
136
|
-
consecutiveFailures: this._consecutiveWriteFailures,
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
/**
|
|
142
|
-
* Write the current status snapshot to feature-level status.json file.
|
|
143
|
-
*
|
|
144
|
-
* Called on run completion, failure, or crash to persist the final state
|
|
145
|
-
* to <featureDir>/status.json. Uses the same NaxStatusFile schema as
|
|
146
|
-
* the project-level status file.
|
|
147
|
-
*
|
|
148
|
-
* No-ops if _prd has not been set.
|
|
149
|
-
* On failure, logs a warning/error but does not throw (non-fatal).
|
|
150
|
-
*
|
|
151
|
-
* @param featureDir - Feature directory (e.g., nax/features/auth-system)
|
|
152
|
-
* @param totalCost - Accumulated cost at this write point
|
|
153
|
-
* @param iterations - Loop iteration count at this write point
|
|
154
|
-
* @param overrides - Optional partial snapshot overrides (spread last)
|
|
155
|
-
*/
|
|
156
|
-
async writeFeatureStatus(
|
|
157
|
-
featureDir: string,
|
|
158
|
-
totalCost: number,
|
|
159
|
-
iterations: number,
|
|
160
|
-
overrides: Partial<RunStateSnapshot> = {},
|
|
161
|
-
): Promise<void> {
|
|
162
|
-
if (!this._prd) return;
|
|
163
|
-
const safeLogger = getSafeLogger();
|
|
164
|
-
const featureStatusPath = join(featureDir, "status.json");
|
|
165
|
-
|
|
166
|
-
try {
|
|
167
|
-
const base = this.getSnapshot(totalCost, iterations);
|
|
168
|
-
if (!base) {
|
|
169
|
-
throw new Error("Failed to get snapshot");
|
|
170
|
-
}
|
|
171
|
-
const state: RunStateSnapshot = { ...base, ...overrides };
|
|
172
|
-
await writeStatusFile(featureStatusPath, buildStatusSnapshot(state));
|
|
173
|
-
safeLogger?.debug("status-file", "Feature status written", { path: featureStatusPath });
|
|
174
|
-
} catch (err) {
|
|
175
|
-
safeLogger?.warn("status-file", "Failed to write feature status file (non-fatal)", {
|
|
176
|
-
path: featureStatusPath,
|
|
177
|
-
error: (err as Error).message,
|
|
178
|
-
});
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
}
|
|
@@ -1,266 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Story Context Building
|
|
3
|
-
*
|
|
4
|
-
* Extracted from helpers.ts: context building and story readiness functions.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import type { NaxConfig } from "../config";
|
|
8
|
-
import { buildContext, formatContextAsMarkdown } from "../context";
|
|
9
|
-
import type { BuiltContext, ContextBudget, StoryContext } from "../context";
|
|
10
|
-
import type { HookContext } from "../hooks";
|
|
11
|
-
import { getLogger } from "../logger";
|
|
12
|
-
import type { PRD, UserStory } from "../prd";
|
|
13
|
-
|
|
14
|
-
/** Safely get logger instance, returns null if not initialized */
|
|
15
|
-
function getSafeLogger() {
|
|
16
|
-
try {
|
|
17
|
-
return getLogger();
|
|
18
|
-
} catch {
|
|
19
|
-
return null;
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
/**
|
|
24
|
-
* Maximum tokens allowed in context budget for Claude agents.
|
|
25
|
-
*
|
|
26
|
-
* Claude has 200k token context window, but we reserve space for system prompt,
|
|
27
|
-
* agent instructions, conversation history, and output buffer. 100k token limit
|
|
28
|
-
* for story context is conservative but safe, leaving 100k tokens for agent
|
|
29
|
-
* reasoning and responses.
|
|
30
|
-
*/
|
|
31
|
-
const CONTEXT_MAX_TOKENS = 100_000;
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Tokens reserved for agent instructions and prompts.
|
|
35
|
-
*
|
|
36
|
-
* Agent needs space for task instructions, TDD prompts, hook output.
|
|
37
|
-
* 10k token reservation is safe upper bound for all instruction scenarios.
|
|
38
|
-
*/
|
|
39
|
-
const CONTEXT_RESERVED_TOKENS = 10_000;
|
|
40
|
-
|
|
41
|
-
/** Result from executing a batch or single story */
|
|
42
|
-
export interface ExecutionResult {
|
|
43
|
-
success: boolean;
|
|
44
|
-
cost: number;
|
|
45
|
-
storiesProcessed: string[];
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
/**
|
|
49
|
-
* Build a hook context object
|
|
50
|
-
*
|
|
51
|
-
* @param feature - Feature name
|
|
52
|
-
* @param opts - Optional context fields to override
|
|
53
|
-
* @returns Hook context with event set to "on-start" (overridden by fireHook)
|
|
54
|
-
*/
|
|
55
|
-
export function hookCtx(feature: string, opts?: Partial<Omit<HookContext, "event" | "feature">>): HookContext {
|
|
56
|
-
return {
|
|
57
|
-
event: "on-start", // overridden by fireHook
|
|
58
|
-
feature,
|
|
59
|
-
...opts,
|
|
60
|
-
};
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Maybe build context if enabled
|
|
65
|
-
*
|
|
66
|
-
* @param prd - PRD to build context from
|
|
67
|
-
* @param story - Current story being executed
|
|
68
|
-
* @param config - Ngent config
|
|
69
|
-
* @param useContext - Whether to build context
|
|
70
|
-
* @returns Context markdown or undefined if disabled/failed
|
|
71
|
-
*/
|
|
72
|
-
export async function maybeGetContext(
|
|
73
|
-
prd: PRD,
|
|
74
|
-
story: UserStory,
|
|
75
|
-
config: NaxConfig,
|
|
76
|
-
useContext: boolean,
|
|
77
|
-
): Promise<string | undefined> {
|
|
78
|
-
if (!useContext) {
|
|
79
|
-
return undefined;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
const logger = getSafeLogger();
|
|
83
|
-
logger?.debug("context", "Building context...");
|
|
84
|
-
const contextMarkdown = await buildStoryContext(prd, story, config);
|
|
85
|
-
if (contextMarkdown) {
|
|
86
|
-
logger?.debug("context", "Context built successfully");
|
|
87
|
-
}
|
|
88
|
-
return contextMarkdown;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
/**
|
|
92
|
-
* Build story context for context builder
|
|
93
|
-
*
|
|
94
|
-
* @param prd - PRD containing all stories
|
|
95
|
-
* @param story - Current story to build context for
|
|
96
|
-
* @param config - Ngent config
|
|
97
|
-
* @returns Context markdown or undefined if no context available
|
|
98
|
-
*/
|
|
99
|
-
export async function buildStoryContext(prd: PRD, story: UserStory, _config: NaxConfig): Promise<string | undefined> {
|
|
100
|
-
try {
|
|
101
|
-
const storyContext: StoryContext = {
|
|
102
|
-
prd,
|
|
103
|
-
currentStoryId: story.id,
|
|
104
|
-
workdir: process.cwd(),
|
|
105
|
-
config: _config,
|
|
106
|
-
};
|
|
107
|
-
|
|
108
|
-
const budget: ContextBudget = {
|
|
109
|
-
maxTokens: CONTEXT_MAX_TOKENS,
|
|
110
|
-
reservedForInstructions: CONTEXT_RESERVED_TOKENS,
|
|
111
|
-
availableForContext: CONTEXT_MAX_TOKENS - CONTEXT_RESERVED_TOKENS,
|
|
112
|
-
};
|
|
113
|
-
|
|
114
|
-
const built = await buildContext(storyContext, budget);
|
|
115
|
-
|
|
116
|
-
if (built.elements.length === 0) {
|
|
117
|
-
return undefined;
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
return formatContextAsMarkdown(built);
|
|
121
|
-
} catch (error) {
|
|
122
|
-
const logger = getSafeLogger();
|
|
123
|
-
logger?.warn("context", "Context builder failed", {
|
|
124
|
-
error: (error as Error).message,
|
|
125
|
-
});
|
|
126
|
-
return undefined;
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
/**
|
|
131
|
-
* Load package-level context.md content if it exists.
|
|
132
|
-
*
|
|
133
|
-
* Reads <packageWorkdir>/nax/context.md and returns its content, or null
|
|
134
|
-
* if the file does not exist.
|
|
135
|
-
*
|
|
136
|
-
* @internal
|
|
137
|
-
*/
|
|
138
|
-
async function loadPackageContextMd(packageWorkdir: string): Promise<string | null> {
|
|
139
|
-
const contextPath = `${packageWorkdir}/nax/context.md`;
|
|
140
|
-
const file = Bun.file(contextPath);
|
|
141
|
-
if (!(await file.exists())) return null;
|
|
142
|
-
return file.text();
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
/**
|
|
146
|
-
* Build story context returning both markdown and element-level data.
|
|
147
|
-
* Used by `nax prompts` CLI for accurate frontmatter token counts.
|
|
148
|
-
*
|
|
149
|
-
* When `packageWorkdir` is provided (absolute path of story.workdir),
|
|
150
|
-
* appends the package-level nax/context.md after the root context.
|
|
151
|
-
*/
|
|
152
|
-
export async function buildStoryContextFull(
|
|
153
|
-
prd: PRD,
|
|
154
|
-
story: UserStory,
|
|
155
|
-
config: NaxConfig,
|
|
156
|
-
packageWorkdir?: string,
|
|
157
|
-
): Promise<{ markdown: string; builtContext: BuiltContext } | undefined> {
|
|
158
|
-
try {
|
|
159
|
-
const storyContext: StoryContext = {
|
|
160
|
-
prd,
|
|
161
|
-
currentStoryId: story.id,
|
|
162
|
-
workdir: process.cwd(),
|
|
163
|
-
config,
|
|
164
|
-
};
|
|
165
|
-
|
|
166
|
-
const budget: ContextBudget = {
|
|
167
|
-
maxTokens: CONTEXT_MAX_TOKENS,
|
|
168
|
-
reservedForInstructions: CONTEXT_RESERVED_TOKENS,
|
|
169
|
-
availableForContext: CONTEXT_MAX_TOKENS - CONTEXT_RESERVED_TOKENS,
|
|
170
|
-
};
|
|
171
|
-
|
|
172
|
-
const built = await buildContext(storyContext, budget);
|
|
173
|
-
|
|
174
|
-
// MW-003: append package-level context.md if workdir is set
|
|
175
|
-
let packageSection = "";
|
|
176
|
-
if (packageWorkdir) {
|
|
177
|
-
const pkgContent = await loadPackageContextMd(packageWorkdir);
|
|
178
|
-
if (pkgContent) {
|
|
179
|
-
packageSection = `\n---\n\n${pkgContent.trim()}`;
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
if (built.elements.length === 0 && !packageSection) {
|
|
184
|
-
return undefined;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
const baseMarkdown = built.elements.length > 0 ? formatContextAsMarkdown(built) : "";
|
|
188
|
-
const markdown = packageSection ? `${baseMarkdown}${packageSection}` : baseMarkdown;
|
|
189
|
-
|
|
190
|
-
return { markdown, builtContext: built };
|
|
191
|
-
} catch (error) {
|
|
192
|
-
const logger = getSafeLogger();
|
|
193
|
-
logger?.warn("context", "Context builder failed", {
|
|
194
|
-
error: (error as Error).message,
|
|
195
|
-
});
|
|
196
|
-
return undefined;
|
|
197
|
-
}
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
/**
|
|
201
|
-
* Get all stories that are ready to execute (pending, dependencies satisfied)
|
|
202
|
-
*
|
|
203
|
-
* @param prd - PRD containing all stories
|
|
204
|
-
* @returns Array of stories that can be executed now
|
|
205
|
-
*/
|
|
206
|
-
export function getAllReadyStories(prd: PRD): UserStory[] {
|
|
207
|
-
const completedIds = new Set(prd.userStories.filter((s) => s.passes || s.status === "skipped").map((s) => s.id));
|
|
208
|
-
|
|
209
|
-
const logger = getSafeLogger();
|
|
210
|
-
logger?.debug("routing", "getAllReadyStories: completed set", {
|
|
211
|
-
completedIds: [...completedIds],
|
|
212
|
-
totalStories: prd.userStories.length,
|
|
213
|
-
});
|
|
214
|
-
|
|
215
|
-
return prd.userStories.filter(
|
|
216
|
-
(s) =>
|
|
217
|
-
!s.passes &&
|
|
218
|
-
s.status !== "skipped" &&
|
|
219
|
-
s.status !== "failed" &&
|
|
220
|
-
s.status !== "paused" &&
|
|
221
|
-
s.status !== "blocked" &&
|
|
222
|
-
s.dependencies.every((dep) => completedIds.has(dep)),
|
|
223
|
-
);
|
|
224
|
-
}
|
|
225
|
-
|
|
226
|
-
/** Story counts for progress display */
|
|
227
|
-
export interface StoryCounts {
|
|
228
|
-
total: number;
|
|
229
|
-
passed: number;
|
|
230
|
-
failed: number;
|
|
231
|
-
pending: number;
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* Format a progress line with counts, cost, and ETA
|
|
236
|
-
*
|
|
237
|
-
* @param counts - Story counts (total, passed, failed, pending)
|
|
238
|
-
* @param totalCost - Total cost so far
|
|
239
|
-
* @param costLimit - Cost limit from config
|
|
240
|
-
* @param elapsedMs - Elapsed time in milliseconds
|
|
241
|
-
* @param totalStories - Total number of stories (for ETA calculation)
|
|
242
|
-
* @returns Formatted progress string
|
|
243
|
-
*/
|
|
244
|
-
export function formatProgress(
|
|
245
|
-
counts: StoryCounts,
|
|
246
|
-
totalCost: number,
|
|
247
|
-
costLimit: number,
|
|
248
|
-
elapsedMs: number,
|
|
249
|
-
totalStories: number,
|
|
250
|
-
): string {
|
|
251
|
-
const completedStories = counts.passed + counts.failed;
|
|
252
|
-
const remainingStories = totalStories - completedStories;
|
|
253
|
-
|
|
254
|
-
// Calculate ETA from average story duration
|
|
255
|
-
let etaText = "calculating...";
|
|
256
|
-
if (completedStories > 0 && remainingStories > 0) {
|
|
257
|
-
const avgDurationPerStory = elapsedMs / completedStories;
|
|
258
|
-
const etaMs = avgDurationPerStory * remainingStories;
|
|
259
|
-
const etaMinutes = Math.round(etaMs / 1000 / 60);
|
|
260
|
-
etaText = `~${etaMinutes} min remaining`;
|
|
261
|
-
} else if (remainingStories === 0) {
|
|
262
|
-
etaText = "complete";
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
return `Progress: ${completedStories}/${totalStories} stories | ${counts.passed} passed | ${counts.failed} failed | $${totalCost.toFixed(2)}/$${costLimit.toFixed(2)} | ${etaText}`;
|
|
266
|
-
}
|