@stackmemoryai/stackmemory 0.4.2 → 0.5.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 (31) hide show
  1. package/dist/cli/commands/ralph.js +305 -0
  2. package/dist/cli/commands/ralph.js.map +2 -2
  3. package/dist/integrations/claude-code/agent-bridge.js +764 -0
  4. package/dist/integrations/claude-code/agent-bridge.js.map +7 -0
  5. package/dist/integrations/claude-code/task-coordinator.js +356 -0
  6. package/dist/integrations/claude-code/task-coordinator.js.map +7 -0
  7. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js +33 -11
  8. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +2 -2
  9. package/dist/integrations/ralph/monitoring/swarm-registry.js +10 -1
  10. package/dist/integrations/ralph/monitoring/swarm-registry.js.map +2 -2
  11. package/dist/integrations/ralph/patterns/compounding-engineering-pattern.js +396 -0
  12. package/dist/integrations/ralph/patterns/compounding-engineering-pattern.js.map +7 -0
  13. package/dist/integrations/ralph/patterns/extended-coherence-sessions.js +469 -0
  14. package/dist/integrations/ralph/patterns/extended-coherence-sessions.js.map +7 -0
  15. package/dist/integrations/ralph/patterns/oracle-worker-pattern.js +384 -0
  16. package/dist/integrations/ralph/patterns/oracle-worker-pattern.js.map +7 -0
  17. package/dist/integrations/ralph/swarm/git-workflow-manager.js +71 -18
  18. package/dist/integrations/ralph/swarm/git-workflow-manager.js.map +2 -2
  19. package/dist/integrations/ralph/swarm/swarm-coordinator.js +243 -49
  20. package/dist/integrations/ralph/swarm/swarm-coordinator.js.map +2 -2
  21. package/package.json +1 -1
  22. package/scripts/simple-swarm-demo.ts +114 -0
  23. package/scripts/test-ralph-iterations.ts +164 -0
  24. package/scripts/test-swarm-fixes.ts +161 -0
  25. package/scripts/testing/ab-test-runner.ts +4 -2
  26. package/scripts/testing/collect-metrics.ts +2 -2
  27. package/scripts/testing/real-performance-test.js +1 -1
  28. package/scripts/testing/results/real-performance-results.json +90 -0
  29. package/scripts/testing/run-effectiveness-tests.sh +1 -1
  30. package/scripts/testing/simple-effectiveness-test.js +1 -1
  31. package/scripts/testing/test-tier-migration.js +100 -0
@@ -0,0 +1,384 @@
1
+ import { v4 as uuidv4 } from "uuid";
2
+ import { logger } from "../../../core/monitoring/logger.js";
3
+ import { SwarmCoordinator } from "../swarm/swarm-coordinator.js";
4
+ import { RalphStackMemoryBridge } from "../bridge/ralph-stackmemory-bridge.js";
5
+ class OracleWorkerCoordinator extends SwarmCoordinator {
6
+ oracle;
7
+ workerPool;
8
+ reviewerPool;
9
+ costTracker;
10
+ constructor(config) {
11
+ super({
12
+ maxAgents: config.maxWorkers + 2,
13
+ // Workers + Oracle + Reviewer
14
+ coordinationInterval: config.coordinationInterval,
15
+ enableDynamicPlanning: true,
16
+ pathologicalBehaviorDetection: true
17
+ });
18
+ this.oracle = config.oracle;
19
+ this.workerPool = config.workers;
20
+ this.reviewerPool = config.reviewers;
21
+ this.costTracker = {
22
+ oracleSpent: 0,
23
+ workerSpent: 0,
24
+ reviewerSpent: 0,
25
+ totalBudget: config.costBudget || 10
26
+ // $10 default
27
+ };
28
+ logger.info("Oracle/Worker coordinator initialized", {
29
+ oracle: this.oracle.model,
30
+ workers: this.workerPool.length,
31
+ budget: this.costTracker.totalBudget
32
+ });
33
+ }
34
+ /**
35
+ * Launch swarm with Oracle/Worker pattern
36
+ */
37
+ async launchOracleWorkerSwarm(projectDescription, taskHints) {
38
+ logger.info("Launching Oracle/Worker swarm", {
39
+ project: projectDescription.substring(0, 100)
40
+ });
41
+ const oracleTaskId = await this.createOracleTask(
42
+ "project_planning",
43
+ projectDescription,
44
+ taskHints
45
+ );
46
+ const decomposition = await this.executeOracleTask(oracleTaskId);
47
+ const workerTasks = this.allocateTasksToWorkers(decomposition);
48
+ const workerPromises = workerTasks.map(
49
+ (task) => this.executeWorkerTask(task)
50
+ );
51
+ const reviewTaskId = await this.scheduleOracleReview(decomposition);
52
+ const [workerResults, reviewResult] = await Promise.all([
53
+ Promise.allSettled(workerPromises),
54
+ this.executeOracleTask(reviewTaskId)
55
+ ]);
56
+ const swarmId = await this.integrateResults(workerResults, reviewResult);
57
+ this.logCostAnalysis();
58
+ return swarmId;
59
+ }
60
+ /**
61
+ * Create planning task for Oracle
62
+ */
63
+ async createOracleTask(type, description, hints) {
64
+ const taskId = uuidv4();
65
+ const oraclePrompt = this.buildOraclePrompt(type, description, hints);
66
+ const estimatedTokens = this.estimateTokens(oraclePrompt);
67
+ logger.info("Oracle task created", {
68
+ taskId,
69
+ type,
70
+ estimatedTokens,
71
+ estimatedCost: estimatedTokens * this.oracle.costPerToken
72
+ });
73
+ return taskId;
74
+ }
75
+ /**
76
+ * Build specialized prompt for Oracle model
77
+ */
78
+ buildOraclePrompt(type, description, hints) {
79
+ const basePrompt = `
80
+ # ORACLE ROLE: Strategic Planning & Coordination
81
+
82
+ You are the Oracle in an Oracle/Worker pattern. Your role:
83
+ - HIGH-LEVEL STRATEGIC thinking
84
+ - TASK DECOMPOSITION for worker agents
85
+ - QUALITY CONTROL and review
86
+ - COORDINATION between workers
87
+ - ERROR CORRECTION and replanning
88
+
89
+ ## Project Context
90
+ ${description}
91
+
92
+ ${hints ? `## Hints & Context
93
+ ${hints.map((h) => `- ${h}`).join("\n")}` : ""}
94
+
95
+ ## Your Oracle Responsibilities
96
+ 1. **Decompose** this project into discrete, parallelizable tasks
97
+ 2. **Assign complexity levels** (low/medium/high) to guide worker selection
98
+ 3. **Define acceptance criteria** for each task
99
+ 4. **Identify dependencies** between tasks
100
+ 5. **Plan coordination touchpoints** for integration
101
+
102
+ ## Worker Constraints
103
+ - Workers are smaller models optimized for focused execution
104
+ - Workers excel at: implementation, testing, documentation, simple analysis
105
+ - Workers struggle with: complex architecture, strategic decisions, cross-cutting concerns
106
+
107
+ ## Output Required
108
+ Provide a detailed task decomposition in JSON format:
109
+
110
+ \`\`\`json
111
+ {
112
+ "project_summary": "Brief overview",
113
+ "task_decomposition": [
114
+ {
115
+ "id": "unique-id",
116
+ "title": "Task name",
117
+ "description": "Detailed description",
118
+ "complexity": "low|medium|high",
119
+ "type": "implementation|testing|documentation|analysis",
120
+ "estimated_effort": "1-5 scale",
121
+ "worker_requirements": ["specific capabilities needed"],
122
+ "acceptance_criteria": ["criterion 1", "criterion 2"],
123
+ "dependencies": ["task-id-1", "task-id-2"]
124
+ }
125
+ ],
126
+ "coordination_plan": {
127
+ "integration_points": ["When to sync between workers"],
128
+ "review_checkpoints": ["When Oracle should review progress"],
129
+ "risk_mitigation": ["Potential issues and solutions"]
130
+ }
131
+ }
132
+ \`\`\`
133
+
134
+ Remember: Your intelligence is expensive. Focus on high-value strategic thinking that workers cannot do effectively.
135
+ `;
136
+ return basePrompt;
137
+ }
138
+ /**
139
+ * Execute Oracle task with high-end model
140
+ */
141
+ async executeOracleTask(taskId) {
142
+ logger.info("Executing Oracle task", { taskId });
143
+ const ralph = new RalphStackMemoryBridge({
144
+ baseDir: `.oracle/${taskId}`,
145
+ maxIterations: 3,
146
+ // Oracle should be efficient
147
+ useStackMemory: true
148
+ });
149
+ const result = await ralph.run();
150
+ const tokens = this.estimateTokens(result);
151
+ const cost = tokens * this.oracle.costPerToken;
152
+ this.costTracker.oracleSpent += cost;
153
+ logger.info("Oracle task completed", {
154
+ taskId,
155
+ tokensUsed: tokens,
156
+ cost: cost.toFixed(4)
157
+ });
158
+ return this.parseTaskDecomposition(result);
159
+ }
160
+ /**
161
+ * Allocate decomposed tasks to worker models
162
+ */
163
+ allocateTasksToWorkers(decomposition) {
164
+ const allocatedTasks = [];
165
+ for (const task of decomposition) {
166
+ const workerModel = this.selectWorkerForTask(task);
167
+ const workerPrompt = this.buildWorkerPrompt(task, workerModel);
168
+ allocatedTasks.push({
169
+ ...task,
170
+ assignedModel: "worker"
171
+ });
172
+ logger.debug("Task allocated to worker", {
173
+ taskId: task.id,
174
+ complexity: task.complexity,
175
+ worker: workerModel.model
176
+ });
177
+ }
178
+ return allocatedTasks;
179
+ }
180
+ /**
181
+ * Select optimal worker model for task complexity
182
+ */
183
+ selectWorkerForTask(task) {
184
+ if (task.complexity === "high") {
185
+ return this.workerPool[0];
186
+ } else {
187
+ return this.workerPool.reduce(
188
+ (cheapest, current) => current.costPerToken < cheapest.costPerToken ? current : cheapest
189
+ );
190
+ }
191
+ }
192
+ /**
193
+ * Build focused prompt for worker models
194
+ */
195
+ buildWorkerPrompt(task, worker) {
196
+ return `
197
+ # WORKER ROLE: Focused Task Execution
198
+
199
+ You are a specialized worker in an Oracle/Worker pattern.
200
+
201
+ ## Your Task
202
+ ${task.type}: ${task.title}
203
+
204
+ ${task.description}
205
+
206
+ ## Success Criteria
207
+ ${task.acceptanceCriteria.map((c) => `- ${c}`).join("\n")}
208
+
209
+ ## Worker Guidelines
210
+ - FOCUS on this specific task only
211
+ - IMPLEMENT according to the specifications provided
212
+ - ASK for clarification if requirements are unclear
213
+ - COMMUNICATE progress through shared context
214
+ - COMPLETE the task efficiently without over-engineering
215
+
216
+ ## Constraints
217
+ - You are optimized for execution, not planning
218
+ - Stay within your assigned scope
219
+ - Collaborate with other workers through shared context
220
+ - The Oracle will handle integration and review
221
+
222
+ Execute your task now.
223
+ `;
224
+ }
225
+ /**
226
+ * Execute worker task with cost tracking
227
+ */
228
+ async executeWorkerTask(task) {
229
+ logger.info("Executing worker task", {
230
+ taskId: task.id,
231
+ complexity: task.complexity
232
+ });
233
+ const ralph = new RalphStackMemoryBridge({
234
+ baseDir: `.workers/${task.id}`,
235
+ maxIterations: task.complexity === "low" ? 3 : 7,
236
+ useStackMemory: true
237
+ });
238
+ const result = await ralph.run();
239
+ const workerModel = this.selectWorkerForTask(task);
240
+ const tokens = this.estimateTokens(result);
241
+ const cost = tokens * workerModel.costPerToken;
242
+ this.costTracker.workerSpent += cost;
243
+ logger.info("Worker task completed", {
244
+ taskId: task.id,
245
+ tokensUsed: tokens,
246
+ cost: cost.toFixed(4)
247
+ });
248
+ return result;
249
+ }
250
+ /**
251
+ * Schedule Oracle review of worker progress
252
+ */
253
+ async scheduleOracleReview(decomposition) {
254
+ const reviewTaskId = uuidv4();
255
+ logger.info("Oracle review scheduled", {
256
+ reviewTaskId,
257
+ tasksToReview: decomposition.length
258
+ });
259
+ return reviewTaskId;
260
+ }
261
+ /**
262
+ * Integrate worker results under Oracle coordination
263
+ */
264
+ async integrateResults(workerResults, reviewResult) {
265
+ const swarmId = uuidv4();
266
+ const successfulTasks = workerResults.filter(
267
+ (result) => result.status === "fulfilled"
268
+ ).length;
269
+ logger.info("Integration completed", {
270
+ swarmId,
271
+ totalTasks: workerResults.length,
272
+ successfulTasks,
273
+ successRate: (successfulTasks / workerResults.length * 100).toFixed(1)
274
+ });
275
+ return swarmId;
276
+ }
277
+ /**
278
+ * Parse task decomposition from Oracle output
279
+ */
280
+ parseTaskDecomposition(output) {
281
+ return [];
282
+ }
283
+ /**
284
+ * Estimate token usage for cost calculation
285
+ */
286
+ estimateTokens(text) {
287
+ return Math.ceil(text.length / 4);
288
+ }
289
+ /**
290
+ * Log cost analysis and efficiency metrics
291
+ */
292
+ logCostAnalysis() {
293
+ const total = this.costTracker.oracleSpent + this.costTracker.workerSpent + this.costTracker.reviewerSpent;
294
+ const savings = this.calculateTraditionalCost() - total;
295
+ logger.info("Oracle/Worker Cost Analysis", {
296
+ oracleSpent: `$${this.costTracker.oracleSpent.toFixed(4)}`,
297
+ workerSpent: `$${this.costTracker.workerSpent.toFixed(4)}`,
298
+ totalSpent: `$${total.toFixed(4)}`,
299
+ budgetUsed: `${(total / this.costTracker.totalBudget * 100).toFixed(1)}%`,
300
+ estimatedSavings: `$${savings.toFixed(4)}`,
301
+ efficiency: `${(this.costTracker.workerSpent / total * 100).toFixed(1)}% worker tasks`
302
+ });
303
+ }
304
+ /**
305
+ * Calculate what this would cost with all-Oracle approach
306
+ */
307
+ calculateTraditionalCost() {
308
+ const totalSpent = this.costTracker.oracleSpent + this.costTracker.workerSpent + this.costTracker.reviewerSpent;
309
+ const avgWorkerCost = this.workerPool[0]?.costPerToken || 1e-3;
310
+ const workerTokensAsOracle = this.costTracker.workerSpent / avgWorkerCost;
311
+ return this.costTracker.oracleSpent + workerTokensAsOracle * this.oracle.costPerToken;
312
+ }
313
+ }
314
+ const defaultModelConfigs = {
315
+ oracle: [
316
+ {
317
+ tier: "oracle",
318
+ provider: "claude",
319
+ model: "claude-3-opus-20240229",
320
+ costPerToken: 0.015,
321
+ // $15/1M input tokens
322
+ capabilities: [
323
+ "strategic_planning",
324
+ "complex_reasoning",
325
+ "task_decomposition",
326
+ "quality_review",
327
+ "error_correction"
328
+ ]
329
+ }
330
+ ],
331
+ worker: [
332
+ {
333
+ tier: "worker",
334
+ provider: "claude",
335
+ model: "claude-3-haiku-20240307",
336
+ costPerToken: 25e-5,
337
+ // $0.25/1M input tokens
338
+ capabilities: [
339
+ "code_implementation",
340
+ "unit_testing",
341
+ "documentation",
342
+ "simple_analysis",
343
+ "data_processing"
344
+ ]
345
+ },
346
+ {
347
+ tier: "worker",
348
+ provider: "openai",
349
+ model: "gpt-4o-mini",
350
+ costPerToken: 15e-5,
351
+ // $0.15/1M input tokens
352
+ capabilities: [
353
+ "rapid_prototyping",
354
+ "script_writing",
355
+ "basic_testing",
356
+ "formatting",
357
+ "simple_refactoring"
358
+ ]
359
+ }
360
+ ],
361
+ reviewer: [
362
+ {
363
+ tier: "reviewer",
364
+ provider: "claude",
365
+ model: "claude-3-sonnet-20240229",
366
+ costPerToken: 3e-3,
367
+ // $3/1M input tokens
368
+ capabilities: [
369
+ "code_review",
370
+ "quality_assessment",
371
+ "integration_testing",
372
+ "performance_analysis",
373
+ "security_review"
374
+ ]
375
+ }
376
+ ]
377
+ };
378
+ var oracle_worker_pattern_default = OracleWorkerCoordinator;
379
+ export {
380
+ OracleWorkerCoordinator,
381
+ oracle_worker_pattern_default as default,
382
+ defaultModelConfigs
383
+ };
384
+ //# sourceMappingURL=oracle-worker-pattern.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/integrations/ralph/patterns/oracle-worker-pattern.ts"],
4
+ "sourcesContent": ["/**\n * Oracle/Worker Pattern Implementation for StackMemory Swarms\n * \n * Uses high-end model (Oracle) for planning, review, and coordination\n * Uses smaller models (Workers) for task execution and implementation\n * \n * Cost-effective scaling: Intelligence where needed, efficiency for execution\n */\n\nimport { v4 as uuidv4 } from 'uuid';\nimport { logger } from '../../../core/monitoring/logger.js';\nimport { SwarmCoordinator } from '../swarm/swarm-coordinator.js';\nimport { RalphStackMemoryBridge } from '../bridge/ralph-stackmemory-bridge.js';\n\n// Model tiers based on capability and cost\nexport type ModelTier = 'oracle' | 'worker' | 'reviewer';\n\nexport interface ModelConfig {\n tier: ModelTier;\n provider: 'claude' | 'openai' | 'anthropic';\n model: string;\n costPerToken: number;\n capabilities: string[];\n}\n\nexport interface OracleWorkerConfig {\n oracle: ModelConfig;\n workers: ModelConfig[];\n reviewers: ModelConfig[];\n maxWorkers: number;\n coordinationInterval: number;\n costBudget?: number;\n}\n\nexport interface TaskDecomposition {\n id: string;\n type: 'planning' | 'implementation' | 'review' | 'coordination';\n priority: number;\n complexity: 'low' | 'medium' | 'high';\n estimatedTokens: number;\n assignedModel: ModelTier;\n dependencies: string[];\n acceptanceCriteria: string[];\n}\n\n/**\n * Oracle/Worker Pattern Coordinator\n * Implements cost-effective multi-model orchestration\n */\nexport class OracleWorkerCoordinator extends SwarmCoordinator {\n private oracle: ModelConfig;\n private workerPool: ModelConfig[];\n private reviewerPool: ModelConfig[];\n private costTracker: {\n oracleSpent: number;\n workerSpent: number;\n reviewerSpent: number;\n totalBudget: number;\n };\n \n constructor(config: OracleWorkerConfig) {\n super({\n maxAgents: config.maxWorkers + 2, // Workers + Oracle + Reviewer\n coordinationInterval: config.coordinationInterval,\n enableDynamicPlanning: true,\n pathologicalBehaviorDetection: true,\n });\n\n this.oracle = config.oracle;\n this.workerPool = config.workers;\n this.reviewerPool = config.reviewers;\n \n this.costTracker = {\n oracleSpent: 0,\n workerSpent: 0,\n reviewerSpent: 0,\n totalBudget: config.costBudget || 10.0, // $10 default\n };\n\n logger.info('Oracle/Worker coordinator initialized', {\n oracle: this.oracle.model,\n workers: this.workerPool.length,\n budget: this.costTracker.totalBudget,\n });\n }\n\n /**\n * Launch swarm with Oracle/Worker pattern\n */\n async launchOracleWorkerSwarm(\n projectDescription: string,\n taskHints?: string[]\n ): Promise<string> {\n logger.info('Launching Oracle/Worker swarm', {\n project: projectDescription.substring(0, 100),\n });\n\n // Phase 1: Oracle Planning\n const oracleTaskId = await this.createOracleTask(\n 'project_planning',\n projectDescription,\n taskHints\n );\n \n const decomposition = await this.executeOracleTask(oracleTaskId);\n \n // Phase 2: Worker Assignment\n const workerTasks = this.allocateTasksToWorkers(decomposition);\n \n // Phase 3: Parallel Worker Execution\n const workerPromises = workerTasks.map(task => \n this.executeWorkerTask(task)\n );\n \n // Phase 4: Oracle Review & Coordination\n const reviewTaskId = await this.scheduleOracleReview(decomposition);\n \n // Execute workers in parallel, oracle coordinates\n const [workerResults, reviewResult] = await Promise.all([\n Promise.allSettled(workerPromises),\n this.executeOracleTask(reviewTaskId)\n ]);\n\n // Phase 5: Final Integration\n const swarmId = await this.integrateResults(workerResults, reviewResult);\n \n this.logCostAnalysis();\n return swarmId;\n }\n\n /**\n * Create planning task for Oracle\n */\n private async createOracleTask(\n type: string,\n description: string,\n hints?: string[]\n ): Promise<string> {\n const taskId = uuidv4();\n \n const oraclePrompt = this.buildOraclePrompt(type, description, hints);\n const estimatedTokens = this.estimateTokens(oraclePrompt);\n \n logger.info('Oracle task created', {\n taskId,\n type,\n estimatedTokens,\n estimatedCost: estimatedTokens * this.oracle.costPerToken,\n });\n\n return taskId;\n }\n\n /**\n * Build specialized prompt for Oracle model\n */\n private buildOraclePrompt(\n type: string,\n description: string,\n hints?: string[]\n ): string {\n const basePrompt = `\n# ORACLE ROLE: Strategic Planning & Coordination\n\nYou are the Oracle in an Oracle/Worker pattern. Your role:\n- HIGH-LEVEL STRATEGIC thinking\n- TASK DECOMPOSITION for worker agents \n- QUALITY CONTROL and review\n- COORDINATION between workers\n- ERROR CORRECTION and replanning\n\n## Project Context\n${description}\n\n${hints ? `## Hints & Context\\n${hints.map(h => `- ${h}`).join('\\n')}` : ''}\n\n## Your Oracle Responsibilities\n1. **Decompose** this project into discrete, parallelizable tasks\n2. **Assign complexity levels** (low/medium/high) to guide worker selection\n3. **Define acceptance criteria** for each task\n4. **Identify dependencies** between tasks\n5. **Plan coordination touchpoints** for integration\n\n## Worker Constraints\n- Workers are smaller models optimized for focused execution\n- Workers excel at: implementation, testing, documentation, simple analysis\n- Workers struggle with: complex architecture, strategic decisions, cross-cutting concerns\n\n## Output Required\nProvide a detailed task decomposition in JSON format:\n\n\\`\\`\\`json\n{\n \"project_summary\": \"Brief overview\",\n \"task_decomposition\": [\n {\n \"id\": \"unique-id\",\n \"title\": \"Task name\",\n \"description\": \"Detailed description\",\n \"complexity\": \"low|medium|high\",\n \"type\": \"implementation|testing|documentation|analysis\",\n \"estimated_effort\": \"1-5 scale\",\n \"worker_requirements\": [\"specific capabilities needed\"],\n \"acceptance_criteria\": [\"criterion 1\", \"criterion 2\"],\n \"dependencies\": [\"task-id-1\", \"task-id-2\"]\n }\n ],\n \"coordination_plan\": {\n \"integration_points\": [\"When to sync between workers\"],\n \"review_checkpoints\": [\"When Oracle should review progress\"],\n \"risk_mitigation\": [\"Potential issues and solutions\"]\n }\n}\n\\`\\`\\`\n\nRemember: Your intelligence is expensive. Focus on high-value strategic thinking that workers cannot do effectively.\n `;\n\n return basePrompt;\n }\n\n /**\n * Execute Oracle task with high-end model\n */\n private async executeOracleTask(taskId: string): Promise<TaskDecomposition[]> {\n logger.info('Executing Oracle task', { taskId });\n \n // Create Ralph loop with Oracle model configuration\n const ralph = new RalphStackMemoryBridge({\n baseDir: `.oracle/${taskId}`,\n maxIterations: 3, // Oracle should be efficient\n useStackMemory: true,\n });\n\n // Execute with Oracle model (implementation would integrate with actual model APIs)\n const result = await ralph.run();\n \n // Track Oracle costs\n const tokens = this.estimateTokens(result);\n const cost = tokens * this.oracle.costPerToken;\n this.costTracker.oracleSpent += cost;\n \n logger.info('Oracle task completed', {\n taskId,\n tokensUsed: tokens,\n cost: cost.toFixed(4),\n });\n\n return this.parseTaskDecomposition(result);\n }\n\n /**\n * Allocate decomposed tasks to worker models\n */\n private allocateTasksToWorkers(\n decomposition: TaskDecomposition[]\n ): TaskDecomposition[] {\n const allocatedTasks: TaskDecomposition[] = [];\n\n for (const task of decomposition) {\n // Select optimal worker model based on task complexity\n const workerModel = this.selectWorkerForTask(task);\n \n // Create worker-specific prompt\n const workerPrompt = this.buildWorkerPrompt(task, workerModel);\n \n allocatedTasks.push({\n ...task,\n assignedModel: 'worker' as ModelTier,\n });\n\n logger.debug('Task allocated to worker', {\n taskId: task.id,\n complexity: task.complexity,\n worker: workerModel.model,\n });\n }\n\n return allocatedTasks;\n }\n\n /**\n * Select optimal worker model for task complexity\n */\n private selectWorkerForTask(task: TaskDecomposition): ModelConfig {\n // Simple allocation strategy - can be enhanced\n if (task.complexity === 'high') {\n // Use best available worker for complex tasks\n return this.workerPool[0];\n } else {\n // Use cheapest worker for simple tasks\n return this.workerPool.reduce((cheapest, current) => \n current.costPerToken < cheapest.costPerToken ? current : cheapest\n );\n }\n }\n\n /**\n * Build focused prompt for worker models\n */\n private buildWorkerPrompt(\n task: TaskDecomposition,\n worker: ModelConfig\n ): string {\n return `\n# WORKER ROLE: Focused Task Execution\n\nYou are a specialized worker in an Oracle/Worker pattern.\n\n## Your Task\n${task.type}: ${task.title}\n\n${task.description}\n\n## Success Criteria\n${task.acceptanceCriteria.map(c => `- ${c}`).join('\\n')}\n\n## Worker Guidelines\n- FOCUS on this specific task only\n- IMPLEMENT according to the specifications provided\n- ASK for clarification if requirements are unclear\n- COMMUNICATE progress through shared context\n- COMPLETE the task efficiently without over-engineering\n\n## Constraints\n- You are optimized for execution, not planning\n- Stay within your assigned scope\n- Collaborate with other workers through shared context\n- The Oracle will handle integration and review\n\nExecute your task now.\n `;\n }\n\n /**\n * Execute worker task with cost tracking\n */\n private async executeWorkerTask(task: TaskDecomposition): Promise<any> {\n logger.info('Executing worker task', { \n taskId: task.id,\n complexity: task.complexity \n });\n\n const ralph = new RalphStackMemoryBridge({\n baseDir: `.workers/${task.id}`,\n maxIterations: task.complexity === 'low' ? 3 : 7,\n useStackMemory: true,\n });\n\n const result = await ralph.run();\n \n // Track worker costs\n const workerModel = this.selectWorkerForTask(task);\n const tokens = this.estimateTokens(result);\n const cost = tokens * workerModel.costPerToken;\n this.costTracker.workerSpent += cost;\n\n logger.info('Worker task completed', {\n taskId: task.id,\n tokensUsed: tokens,\n cost: cost.toFixed(4),\n });\n\n return result;\n }\n\n /**\n * Schedule Oracle review of worker progress\n */\n private async scheduleOracleReview(\n decomposition: TaskDecomposition[]\n ): Promise<string> {\n const reviewTaskId = uuidv4();\n \n // Oracle reviews worker outputs and coordinates integration\n logger.info('Oracle review scheduled', { \n reviewTaskId,\n tasksToReview: decomposition.length \n });\n\n return reviewTaskId;\n }\n\n /**\n * Integrate worker results under Oracle coordination\n */\n private async integrateResults(\n workerResults: PromiseSettledResult<any>[],\n reviewResult: any\n ): Promise<string> {\n const swarmId = uuidv4();\n \n const successfulTasks = workerResults.filter(\n result => result.status === 'fulfilled'\n ).length;\n\n logger.info('Integration completed', {\n swarmId,\n totalTasks: workerResults.length,\n successfulTasks,\n successRate: (successfulTasks / workerResults.length * 100).toFixed(1),\n });\n\n return swarmId;\n }\n\n /**\n * Parse task decomposition from Oracle output\n */\n private parseTaskDecomposition(output: string): TaskDecomposition[] {\n // Parse JSON from Oracle output\n // Implementation would extract the JSON task decomposition\n return [];\n }\n\n /**\n * Estimate token usage for cost calculation\n */\n private estimateTokens(text: string): number {\n // Rough estimation: ~4 characters per token\n return Math.ceil(text.length / 4);\n }\n\n /**\n * Log cost analysis and efficiency metrics\n */\n private logCostAnalysis(): void {\n const total = this.costTracker.oracleSpent + \n this.costTracker.workerSpent + \n this.costTracker.reviewerSpent;\n\n const savings = this.calculateTraditionalCost() - total;\n\n logger.info('Oracle/Worker Cost Analysis', {\n oracleSpent: `$${this.costTracker.oracleSpent.toFixed(4)}`,\n workerSpent: `$${this.costTracker.workerSpent.toFixed(4)}`,\n totalSpent: `$${total.toFixed(4)}`,\n budgetUsed: `${(total / this.costTracker.totalBudget * 100).toFixed(1)}%`,\n estimatedSavings: `$${savings.toFixed(4)}`,\n efficiency: `${(this.costTracker.workerSpent / total * 100).toFixed(1)}% worker tasks`,\n });\n }\n\n /**\n * Calculate what this would cost with all-Oracle approach\n */\n private calculateTraditionalCost(): number {\n const totalSpent = this.costTracker.oracleSpent + \n this.costTracker.workerSpent + \n this.costTracker.reviewerSpent;\n \n // Estimate if everything was done with Oracle model\n const avgWorkerCost = this.workerPool[0]?.costPerToken || 0.001;\n const workerTokensAsOracle = this.costTracker.workerSpent / avgWorkerCost;\n \n return this.costTracker.oracleSpent + \n (workerTokensAsOracle * this.oracle.costPerToken);\n }\n}\n\n/**\n * Default model configurations for Oracle/Worker pattern\n */\nexport const defaultModelConfigs: Record<ModelTier, ModelConfig[]> = {\n oracle: [\n {\n tier: 'oracle',\n provider: 'claude',\n model: 'claude-3-opus-20240229',\n costPerToken: 0.015, // $15/1M input tokens\n capabilities: [\n 'strategic_planning',\n 'complex_reasoning',\n 'task_decomposition',\n 'quality_review',\n 'error_correction'\n ]\n }\n ],\n \n worker: [\n {\n tier: 'worker',\n provider: 'claude', \n model: 'claude-3-haiku-20240307',\n costPerToken: 0.00025, // $0.25/1M input tokens\n capabilities: [\n 'code_implementation',\n 'unit_testing',\n 'documentation',\n 'simple_analysis',\n 'data_processing'\n ]\n },\n {\n tier: 'worker',\n provider: 'openai',\n model: 'gpt-4o-mini',\n costPerToken: 0.00015, // $0.15/1M input tokens \n capabilities: [\n 'rapid_prototyping',\n 'script_writing',\n 'basic_testing',\n 'formatting',\n 'simple_refactoring'\n ]\n }\n ],\n\n reviewer: [\n {\n tier: 'reviewer',\n provider: 'claude',\n model: 'claude-3-sonnet-20240229', \n costPerToken: 0.003, // $3/1M input tokens\n capabilities: [\n 'code_review',\n 'quality_assessment',\n 'integration_testing',\n 'performance_analysis',\n 'security_review'\n ]\n }\n ]\n};\n\nexport default OracleWorkerCoordinator;"],
5
+ "mappings": "AASA,SAAS,MAAM,cAAc;AAC7B,SAAS,cAAc;AACvB,SAAS,wBAAwB;AACjC,SAAS,8BAA8B;AAqChC,MAAM,gCAAgC,iBAAiB;AAAA,EACpD;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAOR,YAAY,QAA4B;AACtC,UAAM;AAAA,MACJ,WAAW,OAAO,aAAa;AAAA;AAAA,MAC/B,sBAAsB,OAAO;AAAA,MAC7B,uBAAuB;AAAA,MACvB,+BAA+B;AAAA,IACjC,CAAC;AAED,SAAK,SAAS,OAAO;AACrB,SAAK,aAAa,OAAO;AACzB,SAAK,eAAe,OAAO;AAE3B,SAAK,cAAc;AAAA,MACjB,aAAa;AAAA,MACb,aAAa;AAAA,MACb,eAAe;AAAA,MACf,aAAa,OAAO,cAAc;AAAA;AAAA,IACpC;AAEA,WAAO,KAAK,yCAAyC;AAAA,MACnD,QAAQ,KAAK,OAAO;AAAA,MACpB,SAAS,KAAK,WAAW;AAAA,MACzB,QAAQ,KAAK,YAAY;AAAA,IAC3B,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,wBACJ,oBACA,WACiB;AACjB,WAAO,KAAK,iCAAiC;AAAA,MAC3C,SAAS,mBAAmB,UAAU,GAAG,GAAG;AAAA,IAC9C,CAAC;AAGD,UAAM,eAAe,MAAM,KAAK;AAAA,MAC9B;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,UAAM,gBAAgB,MAAM,KAAK,kBAAkB,YAAY;AAG/D,UAAM,cAAc,KAAK,uBAAuB,aAAa;AAG7D,UAAM,iBAAiB,YAAY;AAAA,MAAI,UACrC,KAAK,kBAAkB,IAAI;AAAA,IAC7B;AAGA,UAAM,eAAe,MAAM,KAAK,qBAAqB,aAAa;AAGlE,UAAM,CAAC,eAAe,YAAY,IAAI,MAAM,QAAQ,IAAI;AAAA,MACtD,QAAQ,WAAW,cAAc;AAAA,MACjC,KAAK,kBAAkB,YAAY;AAAA,IACrC,CAAC;AAGD,UAAM,UAAU,MAAM,KAAK,iBAAiB,eAAe,YAAY;AAEvE,SAAK,gBAAgB;AACrB,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,iBACZ,MACA,aACA,OACiB;AACjB,UAAM,SAAS,OAAO;AAEtB,UAAM,eAAe,KAAK,kBAAkB,MAAM,aAAa,KAAK;AACpE,UAAM,kBAAkB,KAAK,eAAe,YAAY;AAExD,WAAO,KAAK,uBAAuB;AAAA,MACjC;AAAA,MACA;AAAA,MACA;AAAA,MACA,eAAe,kBAAkB,KAAK,OAAO;AAAA,IAC/C,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,kBACN,MACA,aACA,OACQ;AACR,UAAM,aAAa;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWrB,WAAW;AAAA;AAAA,EAEX,QAAQ;AAAA,EAAuB,MAAM,IAAI,OAAK,KAAK,CAAC,EAAE,EAAE,KAAK,IAAI,CAAC,KAAK,EAAE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AA4CvE,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,kBAAkB,QAA8C;AAC5E,WAAO,KAAK,yBAAyB,EAAE,OAAO,CAAC;AAG/C,UAAM,QAAQ,IAAI,uBAAuB;AAAA,MACvC,SAAS,WAAW,MAAM;AAAA,MAC1B,eAAe;AAAA;AAAA,MACf,gBAAgB;AAAA,IAClB,CAAC;AAGD,UAAM,SAAS,MAAM,MAAM,IAAI;AAG/B,UAAM,SAAS,KAAK,eAAe,MAAM;AACzC,UAAM,OAAO,SAAS,KAAK,OAAO;AAClC,SAAK,YAAY,eAAe;AAEhC,WAAO,KAAK,yBAAyB;AAAA,MACnC;AAAA,MACA,YAAY;AAAA,MACZ,MAAM,KAAK,QAAQ,CAAC;AAAA,IACtB,CAAC;AAED,WAAO,KAAK,uBAAuB,MAAM;AAAA,EAC3C;AAAA;AAAA;AAAA;AAAA,EAKQ,uBACN,eACqB;AACrB,UAAM,iBAAsC,CAAC;AAE7C,eAAW,QAAQ,eAAe;AAEhC,YAAM,cAAc,KAAK,oBAAoB,IAAI;AAGjD,YAAM,eAAe,KAAK,kBAAkB,MAAM,WAAW;AAE7D,qBAAe,KAAK;AAAA,QAClB,GAAG;AAAA,QACH,eAAe;AAAA,MACjB,CAAC;AAED,aAAO,MAAM,4BAA4B;AAAA,QACvC,QAAQ,KAAK;AAAA,QACb,YAAY,KAAK;AAAA,QACjB,QAAQ,YAAY;AAAA,MACtB,CAAC;AAAA,IACH;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAoB,MAAsC;AAEhE,QAAI,KAAK,eAAe,QAAQ;AAE9B,aAAO,KAAK,WAAW,CAAC;AAAA,IAC1B,OAAO;AAEL,aAAO,KAAK,WAAW;AAAA,QAAO,CAAC,UAAU,YACvC,QAAQ,eAAe,SAAS,eAAe,UAAU;AAAA,MAC3D;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,kBACN,MACA,QACQ;AACR,WAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMT,KAAK,IAAI,KAAK,KAAK,KAAK;AAAA;AAAA,EAExB,KAAK,WAAW;AAAA;AAAA;AAAA,EAGhB,KAAK,mBAAmB,IAAI,OAAK,KAAK,CAAC,EAAE,EAAE,KAAK,IAAI,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAiBrD;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,kBAAkB,MAAuC;AACrE,WAAO,KAAK,yBAAyB;AAAA,MACnC,QAAQ,KAAK;AAAA,MACb,YAAY,KAAK;AAAA,IACnB,CAAC;AAED,UAAM,QAAQ,IAAI,uBAAuB;AAAA,MACvC,SAAS,YAAY,KAAK,EAAE;AAAA,MAC5B,eAAe,KAAK,eAAe,QAAQ,IAAI;AAAA,MAC/C,gBAAgB;AAAA,IAClB,CAAC;AAED,UAAM,SAAS,MAAM,MAAM,IAAI;AAG/B,UAAM,cAAc,KAAK,oBAAoB,IAAI;AACjD,UAAM,SAAS,KAAK,eAAe,MAAM;AACzC,UAAM,OAAO,SAAS,YAAY;AAClC,SAAK,YAAY,eAAe;AAEhC,WAAO,KAAK,yBAAyB;AAAA,MACnC,QAAQ,KAAK;AAAA,MACb,YAAY;AAAA,MACZ,MAAM,KAAK,QAAQ,CAAC;AAAA,IACtB,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,qBACZ,eACiB;AACjB,UAAM,eAAe,OAAO;AAG5B,WAAO,KAAK,2BAA2B;AAAA,MACrC;AAAA,MACA,eAAe,cAAc;AAAA,IAC/B,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,iBACZ,eACA,cACiB;AACjB,UAAM,UAAU,OAAO;AAEvB,UAAM,kBAAkB,cAAc;AAAA,MACpC,YAAU,OAAO,WAAW;AAAA,IAC9B,EAAE;AAEF,WAAO,KAAK,yBAAyB;AAAA,MACnC;AAAA,MACA,YAAY,cAAc;AAAA,MAC1B;AAAA,MACA,cAAc,kBAAkB,cAAc,SAAS,KAAK,QAAQ,CAAC;AAAA,IACvE,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,uBAAuB,QAAqC;AAGlE,WAAO,CAAC;AAAA,EACV;AAAA;AAAA;AAAA;AAAA,EAKQ,eAAe,MAAsB;AAE3C,WAAO,KAAK,KAAK,KAAK,SAAS,CAAC;AAAA,EAClC;AAAA;AAAA;AAAA;AAAA,EAKQ,kBAAwB;AAC9B,UAAM,QAAQ,KAAK,YAAY,cAClB,KAAK,YAAY,cACjB,KAAK,YAAY;AAE9B,UAAM,UAAU,KAAK,yBAAyB,IAAI;AAElD,WAAO,KAAK,+BAA+B;AAAA,MACzC,aAAa,IAAI,KAAK,YAAY,YAAY,QAAQ,CAAC,CAAC;AAAA,MACxD,aAAa,IAAI,KAAK,YAAY,YAAY,QAAQ,CAAC,CAAC;AAAA,MACxD,YAAY,IAAI,MAAM,QAAQ,CAAC,CAAC;AAAA,MAChC,YAAY,IAAI,QAAQ,KAAK,YAAY,cAAc,KAAK,QAAQ,CAAC,CAAC;AAAA,MACtE,kBAAkB,IAAI,QAAQ,QAAQ,CAAC,CAAC;AAAA,MACxC,YAAY,IAAI,KAAK,YAAY,cAAc,QAAQ,KAAK,QAAQ,CAAC,CAAC;AAAA,IACxE,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKQ,2BAAmC;AACzC,UAAM,aAAa,KAAK,YAAY,cAClB,KAAK,YAAY,cACjB,KAAK,YAAY;AAGnC,UAAM,gBAAgB,KAAK,WAAW,CAAC,GAAG,gBAAgB;AAC1D,UAAM,uBAAuB,KAAK,YAAY,cAAc;AAE5D,WAAO,KAAK,YAAY,cAChB,uBAAuB,KAAK,OAAO;AAAA,EAC7C;AACF;AAKO,MAAM,sBAAwD;AAAA,EACnE,QAAQ;AAAA,IACN;AAAA,MACE,MAAM;AAAA,MACN,UAAU;AAAA,MACV,OAAO;AAAA,MACP,cAAc;AAAA;AAAA,MACd,cAAc;AAAA,QACZ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,QAAQ;AAAA,IACN;AAAA,MACE,MAAM;AAAA,MACN,UAAU;AAAA,MACV,OAAO;AAAA,MACP,cAAc;AAAA;AAAA,MACd,cAAc;AAAA,QACZ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAAA,IACA;AAAA,MACE,MAAM;AAAA,MACN,UAAU;AAAA,MACV,OAAO;AAAA,MACP,cAAc;AAAA;AAAA,MACd,cAAc;AAAA,QACZ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,UAAU;AAAA,IACR;AAAA,MACE,MAAM;AAAA,MACN,UAAU;AAAA,MACV,OAAO;AAAA,MACP,cAAc;AAAA;AAAA,MACd,cAAc;AAAA,QACZ;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;AAEA,IAAO,gCAAQ;",
6
+ "names": []
7
+ }
@@ -28,16 +28,38 @@ class GitWorkflowManager {
28
28
  */
29
29
  async initializeAgentWorkflow(agent, task) {
30
30
  if (!this.config.enableGitWorkflow) return;
31
- const branchName = this.generateBranchName(agent, task);
31
+ let branchName = this.generateBranchName(agent, task);
32
32
  try {
33
- this.createBranch(branchName);
33
+ if (this.branchExists(branchName)) {
34
+ const existingHasChanges = this.branchHasUnmergedChanges(branchName);
35
+ if (existingHasChanges) {
36
+ const timestamp = Date.now();
37
+ branchName = `${branchName}-${timestamp}`;
38
+ logger.info(
39
+ `Branch already exists with changes, creating unique branch: ${branchName}`
40
+ );
41
+ this.createBranch(branchName);
42
+ } else {
43
+ logger.info(
44
+ `Reusing existing branch for agent ${agent.role}: ${branchName}`
45
+ );
46
+ this.checkoutBranch(branchName);
47
+ }
48
+ } else {
49
+ this.createBranch(branchName);
50
+ logger.info(
51
+ `Created git branch for agent ${agent.role}: ${branchName}`
52
+ );
53
+ }
34
54
  this.agentBranches.set(agent.id, branchName);
35
- logger.info(`Created git branch for agent ${agent.role}: ${branchName}`);
36
55
  if (this.config.autoCommit) {
37
56
  this.scheduleAutoCommit(agent, task);
38
57
  }
39
58
  } catch (error) {
40
- logger.error(`Failed to initialize git workflow for agent ${agent.role}`, error);
59
+ logger.error(
60
+ `Failed to initialize git workflow for agent ${agent.role}`,
61
+ error
62
+ );
41
63
  }
42
64
  }
43
65
  /**
@@ -122,7 +144,9 @@ class GitWorkflowManager {
122
144
  this.mergeBranch(integrationBranch);
123
145
  logger.info("Successfully integrated all agent work");
124
146
  } else {
125
- logger.warn("Integration tests failed, keeping changes in branch: " + integrationBranch);
147
+ logger.warn(
148
+ "Integration tests failed, keeping changes in branch: " + integrationBranch
149
+ );
126
150
  }
127
151
  }
128
152
  /**
@@ -131,7 +155,9 @@ class GitWorkflowManager {
131
155
  async resolveConflicts(agent) {
132
156
  const conflicts = this.getConflictedFiles();
133
157
  if (conflicts.length === 0) return;
134
- logger.warn(`Agent ${agent.role} encountering merge conflicts: ${conflicts.join(", ")}`);
158
+ logger.warn(
159
+ `Agent ${agent.role} encountering merge conflicts: ${conflicts.join(", ")}`
160
+ );
135
161
  for (const file of conflicts) {
136
162
  try {
137
163
  if (this.isAgentFile(file, agent)) {
@@ -150,12 +176,16 @@ class GitWorkflowManager {
150
176
  execSync("git commit --no-edit", { encoding: "utf8" });
151
177
  logger.info("All conflicts resolved automatically");
152
178
  } else {
153
- logger.error(`Manual intervention needed for: ${remainingConflicts.join(", ")}`);
179
+ logger.error(
180
+ `Manual intervention needed for: ${remainingConflicts.join(", ")}`
181
+ );
154
182
  }
155
183
  }
156
184
  // Private helper methods
157
185
  getCurrentBranch() {
158
- return execSync("git rev-parse --abbrev-ref HEAD", { encoding: "utf8" }).trim();
186
+ return execSync("git rev-parse --abbrev-ref HEAD", {
187
+ encoding: "utf8"
188
+ }).trim();
159
189
  }
160
190
  getMainBranch() {
161
191
  try {
@@ -207,7 +237,9 @@ class GitWorkflowManager {
207
237
  switch (strategy) {
208
238
  case "squash":
209
239
  execSync(`git merge --squash ${branchName}`, { encoding: "utf8" });
210
- execSync('git commit -m "Squashed agent changes"', { encoding: "utf8" });
240
+ execSync('git commit -m "Squashed agent changes"', {
241
+ encoding: "utf8"
242
+ });
211
243
  break;
212
244
  case "rebase":
213
245
  execSync(`git rebase ${branchName}`, { encoding: "utf8" });
@@ -249,7 +281,11 @@ class GitWorkflowManager {
249
281
  scheduleAutoCommit(agent, task) {
250
282
  const intervalMs = this.config.commitFrequency * 60 * 1e3;
251
283
  setInterval(async () => {
252
- await this.commitAgentWork(agent, task, `[${agent.role}] Auto-commit: ${task.title}`);
284
+ await this.commitAgentWork(
285
+ agent,
286
+ task,
287
+ `[${agent.role}] Auto-commit: ${task.title}`
288
+ );
253
289
  }, intervalMs);
254
290
  }
255
291
  async createPullRequest(agent, task, branchName) {
@@ -268,9 +304,12 @@ ${task.acceptanceCriteria.map((c) => `- ${c}`).join("\n")}
268
304
 
269
305
  Generated by Swarm Coordinator
270
306
  `;
271
- execSync(`gh pr create --title "${title}" --body "${body}" --base ${this.baselineBranch}`, {
272
- encoding: "utf8"
273
- });
307
+ execSync(
308
+ `gh pr create --title "${title}" --body "${body}" --base ${this.baselineBranch}`,
309
+ {
310
+ encoding: "utf8"
311
+ }
312
+ );
274
313
  logger.info(`Created PR for agent ${agent.role}`);
275
314
  } catch (error) {
276
315
  logger.warn(`Could not create PR: ${error}`);
@@ -284,6 +323,18 @@ Generated by Swarm Coordinator
284
323
  return false;
285
324
  }
286
325
  }
326
+ branchHasUnmergedChanges(branchName) {
327
+ try {
328
+ const currentBranch = this.getCurrentBranch();
329
+ const unmerged = execSync(
330
+ `git log ${currentBranch}..${branchName} --oneline`,
331
+ { encoding: "utf8", stdio: "pipe" }
332
+ );
333
+ return unmerged.trim().length > 0;
334
+ } catch {
335
+ return true;
336
+ }
337
+ }
287
338
  /**
288
339
  * Get status of all agent branches
289
340
  */
@@ -291,11 +342,13 @@ Generated by Swarm Coordinator
291
342
  const status = {
292
343
  enabled: this.config.enableGitWorkflow,
293
344
  currentBranch: this.getCurrentBranch(),
294
- agentBranches: Array.from(this.agentBranches.entries()).map(([agentId, branch]) => ({
295
- agentId,
296
- branch,
297
- exists: this.branchExists(branch)
298
- })),
345
+ agentBranches: Array.from(this.agentBranches.entries()).map(
346
+ ([agentId, branch]) => ({
347
+ agentId,
348
+ branch,
349
+ exists: this.branchExists(branch)
350
+ })
351
+ ),
299
352
  hasUncommittedChanges: this.hasUncommittedChanges()
300
353
  };
301
354
  return status;