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'; import { readSessionRecords } from '../helpers/session-log'; const __filename = fileURLToPath(import.meta.url); const __dirname = dirname(__filename); // E2E更新時は docs/testing/e2e.md も更新すること describe('E2E: Session NDJSON log output (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 write piece_start, step_complete, and piece_complete on success', () => { const piecePath = resolve(__dirname, '../fixtures/pieces/mock-single-step.yaml'); const scenarioPath = resolve(__dirname, '../fixtures/scenarios/execute-done.json'); const result = runTakt({ args: [ '--task', 'Test session log success', '--piece', piecePath, '--provider', 'mock', ], cwd: repo.path, env: { ...isolatedEnv.env, TAKT_MOCK_SCENARIO: scenarioPath, }, timeout: 240_000, }); expect(result.exitCode).toBe(0); const records = readSessionRecords(repo.path); expect(records.some((r) => r.type === 'piece_start')).toBe(true); expect(records.some((r) => r.type === 'step_complete')).toBe(true); expect(records.some((r) => r.type === 'piece_complete')).toBe(true); }, 240_000); it('should write piece_abort with reason on failure', () => { const piecePath = resolve(__dirname, '../fixtures/pieces/mock-no-match.yaml'); const scenarioPath = resolve(__dirname, '../fixtures/scenarios/no-match.json'); const result = runTakt({ args: [ '--task', 'Test session log abort', '--piece', piecePath, '--provider', 'mock', ], cwd: repo.path, env: { ...isolatedEnv.env, TAKT_MOCK_SCENARIO: scenarioPath, }, timeout: 240_000, }); expect(result.exitCode).not.toBe(0); const records = readSessionRecords(repo.path); const abortRecord = records.find((r) => r.type === 'piece_abort'); expect(abortRecord).toBeDefined(); expect(typeof abortRecord?.reason).toBe('string'); expect((abortRecord?.reason as string).length).toBeGreaterThan(0); }, 240_000); });