coderev-cli 1.0.10 โ 1.0.11
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.
- package/package.json +1 -1
- package/src/cli.js +77 -77
package/package.json
CHANGED
package/src/cli.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
1
|
+
๏ปฟ#!/usr/bin/env node
|
|
2
2
|
|
|
3
3
|
const { program } = require('commander');
|
|
4
4
|
const chalk = require('chalk');
|
|
@@ -71,7 +71,7 @@ program
|
|
|
71
71
|
return;
|
|
72
72
|
}
|
|
73
73
|
|
|
74
|
-
console.error(chalk.bold(`\n
|
|
74
|
+
console.error(chalk.bold(`\n้ฆๆต Found ${prList.length} open PRs in ${prRef.owner}/${prRef.repo}:`));
|
|
75
75
|
for (const pr of prList) {
|
|
76
76
|
console.error(` #${pr.number} ${pr.title} (${pr.draft ? 'draft' : 'open'})`);
|
|
77
77
|
}
|
|
@@ -79,7 +79,7 @@ program
|
|
|
79
79
|
|
|
80
80
|
const results = [];
|
|
81
81
|
for (const pr of prList) {
|
|
82
|
-
console.error(chalk.blue(
|
|
82
|
+
console.error(chalk.blue(`้ซ?Reviewing PR #${pr.number}...`));
|
|
83
83
|
const fullRef = { owner: prRef.owner, repo: prRef.repo, pr: pr.number };
|
|
84
84
|
try {
|
|
85
85
|
const prDiff = await fetchPrDiff(fullRef, token);
|
|
@@ -89,7 +89,7 @@ program
|
|
|
89
89
|
if (options.post) {
|
|
90
90
|
const md = formatMarkdown(result);
|
|
91
91
|
await postPrComment(fullRef, md, token);
|
|
92
|
-
console.error(chalk.green(`
|
|
92
|
+
console.error(chalk.green(` ้?#${pr.number} reviewed & posted`));
|
|
93
93
|
} else {
|
|
94
94
|
const scoreColor = result.score >= 80 ? chalk.green : result.score >= 50 ? chalk.yellow : chalk.red;
|
|
95
95
|
const scoreStr = scoreColor(`${result.score}/100`);
|
|
@@ -97,7 +97,7 @@ program
|
|
|
97
97
|
console.error(` ${scoreStr} (${issueCount} issues) - ${result.summary || ''}`);
|
|
98
98
|
}
|
|
99
99
|
} catch (err) {
|
|
100
|
-
console.error(chalk.red(`
|
|
100
|
+
console.error(chalk.red(` ้?#${pr.number}: ${err.message}`));
|
|
101
101
|
}
|
|
102
102
|
}
|
|
103
103
|
|
|
@@ -105,7 +105,7 @@ program
|
|
|
105
105
|
const scores = results.filter(r => r.result).map(r => r.result.score);
|
|
106
106
|
if (scores.length > 0) {
|
|
107
107
|
const avg = (scores.reduce((a, b) => a + b, 0) / scores.length).toFixed(1);
|
|
108
|
-
console.error(chalk.bold(`\n
|
|
108
|
+
console.error(chalk.bold(`\n้ฆๆณ Batch Summary: ${results.length}/${prList.length} reviewed, avg score: ${avg}`));
|
|
109
109
|
}
|
|
110
110
|
|
|
111
111
|
if (options.output === 'json') {
|
|
@@ -124,35 +124,35 @@ program
|
|
|
124
124
|
const { resolveMrRef, fetchMrDiff } = require('./gitlab');
|
|
125
125
|
const glRef = resolveMrRef(options.gl, options.repo);
|
|
126
126
|
const glToken = options.gitlabToken || process.env.GITLAB_TOKEN;
|
|
127
|
-
console.error(chalk.blue(
|
|
127
|
+
console.error(chalk.blue(`้ซ?Fetching GitLab MR ${glRef.owner}/${glRef.repo}!${glRef.mr}...`));
|
|
128
128
|
diff = await fetchMrDiff(glRef, glToken);
|
|
129
|
-
console.error(chalk.green(
|
|
129
|
+
console.error(chalk.green(`้?Diff fetched (${diff.length} chars)`));
|
|
130
130
|
} else if (options.gee) {
|
|
131
131
|
const { resolvePrRef: resolveGiteeRef, fetchPrDiff: fetchGiteeDiff } = require('./gitee');
|
|
132
132
|
const geeRef = resolveGiteeRef(options.gee, options.repo);
|
|
133
133
|
const geeToken = options.geeToken || process.env.GITEE_TOKEN;
|
|
134
|
-
console.error(chalk.blue(
|
|
134
|
+
console.error(chalk.blue(`้ซ?Fetching Gitee PR ${geeRef.owner}/${geeRef.repo}!${geeRef.pr}...`));
|
|
135
135
|
diff = await fetchGiteeDiff(geeRef, geeToken);
|
|
136
|
-
console.error(chalk.green(
|
|
136
|
+
console.error(chalk.green(`้?Diff fetched (${diff.length} chars)`));
|
|
137
137
|
} else if (options.gc) {
|
|
138
138
|
const { resolveMrRef: resolveGcRef, fetchMrDiff: fetchGcDiff } = require('./gitcode');
|
|
139
139
|
const gcRef = resolveGcRef(options.gc, options.repo);
|
|
140
140
|
const gcToken = options.gcToken || process.env.GITCODE_TOKEN;
|
|
141
|
-
console.error(chalk.blue(
|
|
141
|
+
console.error(chalk.blue(`้ซ?Fetching GitCode MR ${gcRef.owner}/${gcRef.repo}!${gcRef.mr}...`));
|
|
142
142
|
diff = await fetchGcDiff(gcRef, gcToken);
|
|
143
|
-
console.error(chalk.green(
|
|
143
|
+
console.error(chalk.green(`้?Diff fetched (${diff.length} chars)`));
|
|
144
144
|
} else if (options.bb) {
|
|
145
145
|
const { resolvePrRef: resolveBbRef, fetchPrDiff: fetchBbDiff } = require('./bitbucket');
|
|
146
146
|
const bbRef = resolveBbRef(options.bb, options.repo);
|
|
147
147
|
if (options.bbToken) process.env.BITBUCKET_USERNAME = options.bbToken.split(':')[0] || '';
|
|
148
148
|
const bbToken = options.bbToken || process.env.BITBUCKET_APP_PASSWORD;
|
|
149
|
-
console.error(chalk.blue(
|
|
149
|
+
console.error(chalk.blue(`้ซ?Fetching Bitbucket PR ${bbRef.owner}/${bbRef.repo}#${bbRef.pr}...`));
|
|
150
150
|
diff = await fetchBbDiff(bbRef, bbToken);
|
|
151
|
-
console.error(chalk.green(
|
|
151
|
+
console.error(chalk.green(`้?Diff fetched (${diff.length} chars)`));
|
|
152
152
|
} else if (options.pr) {
|
|
153
153
|
prRef = resolvePrRef(options.pr, options.repo);
|
|
154
154
|
const token = resolveToken(options.githubToken, config);
|
|
155
|
-
console.error(chalk.blue(
|
|
155
|
+
console.error(chalk.blue(`้ซ?Fetching PR ${prRef.owner}/${prRef.repo}#${prRef.pr}...`));
|
|
156
156
|
diff = await fetchPrDiff(prRef, token);
|
|
157
157
|
} else if (options.file) {
|
|
158
158
|
const fs = require('fs');
|
|
@@ -166,7 +166,7 @@ program
|
|
|
166
166
|
if (stdinBuffer.trim()) {
|
|
167
167
|
diff = stdinBuffer;
|
|
168
168
|
} else {
|
|
169
|
-
console.error(chalk.red('
|
|
169
|
+
console.error(chalk.red('้?No diff input provided. Pipe a diff, use --file, use --repo, or use --pr.'));
|
|
170
170
|
process.exit(1);
|
|
171
171
|
}
|
|
172
172
|
}
|
|
@@ -191,22 +191,22 @@ program
|
|
|
191
191
|
if (options.post && prRef) {
|
|
192
192
|
const token = resolveToken(options.githubToken, config);
|
|
193
193
|
if (!token) {
|
|
194
|
-
console.error(chalk.red('
|
|
194
|
+
console.error(chalk.red('้?--post requires --github-token or GITHUB_TOKEN env var'));
|
|
195
195
|
process.exit(1);
|
|
196
196
|
}
|
|
197
197
|
const mdReport = formatMarkdown(result);
|
|
198
|
-
console.error(chalk.blue(
|
|
198
|
+
console.error(chalk.blue(`้ซ?Posting review to PR ${prRef.owner}/${prRef.repo}#${prRef.pr}...`));
|
|
199
199
|
await postPrComment(prRef, mdReport, token);
|
|
200
|
-
console.error(chalk.green('
|
|
200
|
+
console.error(chalk.green('้?Review posted as PR comment!'));
|
|
201
201
|
}
|
|
202
202
|
|
|
203
203
|
if (options.inline && prRef) {
|
|
204
204
|
const token = resolveToken(options.githubToken, config);
|
|
205
205
|
if (!token) {
|
|
206
|
-
console.error(chalk.red('
|
|
206
|
+
console.error(chalk.red('้?--inline requires --github-token or GITHUB_TOKEN env var'));
|
|
207
207
|
process.exit(1);
|
|
208
208
|
}
|
|
209
|
-
console.error(chalk.blue(
|
|
209
|
+
console.error(chalk.blue(`้ซ?Posting inline review to PR ${prRef.owner}/${prRef.repo}#${prRef.pr}...`));
|
|
210
210
|
|
|
211
211
|
// Get PR files for commit SHA and file mapping
|
|
212
212
|
const prFiles = await fetchPrFiles(prRef, token);
|
|
@@ -246,23 +246,23 @@ program
|
|
|
246
246
|
const headSha = prInfo?.head?.sha;
|
|
247
247
|
if (headSha) {
|
|
248
248
|
await postInlineComments(prRef, headSha, inlineComments, token);
|
|
249
|
-
console.error(chalk.green(
|
|
249
|
+
console.error(chalk.green(`้?${inlineComments.length} inline comments posted!`));
|
|
250
250
|
} else {
|
|
251
|
-
console.error(chalk.red('
|
|
251
|
+
console.error(chalk.red('้?Could not resolve PR head commit SHA'));
|
|
252
252
|
}
|
|
253
253
|
} else {
|
|
254
|
-
console.error(chalk.yellow('
|
|
254
|
+
console.error(chalk.yellow('้ฟ?No line-level issues to post inline'));
|
|
255
255
|
}
|
|
256
256
|
}
|
|
257
257
|
|
|
258
258
|
console.log(output);
|
|
259
259
|
} catch (err) {
|
|
260
|
-
console.error(chalk.red(
|
|
260
|
+
console.error(chalk.red(`้?${err.message}`));
|
|
261
261
|
process.exit(1);
|
|
262
262
|
}
|
|
263
263
|
});
|
|
264
264
|
|
|
265
|
-
//
|
|
265
|
+
// ้นโฌ้นโฌ Cache Management ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
266
266
|
program
|
|
267
267
|
.command('cache')
|
|
268
268
|
.description('Manage review cache')
|
|
@@ -274,7 +274,7 @@ program
|
|
|
274
274
|
|
|
275
275
|
if (action === 'clear') {
|
|
276
276
|
const count = cleanCache();
|
|
277
|
-
console.log(chalk.green(
|
|
277
|
+
console.log(chalk.green(`้?Cache cleared (${count} entries removed)`));
|
|
278
278
|
} else if (action === 'status') {
|
|
279
279
|
if (!fs.existsSync(cacheDir)) {
|
|
280
280
|
console.log(chalk.blue(' Cache is empty'));
|
|
@@ -282,11 +282,11 @@ program
|
|
|
282
282
|
}
|
|
283
283
|
const files = fs.readdirSync(cacheDir).filter(f => f.endsWith('.json'));
|
|
284
284
|
const totalSize = files.reduce((sum, f) => sum + fs.statSync(path.join(cacheDir, f)).size, 0);
|
|
285
|
-
console.log(chalk.bold(`\n
|
|
285
|
+
console.log(chalk.bold(`\n้ฆๆ Cache: ${files.length} entries, ${(totalSize / 1024).toFixed(1)} KB`));
|
|
286
286
|
}
|
|
287
287
|
});
|
|
288
288
|
|
|
289
|
-
//
|
|
289
|
+
// ้นโฌ้นโฌ Fix ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
290
290
|
program
|
|
291
291
|
.command('fix')
|
|
292
292
|
.description('Generate a fix patch for issues found in a diff')
|
|
@@ -305,7 +305,7 @@ program
|
|
|
305
305
|
const { resolvePrRef, fetchPrDiff } = require('./github');
|
|
306
306
|
prRef = resolvePrRef(options.pr, options.repo);
|
|
307
307
|
const token = resolveToken(options.githubToken, config);
|
|
308
|
-
console.error(chalk.blue(
|
|
308
|
+
console.error(chalk.blue(`้ซ?Fetching PR ${prRef.owner}/${prRef.repo}#${prRef.pr}...`));
|
|
309
309
|
diff = await fetchPrDiff(prRef, token);
|
|
310
310
|
} else if (options.file) {
|
|
311
311
|
const fs = require('fs');
|
|
@@ -314,13 +314,13 @@ program
|
|
|
314
314
|
const fs = require('fs');
|
|
315
315
|
const stdinBuffer = fs.readFileSync(0, 'utf-8');
|
|
316
316
|
if (!stdinBuffer.trim()) {
|
|
317
|
-
console.error(chalk.red('
|
|
317
|
+
console.error(chalk.red('้?No diff input provided.'));
|
|
318
318
|
process.exit(1);
|
|
319
319
|
}
|
|
320
320
|
diff = stdinBuffer;
|
|
321
321
|
}
|
|
322
322
|
|
|
323
|
-
console.error(chalk.blue('
|
|
323
|
+
console.error(chalk.blue('้ซ?Generating fix patch...'));
|
|
324
324
|
const { reviewDiff } = require('./reviewer');
|
|
325
325
|
const result = await reviewDiff(diff, config, { noCache: true, single: true });
|
|
326
326
|
|
|
@@ -343,31 +343,31 @@ program
|
|
|
343
343
|
const patchMatch = aiResponse.match(/\`\`\`diff\n([\s\S]*?)\n\`\`\`/);
|
|
344
344
|
const patch = patchMatch ? patchMatch[1] : aiResponse;
|
|
345
345
|
|
|
346
|
-
console.log('\n' + chalk.bold('
|
|
347
|
-
console.log('
|
|
346
|
+
console.log('\n' + chalk.bold('้ฆโ Fix Patch:'));
|
|
347
|
+
console.log('้น?.repeat(50));
|
|
348
348
|
console.log(patch);
|
|
349
349
|
|
|
350
350
|
if (options.apply) {
|
|
351
351
|
const fs = require('fs');
|
|
352
352
|
const tmpFile = path.join(require('os').tmpdir(), 'coderev-fix.patch');
|
|
353
353
|
fs.writeFileSync(tmpFile, patch);
|
|
354
|
-
console.error(chalk.blue(
|
|
354
|
+
console.error(chalk.blue(`้ซ?Applying patch from ${tmpFile}...`));
|
|
355
355
|
try {
|
|
356
356
|
const { execSync } = require('child_process');
|
|
357
357
|
const cwd = prRef ? undefined : process.cwd();
|
|
358
358
|
execSync(`git apply "${tmpFile}"`, { cwd, stdio: 'pipe' });
|
|
359
|
-
console.log(chalk.green('
|
|
359
|
+
console.log(chalk.green('้?Patch applied successfully!'));
|
|
360
360
|
} catch (applyErr) {
|
|
361
|
-
console.error(chalk.red(
|
|
361
|
+
console.error(chalk.red(`้?Failed to apply patch: ${applyErr.stderr || applyErr.message}`));
|
|
362
362
|
}
|
|
363
363
|
}
|
|
364
364
|
} catch (err) {
|
|
365
|
-
console.error(chalk.red(
|
|
365
|
+
console.error(chalk.red(`้?${err.message}`));
|
|
366
366
|
process.exit(1);
|
|
367
367
|
}
|
|
368
368
|
});
|
|
369
369
|
|
|
370
|
-
//
|
|
370
|
+
// ้นโฌ้นโฌ Config ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
371
371
|
program
|
|
372
372
|
.command('config')
|
|
373
373
|
.description('Manage configuration')
|
|
@@ -381,8 +381,8 @@ program
|
|
|
381
381
|
const masked = JSON.parse(JSON.stringify(config));
|
|
382
382
|
if (masked.ai?.apiKey) masked.ai.apiKey = masked.ai.apiKey.slice(0, 8) + '...' + masked.ai.apiKey.slice(-4);
|
|
383
383
|
if (masked.github?.token) masked.github.token = masked.github.token.slice(0, 8) + '...' + masked.github.token.slice(-4);
|
|
384
|
-
console.log(chalk.bold('\n
|
|
385
|
-
console.log('
|
|
384
|
+
console.log(chalk.bold('\n้ฟ?Active Configuration:'));
|
|
385
|
+
console.log('้น?.repeat(50));
|
|
386
386
|
console.log(JSON.stringify(masked, null, 2));
|
|
387
387
|
} else if (action === 'validate') {
|
|
388
388
|
const fs = require('fs');
|
|
@@ -407,13 +407,13 @@ program
|
|
|
407
407
|
if (!parsed.ai?.provider) errors.push('Missing "ai.provider"');
|
|
408
408
|
if (!parsed.ai?.model) errors.push('Missing "ai.model"');
|
|
409
409
|
if (errors.length === 0) {
|
|
410
|
-
console.log(chalk.green(
|
|
410
|
+
console.log(chalk.green(`้?Config valid: ${found}`));
|
|
411
411
|
} else {
|
|
412
|
-
console.log(chalk.yellow(
|
|
412
|
+
console.log(chalk.yellow(`้ฟ?Config found but has issues:`));
|
|
413
413
|
for (const e of errors) console.log(chalk.yellow(` ${e}`));
|
|
414
414
|
}
|
|
415
415
|
} catch (parseErr) {
|
|
416
|
-
console.error(chalk.red(
|
|
416
|
+
console.error(chalk.red(`้?Invalid JSON in ${found}: ${parseErr.message}`));
|
|
417
417
|
}
|
|
418
418
|
} else {
|
|
419
419
|
console.log(chalk.blue(' No config file found in current or parent directories.'));
|
|
@@ -437,7 +437,7 @@ program
|
|
|
437
437
|
}
|
|
438
438
|
});
|
|
439
439
|
|
|
440
|
-
//
|
|
440
|
+
// ้นโฌ้นโฌ Stats ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
441
441
|
program
|
|
442
442
|
.command('stats')
|
|
443
443
|
.description('Review statistics and trends')
|
|
@@ -448,9 +448,9 @@ program
|
|
|
448
448
|
|
|
449
449
|
if (options.clear) {
|
|
450
450
|
if (clearHistory()) {
|
|
451
|
-
console.log(chalk.green('
|
|
451
|
+
console.log(chalk.green('้?Review history cleared'));
|
|
452
452
|
} else {
|
|
453
|
-
console.error(chalk.red('
|
|
453
|
+
console.error(chalk.red('้?Failed to clear history'));
|
|
454
454
|
}
|
|
455
455
|
return;
|
|
456
456
|
}
|
|
@@ -463,8 +463,8 @@ program
|
|
|
463
463
|
return;
|
|
464
464
|
}
|
|
465
465
|
|
|
466
|
-
console.log(chalk.bold('\n
|
|
467
|
-
console.log('
|
|
466
|
+
console.log(chalk.bold('\n้ฆๆณ Review Statistics'));
|
|
467
|
+
console.log('้น?.repeat(50));
|
|
468
468
|
console.log(` Period: ${chalk.bold(period)}`);
|
|
469
469
|
console.log(` Total reviews: ${stats.total}`);
|
|
470
470
|
if (stats.totalAllTime > stats.total) {
|
|
@@ -478,7 +478,7 @@ program
|
|
|
478
478
|
if (Object.keys(stats.issueTypes).length > 0) {
|
|
479
479
|
console.log(chalk.bold('\n Issue Types:'));
|
|
480
480
|
for (const [type, count] of Object.entries(stats.issueTypes)) {
|
|
481
|
-
const icon = type === 'error' ? chalk.red('
|
|
481
|
+
const icon = type === 'error' ? chalk.red('้?) : type === 'warning' ? chalk.yellow('้ฟ?) : chalk.blue('้ฉ?);
|
|
482
482
|
console.log(` ${icon} ${type}: ${count}`);
|
|
483
483
|
}
|
|
484
484
|
}
|
|
@@ -487,14 +487,14 @@ program
|
|
|
487
487
|
console.log(chalk.bold('\n Severity:'));
|
|
488
488
|
for (const [sev, count] of Object.entries(stats.severityBreakdown)) {
|
|
489
489
|
const color = sev === 'high' ? chalk.red : sev === 'medium' ? chalk.yellow : chalk.blue;
|
|
490
|
-
console.log(` ${color('
|
|
490
|
+
console.log(` ${color('้ผ?)} ${sev}: ${count}`);
|
|
491
491
|
}
|
|
492
492
|
}
|
|
493
493
|
|
|
494
494
|
if (stats.trend.length > 0) {
|
|
495
495
|
console.log(chalk.bold('\n Trend (last ' + stats.trend.length + ' reviews):'));
|
|
496
496
|
for (const t of stats.trend) {
|
|
497
|
-
const bar = '
|
|
497
|
+
const bar = '้ป?.repeat(Math.max(1, Math.round(t.score / 10)));
|
|
498
498
|
const color = t.score >= 80 ? chalk.green : t.score >= 50 ? chalk.yellow : chalk.red;
|
|
499
499
|
console.log(` ${t.date} ${color(bar)} ${t.score} (${t.issues} issues)`);
|
|
500
500
|
}
|
|
@@ -502,7 +502,7 @@ program
|
|
|
502
502
|
console.log('');
|
|
503
503
|
});
|
|
504
504
|
|
|
505
|
-
//
|
|
505
|
+
// ้นโฌ้นโฌ Hook ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
506
506
|
program
|
|
507
507
|
.command('hook')
|
|
508
508
|
.description('Install or remove a git hook (pre-commit / pre-push)')
|
|
@@ -517,20 +517,20 @@ program
|
|
|
517
517
|
|
|
518
518
|
if (action === 'install') {
|
|
519
519
|
if (!fs.existsSync(gitDir)) {
|
|
520
|
-
console.error(chalk.red('
|
|
520
|
+
console.error(chalk.red('้?Not a git repository: ' + process.cwd()));
|
|
521
521
|
process.exit(1);
|
|
522
522
|
}
|
|
523
523
|
|
|
524
524
|
const hookScript = `#!/bin/sh
|
|
525
525
|
# coderev ${hookType} hook
|
|
526
526
|
export PATH="$PATH:$(npm root -g)/../.bin"
|
|
527
|
-
echo "
|
|
527
|
+
echo "้ซ?Running coderev ${hookType} hook..."
|
|
528
528
|
coderev review --repo . --output markdown > /tmp/coderev-hook-report.md 2>/dev/null
|
|
529
529
|
SCORE=$(grep -oP 'Score: \\K\\d+' /tmp/coderev-hook-report.md || echo 0)
|
|
530
530
|
echo "Score: $SCORE/100"
|
|
531
531
|
MIN_SCORE=${minScore}
|
|
532
532
|
if [ "$SCORE" -lt "$MIN_SCORE" ]; then
|
|
533
|
-
echo "
|
|
533
|
+
echo "้?Score below threshold ($MIN_SCORE). Aborting ${hookType}."
|
|
534
534
|
cat /tmp/coderev-hook-report.md
|
|
535
535
|
exit 1
|
|
536
536
|
fi
|
|
@@ -540,21 +540,21 @@ fi
|
|
|
540
540
|
try {
|
|
541
541
|
fs.chmodSync(hookPath, '755');
|
|
542
542
|
} catch {}
|
|
543
|
-
console.log(chalk.green(
|
|
543
|
+
console.log(chalk.green(`้?${hookType} hook installed at ${hookPath}`));
|
|
544
544
|
} else if (action === 'remove') {
|
|
545
545
|
if (fs.existsSync(hookPath)) {
|
|
546
546
|
fs.unlinkSync(hookPath);
|
|
547
|
-
console.log(chalk.green(
|
|
547
|
+
console.log(chalk.green(`้?${hookType} hook removed`));
|
|
548
548
|
} else {
|
|
549
549
|
console.log(chalk.blue(' No hook to remove'));
|
|
550
550
|
}
|
|
551
551
|
} else {
|
|
552
|
-
console.error(chalk.red('
|
|
552
|
+
console.error(chalk.red('้?Unknown action. Use "install" or "remove".'));
|
|
553
553
|
process.exit(1);
|
|
554
554
|
}
|
|
555
555
|
});
|
|
556
556
|
|
|
557
|
-
//
|
|
557
|
+
// ้นโฌ้นโฌ Init / Setup ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
558
558
|
program
|
|
559
559
|
.command('init')
|
|
560
560
|
.description('Create a default coderev config file')
|
|
@@ -567,7 +567,7 @@ program
|
|
|
567
567
|
model: 'deepseek-chat',
|
|
568
568
|
temperature: 0.3,
|
|
569
569
|
maxTokens: 4096,
|
|
570
|
-
//
|
|
570
|
+
// ๆฟ๎ขๅๆตฃ็ตๆฎ API Key ้ดๆ ญโฌๆฐณ็น้๎๎จ้ๆฉๅบ็ๅง็
|
|
571
571
|
// apiKey: "sk-xxx",
|
|
572
572
|
// apiKeyEnv: "DEEPSEEK_API_KEY",
|
|
573
573
|
},
|
|
@@ -583,7 +583,7 @@ program
|
|
|
583
583
|
};
|
|
584
584
|
const configPath = path.join(process.cwd(), '.coderevrc.json');
|
|
585
585
|
fs.writeFileSync(configPath, JSON.stringify(defaultConfig, null, 2));
|
|
586
|
-
console.log(chalk.green(
|
|
586
|
+
console.log(chalk.green(`้?Default config created at ${configPath}`));
|
|
587
587
|
|
|
588
588
|
// Also create .coderevignore if it doesn't exist
|
|
589
589
|
const ignorePath = path.join(process.cwd(), '.coderevignore');
|
|
@@ -600,7 +600,7 @@ dist/
|
|
|
600
600
|
build/
|
|
601
601
|
`;
|
|
602
602
|
fs.writeFileSync(ignorePath, ignoreContent);
|
|
603
|
-
console.log(chalk.green(
|
|
603
|
+
console.log(chalk.green(`้?Default .coderevignore created at ${ignorePath}`));
|
|
604
604
|
}
|
|
605
605
|
|
|
606
606
|
// Also create .coderevhint if it doesn't exist
|
|
@@ -619,13 +619,13 @@ build/
|
|
|
619
619
|
- Avoid:
|
|
620
620
|
`;
|
|
621
621
|
fs.writeFileSync(hintPath, hintContent);
|
|
622
|
-
console.log(chalk.green(
|
|
622
|
+
console.log(chalk.green(`้?Default .coderevhint created at ${hintPath}`));
|
|
623
623
|
}
|
|
624
624
|
});
|
|
625
625
|
|
|
626
626
|
program.parse(process.argv);
|
|
627
627
|
|
|
628
|
-
//
|
|
628
|
+
// ้นโฌ้นโฌ Helpers ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ้นโฌ
|
|
629
629
|
async function getGitDiff(repoPath, base = 'main', head) {
|
|
630
630
|
const { execSync } = require('child_process');
|
|
631
631
|
const args = ['git', 'diff'];
|
|
@@ -640,8 +640,8 @@ async function getGitDiff(repoPath, base = 'main', head) {
|
|
|
640
640
|
|
|
641
641
|
function formatTerminal(result) {
|
|
642
642
|
const lines = [];
|
|
643
|
-
lines.push(chalk.bold('\n
|
|
644
|
-
lines.push('
|
|
643
|
+
lines.push(chalk.bold('\n้ฆๆต Code Review Report'));
|
|
644
|
+
lines.push('้น?.repeat(50));
|
|
645
645
|
|
|
646
646
|
if (result.summary) {
|
|
647
647
|
lines.push(`\n${chalk.bold('Summary:')} ${result.summary}`);
|
|
@@ -656,8 +656,8 @@ function formatTerminal(result) {
|
|
|
656
656
|
lines.push(`\n${chalk.bold(`Issues (${result.issues.length}):`)}`);
|
|
657
657
|
for (const issue of result.issues) {
|
|
658
658
|
const typeLabel =
|
|
659
|
-
issue.type === 'error' ? chalk.red('
|
|
660
|
-
issue.type === 'warning' ? chalk.yellow('
|
|
659
|
+
issue.type === 'error' ? chalk.red('้?) :
|
|
660
|
+
issue.type === 'warning' ? chalk.yellow('้ฟ?) : chalk.blue('้ฉ?);
|
|
661
661
|
const severity = issue.severity ? ` [${issue.severity}]` : '';
|
|
662
662
|
lines.push(` ${typeLabel}${severity} ${issue.message}`);
|
|
663
663
|
if (issue.file) lines.push(` File: ${issue.file}`);
|
|
@@ -669,23 +669,23 @@ function formatTerminal(result) {
|
|
|
669
669
|
if (result.suggestions && result.suggestions.length > 0) {
|
|
670
670
|
lines.push(`\n${chalk.bold('Suggestions:')}`);
|
|
671
671
|
for (const s of result.suggestions) {
|
|
672
|
-
lines.push(`
|
|
672
|
+
lines.push(` ้ฆๆ ${s}`);
|
|
673
673
|
}
|
|
674
674
|
}
|
|
675
675
|
|
|
676
676
|
if (result.praise && result.praise.length > 0) {
|
|
677
|
-
lines.push(`\n${chalk.bold('
|
|
677
|
+
lines.push(`\n${chalk.bold('้ฆๆค Good Practices:')}`);
|
|
678
678
|
for (const p of result.praise) {
|
|
679
|
-
lines.push(`
|
|
679
|
+
lines.push(` ้?${p}`);
|
|
680
680
|
}
|
|
681
681
|
}
|
|
682
682
|
|
|
683
|
-
lines.push('\n' + '
|
|
683
|
+
lines.push('\n' + '้น?.repeat(50));
|
|
684
684
|
return lines.join('\n');
|
|
685
685
|
}
|
|
686
686
|
|
|
687
687
|
function formatMarkdown(result) {
|
|
688
|
-
let md = '#
|
|
688
|
+
let md = '# ้ฆๆต Code Review Report\n\n';
|
|
689
689
|
|
|
690
690
|
if (result.summary) md += `**Summary:** ${result.summary}\n\n`;
|
|
691
691
|
if (result.score !== undefined) md += `**Score:** ${result.score}/100\n\n`;
|
|
@@ -704,12 +704,12 @@ function formatMarkdown(result) {
|
|
|
704
704
|
|
|
705
705
|
if (result.suggestions?.length) {
|
|
706
706
|
md += `\n## Suggestions\n\n`;
|
|
707
|
-
for (const s of result.suggestions) md += `-
|
|
707
|
+
for (const s of result.suggestions) md += `- ้ฆๆ ${s}\n`;
|
|
708
708
|
}
|
|
709
709
|
|
|
710
710
|
if (result.praise?.length) {
|
|
711
|
-
md += `\n##
|
|
712
|
-
for (const p of result.praise) md += `-
|
|
711
|
+
md += `\n## ้ฆๆค Good Practices\n\n`;
|
|
712
|
+
for (const p of result.praise) md += `- ้?${p}\n`;
|
|
713
713
|
}
|
|
714
714
|
|
|
715
715
|
return md;
|