@mknightzzz/stw 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +277 -0
- package/dist/agentic-fallback.d.ts +3 -0
- package/dist/agentic-fallback.js +32 -0
- package/dist/agentic-fallback.js.map +1 -0
- package/dist/agentic-prompt.d.ts +2 -0
- package/dist/agentic-prompt.js +68 -0
- package/dist/agentic-prompt.js.map +1 -0
- package/dist/agentic-runtime.d.ts +48 -0
- package/dist/agentic-runtime.js +149 -0
- package/dist/agentic-runtime.js.map +1 -0
- package/dist/agentic-types.d.ts +37 -0
- package/dist/agentic-types.js +2 -0
- package/dist/agentic-types.js.map +1 -0
- package/dist/agents.d.ts +7 -0
- package/dist/agents.js +2 -0
- package/dist/agents.js.map +1 -0
- package/dist/assignments.d.ts +7 -0
- package/dist/assignments.js +125 -0
- package/dist/assignments.js.map +1 -0
- package/dist/checkpoint.d.ts +35 -0
- package/dist/checkpoint.js +78 -0
- package/dist/checkpoint.js.map +1 -0
- package/dist/circuit-breaker.d.ts +17 -0
- package/dist/circuit-breaker.js +65 -0
- package/dist/circuit-breaker.js.map +1 -0
- package/dist/claim.d.ts +6 -0
- package/dist/claim.js +135 -0
- package/dist/claim.js.map +1 -0
- package/dist/clarity-gate.d.ts +12 -0
- package/dist/clarity-gate.js +83 -0
- package/dist/clarity-gate.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +38 -0
- package/dist/cli.js.map +1 -0
- package/dist/command-dispatch.d.ts +45 -0
- package/dist/command-dispatch.js +206 -0
- package/dist/command-dispatch.js.map +1 -0
- package/dist/command-parser.d.ts +11 -0
- package/dist/command-parser.js +101 -0
- package/dist/command-parser.js.map +1 -0
- package/dist/commands/clean.d.ts +10 -0
- package/dist/commands/clean.js +133 -0
- package/dist/commands/clean.js.map +1 -0
- package/dist/commands/execution.d.ts +2 -0
- package/dist/commands/execution.js +327 -0
- package/dist/commands/execution.js.map +1 -0
- package/dist/commands/go.d.ts +2 -0
- package/dist/commands/go.js +197 -0
- package/dist/commands/go.js.map +1 -0
- package/dist/commands/helpers.d.ts +44 -0
- package/dist/commands/helpers.js +231 -0
- package/dist/commands/helpers.js.map +1 -0
- package/dist/commands/idea.d.ts +2 -0
- package/dist/commands/idea.js +89 -0
- package/dist/commands/idea.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.js +94 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/integration.d.ts +7 -0
- package/dist/commands/integration.js +139 -0
- package/dist/commands/integration.js.map +1 -0
- package/dist/commands/maintenance.d.ts +2 -0
- package/dist/commands/maintenance.js +301 -0
- package/dist/commands/maintenance.js.map +1 -0
- package/dist/commands/run.d.ts +2 -0
- package/dist/commands/run.js +356 -0
- package/dist/commands/run.js.map +1 -0
- package/dist/commands/setup.d.ts +2 -0
- package/dist/commands/setup.js +198 -0
- package/dist/commands/setup.js.map +1 -0
- package/dist/commands/spec.d.ts +2 -0
- package/dist/commands/spec.js +35 -0
- package/dist/commands/spec.js.map +1 -0
- package/dist/commands/stats.d.ts +2 -0
- package/dist/commands/stats.js +80 -0
- package/dist/commands/stats.js.map +1 -0
- package/dist/commands/task-ops.d.ts +2 -0
- package/dist/commands/task-ops.js +406 -0
- package/dist/commands/task-ops.js.map +1 -0
- package/dist/config.d.ts +18 -0
- package/dist/config.js +338 -0
- package/dist/config.js.map +1 -0
- package/dist/cost.d.ts +30 -0
- package/dist/cost.js +167 -0
- package/dist/cost.js.map +1 -0
- package/dist/crash-recovery.d.ts +9 -0
- package/dist/crash-recovery.js +42 -0
- package/dist/crash-recovery.js.map +1 -0
- package/dist/diagnostic.d.ts +48 -0
- package/dist/diagnostic.js +328 -0
- package/dist/diagnostic.js.map +1 -0
- package/dist/doctor.d.ts +31 -0
- package/dist/doctor.js +225 -0
- package/dist/doctor.js.map +1 -0
- package/dist/drift.d.ts +11 -0
- package/dist/drift.js +57 -0
- package/dist/drift.js.map +1 -0
- package/dist/git-utils.d.ts +20 -0
- package/dist/git-utils.js +206 -0
- package/dist/git-utils.js.map +1 -0
- package/dist/gitlab.d.ts +54 -0
- package/dist/gitlab.js +101 -0
- package/dist/gitlab.js.map +1 -0
- package/dist/idea.d.ts +35 -0
- package/dist/idea.js +251 -0
- package/dist/idea.js.map +1 -0
- package/dist/import-resolution.d.ts +13 -0
- package/dist/import-resolution.js +111 -0
- package/dist/import-resolution.js.map +1 -0
- package/dist/inbox-renderer.d.ts +2 -0
- package/dist/inbox-renderer.js +67 -0
- package/dist/inbox-renderer.js.map +1 -0
- package/dist/init.d.ts +105 -0
- package/dist/init.js +235 -0
- package/dist/init.js.map +1 -0
- package/dist/llm-reviewer.d.ts +14 -0
- package/dist/llm-reviewer.js +109 -0
- package/dist/llm-reviewer.js.map +1 -0
- package/dist/lock.d.ts +26 -0
- package/dist/lock.js +76 -0
- package/dist/lock.js.map +1 -0
- package/dist/logger.d.ts +24 -0
- package/dist/logger.js +40 -0
- package/dist/logger.js.map +1 -0
- package/dist/math-utils.d.ts +2 -0
- package/dist/math-utils.js +7 -0
- package/dist/math-utils.js.map +1 -0
- package/dist/mechanical-review.d.ts +30 -0
- package/dist/mechanical-review.js +76 -0
- package/dist/mechanical-review.js.map +1 -0
- package/dist/merge.d.ts +83 -0
- package/dist/merge.js +363 -0
- package/dist/merge.js.map +1 -0
- package/dist/parallel.d.ts +35 -0
- package/dist/parallel.js +214 -0
- package/dist/parallel.js.map +1 -0
- package/dist/plan-validation.d.ts +19 -0
- package/dist/plan-validation.js +253 -0
- package/dist/plan-validation.js.map +1 -0
- package/dist/planner-prompt.d.ts +33 -0
- package/dist/planner-prompt.js +244 -0
- package/dist/planner-prompt.js.map +1 -0
- package/dist/planner.d.ts +29 -0
- package/dist/planner.js +511 -0
- package/dist/planner.js.map +1 -0
- package/dist/poller.d.ts +34 -0
- package/dist/poller.js +91 -0
- package/dist/poller.js.map +1 -0
- package/dist/progress.d.ts +34 -0
- package/dist/progress.js +122 -0
- package/dist/progress.js.map +1 -0
- package/dist/prompt-builder.d.ts +51 -0
- package/dist/prompt-builder.js +481 -0
- package/dist/prompt-builder.js.map +1 -0
- package/dist/provider.d.ts +14 -0
- package/dist/provider.js +278 -0
- package/dist/provider.js.map +1 -0
- package/dist/question-handler.d.ts +18 -0
- package/dist/question-handler.js +154 -0
- package/dist/question-handler.js.map +1 -0
- package/dist/question-triage.d.ts +31 -0
- package/dist/question-triage.js +175 -0
- package/dist/question-triage.js.map +1 -0
- package/dist/repo-detection.d.ts +8 -0
- package/dist/repo-detection.js +18 -0
- package/dist/repo-detection.js.map +1 -0
- package/dist/retry-context.d.ts +2 -0
- package/dist/retry-context.js +196 -0
- package/dist/retry-context.js.map +1 -0
- package/dist/router.d.ts +18 -0
- package/dist/router.js +137 -0
- package/dist/router.js.map +1 -0
- package/dist/run-artifact-types.d.ts +43 -0
- package/dist/run-artifact-types.js +2 -0
- package/dist/run-artifact-types.js.map +1 -0
- package/dist/run-summary.d.ts +14 -0
- package/dist/run-summary.js +347 -0
- package/dist/run-summary.js.map +1 -0
- package/dist/run-sync.d.ts +11 -0
- package/dist/run-sync.js +110 -0
- package/dist/run-sync.js.map +1 -0
- package/dist/run.d.ts +26 -0
- package/dist/run.js +150 -0
- package/dist/run.js.map +1 -0
- package/dist/scope-expansion.d.ts +10 -0
- package/dist/scope-expansion.js +117 -0
- package/dist/scope-expansion.js.map +1 -0
- package/dist/scope.d.ts +4 -0
- package/dist/scope.js +37 -0
- package/dist/scope.js.map +1 -0
- package/dist/scorecard.d.ts +18 -0
- package/dist/scorecard.js +128 -0
- package/dist/scorecard.js.map +1 -0
- package/dist/spec-templates.d.ts +2 -0
- package/dist/spec-templates.js +285 -0
- package/dist/spec-templates.js.map +1 -0
- package/dist/spec-validator.d.ts +8 -0
- package/dist/spec-validator.js +144 -0
- package/dist/spec-validator.js.map +1 -0
- package/dist/status.d.ts +68 -0
- package/dist/status.js +261 -0
- package/dist/status.js.map +1 -0
- package/dist/storage.d.ts +9 -0
- package/dist/storage.js +35 -0
- package/dist/storage.js.map +1 -0
- package/dist/task-executor-completion.d.ts +12 -0
- package/dist/task-executor-completion.js +67 -0
- package/dist/task-executor-completion.js.map +1 -0
- package/dist/task-executor-fallback.d.ts +20 -0
- package/dist/task-executor-fallback.js +12 -0
- package/dist/task-executor-fallback.js.map +1 -0
- package/dist/task-executor.d.ts +34 -0
- package/dist/task-executor.js +521 -0
- package/dist/task-executor.js.map +1 -0
- package/dist/task-graph.d.ts +11 -0
- package/dist/task-graph.js +226 -0
- package/dist/task-graph.js.map +1 -0
- package/dist/task-pipeline-helpers.d.ts +45 -0
- package/dist/task-pipeline-helpers.js +160 -0
- package/dist/task-pipeline-helpers.js.map +1 -0
- package/dist/task-review.d.ts +51 -0
- package/dist/task-review.js +410 -0
- package/dist/task-review.js.map +1 -0
- package/dist/transitions.d.ts +13 -0
- package/dist/transitions.js +104 -0
- package/dist/transitions.js.map +1 -0
- package/dist/types.d.ts +405 -0
- package/dist/types.js +101 -0
- package/dist/types.js.map +1 -0
- package/dist/utils.d.ts +1 -0
- package/dist/utils.js +23 -0
- package/dist/utils.js.map +1 -0
- package/dist/validation.d.ts +19 -0
- package/dist/validation.js +73 -0
- package/dist/validation.js.map +1 -0
- package/dist/worker-response.d.ts +12 -0
- package/dist/worker-response.js +60 -0
- package/dist/worker-response.js.map +1 -0
- package/dist/worker-runner.d.ts +19 -0
- package/dist/worker-runner.js +347 -0
- package/dist/worker-runner.js.map +1 -0
- package/dist/worktree-cleanup.d.ts +44 -0
- package/dist/worktree-cleanup.js +325 -0
- package/dist/worktree-cleanup.js.map +1 -0
- package/dist/worktree.d.ts +22 -0
- package/dist/worktree.js +213 -0
- package/dist/worktree.js.map +1 -0
- package/examples/spec.md +58 -0
- package/package.json +66 -0
|
@@ -0,0 +1,327 @@
|
|
|
1
|
+
import { readdirSync, cpSync, mkdirSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { unlinkSync } from 'node:fs';
|
|
4
|
+
import { loadConfig } from '../config.js';
|
|
5
|
+
import { getResettableSkippedTaskIds } from '../task-graph.js';
|
|
6
|
+
import { buildRunRecoveryReport, readTaskStatus, recordRunEvent, writeRunState, writeTaskStatus } from '../status.js';
|
|
7
|
+
import { readJson, writeJson, writeAtomic, fileExists, runDir, taskDir, ensureDir } from '../storage.js';
|
|
8
|
+
import { executeTask } from '../task-executor.js';
|
|
9
|
+
import { clearCheckpointsFrom } from '../checkpoint.js';
|
|
10
|
+
import { acquireLock, releaseLock } from '../lock.js';
|
|
11
|
+
import { createLogger } from '../logger.js';
|
|
12
|
+
import { claimTask } from '../claim.js';
|
|
13
|
+
import { getStwRoot } from './helpers.js';
|
|
14
|
+
const STOPPABLE_RUN_STATES = new Set(['running', 'pending', 'planning', 'routing']);
|
|
15
|
+
const RETRYABLE_RUN_STATES = new Set(['paused', 'human_review_required', 'running']);
|
|
16
|
+
const RETRYABLE_TASK_STATUSES = new Set(['escalated', 'failed', 'skipped']);
|
|
17
|
+
function findCheckpointTask(stwRoot, runId, tasks) {
|
|
18
|
+
for (const task of tasks) {
|
|
19
|
+
const status = readTaskStatus(stwRoot, runId, task.id);
|
|
20
|
+
const hasStatus = status && !['complete', 'blocked', 'failed', 'escalated'].includes(status.status);
|
|
21
|
+
if (!status || hasStatus) {
|
|
22
|
+
const td = taskDir(stwRoot, runId, task.id);
|
|
23
|
+
if (fileExists(join(td, 'checkpoints'))) {
|
|
24
|
+
return task.id;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
function archiveCurrentAttempt(td) {
|
|
31
|
+
if (!fileExists(td)) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
const candidateNames = [
|
|
35
|
+
'status.json',
|
|
36
|
+
'state.json',
|
|
37
|
+
'question.json',
|
|
38
|
+
'response_raw.txt',
|
|
39
|
+
'checkpoints',
|
|
40
|
+
'diagnostic.json',
|
|
41
|
+
'validation.json',
|
|
42
|
+
'review.json',
|
|
43
|
+
];
|
|
44
|
+
const existing = candidateNames.filter((name) => fileExists(join(td, name)));
|
|
45
|
+
const answerFiles = safeListAnswerFiles(td);
|
|
46
|
+
const filesToArchive = [...existing, ...answerFiles];
|
|
47
|
+
if (filesToArchive.length === 0) {
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
const attemptsDir = join(td, 'attempts');
|
|
51
|
+
ensureDir(attemptsDir);
|
|
52
|
+
const attemptNumber = getNextAttemptNumber(attemptsDir);
|
|
53
|
+
const archiveDir = join(attemptsDir, String(attemptNumber));
|
|
54
|
+
mkdirSync(archiveDir, { recursive: true });
|
|
55
|
+
for (const name of filesToArchive) {
|
|
56
|
+
cpSync(join(td, name), join(archiveDir, name), { recursive: true });
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
function safeListAnswerFiles(td) {
|
|
60
|
+
try {
|
|
61
|
+
return readdirSync(td, { withFileTypes: true })
|
|
62
|
+
.filter((entry) => entry.isFile() && /^answer-\d+\.json$/.test(entry.name))
|
|
63
|
+
.map((entry) => entry.name)
|
|
64
|
+
.sort();
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
return [];
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
function getNextAttemptNumber(attemptsDir) {
|
|
71
|
+
let max = 0;
|
|
72
|
+
try {
|
|
73
|
+
for (const entry of readdirSync(attemptsDir, { withFileTypes: true })) {
|
|
74
|
+
if (!entry.isDirectory())
|
|
75
|
+
continue;
|
|
76
|
+
const value = Number.parseInt(entry.name, 10);
|
|
77
|
+
if (Number.isFinite(value) && value > max) {
|
|
78
|
+
max = value;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
catch { }
|
|
83
|
+
return max + 1;
|
|
84
|
+
}
|
|
85
|
+
function deleteStaleTaskFiles(td) {
|
|
86
|
+
const filesToDelete = ['status.json', 'state.json', 'claim.lock'];
|
|
87
|
+
for (const name of filesToDelete) {
|
|
88
|
+
const filePath = join(td, name);
|
|
89
|
+
try {
|
|
90
|
+
unlinkSync(filePath);
|
|
91
|
+
}
|
|
92
|
+
catch { }
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
function reportInterruptedRunIfDetected(stwRoot, runId, graph) {
|
|
96
|
+
const report = buildRunRecoveryReport(stwRoot, runId, graph.tasks.map((task) => task.id));
|
|
97
|
+
if (!report.stale) {
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
console.error(`Interrupted run detected for ${runId}.`);
|
|
101
|
+
if (report.reason === 'stale_lock') {
|
|
102
|
+
console.error('Primary signal: stale lock file for this run.');
|
|
103
|
+
}
|
|
104
|
+
if (report.reason === 'persisted_in_progress_without_live_lock') {
|
|
105
|
+
console.error('Primary signal: persisted in_progress task status without a live lock for this run.');
|
|
106
|
+
}
|
|
107
|
+
if (report.staleTasks.length > 0) {
|
|
108
|
+
console.error(`Affected tasks: ${report.staleTasks.map((task) => `${task.taskId}${task.taskState ? ` (${task.taskState.state})` : ''}`).join(', ')}`);
|
|
109
|
+
}
|
|
110
|
+
for (const line of report.guidance) {
|
|
111
|
+
console.error(`Guidance: ${line}`);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
function mapExecutionResultToStatus(finalState) {
|
|
115
|
+
switch (finalState) {
|
|
116
|
+
case 'complete':
|
|
117
|
+
return 'complete';
|
|
118
|
+
case 'escalated':
|
|
119
|
+
return 'escalated';
|
|
120
|
+
case 'rejected':
|
|
121
|
+
return 'failed';
|
|
122
|
+
case 'stopped':
|
|
123
|
+
return 'escalated';
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
export function registerExecutionCommands(program) {
|
|
127
|
+
program
|
|
128
|
+
.command('stop <run-id>')
|
|
129
|
+
.description('Stop a running run and transition to paused')
|
|
130
|
+
.action((runId) => {
|
|
131
|
+
const stwRoot = getStwRoot();
|
|
132
|
+
const manifestPath = join(runDir(stwRoot, runId), 'manifest.json');
|
|
133
|
+
if (!fileExists(manifestPath)) {
|
|
134
|
+
console.error(`Run not found: ${runId}`);
|
|
135
|
+
process.exit(1);
|
|
136
|
+
}
|
|
137
|
+
const manifest = readJson(manifestPath);
|
|
138
|
+
if (!STOPPABLE_RUN_STATES.has(manifest.status)) {
|
|
139
|
+
console.error(`Cannot stop run in state: ${manifest.status}`);
|
|
140
|
+
process.exit(1);
|
|
141
|
+
}
|
|
142
|
+
const sentinelPath = join(runDir(stwRoot, runId), 'STOP');
|
|
143
|
+
writeAtomic(sentinelPath, `stopped_at=${new Date().toISOString()}\n`);
|
|
144
|
+
writeRunState(stwRoot, runId, 'paused', { pause_reason: 'manual' });
|
|
145
|
+
recordRunEvent(stwRoot, runId, {
|
|
146
|
+
event_type: 'stop',
|
|
147
|
+
actor: 'human',
|
|
148
|
+
summary: 'Run manually stopped',
|
|
149
|
+
details: 'STOP sentinel written and run paused',
|
|
150
|
+
});
|
|
151
|
+
releaseLock(stwRoot);
|
|
152
|
+
const graphPath = join(runDir(stwRoot, runId), 'task_graph.json');
|
|
153
|
+
if (fileExists(graphPath)) {
|
|
154
|
+
const graph = readJson(graphPath);
|
|
155
|
+
const inProgress = graph.tasks.filter((t) => {
|
|
156
|
+
const st = readTaskStatus(stwRoot, runId, t.id);
|
|
157
|
+
return st && !['complete', 'failed', 'escalated', 'skipped'].includes(st.status);
|
|
158
|
+
});
|
|
159
|
+
if (inProgress.length > 0) {
|
|
160
|
+
console.log(`In-progress tasks: ${inProgress.map((t) => t.id).join(', ')}`);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
console.log(`Run ${runId} stopped (paused)`);
|
|
164
|
+
});
|
|
165
|
+
program
|
|
166
|
+
.command('retry <run-id> <task-id>')
|
|
167
|
+
.description('Reset a failed/escalated task for re-execution')
|
|
168
|
+
.option('--tier <tier>', 'Override model tier for retry')
|
|
169
|
+
.action((runId, taskIdArg, opts) => {
|
|
170
|
+
const stwRoot = getStwRoot();
|
|
171
|
+
const manifestPath = join(runDir(stwRoot, runId), 'manifest.json');
|
|
172
|
+
if (!fileExists(manifestPath)) {
|
|
173
|
+
console.error(`Run not found: ${runId}`);
|
|
174
|
+
process.exit(1);
|
|
175
|
+
}
|
|
176
|
+
const manifest = readJson(manifestPath);
|
|
177
|
+
if (!RETRYABLE_RUN_STATES.has(manifest.status)) {
|
|
178
|
+
console.error(`Cannot retry in run state: ${manifest.status}. Must be paused or human_review_required.`);
|
|
179
|
+
process.exit(1);
|
|
180
|
+
}
|
|
181
|
+
const taskStatus = readTaskStatus(stwRoot, runId, taskIdArg);
|
|
182
|
+
if (!taskStatus) {
|
|
183
|
+
console.error(`No status found for task ${taskIdArg}`);
|
|
184
|
+
process.exit(1);
|
|
185
|
+
}
|
|
186
|
+
if (!RETRYABLE_TASK_STATUSES.has(taskStatus.status)) {
|
|
187
|
+
console.error(`Task ${taskIdArg} is ${taskStatus.status}, not retryable`);
|
|
188
|
+
process.exit(1);
|
|
189
|
+
}
|
|
190
|
+
const td = taskDir(stwRoot, runId, taskIdArg);
|
|
191
|
+
archiveCurrentAttempt(td);
|
|
192
|
+
deleteStaleTaskFiles(td);
|
|
193
|
+
clearCheckpointsFrom(td, 'pre_worker');
|
|
194
|
+
const graphPath = join(runDir(stwRoot, runId), 'task_graph.json');
|
|
195
|
+
if (fileExists(graphPath)) {
|
|
196
|
+
const graph = readJson(graphPath);
|
|
197
|
+
if (opts.tier) {
|
|
198
|
+
const task = graph.tasks.find((t) => t.id === taskIdArg);
|
|
199
|
+
if (task) {
|
|
200
|
+
task.model_tier = opts.tier;
|
|
201
|
+
writeJson(graphPath, graph);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
const currentStates = new Map();
|
|
205
|
+
for (const t of graph.tasks) {
|
|
206
|
+
const st = readTaskStatus(stwRoot, runId, t.id);
|
|
207
|
+
currentStates.set(t.id, st?.status ?? 'pending');
|
|
208
|
+
}
|
|
209
|
+
currentStates.set(taskIdArg, 'pending');
|
|
210
|
+
const toReset = getResettableSkippedTaskIds(graph, taskIdArg, currentStates);
|
|
211
|
+
for (const resetId of toReset) {
|
|
212
|
+
const resetTd = taskDir(stwRoot, runId, resetId);
|
|
213
|
+
archiveCurrentAttempt(resetTd);
|
|
214
|
+
deleteStaleTaskFiles(resetTd);
|
|
215
|
+
}
|
|
216
|
+
if (toReset.length > 0) {
|
|
217
|
+
console.log(`Un-skipped ${toReset.length} downstream task(s): ${toReset.join(', ')}`);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
const sentinelPath = join(runDir(stwRoot, runId), 'STOP');
|
|
221
|
+
try {
|
|
222
|
+
unlinkSync(sentinelPath);
|
|
223
|
+
}
|
|
224
|
+
catch { }
|
|
225
|
+
writeRunState(stwRoot, runId, 'running');
|
|
226
|
+
recordRunEvent(stwRoot, runId, {
|
|
227
|
+
event_type: 'retry',
|
|
228
|
+
actor: 'human',
|
|
229
|
+
task_id: taskIdArg,
|
|
230
|
+
summary: 'Task reset for retry',
|
|
231
|
+
details: opts.tier ? `Tier override: ${opts.tier}` : undefined,
|
|
232
|
+
});
|
|
233
|
+
console.log(`Task ${taskIdArg} reset for retry. Run 'stw run ${runId}' to re-execute.`);
|
|
234
|
+
});
|
|
235
|
+
program
|
|
236
|
+
.command('resume <run-id> [task-id]')
|
|
237
|
+
.description('Resume a task from its last checkpoint')
|
|
238
|
+
.action(async (runId, taskId) => {
|
|
239
|
+
const stwRoot = getStwRoot();
|
|
240
|
+
const runRoot = join(stwRoot, 'runs', runId);
|
|
241
|
+
const manifestPath = join(runRoot, 'manifest.json');
|
|
242
|
+
const graphPath = join(runRoot, 'task_graph.json');
|
|
243
|
+
if (!fileExists(graphPath)) {
|
|
244
|
+
console.error(`No task graph found for run ${runId}`);
|
|
245
|
+
process.exit(1);
|
|
246
|
+
}
|
|
247
|
+
if (!fileExists(manifestPath)) {
|
|
248
|
+
console.error(`Run not found: ${runId}`);
|
|
249
|
+
process.exit(1);
|
|
250
|
+
}
|
|
251
|
+
const manifest = readJson(manifestPath);
|
|
252
|
+
const graph = readJson(graphPath);
|
|
253
|
+
const config = loadConfig(join(stwRoot, 'config.yml'));
|
|
254
|
+
reportInterruptedRunIfDetected(stwRoot, runId, graph);
|
|
255
|
+
const targetId = taskId ?? findCheckpointTask(stwRoot, runId, graph.tasks);
|
|
256
|
+
if (!targetId) {
|
|
257
|
+
console.error('No task with checkpoint found');
|
|
258
|
+
process.exit(1);
|
|
259
|
+
}
|
|
260
|
+
const task = graph.tasks.find((t) => t.id === targetId);
|
|
261
|
+
if (!task) {
|
|
262
|
+
console.error(`Task ${targetId} not found`);
|
|
263
|
+
process.exit(1);
|
|
264
|
+
}
|
|
265
|
+
const lockResult = acquireLock(stwRoot, runId);
|
|
266
|
+
if (!lockResult.acquired) {
|
|
267
|
+
console.error(`Run is locked by PID ${lockResult.existingLock?.data.pid}. Use 'stw unlock' to remove.`);
|
|
268
|
+
process.exit(1);
|
|
269
|
+
}
|
|
270
|
+
const logger = createLogger(join(runDir(stwRoot, runId), 'logs.ndjson'), runId);
|
|
271
|
+
try {
|
|
272
|
+
const claim = await claimTask(stwRoot, runId, task.id, 'stw-executor');
|
|
273
|
+
if (!claim.claimed) {
|
|
274
|
+
console.error(`Task ${task.id} already claimed by ${claim.holder}. Use 'stw unlock' if the holder is dead.`);
|
|
275
|
+
process.exit(1);
|
|
276
|
+
}
|
|
277
|
+
console.log(`Resuming task: ${task.id}`);
|
|
278
|
+
const result = await executeTask(task, config, {
|
|
279
|
+
taskDir: taskDir(stwRoot, runId, task.id),
|
|
280
|
+
repoRoot: manifest.repo_root,
|
|
281
|
+
runBranch: manifest.branch,
|
|
282
|
+
resume: true,
|
|
283
|
+
stwRoot,
|
|
284
|
+
runId,
|
|
285
|
+
logger,
|
|
286
|
+
pricing: config.pricing,
|
|
287
|
+
budget: config.budget,
|
|
288
|
+
});
|
|
289
|
+
const previousStatus = readTaskStatus(stwRoot, runId, task.id);
|
|
290
|
+
const preservedNote = result.preservedWorktreePath ? ` | worktree: ${result.preservedWorktreePath}` : '';
|
|
291
|
+
writeTaskStatus(stwRoot, runId, task.id, {
|
|
292
|
+
agent_id: 'stw-executor',
|
|
293
|
+
status: mapExecutionResultToStatus(result.finalState),
|
|
294
|
+
notes: `attempts: ${result.attempts}${preservedNote}`,
|
|
295
|
+
preserved_worktree_path: result.preservedWorktreePath ?? previousStatus?.preserved_worktree_path,
|
|
296
|
+
});
|
|
297
|
+
recordRunEvent(stwRoot, runId, {
|
|
298
|
+
event_type: 'resume',
|
|
299
|
+
actor: 'stw-executor',
|
|
300
|
+
task_id: task.id,
|
|
301
|
+
summary: 'Task resumed from checkpoint',
|
|
302
|
+
details: `Final state: ${result.finalState}; attempts: ${result.attempts}`,
|
|
303
|
+
metadata: task.planner_metadata?.execution_mode === 'agentic'
|
|
304
|
+
? {
|
|
305
|
+
execution_mode: 'agentic',
|
|
306
|
+
agentic_backend: task.planner_metadata.agentic_backend ?? config.defaults.agentic?.default_backend ?? null,
|
|
307
|
+
}
|
|
308
|
+
: undefined,
|
|
309
|
+
});
|
|
310
|
+
if (result.finalState !== 'complete') {
|
|
311
|
+
recordRunEvent(stwRoot, runId, {
|
|
312
|
+
event_type: 'escalation',
|
|
313
|
+
actor: 'stw-executor',
|
|
314
|
+
task_id: task.id,
|
|
315
|
+
summary: `Task ended in ${result.finalState} after resume`,
|
|
316
|
+
escalation_category: 'human_review',
|
|
317
|
+
});
|
|
318
|
+
}
|
|
319
|
+
console.log(`Task ${task.id} result: ${result.finalState}`);
|
|
320
|
+
console.log(`Attempts: ${result.attempts}`);
|
|
321
|
+
}
|
|
322
|
+
finally {
|
|
323
|
+
releaseLock(stwRoot);
|
|
324
|
+
}
|
|
325
|
+
});
|
|
326
|
+
}
|
|
327
|
+
//# sourceMappingURL=execution.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"execution.js","sourceRoot":"","sources":["../../src/commands/execution.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
import { readFileSync } from 'node:fs';
|
|
2
|
+
import { join, resolve } from 'node:path';
|
|
3
|
+
import { generateSpec, slugify, KNOWN_STACK_PRESETS, ClarityGateError } from '../idea.js';
|
|
4
|
+
import { loadConfig } from '../config.js';
|
|
5
|
+
import { validateSeedDir } from '../worktree.js';
|
|
6
|
+
import { generatePlan, validatePlanScope } from '../planner.js';
|
|
7
|
+
import { estimateCost, recordCost } from '../cost.js';
|
|
8
|
+
import { writeAtomic, writeJson, readJson, fileExists, runDir } from '../storage.js';
|
|
9
|
+
import { getStwRoot, ingestRunFromContext, setupRunArtifacts } from './helpers.js';
|
|
10
|
+
export function registerGoCommand(program) {
|
|
11
|
+
program
|
|
12
|
+
.command('go')
|
|
13
|
+
.description('Run the full pipeline: idea → ingest → plan → assign → run')
|
|
14
|
+
.argument('<description>', 'The rough idea in natural language')
|
|
15
|
+
.option('--template <name>', 'Template override (bugfix, feature, refactor, cleanup, greenfield)')
|
|
16
|
+
.option('--greenfield', 'Generate a spec for a brand-new project (no existing codebase)')
|
|
17
|
+
.option('--stack <preset>', `Stack preset (${KNOWN_STACK_PRESETS.join(', ')}). Implies --greenfield.`)
|
|
18
|
+
.option('--from <dir>', 'Seed greenfield worktree from a local directory. Implies --greenfield.')
|
|
19
|
+
.option('--spec <path>', 'Use an existing spec instead of generating one')
|
|
20
|
+
.option('--dry-run', 'Stop after plan (do not assign or run)')
|
|
21
|
+
.option('--skip-budget-check', 'Skip budget check during run')
|
|
22
|
+
.option('--verbose', 'Stream worker output to console during run')
|
|
23
|
+
.option('--no-clarity-check', 'Skip the idea clarity check (for power users)')
|
|
24
|
+
.action(async (description, opts) => {
|
|
25
|
+
if (opts.stack && !KNOWN_STACK_PRESETS.includes(opts.stack)) {
|
|
26
|
+
console.error(`Unknown stack preset "${opts.stack}". Available: ${KNOWN_STACK_PRESETS.join(', ')}`);
|
|
27
|
+
process.exit(1);
|
|
28
|
+
}
|
|
29
|
+
const seedDir = opts.from ? resolve(opts.from) : undefined;
|
|
30
|
+
if (seedDir) {
|
|
31
|
+
try {
|
|
32
|
+
validateSeedDir(seedDir);
|
|
33
|
+
}
|
|
34
|
+
catch (err) {
|
|
35
|
+
console.error(err instanceof Error ? err.message : String(err));
|
|
36
|
+
process.exit(1);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
const isGreenfield = opts.greenfield || !!opts.stack || !!seedDir;
|
|
40
|
+
const stwRoot = getStwRoot();
|
|
41
|
+
const config = loadConfig(join(stwRoot, 'config.yml'));
|
|
42
|
+
const repoRoot = resolve('.');
|
|
43
|
+
// --- Step 1: Idea → Spec ---
|
|
44
|
+
let specPath;
|
|
45
|
+
if (opts.spec) {
|
|
46
|
+
specPath = opts.spec;
|
|
47
|
+
console.log(`[go] Using existing spec: ${specPath}`);
|
|
48
|
+
}
|
|
49
|
+
else {
|
|
50
|
+
console.log(`[go] Generating spec from idea: "${description}"`);
|
|
51
|
+
if (opts.stack) {
|
|
52
|
+
console.log(`[go] Stack: ${opts.stack}`);
|
|
53
|
+
}
|
|
54
|
+
let ideaResult;
|
|
55
|
+
try {
|
|
56
|
+
ideaResult = await generateSpec({
|
|
57
|
+
idea: description,
|
|
58
|
+
repoRoot,
|
|
59
|
+
config,
|
|
60
|
+
templateOverride: opts.template,
|
|
61
|
+
isGreenfield,
|
|
62
|
+
stack: opts.stack,
|
|
63
|
+
seedDir,
|
|
64
|
+
skipClarityCheck: opts.clarityCheck === false,
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
catch (err) {
|
|
68
|
+
if (err instanceof ClarityGateError) {
|
|
69
|
+
console.error(`\n[go] Idea rejected — not specific enough to generate a spec.\n`);
|
|
70
|
+
console.error(`Feedback: ${err.feedback}`);
|
|
71
|
+
if (err.suggestedIdea) {
|
|
72
|
+
console.error(`\nYou can use this as a starting point:\n "${err.suggestedIdea}"`);
|
|
73
|
+
}
|
|
74
|
+
process.exit(1);
|
|
75
|
+
}
|
|
76
|
+
throw err;
|
|
77
|
+
}
|
|
78
|
+
specPath = join('specs', `${slugify(description)}.md`);
|
|
79
|
+
writeAtomic(specPath, ideaResult.spec);
|
|
80
|
+
console.log(`[go] Spec generated → ${specPath}`);
|
|
81
|
+
if (ideaResult.model) {
|
|
82
|
+
console.log(`[go] Model: ${ideaResult.model}`);
|
|
83
|
+
}
|
|
84
|
+
if (ideaResult.usage) {
|
|
85
|
+
console.log(`[go] Tokens: ${ideaResult.usage.input_tokens} in / ${ideaResult.usage.output_tokens} out`);
|
|
86
|
+
}
|
|
87
|
+
if (ideaResult.latency_ms) {
|
|
88
|
+
console.log(`[go] Latency: ${(ideaResult.latency_ms / 1000).toFixed(1)}s`);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
// --- Step 2: Ingest → Run ---
|
|
92
|
+
console.log(`[go] Ingesting spec...`);
|
|
93
|
+
const ingestResult = ingestRunFromContext({
|
|
94
|
+
repoRoot,
|
|
95
|
+
stwRoot,
|
|
96
|
+
fallbackSpecPath: specPath,
|
|
97
|
+
greenfield: isGreenfield,
|
|
98
|
+
seedDir,
|
|
99
|
+
});
|
|
100
|
+
if (!ingestResult.ok || !ingestResult.manifest) {
|
|
101
|
+
console.error(`[go] Ingest failed: ${ingestResult.message}`);
|
|
102
|
+
for (const detail of ingestResult.details ?? []) {
|
|
103
|
+
console.error(` - ${detail}`);
|
|
104
|
+
}
|
|
105
|
+
process.exit(1);
|
|
106
|
+
}
|
|
107
|
+
const runId = ingestResult.manifest.run_id;
|
|
108
|
+
console.log(`[go] Run created: ${runId}`);
|
|
109
|
+
// --- Step 3: Plan ---
|
|
110
|
+
console.log(`[go] Planning...`);
|
|
111
|
+
const spec = readFileSync(resolve(repoRoot, specPath), 'utf-8');
|
|
112
|
+
const planResult = await generatePlan({
|
|
113
|
+
spec,
|
|
114
|
+
repoRoot,
|
|
115
|
+
config,
|
|
116
|
+
runId,
|
|
117
|
+
});
|
|
118
|
+
if (planResult.warnings.length > 0) {
|
|
119
|
+
for (const warning of planResult.warnings) {
|
|
120
|
+
console.warn(`[go] Plan warning: ${warning}`);
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
const scopeValidation = validatePlanScope(planResult.task_graph, repoRoot, {
|
|
124
|
+
max_scope_files_per_task: config.defaults.max_scope_files_per_task,
|
|
125
|
+
max_tasks_per_spec: config.defaults.max_tasks_per_spec,
|
|
126
|
+
});
|
|
127
|
+
if (!scopeValidation.valid) {
|
|
128
|
+
// The planner already retried scope errors — demote remaining errors to warnings
|
|
129
|
+
// so the pipeline can proceed. Workers will still enforce scope at execution time.
|
|
130
|
+
for (const error of scopeValidation.errors) {
|
|
131
|
+
console.warn(`[go] Scope warning (post-retry): ${error}`);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
writeJson(join(stwRoot, 'runs', runId, 'task_graph.json'), planResult.task_graph);
|
|
135
|
+
if (planResult.usage && config.pricing) {
|
|
136
|
+
const plannerModel = planResult.model ?? 'unknown';
|
|
137
|
+
const plannerCostUsd = estimateCost(config.pricing, plannerModel, planResult.usage.input_tokens, planResult.usage.output_tokens);
|
|
138
|
+
try {
|
|
139
|
+
recordCost(stwRoot, runId, '_planner', {
|
|
140
|
+
task_id: '_planner',
|
|
141
|
+
model: plannerModel,
|
|
142
|
+
provider: 'unknown',
|
|
143
|
+
input_tokens: planResult.usage.input_tokens,
|
|
144
|
+
output_tokens: planResult.usage.output_tokens,
|
|
145
|
+
cost_usd: plannerCostUsd,
|
|
146
|
+
tier: 'strong',
|
|
147
|
+
latency_ms: planResult.latency_ms ?? 0,
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
catch { }
|
|
151
|
+
}
|
|
152
|
+
const taskCount = planResult.task_graph.tasks.length;
|
|
153
|
+
const taskIds = planResult.task_graph.tasks.map((t) => t.id).join(', ');
|
|
154
|
+
console.log(`[go] Plan: ${taskCount} tasks (${taskIds})`);
|
|
155
|
+
if (opts.dryRun) {
|
|
156
|
+
console.log(`[go] Dry run — stopping before assign/run.`);
|
|
157
|
+
console.log(`[go] To continue: stw assign ${runId} && stw run ${runId}`);
|
|
158
|
+
return;
|
|
159
|
+
}
|
|
160
|
+
// --- Step 4: Assign ---
|
|
161
|
+
console.log(`[go] Assigning...`);
|
|
162
|
+
const setupResult = setupRunArtifacts({ stwRoot, runId, repoRoot });
|
|
163
|
+
if (!setupResult.assignmentsWritten) {
|
|
164
|
+
console.error(`[go] Assignment failed — no task graph found after planning.`);
|
|
165
|
+
process.exit(1);
|
|
166
|
+
}
|
|
167
|
+
console.log(`[go] Assigned ${taskCount} tasks`);
|
|
168
|
+
// --- Step 5: Run ---
|
|
169
|
+
console.log(`[go] Running...`);
|
|
170
|
+
// Shell out to `stw run` reusing the current entry point (works both in dev via tsx and after npm install)
|
|
171
|
+
const { execFileSync } = await import('node:child_process');
|
|
172
|
+
const entryScript = process.argv[1] ?? 'stw';
|
|
173
|
+
const args = [entryScript, 'run', runId];
|
|
174
|
+
if (opts.skipBudgetCheck)
|
|
175
|
+
args.push('--skip-budget-check');
|
|
176
|
+
if (opts.verbose)
|
|
177
|
+
args.push('--verbose');
|
|
178
|
+
try {
|
|
179
|
+
execFileSync(process.execPath, args, {
|
|
180
|
+
cwd: repoRoot,
|
|
181
|
+
encoding: 'utf-8',
|
|
182
|
+
stdio: 'inherit',
|
|
183
|
+
timeout: 30 * 60 * 1000, // 30 min max
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
catch {
|
|
187
|
+
// run command exits non-zero on escalation — check final status
|
|
188
|
+
const manifestPath = join(runDir(stwRoot, runId), 'manifest.json');
|
|
189
|
+
if (fileExists(manifestPath)) {
|
|
190
|
+
const manifest = readJson(manifestPath);
|
|
191
|
+
console.log(`[go] Run finished with status: ${manifest.status}`);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
console.log(`[go] Pipeline complete for run ${runId}`);
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
//# sourceMappingURL=go.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"go.js","sourceRoot":"","sources":["../../src/commands/go.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import type { RunManifest, TaskGraph } from '../types.js';
|
|
2
|
+
import type { AgentDefinition } from '../agents.js';
|
|
3
|
+
export declare function getStwRoot(): string;
|
|
4
|
+
export declare function delay(ms: number): Promise<void>;
|
|
5
|
+
export declare const LOG_LEVELS: readonly ["debug", "info", "warn", "error"];
|
|
6
|
+
export type LogLevel = (typeof LOG_LEVELS)[number];
|
|
7
|
+
export interface LogEntry {
|
|
8
|
+
ts?: string;
|
|
9
|
+
level?: string;
|
|
10
|
+
task_id?: string | null;
|
|
11
|
+
message?: string;
|
|
12
|
+
}
|
|
13
|
+
export interface IngestContext {
|
|
14
|
+
repoRoot: string;
|
|
15
|
+
stwRoot: string;
|
|
16
|
+
noteBody?: string;
|
|
17
|
+
notePath?: string;
|
|
18
|
+
fallbackSpecPath?: string;
|
|
19
|
+
baseBranch?: string;
|
|
20
|
+
greenfield?: boolean;
|
|
21
|
+
seedDir?: string;
|
|
22
|
+
}
|
|
23
|
+
export interface IngestResult {
|
|
24
|
+
ok: boolean;
|
|
25
|
+
manifest?: RunManifest;
|
|
26
|
+
message: string;
|
|
27
|
+
details?: string[];
|
|
28
|
+
}
|
|
29
|
+
export declare function getDefaultAgents(): AgentDefinition[];
|
|
30
|
+
export declare function setupRunArtifacts(options: {
|
|
31
|
+
stwRoot: string;
|
|
32
|
+
runId: string;
|
|
33
|
+
repoRoot: string;
|
|
34
|
+
agents?: AgentDefinition[];
|
|
35
|
+
}): {
|
|
36
|
+
graph: TaskGraph | null;
|
|
37
|
+
assignmentsWritten: boolean;
|
|
38
|
+
inboxesWritten: number;
|
|
39
|
+
warnings: string[];
|
|
40
|
+
};
|
|
41
|
+
export declare function ingestRunFromContext(options: IngestContext): IngestResult;
|
|
42
|
+
export declare function findCurrentGitBranch(repoRoot: string): string | null;
|
|
43
|
+
export declare function getRepoRootFromCwd(): string;
|
|
44
|
+
export declare function getFixturesConfigPath(): string;
|