cc-devflow 2.4.6 → 4.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/.claude/CLAUDE.md +1065 -48
- package/.claude/agents/dev-implementer.md +195 -0
- package/.claude/commands/{flow-archive.md → flow/archive.md} +46 -11
- package/.claude/commands/flow/context.md +150 -0
- package/.claude/commands/flow/delta.md +245 -0
- package/.claude/commands/{flow-dev.md → flow/dev.md} +112 -11
- package/.claude/commands/flow/init.md +45 -0
- package/.claude/commands/flow/quality.md +159 -0
- package/.claude/commands/flow/spec.md +186 -0
- package/.claude/commands/flow/workspace.md +146 -0
- package/.claude/commands/{cancel-ralph.md → util/cancel-ralph.md} +1 -0
- package/.claude/config/quality-gates.yml +305 -0
- package/.claude/docs/guides/TEAM_MODE_GUIDE.md +313 -0
- package/.claude/docs/templates/DELTA_SPEC_TEMPLATE.md +91 -0
- package/.claude/docs/templates/DESIGN_DECISIONS_TEMPLATE.md +151 -0
- package/.claude/docs/templates/JOURNAL_TEMPLATE.md +75 -0
- package/.claude/docs/templates/_shared/CLAUDE.md +36 -0
- package/.claude/docs/templates/_shared/CONSTITUTION_CHECK.md +125 -0
- package/.claude/docs/templates/_shared/VALIDATION_CHECKLIST.md +187 -0
- package/.claude/docs/templates/_shared/YAML_FRONTMATTER.md +164 -0
- package/.claude/docs/templates/context/dev.jsonl.template +6 -0
- package/.claude/docs/templates/context/epic.jsonl.template +5 -0
- package/.claude/docs/templates/context/prd.jsonl.template +4 -0
- package/.claude/docs/templates/context/research.jsonl.template +4 -0
- package/.claude/docs/templates/context/review.jsonl.template +5 -0
- package/.claude/docs/templates/context/tech.jsonl.template +5 -0
- package/.claude/hooks/CLAUDE.md +342 -0
- package/.claude/hooks/inject-agent-context.ts +480 -0
- package/.claude/hooks/inject-skill-context.ts +359 -0
- package/.claude/hooks/ralph-loop.ts +931 -0
- package/.claude/hooks/task-completed-hook.ts +593 -0
- package/.claude/hooks/teammate-idle-hook.ts +690 -0
- package/.claude/hooks/types/team-types.d.ts +238 -0
- package/.claude/rules/devflow-conventions.md +82 -9
- package/.claude/scripts/archive-requirement.sh +44 -1
- package/.claude/scripts/common.sh +670 -3
- package/.claude/scripts/delta-parser.ts +527 -0
- package/.claude/scripts/detect-file-conflicts.sh +151 -0
- package/.claude/scripts/flow-context-add.sh +134 -0
- package/.claude/scripts/flow-context-init.sh +133 -0
- package/.claude/scripts/flow-context-validate.sh +144 -0
- package/.claude/scripts/flow-delta-apply.sh +297 -0
- package/.claude/scripts/flow-delta-archive.sh +71 -0
- package/.claude/scripts/flow-delta-create.sh +202 -0
- package/.claude/scripts/flow-delta-list.sh +142 -0
- package/.claude/scripts/flow-delta-status.sh +235 -0
- package/.claude/scripts/flow-quality-full.sh +184 -0
- package/.claude/scripts/flow-quality-quick.sh +64 -0
- package/.claude/scripts/flow-workspace-init.sh +117 -0
- package/.claude/scripts/flow-workspace-record.sh +164 -0
- package/.claude/scripts/flow-workspace-start.sh +88 -0
- package/.claude/scripts/get-workflow-status.sh +415 -0
- package/.claude/scripts/parse-task-dependencies.js +334 -0
- package/.claude/scripts/record-quality-error.sh +165 -0
- package/.claude/scripts/run-quality-gates.sh +242 -0
- package/.claude/scripts/team-dev-init.sh +319 -0
- package/.claude/scripts/team-state-recovery.sh +229 -0
- package/.claude/scripts/workflow-status.ts +433 -0
- package/.claude/settings.json +19 -0
- package/.claude/skills/cc-devflow-orchestrator/SKILL.md +85 -200
- package/.claude/skills/domain/using-git-worktrees/SKILL.md +252 -0
- package/.claude/skills/domain/using-git-worktrees/assets/SHELL_ALIASES.md +133 -0
- package/.claude/skills/domain/using-git-worktrees/context.jsonl +4 -0
- package/.claude/skills/domain/using-git-worktrees/scripts/worktree-cleanup.sh +218 -0
- package/.claude/skills/domain/using-git-worktrees/scripts/worktree-create.sh +232 -0
- package/.claude/skills/domain/using-git-worktrees/scripts/worktree-list.sh +130 -0
- package/.claude/skills/domain/using-git-worktrees/scripts/worktree-status.sh +140 -0
- package/.claude/skills/domain/using-git-worktrees/scripts/worktree-switch.sh +70 -0
- package/.claude/skills/skill-rules.json +72 -1
- package/.claude/skills/utility/journey-checker/SKILL.md +199 -0
- package/.claude/skills/utility/journey-checker/pressure-scenarios.md +164 -0
- package/.claude/skills/utility/skill-creator/LICENSE.txt +202 -0
- package/.claude/skills/utility/skill-creator/SKILL.md +356 -0
- package/.claude/skills/utility/skill-creator/references/output-patterns.md +82 -0
- package/.claude/skills/utility/skill-creator/references/workflows.md +28 -0
- package/.claude/skills/utility/skill-creator/scripts/init_skill.py +303 -0
- package/.claude/skills/utility/skill-creator/scripts/package_skill.py +110 -0
- package/.claude/skills/utility/skill-creator/scripts/quick_validate.py +95 -0
- package/.claude/skills/workflow/flow-dev/CLAUDE.md +78 -0
- package/.claude/skills/workflow/flow-dev/SKILL.md +96 -0
- package/.claude/skills/workflow/flow-dev/assets/IMPLEMENTATION_PLAN_TEMPLATE.md +71 -0
- package/.claude/skills/workflow/flow-dev/context.jsonl +8 -0
- package/.claude/skills/workflow/flow-dev/dev-implementer.jsonl +8 -0
- package/.claude/skills/workflow/flow-dev/scripts/entry-gate.sh +116 -0
- package/.claude/skills/workflow/flow-dev/scripts/exit-gate.sh +101 -0
- package/.claude/skills/workflow/flow-dev/scripts/task-orchestrator.sh +106 -0
- package/.claude/skills/workflow/flow-fix/SKILL.md +105 -0
- package/.claude/skills/workflow/flow-fix/context.jsonl +6 -0
- package/.claude/skills/workflow/flow-fix/references/bug-analyzer.md +381 -0
- package/.claude/skills/workflow/flow-init/SKILL.md +211 -0
- package/.claude/skills/workflow/flow-init/assets/BRAINSTORM_TEMPLATE.md +148 -0
- package/.claude/skills/workflow/flow-init/assets/INIT_FLOW_TEMPLATE.md +198 -0
- package/.claude/skills/workflow/flow-init/assets/RESEARCH_TEMPLATE.md +276 -0
- package/.claude/skills/workflow/flow-init/context.jsonl +5 -0
- package/.claude/skills/workflow/flow-init/references/flow-researcher.md +132 -0
- package/.claude/skills/workflow/flow-init/scripts/check-prerequisites.sh +232 -0
- package/.claude/skills/workflow/flow-init/scripts/consolidate-research.sh +182 -0
- package/.claude/skills/workflow/flow-init/scripts/create-requirement.sh +515 -0
- package/.claude/skills/workflow/flow-init/scripts/generate-research-tasks.sh +157 -0
- package/.claude/skills/workflow/flow-init/scripts/populate-research-tasks.sh +284 -0
- package/.claude/skills/workflow/flow-init/scripts/validate-research.sh +332 -0
- package/.claude/skills/workflow/flow-quality/SKILL.md +94 -0
- package/.claude/skills/workflow/flow-quality/context.jsonl +6 -0
- package/.claude/skills/workflow/flow-quality/references/code-quality-reviewer.md +205 -0
- package/.claude/skills/workflow/flow-quality/references/qa-tester.md +313 -0
- package/.claude/skills/workflow/flow-quality/references/security-reviewer.md +314 -0
- package/.claude/skills/workflow/flow-quality/references/spec-reviewer.md +221 -0
- package/.claude/skills/workflow/flow-release/SKILL.md +126 -0
- package/.claude/skills/workflow/flow-release/context.jsonl +7 -0
- package/.claude/skills/workflow/flow-release/references/release-manager.md +295 -0
- package/.claude/skills/workflow/flow-spec/CLAUDE.md +103 -0
- package/.claude/skills/workflow/flow-spec/SKILL.md +545 -0
- package/.claude/skills/workflow/flow-spec/context.jsonl +7 -0
- package/.claude/skills/workflow/flow-spec/scripts/entry-gate.sh +194 -0
- package/.claude/skills/workflow/flow-spec/scripts/exit-gate.sh +244 -0
- package/.claude/skills/workflow/flow-spec/scripts/parallel-orchestrator.sh +205 -0
- package/.claude/skills/workflow/flow-spec/scripts/team-communication.sh +353 -0
- package/.claude/skills/workflow/flow-spec/scripts/team-init.sh +195 -0
- package/.claude/skills/workflow/flow-spec/scripts/test-team-mode.sh +496 -0
- package/.claude/skills/workflow/flow-spec/team-config.json +165 -0
- package/.claude/skills/workflow.yaml +417 -0
- package/CHANGELOG.md +254 -0
- package/README.md +193 -33
- package/README.zh-CN.md +206 -46
- package/lib/compiler/CLAUDE.md +77 -46
- package/lib/compiler/__tests__/multi-module-emitters.test.js +508 -0
- package/lib/compiler/context-expander.js +179 -0
- package/lib/compiler/emitters/antigravity-emitter.js +195 -5
- package/lib/compiler/emitters/base-emitter.js +217 -2
- package/lib/compiler/emitters/codex-emitter.js +200 -4
- package/lib/compiler/emitters/cursor-emitter.js +307 -3
- package/lib/compiler/emitters/qwen-emitter.js +196 -4
- package/lib/compiler/index.js +197 -2
- package/lib/compiler/platforms.js +270 -21
- package/package.json +1 -1
- package/.claude/commands/flow-epic.md +0 -183
- package/.claude/commands/flow-init.md +0 -370
- package/.claude/commands/flow-prd.md +0 -144
- package/.claude/commands/flow-qa.md +0 -93
- package/.claude/commands/flow-review.md +0 -257
- package/.claude/commands/flow-tech.md +0 -142
- package/.claude/commands/flow-ui.md +0 -189
- package/.claude/skills/file-header-guardian/SKILL.md +0 -56
- package/.claude/skills/skill-developer/ADVANCED.md +0 -197
- package/.claude/skills/skill-developer/HOOK_MECHANISMS.md +0 -306
- package/.claude/skills/skill-developer/PATTERNS_LIBRARY.md +0 -152
- package/.claude/skills/skill-developer/SKILL.md +0 -426
- package/.claude/skills/skill-developer/SKILL_RULES_REFERENCE.md +0 -315
- package/.claude/skills/skill-developer/TRIGGER_TYPES.md +0 -305
- package/.claude/skills/skill-developer/TROUBLESHOOTING.md +0 -514
- package/.claude/skills/writing-skills/SKILL.md +0 -655
- package/.claude/skills/writing-skills/anthropic-best-practices.md +0 -1150
- package/.claude/skills/writing-skills/examples/CLAUDE_MD_TESTING.md +0 -189
- package/.claude/skills/writing-skills/graphviz-conventions.dot +0 -172
- package/.claude/skills/writing-skills/persuasion-principles.md +0 -187
- package/.claude/skills/writing-skills/render-graphs.js +0 -168
- package/.claude/skills/writing-skills/testing-skills-with-subagents.md +0 -384
- package/.claude/tsc-cache/795ba6e3-b98a-423b-bab2-51aa62812569/affected-repos.txt +0 -1
- package/.claude/tsc-cache/ae335694-be5a-4ba4-a1a0-b676c09a7906/affected-repos.txt +0 -1
- /package/.claude/commands/{core-architecture.md → core/architecture.md} +0 -0
- /package/.claude/commands/{core-guidelines.md → core/guidelines.md} +0 -0
- /package/.claude/commands/{core-roadmap.md → core/roadmap.md} +0 -0
- /package/.claude/commands/{core-style.md → core/style.md} +0 -0
- /package/.claude/commands/{flow-checklist.md → flow/checklist.md} +0 -0
- /package/.claude/commands/{flow-clarify.md → flow/clarify.md} +0 -0
- /package/.claude/commands/{flow-constitution.md → flow/constitution.md} +0 -0
- /package/.claude/commands/{flow-fix.md → flow/fix.md} +0 -0
- /package/.claude/commands/{flow-ideate.md → flow/ideate.md} +0 -0
- /package/.claude/commands/{flow-new.md → flow/new.md} +0 -0
- /package/.claude/commands/{flow-release.md → flow/release.md} +0 -0
- /package/.claude/commands/{flow-restart.md → flow/restart.md} +0 -0
- /package/.claude/commands/{flow-status.md → flow/status.md} +0 -0
- /package/.claude/commands/{flow-update.md → flow/update.md} +0 -0
- /package/.claude/commands/{flow-upgrade.md → flow/upgrade.md} +0 -0
- /package/.claude/commands/{flow-verify.md → flow/verify.md} +0 -0
- /package/.claude/commands/{code-review-high.md → util/code-review.md} +0 -0
- /package/.claude/commands/{git-commit.md → util/git-commit.md} +0 -0
- /package/.claude/commands/{problem-analyzer.md → util/problem-analyzer.md} +0 -0
- /package/.claude/skills/{flow-attention-refresh → domain/attention-refresh}/SKILL.md +0 -0
- /package/.claude/skills/{flow-brainstorming → domain/brainstorming}/SKILL.md +0 -0
- /package/.claude/skills/{flow-debugging → domain/debugging}/SKILL.md +0 -0
- /package/.claude/skills/{flow-finishing-branch → domain/finishing-branch}/SKILL.md +0 -0
- /package/.claude/skills/{flow-receiving-review → domain/receiving-review}/SKILL.md +0 -0
- /package/.claude/skills/{flow-tdd → domain/tdd}/SKILL.md +0 -0
- /package/.claude/skills/{verification-before-completion → domain/verification}/SKILL.md +0 -0
- /package/.claude/skills/{constitution-guardian → guardrail/constitution-guardian}/SKILL.md +0 -0
- /package/.claude/skills/{devflow-tdd-enforcer → guardrail/tdd-enforcer}/SKILL.md +0 -0
- /package/.claude/skills/{devflow-constitution-quick-ref → utility/constitution-quick-ref}/SKILL.md +0 -0
- /package/.claude/skills/{devflow-file-standards → utility/file-standards}/SKILL.md +0 -0
- /package/.claude/skills/{fractal-docs-generator → utility/fractal-docs}/SKILL.md +0 -0
- /package/.claude/skills/{npm-release → utility/npm-release}/SKILL.md +0 -0
|
@@ -0,0 +1,931 @@
|
|
|
1
|
+
#!/usr/bin/env npx ts-node
|
|
2
|
+
/**
|
|
3
|
+
* [INPUT]: 依赖 quality-gates.yml 的 verify 命令配置, orchestration_status.json 的 Team 状态
|
|
4
|
+
* [OUTPUT]: 对外提供 SubagentStop 钩子,拦截子 Agent 停止并执行程序化验证
|
|
5
|
+
* [POS]: hooks/ 的 Ralph Loop 控制器,支持单 Agent 和多 Teammate 模式
|
|
6
|
+
* [PROTOCOL]: 变更时更新此头部,然后检查 CLAUDE.md
|
|
7
|
+
*
|
|
8
|
+
* =============================================================================
|
|
9
|
+
* CC-DevFlow Ralph Loop - SubagentStop Hook (借鉴 Trellis)
|
|
10
|
+
* =============================================================================
|
|
11
|
+
*
|
|
12
|
+
* 核心设计哲学:
|
|
13
|
+
* - 子 Agent 尝试停止时执行程序化验证
|
|
14
|
+
* - 验证失败则阻止停止,返回错误信息
|
|
15
|
+
* - 最多迭代 N 次,防止无限循环
|
|
16
|
+
* - 状态持久化到 .ralph-state.json (单 Agent) 或 orchestration_status.json (Team)
|
|
17
|
+
*
|
|
18
|
+
* v4.7 Team 模式支持:
|
|
19
|
+
* - Teammate 级别验证:每个 teammate 停止时验证自己的改动
|
|
20
|
+
* - 全局验证:最后一个 teammate 停止时执行全局验证
|
|
21
|
+
* - 分布式迭代控制:每个 teammate 独立计数 + 全局计数
|
|
22
|
+
*
|
|
23
|
+
* 触发条件:SubagentStop 事件
|
|
24
|
+
*/
|
|
25
|
+
|
|
26
|
+
import * as fs from 'fs';
|
|
27
|
+
import * as path from 'path';
|
|
28
|
+
import { execSync } from 'child_process';
|
|
29
|
+
|
|
30
|
+
// =============================================================================
|
|
31
|
+
// 类型定义
|
|
32
|
+
// =============================================================================
|
|
33
|
+
|
|
34
|
+
interface RalphState {
|
|
35
|
+
agent_id: string;
|
|
36
|
+
iteration: number;
|
|
37
|
+
last_failures: FailureRecord[];
|
|
38
|
+
started_at: string;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
interface FailureRecord {
|
|
42
|
+
command: string;
|
|
43
|
+
output: string;
|
|
44
|
+
timestamp: string;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
interface VerifyCommand {
|
|
48
|
+
name?: string;
|
|
49
|
+
command: string;
|
|
50
|
+
required?: boolean;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Team 模式类型
|
|
54
|
+
interface TeammateRalphState {
|
|
55
|
+
iteration: number;
|
|
56
|
+
lastVerifyResult: 'passed' | 'failed' | 'skipped';
|
|
57
|
+
lastVerifyAt?: string;
|
|
58
|
+
lastFailures?: FailureRecord[];
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
interface RalphLoopTeamState {
|
|
62
|
+
enabled: boolean;
|
|
63
|
+
teammates: Record<string, TeammateRalphState>;
|
|
64
|
+
globalIteration: number;
|
|
65
|
+
maxIterations: number;
|
|
66
|
+
startedAt: string;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
interface TeamState {
|
|
70
|
+
mode: 'sequential' | 'parallel';
|
|
71
|
+
lead: string;
|
|
72
|
+
teammates: Array<{
|
|
73
|
+
id: string;
|
|
74
|
+
role: string;
|
|
75
|
+
status: string;
|
|
76
|
+
currentTask: string | null;
|
|
77
|
+
completedTasks: string[];
|
|
78
|
+
lastActiveAt: string;
|
|
79
|
+
}>;
|
|
80
|
+
taskAssignments: Record<string, string>;
|
|
81
|
+
createdAt: string;
|
|
82
|
+
updatedAt: string;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
interface OrchestrationStatus {
|
|
86
|
+
reqId: string;
|
|
87
|
+
status: string;
|
|
88
|
+
phase: string;
|
|
89
|
+
team?: TeamState;
|
|
90
|
+
ralphLoop?: RalphLoopTeamState;
|
|
91
|
+
updatedAt: string;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
interface TeamModeConfig {
|
|
95
|
+
enabled?: boolean;
|
|
96
|
+
scope?: 'teammate' | 'global';
|
|
97
|
+
teammate_verify?: Record<string, string[]>;
|
|
98
|
+
global_verify?: string[];
|
|
99
|
+
max_iterations_per_teammate?: number;
|
|
100
|
+
max_global_iterations?: number;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
interface QualityGatesConfig {
|
|
104
|
+
verify?: (string | VerifyCommand)[];
|
|
105
|
+
ralph_loop?: {
|
|
106
|
+
max_iterations?: number;
|
|
107
|
+
timeout_minutes?: number;
|
|
108
|
+
team_mode?: TeamModeConfig;
|
|
109
|
+
};
|
|
110
|
+
global?: {
|
|
111
|
+
max_iterations?: number;
|
|
112
|
+
timeout_minutes?: number;
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
interface HookInput {
|
|
117
|
+
hook_event_name: string;
|
|
118
|
+
subagent_type?: string;
|
|
119
|
+
agent_output?: string;
|
|
120
|
+
prompt?: string;
|
|
121
|
+
cwd?: string;
|
|
122
|
+
session_id?: string;
|
|
123
|
+
// Team 模式扩展字段
|
|
124
|
+
teammate_id?: string;
|
|
125
|
+
teammate_role?: string;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
interface HookOutput {
|
|
129
|
+
decision: 'allow' | 'block';
|
|
130
|
+
reason: string;
|
|
131
|
+
systemMessage?: string;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// =============================================================================
|
|
135
|
+
// 配置常量
|
|
136
|
+
// =============================================================================
|
|
137
|
+
|
|
138
|
+
const DEFAULT_MAX_ITERATIONS = 5;
|
|
139
|
+
const DEFAULT_TIMEOUT_MINUTES = 30;
|
|
140
|
+
const DEFAULT_MAX_ITERATIONS_PER_TEAMMATE = 3;
|
|
141
|
+
const DEFAULT_MAX_GLOBAL_ITERATIONS = 10;
|
|
142
|
+
const STATE_FILE = '.ralph-state.json';
|
|
143
|
+
const CONFIG_FILE = '.claude/config/quality-gates.yml';
|
|
144
|
+
const COMMAND_TIMEOUT_MS = 120000; // 2 minutes per command
|
|
145
|
+
|
|
146
|
+
// =============================================================================
|
|
147
|
+
// 工具函数
|
|
148
|
+
// =============================================================================
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* 查找 Git 仓库根目录
|
|
152
|
+
*/
|
|
153
|
+
function findRepoRoot(startPath: string): string | null {
|
|
154
|
+
let current = path.resolve(startPath);
|
|
155
|
+
while (current !== path.dirname(current)) {
|
|
156
|
+
if (fs.existsSync(path.join(current, '.git'))) {
|
|
157
|
+
return current;
|
|
158
|
+
}
|
|
159
|
+
current = path.dirname(current);
|
|
160
|
+
}
|
|
161
|
+
return null;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* 简单 YAML 解析器 (无外部依赖)
|
|
166
|
+
* 只解析 verify 列表和 ralph_loop 配置
|
|
167
|
+
*/
|
|
168
|
+
function parseSimpleYaml(content: string): QualityGatesConfig {
|
|
169
|
+
const config: QualityGatesConfig = {};
|
|
170
|
+
const lines = content.split('\n');
|
|
171
|
+
|
|
172
|
+
let currentSection = '';
|
|
173
|
+
let inVerifySection = false;
|
|
174
|
+
let inRalphLoopSection = false;
|
|
175
|
+
let inGlobalSection = false;
|
|
176
|
+
let inTeamModeSection = false;
|
|
177
|
+
let inTeammateVerifySection = false;
|
|
178
|
+
let inGlobalVerifySection = false;
|
|
179
|
+
let currentTeammateRole = '';
|
|
180
|
+
const verifyCommands: (string | VerifyCommand)[] = [];
|
|
181
|
+
|
|
182
|
+
for (const line of lines) {
|
|
183
|
+
const trimmed = line.trim();
|
|
184
|
+
|
|
185
|
+
// 跳过注释和空行
|
|
186
|
+
if (trimmed.startsWith('#') || trimmed === '') continue;
|
|
187
|
+
|
|
188
|
+
// 检测顶级 section
|
|
189
|
+
if (!line.startsWith(' ') && !line.startsWith('\t') && trimmed.endsWith(':')) {
|
|
190
|
+
currentSection = trimmed.slice(0, -1);
|
|
191
|
+
inVerifySection = false;
|
|
192
|
+
inRalphLoopSection = currentSection === 'ralph_loop';
|
|
193
|
+
inGlobalSection = currentSection === 'global';
|
|
194
|
+
inTeamModeSection = false;
|
|
195
|
+
inTeammateVerifySection = false;
|
|
196
|
+
inGlobalVerifySection = false;
|
|
197
|
+
continue;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
// 检测 verify 子 section (顶级)
|
|
201
|
+
if (trimmed === 'verify:' && !inRalphLoopSection) {
|
|
202
|
+
inVerifySection = true;
|
|
203
|
+
continue;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
// 解析 verify 列表项
|
|
207
|
+
if (inVerifySection && trimmed.startsWith('- ')) {
|
|
208
|
+
const value = trimmed.slice(2).trim();
|
|
209
|
+
// 简单字符串命令
|
|
210
|
+
if (!value.startsWith('name:') && !value.includes(':')) {
|
|
211
|
+
verifyCommands.push(value);
|
|
212
|
+
}
|
|
213
|
+
continue;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
// 解析 ralph_loop 配置
|
|
217
|
+
if (inRalphLoopSection) {
|
|
218
|
+
if (trimmed.startsWith('max_iterations:')) {
|
|
219
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
220
|
+
if (!isNaN(val)) {
|
|
221
|
+
config.ralph_loop = config.ralph_loop || {};
|
|
222
|
+
config.ralph_loop.max_iterations = val;
|
|
223
|
+
}
|
|
224
|
+
} else if (trimmed.startsWith('timeout_minutes:')) {
|
|
225
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
226
|
+
if (!isNaN(val)) {
|
|
227
|
+
config.ralph_loop = config.ralph_loop || {};
|
|
228
|
+
config.ralph_loop.timeout_minutes = val;
|
|
229
|
+
}
|
|
230
|
+
} else if (trimmed === 'team_mode:') {
|
|
231
|
+
inTeamModeSection = true;
|
|
232
|
+
config.ralph_loop = config.ralph_loop || {};
|
|
233
|
+
config.ralph_loop.team_mode = {};
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
// 解析 team_mode 配置
|
|
238
|
+
if (inTeamModeSection) {
|
|
239
|
+
if (trimmed.startsWith('enabled:')) {
|
|
240
|
+
const val = trimmed.split(':')[1].trim();
|
|
241
|
+
config.ralph_loop!.team_mode!.enabled = val === 'true';
|
|
242
|
+
} else if (trimmed.startsWith('scope:')) {
|
|
243
|
+
const val = trimmed.split(':')[1].trim() as 'teammate' | 'global';
|
|
244
|
+
config.ralph_loop!.team_mode!.scope = val;
|
|
245
|
+
} else if (trimmed.startsWith('max_iterations_per_teammate:')) {
|
|
246
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
247
|
+
if (!isNaN(val)) {
|
|
248
|
+
config.ralph_loop!.team_mode!.max_iterations_per_teammate = val;
|
|
249
|
+
}
|
|
250
|
+
} else if (trimmed.startsWith('max_global_iterations:')) {
|
|
251
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
252
|
+
if (!isNaN(val)) {
|
|
253
|
+
config.ralph_loop!.team_mode!.max_global_iterations = val;
|
|
254
|
+
}
|
|
255
|
+
} else if (trimmed === 'teammate_verify:') {
|
|
256
|
+
inTeammateVerifySection = true;
|
|
257
|
+
inGlobalVerifySection = false;
|
|
258
|
+
config.ralph_loop!.team_mode!.teammate_verify = {};
|
|
259
|
+
} else if (trimmed === 'global_verify:') {
|
|
260
|
+
inGlobalVerifySection = true;
|
|
261
|
+
inTeammateVerifySection = false;
|
|
262
|
+
config.ralph_loop!.team_mode!.global_verify = [];
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
// 解析 teammate_verify 配置
|
|
267
|
+
if (inTeammateVerifySection && !inGlobalVerifySection) {
|
|
268
|
+
// 检测 teammate role (e.g., "dev-frontend:")
|
|
269
|
+
if (trimmed.endsWith(':') && !trimmed.startsWith('-')) {
|
|
270
|
+
currentTeammateRole = trimmed.slice(0, -1);
|
|
271
|
+
config.ralph_loop!.team_mode!.teammate_verify![currentTeammateRole] = [];
|
|
272
|
+
} else if (trimmed.startsWith('- ') && currentTeammateRole) {
|
|
273
|
+
const cmd = trimmed.slice(2).trim();
|
|
274
|
+
config.ralph_loop!.team_mode!.teammate_verify![currentTeammateRole].push(cmd);
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
// 解析 global_verify 配置
|
|
279
|
+
if (inGlobalVerifySection && trimmed.startsWith('- ')) {
|
|
280
|
+
const cmd = trimmed.slice(2).trim();
|
|
281
|
+
config.ralph_loop!.team_mode!.global_verify!.push(cmd);
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
// 解析 global 配置
|
|
285
|
+
if (inGlobalSection) {
|
|
286
|
+
if (trimmed.startsWith('max_iterations:')) {
|
|
287
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
288
|
+
if (!isNaN(val)) {
|
|
289
|
+
config.global = config.global || {};
|
|
290
|
+
config.global.max_iterations = val;
|
|
291
|
+
}
|
|
292
|
+
} else if (trimmed.startsWith('timeout_minutes:')) {
|
|
293
|
+
const val = parseInt(trimmed.split(':')[1].trim(), 10);
|
|
294
|
+
if (!isNaN(val)) {
|
|
295
|
+
config.global = config.global || {};
|
|
296
|
+
config.global.timeout_minutes = val;
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
if (verifyCommands.length > 0) {
|
|
303
|
+
config.verify = verifyCommands;
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
return config;
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
/**
|
|
310
|
+
* 读取 quality-gates.yml 配置
|
|
311
|
+
*/
|
|
312
|
+
function loadConfig(repoRoot: string): QualityGatesConfig {
|
|
313
|
+
const configPath = path.join(repoRoot, CONFIG_FILE);
|
|
314
|
+
if (!fs.existsSync(configPath)) {
|
|
315
|
+
return {};
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
try {
|
|
319
|
+
const content = fs.readFileSync(configPath, 'utf-8');
|
|
320
|
+
return parseSimpleYaml(content);
|
|
321
|
+
} catch {
|
|
322
|
+
return {};
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
/**
|
|
327
|
+
* 获取 verify 命令列表
|
|
328
|
+
*/
|
|
329
|
+
function getVerifyCommands(config: QualityGatesConfig): string[] {
|
|
330
|
+
if (!config.verify) {
|
|
331
|
+
// 默认验证命令
|
|
332
|
+
return [
|
|
333
|
+
'npm run lint --if-present',
|
|
334
|
+
'npm run typecheck --if-present',
|
|
335
|
+
'npm test -- --passWithNoTests'
|
|
336
|
+
];
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
return config.verify.map(cmd => {
|
|
340
|
+
if (typeof cmd === 'string') return cmd;
|
|
341
|
+
return cmd.command;
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
/**
|
|
346
|
+
* 获取最大迭代次数
|
|
347
|
+
*/
|
|
348
|
+
function getMaxIterations(config: QualityGatesConfig): number {
|
|
349
|
+
return config.ralph_loop?.max_iterations
|
|
350
|
+
?? config.global?.max_iterations
|
|
351
|
+
?? DEFAULT_MAX_ITERATIONS;
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
/**
|
|
355
|
+
* 获取超时时间 (分钟)
|
|
356
|
+
*/
|
|
357
|
+
function getTimeoutMinutes(config: QualityGatesConfig): number {
|
|
358
|
+
return config.ralph_loop?.timeout_minutes
|
|
359
|
+
?? config.global?.timeout_minutes
|
|
360
|
+
?? DEFAULT_TIMEOUT_MINUTES;
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
/**
|
|
364
|
+
* 加载 Ralph 状态
|
|
365
|
+
*/
|
|
366
|
+
function loadState(repoRoot: string, agentId: string): RalphState {
|
|
367
|
+
const statePath = path.join(repoRoot, STATE_FILE);
|
|
368
|
+
|
|
369
|
+
if (!fs.existsSync(statePath)) {
|
|
370
|
+
return createNewState(agentId);
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
try {
|
|
374
|
+
const content = fs.readFileSync(statePath, 'utf-8');
|
|
375
|
+
const state: RalphState = JSON.parse(content);
|
|
376
|
+
|
|
377
|
+
// 检查是否是同一个 agent 会话
|
|
378
|
+
if (state.agent_id !== agentId) {
|
|
379
|
+
return createNewState(agentId);
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
return state;
|
|
383
|
+
} catch {
|
|
384
|
+
return createNewState(agentId);
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
/**
|
|
389
|
+
* 创建新状态
|
|
390
|
+
*/
|
|
391
|
+
function createNewState(agentId: string): RalphState {
|
|
392
|
+
return {
|
|
393
|
+
agent_id: agentId,
|
|
394
|
+
iteration: 0,
|
|
395
|
+
last_failures: [],
|
|
396
|
+
started_at: new Date().toISOString()
|
|
397
|
+
};
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
/**
|
|
401
|
+
* 保存 Ralph 状态
|
|
402
|
+
*/
|
|
403
|
+
function saveState(repoRoot: string, state: RalphState): void {
|
|
404
|
+
const statePath = path.join(repoRoot, STATE_FILE);
|
|
405
|
+
try {
|
|
406
|
+
fs.writeFileSync(statePath, JSON.stringify(state, null, 2), 'utf-8');
|
|
407
|
+
} catch {
|
|
408
|
+
// 忽略写入错误
|
|
409
|
+
}
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
/**
|
|
413
|
+
* 清理 Ralph 状态
|
|
414
|
+
*/
|
|
415
|
+
function clearState(repoRoot: string): void {
|
|
416
|
+
const statePath = path.join(repoRoot, STATE_FILE);
|
|
417
|
+
try {
|
|
418
|
+
if (fs.existsSync(statePath)) {
|
|
419
|
+
fs.unlinkSync(statePath);
|
|
420
|
+
}
|
|
421
|
+
} catch {
|
|
422
|
+
// 忽略删除错误
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
/**
|
|
427
|
+
* 检查是否超时
|
|
428
|
+
*/
|
|
429
|
+
function isTimedOut(state: RalphState, timeoutMinutes: number): boolean {
|
|
430
|
+
const startedAt = new Date(state.started_at);
|
|
431
|
+
const now = new Date();
|
|
432
|
+
const elapsedMinutes = (now.getTime() - startedAt.getTime()) / (1000 * 60);
|
|
433
|
+
return elapsedMinutes >= timeoutMinutes;
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
/**
|
|
437
|
+
* 执行验证命令
|
|
438
|
+
*/
|
|
439
|
+
function runVerifyCommands(
|
|
440
|
+
repoRoot: string,
|
|
441
|
+
commands: string[]
|
|
442
|
+
): { passed: boolean; failures: FailureRecord[] } {
|
|
443
|
+
const failures: FailureRecord[] = [];
|
|
444
|
+
|
|
445
|
+
for (const cmd of commands) {
|
|
446
|
+
try {
|
|
447
|
+
execSync(cmd, {
|
|
448
|
+
cwd: repoRoot,
|
|
449
|
+
timeout: COMMAND_TIMEOUT_MS,
|
|
450
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
451
|
+
encoding: 'utf-8'
|
|
452
|
+
});
|
|
453
|
+
} catch (error: unknown) {
|
|
454
|
+
let output = '';
|
|
455
|
+
if (error && typeof error === 'object') {
|
|
456
|
+
const execError = error as { stderr?: string; stdout?: string; message?: string };
|
|
457
|
+
output = execError.stderr || execError.stdout || execError.message || 'Unknown error';
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// 截断过长输出
|
|
461
|
+
if (output.length > 500) {
|
|
462
|
+
output = output.slice(0, 500) + '...';
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
failures.push({
|
|
466
|
+
command: cmd,
|
|
467
|
+
output,
|
|
468
|
+
timestamp: new Date().toISOString()
|
|
469
|
+
});
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
return {
|
|
474
|
+
passed: failures.length === 0,
|
|
475
|
+
failures
|
|
476
|
+
};
|
|
477
|
+
}
|
|
478
|
+
|
|
479
|
+
/**
|
|
480
|
+
* 格式化失败信息
|
|
481
|
+
*/
|
|
482
|
+
function formatFailures(failures: FailureRecord[]): string {
|
|
483
|
+
return failures.map(f => `Command: ${f.command}\nOutput: ${f.output}`).join('\n\n');
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
// =============================================================================
|
|
487
|
+
// Team 模式工具函数
|
|
488
|
+
// =============================================================================
|
|
489
|
+
|
|
490
|
+
/**
|
|
491
|
+
* 查找当前需求的 orchestration_status.json
|
|
492
|
+
*/
|
|
493
|
+
function findOrchestrationStatus(repoRoot: string): string | null {
|
|
494
|
+
const devflowDir = path.join(repoRoot, 'devflow', 'requirements');
|
|
495
|
+
if (!fs.existsSync(devflowDir)) return null;
|
|
496
|
+
|
|
497
|
+
// 查找最近修改的需求目录
|
|
498
|
+
const reqDirs = fs.readdirSync(devflowDir)
|
|
499
|
+
.filter(d => d.startsWith('REQ-') || d.startsWith('BUG-'))
|
|
500
|
+
.map(d => ({
|
|
501
|
+
name: d,
|
|
502
|
+
path: path.join(devflowDir, d),
|
|
503
|
+
mtime: fs.statSync(path.join(devflowDir, d)).mtime.getTime()
|
|
504
|
+
}))
|
|
505
|
+
.sort((a, b) => b.mtime - a.mtime);
|
|
506
|
+
|
|
507
|
+
for (const reqDir of reqDirs) {
|
|
508
|
+
const statusFile = path.join(reqDir.path, 'orchestration_status.json');
|
|
509
|
+
if (fs.existsSync(statusFile)) {
|
|
510
|
+
return statusFile;
|
|
511
|
+
}
|
|
512
|
+
}
|
|
513
|
+
|
|
514
|
+
return null;
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
/**
|
|
518
|
+
* 加载 orchestration_status.json
|
|
519
|
+
*/
|
|
520
|
+
function loadOrchestrationStatus(statusPath: string): OrchestrationStatus | null {
|
|
521
|
+
try {
|
|
522
|
+
const content = fs.readFileSync(statusPath, 'utf-8');
|
|
523
|
+
return JSON.parse(content);
|
|
524
|
+
} catch {
|
|
525
|
+
return null;
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
/**
|
|
530
|
+
* 保存 orchestration_status.json
|
|
531
|
+
*/
|
|
532
|
+
function saveOrchestrationStatus(statusPath: string, status: OrchestrationStatus): void {
|
|
533
|
+
try {
|
|
534
|
+
status.updatedAt = new Date().toISOString();
|
|
535
|
+
fs.writeFileSync(statusPath, JSON.stringify(status, null, 2), 'utf-8');
|
|
536
|
+
} catch {
|
|
537
|
+
// 忽略写入错误
|
|
538
|
+
}
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
/**
|
|
542
|
+
* 检查是否启用 Team 模式
|
|
543
|
+
*/
|
|
544
|
+
function isTeamModeEnabled(status: OrchestrationStatus | null, config: QualityGatesConfig): boolean {
|
|
545
|
+
// 检查 orchestration_status.json 中是否有 team 状态
|
|
546
|
+
if (!status?.team) return false;
|
|
547
|
+
|
|
548
|
+
// 检查配置中是否启用 team_mode
|
|
549
|
+
return config.ralph_loop?.team_mode?.enabled ?? false;
|
|
550
|
+
}
|
|
551
|
+
|
|
552
|
+
/**
|
|
553
|
+
* 获取 Teammate 特定的验证命令
|
|
554
|
+
*/
|
|
555
|
+
function getTeammateVerifyCommands(
|
|
556
|
+
config: QualityGatesConfig,
|
|
557
|
+
teammateRole: string
|
|
558
|
+
): string[] {
|
|
559
|
+
const teamModeConfig = config.ralph_loop?.team_mode;
|
|
560
|
+
|
|
561
|
+
// 优先使用 teammate 特定命令
|
|
562
|
+
if (teamModeConfig?.teammate_verify?.[teammateRole]) {
|
|
563
|
+
return teamModeConfig.teammate_verify[teammateRole];
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
// 回退到默认验证命令
|
|
567
|
+
return getVerifyCommands(config);
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
/**
|
|
571
|
+
* 获取全局验证命令
|
|
572
|
+
*/
|
|
573
|
+
function getGlobalVerifyCommands(config: QualityGatesConfig): string[] {
|
|
574
|
+
const teamModeConfig = config.ralph_loop?.team_mode;
|
|
575
|
+
|
|
576
|
+
if (teamModeConfig?.global_verify && teamModeConfig.global_verify.length > 0) {
|
|
577
|
+
return teamModeConfig.global_verify;
|
|
578
|
+
}
|
|
579
|
+
|
|
580
|
+
// 回退到默认验证命令
|
|
581
|
+
return getVerifyCommands(config);
|
|
582
|
+
}
|
|
583
|
+
|
|
584
|
+
/**
|
|
585
|
+
* 检查是否是最后一个活跃的 Teammate
|
|
586
|
+
*/
|
|
587
|
+
function isLastActiveTeammate(
|
|
588
|
+
status: OrchestrationStatus,
|
|
589
|
+
currentTeammateId: string
|
|
590
|
+
): boolean {
|
|
591
|
+
if (!status.team?.teammates) return true;
|
|
592
|
+
|
|
593
|
+
const activeTeammates = status.team.teammates.filter(
|
|
594
|
+
t => t.status === 'working' && t.id !== currentTeammateId
|
|
595
|
+
);
|
|
596
|
+
|
|
597
|
+
return activeTeammates.length === 0;
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
/**
|
|
601
|
+
* 初始化 Teammate 的 Ralph 状态
|
|
602
|
+
*/
|
|
603
|
+
function initTeammateRalphState(): TeammateRalphState {
|
|
604
|
+
return {
|
|
605
|
+
iteration: 0,
|
|
606
|
+
lastVerifyResult: 'skipped'
|
|
607
|
+
};
|
|
608
|
+
}
|
|
609
|
+
|
|
610
|
+
/**
|
|
611
|
+
* 更新 Teammate 的 Ralph 状态
|
|
612
|
+
*/
|
|
613
|
+
function updateTeammateRalphState(
|
|
614
|
+
status: OrchestrationStatus,
|
|
615
|
+
teammateId: string,
|
|
616
|
+
verifyResult: 'passed' | 'failed',
|
|
617
|
+
failures?: FailureRecord[]
|
|
618
|
+
): void {
|
|
619
|
+
if (!status.ralphLoop) {
|
|
620
|
+
status.ralphLoop = {
|
|
621
|
+
enabled: true,
|
|
622
|
+
teammates: {},
|
|
623
|
+
globalIteration: 0,
|
|
624
|
+
maxIterations: DEFAULT_MAX_GLOBAL_ITERATIONS,
|
|
625
|
+
startedAt: new Date().toISOString()
|
|
626
|
+
};
|
|
627
|
+
}
|
|
628
|
+
|
|
629
|
+
if (!status.ralphLoop.teammates[teammateId]) {
|
|
630
|
+
status.ralphLoop.teammates[teammateId] = initTeammateRalphState();
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
const teammateState = status.ralphLoop.teammates[teammateId];
|
|
634
|
+
teammateState.iteration += 1;
|
|
635
|
+
teammateState.lastVerifyResult = verifyResult;
|
|
636
|
+
teammateState.lastVerifyAt = new Date().toISOString();
|
|
637
|
+
|
|
638
|
+
if (failures && failures.length > 0) {
|
|
639
|
+
teammateState.lastFailures = failures;
|
|
640
|
+
} else {
|
|
641
|
+
delete teammateState.lastFailures;
|
|
642
|
+
}
|
|
643
|
+
|
|
644
|
+
// 更新全局迭代计数
|
|
645
|
+
status.ralphLoop.globalIteration += 1;
|
|
646
|
+
}
|
|
647
|
+
|
|
648
|
+
/**
|
|
649
|
+
* 获取 Teammate 的迭代次数
|
|
650
|
+
*/
|
|
651
|
+
function getTeammateIteration(status: OrchestrationStatus, teammateId: string): number {
|
|
652
|
+
return status.ralphLoop?.teammates?.[teammateId]?.iteration ?? 0;
|
|
653
|
+
}
|
|
654
|
+
|
|
655
|
+
/**
|
|
656
|
+
* 获取全局迭代次数
|
|
657
|
+
*/
|
|
658
|
+
function getGlobalIteration(status: OrchestrationStatus): number {
|
|
659
|
+
return status.ralphLoop?.globalIteration ?? 0;
|
|
660
|
+
}
|
|
661
|
+
|
|
662
|
+
/**
|
|
663
|
+
* 获取 Teammate 最大迭代次数
|
|
664
|
+
*/
|
|
665
|
+
function getMaxIterationsPerTeammate(config: QualityGatesConfig): number {
|
|
666
|
+
return config.ralph_loop?.team_mode?.max_iterations_per_teammate
|
|
667
|
+
?? DEFAULT_MAX_ITERATIONS_PER_TEAMMATE;
|
|
668
|
+
}
|
|
669
|
+
|
|
670
|
+
/**
|
|
671
|
+
* 获取全局最大迭代次数
|
|
672
|
+
*/
|
|
673
|
+
function getMaxGlobalIterations(config: QualityGatesConfig): number {
|
|
674
|
+
return config.ralph_loop?.team_mode?.max_global_iterations
|
|
675
|
+
?? DEFAULT_MAX_GLOBAL_ITERATIONS;
|
|
676
|
+
}
|
|
677
|
+
|
|
678
|
+
// =============================================================================
|
|
679
|
+
// 主函数
|
|
680
|
+
// =============================================================================
|
|
681
|
+
|
|
682
|
+
function main(): void {
|
|
683
|
+
let inputData: HookInput;
|
|
684
|
+
|
|
685
|
+
try {
|
|
686
|
+
const stdin = fs.readFileSync(0, 'utf-8');
|
|
687
|
+
inputData = JSON.parse(stdin);
|
|
688
|
+
} catch {
|
|
689
|
+
// 无法解析输入,允许停止
|
|
690
|
+
process.exit(0);
|
|
691
|
+
}
|
|
692
|
+
|
|
693
|
+
const hookEvent = inputData.hook_event_name || '';
|
|
694
|
+
|
|
695
|
+
// 只处理 SubagentStop 事件
|
|
696
|
+
if (hookEvent !== 'SubagentStop') {
|
|
697
|
+
process.exit(0);
|
|
698
|
+
}
|
|
699
|
+
|
|
700
|
+
const cwd = inputData.cwd || process.cwd();
|
|
701
|
+
const agentId = inputData.session_id || 'default';
|
|
702
|
+
const teammateId = inputData.teammate_id;
|
|
703
|
+
const teammateRole = inputData.teammate_role || 'default';
|
|
704
|
+
|
|
705
|
+
// 查找仓库根目录
|
|
706
|
+
const repoRoot = findRepoRoot(cwd);
|
|
707
|
+
if (!repoRoot) {
|
|
708
|
+
process.exit(0);
|
|
709
|
+
}
|
|
710
|
+
|
|
711
|
+
// 加载配置
|
|
712
|
+
const config = loadConfig(repoRoot);
|
|
713
|
+
|
|
714
|
+
// 查找 orchestration_status.json
|
|
715
|
+
const statusPath = findOrchestrationStatus(repoRoot);
|
|
716
|
+
const orchestrationStatus = statusPath ? loadOrchestrationStatus(statusPath) : null;
|
|
717
|
+
|
|
718
|
+
// 检查是否启用 Team 模式
|
|
719
|
+
if (teammateId && isTeamModeEnabled(orchestrationStatus, config)) {
|
|
720
|
+
handleTeamMode(
|
|
721
|
+
repoRoot,
|
|
722
|
+
config,
|
|
723
|
+
statusPath!,
|
|
724
|
+
orchestrationStatus!,
|
|
725
|
+
teammateId,
|
|
726
|
+
teammateRole
|
|
727
|
+
);
|
|
728
|
+
} else {
|
|
729
|
+
handleSingleAgentMode(repoRoot, config, agentId);
|
|
730
|
+
}
|
|
731
|
+
}
|
|
732
|
+
|
|
733
|
+
/**
|
|
734
|
+
* 处理 Team 模式的 SubagentStop
|
|
735
|
+
*/
|
|
736
|
+
function handleTeamMode(
|
|
737
|
+
repoRoot: string,
|
|
738
|
+
config: QualityGatesConfig,
|
|
739
|
+
statusPath: string,
|
|
740
|
+
status: OrchestrationStatus,
|
|
741
|
+
teammateId: string,
|
|
742
|
+
teammateRole: string
|
|
743
|
+
): void {
|
|
744
|
+
const maxIterationsPerTeammate = getMaxIterationsPerTeammate(config);
|
|
745
|
+
const maxGlobalIterations = getMaxGlobalIterations(config);
|
|
746
|
+
const timeoutMinutes = getTimeoutMinutes(config);
|
|
747
|
+
|
|
748
|
+
// 检查超时
|
|
749
|
+
if (status.ralphLoop?.startedAt) {
|
|
750
|
+
const startedAt = new Date(status.ralphLoop.startedAt);
|
|
751
|
+
const now = new Date();
|
|
752
|
+
const elapsedMinutes = (now.getTime() - startedAt.getTime()) / (1000 * 60);
|
|
753
|
+
if (elapsedMinutes >= timeoutMinutes) {
|
|
754
|
+
const output: HookOutput = {
|
|
755
|
+
decision: 'allow',
|
|
756
|
+
reason: `Team mode timeout (${timeoutMinutes} minutes) reached. Allowing stop.`
|
|
757
|
+
};
|
|
758
|
+
console.log(JSON.stringify(output, null, 0));
|
|
759
|
+
process.exit(0);
|
|
760
|
+
}
|
|
761
|
+
}
|
|
762
|
+
|
|
763
|
+
// 检查 Teammate 迭代次数
|
|
764
|
+
const teammateIteration = getTeammateIteration(status, teammateId);
|
|
765
|
+
if (teammateIteration >= maxIterationsPerTeammate) {
|
|
766
|
+
const output: HookOutput = {
|
|
767
|
+
decision: 'allow',
|
|
768
|
+
reason: `Teammate ${teammateId} max iterations (${maxIterationsPerTeammate}) reached. Allowing stop.`
|
|
769
|
+
};
|
|
770
|
+
console.log(JSON.stringify(output, null, 0));
|
|
771
|
+
process.exit(0);
|
|
772
|
+
}
|
|
773
|
+
|
|
774
|
+
// 检查全局迭代次数
|
|
775
|
+
const globalIteration = getGlobalIteration(status);
|
|
776
|
+
if (globalIteration >= maxGlobalIterations) {
|
|
777
|
+
const output: HookOutput = {
|
|
778
|
+
decision: 'allow',
|
|
779
|
+
reason: `Global max iterations (${maxGlobalIterations}) reached. Allowing stop.`
|
|
780
|
+
};
|
|
781
|
+
console.log(JSON.stringify(output, null, 0));
|
|
782
|
+
process.exit(0);
|
|
783
|
+
}
|
|
784
|
+
|
|
785
|
+
// 获取验证命令
|
|
786
|
+
const verifyCommands = getTeammateVerifyCommands(config, teammateRole);
|
|
787
|
+
|
|
788
|
+
// 执行 Teammate 级别验证
|
|
789
|
+
const { passed: teammatePassed, failures: teammateFailures } = runVerifyCommands(
|
|
790
|
+
repoRoot,
|
|
791
|
+
verifyCommands
|
|
792
|
+
);
|
|
793
|
+
|
|
794
|
+
// 更新 Teammate Ralph 状态
|
|
795
|
+
updateTeammateRalphState(
|
|
796
|
+
status,
|
|
797
|
+
teammateId,
|
|
798
|
+
teammatePassed ? 'passed' : 'failed',
|
|
799
|
+
teammateFailures
|
|
800
|
+
);
|
|
801
|
+
|
|
802
|
+
if (!teammatePassed) {
|
|
803
|
+
// Teammate 验证失败
|
|
804
|
+
saveOrchestrationStatus(statusPath, status);
|
|
805
|
+
|
|
806
|
+
const failureDetails = formatFailures(teammateFailures);
|
|
807
|
+
const currentIteration = getTeammateIteration(status, teammateId);
|
|
808
|
+
const output: HookOutput = {
|
|
809
|
+
decision: 'block',
|
|
810
|
+
reason: `Teammate ${teammateId} iteration ${currentIteration}/${maxIterationsPerTeammate}. Verification failed:\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
|
|
811
|
+
systemMessage: `Ralph Loop (Team Mode) - Teammate ${teammateId} iteration ${currentIteration}/${maxIterationsPerTeammate}`
|
|
812
|
+
};
|
|
813
|
+
console.log(JSON.stringify(output, null, 0));
|
|
814
|
+
process.exit(0);
|
|
815
|
+
}
|
|
816
|
+
|
|
817
|
+
// Teammate 验证通过,检查是否是最后一个活跃的 Teammate
|
|
818
|
+
const isLast = isLastActiveTeammate(status, teammateId);
|
|
819
|
+
|
|
820
|
+
if (isLast) {
|
|
821
|
+
// 执行全局验证
|
|
822
|
+
const globalVerifyCommands = getGlobalVerifyCommands(config);
|
|
823
|
+
const { passed: globalPassed, failures: globalFailures } = runVerifyCommands(
|
|
824
|
+
repoRoot,
|
|
825
|
+
globalVerifyCommands
|
|
826
|
+
);
|
|
827
|
+
|
|
828
|
+
if (!globalPassed) {
|
|
829
|
+
// 全局验证失败
|
|
830
|
+
saveOrchestrationStatus(statusPath, status);
|
|
831
|
+
|
|
832
|
+
const failureDetails = formatFailures(globalFailures);
|
|
833
|
+
const output: HookOutput = {
|
|
834
|
+
decision: 'block',
|
|
835
|
+
reason: `Global verification failed (last teammate ${teammateId}):\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
|
|
836
|
+
systemMessage: `Ralph Loop (Team Mode) - Global verification failed`
|
|
837
|
+
};
|
|
838
|
+
console.log(JSON.stringify(output, null, 0));
|
|
839
|
+
process.exit(0);
|
|
840
|
+
}
|
|
841
|
+
|
|
842
|
+
// 全局验证通过
|
|
843
|
+
saveOrchestrationStatus(statusPath, status);
|
|
844
|
+
const output: HookOutput = {
|
|
845
|
+
decision: 'allow',
|
|
846
|
+
reason: `All verifications passed. Teammate ${teammateId} (last active) completed successfully.`
|
|
847
|
+
};
|
|
848
|
+
console.log(JSON.stringify(output, null, 0));
|
|
849
|
+
process.exit(0);
|
|
850
|
+
}
|
|
851
|
+
|
|
852
|
+
// 不是最后一个 Teammate,允许停止
|
|
853
|
+
saveOrchestrationStatus(statusPath, status);
|
|
854
|
+
const output: HookOutput = {
|
|
855
|
+
decision: 'allow',
|
|
856
|
+
reason: `Teammate ${teammateId} verification passed. Other teammates still active.`
|
|
857
|
+
};
|
|
858
|
+
console.log(JSON.stringify(output, null, 0));
|
|
859
|
+
process.exit(0);
|
|
860
|
+
}
|
|
861
|
+
|
|
862
|
+
/**
|
|
863
|
+
* 处理单 Agent 模式的 SubagentStop (原有逻辑)
|
|
864
|
+
*/
|
|
865
|
+
function handleSingleAgentMode(
|
|
866
|
+
repoRoot: string,
|
|
867
|
+
config: QualityGatesConfig,
|
|
868
|
+
agentId: string
|
|
869
|
+
): void {
|
|
870
|
+
const maxIterations = getMaxIterations(config);
|
|
871
|
+
const timeoutMinutes = getTimeoutMinutes(config);
|
|
872
|
+
const verifyCommands = getVerifyCommands(config);
|
|
873
|
+
|
|
874
|
+
// 加载状态
|
|
875
|
+
const state = loadState(repoRoot, agentId);
|
|
876
|
+
|
|
877
|
+
// 检查超时
|
|
878
|
+
if (isTimedOut(state, timeoutMinutes)) {
|
|
879
|
+
clearState(repoRoot);
|
|
880
|
+
const output: HookOutput = {
|
|
881
|
+
decision: 'allow',
|
|
882
|
+
reason: `Timeout (${timeoutMinutes} minutes) reached. Stopping Ralph loop.`
|
|
883
|
+
};
|
|
884
|
+
console.log(JSON.stringify(output, null, 0));
|
|
885
|
+
process.exit(0);
|
|
886
|
+
}
|
|
887
|
+
|
|
888
|
+
// 增加迭代计数
|
|
889
|
+
state.iteration += 1;
|
|
890
|
+
|
|
891
|
+
// 检查最大迭代次数
|
|
892
|
+
if (state.iteration >= maxIterations) {
|
|
893
|
+
clearState(repoRoot);
|
|
894
|
+
const output: HookOutput = {
|
|
895
|
+
decision: 'allow',
|
|
896
|
+
reason: `Max iterations (${maxIterations}) reached. Stopping to prevent infinite loop.`
|
|
897
|
+
};
|
|
898
|
+
console.log(JSON.stringify(output, null, 0));
|
|
899
|
+
process.exit(0);
|
|
900
|
+
}
|
|
901
|
+
|
|
902
|
+
// 执行验证命令
|
|
903
|
+
const { passed, failures } = runVerifyCommands(repoRoot, verifyCommands);
|
|
904
|
+
|
|
905
|
+
if (passed) {
|
|
906
|
+
// 所有验证通过,允许停止
|
|
907
|
+
clearState(repoRoot);
|
|
908
|
+
const output: HookOutput = {
|
|
909
|
+
decision: 'allow',
|
|
910
|
+
reason: 'All verify commands passed. Quality check complete.'
|
|
911
|
+
};
|
|
912
|
+
console.log(JSON.stringify(output, null, 0));
|
|
913
|
+
process.exit(0);
|
|
914
|
+
}
|
|
915
|
+
|
|
916
|
+
// 验证失败,更新状态并阻止停止
|
|
917
|
+
state.last_failures = failures;
|
|
918
|
+
saveState(repoRoot, state);
|
|
919
|
+
|
|
920
|
+
const failureDetails = formatFailures(failures);
|
|
921
|
+
const output: HookOutput = {
|
|
922
|
+
decision: 'block',
|
|
923
|
+
reason: `Iteration ${state.iteration}/${maxIterations}. Verification failed:\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
|
|
924
|
+
systemMessage: `Ralph Loop iteration ${state.iteration}/${maxIterations} - Fix the errors above to proceed.`
|
|
925
|
+
};
|
|
926
|
+
|
|
927
|
+
console.log(JSON.stringify(output, null, 0));
|
|
928
|
+
process.exit(0);
|
|
929
|
+
}
|
|
930
|
+
|
|
931
|
+
main();
|