@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.
Files changed (250) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +277 -0
  3. package/dist/agentic-fallback.d.ts +3 -0
  4. package/dist/agentic-fallback.js +32 -0
  5. package/dist/agentic-fallback.js.map +1 -0
  6. package/dist/agentic-prompt.d.ts +2 -0
  7. package/dist/agentic-prompt.js +68 -0
  8. package/dist/agentic-prompt.js.map +1 -0
  9. package/dist/agentic-runtime.d.ts +48 -0
  10. package/dist/agentic-runtime.js +149 -0
  11. package/dist/agentic-runtime.js.map +1 -0
  12. package/dist/agentic-types.d.ts +37 -0
  13. package/dist/agentic-types.js +2 -0
  14. package/dist/agentic-types.js.map +1 -0
  15. package/dist/agents.d.ts +7 -0
  16. package/dist/agents.js +2 -0
  17. package/dist/agents.js.map +1 -0
  18. package/dist/assignments.d.ts +7 -0
  19. package/dist/assignments.js +125 -0
  20. package/dist/assignments.js.map +1 -0
  21. package/dist/checkpoint.d.ts +35 -0
  22. package/dist/checkpoint.js +78 -0
  23. package/dist/checkpoint.js.map +1 -0
  24. package/dist/circuit-breaker.d.ts +17 -0
  25. package/dist/circuit-breaker.js +65 -0
  26. package/dist/circuit-breaker.js.map +1 -0
  27. package/dist/claim.d.ts +6 -0
  28. package/dist/claim.js +135 -0
  29. package/dist/claim.js.map +1 -0
  30. package/dist/clarity-gate.d.ts +12 -0
  31. package/dist/clarity-gate.js +83 -0
  32. package/dist/clarity-gate.js.map +1 -0
  33. package/dist/cli.d.ts +2 -0
  34. package/dist/cli.js +38 -0
  35. package/dist/cli.js.map +1 -0
  36. package/dist/command-dispatch.d.ts +45 -0
  37. package/dist/command-dispatch.js +206 -0
  38. package/dist/command-dispatch.js.map +1 -0
  39. package/dist/command-parser.d.ts +11 -0
  40. package/dist/command-parser.js +101 -0
  41. package/dist/command-parser.js.map +1 -0
  42. package/dist/commands/clean.d.ts +10 -0
  43. package/dist/commands/clean.js +133 -0
  44. package/dist/commands/clean.js.map +1 -0
  45. package/dist/commands/execution.d.ts +2 -0
  46. package/dist/commands/execution.js +327 -0
  47. package/dist/commands/execution.js.map +1 -0
  48. package/dist/commands/go.d.ts +2 -0
  49. package/dist/commands/go.js +197 -0
  50. package/dist/commands/go.js.map +1 -0
  51. package/dist/commands/helpers.d.ts +44 -0
  52. package/dist/commands/helpers.js +231 -0
  53. package/dist/commands/helpers.js.map +1 -0
  54. package/dist/commands/idea.d.ts +2 -0
  55. package/dist/commands/idea.js +89 -0
  56. package/dist/commands/idea.js.map +1 -0
  57. package/dist/commands/init.d.ts +2 -0
  58. package/dist/commands/init.js +94 -0
  59. package/dist/commands/init.js.map +1 -0
  60. package/dist/commands/integration.d.ts +7 -0
  61. package/dist/commands/integration.js +139 -0
  62. package/dist/commands/integration.js.map +1 -0
  63. package/dist/commands/maintenance.d.ts +2 -0
  64. package/dist/commands/maintenance.js +301 -0
  65. package/dist/commands/maintenance.js.map +1 -0
  66. package/dist/commands/run.d.ts +2 -0
  67. package/dist/commands/run.js +356 -0
  68. package/dist/commands/run.js.map +1 -0
  69. package/dist/commands/setup.d.ts +2 -0
  70. package/dist/commands/setup.js +198 -0
  71. package/dist/commands/setup.js.map +1 -0
  72. package/dist/commands/spec.d.ts +2 -0
  73. package/dist/commands/spec.js +35 -0
  74. package/dist/commands/spec.js.map +1 -0
  75. package/dist/commands/stats.d.ts +2 -0
  76. package/dist/commands/stats.js +80 -0
  77. package/dist/commands/stats.js.map +1 -0
  78. package/dist/commands/task-ops.d.ts +2 -0
  79. package/dist/commands/task-ops.js +406 -0
  80. package/dist/commands/task-ops.js.map +1 -0
  81. package/dist/config.d.ts +18 -0
  82. package/dist/config.js +338 -0
  83. package/dist/config.js.map +1 -0
  84. package/dist/cost.d.ts +30 -0
  85. package/dist/cost.js +167 -0
  86. package/dist/cost.js.map +1 -0
  87. package/dist/crash-recovery.d.ts +9 -0
  88. package/dist/crash-recovery.js +42 -0
  89. package/dist/crash-recovery.js.map +1 -0
  90. package/dist/diagnostic.d.ts +48 -0
  91. package/dist/diagnostic.js +328 -0
  92. package/dist/diagnostic.js.map +1 -0
  93. package/dist/doctor.d.ts +31 -0
  94. package/dist/doctor.js +225 -0
  95. package/dist/doctor.js.map +1 -0
  96. package/dist/drift.d.ts +11 -0
  97. package/dist/drift.js +57 -0
  98. package/dist/drift.js.map +1 -0
  99. package/dist/git-utils.d.ts +20 -0
  100. package/dist/git-utils.js +206 -0
  101. package/dist/git-utils.js.map +1 -0
  102. package/dist/gitlab.d.ts +54 -0
  103. package/dist/gitlab.js +101 -0
  104. package/dist/gitlab.js.map +1 -0
  105. package/dist/idea.d.ts +35 -0
  106. package/dist/idea.js +251 -0
  107. package/dist/idea.js.map +1 -0
  108. package/dist/import-resolution.d.ts +13 -0
  109. package/dist/import-resolution.js +111 -0
  110. package/dist/import-resolution.js.map +1 -0
  111. package/dist/inbox-renderer.d.ts +2 -0
  112. package/dist/inbox-renderer.js +67 -0
  113. package/dist/inbox-renderer.js.map +1 -0
  114. package/dist/init.d.ts +105 -0
  115. package/dist/init.js +235 -0
  116. package/dist/init.js.map +1 -0
  117. package/dist/llm-reviewer.d.ts +14 -0
  118. package/dist/llm-reviewer.js +109 -0
  119. package/dist/llm-reviewer.js.map +1 -0
  120. package/dist/lock.d.ts +26 -0
  121. package/dist/lock.js +76 -0
  122. package/dist/lock.js.map +1 -0
  123. package/dist/logger.d.ts +24 -0
  124. package/dist/logger.js +40 -0
  125. package/dist/logger.js.map +1 -0
  126. package/dist/math-utils.d.ts +2 -0
  127. package/dist/math-utils.js +7 -0
  128. package/dist/math-utils.js.map +1 -0
  129. package/dist/mechanical-review.d.ts +30 -0
  130. package/dist/mechanical-review.js +76 -0
  131. package/dist/mechanical-review.js.map +1 -0
  132. package/dist/merge.d.ts +83 -0
  133. package/dist/merge.js +363 -0
  134. package/dist/merge.js.map +1 -0
  135. package/dist/parallel.d.ts +35 -0
  136. package/dist/parallel.js +214 -0
  137. package/dist/parallel.js.map +1 -0
  138. package/dist/plan-validation.d.ts +19 -0
  139. package/dist/plan-validation.js +253 -0
  140. package/dist/plan-validation.js.map +1 -0
  141. package/dist/planner-prompt.d.ts +33 -0
  142. package/dist/planner-prompt.js +244 -0
  143. package/dist/planner-prompt.js.map +1 -0
  144. package/dist/planner.d.ts +29 -0
  145. package/dist/planner.js +511 -0
  146. package/dist/planner.js.map +1 -0
  147. package/dist/poller.d.ts +34 -0
  148. package/dist/poller.js +91 -0
  149. package/dist/poller.js.map +1 -0
  150. package/dist/progress.d.ts +34 -0
  151. package/dist/progress.js +122 -0
  152. package/dist/progress.js.map +1 -0
  153. package/dist/prompt-builder.d.ts +51 -0
  154. package/dist/prompt-builder.js +481 -0
  155. package/dist/prompt-builder.js.map +1 -0
  156. package/dist/provider.d.ts +14 -0
  157. package/dist/provider.js +278 -0
  158. package/dist/provider.js.map +1 -0
  159. package/dist/question-handler.d.ts +18 -0
  160. package/dist/question-handler.js +154 -0
  161. package/dist/question-handler.js.map +1 -0
  162. package/dist/question-triage.d.ts +31 -0
  163. package/dist/question-triage.js +175 -0
  164. package/dist/question-triage.js.map +1 -0
  165. package/dist/repo-detection.d.ts +8 -0
  166. package/dist/repo-detection.js +18 -0
  167. package/dist/repo-detection.js.map +1 -0
  168. package/dist/retry-context.d.ts +2 -0
  169. package/dist/retry-context.js +196 -0
  170. package/dist/retry-context.js.map +1 -0
  171. package/dist/router.d.ts +18 -0
  172. package/dist/router.js +137 -0
  173. package/dist/router.js.map +1 -0
  174. package/dist/run-artifact-types.d.ts +43 -0
  175. package/dist/run-artifact-types.js +2 -0
  176. package/dist/run-artifact-types.js.map +1 -0
  177. package/dist/run-summary.d.ts +14 -0
  178. package/dist/run-summary.js +347 -0
  179. package/dist/run-summary.js.map +1 -0
  180. package/dist/run-sync.d.ts +11 -0
  181. package/dist/run-sync.js +110 -0
  182. package/dist/run-sync.js.map +1 -0
  183. package/dist/run.d.ts +26 -0
  184. package/dist/run.js +150 -0
  185. package/dist/run.js.map +1 -0
  186. package/dist/scope-expansion.d.ts +10 -0
  187. package/dist/scope-expansion.js +117 -0
  188. package/dist/scope-expansion.js.map +1 -0
  189. package/dist/scope.d.ts +4 -0
  190. package/dist/scope.js +37 -0
  191. package/dist/scope.js.map +1 -0
  192. package/dist/scorecard.d.ts +18 -0
  193. package/dist/scorecard.js +128 -0
  194. package/dist/scorecard.js.map +1 -0
  195. package/dist/spec-templates.d.ts +2 -0
  196. package/dist/spec-templates.js +285 -0
  197. package/dist/spec-templates.js.map +1 -0
  198. package/dist/spec-validator.d.ts +8 -0
  199. package/dist/spec-validator.js +144 -0
  200. package/dist/spec-validator.js.map +1 -0
  201. package/dist/status.d.ts +68 -0
  202. package/dist/status.js +261 -0
  203. package/dist/status.js.map +1 -0
  204. package/dist/storage.d.ts +9 -0
  205. package/dist/storage.js +35 -0
  206. package/dist/storage.js.map +1 -0
  207. package/dist/task-executor-completion.d.ts +12 -0
  208. package/dist/task-executor-completion.js +67 -0
  209. package/dist/task-executor-completion.js.map +1 -0
  210. package/dist/task-executor-fallback.d.ts +20 -0
  211. package/dist/task-executor-fallback.js +12 -0
  212. package/dist/task-executor-fallback.js.map +1 -0
  213. package/dist/task-executor.d.ts +34 -0
  214. package/dist/task-executor.js +521 -0
  215. package/dist/task-executor.js.map +1 -0
  216. package/dist/task-graph.d.ts +11 -0
  217. package/dist/task-graph.js +226 -0
  218. package/dist/task-graph.js.map +1 -0
  219. package/dist/task-pipeline-helpers.d.ts +45 -0
  220. package/dist/task-pipeline-helpers.js +160 -0
  221. package/dist/task-pipeline-helpers.js.map +1 -0
  222. package/dist/task-review.d.ts +51 -0
  223. package/dist/task-review.js +410 -0
  224. package/dist/task-review.js.map +1 -0
  225. package/dist/transitions.d.ts +13 -0
  226. package/dist/transitions.js +104 -0
  227. package/dist/transitions.js.map +1 -0
  228. package/dist/types.d.ts +405 -0
  229. package/dist/types.js +101 -0
  230. package/dist/types.js.map +1 -0
  231. package/dist/utils.d.ts +1 -0
  232. package/dist/utils.js +23 -0
  233. package/dist/utils.js.map +1 -0
  234. package/dist/validation.d.ts +19 -0
  235. package/dist/validation.js +73 -0
  236. package/dist/validation.js.map +1 -0
  237. package/dist/worker-response.d.ts +12 -0
  238. package/dist/worker-response.js +60 -0
  239. package/dist/worker-response.js.map +1 -0
  240. package/dist/worker-runner.d.ts +19 -0
  241. package/dist/worker-runner.js +347 -0
  242. package/dist/worker-runner.js.map +1 -0
  243. package/dist/worktree-cleanup.d.ts +44 -0
  244. package/dist/worktree-cleanup.js +325 -0
  245. package/dist/worktree-cleanup.js.map +1 -0
  246. package/dist/worktree.d.ts +22 -0
  247. package/dist/worktree.js +213 -0
  248. package/dist/worktree.js.map +1 -0
  249. package/examples/spec.md +58 -0
  250. package/package.json +66 -0
@@ -0,0 +1,231 @@
1
+ import { resolve, isAbsolute, relative, dirname, join } from 'node:path';
2
+ import { existsSync, readFileSync } from 'node:fs';
3
+ import { fileURLToPath } from 'node:url';
4
+ import { execFileSync } from 'node:child_process';
5
+ import { validateSpec } from '../spec-validator.js';
6
+ import { createRun } from '../run.js';
7
+ import { loadConfig } from '../config.js';
8
+ import { validateTaskGraph } from '../task-graph.js';
9
+ import { generateAssignments } from '../assignments.js';
10
+ import { renderInboxes } from '../inbox-renderer.js';
11
+ import { ensureDir, fileExists, readJson, writeAtomic, writeJson, runDir } from '../storage.js';
12
+ export function getStwRoot() {
13
+ return resolve('.stw');
14
+ }
15
+ export function delay(ms) {
16
+ return new Promise((r) => setTimeout(r, ms));
17
+ }
18
+ export const LOG_LEVELS = ['debug', 'info', 'warn', 'error'];
19
+ export function getDefaultAgents() {
20
+ return [
21
+ { agent_id: 'agent-strong-1', tool: 'claude-code', model: 'opus-4.6', tier: 'strong' },
22
+ { agent_id: 'agent-medium-1', tool: 'opencode', model: 'gpt-5.4', tier: 'medium' },
23
+ { agent_id: 'agent-cheap-1', tool: 'opencode', model: 'minimax-2.5', tier: 'cheap' },
24
+ ];
25
+ }
26
+ export function setupRunArtifacts(options) {
27
+ const graphPath = join(runDir(options.stwRoot, options.runId), 'task_graph.json');
28
+ if (!existsSync(graphPath)) {
29
+ return {
30
+ graph: null,
31
+ assignmentsWritten: false,
32
+ inboxesWritten: 0,
33
+ warnings: [],
34
+ };
35
+ }
36
+ const graph = readJson(graphPath);
37
+ const graphValidation = validateTaskGraph(graph);
38
+ if (!graphValidation.valid) {
39
+ throw new Error(`Task graph validation failed: ${graphValidation.errors.join('; ')}`);
40
+ }
41
+ const config = loadConfig(join(options.stwRoot, 'config.yml'));
42
+ const agents = options.agents ?? getDefaultAgents();
43
+ const assignmentResult = generateAssignments(graph, agents, config);
44
+ writeJson(join(runDir(options.stwRoot, options.runId), 'assignments.json'), assignmentResult.assignments);
45
+ const inboxes = renderInboxes(assignmentResult.assignments, graph);
46
+ const inboxDir = join(runDir(options.stwRoot, options.runId), 'inboxes');
47
+ ensureDir(inboxDir);
48
+ for (const [agentId, content] of Object.entries(inboxes)) {
49
+ writeAtomic(join(inboxDir, `${agentId}.md`), content);
50
+ }
51
+ return {
52
+ graph,
53
+ assignmentsWritten: true,
54
+ inboxesWritten: Object.keys(inboxes).length,
55
+ warnings: [...graphValidation.warnings, ...assignmentResult.warnings],
56
+ };
57
+ }
58
+ export function ingestRunFromContext(options) {
59
+ const config = loadConfig(join(options.stwRoot, 'config.yml'));
60
+ const specResolution = resolveSpecPathFromContext({
61
+ repoRoot: options.repoRoot,
62
+ noteBody: options.noteBody,
63
+ notePath: options.notePath,
64
+ fallbackSpecPath: options.fallbackSpecPath,
65
+ });
66
+ if (!specResolution.ok) {
67
+ return {
68
+ ok: false,
69
+ message: specResolution.message,
70
+ details: specResolution.details,
71
+ };
72
+ }
73
+ let markdown;
74
+ try {
75
+ markdown = readFileSync(specResolution.absolutePath, 'utf-8');
76
+ }
77
+ catch (error) {
78
+ const message = error instanceof Error ? error.message : String(error);
79
+ return {
80
+ ok: false,
81
+ message: `Failed to read spec: ${specResolution.displayPath}`,
82
+ details: [message],
83
+ };
84
+ }
85
+ const validation = validateSpec(markdown);
86
+ if (!validation.valid) {
87
+ return {
88
+ ok: false,
89
+ message: `Spec validation failed for ${specResolution.displayPath}`,
90
+ details: validation.errors,
91
+ };
92
+ }
93
+ const specStartedAt = new Date().toISOString();
94
+ const manifest = createRun({
95
+ stwRoot: options.stwRoot,
96
+ specPath: specResolution.repoRelativePath,
97
+ repoRoot: options.repoRoot,
98
+ baseBranch: options.baseBranch ?? config.gitlab?.base_branch ?? 'main',
99
+ branchPrefix: config.gitlab?.branch_prefix ?? 'ai/',
100
+ greenfield: options.greenfield,
101
+ seedDir: options.seedDir,
102
+ });
103
+ // Record spec timing in manifest
104
+ const manifestPath = join(runDir(options.stwRoot, manifest.run_id), 'manifest.json');
105
+ if (fileExists(manifestPath)) {
106
+ const savedManifest = readJson(manifestPath);
107
+ savedManifest.spec_started_at = specStartedAt;
108
+ savedManifest.spec_completed_at = new Date().toISOString();
109
+ writeJson(manifestPath, savedManifest);
110
+ }
111
+ let setupSummary = 'No task graph found yet; run created only.';
112
+ const details = [`Spec: ${specResolution.repoRelativePath}`, `Branch: ${manifest.branch}`];
113
+ if (validation.warnings.length > 0) {
114
+ details.push(...validation.warnings.map((warning) => `Spec warning: ${warning}`));
115
+ }
116
+ try {
117
+ const setupResult = setupRunArtifacts({
118
+ stwRoot: options.stwRoot,
119
+ runId: manifest.run_id,
120
+ repoRoot: options.repoRoot,
121
+ });
122
+ if (setupResult.graph) {
123
+ setupSummary = `Assignments written; inboxes generated: ${setupResult.inboxesWritten}.`;
124
+ for (const warning of setupResult.warnings) {
125
+ details.push(`Setup warning: ${warning}`);
126
+ }
127
+ }
128
+ }
129
+ catch (error) {
130
+ const message = error instanceof Error ? error.message : String(error);
131
+ setupSummary = `Run created, but setup bootstrap was incomplete: ${message}`;
132
+ }
133
+ details.push(setupSummary);
134
+ return {
135
+ ok: true,
136
+ manifest,
137
+ message: `Run created: ${manifest.run_id}`,
138
+ details,
139
+ };
140
+ }
141
+ function resolveSpecPathFromContext(options) {
142
+ const candidates = collectSpecCandidates(options.noteBody, options.fallbackSpecPath);
143
+ if (options.notePath) {
144
+ const noteDir = dirname(options.notePath);
145
+ for (const candidate of candidates) {
146
+ const maybeAbsolute = isAbsolute(candidate) ? candidate : resolve(noteDir, candidate);
147
+ if (existsSync(maybeAbsolute)) {
148
+ const repoRelativePath = relative(options.repoRoot, maybeAbsolute).replaceAll('\\', '/');
149
+ return {
150
+ ok: true,
151
+ absolutePath: maybeAbsolute,
152
+ repoRelativePath,
153
+ displayPath: repoRelativePath,
154
+ };
155
+ }
156
+ }
157
+ }
158
+ for (const candidate of candidates) {
159
+ const maybeAbsolute = isAbsolute(candidate) ? candidate : resolve(options.repoRoot, candidate);
160
+ if (existsSync(maybeAbsolute)) {
161
+ const repoRelativePath = relative(options.repoRoot, maybeAbsolute).replaceAll('\\', '/');
162
+ return {
163
+ ok: true,
164
+ absolutePath: maybeAbsolute,
165
+ repoRelativePath,
166
+ displayPath: repoRelativePath,
167
+ };
168
+ }
169
+ }
170
+ if (candidates.length === 0) {
171
+ return {
172
+ ok: false,
173
+ message: 'No spec path found in /ingest command or note context.',
174
+ details: ['Provide `/ingest path/to/spec.md` or include a repo-relative spec path in the note.'],
175
+ };
176
+ }
177
+ return {
178
+ ok: false,
179
+ message: `Spec path not found: ${candidates[0]}`,
180
+ details: candidates.slice(1).map((candidate) => `Also tried: ${candidate}`),
181
+ };
182
+ }
183
+ function collectSpecCandidates(noteBody, fallbackSpecPath) {
184
+ const values = [];
185
+ if (fallbackSpecPath?.trim()) {
186
+ values.push(fallbackSpecPath.trim());
187
+ }
188
+ if (noteBody) {
189
+ const commandMatch = noteBody.match(/(?:^|\s)\/ingest\s+([^\s]+)/i);
190
+ if (commandMatch?.[1]) {
191
+ values.push(commandMatch[1].trim());
192
+ }
193
+ const markdownMatches = noteBody.matchAll(/`([^`]+\.md)`/g);
194
+ for (const match of markdownMatches) {
195
+ values.push(match[1].trim());
196
+ }
197
+ const pathMatches = noteBody.matchAll(/(?:^|\s)([A-Za-z0-9_./-]+\.md)(?=$|\s)/g);
198
+ for (const match of pathMatches) {
199
+ values.push(match[1].trim());
200
+ }
201
+ }
202
+ const unique = new Set();
203
+ const normalized = [];
204
+ for (const value of values) {
205
+ if (!value || unique.has(value))
206
+ continue;
207
+ unique.add(value);
208
+ normalized.push(value);
209
+ }
210
+ return normalized;
211
+ }
212
+ export function findCurrentGitBranch(repoRoot) {
213
+ try {
214
+ return execFileSync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], {
215
+ cwd: repoRoot,
216
+ encoding: 'utf-8',
217
+ stdio: 'pipe',
218
+ }).trim();
219
+ }
220
+ catch {
221
+ return null;
222
+ }
223
+ }
224
+ export function getRepoRootFromCwd() {
225
+ return resolve('.');
226
+ }
227
+ export function getFixturesConfigPath() {
228
+ const here = dirname(fileURLToPath(import.meta.url));
229
+ return resolve(here, '../../fixtures/config.yml');
230
+ }
231
+ //# sourceMappingURL=helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../src/commands/helpers.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from 'commander';
2
+ export declare function registerIdeaCommand(program: Command): void;
@@ -0,0 +1,89 @@
1
+ import { join, resolve } from 'node:path';
2
+ import { generateSpec, slugify, KNOWN_STACK_PRESETS, ClarityGateError } from '../idea.js';
3
+ import { loadConfig } from '../config.js';
4
+ import { writeAtomic } from '../storage.js';
5
+ import { validateSeedDir } from '../worktree.js';
6
+ export function registerIdeaCommand(program) {
7
+ program
8
+ .command('idea')
9
+ .description('Generate a spec from a rough idea using a strong-tier model')
10
+ .argument('<description>', 'The rough idea in natural language')
11
+ .option('--template <name>', 'Template override (bugfix, feature, refactor, cleanup, greenfield)')
12
+ .option('--greenfield', 'Generate a spec for a brand-new project (no existing codebase)')
13
+ .option('--stack <preset>', `Stack preset (${KNOWN_STACK_PRESETS.join(', ')}). Implies --greenfield.`)
14
+ .option('--from <dir>', 'Seed greenfield worktree from a local directory. Implies --greenfield.')
15
+ .option('--no-clarity-check', 'Skip the idea clarity check (for power users)')
16
+ .option('--output <path>', 'Output path for generated spec')
17
+ .action(async (description, opts) => {
18
+ if (opts.stack && !KNOWN_STACK_PRESETS.includes(opts.stack)) {
19
+ console.error(`Unknown stack preset "${opts.stack}". Available: ${KNOWN_STACK_PRESETS.join(', ')}`);
20
+ process.exit(1);
21
+ }
22
+ const seedDir = opts.from ? resolve(opts.from) : undefined;
23
+ if (seedDir) {
24
+ try {
25
+ validateSeedDir(seedDir);
26
+ }
27
+ catch (err) {
28
+ console.error(err instanceof Error ? err.message : String(err));
29
+ process.exit(1);
30
+ }
31
+ }
32
+ const isGreenfield = opts.greenfield || !!opts.stack || !!seedDir;
33
+ const stwRoot = join(process.cwd(), '.stw');
34
+ const config = loadConfig(join(stwRoot, 'config.yml'));
35
+ const outputPath = opts.output ?? join('specs', `${slugify(description)}.md`);
36
+ console.log(`Generating spec from idea: "${description}"`);
37
+ if (isGreenfield) {
38
+ console.log(`Mode: greenfield (no existing codebase)`);
39
+ }
40
+ else if (opts.template) {
41
+ console.log(`Using template: ${opts.template}`);
42
+ }
43
+ if (opts.stack) {
44
+ console.log(`Stack: ${opts.stack}`);
45
+ }
46
+ if (seedDir) {
47
+ console.log(`Seed directory: ${seedDir}`);
48
+ }
49
+ let result;
50
+ try {
51
+ result = await generateSpec({
52
+ idea: description,
53
+ repoRoot: process.cwd(),
54
+ config,
55
+ templateOverride: opts.template,
56
+ isGreenfield,
57
+ stack: opts.stack,
58
+ seedDir,
59
+ skipClarityCheck: opts.clarityCheck === false,
60
+ });
61
+ }
62
+ catch (err) {
63
+ if (err instanceof ClarityGateError) {
64
+ console.error(`\nIdea rejected — not specific enough to generate a spec.\n`);
65
+ console.error(`Feedback: ${err.feedback}`);
66
+ if (err.suggestedIdea) {
67
+ console.error(`\nYou can use this as a starting point:\n "${err.suggestedIdea}"`);
68
+ }
69
+ process.exit(1);
70
+ }
71
+ throw err;
72
+ }
73
+ writeAtomic(outputPath, result.spec);
74
+ console.log(`\nSpec generated successfully!`);
75
+ console.log(` Template: ${result.templateUsed}`);
76
+ console.log(` Output: ${outputPath}`);
77
+ if (result.model) {
78
+ console.log(` Model: ${result.model}`);
79
+ }
80
+ if (result.usage) {
81
+ console.log(` Tokens: ${result.usage.input_tokens} in / ${result.usage.output_tokens} out`);
82
+ }
83
+ if (result.latency_ms) {
84
+ console.log(` Latency: ${(result.latency_ms / 1000).toFixed(1)}s`);
85
+ }
86
+ console.log(`\nReview the spec, then run: stw ingest ${outputPath}`);
87
+ });
88
+ }
89
+ //# sourceMappingURL=idea.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"idea.js","sourceRoot":"","sources":["../../src/commands/idea.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAE1C,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAC1F,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAEjD,MAAM,UAAU,mBAAmB,CAAC,OAAgB;IAClD,OAAO;SACJ,OAAO,CAAC,MAAM,CAAC;SACf,WAAW,CAAC,6DAA6D,CAAC;SAC1E,QAAQ,CAAC,eAAe,EAAE,oCAAoC,CAAC;SAC/D,MAAM,CAAC,mBAAmB,EAAE,oEAAoE,CAAC;SACjG,MAAM,CAAC,cAAc,EAAE,gEAAgE,CAAC;SACxF,MAAM,CAAC,kBAAkB,EAAE,iBAAiB,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,0BAA0B,CAAC;SACrG,MAAM,CAAC,cAAc,EAAE,wEAAwE,CAAC;SAChG,MAAM,CAAC,oBAAoB,EAAE,+CAA+C,CAAC;SAC7E,MAAM,CAAC,iBAAiB,EAAE,gCAAgC,CAAC;SAC3D,MAAM,CACL,KAAK,EACH,WAAmB,EACnB,IAOC,EACD,EAAE;QACF,IAAI,IAAI,CAAC,KAAK,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,IAAI,CAAC,KAA6C,CAAC,EAAE,CAAC;YACpG,OAAO,CAAC,KAAK,CAAC,yBAAyB,IAAI,CAAC,KAAK,iBAAiB,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACpG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAC3D,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC;gBACH,eAAe,CAAC,OAAO,CAAC,CAAC;YAC3B,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,OAAO,CAAC,KAAK,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;gBAChE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,UAAU,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,OAAO,CAAC;QAElE,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,MAAM,CAAC,CAAC;QAC5C,MAAM,MAAM,GAAG,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC;QAEvD,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QAE9E,OAAO,CAAC,GAAG,CAAC,+BAA+B,WAAW,GAAG,CAAC,CAAC;QAC3D,IAAI,YAAY,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAC;QACzD,CAAC;aAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACzB,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;QAClD,CAAC;QACD,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACtC,CAAC;QACD,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,CAAC,GAAG,CAAC,mBAAmB,OAAO,EAAE,CAAC,CAAC;QAC5C,CAAC;QAED,IAAI,MAAM,CAAC;QACX,IAAI,CAAC;YACH,MAAM,GAAG,MAAM,YAAY,CAAC;gBAC1B,IAAI,EAAE,WAAW;gBACjB,QAAQ,EAAE,OAAO,CAAC,GAAG,EAAE;gBACvB,MAAM;gBACN,gBAAgB,EAAE,IAAI,CAAC,QAAQ;gBAC/B,YAAY;gBACZ,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,OAAO;gBACP,gBAAgB,EAAE,IAAI,CAAC,YAAY,KAAK,KAAK;aAC9C,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,gBAAgB,EAAE,CAAC;gBACpC,OAAO,CAAC,KAAK,CAAC,6DAA6D,CAAC,CAAC;gBAC7E,OAAO,CAAC,KAAK,CAAC,aAAa,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAC3C,IAAI,GAAG,CAAC,aAAa,EAAE,CAAC;oBACtB,OAAO,CAAC,KAAK,CAAC,+CAA+C,GAAG,CAAC,aAAa,GAAG,CAAC,CAAC;gBACrF,CAAC;gBACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;QAED,WAAW,CAAC,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;QAErC,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;QAC9C,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,YAAY,EAAE,CAAC,CAAC;QAClD,OAAO,CAAC,GAAG,CAAC,eAAe,UAAU,EAAE,CAAC,CAAC;QACzC,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;QAC7C,CAAC;QACD,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,KAAK,CAAC,YAAY,SAAS,MAAM,CAAC,KAAK,CAAC,aAAa,MAAM,CAAC,CAAC;QACjG,CAAC;QACD,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YACtB,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,MAAM,CAAC,UAAU,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACvE,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,2CAA2C,UAAU,EAAE,CAAC,CAAC;IACvE,CAAC,CACF,CAAC;AACN,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { Command } from 'commander';
2
+ export declare function registerInitCommand(program: Command): void;
@@ -0,0 +1,94 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { resolve, join } from 'node:path';
3
+ import * as yaml from 'js-yaml';
4
+ import { detectRepo, generateConfig, detectBackends } from '../init.js';
5
+ import { writeAtomic, ensureDir } from '../storage.js';
6
+ import { runDoctor } from '../doctor.js';
7
+ export function registerInitCommand(program) {
8
+ program
9
+ .command('init')
10
+ .description('Initialize a repository for STW orchestration')
11
+ .option('--force', 'Overwrite existing config')
12
+ .option('--provider <provider>', 'Provider name (default: openrouter)')
13
+ .option('--backend <backend>', 'Agentic backend (default: opencode)')
14
+ .option('--max-run-usd <amount>', 'Max budget per run in USD', parseFloat)
15
+ .option('--preset <preset>', 'Model preset: glm5 (budget), minimax (balanced), anthropic (premium), mixed (multi-provider) (default: minimax)')
16
+ .action((opts) => {
17
+ const repoRoot = resolve('.');
18
+ if (!existsSync(join(repoRoot, '.git'))) {
19
+ console.error('Error: Not a git repository. Run "git init" first.');
20
+ process.exit(1);
21
+ }
22
+ const configPath = join(repoRoot, '.stw', 'config.yml');
23
+ if (existsSync(configPath) && !opts.force) {
24
+ console.error('Error: .stw/config.yml already exists. Use --force to overwrite.');
25
+ process.exit(1);
26
+ }
27
+ const backends = detectBackends();
28
+ const availableBackends = [];
29
+ if (backends.opencode)
30
+ availableBackends.push('opencode');
31
+ if (backends.claude)
32
+ availableBackends.push('claude');
33
+ console.log(`Detected backends: ${availableBackends.length > 0 ? availableBackends.join(', ') : 'none'}`);
34
+ let backend = opts.backend;
35
+ if (!backend && backends.claude) {
36
+ backend = 'claude-code';
37
+ console.log('Defaulting to claude-code backend (claude detected)');
38
+ }
39
+ const profile = detectRepo(repoRoot);
40
+ console.log(`Detected repo type: ${profile.type}`);
41
+ if (profile.detectedFiles.length > 0) {
42
+ console.log(` Files: ${profile.detectedFiles.join(', ')}`);
43
+ }
44
+ if (profile.testCommand) {
45
+ console.log(` Test command: ${profile.testCommand}`);
46
+ }
47
+ if (profile.lintCommand) {
48
+ console.log(` Lint command: ${profile.lintCommand}`);
49
+ }
50
+ const preset = opts.preset;
51
+ const config = generateConfig(repoRoot, {
52
+ provider: opts.provider,
53
+ backend,
54
+ maxRunUsd: opts.maxRunUsd,
55
+ preset,
56
+ });
57
+ ensureDir(join(repoRoot, '.stw'));
58
+ const yamlContent = yaml.dump(config, { lineWidth: 120, noRefs: true });
59
+ writeAtomic(configPath, yamlContent);
60
+ console.log(`\nConfig written to: ${configPath}`);
61
+ // Post-init: run doctor to show readiness
62
+ console.log('\nRunning readiness checks...\n');
63
+ const doctorChecks = runDoctor({ repoRoot, stwRoot: join(repoRoot, '.stw') });
64
+ const failed = doctorChecks.filter((c) => !c.passed);
65
+ const passed = doctorChecks.filter((c) => c.passed);
66
+ for (const check of passed) {
67
+ console.log(` ✓ ${check.message}`);
68
+ }
69
+ for (const check of failed) {
70
+ console.log(` ✗ ${check.message}`);
71
+ }
72
+ if (failed.length === 0) {
73
+ console.log("\nAll checks passed! You're ready to go.");
74
+ }
75
+ else {
76
+ console.log(`\n${failed.length} issue(s) to fix before running stw.`);
77
+ }
78
+ // Show next steps based on what's missing
79
+ const apiKeyMissing = failed.some((c) => c.name.startsWith('api-key-'));
80
+ const steps = [];
81
+ if (apiKeyMissing) {
82
+ const envVar = Object.values(config.providers)[0]?.api_key_env ?? 'OPENROUTER_API_KEY';
83
+ steps.push(`Set your API key: export ${envVar}=your-key`);
84
+ }
85
+ steps.push('Verify readiness: stw doctor');
86
+ steps.push('Run from an idea: stw go "add a health check endpoint"');
87
+ steps.push('Or write a spec first: stw idea "add rate limiting" → edit → stw go "rate limiting" --spec specs/add-rate-limiting.md');
88
+ console.log('\nNext steps:');
89
+ for (let i = 0; i < steps.length; i++) {
90
+ console.log(` ${i + 1}. ${steps[i]}`);
91
+ }
92
+ });
93
+ }
94
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import type { Command } from 'commander';
2
+ export declare function pollRunComments(options: {
3
+ runId: string;
4
+ stwRoot: string;
5
+ watch?: boolean;
6
+ }): Promise<void>;
7
+ export declare function registerIntegrationCommands(program: Command): void;
@@ -0,0 +1,139 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { join, resolve } from 'node:path';
3
+ import { loadConfig } from '../config.js';
4
+ import { readTaskStatus } from '../status.js';
5
+ import { readJson, writeJson, fileExists, runDir } from '../storage.js';
6
+ import { loadCosts } from '../cost.js';
7
+ import { pollAndDispatch } from '../poller.js';
8
+ import { syncRunToGitlab } from '../run-sync.js';
9
+ import { getStwRoot, delay } from './helpers.js';
10
+ function buildTaskStates(stwRoot, runId) {
11
+ const graphPath = join(runDir(stwRoot, runId), 'task_graph.json');
12
+ if (!fileExists(graphPath)) {
13
+ return undefined;
14
+ }
15
+ const graph = readJson(graphPath);
16
+ const taskStates = {};
17
+ for (const task of graph.tasks) {
18
+ const status = readTaskStatus(stwRoot, runId, task.id);
19
+ taskStates[task.id] = status?.status ?? task.state;
20
+ }
21
+ return taskStates;
22
+ }
23
+ export async function pollRunComments(options) {
24
+ const manifestPath = join(runDir(options.stwRoot, options.runId), 'manifest.json');
25
+ if (!fileExists(manifestPath)) {
26
+ console.error(`Run not found: ${options.runId}`);
27
+ process.exit(1);
28
+ }
29
+ const config = loadConfig(join(options.stwRoot, 'config.yml'));
30
+ const allowedUsers = config.gitlab?.allowed_command_users ?? [];
31
+ const pollIntervalMs = Math.max(1, config.defaults.poll_interval_seconds) * 1000;
32
+ const commandsFilePath = join(runDir(options.stwRoot, options.runId), 'commands.ndjson');
33
+ const stopSentinelPath = join(runDir(options.stwRoot, options.runId), 'STOP_POLL');
34
+ while (true) {
35
+ const currentManifest = readJson(manifestPath);
36
+ if (!currentManifest.pr_number) {
37
+ console.error(`Run ${options.runId} has no merge request`);
38
+ process.exit(1);
39
+ }
40
+ const pollResult = await pollAndDispatch({
41
+ repoRoot: resolve('.'),
42
+ mrIid: currentManifest.pr_number,
43
+ since: currentManifest.created_at,
44
+ allowedUsers,
45
+ afterCommentId: currentManifest.last_polled_comment_id,
46
+ stwRoot: options.stwRoot,
47
+ runId: options.runId,
48
+ dispatchContext: {
49
+ commandsFilePath,
50
+ allowedUsers,
51
+ runId: options.runId,
52
+ stwRoot: options.stwRoot,
53
+ runState: currentManifest.status,
54
+ taskStates: buildTaskStates(options.stwRoot, options.runId),
55
+ },
56
+ });
57
+ for (const entry of pollResult.commands) {
58
+ console.log(`#${entry.noteId} ${entry.author}: /${entry.command.type}`);
59
+ }
60
+ for (const entry of pollResult.unauthorized) {
61
+ console.log(`#${entry.noteId} ${entry.author}: unauthorized /${entry.command.type}`);
62
+ }
63
+ if (pollResult.lastAppliedCommentId != null) {
64
+ const updatedManifest = readJson(manifestPath);
65
+ updatedManifest.last_polled_comment_id = pollResult.lastAppliedCommentId;
66
+ writeJson(manifestPath, updatedManifest);
67
+ }
68
+ if (!options.watch) {
69
+ break;
70
+ }
71
+ if (existsSync(stopSentinelPath)) {
72
+ break;
73
+ }
74
+ await delay(pollIntervalMs);
75
+ }
76
+ }
77
+ export function registerIntegrationCommands(program) {
78
+ program
79
+ .command('sync-gitlab <run-id>')
80
+ .description('Push run branch and sync MR state to GitLab')
81
+ .action(async (runId) => {
82
+ const stwRoot = getStwRoot();
83
+ const manifestPath = join(runDir(stwRoot, runId), 'manifest.json');
84
+ const manifest = readJson(manifestPath);
85
+ const graph = readJson(join(runDir(stwRoot, runId), 'task_graph.json'));
86
+ const config = loadConfig(join(stwRoot, 'config.yml'));
87
+ const costs = loadCosts(stwRoot, runId);
88
+ try {
89
+ syncRunToGitlab({ stwRoot, runId, config, manifest, manifestPath, graph, costs });
90
+ }
91
+ catch (error) {
92
+ const message = error instanceof Error ? error.message : String(error);
93
+ console.error(`sync-gitlab failed: ${message}`);
94
+ process.exit(1);
95
+ }
96
+ });
97
+ program
98
+ .command('merge <run-id>')
99
+ .description('Push a run branch and create its GitLab merge request')
100
+ .action(async (runId) => {
101
+ const stwRoot = getStwRoot();
102
+ const { executeMergeForRun } = await import('../merge.js');
103
+ try {
104
+ const result = await executeMergeForRun({
105
+ stwRoot,
106
+ runId,
107
+ repoRoot: resolve('.'),
108
+ });
109
+ for (const line of result.messages) {
110
+ if (line.level === 'error') {
111
+ console.error(line.text);
112
+ }
113
+ else {
114
+ console.log(line.text);
115
+ }
116
+ }
117
+ if (!result.ok) {
118
+ process.exit(1);
119
+ }
120
+ }
121
+ catch (error) {
122
+ const message = error instanceof Error ? error.message : String(error);
123
+ console.error(`merge failed: ${message}`);
124
+ process.exit(1);
125
+ }
126
+ });
127
+ program
128
+ .command('poll <run-id>')
129
+ .description('Poll MR comments for slash commands')
130
+ .option('--watch', 'Continuously poll for new comments')
131
+ .action(async (runId, opts) => {
132
+ await pollRunComments({
133
+ runId,
134
+ stwRoot: getStwRoot(),
135
+ watch: opts.watch,
136
+ });
137
+ });
138
+ }
139
+ //# sourceMappingURL=integration.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"integration.js","sourceRoot":"","sources":["../../src/commands/integration.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { Command } from 'commander';
2
+ export declare function registerMaintenanceCommands(program: Command): void;