takt/e2e/specs/eject.e2e.ts
2026-02-07 22:59:19 +09:00

188 lines
5.9 KiB
TypeScript

import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { existsSync, readFileSync, mkdirSync, writeFileSync, mkdtempSync, rmSync } from 'node:fs';
import { join } from 'node:path';
import { tmpdir } from 'node:os';
import { execFileSync } from 'node:child_process';
import { createIsolatedEnv, type IsolatedEnv } from '../helpers/isolated-env';
import { runTakt } from '../helpers/takt-runner';
/**
* Create a minimal local git repository for eject tests.
* No GitHub access needed — just a local git init.
*/
function createLocalRepo(): { path: string; cleanup: () => void } {
const repoPath = mkdtempSync(join(tmpdir(), 'takt-eject-e2e-'));
execFileSync('git', ['init'], { cwd: repoPath, stdio: 'pipe' });
execFileSync('git', ['config', 'user.email', 'test@example.com'], { cwd: repoPath, stdio: 'pipe' });
execFileSync('git', ['config', 'user.name', 'Test'], { cwd: repoPath, stdio: 'pipe' });
// Create initial commit so branch exists
writeFileSync(join(repoPath, 'README.md'), '# test\n');
execFileSync('git', ['add', '.'], { cwd: repoPath, stdio: 'pipe' });
execFileSync('git', ['commit', '-m', 'init'], { cwd: repoPath, stdio: 'pipe' });
return {
path: repoPath,
cleanup: () => {
try {
rmSync(repoPath, { recursive: true, force: true });
} catch {
// best-effort
}
},
};
}
// E2E更新時は docs/testing/e2e.md も更新すること
describe('E2E: Eject builtin pieces (takt eject)', () => {
let isolatedEnv: IsolatedEnv;
let repo: { path: string; cleanup: () => void };
beforeEach(() => {
isolatedEnv = createIsolatedEnv();
repo = createLocalRepo();
});
afterEach(() => {
try {
repo.cleanup();
} catch {
// best-effort
}
try {
isolatedEnv.cleanup();
} catch {
// best-effort
}
});
it('should list available builtin pieces when no name given', () => {
const result = runTakt({
args: ['eject'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
expect(result.stdout).toContain('default');
expect(result.stdout).toContain('Available builtin pieces');
});
it('should eject piece to project .takt/ by default', () => {
const result = runTakt({
args: ['eject', 'default'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
// Piece YAML should be in project .takt/pieces/
const piecePath = join(repo.path, '.takt', 'pieces', 'default.yaml');
expect(existsSync(piecePath)).toBe(true);
// Personas should be in project .takt/personas/
const personasDir = join(repo.path, '.takt', 'personas');
expect(existsSync(personasDir)).toBe(true);
expect(existsSync(join(personasDir, 'coder.md'))).toBe(true);
expect(existsSync(join(personasDir, 'planner.md'))).toBe(true);
});
it('should preserve relative persona paths in ejected piece (no rewriting)', () => {
runTakt({
args: ['eject', 'default'],
cwd: repo.path,
env: isolatedEnv.env,
});
const piecePath = join(repo.path, '.takt', 'pieces', 'default.yaml');
const content = readFileSync(piecePath, 'utf-8');
// Relative paths should be preserved as ../personas/
expect(content).toContain('../personas/');
// Should NOT contain rewritten absolute paths
expect(content).not.toContain('~/.takt/personas/');
});
it('should eject piece to global ~/.takt/ with --global flag', () => {
const result = runTakt({
args: ['eject', 'default', '--global'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
// Piece YAML should be in global dir (TAKT_CONFIG_DIR from isolated env)
const piecePath = join(isolatedEnv.taktDir, 'pieces', 'default.yaml');
expect(existsSync(piecePath)).toBe(true);
// Personas should be in global personas dir
const personasDir = join(isolatedEnv.taktDir, 'personas');
expect(existsSync(personasDir)).toBe(true);
expect(existsSync(join(personasDir, 'coder.md'))).toBe(true);
// Should NOT be in project dir
const projectPiecePath = join(repo.path, '.takt', 'pieces', 'default.yaml');
expect(existsSync(projectPiecePath)).toBe(false);
});
it('should warn and skip when piece already exists', () => {
// First eject
runTakt({
args: ['eject', 'default'],
cwd: repo.path,
env: isolatedEnv.env,
});
// Second eject — should skip
const result = runTakt({
args: ['eject', 'default'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
expect(result.stdout).toContain('already exists');
});
it('should report error for non-existent builtin', () => {
const result = runTakt({
args: ['eject', 'nonexistent-piece-xyz'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
expect(result.stdout).toContain('not found');
});
it('should correctly eject personas for pieces with unique personas', () => {
const result = runTakt({
args: ['eject', 'magi'],
cwd: repo.path,
env: isolatedEnv.env,
});
expect(result.exitCode).toBe(0);
// MAGI piece should have its personas ejected
const personasDir = join(repo.path, '.takt', 'personas');
expect(existsSync(join(personasDir, 'melchior.md'))).toBe(true);
expect(existsSync(join(personasDir, 'balthasar.md'))).toBe(true);
expect(existsSync(join(personasDir, 'casper.md'))).toBe(true);
});
it('should preserve relative paths for global eject too', () => {
runTakt({
args: ['eject', 'magi', '--global'],
cwd: repo.path,
env: isolatedEnv.env,
});
const piecePath = join(isolatedEnv.taktDir, 'pieces', 'magi.yaml');
const content = readFileSync(piecePath, 'utf-8');
expect(content).toContain('../personas/');
expect(content).not.toContain('~/.takt/personas/');
});
});