import { describe, it, expect, beforeEach, afterEach } from 'vitest'; import { resolve, dirname } from 'node:path'; import { fileURLToPath } from 'node:url'; import { createIsolatedEnv, 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: Error handling edge cases (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 error when --piece points to a nonexistent file path', () => { // Given: a nonexistent piece file path // When: running with a bad piece path const result = runTakt({ args: [ '--task', 'test', '--piece', '/nonexistent/path/to/piece.yaml', '--provider', 'mock', ], cwd: repo.path, env: isolatedEnv.env, timeout: 240_000, }); // Then: exits with error expect(result.exitCode).not.toBe(0); const combined = result.stdout + result.stderr; expect(combined).toMatch(/not found|does not exist|ENOENT/i); }, 240_000); it('should report error when --piece specifies a nonexistent piece name', () => { // Given: a nonexistent piece name // When: running with a bad piece name const result = runTakt({ args: [ '--task', 'test', '--piece', 'nonexistent-piece-name-xyz', '--provider', 'mock', ], cwd: repo.path, env: isolatedEnv.env, timeout: 240_000, }); // Then: output contains error about piece not found // Note: takt reports the error but currently exits with code 0 const combined = result.stdout + result.stderr; expect(combined).toMatch(/not found/i); }, 240_000); it('should error when --pipeline is used without --task or --issue', () => { // Given: pipeline mode with no task or issue const piecePath = resolve(__dirname, '../fixtures/pieces/mock-single-step.yaml'); // When: running in pipeline mode without a task const result = runTakt({ args: [ '--pipeline', '--piece', piecePath, '--skip-git', '--provider', 'mock', ], cwd: repo.path, env: isolatedEnv.env, timeout: 240_000, }); // Then: exits with error (should not hang in interactive mode due to TAKT_NO_TTY=1) expect(result.exitCode).not.toBe(0); const combined = result.stdout + result.stderr; expect(combined).toMatch(/task|issue|required/i); }, 240_000); it('should error when deprecated --create-worktree option is used', () => { // Given: deprecated option value const piecePath = resolve(__dirname, '../fixtures/pieces/mock-single-step.yaml'); // When: running with invalid worktree option const result = runTakt({ args: [ '--task', 'test', '--piece', piecePath, '--create-worktree', 'invalid-value', '--provider', 'mock', ], cwd: repo.path, env: isolatedEnv.env, timeout: 240_000, }); // Then: exits with migration error const combined = result.stdout + result.stderr; expect(result.exitCode).not.toBe(0); expect(combined).toContain('--create-worktree has been removed'); }, 240_000); it('should error when piece file contains invalid YAML', () => { // Given: a broken YAML piece file const brokenPiecePath = resolve(__dirname, '../fixtures/pieces/broken.yaml'); // When: running with the broken piece const result = runTakt({ args: [ '--task', 'test', '--piece', brokenPiecePath, '--provider', 'mock', ], cwd: repo.path, env: isolatedEnv.env, timeout: 240_000, }); // Then: exits with error about parsing expect(result.exitCode).not.toBe(0); const combined = result.stdout + result.stderr; expect(combined).toMatch(/parse|invalid|error|validation/i); }, 240_000); });