takt/e2e/specs/provider-error.e2e.ts

113 lines
3.6 KiB
TypeScript

import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { resolve, dirname } from 'node:path';
import { fileURLToPath } from 'node:url';
import {
createIsolatedEnv,
updateIsolatedConfig,
type IsolatedEnv,
} from '../helpers/isolated-env';
import { runTakt } from '../helpers/takt-runner';
import { createLocalRepo, type LocalRepo } from '../helpers/test-repo';
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
// E2E更新時は docs/testing/e2e.md も更新すること
describe('E2E: Provider error handling (mock)', () => {
let isolatedEnv: IsolatedEnv;
let repo: LocalRepo;
beforeEach(() => {
isolatedEnv = createIsolatedEnv();
repo = createLocalRepo();
});
afterEach(() => {
try { repo.cleanup(); } catch { /* best-effort */ }
try { isolatedEnv.cleanup(); } catch { /* best-effort */ }
});
it('should override config provider with --provider flag', () => {
// Given: config.yaml has provider: claude, but CLI flag specifies mock
updateIsolatedConfig(isolatedEnv.taktDir, {
provider: 'claude',
});
const piecePath = resolve(__dirname, '../fixtures/pieces/mock-single-step.yaml');
const scenarioPath = resolve(__dirname, '../fixtures/scenarios/execute-done.json');
// When: running with --provider mock
const result = runTakt({
args: [
'--task', 'Test provider override',
'--piece', piecePath,
'--create-worktree', 'no',
'--provider', 'mock',
],
cwd: repo.path,
env: {
...isolatedEnv.env,
TAKT_MOCK_SCENARIO: scenarioPath,
},
timeout: 240_000,
});
// Then: executes successfully with mock provider
expect(result.exitCode).toBe(0);
expect(result.stdout).toContain('Piece completed');
}, 240_000);
it('should use default mock response when scenario entries are exhausted', () => {
// Given: a two-step piece with only 1 scenario entry
const piecePath = resolve(__dirname, '../fixtures/pieces/mock-two-step.yaml');
const scenarioPath = resolve(__dirname, '../fixtures/scenarios/one-entry-only.json');
// When: executing the piece (step-2 will have no scenario entry)
const result = runTakt({
args: [
'--task', 'Test scenario exhaustion',
'--piece', piecePath,
'--create-worktree', 'no',
'--provider', 'mock',
],
cwd: repo.path,
env: {
...isolatedEnv.env,
TAKT_MOCK_SCENARIO: scenarioPath,
},
timeout: 240_000,
});
// Then: does not crash; either completes or aborts gracefully
const combined = result.stdout + result.stderr;
expect(combined).not.toContain('UnhandledPromiseRejection');
expect(combined).not.toContain('SIGTERM');
}, 240_000);
it('should error when scenario file does not exist', () => {
// Given: TAKT_MOCK_SCENARIO pointing to a non-existent file
const piecePath = resolve(__dirname, '../fixtures/pieces/mock-single-step.yaml');
// When: executing with a bad scenario path
const result = runTakt({
args: [
'--task', 'Test bad scenario',
'--piece', piecePath,
'--create-worktree', 'no',
'--provider', 'mock',
],
cwd: repo.path,
env: {
...isolatedEnv.env,
TAKT_MOCK_SCENARIO: '/nonexistent/path/scenario.json',
},
timeout: 240_000,
});
// Then: exits with error and clear message
expect(result.exitCode).not.toBe(0);
const combined = result.stdout + result.stderr;
expect(combined).toMatch(/[Ss]cenario file not found|ENOENT/);
}, 240_000);
});