mirror of
https://github.com/AutoMaker-Org/automaker.git
synced 2026-02-03 08:53:36 +00:00
fix(server): Address PR #733 review feedback and fix cross-platform tests
- Extract merge logic from pipeline-orchestrator to merge-service.ts to avoid HTTP self-call - Make agent-executor error handling provider-agnostic using shared isAuthenticationError utility - Fix cross-platform path handling in tests using path.normalize/path.resolve helpers - Add catch handlers in plan-approval-service tests to prevent unhandled promise rejection warnings Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import type { ExecuteOptions, ParsedTask } from '@automaker/types';
|
import type { ExecuteOptions, ParsedTask } from '@automaker/types';
|
||||||
import { buildPromptWithImages, createLogger } from '@automaker/utils';
|
import { buildPromptWithImages, createLogger, isAuthenticationError } from '@automaker/utils';
|
||||||
import { getFeatureDir } from '@automaker/platform';
|
import { getFeatureDir } from '@automaker/platform';
|
||||||
import * as secureFs from '../lib/secure-fs.js';
|
import * as secureFs from '../lib/secure-fs.js';
|
||||||
import { TypedEventBus } from './typed-event-bus.js';
|
import { TypedEventBus } from './typed-event-bus.js';
|
||||||
@@ -206,14 +206,10 @@ export class AgentExecutor {
|
|||||||
responseText += '\n\n';
|
responseText += '\n\n';
|
||||||
}
|
}
|
||||||
responseText += newText;
|
responseText += newText;
|
||||||
if (
|
// Check for authentication errors using provider-agnostic utility
|
||||||
block.text &&
|
if (block.text && isAuthenticationError(block.text))
|
||||||
(block.text.includes('Invalid API key') ||
|
|
||||||
block.text.includes('authentication_failed') ||
|
|
||||||
block.text.includes('Fix external API key'))
|
|
||||||
)
|
|
||||||
throw new Error(
|
throw new Error(
|
||||||
"Authentication failed: Invalid or expired API key. Please check your ANTHROPIC_API_KEY, or run 'claude login' to re-authenticate."
|
'Authentication failed: Invalid or expired API key. Please check your API key configuration or re-authenticate with your provider.'
|
||||||
);
|
);
|
||||||
scheduleWrite();
|
scheduleWrite();
|
||||||
const hasExplicitMarker = responseText.includes('[SPEC_GENERATED]'),
|
const hasExplicitMarker = responseText.includes('[SPEC_GENERATED]'),
|
||||||
|
|||||||
175
apps/server/src/services/merge-service.ts
Normal file
175
apps/server/src/services/merge-service.ts
Normal file
@@ -0,0 +1,175 @@
|
|||||||
|
/**
|
||||||
|
* MergeService - Direct merge operations without HTTP
|
||||||
|
*
|
||||||
|
* Extracted from worktree merge route to allow internal service calls.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { exec } from 'child_process';
|
||||||
|
import { promisify } from 'util';
|
||||||
|
import { createLogger } from '@automaker/utils';
|
||||||
|
import { spawnProcess } from '@automaker/platform';
|
||||||
|
|
||||||
|
const execAsync = promisify(exec);
|
||||||
|
const logger = createLogger('MergeService');
|
||||||
|
|
||||||
|
export interface MergeOptions {
|
||||||
|
squash?: boolean;
|
||||||
|
message?: string;
|
||||||
|
deleteWorktreeAndBranch?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MergeServiceResult {
|
||||||
|
success: boolean;
|
||||||
|
error?: string;
|
||||||
|
hasConflicts?: boolean;
|
||||||
|
mergedBranch?: string;
|
||||||
|
targetBranch?: string;
|
||||||
|
deleted?: {
|
||||||
|
worktreeDeleted: boolean;
|
||||||
|
branchDeleted: boolean;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute git command with array arguments to prevent command injection.
|
||||||
|
*/
|
||||||
|
async function execGitCommand(args: string[], cwd: string): Promise<string> {
|
||||||
|
const result = await spawnProcess({
|
||||||
|
command: 'git',
|
||||||
|
args,
|
||||||
|
cwd,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result.exitCode === 0) {
|
||||||
|
return result.stdout;
|
||||||
|
} else {
|
||||||
|
const errorMessage = result.stderr || `Git command failed with code ${result.exitCode}`;
|
||||||
|
throw new Error(errorMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate branch name to prevent command injection.
|
||||||
|
*/
|
||||||
|
function isValidBranchName(name: string): boolean {
|
||||||
|
return /^[a-zA-Z0-9._\-/]+$/.test(name) && name.length < 250;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Perform a git merge operation directly without HTTP.
|
||||||
|
*
|
||||||
|
* @param projectPath - Path to the git repository
|
||||||
|
* @param branchName - Source branch to merge
|
||||||
|
* @param worktreePath - Path to the worktree (used for deletion if requested)
|
||||||
|
* @param targetBranch - Branch to merge into (defaults to 'main')
|
||||||
|
* @param options - Merge options (squash, message, deleteWorktreeAndBranch)
|
||||||
|
*/
|
||||||
|
export async function performMerge(
|
||||||
|
projectPath: string,
|
||||||
|
branchName: string,
|
||||||
|
worktreePath: string,
|
||||||
|
targetBranch: string = 'main',
|
||||||
|
options?: MergeOptions
|
||||||
|
): Promise<MergeServiceResult> {
|
||||||
|
if (!projectPath || !branchName || !worktreePath) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: 'projectPath, branchName, and worktreePath are required',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const mergeTo = targetBranch || 'main';
|
||||||
|
|
||||||
|
// Validate source branch exists
|
||||||
|
try {
|
||||||
|
await execAsync(`git rev-parse --verify ${branchName}`, { cwd: projectPath });
|
||||||
|
} catch {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: `Branch "${branchName}" does not exist`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate target branch exists
|
||||||
|
try {
|
||||||
|
await execAsync(`git rev-parse --verify ${mergeTo}`, { cwd: projectPath });
|
||||||
|
} catch {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: `Target branch "${mergeTo}" does not exist`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Merge the feature branch into the target branch
|
||||||
|
const mergeCmd = options?.squash
|
||||||
|
? `git merge --squash ${branchName}`
|
||||||
|
: `git merge ${branchName} -m "${options?.message || `Merge ${branchName} into ${mergeTo}`}"`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await execAsync(mergeCmd, { cwd: projectPath });
|
||||||
|
} catch (mergeError: unknown) {
|
||||||
|
// Check if this is a merge conflict
|
||||||
|
const err = mergeError as { stdout?: string; stderr?: string; message?: string };
|
||||||
|
const output = `${err.stdout || ''} ${err.stderr || ''} ${err.message || ''}`;
|
||||||
|
const hasConflicts = output.includes('CONFLICT') || output.includes('Automatic merge failed');
|
||||||
|
|
||||||
|
if (hasConflicts) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: `Merge CONFLICT: Automatic merge of "${branchName}" into "${mergeTo}" failed. Please resolve conflicts manually.`,
|
||||||
|
hasConflicts: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Re-throw non-conflict errors
|
||||||
|
throw mergeError;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If squash merge, need to commit
|
||||||
|
if (options?.squash) {
|
||||||
|
await execAsync(`git commit -m "${options?.message || `Merge ${branchName} (squash)`}"`, {
|
||||||
|
cwd: projectPath,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Optionally delete the worktree and branch after merging
|
||||||
|
let worktreeDeleted = false;
|
||||||
|
let branchDeleted = false;
|
||||||
|
|
||||||
|
if (options?.deleteWorktreeAndBranch) {
|
||||||
|
// Remove the worktree
|
||||||
|
try {
|
||||||
|
await execGitCommand(['worktree', 'remove', worktreePath, '--force'], projectPath);
|
||||||
|
worktreeDeleted = true;
|
||||||
|
} catch {
|
||||||
|
// Try with prune if remove fails
|
||||||
|
try {
|
||||||
|
await execGitCommand(['worktree', 'prune'], projectPath);
|
||||||
|
worktreeDeleted = true;
|
||||||
|
} catch {
|
||||||
|
logger.warn(`Failed to remove worktree: ${worktreePath}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the branch (but not main/master)
|
||||||
|
if (branchName !== 'main' && branchName !== 'master') {
|
||||||
|
if (!isValidBranchName(branchName)) {
|
||||||
|
logger.warn(`Invalid branch name detected, skipping deletion: ${branchName}`);
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
await execGitCommand(['branch', '-D', branchName], projectPath);
|
||||||
|
branchDeleted = true;
|
||||||
|
} catch {
|
||||||
|
logger.warn(`Failed to delete branch: ${branchName}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
mergedBranch: branchName,
|
||||||
|
targetBranch: mergeTo,
|
||||||
|
deleted: options?.deleteWorktreeAndBranch ? { worktreeDeleted, branchDeleted } : undefined,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -27,6 +27,7 @@ import type { SettingsService } from './settings-service.js';
|
|||||||
import type { ConcurrencyManager } from './concurrency-manager.js';
|
import type { ConcurrencyManager } from './concurrency-manager.js';
|
||||||
import { pipelineService } from './pipeline-service.js';
|
import { pipelineService } from './pipeline-service.js';
|
||||||
import type { TestRunnerService, TestRunStatus } from './test-runner-service.js';
|
import type { TestRunnerService, TestRunStatus } from './test-runner-service.js';
|
||||||
|
import { performMerge } from './merge-service.js';
|
||||||
import type {
|
import type {
|
||||||
PipelineContext,
|
PipelineContext,
|
||||||
PipelineStatusInfo,
|
PipelineStatusInfo,
|
||||||
@@ -65,8 +66,7 @@ export class PipelineOrchestrator {
|
|||||||
private loadContextFilesFn: typeof loadContextFiles,
|
private loadContextFilesFn: typeof loadContextFiles,
|
||||||
private buildFeaturePromptFn: BuildFeaturePromptFn,
|
private buildFeaturePromptFn: BuildFeaturePromptFn,
|
||||||
private executeFeatureFn: ExecuteFeatureFn,
|
private executeFeatureFn: ExecuteFeatureFn,
|
||||||
private runAgentFn: RunAgentFn,
|
private runAgentFn: RunAgentFn
|
||||||
private serverPort = 3008
|
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async executePipeline(ctx: PipelineContext): Promise<void> {
|
async executePipeline(ctx: PipelineContext): Promise<void> {
|
||||||
@@ -483,37 +483,19 @@ export class PipelineOrchestrator {
|
|||||||
|
|
||||||
logger.info(`Attempting auto-merge for feature ${featureId} (branch: ${branchName})`);
|
logger.info(`Attempting auto-merge for feature ${featureId} (branch: ${branchName})`);
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`http://localhost:${this.serverPort}/api/worktree/merge`, {
|
// Call merge service directly instead of HTTP fetch
|
||||||
method: 'POST',
|
const result = await performMerge(
|
||||||
headers: { 'Content-Type': 'application/json' },
|
projectPath,
|
||||||
body: JSON.stringify({
|
branchName,
|
||||||
projectPath,
|
worktreePath || projectPath,
|
||||||
branchName,
|
'main',
|
||||||
worktreePath,
|
{
|
||||||
targetBranch: 'main',
|
deleteWorktreeAndBranch: false,
|
||||||
options: { deleteWorktreeAndBranch: false },
|
}
|
||||||
}),
|
);
|
||||||
});
|
|
||||||
|
|
||||||
if (!response) {
|
if (!result.success) {
|
||||||
return { success: false, error: 'No response from merge endpoint' };
|
if (result.hasConflicts) {
|
||||||
}
|
|
||||||
|
|
||||||
// Defensively parse JSON response
|
|
||||||
let data: { success: boolean; hasConflicts?: boolean; error?: string };
|
|
||||||
try {
|
|
||||||
data = (await response.json()) as {
|
|
||||||
success: boolean;
|
|
||||||
hasConflicts?: boolean;
|
|
||||||
error?: string;
|
|
||||||
};
|
|
||||||
} catch (parseError) {
|
|
||||||
logger.error(`Failed to parse merge response:`, parseError);
|
|
||||||
return { success: false, error: 'Invalid response from merge endpoint' };
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
if (data.hasConflicts) {
|
|
||||||
await this.updateFeatureStatusFn(projectPath, featureId, 'merge_conflict');
|
await this.updateFeatureStatusFn(projectPath, featureId, 'merge_conflict');
|
||||||
this.eventBus.emitAutoModeEvent('pipeline_merge_conflict', {
|
this.eventBus.emitAutoModeEvent('pipeline_merge_conflict', {
|
||||||
featureId,
|
featureId,
|
||||||
@@ -522,7 +504,7 @@ export class PipelineOrchestrator {
|
|||||||
});
|
});
|
||||||
return { success: false, hasConflicts: true, needsAgentResolution: true };
|
return { success: false, hasConflicts: true, needsAgentResolution: true };
|
||||||
}
|
}
|
||||||
return { success: false, error: data.error };
|
return { success: false, error: result.error };
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`Auto-merge successful for feature ${featureId}`);
|
logger.info(`Auto-merge successful for feature ${featureId}`);
|
||||||
|
|||||||
@@ -1,5 +1,11 @@
|
|||||||
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
||||||
|
import path from 'path';
|
||||||
import type { Feature } from '@automaker/types';
|
import type { Feature } from '@automaker/types';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper to normalize paths for cross-platform test compatibility.
|
||||||
|
*/
|
||||||
|
const normalizePath = (p: string): string => path.resolve(p);
|
||||||
import {
|
import {
|
||||||
ExecutionService,
|
ExecutionService,
|
||||||
type RunAgentFn,
|
type RunAgentFn,
|
||||||
@@ -931,8 +937,8 @@ describe('execution-service.ts', () => {
|
|||||||
// Should still run agent, just with project path
|
// Should still run agent, just with project path
|
||||||
expect(mockRunAgentFn).toHaveBeenCalled();
|
expect(mockRunAgentFn).toHaveBeenCalled();
|
||||||
const callArgs = mockRunAgentFn.mock.calls[0];
|
const callArgs = mockRunAgentFn.mock.calls[0];
|
||||||
// First argument is workDir - should end with /test/project
|
// First argument is workDir - should be normalized path to /test/project
|
||||||
expect(callArgs[0]).toMatch(/\/test\/project$/);
|
expect(callArgs[0]).toBe(normalizePath('/test/project'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('skips worktree resolution when useWorktrees is false', async () => {
|
it('skips worktree resolution when useWorktrees is false', async () => {
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { describe, it, expect, beforeEach, vi, type Mock } from 'vitest';
|
import { describe, it, expect, beforeEach, vi, type Mock } from 'vitest';
|
||||||
|
import path from 'path';
|
||||||
import { FeatureStateManager } from '@/services/feature-state-manager.js';
|
import { FeatureStateManager } from '@/services/feature-state-manager.js';
|
||||||
import type { Feature } from '@automaker/types';
|
import type { Feature } from '@automaker/types';
|
||||||
import type { EventEmitter } from '@/lib/events.js';
|
import type { EventEmitter } from '@/lib/events.js';
|
||||||
@@ -8,6 +9,12 @@ import { atomicWriteJson, readJsonWithRecovery } from '@automaker/utils';
|
|||||||
import { getFeatureDir, getFeaturesDir } from '@automaker/platform';
|
import { getFeatureDir, getFeaturesDir } from '@automaker/platform';
|
||||||
import { getNotificationService } from '@/services/notification-service.js';
|
import { getNotificationService } from '@/services/notification-service.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper to normalize paths for cross-platform test compatibility.
|
||||||
|
* Uses path.normalize (not path.resolve) to match path.join behavior in production code.
|
||||||
|
*/
|
||||||
|
const normalizePath = (p: string): string => path.normalize(p);
|
||||||
|
|
||||||
// Mock dependencies
|
// Mock dependencies
|
||||||
vi.mock('@/lib/secure-fs.js', () => ({
|
vi.mock('@/lib/secure-fs.js', () => ({
|
||||||
readFile: vi.fn(),
|
readFile: vi.fn(),
|
||||||
@@ -78,7 +85,7 @@ describe('FeatureStateManager', () => {
|
|||||||
expect(feature).toEqual(mockFeature);
|
expect(feature).toEqual(mockFeature);
|
||||||
expect(getFeatureDir).toHaveBeenCalledWith('/project', 'feature-123');
|
expect(getFeatureDir).toHaveBeenCalledWith('/project', 'feature-123');
|
||||||
expect(readJsonWithRecovery).toHaveBeenCalledWith(
|
expect(readJsonWithRecovery).toHaveBeenCalledWith(
|
||||||
'/project/.automaker/features/feature-123/feature.json',
|
normalizePath('/project/.automaker/features/feature-123/feature.json'),
|
||||||
null,
|
null,
|
||||||
expect.objectContaining({ autoRestore: true })
|
expect.objectContaining({ autoRestore: true })
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -35,6 +35,13 @@ vi.mock('../../../src/services/pipeline-service.js', () => ({
|
|||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
// Mock merge-service
|
||||||
|
vi.mock('../../../src/services/merge-service.js', () => ({
|
||||||
|
performMerge: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
import { performMerge } from '../../../src/services/merge-service.js';
|
||||||
|
|
||||||
// Mock secureFs
|
// Mock secureFs
|
||||||
vi.mock('../../../src/lib/secure-fs.js', () => ({
|
vi.mock('../../../src/lib/secure-fs.js', () => ({
|
||||||
readFile: vi.fn(),
|
readFile: vi.fn(),
|
||||||
@@ -470,36 +477,26 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
global.fetch = vi.fn();
|
vi.mocked(performMerge).mockReset();
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
it('should call performMerge with correct parameters', async () => {
|
||||||
vi.mocked(global.fetch).mockReset();
|
vi.mocked(performMerge).mockResolvedValue({ success: true });
|
||||||
});
|
|
||||||
|
|
||||||
it('should call merge endpoint with correct parameters', async () => {
|
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
|
||||||
ok: true,
|
|
||||||
json: vi.fn().mockResolvedValue({ success: true }),
|
|
||||||
} as never);
|
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
await orchestrator.attemptMerge(context);
|
await orchestrator.attemptMerge(context);
|
||||||
|
|
||||||
expect(global.fetch).toHaveBeenCalledWith(
|
expect(performMerge).toHaveBeenCalledWith(
|
||||||
expect.stringContaining('/api/worktree/merge'),
|
'/test/project',
|
||||||
expect.objectContaining({
|
'feature/test-1',
|
||||||
method: 'POST',
|
'/test/worktree',
|
||||||
body: expect.stringContaining('feature/test-1'),
|
'main',
|
||||||
})
|
{ deleteWorktreeAndBranch: false }
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return success on clean merge', async () => {
|
it('should return success on clean merge', async () => {
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({ success: true });
|
||||||
ok: true,
|
|
||||||
json: vi.fn().mockResolvedValue({ success: true }),
|
|
||||||
} as never);
|
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
const result = await orchestrator.attemptMerge(context);
|
const result = await orchestrator.attemptMerge(context);
|
||||||
@@ -509,10 +506,11 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should set merge_conflict status when hasConflicts is true', async () => {
|
it('should set merge_conflict status when hasConflicts is true', async () => {
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({
|
||||||
ok: false,
|
success: false,
|
||||||
json: vi.fn().mockResolvedValue({ success: false, hasConflicts: true }),
|
hasConflicts: true,
|
||||||
} as never);
|
error: 'Merge conflict',
|
||||||
|
});
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
await orchestrator.attemptMerge(context);
|
await orchestrator.attemptMerge(context);
|
||||||
@@ -525,10 +523,11 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should emit pipeline_merge_conflict event on conflict', async () => {
|
it('should emit pipeline_merge_conflict event on conflict', async () => {
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({
|
||||||
ok: false,
|
success: false,
|
||||||
json: vi.fn().mockResolvedValue({ success: false, hasConflicts: true }),
|
hasConflicts: true,
|
||||||
} as never);
|
error: 'Merge conflict',
|
||||||
|
});
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
await orchestrator.attemptMerge(context);
|
await orchestrator.attemptMerge(context);
|
||||||
@@ -540,10 +539,7 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should emit auto_mode_feature_complete on success', async () => {
|
it('should emit auto_mode_feature_complete on success', async () => {
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({ success: true });
|
||||||
ok: true,
|
|
||||||
json: vi.fn().mockResolvedValue({ success: true }),
|
|
||||||
} as never);
|
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
await orchestrator.attemptMerge(context);
|
await orchestrator.attemptMerge(context);
|
||||||
@@ -555,10 +551,11 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should return needsAgentResolution true on conflict', async () => {
|
it('should return needsAgentResolution true on conflict', async () => {
|
||||||
vi.mocked(global.fetch).mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({
|
||||||
ok: false,
|
success: false,
|
||||||
json: vi.fn().mockResolvedValue({ success: false, hasConflicts: true }),
|
hasConflicts: true,
|
||||||
} as never);
|
error: 'Merge conflict',
|
||||||
|
});
|
||||||
|
|
||||||
const context = createMergeContext();
|
const context = createMergeContext();
|
||||||
const result = await orchestrator.attemptMerge(context);
|
const result = await orchestrator.attemptMerge(context);
|
||||||
@@ -728,10 +725,7 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
global.fetch = vi.fn().mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({ success: true });
|
||||||
ok: true,
|
|
||||||
json: vi.fn().mockResolvedValue({ success: true }),
|
|
||||||
} as never);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should execute steps in sequence', async () => {
|
it('should execute steps in sequence', async () => {
|
||||||
@@ -792,9 +786,12 @@ describe('PipelineOrchestrator', () => {
|
|||||||
const context = createPipelineContext();
|
const context = createPipelineContext();
|
||||||
await orchestrator.executePipeline(context);
|
await orchestrator.executePipeline(context);
|
||||||
|
|
||||||
expect(global.fetch).toHaveBeenCalledWith(
|
expect(performMerge).toHaveBeenCalledWith(
|
||||||
expect.stringContaining('/api/worktree/merge'),
|
'/test/project',
|
||||||
expect.any(Object)
|
'feature/test-1',
|
||||||
|
'/test/project', // Falls back to projectPath when worktreePath is null
|
||||||
|
'main',
|
||||||
|
{ deleteWorktreeAndBranch: false }
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -816,10 +813,7 @@ describe('PipelineOrchestrator', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
global.fetch = vi.fn().mockResolvedValue({
|
vi.mocked(performMerge).mockResolvedValue({ success: true });
|
||||||
ok: true,
|
|
||||||
json: vi.fn().mockResolvedValue({ success: true }),
|
|
||||||
} as never);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('builds PipelineContext with correct fields from executeFeature', async () => {
|
it('builds PipelineContext with correct fields from executeFeature', async () => {
|
||||||
@@ -845,11 +839,12 @@ describe('PipelineOrchestrator', () => {
|
|||||||
await orchestrator.executePipeline(context);
|
await orchestrator.executePipeline(context);
|
||||||
|
|
||||||
// Merge should receive the worktree path
|
// Merge should receive the worktree path
|
||||||
expect(global.fetch).toHaveBeenCalledWith(
|
expect(performMerge).toHaveBeenCalledWith(
|
||||||
expect.stringContaining('/api/worktree/merge'),
|
'/test/project',
|
||||||
expect.objectContaining({
|
'feature/test-1',
|
||||||
body: expect.stringContaining('/test/custom-worktree'),
|
'/test/custom-worktree',
|
||||||
})
|
'main',
|
||||||
|
{ deleteWorktreeAndBranch: false }
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -860,11 +855,12 @@ describe('PipelineOrchestrator', () => {
|
|||||||
|
|
||||||
await orchestrator.executePipeline(context);
|
await orchestrator.executePipeline(context);
|
||||||
|
|
||||||
expect(global.fetch).toHaveBeenCalledWith(
|
expect(performMerge).toHaveBeenCalledWith(
|
||||||
expect.stringContaining('/api/worktree/merge'),
|
'/test/project',
|
||||||
expect.objectContaining({
|
'feature/custom-branch',
|
||||||
body: expect.stringContaining('feature/custom-branch'),
|
'/test/worktree',
|
||||||
})
|
'main',
|
||||||
|
{ deleteWorktreeAndBranch: false }
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -54,6 +54,8 @@ describe('PlanApprovalService', () => {
|
|||||||
|
|
||||||
it('should timeout and reject after configured period', async () => {
|
it('should timeout and reject after configured period', async () => {
|
||||||
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
// Flush the async initialization
|
// Flush the async initialization
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
@@ -73,6 +75,8 @@ describe('PlanApprovalService', () => {
|
|||||||
} as never);
|
} as never);
|
||||||
|
|
||||||
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
// Flush the async initialization
|
// Flush the async initialization
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
@@ -93,6 +97,8 @@ describe('PlanApprovalService', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const approvalPromise = serviceNoSettings.waitForApproval('feature-1', '/project');
|
const approvalPromise = serviceNoSettings.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
// Flush async
|
// Flush async
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
@@ -417,6 +423,8 @@ describe('PlanApprovalService', () => {
|
|||||||
} as never);
|
} as never);
|
||||||
|
|
||||||
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
// Should not timeout at 4 minutes
|
// Should not timeout at 4 minutes
|
||||||
@@ -432,6 +440,8 @@ describe('PlanApprovalService', () => {
|
|||||||
vi.mocked(mockSettingsService!.getProjectSettings).mockRejectedValue(new Error('Failed'));
|
vi.mocked(mockSettingsService!.getProjectSettings).mockRejectedValue(new Error('Failed'));
|
||||||
|
|
||||||
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
// Should use default 30 minute timeout
|
// Should use default 30 minute timeout
|
||||||
@@ -448,6 +458,8 @@ describe('PlanApprovalService', () => {
|
|||||||
} as never);
|
} as never);
|
||||||
|
|
||||||
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
const approvalPromise = service.waitForApproval('feature-1', '/project');
|
||||||
|
// Attach catch to prevent unhandled rejection warning (will be properly asserted below)
|
||||||
|
approvalPromise.catch(() => {});
|
||||||
await vi.advanceTimersByTimeAsync(0);
|
await vi.advanceTimersByTimeAsync(0);
|
||||||
|
|
||||||
// Should use default 30 minute timeout
|
// Should use default 30 minute timeout
|
||||||
|
|||||||
@@ -9,9 +9,16 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
||||||
|
import path from 'path';
|
||||||
import { RecoveryService, DEFAULT_EXECUTION_STATE } from '@/services/recovery-service.js';
|
import { RecoveryService, DEFAULT_EXECUTION_STATE } from '@/services/recovery-service.js';
|
||||||
import type { Feature } from '@automaker/types';
|
import type { Feature } from '@automaker/types';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper to normalize paths for cross-platform test compatibility.
|
||||||
|
* Uses path.normalize (not path.resolve) to match path.join behavior in production code.
|
||||||
|
*/
|
||||||
|
const normalizePath = (p: string): string => path.normalize(p);
|
||||||
|
|
||||||
// Mock dependencies
|
// Mock dependencies
|
||||||
vi.mock('@automaker/utils', () => ({
|
vi.mock('@automaker/utils', () => ({
|
||||||
createLogger: () => ({
|
createLogger: () => ({
|
||||||
@@ -288,7 +295,7 @@ describe('recovery-service.ts', () => {
|
|||||||
|
|
||||||
expect(result).toBe(true);
|
expect(result).toBe(true);
|
||||||
expect(secureFs.access).toHaveBeenCalledWith(
|
expect(secureFs.access).toHaveBeenCalledWith(
|
||||||
'/test/project/.automaker/features/feature-1/agent-output.md'
|
normalizePath('/test/project/.automaker/features/feature-1/agent-output.md')
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,20 @@
|
|||||||
import { describe, it, expect, beforeEach, vi, type Mock } from 'vitest';
|
import { describe, it, expect, beforeEach, vi, type Mock } from 'vitest';
|
||||||
import { WorktreeResolver, type WorktreeInfo } from '@/services/worktree-resolver.js';
|
import { WorktreeResolver, type WorktreeInfo } from '@/services/worktree-resolver.js';
|
||||||
import { exec } from 'child_process';
|
import { exec } from 'child_process';
|
||||||
|
import path from 'path';
|
||||||
|
|
||||||
// Mock child_process
|
// Mock child_process
|
||||||
vi.mock('child_process', () => ({
|
vi.mock('child_process', () => ({
|
||||||
exec: vi.fn(),
|
exec: vi.fn(),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper to normalize paths for cross-platform test compatibility.
|
||||||
|
* On Windows, path.resolve('/Users/dev/project') returns 'C:\Users\dev\project' (with current drive).
|
||||||
|
* This helper ensures test expectations match the actual platform behavior.
|
||||||
|
*/
|
||||||
|
const normalizePath = (p: string): string => path.resolve(p);
|
||||||
|
|
||||||
// Create promisified mock helper
|
// Create promisified mock helper
|
||||||
const mockExecAsync = (
|
const mockExecAsync = (
|
||||||
impl: (cmd: string, options?: { cwd?: string }) => Promise<{ stdout: string; stderr: string }>
|
impl: (cmd: string, options?: { cwd?: string }) => Promise<{ stdout: string; stderr: string }>
|
||||||
@@ -94,9 +102,9 @@ branch refs/heads/feature-y
|
|||||||
it('should find worktree by branch name', async () => {
|
it('should find worktree by branch name', async () => {
|
||||||
mockExecAsync(async () => ({ stdout: porcelainOutput, stderr: '' }));
|
mockExecAsync(async () => ({ stdout: porcelainOutput, stderr: '' }));
|
||||||
|
|
||||||
const path = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-x');
|
const result = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-x');
|
||||||
|
|
||||||
expect(path).toBe('/Users/dev/project/.worktrees/feature-x');
|
expect(result).toBe(normalizePath('/Users/dev/project/.worktrees/feature-x'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return null when branch not found', async () => {
|
it('should return null when branch not found', async () => {
|
||||||
@@ -120,9 +128,9 @@ branch refs/heads/feature-y
|
|||||||
it('should find main worktree', async () => {
|
it('should find main worktree', async () => {
|
||||||
mockExecAsync(async () => ({ stdout: porcelainOutput, stderr: '' }));
|
mockExecAsync(async () => ({ stdout: porcelainOutput, stderr: '' }));
|
||||||
|
|
||||||
const path = await resolver.findWorktreeForBranch('/Users/dev/project', 'main');
|
const result = await resolver.findWorktreeForBranch('/Users/dev/project', 'main');
|
||||||
|
|
||||||
expect(path).toBe('/Users/dev/project');
|
expect(result).toBe(normalizePath('/Users/dev/project'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should handle porcelain output without trailing newline', async () => {
|
it('should handle porcelain output without trailing newline', async () => {
|
||||||
@@ -134,9 +142,9 @@ branch refs/heads/feature-x`;
|
|||||||
|
|
||||||
mockExecAsync(async () => ({ stdout: noTrailingNewline, stderr: '' }));
|
mockExecAsync(async () => ({ stdout: noTrailingNewline, stderr: '' }));
|
||||||
|
|
||||||
const path = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-x');
|
const result = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-x');
|
||||||
|
|
||||||
expect(path).toBe('/Users/dev/project/.worktrees/feature-x');
|
expect(result).toBe(normalizePath('/Users/dev/project/.worktrees/feature-x'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should resolve relative paths to absolute', async () => {
|
it('should resolve relative paths to absolute', async () => {
|
||||||
@@ -151,8 +159,8 @@ branch refs/heads/feature-relative
|
|||||||
|
|
||||||
const result = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-relative');
|
const result = await resolver.findWorktreeForBranch('/Users/dev/project', 'feature-relative');
|
||||||
|
|
||||||
// Should resolve to absolute path
|
// Should resolve to absolute path (platform-specific)
|
||||||
expect(result).toBe('/Users/dev/project/.worktrees/feature-relative');
|
expect(result).toBe(normalizePath('/Users/dev/project/.worktrees/feature-relative'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use projectPath as cwd for git command', async () => {
|
it('should use projectPath as cwd for git command', async () => {
|
||||||
@@ -186,17 +194,17 @@ branch refs/heads/feature-y
|
|||||||
|
|
||||||
expect(worktrees).toHaveLength(3);
|
expect(worktrees).toHaveLength(3);
|
||||||
expect(worktrees[0]).toEqual({
|
expect(worktrees[0]).toEqual({
|
||||||
path: '/Users/dev/project',
|
path: normalizePath('/Users/dev/project'),
|
||||||
branch: 'main',
|
branch: 'main',
|
||||||
isMain: true,
|
isMain: true,
|
||||||
});
|
});
|
||||||
expect(worktrees[1]).toEqual({
|
expect(worktrees[1]).toEqual({
|
||||||
path: '/Users/dev/project/.worktrees/feature-x',
|
path: normalizePath('/Users/dev/project/.worktrees/feature-x'),
|
||||||
branch: 'feature-x',
|
branch: 'feature-x',
|
||||||
isMain: false,
|
isMain: false,
|
||||||
});
|
});
|
||||||
expect(worktrees[2]).toEqual({
|
expect(worktrees[2]).toEqual({
|
||||||
path: '/Users/dev/project/.worktrees/feature-y',
|
path: normalizePath('/Users/dev/project/.worktrees/feature-y'),
|
||||||
branch: 'feature-y',
|
branch: 'feature-y',
|
||||||
isMain: false,
|
isMain: false,
|
||||||
});
|
});
|
||||||
@@ -226,7 +234,7 @@ detached
|
|||||||
|
|
||||||
expect(worktrees).toHaveLength(2);
|
expect(worktrees).toHaveLength(2);
|
||||||
expect(worktrees[1]).toEqual({
|
expect(worktrees[1]).toEqual({
|
||||||
path: '/Users/dev/project/.worktrees/detached-wt',
|
path: normalizePath('/Users/dev/project/.worktrees/detached-wt'),
|
||||||
branch: null, // Detached HEAD has no branch
|
branch: null, // Detached HEAD has no branch
|
||||||
isMain: false,
|
isMain: false,
|
||||||
});
|
});
|
||||||
@@ -264,7 +272,7 @@ branch refs/heads/relative-branch
|
|||||||
|
|
||||||
const worktrees = await resolver.listWorktrees('/Users/dev/project');
|
const worktrees = await resolver.listWorktrees('/Users/dev/project');
|
||||||
|
|
||||||
expect(worktrees[1].path).toBe('/Users/dev/project/.worktrees/relative-wt');
|
expect(worktrees[1].path).toBe(normalizePath('/Users/dev/project/.worktrees/relative-wt'));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should handle single worktree (main only)', async () => {
|
it('should handle single worktree (main only)', async () => {
|
||||||
@@ -278,7 +286,7 @@ branch refs/heads/main
|
|||||||
|
|
||||||
expect(worktrees).toHaveLength(1);
|
expect(worktrees).toHaveLength(1);
|
||||||
expect(worktrees[0]).toEqual({
|
expect(worktrees[0]).toEqual({
|
||||||
path: '/Users/dev/project',
|
path: normalizePath('/Users/dev/project'),
|
||||||
branch: 'main',
|
branch: 'main',
|
||||||
isMain: true,
|
isMain: true,
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user