@hivehub/rulebook 5.2.0 → 5.3.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/.claude/commands/analysis.md +35 -0
- package/.claude/commands/rulebook-task-apply.md +7 -25
- package/.claude/commands/rulebook-task-archive.md +10 -19
- package/.claude/commands/rulebook-task-create.md +1 -1
- package/README.md +354 -965
- package/dist/cli/commands/analysis.d.ts +8 -0
- package/dist/cli/commands/analysis.d.ts.map +1 -0
- package/dist/cli/commands/analysis.js +78 -0
- package/dist/cli/commands/analysis.js.map +1 -0
- package/dist/cli/commands/context-intelligence.d.ts +33 -0
- package/dist/cli/commands/context-intelligence.d.ts.map +1 -0
- package/dist/cli/commands/context-intelligence.js +181 -0
- package/dist/cli/commands/context-intelligence.js.map +1 -0
- package/dist/cli/commands/index.d.ts +19 -0
- package/dist/cli/commands/index.d.ts.map +1 -0
- package/dist/cli/commands/index.js +19 -0
- package/dist/cli/commands/index.js.map +1 -0
- package/dist/cli/commands/init.d.ts +15 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +608 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/mcp.d.ts +10 -0
- package/dist/cli/commands/mcp.d.ts.map +1 -0
- package/dist/cli/commands/mcp.js +128 -0
- package/dist/cli/commands/mcp.js.map +1 -0
- package/dist/cli/commands/memory.d.ts +24 -0
- package/dist/cli/commands/memory.d.ts.map +1 -0
- package/dist/cli/commands/memory.js +265 -0
- package/dist/cli/commands/memory.js.map +1 -0
- package/dist/cli/commands/misc.d.ts +33 -0
- package/dist/cli/commands/misc.d.ts.map +1 -0
- package/dist/cli/commands/misc.js +576 -0
- package/dist/cli/commands/misc.js.map +1 -0
- package/dist/cli/commands/plans.d.ts +15 -0
- package/dist/cli/commands/plans.d.ts.map +1 -0
- package/dist/cli/commands/plans.js +266 -0
- package/dist/cli/commands/plans.js.map +1 -0
- package/dist/cli/commands/ralph.d.ts +45 -0
- package/dist/cli/commands/ralph.d.ts.map +1 -0
- package/dist/cli/commands/ralph.js +694 -0
- package/dist/cli/commands/ralph.js.map +1 -0
- package/dist/cli/commands/skills.d.ts +9 -0
- package/dist/cli/commands/skills.d.ts.map +1 -0
- package/dist/cli/commands/skills.js +249 -0
- package/dist/cli/commands/skills.js.map +1 -0
- package/dist/cli/commands/task.d.ts +16 -0
- package/dist/cli/commands/task.d.ts.map +1 -0
- package/dist/cli/commands/task.js +256 -0
- package/dist/cli/commands/task.js.map +1 -0
- package/dist/cli/commands/update.d.ts +14 -0
- package/dist/cli/commands/update.d.ts.map +1 -0
- package/dist/cli/commands/update.js +636 -0
- package/dist/cli/commands/update.js.map +1 -0
- package/dist/cli/commands/workspace.d.ts +6 -0
- package/dist/cli/commands/workspace.d.ts.map +1 -0
- package/dist/cli/commands/workspace.js +141 -0
- package/dist/cli/commands/workspace.js.map +1 -0
- package/dist/core/agent-template-engine.js +28 -28
- package/dist/core/analysis-manager.d.ts +56 -0
- package/dist/core/analysis-manager.d.ts.map +1 -0
- package/dist/core/analysis-manager.js +218 -0
- package/dist/core/analysis-manager.js.map +1 -0
- package/dist/core/claude-md-generator.d.ts +52 -0
- package/dist/core/claude-md-generator.d.ts.map +1 -0
- package/dist/core/claude-md-generator.js +104 -0
- package/dist/core/claude-md-generator.js.map +1 -0
- package/dist/core/claude-settings-manager.d.ts +37 -0
- package/dist/core/claude-settings-manager.d.ts.map +1 -0
- package/dist/core/claude-settings-manager.js +168 -0
- package/dist/core/claude-settings-manager.js.map +1 -0
- package/dist/core/compact-context-manager.d.ts +34 -0
- package/dist/core/compact-context-manager.d.ts.map +1 -0
- package/dist/core/compact-context-manager.js +60 -0
- package/dist/core/compact-context-manager.js.map +1 -0
- package/dist/core/doctor.d.ts +19 -0
- package/dist/core/doctor.d.ts.map +1 -0
- package/dist/core/doctor.js +163 -0
- package/dist/core/doctor.js.map +1 -0
- package/dist/core/generator.js +28 -28
- package/dist/core/mcp-reference-generator.d.ts +13 -0
- package/dist/core/mcp-reference-generator.d.ts.map +1 -0
- package/dist/core/mcp-reference-generator.js +66 -0
- package/dist/core/mcp-reference-generator.js.map +1 -0
- package/dist/core/merger.d.ts +35 -0
- package/dist/core/merger.d.ts.map +1 -1
- package/dist/core/merger.js +120 -0
- package/dist/core/merger.js.map +1 -1
- package/dist/core/prd-generator.d.ts.map +1 -1
- package/dist/core/prd-generator.js +7 -1
- package/dist/core/prd-generator.js.map +1 -1
- package/dist/core/ralph-manager.d.ts.map +1 -1
- package/dist/core/ralph-manager.js +17 -0
- package/dist/core/ralph-manager.js.map +1 -1
- package/dist/core/rules-generator.d.ts +73 -0
- package/dist/core/rules-generator.d.ts.map +1 -0
- package/dist/core/rules-generator.js +201 -0
- package/dist/core/rules-generator.js.map +1 -0
- package/dist/core/state-writer.d.ts +35 -0
- package/dist/core/state-writer.d.ts.map +1 -0
- package/dist/core/state-writer.js +81 -0
- package/dist/core/state-writer.js.map +1 -0
- package/dist/core/task-manager.d.ts +35 -0
- package/dist/core/task-manager.d.ts.map +1 -1
- package/dist/core/task-manager.js +137 -38
- package/dist/core/task-manager.js.map +1 -1
- package/dist/core/telemetry.d.ts +29 -0
- package/dist/core/telemetry.d.ts.map +1 -0
- package/dist/core/telemetry.js +57 -0
- package/dist/core/telemetry.js.map +1 -0
- package/dist/core/workflow-generator.d.ts.map +1 -1
- package/dist/core/workflow-generator.js +2 -177
- package/dist/core/workflow-generator.js.map +1 -1
- package/dist/index.js +28 -1
- package/dist/index.js.map +1 -1
- package/dist/mcp/rulebook-server.d.ts.map +1 -1
- package/dist/mcp/rulebook-server.js +190 -7
- package/dist/mcp/rulebook-server.js.map +1 -1
- package/dist/memory/memory-store.js +91 -91
- package/dist/types.d.ts +11 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/utils/gitignore.d.ts +10 -0
- package/dist/utils/gitignore.d.ts.map +1 -0
- package/dist/utils/gitignore.js +38 -0
- package/dist/utils/gitignore.js.map +1 -0
- package/package.json +1 -1
- package/templates/compact-context/_default.md +23 -0
- package/templates/compact-context/cpp.md +26 -0
- package/templates/compact-context/go.md +26 -0
- package/templates/compact-context/python.md +26 -0
- package/templates/compact-context/rust.md +28 -0
- package/templates/compact-context/typescript.md +29 -0
- package/templates/core/CLAUDE_MD_v2.md +71 -0
- package/templates/hooks/check-context-and-handoff.ps1 +50 -0
- package/templates/hooks/check-context-and-handoff.sh +69 -0
- package/templates/hooks/enforce-mcp-for-tasks.sh +31 -0
- package/templates/hooks/enforce-no-deferred.sh +21 -0
- package/templates/hooks/enforce-no-shortcuts.sh +31 -0
- package/templates/hooks/enforce-team-for-background-agents.ps1 +63 -0
- package/templates/hooks/enforce-team-for-background-agents.sh +55 -0
- package/templates/hooks/on-compact-reinject.sh +34 -0
- package/templates/hooks/resume-from-handoff.ps1 +33 -0
- package/templates/hooks/resume-from-handoff.sh +55 -0
- package/templates/rules/consult-analysis-before-implementing.md +23 -0
- package/templates/rules/cpp.md +46 -0
- package/templates/rules/csharp.md +44 -0
- package/templates/rules/diagnostic-first.md +39 -0
- package/templates/rules/fail-twice-escalate.md +46 -0
- package/templates/rules/go.md +40 -0
- package/templates/rules/java.md +43 -0
- package/templates/rules/javascript.md +39 -0
- package/templates/rules/multi-agent-teams.md +75 -0
- package/templates/rules/python.md +43 -0
- package/templates/rules/respect-handoff-trigger.md +41 -0
- package/templates/rules/rust.md +40 -0
- package/templates/rules/typescript.md +40 -0
- package/templates/skills/dev/analysis/SKILL.md +19 -0
- package/templates/skills/dev/handoff/SKILL.md +27 -0
|
@@ -0,0 +1,694 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
import { writeFile, ensureDir } from '../../utils/file-system.js';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
/**
|
|
5
|
+
* Build prompt for AI agent from user story context
|
|
6
|
+
*/
|
|
7
|
+
export function ralphBuildPrompt(task, prd, contextHistory, plansContext) {
|
|
8
|
+
const criteria = (task.acceptanceCriteria || []).map((c) => `- ${c}`).join('\n');
|
|
9
|
+
return [
|
|
10
|
+
`You are working on project: ${prd?.project || 'unknown'}`,
|
|
11
|
+
``,
|
|
12
|
+
plansContext ? `## Session Context (PLANS.md)\n${plansContext}\n` : '',
|
|
13
|
+
contextHistory && contextHistory !== 'No iteration history available.'
|
|
14
|
+
? `## Iteration History\n${contextHistory}\n`
|
|
15
|
+
: '',
|
|
16
|
+
`## Current Task: ${task.title}`,
|
|
17
|
+
`ID: ${task.id}`,
|
|
18
|
+
``,
|
|
19
|
+
`## Description`,
|
|
20
|
+
task.description,
|
|
21
|
+
``,
|
|
22
|
+
`## Acceptance Criteria`,
|
|
23
|
+
criteria,
|
|
24
|
+
``,
|
|
25
|
+
task.notes ? `## Notes\n${task.notes}\n` : '',
|
|
26
|
+
`## Instructions`,
|
|
27
|
+
`1. Implement the changes described above`,
|
|
28
|
+
`2. Ensure all acceptance criteria are met`,
|
|
29
|
+
`3. Run quality checks: type-check, lint, tests`,
|
|
30
|
+
`4. Fix any issues found by quality checks`,
|
|
31
|
+
`5. When done, summarize what was changed`,
|
|
32
|
+
]
|
|
33
|
+
.filter(Boolean)
|
|
34
|
+
.join('\n');
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Execute AI agent and capture output
|
|
38
|
+
*/
|
|
39
|
+
export async function ralphExecuteAgent(tool, prompt, cwd, spawn) {
|
|
40
|
+
const toolCommands = {
|
|
41
|
+
claude: {
|
|
42
|
+
cmd: 'claude',
|
|
43
|
+
args: ['-p', '--dangerously-skip-permissions', '--verbose'],
|
|
44
|
+
stdinPrompt: true,
|
|
45
|
+
},
|
|
46
|
+
amp: { cmd: 'amp', args: ['-p', prompt], stdinPrompt: false },
|
|
47
|
+
gemini: { cmd: 'gemini', args: ['-p', prompt], stdinPrompt: false },
|
|
48
|
+
};
|
|
49
|
+
const config = toolCommands[tool] || toolCommands.claude;
|
|
50
|
+
return new Promise((resolve, reject) => {
|
|
51
|
+
let output = '';
|
|
52
|
+
let errorOutput = '';
|
|
53
|
+
const proc = spawn(config.cmd, config.args, {
|
|
54
|
+
cwd,
|
|
55
|
+
shell: true,
|
|
56
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
57
|
+
env: { ...process.env },
|
|
58
|
+
});
|
|
59
|
+
if (config.stdinPrompt && proc.stdin) {
|
|
60
|
+
proc.stdin.write(prompt);
|
|
61
|
+
proc.stdin.end();
|
|
62
|
+
}
|
|
63
|
+
proc.stdout?.on('data', (data) => {
|
|
64
|
+
const text = data.toString();
|
|
65
|
+
output += text;
|
|
66
|
+
process.stdout.write(text);
|
|
67
|
+
});
|
|
68
|
+
proc.stderr?.on('data', (data) => {
|
|
69
|
+
errorOutput += data.toString();
|
|
70
|
+
});
|
|
71
|
+
proc.on('close', (code) => {
|
|
72
|
+
if (code === 0 || output.length > 0) {
|
|
73
|
+
resolve(output || errorOutput);
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
reject(new Error(`Agent ${tool} exited with code ${code}: ${errorOutput.slice(0, 500)}`));
|
|
77
|
+
}
|
|
78
|
+
});
|
|
79
|
+
proc.on('error', (err) => {
|
|
80
|
+
reject(new Error(`Failed to start ${tool}: ${err.message}`));
|
|
81
|
+
});
|
|
82
|
+
setTimeout(() => {
|
|
83
|
+
proc.kill('SIGTERM');
|
|
84
|
+
resolve(output || 'Agent execution timed out after 10 minutes');
|
|
85
|
+
}, 600000);
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Run quality gates and return results
|
|
90
|
+
*/
|
|
91
|
+
export async function ralphRunQualityGates(cwd, spawn) {
|
|
92
|
+
const runGate = (cmd, args) => {
|
|
93
|
+
return new Promise((resolve) => {
|
|
94
|
+
const proc = spawn(cmd, args, {
|
|
95
|
+
cwd,
|
|
96
|
+
shell: true,
|
|
97
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
98
|
+
});
|
|
99
|
+
proc.on('close', (code) => {
|
|
100
|
+
resolve(code === 0);
|
|
101
|
+
});
|
|
102
|
+
proc.on('error', () => {
|
|
103
|
+
resolve(false);
|
|
104
|
+
});
|
|
105
|
+
setTimeout(() => {
|
|
106
|
+
proc.kill('SIGTERM');
|
|
107
|
+
resolve(false);
|
|
108
|
+
}, 120000);
|
|
109
|
+
});
|
|
110
|
+
};
|
|
111
|
+
const { detectMonorepo } = await import('../../core/detector.js');
|
|
112
|
+
const monorepo = await detectMonorepo(cwd).catch(() => ({
|
|
113
|
+
detected: false,
|
|
114
|
+
tool: null,
|
|
115
|
+
packages: [],
|
|
116
|
+
}));
|
|
117
|
+
let testCmd = ['npm', ['test']];
|
|
118
|
+
if (monorepo.detected) {
|
|
119
|
+
if (monorepo.tool === 'turborepo')
|
|
120
|
+
testCmd = ['turbo', ['run', 'test']];
|
|
121
|
+
else if (monorepo.tool === 'nx')
|
|
122
|
+
testCmd = ['nx', ['run-many', '--target=test']];
|
|
123
|
+
}
|
|
124
|
+
const [typeCheck, lint, tests] = await Promise.all([
|
|
125
|
+
runGate('npm', ['run', 'type-check']),
|
|
126
|
+
runGate('npm', ['run', 'lint']),
|
|
127
|
+
runGate(testCmd[0], testCmd[1]),
|
|
128
|
+
]);
|
|
129
|
+
return {
|
|
130
|
+
type_check: typeCheck,
|
|
131
|
+
lint: lint,
|
|
132
|
+
tests: tests,
|
|
133
|
+
coverage_met: tests,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Create git branch from PRD branchName
|
|
138
|
+
*/
|
|
139
|
+
export async function ralphCreateBranch(cwd, branchName) {
|
|
140
|
+
const { readFileSync } = await import('fs');
|
|
141
|
+
const { spawn } = await import('child_process');
|
|
142
|
+
try {
|
|
143
|
+
const gitHeadPath = path.join(cwd, '.git', 'HEAD');
|
|
144
|
+
const head = readFileSync(gitHeadPath, 'utf8').trim();
|
|
145
|
+
const currentBranch = head.replace('ref: refs/heads/', '');
|
|
146
|
+
if (currentBranch === branchName) {
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
catch {
|
|
151
|
+
return;
|
|
152
|
+
}
|
|
153
|
+
await new Promise((resolve) => {
|
|
154
|
+
const proc = spawn('git', ['checkout', '-B', branchName], {
|
|
155
|
+
cwd,
|
|
156
|
+
shell: true,
|
|
157
|
+
stdio: 'pipe',
|
|
158
|
+
});
|
|
159
|
+
proc.on('close', () => resolve());
|
|
160
|
+
proc.on('error', () => resolve());
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Commit changes after successful iteration
|
|
165
|
+
*/
|
|
166
|
+
export async function ralphGitCommit(cwd, task, iteration, spawn) {
|
|
167
|
+
await new Promise((resolve) => {
|
|
168
|
+
const proc = spawn('git', ['add', '-A'], { cwd, shell: true, stdio: 'pipe' });
|
|
169
|
+
proc.on('close', () => resolve());
|
|
170
|
+
proc.on('error', () => resolve());
|
|
171
|
+
});
|
|
172
|
+
const hasChanges = await new Promise((resolve) => {
|
|
173
|
+
let output = '';
|
|
174
|
+
const proc = spawn('git', ['diff', '--cached', '--stat'], {
|
|
175
|
+
cwd,
|
|
176
|
+
shell: true,
|
|
177
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
178
|
+
});
|
|
179
|
+
proc.stdout?.on('data', (data) => {
|
|
180
|
+
output += data.toString();
|
|
181
|
+
});
|
|
182
|
+
proc.on('close', () => resolve(output.trim().length > 0));
|
|
183
|
+
proc.on('error', () => resolve(false));
|
|
184
|
+
});
|
|
185
|
+
if (!hasChanges) {
|
|
186
|
+
return undefined;
|
|
187
|
+
}
|
|
188
|
+
const commitMsg = `ralph(${task.id}): ${task.title}\n\nIteration ${iteration} - Ralph autonomous loop`;
|
|
189
|
+
const commitHash = await new Promise((resolve) => {
|
|
190
|
+
let output = '';
|
|
191
|
+
const proc = spawn('git', ['commit', '-m', commitMsg], {
|
|
192
|
+
cwd,
|
|
193
|
+
shell: true,
|
|
194
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
195
|
+
});
|
|
196
|
+
proc.stdout?.on('data', (data) => {
|
|
197
|
+
output += data.toString();
|
|
198
|
+
});
|
|
199
|
+
proc.on('close', (code) => {
|
|
200
|
+
if (code === 0) {
|
|
201
|
+
const hashMatch = output.match(/\[[\w/.-]+ ([a-f0-9]+)\]/);
|
|
202
|
+
resolve(hashMatch ? hashMatch[1] : undefined);
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
resolve(undefined);
|
|
206
|
+
}
|
|
207
|
+
});
|
|
208
|
+
proc.on('error', () => resolve(undefined));
|
|
209
|
+
});
|
|
210
|
+
return commitHash;
|
|
211
|
+
}
|
|
212
|
+
export async function ralphInitCommand() {
|
|
213
|
+
const oraModule = await import('ora');
|
|
214
|
+
const ora = oraModule.default;
|
|
215
|
+
const spinner = ora('Initializing Ralph autonomous loop...').start();
|
|
216
|
+
try {
|
|
217
|
+
const cwd = process.cwd();
|
|
218
|
+
const { Logger } = await import('../../core/logger.js');
|
|
219
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
220
|
+
const { PRDGenerator } = await import('../../core/prd-generator.js');
|
|
221
|
+
const { createConfigManager } = await import('../../core/config-manager.js');
|
|
222
|
+
const logger = new Logger(cwd);
|
|
223
|
+
const configManager = createConfigManager(cwd);
|
|
224
|
+
const config = await configManager.loadConfig();
|
|
225
|
+
const ralphManager = new RalphManager(cwd, logger);
|
|
226
|
+
const prdGenerator = new PRDGenerator(cwd, logger);
|
|
227
|
+
const maxIterations = config.ralph?.maxIterations || 10;
|
|
228
|
+
const tool = (config.ralph?.tool || 'claude');
|
|
229
|
+
await ralphManager.initialize(maxIterations, tool);
|
|
230
|
+
const prd = await prdGenerator.generatePRD(path.basename(cwd));
|
|
231
|
+
const prdPath = path.join(cwd, '.rulebook', 'ralph', 'prd.json');
|
|
232
|
+
await writeFile(prdPath, JSON.stringify(prd, null, 2));
|
|
233
|
+
spinner.succeed(`Ralph initialized: ${prd.userStories.length} user stories loaded`);
|
|
234
|
+
console.log(`\n 📋 PRD: ${prdPath}`);
|
|
235
|
+
console.log(` 🔄 Max iterations: ${maxIterations}`);
|
|
236
|
+
console.log(` 🤖 AI Tool: ${tool}`);
|
|
237
|
+
console.log(`\n Run: ${chalk.bold('rulebook ralph run')}\n`);
|
|
238
|
+
}
|
|
239
|
+
catch (error) {
|
|
240
|
+
spinner.fail('Ralph initialization failed');
|
|
241
|
+
console.error(chalk.red(String(error)));
|
|
242
|
+
process.exit(1);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
export async function ralphRunCommand(options) {
|
|
246
|
+
const oraModule = await import('ora');
|
|
247
|
+
const ora = oraModule.default;
|
|
248
|
+
const spinner = ora('Starting Ralph autonomous loop...').start();
|
|
249
|
+
try {
|
|
250
|
+
const cwd = process.cwd();
|
|
251
|
+
const { Logger } = await import('../../core/logger.js');
|
|
252
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
253
|
+
const { RalphParser } = await import('../../agents/ralph-parser.js');
|
|
254
|
+
const { createConfigManager } = await import('../../core/config-manager.js');
|
|
255
|
+
const { IterationTracker } = await import('../../core/iteration-tracker.js');
|
|
256
|
+
const childProcess = await import('child_process');
|
|
257
|
+
const logger = new Logger(cwd);
|
|
258
|
+
const configManager = createConfigManager(cwd);
|
|
259
|
+
const config = await configManager.loadConfig();
|
|
260
|
+
const ralphManager = new RalphManager(cwd, logger);
|
|
261
|
+
const maxIterations = options.maxIterations || config.ralph?.maxIterations || 10;
|
|
262
|
+
const tool = options.tool || config.ralph?.tool || 'claude';
|
|
263
|
+
const parallelWorkers = options.parallel ??
|
|
264
|
+
(config.ralph?.parallel?.enabled ? config.ralph.parallel.maxWorkers : undefined);
|
|
265
|
+
const planCheckpointConfig = {
|
|
266
|
+
enabled: options.planFirst ?? config.ralph?.planCheckpoint?.enabled ?? false,
|
|
267
|
+
autoApproveAfterSeconds: config.ralph?.planCheckpoint?.autoApproveAfterSeconds ?? 0,
|
|
268
|
+
requireApprovalForStories: config.ralph?.planCheckpoint?.requireApprovalForStories ?? 'all',
|
|
269
|
+
};
|
|
270
|
+
const compressionConfig = config.ralph?.contextCompression;
|
|
271
|
+
const compressionEnabled = compressionConfig?.enabled !== false;
|
|
272
|
+
const compressionRecentCount = compressionConfig?.recentCount ?? 3;
|
|
273
|
+
const compressionThreshold = compressionConfig?.threshold ?? 5;
|
|
274
|
+
const iterationTracker = new IterationTracker(cwd, logger);
|
|
275
|
+
await iterationTracker.initialize();
|
|
276
|
+
await ralphManager.initialize(maxIterations, tool);
|
|
277
|
+
const prd = await ralphManager.loadPRD();
|
|
278
|
+
if (prd?.branchName) {
|
|
279
|
+
await ralphCreateBranch(cwd, prd.branchName);
|
|
280
|
+
}
|
|
281
|
+
let interrupted = false;
|
|
282
|
+
const handleInterrupt = async () => {
|
|
283
|
+
interrupted = true;
|
|
284
|
+
spinner.warn('Pausing after current iteration...');
|
|
285
|
+
await ralphManager.pause();
|
|
286
|
+
};
|
|
287
|
+
process.on('SIGINT', handleInterrupt);
|
|
288
|
+
await ralphManager.refreshTaskCount();
|
|
289
|
+
if (parallelWorkers && parallelWorkers > 1) {
|
|
290
|
+
spinner.text = `Ralph parallel mode (${parallelWorkers} workers)...`;
|
|
291
|
+
const batches = await ralphManager.getParallelBatches(parallelWorkers);
|
|
292
|
+
spinner.stop();
|
|
293
|
+
console.log(chalk.bold.cyan(`\n Parallel mode: ${batches.length} batch(es), max ${parallelWorkers} workers\n`));
|
|
294
|
+
let iterationCount = 0;
|
|
295
|
+
for (const batch of batches) {
|
|
296
|
+
if (interrupted)
|
|
297
|
+
break;
|
|
298
|
+
console.log(chalk.bold(` ── Batch: ${batch.map((s) => s.id).join(', ')} (${batch.length} stories) ──`));
|
|
299
|
+
const batchResults = await Promise.allSettled(batch.map(async (task) => {
|
|
300
|
+
iterationCount++;
|
|
301
|
+
const localIteration = iterationCount;
|
|
302
|
+
const startTime = Date.now();
|
|
303
|
+
let contextHistory = '';
|
|
304
|
+
if (compressionEnabled) {
|
|
305
|
+
contextHistory = await iterationTracker.buildCompressedContext(compressionRecentCount, compressionThreshold);
|
|
306
|
+
}
|
|
307
|
+
let plansContext = '';
|
|
308
|
+
try {
|
|
309
|
+
const { readPlans, plansExists } = await import('../../core/plans-manager.js');
|
|
310
|
+
if (plansExists(cwd)) {
|
|
311
|
+
const plans = await readPlans(cwd);
|
|
312
|
+
if (plans?.context && plans.context.trim()) {
|
|
313
|
+
plansContext = plans.context.trim();
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
catch {
|
|
318
|
+
// PLANS.md injection is optional
|
|
319
|
+
}
|
|
320
|
+
const prompt = ralphBuildPrompt(task, prd, contextHistory, plansContext);
|
|
321
|
+
let agentOutput = '';
|
|
322
|
+
try {
|
|
323
|
+
agentOutput = await ralphExecuteAgent(tool, prompt, cwd, childProcess.spawn);
|
|
324
|
+
}
|
|
325
|
+
catch (agentError) {
|
|
326
|
+
agentOutput = `Error executing agent: ${agentError.message || agentError}`;
|
|
327
|
+
}
|
|
328
|
+
const qualityResults = await ralphRunQualityGates(cwd, childProcess.spawn);
|
|
329
|
+
const executionTime = Date.now() - startTime;
|
|
330
|
+
const parsed = RalphParser.parseAgentOutput(agentOutput, localIteration, task.id, task.title, tool);
|
|
331
|
+
const allGatesPass = qualityResults.type_check &&
|
|
332
|
+
qualityResults.lint &&
|
|
333
|
+
qualityResults.tests &&
|
|
334
|
+
qualityResults.coverage_met;
|
|
335
|
+
const passCount = Object.values(qualityResults).filter(Boolean).length;
|
|
336
|
+
const status = allGatesPass
|
|
337
|
+
? 'success'
|
|
338
|
+
: passCount >= 2
|
|
339
|
+
? 'partial'
|
|
340
|
+
: 'failed';
|
|
341
|
+
let gitCommit;
|
|
342
|
+
if (allGatesPass) {
|
|
343
|
+
gitCommit = await ralphGitCommit(cwd, task, localIteration, childProcess.spawn);
|
|
344
|
+
await ralphManager.markStoryComplete(task.id);
|
|
345
|
+
console.log(chalk.green(` [parallel] Story ${task.id} completed`));
|
|
346
|
+
}
|
|
347
|
+
else {
|
|
348
|
+
console.log(chalk.yellow(` [parallel] Story ${task.id} not completed (quality gates failed)`));
|
|
349
|
+
}
|
|
350
|
+
const result = {
|
|
351
|
+
iteration: localIteration,
|
|
352
|
+
timestamp: new Date().toISOString(),
|
|
353
|
+
task_id: task.id,
|
|
354
|
+
task_title: task.title,
|
|
355
|
+
status,
|
|
356
|
+
ai_tool: tool,
|
|
357
|
+
execution_time_ms: executionTime,
|
|
358
|
+
quality_checks: qualityResults,
|
|
359
|
+
output_summary: parsed.output_summary || `Iteration ${localIteration}: ${task.title}`,
|
|
360
|
+
git_commit: gitCommit,
|
|
361
|
+
learnings: parsed.learnings,
|
|
362
|
+
errors: parsed.errors,
|
|
363
|
+
metadata: {
|
|
364
|
+
context_loss_count: parsed.metadata.context_loss_count,
|
|
365
|
+
parsed_completion: parsed.metadata.parsed_completion,
|
|
366
|
+
},
|
|
367
|
+
};
|
|
368
|
+
await ralphManager.recordIteration(result);
|
|
369
|
+
return result;
|
|
370
|
+
}));
|
|
371
|
+
for (const [i, result] of batchResults.entries()) {
|
|
372
|
+
if (result.status === 'rejected') {
|
|
373
|
+
const story = batch[i];
|
|
374
|
+
console.log(chalk.red(` [parallel] Story ${story.id} threw: ${result.reason}`));
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
const currentStatus = await ralphManager.getStatus();
|
|
378
|
+
if (currentStatus?.paused)
|
|
379
|
+
break;
|
|
380
|
+
}
|
|
381
|
+
process.removeListener('SIGINT', handleInterrupt);
|
|
382
|
+
const stats = await ralphManager.getTaskStats();
|
|
383
|
+
console.log(`\n Parallel run complete: ${stats.completed}/${stats.total} tasks completed`);
|
|
384
|
+
console.log(` Iterations: ${iterationCount}`);
|
|
385
|
+
if (interrupted) {
|
|
386
|
+
console.log(chalk.yellow(` Paused by user. Resume: ${chalk.bold('rulebook ralph resume')}`));
|
|
387
|
+
}
|
|
388
|
+
console.log(`\n View history: ${chalk.bold('rulebook ralph history')}\n`);
|
|
389
|
+
return;
|
|
390
|
+
}
|
|
391
|
+
spinner.text = 'Ralph loop running (Ctrl+C to pause)...';
|
|
392
|
+
let iterationCount = 0;
|
|
393
|
+
while (ralphManager.canContinue() && !interrupted) {
|
|
394
|
+
iterationCount++;
|
|
395
|
+
const task = await ralphManager.getNextTask();
|
|
396
|
+
if (!task) {
|
|
397
|
+
break;
|
|
398
|
+
}
|
|
399
|
+
spinner.stop();
|
|
400
|
+
console.log(chalk.bold.cyan(`\n ── Iteration ${iterationCount}: ${task.title} ──\n`));
|
|
401
|
+
const startTime = Date.now();
|
|
402
|
+
if (planCheckpointConfig.enabled) {
|
|
403
|
+
const checkpoint = await ralphManager.runCheckpoint(task, tool, planCheckpointConfig);
|
|
404
|
+
if (!checkpoint.proceed) {
|
|
405
|
+
console.log(chalk.yellow(` Plan rejected for ${task.id}. Skipping implementation.`));
|
|
406
|
+
if (checkpoint.feedback) {
|
|
407
|
+
console.log(chalk.gray(` Feedback: ${checkpoint.feedback}`));
|
|
408
|
+
}
|
|
409
|
+
spinner.start('Preparing next iteration...');
|
|
410
|
+
continue;
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
let contextHistory = '';
|
|
414
|
+
if (compressionEnabled) {
|
|
415
|
+
contextHistory = await iterationTracker.buildCompressedContext(compressionRecentCount, compressionThreshold);
|
|
416
|
+
}
|
|
417
|
+
let plansContext = '';
|
|
418
|
+
try {
|
|
419
|
+
const { readPlans, plansExists } = await import('../../core/plans-manager.js');
|
|
420
|
+
if (plansExists(cwd)) {
|
|
421
|
+
const plans = await readPlans(cwd);
|
|
422
|
+
if (plans?.context && plans.context.trim()) {
|
|
423
|
+
plansContext = plans.context.trim();
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
catch {
|
|
428
|
+
// PLANS.md injection is optional — skip on error
|
|
429
|
+
}
|
|
430
|
+
const prompt = ralphBuildPrompt(task, prd, contextHistory, plansContext);
|
|
431
|
+
let agentOutput = '';
|
|
432
|
+
try {
|
|
433
|
+
agentOutput = await ralphExecuteAgent(tool, prompt, cwd, childProcess.spawn);
|
|
434
|
+
}
|
|
435
|
+
catch (agentError) {
|
|
436
|
+
agentOutput = `Error executing agent: ${agentError.message || agentError}`;
|
|
437
|
+
console.log(chalk.red(` Agent error: ${agentError.message || agentError}`));
|
|
438
|
+
}
|
|
439
|
+
spinner.start('Running quality gates...');
|
|
440
|
+
const qualityResults = await ralphRunQualityGates(cwd, childProcess.spawn);
|
|
441
|
+
spinner.stop();
|
|
442
|
+
const gateIcon = (pass) => (pass ? chalk.green('✓') : chalk.red('✗'));
|
|
443
|
+
console.log(` ${gateIcon(qualityResults.type_check)} type-check`);
|
|
444
|
+
console.log(` ${gateIcon(qualityResults.lint)} lint`);
|
|
445
|
+
console.log(` ${gateIcon(qualityResults.tests)} tests`);
|
|
446
|
+
console.log(` ${gateIcon(qualityResults.coverage_met)} coverage`);
|
|
447
|
+
const executionTime = Date.now() - startTime;
|
|
448
|
+
const parsed = RalphParser.parseAgentOutput(agentOutput, iterationCount, task.id, task.title, tool);
|
|
449
|
+
const allGatesPass = qualityResults.type_check &&
|
|
450
|
+
qualityResults.lint &&
|
|
451
|
+
qualityResults.tests &&
|
|
452
|
+
qualityResults.coverage_met;
|
|
453
|
+
const passCount = Object.values(qualityResults).filter(Boolean).length;
|
|
454
|
+
const status = allGatesPass
|
|
455
|
+
? 'success'
|
|
456
|
+
: passCount >= 2
|
|
457
|
+
? 'partial'
|
|
458
|
+
: 'failed';
|
|
459
|
+
let gitCommit;
|
|
460
|
+
if (allGatesPass) {
|
|
461
|
+
gitCommit = await ralphGitCommit(cwd, task, iterationCount, childProcess.spawn);
|
|
462
|
+
await ralphManager.markStoryComplete(task.id);
|
|
463
|
+
console.log(chalk.green(`\n ✅ Story ${task.id} completed`));
|
|
464
|
+
}
|
|
465
|
+
else {
|
|
466
|
+
console.log(chalk.yellow(`\n ⚠ Story ${task.id} not completed (quality gates failed)`));
|
|
467
|
+
}
|
|
468
|
+
const result = {
|
|
469
|
+
iteration: iterationCount,
|
|
470
|
+
timestamp: new Date().toISOString(),
|
|
471
|
+
task_id: task.id,
|
|
472
|
+
task_title: task.title,
|
|
473
|
+
status,
|
|
474
|
+
ai_tool: tool,
|
|
475
|
+
execution_time_ms: executionTime,
|
|
476
|
+
quality_checks: qualityResults,
|
|
477
|
+
output_summary: parsed.output_summary || `Iteration ${iterationCount}: ${task.title}`,
|
|
478
|
+
git_commit: gitCommit,
|
|
479
|
+
learnings: parsed.learnings,
|
|
480
|
+
errors: parsed.errors,
|
|
481
|
+
metadata: {
|
|
482
|
+
context_loss_count: parsed.metadata.context_loss_count,
|
|
483
|
+
parsed_completion: parsed.metadata.parsed_completion,
|
|
484
|
+
},
|
|
485
|
+
};
|
|
486
|
+
await ralphManager.recordIteration(result);
|
|
487
|
+
spinner.start('Preparing next iteration...');
|
|
488
|
+
}
|
|
489
|
+
process.removeListener('SIGINT', handleInterrupt);
|
|
490
|
+
const stats = await ralphManager.getTaskStats();
|
|
491
|
+
spinner.succeed(`Ralph loop complete: ${stats.completed}/${stats.total} tasks completed`);
|
|
492
|
+
console.log(`\n ✅ Iterations: ${iterationCount}`);
|
|
493
|
+
console.log(` 📊 Completed: ${stats.completed}/${stats.total}`);
|
|
494
|
+
if (interrupted) {
|
|
495
|
+
console.log(chalk.yellow(` ⏸ Paused by user. Resume: ${chalk.bold('rulebook ralph resume')}`));
|
|
496
|
+
}
|
|
497
|
+
console.log(`\n View history: ${chalk.bold('rulebook ralph history')}\n`);
|
|
498
|
+
}
|
|
499
|
+
catch (error) {
|
|
500
|
+
spinner.fail('Ralph loop failed');
|
|
501
|
+
console.error(chalk.red(String(error)));
|
|
502
|
+
process.exit(1);
|
|
503
|
+
}
|
|
504
|
+
}
|
|
505
|
+
export async function ralphStatusCommand() {
|
|
506
|
+
const oraModule = await import('ora');
|
|
507
|
+
const ora = oraModule.default;
|
|
508
|
+
const spinner = ora('Loading Ralph status...').start();
|
|
509
|
+
try {
|
|
510
|
+
const cwd = process.cwd();
|
|
511
|
+
const { Logger } = await import('../../core/logger.js');
|
|
512
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
513
|
+
const logger = new Logger(cwd);
|
|
514
|
+
const ralphManager = new RalphManager(cwd, logger);
|
|
515
|
+
const status = await ralphManager.getStatus();
|
|
516
|
+
if (!status) {
|
|
517
|
+
spinner.fail('Ralph not initialized');
|
|
518
|
+
console.log(`\n Run: ${chalk.bold('rulebook ralph init')}\n`);
|
|
519
|
+
return;
|
|
520
|
+
}
|
|
521
|
+
spinner.stop();
|
|
522
|
+
const { createConfigManager } = await import('../../core/config-manager.js');
|
|
523
|
+
const configManager = createConfigManager(cwd);
|
|
524
|
+
const cfg = await configManager.loadConfig();
|
|
525
|
+
const agentsMode = cfg.agentsMode ?? 'full';
|
|
526
|
+
console.log(`\n ${chalk.bold('Ralph Loop Status')}`);
|
|
527
|
+
console.log(` Iteration: ${status.current_iteration}/${status.max_iterations}`);
|
|
528
|
+
console.log(` Tasks: ${status.completed_tasks}/${status.total_tasks}`);
|
|
529
|
+
console.log(` Status: ${status.paused ? chalk.yellow('PAUSED') : chalk.green('RUNNING')}`);
|
|
530
|
+
console.log(` AI Tool: ${status.tool}`);
|
|
531
|
+
console.log(` Started: ${new Date(status.started_at).toLocaleString()}`);
|
|
532
|
+
console.log(` Agents Mode: ${agentsMode === 'lean' ? chalk.cyan('lean') : chalk.gray('full')} (rulebook mode set lean|full)`);
|
|
533
|
+
console.log();
|
|
534
|
+
}
|
|
535
|
+
catch (error) {
|
|
536
|
+
spinner.fail('Failed to load status');
|
|
537
|
+
console.error(chalk.red(String(error)));
|
|
538
|
+
process.exit(1);
|
|
539
|
+
}
|
|
540
|
+
}
|
|
541
|
+
export async function ralphHistoryCommand(options) {
|
|
542
|
+
const oraModule = await import('ora');
|
|
543
|
+
const ora = oraModule.default;
|
|
544
|
+
const spinner = ora('Loading iteration history...').start();
|
|
545
|
+
try {
|
|
546
|
+
const cwd = process.cwd();
|
|
547
|
+
const { Logger } = await import('../../core/logger.js');
|
|
548
|
+
const { IterationTracker } = await import('../../core/iteration-tracker.js');
|
|
549
|
+
const logger = new Logger(cwd);
|
|
550
|
+
const tracker = new IterationTracker(cwd, logger);
|
|
551
|
+
const limit = options.limit || 10;
|
|
552
|
+
const history = await tracker.getHistory(limit);
|
|
553
|
+
if (history.length === 0) {
|
|
554
|
+
spinner.fail('No iteration history found');
|
|
555
|
+
return;
|
|
556
|
+
}
|
|
557
|
+
spinner.stop();
|
|
558
|
+
console.log(`\n ${chalk.bold('Recent Iterations')} (${history.length})\n`);
|
|
559
|
+
for (const iter of history) {
|
|
560
|
+
const statusIcon = iter.status === 'success'
|
|
561
|
+
? chalk.green('✓')
|
|
562
|
+
: iter.status === 'partial'
|
|
563
|
+
? chalk.yellow('◐')
|
|
564
|
+
: chalk.red('✗');
|
|
565
|
+
console.log(` ${statusIcon} Iteration ${iter.iteration}: ${iter.task_title}`);
|
|
566
|
+
console.log(` Status: ${iter.status} | Duration: ${(iter.duration_ms || 0) / 1000}s`);
|
|
567
|
+
console.log(` Checks: type=${iter.quality_checks.type_check ? '✓' : '✗'} lint=${iter.quality_checks.lint ? '✓' : '✗'} tests=${iter.quality_checks.tests ? '✓' : '✗'}`);
|
|
568
|
+
if (iter.git_commit) {
|
|
569
|
+
console.log(` Commit: ${iter.git_commit}`);
|
|
570
|
+
}
|
|
571
|
+
console.log();
|
|
572
|
+
}
|
|
573
|
+
const stats = await tracker.getStatistics();
|
|
574
|
+
console.log(` ${chalk.bold('Statistics')}`);
|
|
575
|
+
console.log(` Total: ${stats.total_iterations} | Success: ${stats.successful_iterations} | Failed: ${stats.failed_iterations}`);
|
|
576
|
+
console.log(` Success rate: ${(stats.success_rate * 100).toFixed(1)}%`);
|
|
577
|
+
console.log(` Avg duration: ${stats.average_duration_ms}ms\n`);
|
|
578
|
+
}
|
|
579
|
+
catch (error) {
|
|
580
|
+
spinner.fail('Failed to load history');
|
|
581
|
+
console.error(chalk.red(String(error)));
|
|
582
|
+
process.exit(1);
|
|
583
|
+
}
|
|
584
|
+
}
|
|
585
|
+
export async function ralphPauseCommand() {
|
|
586
|
+
const oraModule = await import('ora');
|
|
587
|
+
const ora = oraModule.default;
|
|
588
|
+
const spinner = ora('Pausing Ralph loop...').start();
|
|
589
|
+
try {
|
|
590
|
+
const cwd = process.cwd();
|
|
591
|
+
const { Logger } = await import('../../core/logger.js');
|
|
592
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
593
|
+
const logger = new Logger(cwd);
|
|
594
|
+
const ralphManager = new RalphManager(cwd, logger);
|
|
595
|
+
const status = await ralphManager.getStatus();
|
|
596
|
+
if (!status) {
|
|
597
|
+
spinner.fail('Ralph not initialized');
|
|
598
|
+
console.log(`\n Run: ${chalk.bold('rulebook ralph init')}\n`);
|
|
599
|
+
return;
|
|
600
|
+
}
|
|
601
|
+
await ralphManager.pause();
|
|
602
|
+
spinner.succeed('Ralph loop paused');
|
|
603
|
+
console.log(`\n Resume with: ${chalk.bold('rulebook ralph resume')}\n`);
|
|
604
|
+
}
|
|
605
|
+
catch (error) {
|
|
606
|
+
spinner.fail('Failed to pause');
|
|
607
|
+
console.error(chalk.red(String(error)));
|
|
608
|
+
process.exit(1);
|
|
609
|
+
}
|
|
610
|
+
}
|
|
611
|
+
export async function ralphResumeCommand() {
|
|
612
|
+
const oraModule = await import('ora');
|
|
613
|
+
const ora = oraModule.default;
|
|
614
|
+
const spinner = ora('Resuming Ralph loop...').start();
|
|
615
|
+
try {
|
|
616
|
+
const cwd = process.cwd();
|
|
617
|
+
const { Logger } = await import('../../core/logger.js');
|
|
618
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
619
|
+
const logger = new Logger(cwd);
|
|
620
|
+
const ralphManager = new RalphManager(cwd, logger);
|
|
621
|
+
const status = await ralphManager.getStatus();
|
|
622
|
+
if (!status) {
|
|
623
|
+
spinner.fail('Ralph not initialized');
|
|
624
|
+
console.log(`\n Run: ${chalk.bold('rulebook ralph init')}\n`);
|
|
625
|
+
return;
|
|
626
|
+
}
|
|
627
|
+
await ralphManager.resume();
|
|
628
|
+
spinner.succeed('Ralph loop resumed');
|
|
629
|
+
console.log(`\n Continue loop: ${chalk.bold('rulebook ralph run')}\n`);
|
|
630
|
+
}
|
|
631
|
+
catch (error) {
|
|
632
|
+
spinner.fail('Failed to resume');
|
|
633
|
+
console.error(chalk.red(String(error)));
|
|
634
|
+
process.exit(1);
|
|
635
|
+
}
|
|
636
|
+
}
|
|
637
|
+
export async function ralphImportIssuesCommand(options) {
|
|
638
|
+
const oraModule = await import('ora');
|
|
639
|
+
const ora = oraModule.default;
|
|
640
|
+
try {
|
|
641
|
+
const { checkGhCliAvailable, fetchGithubIssues, convertIssueToStory, mergeStoriesIntoExistingPrd, } = await import('../../core/github-issues-importer.js');
|
|
642
|
+
const ghAvailable = await checkGhCliAvailable();
|
|
643
|
+
if (!ghAvailable) {
|
|
644
|
+
console.error(chalk.red('GitHub CLI (gh) is not installed. Install it from: https://cli.github.com'));
|
|
645
|
+
return;
|
|
646
|
+
}
|
|
647
|
+
const spinner = ora('Fetching GitHub issues...').start();
|
|
648
|
+
const issues = await fetchGithubIssues({
|
|
649
|
+
label: options.label,
|
|
650
|
+
milestone: options.milestone,
|
|
651
|
+
limit: options.limit ?? 20,
|
|
652
|
+
});
|
|
653
|
+
if (issues.length === 0) {
|
|
654
|
+
spinner.info('No open issues found matching the given filters.');
|
|
655
|
+
return;
|
|
656
|
+
}
|
|
657
|
+
spinner.text = `Converting ${issues.length} issues to Ralph stories...`;
|
|
658
|
+
const cwd = process.cwd();
|
|
659
|
+
let existingPrd = null;
|
|
660
|
+
try {
|
|
661
|
+
const { RalphManager } = await import('../../core/ralph-manager.js');
|
|
662
|
+
const { Logger } = await import('../../core/logger.js');
|
|
663
|
+
const logger = new Logger(cwd);
|
|
664
|
+
const manager = new RalphManager(cwd, logger);
|
|
665
|
+
existingPrd = await manager.loadPRD();
|
|
666
|
+
}
|
|
667
|
+
catch {
|
|
668
|
+
// PRD not initialized — will create a new one
|
|
669
|
+
}
|
|
670
|
+
const newStories = issues.map((issue) => convertIssueToStory(issue));
|
|
671
|
+
const { prd: mergedPrd, result } = mergeStoriesIntoExistingPrd(existingPrd, newStories);
|
|
672
|
+
if (options.dryRun) {
|
|
673
|
+
spinner.stop();
|
|
674
|
+
console.log(chalk.yellow(`Dry run — would import ${result.imported} stories, update ${result.updated}, skip ${result.skipped}`));
|
|
675
|
+
console.log('');
|
|
676
|
+
for (const story of mergedPrd.userStories) {
|
|
677
|
+
const marker = story.passes ? chalk.green('[PASS]') : chalk.gray('[ ]');
|
|
678
|
+
console.log(` ${marker} ${story.id}: ${story.title}`);
|
|
679
|
+
}
|
|
680
|
+
return;
|
|
681
|
+
}
|
|
682
|
+
const prdPath = path.join(cwd, '.rulebook', 'ralph', 'prd.json');
|
|
683
|
+
await ensureDir(path.join(cwd, '.rulebook', 'ralph'));
|
|
684
|
+
await writeFile(prdPath, JSON.stringify(mergedPrd, null, 2));
|
|
685
|
+
spinner.succeed(`Imported ${result.imported} new stories, updated ${result.updated} existing, ${result.skipped} skipped`);
|
|
686
|
+
console.log(`\n PRD saved to: ${prdPath}`);
|
|
687
|
+
console.log(` Total stories: ${mergedPrd.userStories.length}\n`);
|
|
688
|
+
}
|
|
689
|
+
catch (error) {
|
|
690
|
+
console.error(chalk.red(`Failed to import GitHub issues: ${String(error)}`));
|
|
691
|
+
process.exit(1);
|
|
692
|
+
}
|
|
693
|
+
}
|
|
694
|
+
//# sourceMappingURL=ralph.js.map
|