aiag-cli 2.2.2 → 2.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (77) hide show
  1. package/README.md +72 -37
  2. package/dist/cli.js +30 -2
  3. package/dist/cli.js.map +1 -1
  4. package/dist/commands/auto.js +45 -41
  5. package/dist/commands/auto.js.map +1 -1
  6. package/dist/commands/feature.d.ts +11 -0
  7. package/dist/commands/feature.d.ts.map +1 -0
  8. package/dist/commands/feature.js +153 -0
  9. package/dist/commands/feature.js.map +1 -0
  10. package/dist/commands/init.d.ts +1 -1
  11. package/dist/commands/init.d.ts.map +1 -1
  12. package/dist/commands/init.js +29 -78
  13. package/dist/commands/init.js.map +1 -1
  14. package/dist/commands/prd.d.ts +12 -0
  15. package/dist/commands/prd.d.ts.map +1 -0
  16. package/dist/commands/prd.js +179 -0
  17. package/dist/commands/prd.js.map +1 -0
  18. package/dist/prompts/coding.d.ts.map +1 -1
  19. package/dist/prompts/coding.js +12 -0
  20. package/dist/prompts/coding.js.map +1 -1
  21. package/dist/prompts/index.d.ts +2 -0
  22. package/dist/prompts/index.d.ts.map +1 -1
  23. package/dist/prompts/index.js +2 -0
  24. package/dist/prompts/index.js.map +1 -1
  25. package/dist/prompts/initializer.d.ts.map +1 -1
  26. package/dist/prompts/initializer.js +6 -0
  27. package/dist/prompts/initializer.js.map +1 -1
  28. package/dist/prompts/prd.d.ts +28 -0
  29. package/dist/prompts/prd.d.ts.map +1 -0
  30. package/dist/prompts/prd.js +105 -0
  31. package/dist/prompts/prd.js.map +1 -0
  32. package/dist/skills/index.d.ts +12 -0
  33. package/dist/skills/index.d.ts.map +1 -0
  34. package/dist/skills/index.js +12 -0
  35. package/dist/skills/index.js.map +1 -0
  36. package/dist/skills/installer.d.ts +38 -0
  37. package/dist/skills/installer.d.ts.map +1 -0
  38. package/dist/skills/installer.js +153 -0
  39. package/dist/skills/installer.js.map +1 -0
  40. package/dist/skills/loader.d.ts +34 -0
  41. package/dist/skills/loader.d.ts.map +1 -0
  42. package/dist/skills/loader.js +134 -0
  43. package/dist/skills/loader.js.map +1 -0
  44. package/dist/skills/runner.d.ts +14 -0
  45. package/dist/skills/runner.d.ts.map +1 -0
  46. package/dist/skills/runner.js +238 -0
  47. package/dist/skills/runner.js.map +1 -0
  48. package/dist/types.d.ts +127 -0
  49. package/dist/types.d.ts.map +1 -1
  50. package/dist/utils/prd.d.ts +21 -0
  51. package/dist/utils/prd.d.ts.map +1 -1
  52. package/dist/utils/prd.js +69 -0
  53. package/dist/utils/prd.js.map +1 -1
  54. package/dist/utils/taskmasterConverter.d.ts +72 -0
  55. package/dist/utils/taskmasterConverter.d.ts.map +1 -0
  56. package/dist/utils/taskmasterConverter.js +401 -0
  57. package/dist/utils/taskmasterConverter.js.map +1 -0
  58. package/dist/utils/taskmasterParser.d.ts +35 -0
  59. package/dist/utils/taskmasterParser.d.ts.map +1 -0
  60. package/dist/utils/taskmasterParser.js +259 -0
  61. package/dist/utils/taskmasterParser.js.map +1 -0
  62. package/package.json +1 -1
  63. package/templates/skills/prd-taskmaster/.taskmaster/docs/prd.md +2571 -0
  64. package/templates/skills/prd-taskmaster/.taskmaster/scripts/execution-state.py +87 -0
  65. package/templates/skills/prd-taskmaster/.taskmaster/scripts/learn-accuracy.py +113 -0
  66. package/templates/skills/prd-taskmaster/.taskmaster/scripts/rollback.sh +71 -0
  67. package/templates/skills/prd-taskmaster/.taskmaster/scripts/security-audit.py +130 -0
  68. package/templates/skills/prd-taskmaster/.taskmaster/scripts/track-time.py +133 -0
  69. package/templates/skills/prd-taskmaster/LICENSE +21 -0
  70. package/templates/skills/prd-taskmaster/README.md +608 -0
  71. package/templates/skills/prd-taskmaster/SKILL.md +1258 -0
  72. package/templates/skills/prd-taskmaster/reference/taskmaster-integration-guide.md +645 -0
  73. package/templates/skills/prd-taskmaster/reference/validation-checklist.md +394 -0
  74. package/templates/skills/prd-taskmaster/scripts/setup-taskmaster.sh +112 -0
  75. package/templates/skills/prd-taskmaster/templates/CLAUDE.md.template +635 -0
  76. package/templates/skills/prd-taskmaster/templates/taskmaster-prd-comprehensive.md +983 -0
  77. package/templates/skills/prd-taskmaster/templates/taskmaster-prd-minimal.md +103 -0
@@ -0,0 +1,153 @@
1
+ /**
2
+ * 스킬 설치 모듈
3
+ * 번들된 스킬을 ~/.claude/skills/ 디렉토리에 설치합니다.
4
+ */
5
+ import fs from 'fs';
6
+ import path from 'path';
7
+ import { fileURLToPath } from 'url';
8
+ import { SKILLS_DIR, skillExists, loadSkillMetadata } from './loader.js';
9
+ const __filename = fileURLToPath(import.meta.url);
10
+ const __dirname = path.dirname(__filename);
11
+ /**
12
+ * 번들된 스킬 디렉토리 경로
13
+ */
14
+ const BUNDLED_SKILLS_DIR = path.join(__dirname, '../../templates/skills');
15
+ /**
16
+ * 디렉토리 재귀 복사
17
+ */
18
+ function copyDirSync(src, dest) {
19
+ if (!fs.existsSync(dest)) {
20
+ fs.mkdirSync(dest, { recursive: true });
21
+ }
22
+ const entries = fs.readdirSync(src, { withFileTypes: true });
23
+ for (const entry of entries) {
24
+ const srcPath = path.join(src, entry.name);
25
+ const destPath = path.join(dest, entry.name);
26
+ if (entry.isDirectory()) {
27
+ copyDirSync(srcPath, destPath);
28
+ }
29
+ else {
30
+ fs.copyFileSync(srcPath, destPath);
31
+ }
32
+ }
33
+ }
34
+ /**
35
+ * 스킬 디렉토리 생성 (없으면)
36
+ */
37
+ export function ensureSkillsDir() {
38
+ if (!fs.existsSync(SKILLS_DIR)) {
39
+ fs.mkdirSync(SKILLS_DIR, { recursive: true });
40
+ }
41
+ }
42
+ /**
43
+ * 번들된 스킬 존재 여부 확인
44
+ */
45
+ export function bundledSkillExists(skillName) {
46
+ const bundledPath = path.join(BUNDLED_SKILLS_DIR, skillName, 'SKILL.md');
47
+ return fs.existsSync(bundledPath);
48
+ }
49
+ /**
50
+ * 번들된 스킬 목록 조회
51
+ */
52
+ export function listBundledSkills() {
53
+ if (!fs.existsSync(BUNDLED_SKILLS_DIR)) {
54
+ return [];
55
+ }
56
+ return fs.readdirSync(BUNDLED_SKILLS_DIR, { withFileTypes: true })
57
+ .filter(entry => entry.isDirectory())
58
+ .filter(entry => fs.existsSync(path.join(BUNDLED_SKILLS_DIR, entry.name, 'SKILL.md')))
59
+ .map(entry => entry.name);
60
+ }
61
+ /**
62
+ * 스킬 설치 (번들에서 복사)
63
+ */
64
+ export function installSkill(skillName, options = {}) {
65
+ const bundledPath = path.join(BUNDLED_SKILLS_DIR, skillName);
66
+ const targetPath = path.join(SKILLS_DIR, skillName);
67
+ // 번들된 스킬 존재 확인
68
+ if (!fs.existsSync(path.join(bundledPath, 'SKILL.md'))) {
69
+ return {
70
+ status: 'failed',
71
+ path: targetPath,
72
+ error: `번들된 스킬을 찾을 수 없습니다: ${skillName}`,
73
+ };
74
+ }
75
+ // 이미 설치되어 있고 force가 아니면 건너뛰기
76
+ if (skillExists(skillName) && !options.force) {
77
+ const metadata = loadSkillMetadata(skillName);
78
+ return {
79
+ status: 'skipped',
80
+ path: targetPath,
81
+ version: metadata?.version,
82
+ };
83
+ }
84
+ try {
85
+ // 스킬 디렉토리 생성
86
+ ensureSkillsDir();
87
+ // 기존 스킬 백업 (force인 경우)
88
+ if (options.force && fs.existsSync(targetPath)) {
89
+ const backupPath = `${targetPath}.backup-${Date.now()}`;
90
+ fs.renameSync(targetPath, backupPath);
91
+ }
92
+ // 스킬 복사
93
+ copyDirSync(bundledPath, targetPath);
94
+ const metadata = loadSkillMetadata(skillName);
95
+ return {
96
+ status: skillExists(skillName) ? 'updated' : 'installed',
97
+ path: targetPath,
98
+ version: metadata?.version,
99
+ };
100
+ }
101
+ catch (error) {
102
+ return {
103
+ status: 'failed',
104
+ path: targetPath,
105
+ error: error instanceof Error ? error.message : String(error),
106
+ };
107
+ }
108
+ }
109
+ /**
110
+ * 스킬 설치 (필요한 경우에만)
111
+ */
112
+ export function installSkillIfNeeded(skillName) {
113
+ if (skillExists(skillName)) {
114
+ const metadata = loadSkillMetadata(skillName);
115
+ return {
116
+ status: 'skipped',
117
+ path: path.join(SKILLS_DIR, skillName),
118
+ version: metadata?.version,
119
+ };
120
+ }
121
+ return installSkill(skillName);
122
+ }
123
+ /**
124
+ * 스킬 제거
125
+ */
126
+ export function uninstallSkill(skillName) {
127
+ const skillPath = path.join(SKILLS_DIR, skillName);
128
+ if (!fs.existsSync(skillPath)) {
129
+ return false;
130
+ }
131
+ try {
132
+ fs.rmSync(skillPath, { recursive: true, force: true });
133
+ return true;
134
+ }
135
+ catch {
136
+ return false;
137
+ }
138
+ }
139
+ /**
140
+ * 모든 번들 스킬 설치
141
+ */
142
+ export function installAllBundledSkills(options = {}) {
143
+ const results = new Map();
144
+ const bundledSkills = listBundledSkills();
145
+ for (const skillName of bundledSkills) {
146
+ const result = options.force
147
+ ? installSkill(skillName, { force: true })
148
+ : installSkillIfNeeded(skillName);
149
+ results.set(skillName, result);
150
+ }
151
+ return results;
152
+ }
153
+ //# sourceMappingURL=installer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"installer.js","sourceRoot":"","sources":["../../src/skills/installer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,34 @@
1
+ /**
2
+ * 스킬 로더 모듈
3
+ * SKILL.md 파일을 파싱하여 스킬 정보를 로드합니다.
4
+ */
5
+ import type { LoadedSkill, SkillMetadata } from '../types.js';
6
+ /**
7
+ * 스킬 디렉토리 기본 경로
8
+ */
9
+ export declare const SKILLS_DIR: string;
10
+ /**
11
+ * 스킬 존재 여부 확인
12
+ */
13
+ export declare function skillExists(skillName: string): boolean;
14
+ /**
15
+ * 스킬 경로 반환
16
+ */
17
+ export declare function getSkillPath(skillName: string): string;
18
+ /**
19
+ * 스킬 메타데이터만 로드 (빠른 조회용)
20
+ */
21
+ export declare function loadSkillMetadata(skillName: string): SkillMetadata | null;
22
+ /**
23
+ * 스킬 전체 정보 로드
24
+ */
25
+ export declare function loadSkill(skillName: string): LoadedSkill;
26
+ /**
27
+ * 설치된 모든 스킬 목록 조회
28
+ */
29
+ export declare function listInstalledSkills(): SkillMetadata[];
30
+ /**
31
+ * 스킬 템플릿 내용 로드
32
+ */
33
+ export declare function loadSkillTemplate(skillName: string, templateName: string): string | null;
34
+ //# sourceMappingURL=loader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loader.d.ts","sourceRoot":"","sources":["../../src/skills/loader.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAKH,OAAO,KAAK,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE9D;;GAEG;AACH,eAAO,MAAM,UAAU,QAA+C,CAAC;AAqCvE;;GAEG;AACH,wBAAgB,WAAW,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAGtD;AAED;;GAEG;AACH,wBAAgB,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,CAEtD;AAED;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,aAAa,GAAG,IAAI,CAqBzE;AAED;;GAEG;AACH,wBAAgB,SAAS,CAAC,SAAS,EAAE,MAAM,GAAG,WAAW,CA8BxD;AAED;;GAEG;AACH,wBAAgB,mBAAmB,IAAI,aAAa,EAAE,CAkBrD;AAED;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAQxF"}
@@ -0,0 +1,134 @@
1
+ /**
2
+ * 스킬 로더 모듈
3
+ * SKILL.md 파일을 파싱하여 스킬 정보를 로드합니다.
4
+ */
5
+ import fs from 'fs';
6
+ import path from 'path';
7
+ import os from 'os';
8
+ /**
9
+ * 스킬 디렉토리 기본 경로
10
+ */
11
+ export const SKILLS_DIR = path.join(os.homedir(), '.claude', 'skills');
12
+ /**
13
+ * SKILL.md frontmatter 파싱
14
+ * YAML frontmatter에서 스킬 메타데이터를 추출합니다.
15
+ */
16
+ function parseFrontmatter(content) {
17
+ const match = content.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);
18
+ if (!match) {
19
+ return { frontmatter: {}, body: content };
20
+ }
21
+ const yamlContent = match[1];
22
+ const body = match[2];
23
+ const result = {};
24
+ // 간단한 YAML 파싱 (key: value 형식)
25
+ for (const line of yamlContent.split('\n')) {
26
+ const colonIndex = line.indexOf(':');
27
+ if (colonIndex > 0) {
28
+ const key = line.slice(0, colonIndex).trim();
29
+ let value = line.slice(colonIndex + 1).trim();
30
+ // 배열 처리 [a, b, c]
31
+ if (typeof value === 'string' && value.startsWith('[') && value.endsWith(']')) {
32
+ const arrayContent = value.slice(1, -1);
33
+ value = arrayContent.split(',').map(v => v.trim());
34
+ }
35
+ result[key] = value;
36
+ }
37
+ }
38
+ return { frontmatter: result, body };
39
+ }
40
+ /**
41
+ * 스킬 존재 여부 확인
42
+ */
43
+ export function skillExists(skillName) {
44
+ const skillPath = path.join(SKILLS_DIR, skillName, 'SKILL.md');
45
+ return fs.existsSync(skillPath);
46
+ }
47
+ /**
48
+ * 스킬 경로 반환
49
+ */
50
+ export function getSkillPath(skillName) {
51
+ return path.join(SKILLS_DIR, skillName);
52
+ }
53
+ /**
54
+ * 스킬 메타데이터만 로드 (빠른 조회용)
55
+ */
56
+ export function loadSkillMetadata(skillName) {
57
+ const skillPath = path.join(SKILLS_DIR, skillName);
58
+ const skillMdPath = path.join(skillPath, 'SKILL.md');
59
+ if (!fs.existsSync(skillMdPath)) {
60
+ return null;
61
+ }
62
+ try {
63
+ const content = fs.readFileSync(skillMdPath, 'utf-8');
64
+ const { frontmatter } = parseFrontmatter(content);
65
+ return {
66
+ name: frontmatter.name || skillName,
67
+ description: frontmatter.description || '',
68
+ allowedTools: frontmatter['allowed-tools'] || [],
69
+ version: frontmatter.version,
70
+ };
71
+ }
72
+ catch {
73
+ return null;
74
+ }
75
+ }
76
+ /**
77
+ * 스킬 전체 정보 로드
78
+ */
79
+ export function loadSkill(skillName) {
80
+ const skillPath = path.join(SKILLS_DIR, skillName);
81
+ const skillMdPath = path.join(skillPath, 'SKILL.md');
82
+ if (!fs.existsSync(skillMdPath)) {
83
+ throw new Error(`스킬을 찾을 수 없습니다: ${skillName}\n경로: ${skillMdPath}`);
84
+ }
85
+ const content = fs.readFileSync(skillMdPath, 'utf-8');
86
+ const { frontmatter, body } = parseFrontmatter(content);
87
+ // 템플릿 파일 목록 수집
88
+ const templatesDir = path.join(skillPath, 'templates');
89
+ let templates = [];
90
+ if (fs.existsSync(templatesDir)) {
91
+ templates = fs.readdirSync(templatesDir)
92
+ .filter(f => f.endsWith('.md'))
93
+ .map(f => path.join(templatesDir, f));
94
+ }
95
+ return {
96
+ name: frontmatter.name || skillName,
97
+ description: frontmatter.description || '',
98
+ allowedTools: frontmatter['allowed-tools'] || [],
99
+ version: frontmatter.version,
100
+ content: body,
101
+ path: skillPath,
102
+ templates,
103
+ };
104
+ }
105
+ /**
106
+ * 설치된 모든 스킬 목록 조회
107
+ */
108
+ export function listInstalledSkills() {
109
+ if (!fs.existsSync(SKILLS_DIR)) {
110
+ return [];
111
+ }
112
+ const skills = [];
113
+ const entries = fs.readdirSync(SKILLS_DIR, { withFileTypes: true });
114
+ for (const entry of entries) {
115
+ if (entry.isDirectory()) {
116
+ const metadata = loadSkillMetadata(entry.name);
117
+ if (metadata) {
118
+ skills.push(metadata);
119
+ }
120
+ }
121
+ }
122
+ return skills;
123
+ }
124
+ /**
125
+ * 스킬 템플릿 내용 로드
126
+ */
127
+ export function loadSkillTemplate(skillName, templateName) {
128
+ const templatePath = path.join(SKILLS_DIR, skillName, 'templates', templateName);
129
+ if (!fs.existsSync(templatePath)) {
130
+ return null;
131
+ }
132
+ return fs.readFileSync(templatePath, 'utf-8');
133
+ }
134
+ //# sourceMappingURL=loader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loader.js","sourceRoot":"","sources":["../../src/skills/loader.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ /**
2
+ * 스킬 러너 모듈
3
+ * Claude Agent SDK를 사용하여 스킬을 실행합니다.
4
+ */
5
+ import type { SkillRunOptions, SkillRunResult } from '../types.js';
6
+ /**
7
+ * 스킬 실행
8
+ */
9
+ export declare function runSkill(skillName: string, options: SkillRunOptions): Promise<SkillRunResult>;
10
+ /**
11
+ * PRD 스킬 실행 (prd-taskmaster 전용)
12
+ */
13
+ export declare function runPrdSkill(options: SkillRunOptions): Promise<SkillRunResult>;
14
+ //# sourceMappingURL=runner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runner.d.ts","sourceRoot":"","sources":["../../src/skills/runner.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAUH,OAAO,KAAK,EAAe,eAAe,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAuFhF;;GAEG;AACH,wBAAsB,QAAQ,CAC5B,SAAS,EAAE,MAAM,EACjB,OAAO,EAAE,eAAe,GACvB,OAAO,CAAC,cAAc,CAAC,CAiKzB;AAED;;GAEG;AACH,wBAAsB,WAAW,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,cAAc,CAAC,CAEnF"}
@@ -0,0 +1,238 @@
1
+ /**
2
+ * 스킬 러너 모듈
3
+ * Claude Agent SDK를 사용하여 스킬을 실행합니다.
4
+ */
5
+ import { query, } from '@anthropic-ai/claude-agent-sdk';
6
+ import path from 'path';
7
+ import { loadSkill, skillExists, loadSkillTemplate } from './loader.js';
8
+ import { installSkillIfNeeded } from './installer.js';
9
+ /**
10
+ * 스킬 프롬프트 구성
11
+ */
12
+ function buildSkillPrompt(skill, options) {
13
+ const lines = [];
14
+ lines.push('# 스킬 실행 세션');
15
+ lines.push('');
16
+ lines.push('## 언어 규칙');
17
+ lines.push('- **모든 출력과 설명은 한국어로 작성하세요.**');
18
+ lines.push('- 코드, 명령어, 파일명은 영어로 유지하세요.');
19
+ lines.push('');
20
+ lines.push('## 프로젝트 정보');
21
+ lines.push(`- **작업 디렉토리:** ${options.workingDirectory}`);
22
+ lines.push(`- **프로젝트 이름:** ${path.basename(options.workingDirectory)}`);
23
+ lines.push('');
24
+ // 기존 문서가 있는 경우 포함
25
+ if (options.existingDocContent) {
26
+ lines.push('## 기존 요구사항 문서');
27
+ lines.push('');
28
+ lines.push('사용자가 기존에 작성한 요구사항 문서 또는 PRD 초안입니다.');
29
+ lines.push('이 문서의 내용을 분석하여 Taskmaster PRD 형식으로 변환하세요.');
30
+ lines.push('');
31
+ if (options.existingDocPath) {
32
+ lines.push(`**문서 경로:** ${options.existingDocPath}`);
33
+ lines.push('');
34
+ }
35
+ lines.push('<existing-document>');
36
+ lines.push(options.existingDocContent);
37
+ lines.push('</existing-document>');
38
+ lines.push('');
39
+ lines.push('### 변환 지침');
40
+ lines.push('1. 위 문서에서 요구사항, 기능, 목표를 추출하세요.');
41
+ lines.push('2. 누락된 정보가 있다면 사용자에게 질문하세요.');
42
+ lines.push('3. 최종적으로 Taskmaster PRD 형식에 맞게 구조화된 PRD를 생성하세요.');
43
+ lines.push('');
44
+ }
45
+ // 템플릿 선택 (있는 경우)
46
+ if (options.template && skill.templates && skill.templates.length > 0) {
47
+ const templateName = options.template === 'minimal'
48
+ ? 'taskmaster-prd-minimal.md'
49
+ : 'taskmaster-prd-comprehensive.md';
50
+ const templateContent = loadSkillTemplate(skill.name, templateName);
51
+ if (templateContent) {
52
+ lines.push('## 선택된 템플릿');
53
+ lines.push(`템플릿: ${options.template}`);
54
+ lines.push('');
55
+ lines.push('<template>');
56
+ lines.push(templateContent);
57
+ lines.push('</template>');
58
+ lines.push('');
59
+ }
60
+ }
61
+ lines.push('## 스킬 지시사항');
62
+ lines.push('');
63
+ lines.push('다음 스킬 문서를 따라 작업을 수행하세요:');
64
+ lines.push('');
65
+ lines.push('<skill>');
66
+ lines.push(skill.content);
67
+ lines.push('</skill>');
68
+ lines.push('');
69
+ lines.push('## 시작');
70
+ lines.push('');
71
+ if (options.existingDocContent) {
72
+ lines.push('기존 문서가 제공되었습니다. 먼저 문서를 분석하고, 누락된 정보를 파악한 후,');
73
+ lines.push('필요한 추가 질문을 사용자에게 하세요. 최종적으로 Taskmaster PRD 형식으로 변환하세요.');
74
+ }
75
+ else {
76
+ lines.push('위 스킬의 Step 1부터 시작하세요.');
77
+ }
78
+ lines.push('사용자에게 질문할 때는 AskUserQuestion 도구를 사용하여 명확하고 구체적으로 물어보세요.');
79
+ lines.push('');
80
+ return lines.join('\n');
81
+ }
82
+ /**
83
+ * 스킬 실행
84
+ */
85
+ export async function runSkill(skillName, options) {
86
+ const startTime = Date.now();
87
+ // 1. 스킬 로딩 (없으면 설치 시도)
88
+ if (!skillExists(skillName)) {
89
+ const installResult = installSkillIfNeeded(skillName);
90
+ if (installResult.status === 'failed') {
91
+ return {
92
+ success: false,
93
+ error: `스킬 설치 실패: ${installResult.error}`,
94
+ };
95
+ }
96
+ options.onProgress?.(`스킬 설치됨: ${skillName}`);
97
+ }
98
+ const skill = loadSkill(skillName);
99
+ options.onProgress?.(`스킬 로드됨: ${skill.name}`);
100
+ // 2. Dry run 모드
101
+ if (options.dryRun) {
102
+ options.onProgress?.('[DRY RUN] 스킬 실행 시뮬레이션');
103
+ return {
104
+ success: true,
105
+ result: '[DRY RUN] 시뮬레이션 완료',
106
+ durationMs: Date.now() - startTime,
107
+ };
108
+ }
109
+ // 3. 프롬프트 구성
110
+ const prompt = buildSkillPrompt(skill, options);
111
+ // 4. AbortController 설정
112
+ const abortController = new AbortController();
113
+ const timeout = options.timeout || 30 * 60 * 1000; // 30분 기본값
114
+ let timeoutHandle;
115
+ if (timeout > 0) {
116
+ timeoutHandle = setTimeout(() => {
117
+ abortController.abort();
118
+ options.onProgress?.(`[TIMEOUT] ${timeout}ms 초과로 스킬 중단`);
119
+ }, timeout);
120
+ }
121
+ try {
122
+ // 5. SDK 옵션 구성
123
+ const sdkOptions = {
124
+ model: 'claude-sonnet-4-5-20250929',
125
+ cwd: options.workingDirectory,
126
+ maxTurns: 100,
127
+ abortController,
128
+ permissionMode: 'bypassPermissions',
129
+ allowDangerouslySkipPermissions: true,
130
+ settingSources: ['project', 'user'],
131
+ systemPrompt: {
132
+ type: 'preset',
133
+ preset: 'claude_code',
134
+ append: '모든 출력은 한국어로 작성하세요. 사용자와 대화할 때는 AskUserQuestion 도구를 사용하세요.',
135
+ },
136
+ };
137
+ // 6. 스킬 실행
138
+ options.onProgress?.('[SDK] 스킬 세션 시작');
139
+ const response = query({ prompt, options: sdkOptions });
140
+ let sessionId;
141
+ let result;
142
+ const assistantMessages = [];
143
+ const filesCreated = [];
144
+ // 7. 스트리밍 응답 처리
145
+ for await (const message of response) {
146
+ switch (message.type) {
147
+ case 'system':
148
+ if (message.subtype === 'init') {
149
+ sessionId = message.session_id;
150
+ if (options.verbose) {
151
+ options.onProgress?.(`[SDK] 세션: ${sessionId}`);
152
+ }
153
+ }
154
+ break;
155
+ case 'assistant':
156
+ if (message.message?.content) {
157
+ const content = message.message.content;
158
+ if (Array.isArray(content)) {
159
+ for (const block of content) {
160
+ if ('type' in block && block.type === 'text' && 'text' in block) {
161
+ const text = block.text;
162
+ assistantMessages.push(text);
163
+ if (options.verbose) {
164
+ process.stdout.write(text);
165
+ }
166
+ }
167
+ // Write 도구 사용 감지
168
+ if ('type' in block && block.type === 'tool_use') {
169
+ const toolBlock = block;
170
+ if (toolBlock.name === 'Write' && toolBlock.input?.file_path) {
171
+ filesCreated.push(toolBlock.input.file_path);
172
+ }
173
+ }
174
+ }
175
+ }
176
+ }
177
+ break;
178
+ case 'result':
179
+ result = message;
180
+ break;
181
+ }
182
+ }
183
+ // 8. 타임아웃 해제
184
+ if (timeoutHandle) {
185
+ clearTimeout(timeoutHandle);
186
+ }
187
+ const durationMs = Date.now() - startTime;
188
+ // 9. 결과 처리
189
+ if (!result) {
190
+ return {
191
+ success: false,
192
+ error: '스킬 결과를 받지 못함',
193
+ sessionId,
194
+ durationMs,
195
+ };
196
+ }
197
+ if (result.subtype === 'success') {
198
+ return {
199
+ success: true,
200
+ result: result.result || assistantMessages.join('\n'),
201
+ sessionId,
202
+ numTurns: result.num_turns,
203
+ durationMs,
204
+ costUsd: result.total_cost_usd,
205
+ filesCreated,
206
+ };
207
+ }
208
+ else {
209
+ const errorResult = result;
210
+ return {
211
+ success: false,
212
+ error: errorResult.errors?.join('\n') || `스킬 실패: ${result.subtype}`,
213
+ sessionId,
214
+ numTurns: result.num_turns,
215
+ durationMs,
216
+ costUsd: result.total_cost_usd,
217
+ };
218
+ }
219
+ }
220
+ catch (error) {
221
+ if (timeoutHandle) {
222
+ clearTimeout(timeoutHandle);
223
+ }
224
+ const errorMessage = error instanceof Error ? error.message : String(error);
225
+ return {
226
+ success: false,
227
+ error: errorMessage,
228
+ durationMs: Date.now() - startTime,
229
+ };
230
+ }
231
+ }
232
+ /**
233
+ * PRD 스킬 실행 (prd-taskmaster 전용)
234
+ */
235
+ export async function runPrdSkill(options) {
236
+ return runSkill('prd-taskmaster', options);
237
+ }
238
+ //# sourceMappingURL=runner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runner.js","sourceRoot":"","sources":["../../src/skills/runner.ts"],"names":[],"mappings":"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"}