sales-frontend-gemini-cli 0.2.0 → 0.3.1

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 (87) hide show
  1. package/dist/{pr-review → common}/helper.cjs +82 -6
  2. package/dist/common/helper.cjs.map +1 -0
  3. package/dist/{pr-review → common}/helper.d.cts +12 -1
  4. package/dist/{pr-review → common}/helper.d.ts +12 -1
  5. package/dist/{pr-review → common}/helper.js +79 -7
  6. package/dist/common/helper.js.map +1 -0
  7. package/dist/common/types.cjs +4 -0
  8. package/dist/common/types.cjs.map +1 -0
  9. package/dist/common/types.d.cts +3 -0
  10. package/dist/common/types.d.ts +3 -0
  11. package/dist/common/types.js +3 -0
  12. package/dist/common/types.js.map +1 -0
  13. package/dist/pr-review/claude/claude-commander.cjs +59 -0
  14. package/dist/pr-review/claude/claude-commander.cjs.map +1 -0
  15. package/dist/pr-review/claude/claude-commander.d.cts +17 -0
  16. package/dist/pr-review/claude/claude-commander.d.ts +17 -0
  17. package/dist/pr-review/claude/claude-commander.js +51 -0
  18. package/dist/pr-review/claude/claude-commander.js.map +1 -0
  19. package/dist/pr-review/claude/installation-claude.cjs +27 -0
  20. package/dist/pr-review/claude/installation-claude.cjs.map +1 -0
  21. package/dist/pr-review/claude/installation-claude.d.cts +3 -0
  22. package/dist/pr-review/claude/installation-claude.d.ts +3 -0
  23. package/dist/pr-review/claude/installation-claude.js +25 -0
  24. package/dist/pr-review/claude/installation-claude.js.map +1 -0
  25. package/dist/pr-review/{commander.cjs → gemini/gemini-commander.cjs} +12 -12
  26. package/dist/pr-review/gemini/gemini-commander.cjs.map +1 -0
  27. package/dist/pr-review/{commander.d.cts → gemini/gemini-commander.d.cts} +2 -2
  28. package/dist/pr-review/{commander.d.ts → gemini/gemini-commander.d.ts} +2 -2
  29. package/dist/pr-review/{commander.js → gemini/gemini-commander.js} +11 -11
  30. package/dist/pr-review/gemini/gemini-commander.js.map +1 -0
  31. package/dist/{common → pr-review/gemini}/installation-gcloud.cjs +1 -1
  32. package/dist/pr-review/gemini/installation-gcloud.cjs.map +1 -0
  33. package/dist/{common → pr-review/gemini}/installation-gcloud.js +1 -1
  34. package/dist/pr-review/gemini/installation-gcloud.js.map +1 -0
  35. package/dist/{common → pr-review/gemini}/installation-gemini.cjs +1 -1
  36. package/dist/pr-review/gemini/installation-gemini.cjs.map +1 -0
  37. package/dist/{common → pr-review/gemini}/installation-gemini.js +1 -1
  38. package/dist/pr-review/gemini/installation-gemini.js.map +1 -0
  39. package/dist/pr-review/{interactive-version → gemini/interactive-version}/index.cjs +2 -2
  40. package/dist/pr-review/gemini/interactive-version/index.cjs.map +1 -0
  41. package/dist/pr-review/{interactive-version → gemini/interactive-version}/index.js +2 -2
  42. package/dist/pr-review/gemini/interactive-version/index.js.map +1 -0
  43. package/dist/{common → pr-review/gemini}/login.cjs +1 -1
  44. package/dist/pr-review/gemini/login.cjs.map +1 -0
  45. package/dist/{common → pr-review/gemini}/login.js +1 -1
  46. package/dist/pr-review/gemini/login.js.map +1 -0
  47. package/dist/pr-review/{vertex-version → gemini/vertex-version}/index.cjs +1 -1
  48. package/dist/pr-review/gemini/vertex-version/index.cjs.map +1 -0
  49. package/dist/pr-review/{vertex-version → gemini/vertex-version}/index.js +1 -1
  50. package/dist/pr-review/gemini/vertex-version/index.js.map +1 -0
  51. package/dist/pr-review/review-one-by-one.cjs +185 -50
  52. package/dist/pr-review/review-one-by-one.cjs.map +1 -1
  53. package/dist/pr-review/review-one-by-one.js +183 -49
  54. package/dist/pr-review/review-one-by-one.js.map +1 -1
  55. package/dist/pr-review/review.cjs +201 -65
  56. package/dist/pr-review/review.cjs.map +1 -1
  57. package/dist/pr-review/review.js +200 -65
  58. package/dist/pr-review/review.js.map +1 -1
  59. package/package.json +9 -9
  60. package/src/common/rules/review-rules.md +49 -0
  61. package/dist/common/installation-gcloud.cjs.map +0 -1
  62. package/dist/common/installation-gcloud.js.map +0 -1
  63. package/dist/common/installation-gemini.cjs.map +0 -1
  64. package/dist/common/installation-gemini.js.map +0 -1
  65. package/dist/common/login.cjs.map +0 -1
  66. package/dist/common/login.js.map +0 -1
  67. package/dist/pr-review/commander.cjs.map +0 -1
  68. package/dist/pr-review/commander.js.map +0 -1
  69. package/dist/pr-review/helper.cjs.map +0 -1
  70. package/dist/pr-review/helper.js.map +0 -1
  71. package/dist/pr-review/interactive-version/index.cjs.map +0 -1
  72. package/dist/pr-review/interactive-version/index.js.map +0 -1
  73. package/dist/pr-review/vertex-version/index.cjs.map +0 -1
  74. package/dist/pr-review/vertex-version/index.js.map +0 -1
  75. package/src/pr-review/rules/review-rules.md +0 -71
  76. /package/dist/{common → pr-review/gemini}/installation-gcloud.d.cts +0 -0
  77. /package/dist/{common → pr-review/gemini}/installation-gcloud.d.ts +0 -0
  78. /package/dist/{common → pr-review/gemini}/installation-gemini.d.cts +0 -0
  79. /package/dist/{common → pr-review/gemini}/installation-gemini.d.ts +0 -0
  80. /package/dist/pr-review/{interactive-version → gemini/interactive-version}/index.d.cts +0 -0
  81. /package/dist/pr-review/{interactive-version → gemini/interactive-version}/index.d.ts +0 -0
  82. /package/dist/{common → pr-review/gemini}/login.d.cts +0 -0
  83. /package/dist/{common → pr-review/gemini}/login.d.ts +0 -0
  84. /package/dist/pr-review/{vertex-version → gemini/vertex-version}/index.d.cts +0 -0
  85. /package/dist/pr-review/{vertex-version → gemini/vertex-version}/index.d.ts +0 -0
  86. /package/src/{pr-review → common}/form/review-form-one-by-one.md +0 -0
  87. /package/src/{pr-review → common}/form/review-form.md +0 -0
@@ -3,6 +3,7 @@
3
3
  var child_process = require('child_process');
4
4
  var fs = require('fs');
5
5
  var path = require('path');
6
+ var readline = require('readline');
6
7
  var url = require('url');
7
8
 
8
9
  var _documentCurrentScript = typeof document !== 'undefined' ? document.currentScript : null;
@@ -10,16 +11,18 @@ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
10
11
 
11
12
  var fs__default = /*#__PURE__*/_interopDefault(fs);
12
13
  var path__default = /*#__PURE__*/_interopDefault(path);
14
+ var readline__default = /*#__PURE__*/_interopDefault(readline);
13
15
 
14
- // src/pr-review/helper.ts
16
+ // src/common/helper.ts
15
17
  var __dirname$1 = path__default.default.dirname(url.fileURLToPath((typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('helper.cjs', document.baseURI).href))));
16
- var rulesPath = path__default.default.resolve(__dirname$1, "../../src/pr-review/rules/review-rules.md");
17
- var namingRulesPath = path__default.default.resolve(__dirname$1, "../../src/pr-review/rules/naming-rule.md");
18
- var codingConventionRulesPath = path__default.default.resolve(__dirname$1, "../../src/pr-review/rules/coding-convention.md");
19
- var reviewFormPath = path__default.default.resolve(__dirname$1, "../../src/pr-review/form/review-form.md");
20
- var reviewFormOneByOnePath = path__default.default.resolve(__dirname$1, "../../src/pr-review/form/review-form-one-by-one.md");
18
+ var rulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/review-rules.md");
19
+ var namingRulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/naming-rule.md");
20
+ var codingConventionRulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/coding-convention.md");
21
+ var reviewFormPath = path__default.default.resolve(__dirname$1, "../../src/common/form/review-form.md");
22
+ var reviewFormOneByOnePath = path__default.default.resolve(__dirname$1, "../../src/common/form/review-form-one-by-one.md");
21
23
  var REPORT_DIR = ".review-report";
22
24
  var tempDiffPath = "temp_diff.txt";
25
+ var AIServices = ["gemini", "claude", "codex"];
23
26
  var ignoreList = [
24
27
  "package.json",
25
28
  "*.yml",
@@ -116,7 +119,78 @@ function getDiffArgs() {
116
119
  }
117
120
  return diffArgs;
118
121
  }
122
+ function selectAIService() {
123
+ const args = process.argv.slice(2);
124
+ const serviceIndex = args.indexOf("--service");
125
+ const service = args[serviceIndex + 1];
126
+ if (!service) {
127
+ console.error("\u274C \uC11C\uBE44\uC2A4\uAC00 \uC120\uD0DD\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4.");
128
+ process.exit(1);
129
+ }
130
+ return service;
131
+ }
132
+ async function showSelectionAIService() {
133
+ let selectedIndex = 0;
134
+ const rl = readline__default.default.createInterface({
135
+ input: process.stdin,
136
+ output: process.stdout,
137
+ terminal: true
138
+ });
139
+ let firstRender = true;
140
+ process.stdout.write("\x1B[?25l");
141
+ const render = () => {
142
+ if (!firstRender) {
143
+ readline__default.default.moveCursor(process.stdout, 0, -(AIServices.length + 1));
144
+ }
145
+ firstRender = false;
146
+ readline__default.default.clearScreenDown(process.stdout);
147
+ process.stdout.write("\u{1F916} AI \uC11C\uBE44\uC2A4\uB97C \uC120\uD0DD\uD574\uC8FC\uC138\uC694 (\x1B[33m\u2191\u2193 \uBC29\uD5A5\uD0A4\x1B[0m \uC774\uB3D9, \x1B[33mEnter\x1B[0m \uC120\uD0DD):\n");
148
+ AIServices.forEach((service, index) => {
149
+ if (index === selectedIndex) {
150
+ process.stdout.write(` \x1B[36m>\x1B[0m \x1B[36m\u25C9\x1B[0m \x1B[1m${service}\x1B[0m
151
+ `);
152
+ } else {
153
+ process.stdout.write(` \u25EF ${service}
154
+ `);
155
+ }
156
+ });
157
+ };
158
+ render();
159
+ return new Promise((resolve) => {
160
+ const onData = (data) => {
161
+ const key = data.toString();
162
+ if (key === "") {
163
+ process.stdout.write("\x1B[?25h");
164
+ process.exit(0);
165
+ }
166
+ if (key === "\x1B[A") {
167
+ selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;
168
+ render();
169
+ } else if (key === "\x1B[B") {
170
+ selectedIndex = (selectedIndex + 1) % AIServices.length;
171
+ render();
172
+ } else if (key === "\r" || key === "\n") {
173
+ process.stdin.removeListener("data", onData);
174
+ process.stdin.setRawMode(false);
175
+ process.stdin.pause();
176
+ rl.close();
177
+ process.stdout.write("\x1B[?25h");
178
+ console.log(`
179
+ \u2705 \x1B[32m${AIServices[selectedIndex]}\x1B[0m \uC11C\uBE44\uC2A4\uAC00 \uC120\uD0DD\uB418\uC5C8\uC2B5\uB2C8\uB2E4.
180
+ `);
181
+ const result = AIServices[selectedIndex];
182
+ if (result) {
183
+ resolve(result);
184
+ }
185
+ }
186
+ };
187
+ process.stdin.setRawMode(true);
188
+ process.stdin.resume();
189
+ process.stdin.on("data", onData);
190
+ });
191
+ }
119
192
 
193
+ exports.AIServices = AIServices;
120
194
  exports.REPORT_DIR = REPORT_DIR;
121
195
  exports.codingConventionRulesPath = codingConventionRulesPath;
122
196
  exports.createReportDirectory = createReportDirectory;
@@ -132,6 +206,8 @@ exports.openReport = openReport;
132
206
  exports.reviewFormOneByOnePath = reviewFormOneByOnePath;
133
207
  exports.reviewFormPath = reviewFormPath;
134
208
  exports.rulesPath = rulesPath;
209
+ exports.selectAIService = selectAIService;
210
+ exports.showSelectionAIService = showSelectionAIService;
135
211
  exports.tempDiffPath = tempDiffPath;
136
212
  //# sourceMappingURL=helper.cjs.map
137
213
  //# sourceMappingURL=helper.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/common/helper.ts"],"names":["__dirname","path","fileURLToPath","fs","execSync","readline"],"mappings":"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","file":"helper.cjs","sourcesContent":["import { execSync } from 'child_process';\nimport fs from 'fs'\nimport path from 'path';\nimport readline from 'readline';\nimport { fileURLToPath } from 'url';\n\nimport { AIServiceType } from './types';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// 설치된 위치에 맞게 규칙/양식 파일 경로를 계산 (dist에서 src로 이동 등 고려)\n// dist/common/helper.js 가 실행되므로 __dirname은 .../dist/common 입니다.\n// 따라서 ../../src/common/rules 로 이동해야 원본 소스의 규칙 파일을 찾을 수 있습니다.\nexport const rulesPath = path.resolve(__dirname, '../../src/common/rules/review-rules.md');\nexport const namingRulesPath = path.resolve(__dirname, '../../src/common/rules/naming-rule.md');\nexport const codingConventionRulesPath = path.resolve(__dirname, '../../src/common/rules/coding-convention.md');\nexport const reviewFormPath = path.resolve(__dirname, '../../src/common/form/review-form.md');\nexport const reviewFormOneByOnePath = path.resolve(__dirname, '../../src/common/form/review-form-one-by-one.md');\nexport const REPORT_DIR = '.review-report';\nexport const tempDiffPath = 'temp_diff.txt';\nexport const AIServices: AIServiceType[] = ['gemini', 'claude', 'codex'];\nexport const ignoreList = [\n 'package.json',\n '*.yml',\n '*.md',\n '*.lock',\n 'dist/',\n 'node_modules/',\n 'assets/',\n 'public/',\n '*.json',\n '*.yaml',\n '.review-report/' // 생성되는 리포트 폴더도 제외\n];\n\n\nexport function getNextFilePath(dir: string, baseName: string, extension: string) {\n let counter = 1;\n // eslint-disable-next-line no-constant-condition\n while (true) {\n const filePath = path.join(dir, `${baseName}-${counter}${extension}`);\n if (!fs.existsSync(filePath)) {\n return filePath;\n }\n counter++;\n }\n}\n\nexport function deleteFile(filePath: string) {\n if (fs.existsSync(filePath)) {\n fs.unlinkSync(filePath);\n }\n}\n\n\n/**\n * 임시파일 삭제\n */\nexport function deleteTempDiff() {\n deleteFile(tempDiffPath);\n}\n\n\n/**\n * 리뷰 결과 폴더 생성\n */\nexport function createReportDirectory() {\n if (!fs.existsSync(REPORT_DIR)) {\n fs.mkdirSync(REPORT_DIR, { recursive: true });\n }\n}\n\n\n/**\n * 현재 시간 문자열 생성\n */\nexport function getNowString() {\n const now = new Date();\n const YYYY = now.getFullYear();\n const MM = String(now.getMonth() + 1).padStart(2, '0');\n const DD = String(now.getDate()).padStart(2, '0');\n const HH = String(now.getHours()).padStart(2, '0');\n const mm = String(now.getMinutes()).padStart(2, '0');\n const ss = String(now.getSeconds()).padStart(2, '0');\n\n return `${YYYY}-${MM}-${DD}_${HH}-${mm}-${ss}`;\n}\n\nexport function getGitDiffFilter() {\n\n // 1. 리뷰 대상 파일 확장자 정의\n const includeExtensions = ['*.ts', '*.tsx', '*.js', '*.jsx'];\n\n // ignoreList 를 import 하여 재사용하여 작성한다.\n const excludePatterns = ignoreList.map(item => `:(exclude)${item}`);\n\n // const excludePatterns = [':(exclude)*.lock', ':(exclude)dist/', ':(exclude)*.md'];\n\n // 2. 변경된 파일 목록 가져오기 (각 패턴을 따옴표로 감싸서 쉘 에러 방지)\n const quote = (pattern: string) => `\"${pattern}\"`;\n const includeParams = includeExtensions.map(quote).join(' ');\n const excludeParams = excludePatterns.map(quote).join(' ');\n\n return { includeParams, excludeParams };\n\n\n}\n\nexport function openReport(reportPath: string) {\n // 브라우저 열기\n try {\n execSync(`open -a \"Google Chrome\" \"${path.resolve(reportPath)}\"`);\n console.log(`🚀 브라우저에서 리포트를 열었습니다.`);\n } catch (e) {\n console.error('⚠️ 브라우저 열기 실패:', e);\n }\n}\n\n\nexport function getDiffArgs() {\n const args = process.argv.slice(2);\n const commitIndex = args.indexOf('--commit');\n const { includeParams, excludeParams } = getGitDiffFilter();\n\n let diffArgs = '';\n\n if (commitIndex !== -1) {\n // 특정 커밋 (및 이전 n개) 리뷰\n const commitHash = args[commitIndex + 1];\n if (!commitHash) {\n console.error('❌ 커밋 해시가 제공되지 않았습니다.');\n process.exit(1);\n }\n\n // n값 확인 (optional)\n const nextArg = args[commitIndex + 2];\n let n = 0;\n if (nextArg && !nextArg.startsWith('--')) {\n n = parseInt(nextArg, 10);\n if (isNaN(n)) {\n n = 0;\n }\n }\n\n console.log(`ℹ️ 커밋 '${commitHash}' ${n > 0 ? ` 포함 총 ${n + 1}개의 커밋` : ''}을 리뷰합니다...`);\n diffArgs = `${commitHash}~${n + 1} ${commitHash}`;\n } else {\n // 기본 모드:\n // 1. Unstaged 변경사항 확인\n try {\n const check = execSync(`git diff --name-only -- ${includeParams} ${excludeParams}`).toString();\n if (!check.trim()) {\n console.log('ℹ️ Unstaged 변경사항이 없습니다. 마지막 커밋(HEAD)을 리뷰합니다...');\n diffArgs = 'HEAD~1 HEAD';\n }\n } catch {\n // git diff 실패시 무시\n }\n }\n\n return diffArgs;\n}\n\n// export const ServiceType = {\n// GEMINI: 'gemini',\n// CLAUDE: 'claude',\n// CODEX: 'codex'\n\n// }\n\n/**\n * AI 서비스 선택\n */\nexport function selectAIService() {\n const args = process.argv.slice(2);\n const serviceIndex = args.indexOf('--service');\n const service = args[serviceIndex + 1];\n if (!service) {\n console.error('❌ 서비스가 선택되지 않았습니다.');\n process.exit(1);\n }\n\n return service;\n}\n\n/**\n * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.\n */\nexport async function showSelectionAIService(): Promise<AIServiceType> {\n\n let selectedIndex = 0;\n\n // Use readline to handle keypresses\n // 키 입력을 처리하기 위해 readline 인터페이스 사용\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n terminal: true\n });\n\n let firstRender = true;\n\n // Hide cursor\n process.stdout.write('\\u001b[?25l');\n\n const render = () => {\n if (!firstRender) {\n // Move cursor back to the starting line of the selection UI\n // We print (1 question line + services.length lines)\n // 선택 UI의 시작 라인으로 커서 이동 (질문 1줄 + 서비스 목록 N줄)\n readline.moveCursor(process.stdout, 0, -(AIServices.length + 1));\n }\n firstRender = false;\n\n // Clear everything from cursor down to avoid ghosting/overlaps\n // 잔상이나 겹침을 방지하기 위해 커서 위치부터 아래쪽 모두 지움\n readline.clearScreenDown(process.stdout);\n\n process.stdout.write('🤖 AI 서비스를 선택해주세요 (\\u001b[33m↑↓ 방향키\\u001b[0m 이동, \\u001b[33mEnter\\u001b[0m 선택):\\n');\n AIServices.forEach((service, index) => {\n if (index === selectedIndex) {\n process.stdout.write(` \\u001b[36m>\\u001b[0m \\u001b[36m◉\\u001b[0m \\u001b[1m${service}\\u001b[0m\\n`);\n } else {\n process.stdout.write(` ◯ ${service}\\n`);\n }\n });\n };\n\n render();\n\n return new Promise((resolve) => {\n const onData = (data: Buffer) => {\n const key = data.toString();\n if (key === '\\u0003') { // Ctrl+C\n process.stdout.write('\\u001b[?25h'); // Show cursor\n process.exit(0);\n }\n if (key === '\\x1b[A') { // Up arrow\n selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;\n render();\n } else if (key === '\\x1b[B') { // Down arrow\n selectedIndex = (selectedIndex + 1) % AIServices.length;\n render();\n } else if (key === '\\r' || key === '\\n') { // Enter\n process.stdin.removeListener('data', onData);\n process.stdin.setRawMode(false);\n process.stdin.pause();\n rl.close();\n\n // Show cursor\n process.stdout.write('\\u001b[?25h');\n\n console.log(`\\n✅ \\u001b[32m${AIServices[selectedIndex]}\\u001b[0m 서비스가 선택되었습니다.\\n`);\n const result = AIServices[selectedIndex];\n if (result) {\n resolve(result);\n }\n }\n };\n\n process.stdin.setRawMode(true);\n process.stdin.resume();\n process.stdin.on('data', onData);\n });\n}\n\n"]}
@@ -1,3 +1,5 @@
1
+ import { AIServiceType } from './types.cjs';
2
+
1
3
  declare const rulesPath: string;
2
4
  declare const namingRulesPath: string;
3
5
  declare const codingConventionRulesPath: string;
@@ -5,6 +7,7 @@ declare const reviewFormPath: string;
5
7
  declare const reviewFormOneByOnePath: string;
6
8
  declare const REPORT_DIR = ".review-report";
7
9
  declare const tempDiffPath = "temp_diff.txt";
10
+ declare const AIServices: AIServiceType[];
8
11
  declare const ignoreList: string[];
9
12
  declare function getNextFilePath(dir: string, baseName: string, extension: string): string;
10
13
  declare function deleteFile(filePath: string): void;
@@ -26,5 +29,13 @@ declare function getGitDiffFilter(): {
26
29
  };
27
30
  declare function openReport(reportPath: string): void;
28
31
  declare function getDiffArgs(): string;
32
+ /**
33
+ * AI 서비스 선택
34
+ */
35
+ declare function selectAIService(): string;
36
+ /**
37
+ * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.
38
+ */
39
+ declare function showSelectionAIService(): Promise<AIServiceType>;
29
40
 
30
- export { REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, tempDiffPath };
41
+ export { AIServices, REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, selectAIService, showSelectionAIService, tempDiffPath };
@@ -1,3 +1,5 @@
1
+ import { AIServiceType } from './types.js';
2
+
1
3
  declare const rulesPath: string;
2
4
  declare const namingRulesPath: string;
3
5
  declare const codingConventionRulesPath: string;
@@ -5,6 +7,7 @@ declare const reviewFormPath: string;
5
7
  declare const reviewFormOneByOnePath: string;
6
8
  declare const REPORT_DIR = ".review-report";
7
9
  declare const tempDiffPath = "temp_diff.txt";
10
+ declare const AIServices: AIServiceType[];
8
11
  declare const ignoreList: string[];
9
12
  declare function getNextFilePath(dir: string, baseName: string, extension: string): string;
10
13
  declare function deleteFile(filePath: string): void;
@@ -26,5 +29,13 @@ declare function getGitDiffFilter(): {
26
29
  };
27
30
  declare function openReport(reportPath: string): void;
28
31
  declare function getDiffArgs(): string;
32
+ /**
33
+ * AI 서비스 선택
34
+ */
35
+ declare function selectAIService(): string;
36
+ /**
37
+ * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.
38
+ */
39
+ declare function showSelectionAIService(): Promise<AIServiceType>;
29
40
 
30
- export { REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, tempDiffPath };
41
+ export { AIServices, REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, selectAIService, showSelectionAIService, tempDiffPath };
@@ -1,17 +1,19 @@
1
1
  import { execSync } from 'child_process';
2
2
  import fs from 'fs';
3
3
  import path from 'path';
4
+ import readline from 'readline';
4
5
  import { fileURLToPath } from 'url';
5
6
 
6
- // src/pr-review/helper.ts
7
+ // src/common/helper.ts
7
8
  var __dirname = path.dirname(fileURLToPath(import.meta.url));
8
- var rulesPath = path.resolve(__dirname, "../../src/pr-review/rules/review-rules.md");
9
- var namingRulesPath = path.resolve(__dirname, "../../src/pr-review/rules/naming-rule.md");
10
- var codingConventionRulesPath = path.resolve(__dirname, "../../src/pr-review/rules/coding-convention.md");
11
- var reviewFormPath = path.resolve(__dirname, "../../src/pr-review/form/review-form.md");
12
- var reviewFormOneByOnePath = path.resolve(__dirname, "../../src/pr-review/form/review-form-one-by-one.md");
9
+ var rulesPath = path.resolve(__dirname, "../../src/common/rules/review-rules.md");
10
+ var namingRulesPath = path.resolve(__dirname, "../../src/common/rules/naming-rule.md");
11
+ var codingConventionRulesPath = path.resolve(__dirname, "../../src/common/rules/coding-convention.md");
12
+ var reviewFormPath = path.resolve(__dirname, "../../src/common/form/review-form.md");
13
+ var reviewFormOneByOnePath = path.resolve(__dirname, "../../src/common/form/review-form-one-by-one.md");
13
14
  var REPORT_DIR = ".review-report";
14
15
  var tempDiffPath = "temp_diff.txt";
16
+ var AIServices = ["gemini", "claude", "codex"];
15
17
  var ignoreList = [
16
18
  "package.json",
17
19
  "*.yml",
@@ -108,7 +110,77 @@ function getDiffArgs() {
108
110
  }
109
111
  return diffArgs;
110
112
  }
113
+ function selectAIService() {
114
+ const args = process.argv.slice(2);
115
+ const serviceIndex = args.indexOf("--service");
116
+ const service = args[serviceIndex + 1];
117
+ if (!service) {
118
+ console.error("\u274C \uC11C\uBE44\uC2A4\uAC00 \uC120\uD0DD\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4.");
119
+ process.exit(1);
120
+ }
121
+ return service;
122
+ }
123
+ async function showSelectionAIService() {
124
+ let selectedIndex = 0;
125
+ const rl = readline.createInterface({
126
+ input: process.stdin,
127
+ output: process.stdout,
128
+ terminal: true
129
+ });
130
+ let firstRender = true;
131
+ process.stdout.write("\x1B[?25l");
132
+ const render = () => {
133
+ if (!firstRender) {
134
+ readline.moveCursor(process.stdout, 0, -(AIServices.length + 1));
135
+ }
136
+ firstRender = false;
137
+ readline.clearScreenDown(process.stdout);
138
+ process.stdout.write("\u{1F916} AI \uC11C\uBE44\uC2A4\uB97C \uC120\uD0DD\uD574\uC8FC\uC138\uC694 (\x1B[33m\u2191\u2193 \uBC29\uD5A5\uD0A4\x1B[0m \uC774\uB3D9, \x1B[33mEnter\x1B[0m \uC120\uD0DD):\n");
139
+ AIServices.forEach((service, index) => {
140
+ if (index === selectedIndex) {
141
+ process.stdout.write(` \x1B[36m>\x1B[0m \x1B[36m\u25C9\x1B[0m \x1B[1m${service}\x1B[0m
142
+ `);
143
+ } else {
144
+ process.stdout.write(` \u25EF ${service}
145
+ `);
146
+ }
147
+ });
148
+ };
149
+ render();
150
+ return new Promise((resolve) => {
151
+ const onData = (data) => {
152
+ const key = data.toString();
153
+ if (key === "") {
154
+ process.stdout.write("\x1B[?25h");
155
+ process.exit(0);
156
+ }
157
+ if (key === "\x1B[A") {
158
+ selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;
159
+ render();
160
+ } else if (key === "\x1B[B") {
161
+ selectedIndex = (selectedIndex + 1) % AIServices.length;
162
+ render();
163
+ } else if (key === "\r" || key === "\n") {
164
+ process.stdin.removeListener("data", onData);
165
+ process.stdin.setRawMode(false);
166
+ process.stdin.pause();
167
+ rl.close();
168
+ process.stdout.write("\x1B[?25h");
169
+ console.log(`
170
+ \u2705 \x1B[32m${AIServices[selectedIndex]}\x1B[0m \uC11C\uBE44\uC2A4\uAC00 \uC120\uD0DD\uB418\uC5C8\uC2B5\uB2C8\uB2E4.
171
+ `);
172
+ const result = AIServices[selectedIndex];
173
+ if (result) {
174
+ resolve(result);
175
+ }
176
+ }
177
+ };
178
+ process.stdin.setRawMode(true);
179
+ process.stdin.resume();
180
+ process.stdin.on("data", onData);
181
+ });
182
+ }
111
183
 
112
- export { REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, tempDiffPath };
184
+ export { AIServices, REPORT_DIR, codingConventionRulesPath, createReportDirectory, deleteFile, deleteTempDiff, getDiffArgs, getGitDiffFilter, getNextFilePath, getNowString, ignoreList, namingRulesPath, openReport, reviewFormOneByOnePath, reviewFormPath, rulesPath, selectAIService, showSelectionAIService, tempDiffPath };
113
185
  //# sourceMappingURL=helper.js.map
114
186
  //# sourceMappingURL=helper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/common/helper.ts"],"names":[],"mappings":"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","file":"helper.js","sourcesContent":["import { execSync } from 'child_process';\nimport fs from 'fs'\nimport path from 'path';\nimport readline from 'readline';\nimport { fileURLToPath } from 'url';\n\nimport { AIServiceType } from './types';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// 설치된 위치에 맞게 규칙/양식 파일 경로를 계산 (dist에서 src로 이동 등 고려)\n// dist/common/helper.js 가 실행되므로 __dirname은 .../dist/common 입니다.\n// 따라서 ../../src/common/rules 로 이동해야 원본 소스의 규칙 파일을 찾을 수 있습니다.\nexport const rulesPath = path.resolve(__dirname, '../../src/common/rules/review-rules.md');\nexport const namingRulesPath = path.resolve(__dirname, '../../src/common/rules/naming-rule.md');\nexport const codingConventionRulesPath = path.resolve(__dirname, '../../src/common/rules/coding-convention.md');\nexport const reviewFormPath = path.resolve(__dirname, '../../src/common/form/review-form.md');\nexport const reviewFormOneByOnePath = path.resolve(__dirname, '../../src/common/form/review-form-one-by-one.md');\nexport const REPORT_DIR = '.review-report';\nexport const tempDiffPath = 'temp_diff.txt';\nexport const AIServices: AIServiceType[] = ['gemini', 'claude', 'codex'];\nexport const ignoreList = [\n 'package.json',\n '*.yml',\n '*.md',\n '*.lock',\n 'dist/',\n 'node_modules/',\n 'assets/',\n 'public/',\n '*.json',\n '*.yaml',\n '.review-report/' // 생성되는 리포트 폴더도 제외\n];\n\n\nexport function getNextFilePath(dir: string, baseName: string, extension: string) {\n let counter = 1;\n // eslint-disable-next-line no-constant-condition\n while (true) {\n const filePath = path.join(dir, `${baseName}-${counter}${extension}`);\n if (!fs.existsSync(filePath)) {\n return filePath;\n }\n counter++;\n }\n}\n\nexport function deleteFile(filePath: string) {\n if (fs.existsSync(filePath)) {\n fs.unlinkSync(filePath);\n }\n}\n\n\n/**\n * 임시파일 삭제\n */\nexport function deleteTempDiff() {\n deleteFile(tempDiffPath);\n}\n\n\n/**\n * 리뷰 결과 폴더 생성\n */\nexport function createReportDirectory() {\n if (!fs.existsSync(REPORT_DIR)) {\n fs.mkdirSync(REPORT_DIR, { recursive: true });\n }\n}\n\n\n/**\n * 현재 시간 문자열 생성\n */\nexport function getNowString() {\n const now = new Date();\n const YYYY = now.getFullYear();\n const MM = String(now.getMonth() + 1).padStart(2, '0');\n const DD = String(now.getDate()).padStart(2, '0');\n const HH = String(now.getHours()).padStart(2, '0');\n const mm = String(now.getMinutes()).padStart(2, '0');\n const ss = String(now.getSeconds()).padStart(2, '0');\n\n return `${YYYY}-${MM}-${DD}_${HH}-${mm}-${ss}`;\n}\n\nexport function getGitDiffFilter() {\n\n // 1. 리뷰 대상 파일 확장자 정의\n const includeExtensions = ['*.ts', '*.tsx', '*.js', '*.jsx'];\n\n // ignoreList 를 import 하여 재사용하여 작성한다.\n const excludePatterns = ignoreList.map(item => `:(exclude)${item}`);\n\n // const excludePatterns = [':(exclude)*.lock', ':(exclude)dist/', ':(exclude)*.md'];\n\n // 2. 변경된 파일 목록 가져오기 (각 패턴을 따옴표로 감싸서 쉘 에러 방지)\n const quote = (pattern: string) => `\"${pattern}\"`;\n const includeParams = includeExtensions.map(quote).join(' ');\n const excludeParams = excludePatterns.map(quote).join(' ');\n\n return { includeParams, excludeParams };\n\n\n}\n\nexport function openReport(reportPath: string) {\n // 브라우저 열기\n try {\n execSync(`open -a \"Google Chrome\" \"${path.resolve(reportPath)}\"`);\n console.log(`🚀 브라우저에서 리포트를 열었습니다.`);\n } catch (e) {\n console.error('⚠️ 브라우저 열기 실패:', e);\n }\n}\n\n\nexport function getDiffArgs() {\n const args = process.argv.slice(2);\n const commitIndex = args.indexOf('--commit');\n const { includeParams, excludeParams } = getGitDiffFilter();\n\n let diffArgs = '';\n\n if (commitIndex !== -1) {\n // 특정 커밋 (및 이전 n개) 리뷰\n const commitHash = args[commitIndex + 1];\n if (!commitHash) {\n console.error('❌ 커밋 해시가 제공되지 않았습니다.');\n process.exit(1);\n }\n\n // n값 확인 (optional)\n const nextArg = args[commitIndex + 2];\n let n = 0;\n if (nextArg && !nextArg.startsWith('--')) {\n n = parseInt(nextArg, 10);\n if (isNaN(n)) {\n n = 0;\n }\n }\n\n console.log(`ℹ️ 커밋 '${commitHash}' ${n > 0 ? ` 포함 총 ${n + 1}개의 커밋` : ''}을 리뷰합니다...`);\n diffArgs = `${commitHash}~${n + 1} ${commitHash}`;\n } else {\n // 기본 모드:\n // 1. Unstaged 변경사항 확인\n try {\n const check = execSync(`git diff --name-only -- ${includeParams} ${excludeParams}`).toString();\n if (!check.trim()) {\n console.log('ℹ️ Unstaged 변경사항이 없습니다. 마지막 커밋(HEAD)을 리뷰합니다...');\n diffArgs = 'HEAD~1 HEAD';\n }\n } catch {\n // git diff 실패시 무시\n }\n }\n\n return diffArgs;\n}\n\n// export const ServiceType = {\n// GEMINI: 'gemini',\n// CLAUDE: 'claude',\n// CODEX: 'codex'\n\n// }\n\n/**\n * AI 서비스 선택\n */\nexport function selectAIService() {\n const args = process.argv.slice(2);\n const serviceIndex = args.indexOf('--service');\n const service = args[serviceIndex + 1];\n if (!service) {\n console.error('❌ 서비스가 선택되지 않았습니다.');\n process.exit(1);\n }\n\n return service;\n}\n\n/**\n * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.\n */\nexport async function showSelectionAIService(): Promise<AIServiceType> {\n\n let selectedIndex = 0;\n\n // Use readline to handle keypresses\n // 키 입력을 처리하기 위해 readline 인터페이스 사용\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n terminal: true\n });\n\n let firstRender = true;\n\n // Hide cursor\n process.stdout.write('\\u001b[?25l');\n\n const render = () => {\n if (!firstRender) {\n // Move cursor back to the starting line of the selection UI\n // We print (1 question line + services.length lines)\n // 선택 UI의 시작 라인으로 커서 이동 (질문 1줄 + 서비스 목록 N줄)\n readline.moveCursor(process.stdout, 0, -(AIServices.length + 1));\n }\n firstRender = false;\n\n // Clear everything from cursor down to avoid ghosting/overlaps\n // 잔상이나 겹침을 방지하기 위해 커서 위치부터 아래쪽 모두 지움\n readline.clearScreenDown(process.stdout);\n\n process.stdout.write('🤖 AI 서비스를 선택해주세요 (\\u001b[33m↑↓ 방향키\\u001b[0m 이동, \\u001b[33mEnter\\u001b[0m 선택):\\n');\n AIServices.forEach((service, index) => {\n if (index === selectedIndex) {\n process.stdout.write(` \\u001b[36m>\\u001b[0m \\u001b[36m◉\\u001b[0m \\u001b[1m${service}\\u001b[0m\\n`);\n } else {\n process.stdout.write(` ◯ ${service}\\n`);\n }\n });\n };\n\n render();\n\n return new Promise((resolve) => {\n const onData = (data: Buffer) => {\n const key = data.toString();\n if (key === '\\u0003') { // Ctrl+C\n process.stdout.write('\\u001b[?25h'); // Show cursor\n process.exit(0);\n }\n if (key === '\\x1b[A') { // Up arrow\n selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;\n render();\n } else if (key === '\\x1b[B') { // Down arrow\n selectedIndex = (selectedIndex + 1) % AIServices.length;\n render();\n } else if (key === '\\r' || key === '\\n') { // Enter\n process.stdin.removeListener('data', onData);\n process.stdin.setRawMode(false);\n process.stdin.pause();\n rl.close();\n\n // Show cursor\n process.stdout.write('\\u001b[?25h');\n\n console.log(`\\n✅ \\u001b[32m${AIServices[selectedIndex]}\\u001b[0m 서비스가 선택되었습니다.\\n`);\n const result = AIServices[selectedIndex];\n if (result) {\n resolve(result);\n }\n }\n };\n\n process.stdin.setRawMode(true);\n process.stdin.resume();\n process.stdin.on('data', onData);\n });\n}\n\n"]}
@@ -0,0 +1,4 @@
1
+ 'use strict';
2
+
3
+ //# sourceMappingURL=types.cjs.map
4
+ //# sourceMappingURL=types.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"types.cjs"}
@@ -0,0 +1,3 @@
1
+ type AIServiceType = 'gemini' | 'claude' | 'codex';
2
+
3
+ export type { AIServiceType };
@@ -0,0 +1,3 @@
1
+ type AIServiceType = 'gemini' | 'claude' | 'codex';
2
+
3
+ export type { AIServiceType };
@@ -0,0 +1,3 @@
1
+
2
+ //# sourceMappingURL=types.js.map
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"types.js"}
@@ -0,0 +1,59 @@
1
+ 'use strict';
2
+
3
+ var fs = require('fs');
4
+ var path = require('path');
5
+ var url = require('url');
6
+
7
+ var _documentCurrentScript = typeof document !== 'undefined' ? document.currentScript : null;
8
+ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
9
+
10
+ var fs__default = /*#__PURE__*/_interopDefault(fs);
11
+ var path__default = /*#__PURE__*/_interopDefault(path);
12
+
13
+ // src/pr-review/claude/claude-commander.ts
14
+ var __dirname$1 = path__default.default.dirname(url.fileURLToPath((typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('claude-commander.cjs', document.baseURI).href))));
15
+ var rulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/review-rules.md");
16
+ var namingRulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/naming-rule.md");
17
+ var codingConventionRulesPath = path__default.default.resolve(__dirname$1, "../../src/common/rules/coding-convention.md");
18
+ path__default.default.resolve(__dirname$1, "../../src/common/form/review-form.md");
19
+ path__default.default.resolve(__dirname$1, "../../src/common/form/review-form-one-by-one.md");
20
+
21
+ // src/pr-review/claude/claude-commander.ts
22
+ var args = process.argv.slice(2);
23
+ var createClaudeCommand = (tempDiffPath, reviewFormPath2) => {
24
+ let modelOption = "";
25
+ if (args.includes("--review")) {
26
+ modelOption = "--model opus";
27
+ } else if (args.includes("--flash")) {
28
+ modelOption = "--model haiku";
29
+ } else {
30
+ const modelIndex = args.indexOf("--model");
31
+ if (modelIndex !== -1 && args[modelIndex + 1]) {
32
+ console.warn("\u26A0\uFE0F \uC9C0\uC815\uD55C \uBAA8\uB378\uC774 \uC5C6\uB294 \uACBD\uC6B0, \uC5D0\uB7EC\uAC00 \uBC1C\uC0DD\uD558\uB2C8 \uC8FC\uC758\uD558\uC138\uC694.");
33
+ modelOption = `--model ${args[modelIndex + 1]}`;
34
+ } else {
35
+ console.warn("\u26A0\uFE0F \uBAA8\uB378\uC774 \uC9C0\uC815\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4. \uAE30\uBCF8 \uBAA8\uB378\uC778 sonnet\uC744 \uC0AC\uC6A9\uD569\uB2C8\uB2E4.");
36
+ modelOption = "--model sonnet";
37
+ }
38
+ }
39
+ const rules = [
40
+ { path: rulesPath, display: "\uB8F0\uC14B" },
41
+ { path: namingRulesPath, display: "\uB124\uC774\uBC0D \uADDC\uCE59" },
42
+ { path: codingConventionRulesPath, display: "\uCF54\uB529 \uCEE8\uBCA4\uC158" }
43
+ ];
44
+ const systemPromptFiles = rules.filter((rule) => fs__default.default.existsSync(rule.path)).map((rule) => `--append-system-prompt-file ${rule.path}`).join(" ");
45
+ const reviewFormOption = fs__default.default.existsSync(reviewFormPath2) ? `--append-system-prompt-file ${reviewFormPath2}` : "";
46
+ const command = `cat ${tempDiffPath} | claude ${modelOption} ${systemPromptFiles} ${reviewFormOption} -p "\uC704 \uADDC\uCE59\uB4E4\uC744 \uCC38\uACE0\uD558\uC5EC \uC774 diff\uB97C \uCF54\uB4DC\uB9AC\uBDF0\uD574\uC918. \uB9AC\uBDF0\uC591\uC2DD\uC5D0 \uB9DE\uCDB0\uC11C \uC791\uC131\uD574\uC918."`;
47
+ if (args.includes("--test")) {
48
+ const safeCommand = command.replace(/"/g, '\\"');
49
+ return `echo "[TEST MODE] Claude \uBA85\uB839\uC5B4\uAC00 \uC2E4\uD589\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4.
50
+
51
+ \uC0DD\uC131\uB420 \uBA85\uB839\uC5B4 \uBBF8\uB9AC\uBCF4\uAE30:
52
+ ${safeCommand}"`;
53
+ }
54
+ return command;
55
+ };
56
+
57
+ exports.createClaudeCommand = createClaudeCommand;
58
+ //# sourceMappingURL=claude-commander.cjs.map
59
+ //# sourceMappingURL=claude-commander.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/common/helper.ts","../../../src/pr-review/claude/claude-commander.ts"],"names":["__dirname","path","fileURLToPath","reviewFormPath","fs"],"mappings":";;;;;;;;;;;;;AAQA,IAAMA,cAAYC,qBAAK,CAAA,OAAA,CAAQC,iBAAc,CAAA,sQAAe,CAAC,CAAA;AAKtD,IAAM,SAAY,GAAAD,qBAAA,CAAK,OAAQ,CAAAD,WAAA,EAAW,wCAAwC,CAAA;AAClF,IAAM,eAAkB,GAAAC,qBAAA,CAAK,OAAQ,CAAAD,WAAA,EAAW,uCAAuC,CAAA;AACvF,IAAM,yBAA4B,GAAAC,qBAAA,CAAK,OAAQ,CAAAD,WAAA,EAAW,6CAA6C,CAAA;AAChFC,qBAAA,CAAK,OAAQ,CAAAD,WAAA,EAAW,sCAAsC;AACtDC,qBAAA,CAAK,OAAQ,CAAAD,WAAA,EAAW,iDAAiD;;;ACb/G,IAAM,IAAO,GAAA,OAAA,CAAQ,IAAK,CAAA,KAAA,CAAM,CAAC,CAAA;AAgBpB,IAAA,mBAAA,GAAsB,CAAC,YAAA,EAAsBG,eAA2B,KAAA;AAEjF,EAAA,IAAI,WAAc,GAAA,EAAA;AAElB,EAAI,IAAA,IAAA,CAAK,QAAS,CAAA,UAAU,CAAG,EAAA;AAG3B,IAAc,WAAA,GAAA,cAAA;AAAA,GACP,MAAA,IAAA,IAAA,CAAK,QAAS,CAAA,SAAS,CAAG,EAAA;AAGjC,IAAc,WAAA,GAAA,eAAA;AAAA,GACX,MAAA;AAEH,IAAM,MAAA,UAAA,GAAa,IAAK,CAAA,OAAA,CAAQ,SAAS,CAAA;AAGzC,IAAA,IAAI,UAAe,KAAA,EAAA,IAAM,IAAK,CAAA,UAAA,GAAa,CAAC,CAAG,EAAA;AAC3C,MAAA,OAAA,CAAQ,KAAK,2JAAmC,CAAA;AAChD,MAAA,WAAA,GAAc,CAAW,QAAA,EAAA,IAAA,CAAK,UAAa,GAAA,CAAC,CAAC,CAAA,CAAA;AAAA,KAC1C,MAAA;AACH,MAAA,OAAA,CAAQ,KAAK,uKAA0C,CAAA;AACvD,MAAc,WAAA,GAAA,gBAAA;AAAA;AAClB;AAKJ,EAAA,MAAM,KAAQ,GAAA;AAAA,IACV,EAAE,IAAA,EAAM,SAAW,EAAA,OAAA,EAAS,cAAK,EAAA;AAAA,IACjC,EAAE,IAAA,EAAM,eAAiB,EAAA,OAAA,EAAS,iCAAS,EAAA;AAAA,IAC3C,EAAE,IAAA,EAAM,yBAA2B,EAAA,OAAA,EAAS,iCAAS;AAAA,GACzD;AAEA,EAAA,MAAM,oBAAoB,KACrB,CAAA,MAAA,CAAO,UAAQC,mBAAG,CAAA,UAAA,CAAW,KAAK,IAAI,CAAC,CACvC,CAAA,GAAA,CAAI,UAAQ,CAA+B,4BAAA,EAAA,IAAA,CAAK,IAAI,CAAE,CAAA,CAAA,CACtD,KAAK,GAAG,CAAA;AAGb,EAAA,MAAM,mBAAmBA,mBAAG,CAAA,UAAA,CAAWD,eAAc,CAC/C,GAAA,CAAA,4BAAA,EAA+BA,eAAc,CAC7C,CAAA,GAAA,EAAA;AAKN,EAAM,MAAA,OAAA,GAAU,OAAO,YAAY,CAAA,UAAA,EAAa,WAAW,CAAI,CAAA,EAAA,iBAAiB,IAAI,gBAAgB,CAAA,kMAAA,CAAA;AAGpG,EAAI,IAAA,IAAA,CAAK,QAAS,CAAA,QAAQ,CAAG,EAAA;AACzB,IAAA,MAAM,WAAc,GAAA,OAAA,CAAQ,OAAQ,CAAA,IAAA,EAAM,KAAK,CAAA;AAE/C,IAAO,OAAA,CAAA;;AAAA;AAAA,EAA+D,WAAW,CAAA,CAAA,CAAA;AAAA;AAGrF,EAAO,OAAA,OAAA;AACX","file":"claude-commander.cjs","sourcesContent":["import { execSync } from 'child_process';\nimport fs from 'fs'\nimport path from 'path';\nimport readline from 'readline';\nimport { fileURLToPath } from 'url';\n\nimport { AIServiceType } from './types';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// 설치된 위치에 맞게 규칙/양식 파일 경로를 계산 (dist에서 src로 이동 등 고려)\n// dist/common/helper.js 가 실행되므로 __dirname은 .../dist/common 입니다.\n// 따라서 ../../src/common/rules 로 이동해야 원본 소스의 규칙 파일을 찾을 수 있습니다.\nexport const rulesPath = path.resolve(__dirname, '../../src/common/rules/review-rules.md');\nexport const namingRulesPath = path.resolve(__dirname, '../../src/common/rules/naming-rule.md');\nexport const codingConventionRulesPath = path.resolve(__dirname, '../../src/common/rules/coding-convention.md');\nexport const reviewFormPath = path.resolve(__dirname, '../../src/common/form/review-form.md');\nexport const reviewFormOneByOnePath = path.resolve(__dirname, '../../src/common/form/review-form-one-by-one.md');\nexport const REPORT_DIR = '.review-report';\nexport const tempDiffPath = 'temp_diff.txt';\nexport const AIServices: AIServiceType[] = ['gemini', 'claude', 'codex'];\nexport const ignoreList = [\n 'package.json',\n '*.yml',\n '*.md',\n '*.lock',\n 'dist/',\n 'node_modules/',\n 'assets/',\n 'public/',\n '*.json',\n '*.yaml',\n '.review-report/' // 생성되는 리포트 폴더도 제외\n];\n\n\nexport function getNextFilePath(dir: string, baseName: string, extension: string) {\n let counter = 1;\n // eslint-disable-next-line no-constant-condition\n while (true) {\n const filePath = path.join(dir, `${baseName}-${counter}${extension}`);\n if (!fs.existsSync(filePath)) {\n return filePath;\n }\n counter++;\n }\n}\n\nexport function deleteFile(filePath: string) {\n if (fs.existsSync(filePath)) {\n fs.unlinkSync(filePath);\n }\n}\n\n\n/**\n * 임시파일 삭제\n */\nexport function deleteTempDiff() {\n deleteFile(tempDiffPath);\n}\n\n\n/**\n * 리뷰 결과 폴더 생성\n */\nexport function createReportDirectory() {\n if (!fs.existsSync(REPORT_DIR)) {\n fs.mkdirSync(REPORT_DIR, { recursive: true });\n }\n}\n\n\n/**\n * 현재 시간 문자열 생성\n */\nexport function getNowString() {\n const now = new Date();\n const YYYY = now.getFullYear();\n const MM = String(now.getMonth() + 1).padStart(2, '0');\n const DD = String(now.getDate()).padStart(2, '0');\n const HH = String(now.getHours()).padStart(2, '0');\n const mm = String(now.getMinutes()).padStart(2, '0');\n const ss = String(now.getSeconds()).padStart(2, '0');\n\n return `${YYYY}-${MM}-${DD}_${HH}-${mm}-${ss}`;\n}\n\nexport function getGitDiffFilter() {\n\n // 1. 리뷰 대상 파일 확장자 정의\n const includeExtensions = ['*.ts', '*.tsx', '*.js', '*.jsx'];\n\n // ignoreList 를 import 하여 재사용하여 작성한다.\n const excludePatterns = ignoreList.map(item => `:(exclude)${item}`);\n\n // const excludePatterns = [':(exclude)*.lock', ':(exclude)dist/', ':(exclude)*.md'];\n\n // 2. 변경된 파일 목록 가져오기 (각 패턴을 따옴표로 감싸서 쉘 에러 방지)\n const quote = (pattern: string) => `\"${pattern}\"`;\n const includeParams = includeExtensions.map(quote).join(' ');\n const excludeParams = excludePatterns.map(quote).join(' ');\n\n return { includeParams, excludeParams };\n\n\n}\n\nexport function openReport(reportPath: string) {\n // 브라우저 열기\n try {\n execSync(`open -a \"Google Chrome\" \"${path.resolve(reportPath)}\"`);\n console.log(`🚀 브라우저에서 리포트를 열었습니다.`);\n } catch (e) {\n console.error('⚠️ 브라우저 열기 실패:', e);\n }\n}\n\n\nexport function getDiffArgs() {\n const args = process.argv.slice(2);\n const commitIndex = args.indexOf('--commit');\n const { includeParams, excludeParams } = getGitDiffFilter();\n\n let diffArgs = '';\n\n if (commitIndex !== -1) {\n // 특정 커밋 (및 이전 n개) 리뷰\n const commitHash = args[commitIndex + 1];\n if (!commitHash) {\n console.error('❌ 커밋 해시가 제공되지 않았습니다.');\n process.exit(1);\n }\n\n // n값 확인 (optional)\n const nextArg = args[commitIndex + 2];\n let n = 0;\n if (nextArg && !nextArg.startsWith('--')) {\n n = parseInt(nextArg, 10);\n if (isNaN(n)) {\n n = 0;\n }\n }\n\n console.log(`ℹ️ 커밋 '${commitHash}' ${n > 0 ? ` 포함 총 ${n + 1}개의 커밋` : ''}을 리뷰합니다...`);\n diffArgs = `${commitHash}~${n + 1} ${commitHash}`;\n } else {\n // 기본 모드:\n // 1. Unstaged 변경사항 확인\n try {\n const check = execSync(`git diff --name-only -- ${includeParams} ${excludeParams}`).toString();\n if (!check.trim()) {\n console.log('ℹ️ Unstaged 변경사항이 없습니다. 마지막 커밋(HEAD)을 리뷰합니다...');\n diffArgs = 'HEAD~1 HEAD';\n }\n } catch {\n // git diff 실패시 무시\n }\n }\n\n return diffArgs;\n}\n\n// export const ServiceType = {\n// GEMINI: 'gemini',\n// CLAUDE: 'claude',\n// CODEX: 'codex'\n\n// }\n\n/**\n * AI 서비스 선택\n */\nexport function selectAIService() {\n const args = process.argv.slice(2);\n const serviceIndex = args.indexOf('--service');\n const service = args[serviceIndex + 1];\n if (!service) {\n console.error('❌ 서비스가 선택되지 않았습니다.');\n process.exit(1);\n }\n\n return service;\n}\n\n/**\n * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.\n */\nexport async function showSelectionAIService(): Promise<AIServiceType> {\n\n let selectedIndex = 0;\n\n // Use readline to handle keypresses\n // 키 입력을 처리하기 위해 readline 인터페이스 사용\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n terminal: true\n });\n\n let firstRender = true;\n\n // Hide cursor\n process.stdout.write('\\u001b[?25l');\n\n const render = () => {\n if (!firstRender) {\n // Move cursor back to the starting line of the selection UI\n // We print (1 question line + services.length lines)\n // 선택 UI의 시작 라인으로 커서 이동 (질문 1줄 + 서비스 목록 N줄)\n readline.moveCursor(process.stdout, 0, -(AIServices.length + 1));\n }\n firstRender = false;\n\n // Clear everything from cursor down to avoid ghosting/overlaps\n // 잔상이나 겹침을 방지하기 위해 커서 위치부터 아래쪽 모두 지움\n readline.clearScreenDown(process.stdout);\n\n process.stdout.write('🤖 AI 서비스를 선택해주세요 (\\u001b[33m↑↓ 방향키\\u001b[0m 이동, \\u001b[33mEnter\\u001b[0m 선택):\\n');\n AIServices.forEach((service, index) => {\n if (index === selectedIndex) {\n process.stdout.write(` \\u001b[36m>\\u001b[0m \\u001b[36m◉\\u001b[0m \\u001b[1m${service}\\u001b[0m\\n`);\n } else {\n process.stdout.write(` ◯ ${service}\\n`);\n }\n });\n };\n\n render();\n\n return new Promise((resolve) => {\n const onData = (data: Buffer) => {\n const key = data.toString();\n if (key === '\\u0003') { // Ctrl+C\n process.stdout.write('\\u001b[?25h'); // Show cursor\n process.exit(0);\n }\n if (key === '\\x1b[A') { // Up arrow\n selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;\n render();\n } else if (key === '\\x1b[B') { // Down arrow\n selectedIndex = (selectedIndex + 1) % AIServices.length;\n render();\n } else if (key === '\\r' || key === '\\n') { // Enter\n process.stdin.removeListener('data', onData);\n process.stdin.setRawMode(false);\n process.stdin.pause();\n rl.close();\n\n // Show cursor\n process.stdout.write('\\u001b[?25h');\n\n console.log(`\\n✅ \\u001b[32m${AIServices[selectedIndex]}\\u001b[0m 서비스가 선택되었습니다.\\n`);\n const result = AIServices[selectedIndex];\n if (result) {\n resolve(result);\n }\n }\n };\n\n process.stdin.setRawMode(true);\n process.stdin.resume();\n process.stdin.on('data', onData);\n });\n}\n\n","import fs from 'fs';\n\nimport { codingConventionRulesPath, namingRulesPath, rulesPath } from \"../../common/helper\";\n\nconst args = process.argv.slice(2);\n\n/**\n * @description\n * Claude Code CLI를 활용한 코드 리뷰 명령어를 생성합니다.\n * \n * 모델 별칭(Alias):\n * 1. sonnet (기본값, 균형 잡힌 성능) - 일반적인 코드 리뷰에 적합\n * 2. opus (최고 성능) - 복잡한 분석, 긴 컨텍스트 처리에 강점\n * 3. haiku (빠른 속도) - 오타 수정, 간단한 리뷰에 적합\n * \n * Claude CLI 기능 활용:\n * - `-p`: 비대화형 모드(print mode)로 실행하여 결과를 stdout으로 출력\n * - `--append-system-prompt-file`: 룰셋/리뷰양식 파일을 시스템 프롬프트에 추가 (Claude가 \"지침\"으로 인식)\n * - `cat diff | claude`: diff 내용을 stdin으로 파이프하여 \"리뷰 대상\"으로 전달\n */\nexport const createClaudeCommand = (tempDiffPath: string, reviewFormPath: string) => {\n\n let modelOption = '';\n\n if (args.includes('--review')) {\n // 최고 성능 모델 (opus)\n // 복잡한 코드 분석, 긴 컨텍스트 처리에 강점\n modelOption = '--model opus';\n } else if (args.includes('--flash')) {\n // 속도 우선 모델 (haiku)\n // 오타 수정, 간단한 리뷰에 적합\n modelOption = '--model haiku';\n } else {\n // 사용자 직접 지정\n const modelIndex = args.indexOf('--model');\n // args[modelIndex + 1] 에는 사용자가 지정한 모델명이 들어갑니다.\n // 예: sonnet, opus, haiku, claude-sonnet-4-5-20250929 등\n if (modelIndex !== -1 && args[modelIndex + 1]) {\n console.warn('⚠️ 지정한 모델이 없는 경우, 에러가 발생하니 주의하세요.')\n modelOption = `--model ${args[modelIndex + 1]}`;\n } else {\n console.warn('⚠️ 모델이 지정되지 않았습니다. 기본 모델인 sonnet을 사용합니다.')\n modelOption = '--model sonnet';\n }\n }\n\n // 룰셋 파일들을 --append-system-prompt-file 로 시스템 프롬프트에 추가\n // Claude가 \"지침/규칙\"으로 인식하도록 시스템 프롬프트 레벨에 주입\n const rules = [\n { path: rulesPath, display: '룰셋' },\n { path: namingRulesPath, display: '네이밍 규칙' },\n { path: codingConventionRulesPath, display: '코딩 컨벤션' }\n ];\n\n const systemPromptFiles = rules\n .filter(rule => fs.existsSync(rule.path))\n .map(rule => `--append-system-prompt-file ${rule.path}`)\n .join(' ');\n\n // 리뷰 양식도 시스템 프롬프트에 추가 (출력 포맷 지침)\n const reviewFormOption = fs.existsSync(reviewFormPath)\n ? `--append-system-prompt-file ${reviewFormPath}`\n : '';\n\n // Claude CLI 명령어 생성\n // cat으로 diff 내용을 stdin 파이프 → Claude가 \"리뷰 대상 데이터\"로 인식\n // -p: 비대화형 모드로 실행\n const command = `cat ${tempDiffPath} | claude ${modelOption} ${systemPromptFiles} ${reviewFormOption} -p \"위 규칙들을 참고하여 이 diff를 코드리뷰해줘. 리뷰양식에 맞춰서 작성해줘.\"`;\n\n // test mode\n if (args.includes('--test')) {\n const safeCommand = command.replace(/\"/g, '\\\\\"');\n\n return `echo \"[TEST MODE] Claude 명령어가 실행되지 않았습니다.\\n\\n생성될 명령어 미리보기:\\n${safeCommand}\"`;\n }\n\n return command;\n}"]}
@@ -0,0 +1,17 @@
1
+ /**
2
+ * @description
3
+ * Claude Code CLI를 활용한 코드 리뷰 명령어를 생성합니다.
4
+ *
5
+ * 모델 별칭(Alias):
6
+ * 1. sonnet (기본값, 균형 잡힌 성능) - 일반적인 코드 리뷰에 적합
7
+ * 2. opus (최고 성능) - 복잡한 분석, 긴 컨텍스트 처리에 강점
8
+ * 3. haiku (빠른 속도) - 오타 수정, 간단한 리뷰에 적합
9
+ *
10
+ * Claude CLI 기능 활용:
11
+ * - `-p`: 비대화형 모드(print mode)로 실행하여 결과를 stdout으로 출력
12
+ * - `--append-system-prompt-file`: 룰셋/리뷰양식 파일을 시스템 프롬프트에 추가 (Claude가 "지침"으로 인식)
13
+ * - `cat diff | claude`: diff 내용을 stdin으로 파이프하여 "리뷰 대상"으로 전달
14
+ */
15
+ declare const createClaudeCommand: (tempDiffPath: string, reviewFormPath: string) => string;
16
+
17
+ export { createClaudeCommand };
@@ -0,0 +1,17 @@
1
+ /**
2
+ * @description
3
+ * Claude Code CLI를 활용한 코드 리뷰 명령어를 생성합니다.
4
+ *
5
+ * 모델 별칭(Alias):
6
+ * 1. sonnet (기본값, 균형 잡힌 성능) - 일반적인 코드 리뷰에 적합
7
+ * 2. opus (최고 성능) - 복잡한 분석, 긴 컨텍스트 처리에 강점
8
+ * 3. haiku (빠른 속도) - 오타 수정, 간단한 리뷰에 적합
9
+ *
10
+ * Claude CLI 기능 활용:
11
+ * - `-p`: 비대화형 모드(print mode)로 실행하여 결과를 stdout으로 출력
12
+ * - `--append-system-prompt-file`: 룰셋/리뷰양식 파일을 시스템 프롬프트에 추가 (Claude가 "지침"으로 인식)
13
+ * - `cat diff | claude`: diff 내용을 stdin으로 파이프하여 "리뷰 대상"으로 전달
14
+ */
15
+ declare const createClaudeCommand: (tempDiffPath: string, reviewFormPath: string) => string;
16
+
17
+ export { createClaudeCommand };
@@ -0,0 +1,51 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { fileURLToPath } from 'url';
4
+
5
+ // src/pr-review/claude/claude-commander.ts
6
+ var __dirname = path.dirname(fileURLToPath(import.meta.url));
7
+ var rulesPath = path.resolve(__dirname, "../../src/common/rules/review-rules.md");
8
+ var namingRulesPath = path.resolve(__dirname, "../../src/common/rules/naming-rule.md");
9
+ var codingConventionRulesPath = path.resolve(__dirname, "../../src/common/rules/coding-convention.md");
10
+ path.resolve(__dirname, "../../src/common/form/review-form.md");
11
+ path.resolve(__dirname, "../../src/common/form/review-form-one-by-one.md");
12
+
13
+ // src/pr-review/claude/claude-commander.ts
14
+ var args = process.argv.slice(2);
15
+ var createClaudeCommand = (tempDiffPath, reviewFormPath2) => {
16
+ let modelOption = "";
17
+ if (args.includes("--review")) {
18
+ modelOption = "--model opus";
19
+ } else if (args.includes("--flash")) {
20
+ modelOption = "--model haiku";
21
+ } else {
22
+ const modelIndex = args.indexOf("--model");
23
+ if (modelIndex !== -1 && args[modelIndex + 1]) {
24
+ console.warn("\u26A0\uFE0F \uC9C0\uC815\uD55C \uBAA8\uB378\uC774 \uC5C6\uB294 \uACBD\uC6B0, \uC5D0\uB7EC\uAC00 \uBC1C\uC0DD\uD558\uB2C8 \uC8FC\uC758\uD558\uC138\uC694.");
25
+ modelOption = `--model ${args[modelIndex + 1]}`;
26
+ } else {
27
+ console.warn("\u26A0\uFE0F \uBAA8\uB378\uC774 \uC9C0\uC815\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4. \uAE30\uBCF8 \uBAA8\uB378\uC778 sonnet\uC744 \uC0AC\uC6A9\uD569\uB2C8\uB2E4.");
28
+ modelOption = "--model sonnet";
29
+ }
30
+ }
31
+ const rules = [
32
+ { path: rulesPath, display: "\uB8F0\uC14B" },
33
+ { path: namingRulesPath, display: "\uB124\uC774\uBC0D \uADDC\uCE59" },
34
+ { path: codingConventionRulesPath, display: "\uCF54\uB529 \uCEE8\uBCA4\uC158" }
35
+ ];
36
+ const systemPromptFiles = rules.filter((rule) => fs.existsSync(rule.path)).map((rule) => `--append-system-prompt-file ${rule.path}`).join(" ");
37
+ const reviewFormOption = fs.existsSync(reviewFormPath2) ? `--append-system-prompt-file ${reviewFormPath2}` : "";
38
+ const command = `cat ${tempDiffPath} | claude ${modelOption} ${systemPromptFiles} ${reviewFormOption} -p "\uC704 \uADDC\uCE59\uB4E4\uC744 \uCC38\uACE0\uD558\uC5EC \uC774 diff\uB97C \uCF54\uB4DC\uB9AC\uBDF0\uD574\uC918. \uB9AC\uBDF0\uC591\uC2DD\uC5D0 \uB9DE\uCDB0\uC11C \uC791\uC131\uD574\uC918."`;
39
+ if (args.includes("--test")) {
40
+ const safeCommand = command.replace(/"/g, '\\"');
41
+ return `echo "[TEST MODE] Claude \uBA85\uB839\uC5B4\uAC00 \uC2E4\uD589\uB418\uC9C0 \uC54A\uC558\uC2B5\uB2C8\uB2E4.
42
+
43
+ \uC0DD\uC131\uB420 \uBA85\uB839\uC5B4 \uBBF8\uB9AC\uBCF4\uAE30:
44
+ ${safeCommand}"`;
45
+ }
46
+ return command;
47
+ };
48
+
49
+ export { createClaudeCommand };
50
+ //# sourceMappingURL=claude-commander.js.map
51
+ //# sourceMappingURL=claude-commander.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/common/helper.ts","../../../src/pr-review/claude/claude-commander.ts"],"names":["reviewFormPath"],"mappings":";;;;;AAQA,IAAM,YAAY,IAAK,CAAA,OAAA,CAAQ,aAAc,CAAA,MAAA,CAAA,IAAA,CAAY,GAAG,CAAC,CAAA;AAKtD,IAAM,SAAY,GAAA,IAAA,CAAK,OAAQ,CAAA,SAAA,EAAW,wCAAwC,CAAA;AAClF,IAAM,eAAkB,GAAA,IAAA,CAAK,OAAQ,CAAA,SAAA,EAAW,uCAAuC,CAAA;AACvF,IAAM,yBAA4B,GAAA,IAAA,CAAK,OAAQ,CAAA,SAAA,EAAW,6CAA6C,CAAA;AAChF,IAAA,CAAK,OAAQ,CAAA,SAAA,EAAW,sCAAsC;AACtD,IAAA,CAAK,OAAQ,CAAA,SAAA,EAAW,iDAAiD;;;ACb/G,IAAM,IAAO,GAAA,OAAA,CAAQ,IAAK,CAAA,KAAA,CAAM,CAAC,CAAA;AAgBpB,IAAA,mBAAA,GAAsB,CAAC,YAAA,EAAsBA,eAA2B,KAAA;AAEjF,EAAA,IAAI,WAAc,GAAA,EAAA;AAElB,EAAI,IAAA,IAAA,CAAK,QAAS,CAAA,UAAU,CAAG,EAAA;AAG3B,IAAc,WAAA,GAAA,cAAA;AAAA,GACP,MAAA,IAAA,IAAA,CAAK,QAAS,CAAA,SAAS,CAAG,EAAA;AAGjC,IAAc,WAAA,GAAA,eAAA;AAAA,GACX,MAAA;AAEH,IAAM,MAAA,UAAA,GAAa,IAAK,CAAA,OAAA,CAAQ,SAAS,CAAA;AAGzC,IAAA,IAAI,UAAe,KAAA,EAAA,IAAM,IAAK,CAAA,UAAA,GAAa,CAAC,CAAG,EAAA;AAC3C,MAAA,OAAA,CAAQ,KAAK,2JAAmC,CAAA;AAChD,MAAA,WAAA,GAAc,CAAW,QAAA,EAAA,IAAA,CAAK,UAAa,GAAA,CAAC,CAAC,CAAA,CAAA;AAAA,KAC1C,MAAA;AACH,MAAA,OAAA,CAAQ,KAAK,uKAA0C,CAAA;AACvD,MAAc,WAAA,GAAA,gBAAA;AAAA;AAClB;AAKJ,EAAA,MAAM,KAAQ,GAAA;AAAA,IACV,EAAE,IAAA,EAAM,SAAW,EAAA,OAAA,EAAS,cAAK,EAAA;AAAA,IACjC,EAAE,IAAA,EAAM,eAAiB,EAAA,OAAA,EAAS,iCAAS,EAAA;AAAA,IAC3C,EAAE,IAAA,EAAM,yBAA2B,EAAA,OAAA,EAAS,iCAAS;AAAA,GACzD;AAEA,EAAA,MAAM,oBAAoB,KACrB,CAAA,MAAA,CAAO,UAAQ,EAAG,CAAA,UAAA,CAAW,KAAK,IAAI,CAAC,CACvC,CAAA,GAAA,CAAI,UAAQ,CAA+B,4BAAA,EAAA,IAAA,CAAK,IAAI,CAAE,CAAA,CAAA,CACtD,KAAK,GAAG,CAAA;AAGb,EAAA,MAAM,mBAAmB,EAAG,CAAA,UAAA,CAAWA,eAAc,CAC/C,GAAA,CAAA,4BAAA,EAA+BA,eAAc,CAC7C,CAAA,GAAA,EAAA;AAKN,EAAM,MAAA,OAAA,GAAU,OAAO,YAAY,CAAA,UAAA,EAAa,WAAW,CAAI,CAAA,EAAA,iBAAiB,IAAI,gBAAgB,CAAA,kMAAA,CAAA;AAGpG,EAAI,IAAA,IAAA,CAAK,QAAS,CAAA,QAAQ,CAAG,EAAA;AACzB,IAAA,MAAM,WAAc,GAAA,OAAA,CAAQ,OAAQ,CAAA,IAAA,EAAM,KAAK,CAAA;AAE/C,IAAO,OAAA,CAAA;;AAAA;AAAA,EAA+D,WAAW,CAAA,CAAA,CAAA;AAAA;AAGrF,EAAO,OAAA,OAAA;AACX","file":"claude-commander.js","sourcesContent":["import { execSync } from 'child_process';\nimport fs from 'fs'\nimport path from 'path';\nimport readline from 'readline';\nimport { fileURLToPath } from 'url';\n\nimport { AIServiceType } from './types';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// 설치된 위치에 맞게 규칙/양식 파일 경로를 계산 (dist에서 src로 이동 등 고려)\n// dist/common/helper.js 가 실행되므로 __dirname은 .../dist/common 입니다.\n// 따라서 ../../src/common/rules 로 이동해야 원본 소스의 규칙 파일을 찾을 수 있습니다.\nexport const rulesPath = path.resolve(__dirname, '../../src/common/rules/review-rules.md');\nexport const namingRulesPath = path.resolve(__dirname, '../../src/common/rules/naming-rule.md');\nexport const codingConventionRulesPath = path.resolve(__dirname, '../../src/common/rules/coding-convention.md');\nexport const reviewFormPath = path.resolve(__dirname, '../../src/common/form/review-form.md');\nexport const reviewFormOneByOnePath = path.resolve(__dirname, '../../src/common/form/review-form-one-by-one.md');\nexport const REPORT_DIR = '.review-report';\nexport const tempDiffPath = 'temp_diff.txt';\nexport const AIServices: AIServiceType[] = ['gemini', 'claude', 'codex'];\nexport const ignoreList = [\n 'package.json',\n '*.yml',\n '*.md',\n '*.lock',\n 'dist/',\n 'node_modules/',\n 'assets/',\n 'public/',\n '*.json',\n '*.yaml',\n '.review-report/' // 생성되는 리포트 폴더도 제외\n];\n\n\nexport function getNextFilePath(dir: string, baseName: string, extension: string) {\n let counter = 1;\n // eslint-disable-next-line no-constant-condition\n while (true) {\n const filePath = path.join(dir, `${baseName}-${counter}${extension}`);\n if (!fs.existsSync(filePath)) {\n return filePath;\n }\n counter++;\n }\n}\n\nexport function deleteFile(filePath: string) {\n if (fs.existsSync(filePath)) {\n fs.unlinkSync(filePath);\n }\n}\n\n\n/**\n * 임시파일 삭제\n */\nexport function deleteTempDiff() {\n deleteFile(tempDiffPath);\n}\n\n\n/**\n * 리뷰 결과 폴더 생성\n */\nexport function createReportDirectory() {\n if (!fs.existsSync(REPORT_DIR)) {\n fs.mkdirSync(REPORT_DIR, { recursive: true });\n }\n}\n\n\n/**\n * 현재 시간 문자열 생성\n */\nexport function getNowString() {\n const now = new Date();\n const YYYY = now.getFullYear();\n const MM = String(now.getMonth() + 1).padStart(2, '0');\n const DD = String(now.getDate()).padStart(2, '0');\n const HH = String(now.getHours()).padStart(2, '0');\n const mm = String(now.getMinutes()).padStart(2, '0');\n const ss = String(now.getSeconds()).padStart(2, '0');\n\n return `${YYYY}-${MM}-${DD}_${HH}-${mm}-${ss}`;\n}\n\nexport function getGitDiffFilter() {\n\n // 1. 리뷰 대상 파일 확장자 정의\n const includeExtensions = ['*.ts', '*.tsx', '*.js', '*.jsx'];\n\n // ignoreList 를 import 하여 재사용하여 작성한다.\n const excludePatterns = ignoreList.map(item => `:(exclude)${item}`);\n\n // const excludePatterns = [':(exclude)*.lock', ':(exclude)dist/', ':(exclude)*.md'];\n\n // 2. 변경된 파일 목록 가져오기 (각 패턴을 따옴표로 감싸서 쉘 에러 방지)\n const quote = (pattern: string) => `\"${pattern}\"`;\n const includeParams = includeExtensions.map(quote).join(' ');\n const excludeParams = excludePatterns.map(quote).join(' ');\n\n return { includeParams, excludeParams };\n\n\n}\n\nexport function openReport(reportPath: string) {\n // 브라우저 열기\n try {\n execSync(`open -a \"Google Chrome\" \"${path.resolve(reportPath)}\"`);\n console.log(`🚀 브라우저에서 리포트를 열었습니다.`);\n } catch (e) {\n console.error('⚠️ 브라우저 열기 실패:', e);\n }\n}\n\n\nexport function getDiffArgs() {\n const args = process.argv.slice(2);\n const commitIndex = args.indexOf('--commit');\n const { includeParams, excludeParams } = getGitDiffFilter();\n\n let diffArgs = '';\n\n if (commitIndex !== -1) {\n // 특정 커밋 (및 이전 n개) 리뷰\n const commitHash = args[commitIndex + 1];\n if (!commitHash) {\n console.error('❌ 커밋 해시가 제공되지 않았습니다.');\n process.exit(1);\n }\n\n // n값 확인 (optional)\n const nextArg = args[commitIndex + 2];\n let n = 0;\n if (nextArg && !nextArg.startsWith('--')) {\n n = parseInt(nextArg, 10);\n if (isNaN(n)) {\n n = 0;\n }\n }\n\n console.log(`ℹ️ 커밋 '${commitHash}' ${n > 0 ? ` 포함 총 ${n + 1}개의 커밋` : ''}을 리뷰합니다...`);\n diffArgs = `${commitHash}~${n + 1} ${commitHash}`;\n } else {\n // 기본 모드:\n // 1. Unstaged 변경사항 확인\n try {\n const check = execSync(`git diff --name-only -- ${includeParams} ${excludeParams}`).toString();\n if (!check.trim()) {\n console.log('ℹ️ Unstaged 변경사항이 없습니다. 마지막 커밋(HEAD)을 리뷰합니다...');\n diffArgs = 'HEAD~1 HEAD';\n }\n } catch {\n // git diff 실패시 무시\n }\n }\n\n return diffArgs;\n}\n\n// export const ServiceType = {\n// GEMINI: 'gemini',\n// CLAUDE: 'claude',\n// CODEX: 'codex'\n\n// }\n\n/**\n * AI 서비스 선택\n */\nexport function selectAIService() {\n const args = process.argv.slice(2);\n const serviceIndex = args.indexOf('--service');\n const service = args[serviceIndex + 1];\n if (!service) {\n console.error('❌ 서비스가 선택되지 않았습니다.');\n process.exit(1);\n }\n\n return service;\n}\n\n/**\n * 터미널에서 라디오 버튼 형태로 AI 서비스를 선택합니다.\n */\nexport async function showSelectionAIService(): Promise<AIServiceType> {\n\n let selectedIndex = 0;\n\n // Use readline to handle keypresses\n // 키 입력을 처리하기 위해 readline 인터페이스 사용\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n terminal: true\n });\n\n let firstRender = true;\n\n // Hide cursor\n process.stdout.write('\\u001b[?25l');\n\n const render = () => {\n if (!firstRender) {\n // Move cursor back to the starting line of the selection UI\n // We print (1 question line + services.length lines)\n // 선택 UI의 시작 라인으로 커서 이동 (질문 1줄 + 서비스 목록 N줄)\n readline.moveCursor(process.stdout, 0, -(AIServices.length + 1));\n }\n firstRender = false;\n\n // Clear everything from cursor down to avoid ghosting/overlaps\n // 잔상이나 겹침을 방지하기 위해 커서 위치부터 아래쪽 모두 지움\n readline.clearScreenDown(process.stdout);\n\n process.stdout.write('🤖 AI 서비스를 선택해주세요 (\\u001b[33m↑↓ 방향키\\u001b[0m 이동, \\u001b[33mEnter\\u001b[0m 선택):\\n');\n AIServices.forEach((service, index) => {\n if (index === selectedIndex) {\n process.stdout.write(` \\u001b[36m>\\u001b[0m \\u001b[36m◉\\u001b[0m \\u001b[1m${service}\\u001b[0m\\n`);\n } else {\n process.stdout.write(` ◯ ${service}\\n`);\n }\n });\n };\n\n render();\n\n return new Promise((resolve) => {\n const onData = (data: Buffer) => {\n const key = data.toString();\n if (key === '\\u0003') { // Ctrl+C\n process.stdout.write('\\u001b[?25h'); // Show cursor\n process.exit(0);\n }\n if (key === '\\x1b[A') { // Up arrow\n selectedIndex = (selectedIndex - 1 + AIServices.length) % AIServices.length;\n render();\n } else if (key === '\\x1b[B') { // Down arrow\n selectedIndex = (selectedIndex + 1) % AIServices.length;\n render();\n } else if (key === '\\r' || key === '\\n') { // Enter\n process.stdin.removeListener('data', onData);\n process.stdin.setRawMode(false);\n process.stdin.pause();\n rl.close();\n\n // Show cursor\n process.stdout.write('\\u001b[?25h');\n\n console.log(`\\n✅ \\u001b[32m${AIServices[selectedIndex]}\\u001b[0m 서비스가 선택되었습니다.\\n`);\n const result = AIServices[selectedIndex];\n if (result) {\n resolve(result);\n }\n }\n };\n\n process.stdin.setRawMode(true);\n process.stdin.resume();\n process.stdin.on('data', onData);\n });\n}\n\n","import fs from 'fs';\n\nimport { codingConventionRulesPath, namingRulesPath, rulesPath } from \"../../common/helper\";\n\nconst args = process.argv.slice(2);\n\n/**\n * @description\n * Claude Code CLI를 활용한 코드 리뷰 명령어를 생성합니다.\n * \n * 모델 별칭(Alias):\n * 1. sonnet (기본값, 균형 잡힌 성능) - 일반적인 코드 리뷰에 적합\n * 2. opus (최고 성능) - 복잡한 분석, 긴 컨텍스트 처리에 강점\n * 3. haiku (빠른 속도) - 오타 수정, 간단한 리뷰에 적합\n * \n * Claude CLI 기능 활용:\n * - `-p`: 비대화형 모드(print mode)로 실행하여 결과를 stdout으로 출력\n * - `--append-system-prompt-file`: 룰셋/리뷰양식 파일을 시스템 프롬프트에 추가 (Claude가 \"지침\"으로 인식)\n * - `cat diff | claude`: diff 내용을 stdin으로 파이프하여 \"리뷰 대상\"으로 전달\n */\nexport const createClaudeCommand = (tempDiffPath: string, reviewFormPath: string) => {\n\n let modelOption = '';\n\n if (args.includes('--review')) {\n // 최고 성능 모델 (opus)\n // 복잡한 코드 분석, 긴 컨텍스트 처리에 강점\n modelOption = '--model opus';\n } else if (args.includes('--flash')) {\n // 속도 우선 모델 (haiku)\n // 오타 수정, 간단한 리뷰에 적합\n modelOption = '--model haiku';\n } else {\n // 사용자 직접 지정\n const modelIndex = args.indexOf('--model');\n // args[modelIndex + 1] 에는 사용자가 지정한 모델명이 들어갑니다.\n // 예: sonnet, opus, haiku, claude-sonnet-4-5-20250929 등\n if (modelIndex !== -1 && args[modelIndex + 1]) {\n console.warn('⚠️ 지정한 모델이 없는 경우, 에러가 발생하니 주의하세요.')\n modelOption = `--model ${args[modelIndex + 1]}`;\n } else {\n console.warn('⚠️ 모델이 지정되지 않았습니다. 기본 모델인 sonnet을 사용합니다.')\n modelOption = '--model sonnet';\n }\n }\n\n // 룰셋 파일들을 --append-system-prompt-file 로 시스템 프롬프트에 추가\n // Claude가 \"지침/규칙\"으로 인식하도록 시스템 프롬프트 레벨에 주입\n const rules = [\n { path: rulesPath, display: '룰셋' },\n { path: namingRulesPath, display: '네이밍 규칙' },\n { path: codingConventionRulesPath, display: '코딩 컨벤션' }\n ];\n\n const systemPromptFiles = rules\n .filter(rule => fs.existsSync(rule.path))\n .map(rule => `--append-system-prompt-file ${rule.path}`)\n .join(' ');\n\n // 리뷰 양식도 시스템 프롬프트에 추가 (출력 포맷 지침)\n const reviewFormOption = fs.existsSync(reviewFormPath)\n ? `--append-system-prompt-file ${reviewFormPath}`\n : '';\n\n // Claude CLI 명령어 생성\n // cat으로 diff 내용을 stdin 파이프 → Claude가 \"리뷰 대상 데이터\"로 인식\n // -p: 비대화형 모드로 실행\n const command = `cat ${tempDiffPath} | claude ${modelOption} ${systemPromptFiles} ${reviewFormOption} -p \"위 규칙들을 참고하여 이 diff를 코드리뷰해줘. 리뷰양식에 맞춰서 작성해줘.\"`;\n\n // test mode\n if (args.includes('--test')) {\n const safeCommand = command.replace(/\"/g, '\\\\\"');\n\n return `echo \"[TEST MODE] Claude 명령어가 실행되지 않았습니다.\\n\\n생성될 명령어 미리보기:\\n${safeCommand}\"`;\n }\n\n return command;\n}"]}