@kevinrabun/judges 3.94.0 → 3.95.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.
- package/CHANGELOG.md +13 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +63 -0
- package/dist/cli.js.map +1 -1
- package/dist/commands/finding-correlation-map.d.ts +5 -0
- package/dist/commands/finding-correlation-map.d.ts.map +1 -0
- package/dist/commands/finding-correlation-map.js +102 -0
- package/dist/commands/finding-correlation-map.js.map +1 -0
- package/dist/commands/finding-dedup-cross.d.ts +5 -0
- package/dist/commands/finding-dedup-cross.d.ts.map +1 -0
- package/dist/commands/finding-dedup-cross.js +91 -0
- package/dist/commands/finding-dedup-cross.js.map +1 -0
- package/dist/commands/finding-groupby-file.d.ts +5 -0
- package/dist/commands/finding-groupby-file.d.ts.map +1 -0
- package/dist/commands/finding-groupby-file.js +95 -0
- package/dist/commands/finding-groupby-file.js.map +1 -0
- package/dist/commands/review-api-export.d.ts +5 -0
- package/dist/commands/review-api-export.d.ts.map +1 -0
- package/dist/commands/review-api-export.js +99 -0
- package/dist/commands/review-api-export.js.map +1 -0
- package/dist/commands/review-merge-request.d.ts +5 -0
- package/dist/commands/review-merge-request.d.ts.map +1 -0
- package/dist/commands/review-merge-request.js +96 -0
- package/dist/commands/review-merge-request.js.map +1 -0
- package/dist/commands/review-notification-config.d.ts +5 -0
- package/dist/commands/review-notification-config.d.ts.map +1 -0
- package/dist/commands/review-notification-config.js +123 -0
- package/dist/commands/review-notification-config.js.map +1 -0
- package/dist/commands/review-scope-select.d.ts +5 -0
- package/dist/commands/review-scope-select.d.ts.map +1 -0
- package/dist/commands/review-scope-select.js +99 -0
- package/dist/commands/review-scope-select.js.map +1 -0
- package/dist/commands/review-summary-dashboard.d.ts +5 -0
- package/dist/commands/review-summary-dashboard.d.ts.map +1 -0
- package/dist/commands/review-summary-dashboard.js +97 -0
- package/dist/commands/review-summary-dashboard.js.map +1 -0
- package/dist/commands/review-template-library.d.ts +5 -0
- package/dist/commands/review-template-library.d.ts.map +1 -0
- package/dist/commands/review-template-library.js +156 -0
- package/dist/commands/review-template-library.js.map +1 -0
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-correlation-map.d.ts","sourceRoot":"","sources":["../../src/commands/finding-correlation-map.ts"],"names":[],"mappings":"AAAA;;GAEG;AA0DH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAqE7D"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-correlation-map — Map correlations between related findings.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Helpers ────────────────────────────────────────────────────────────────
|
|
6
|
+
function findCorrelations(findings) {
|
|
7
|
+
const correlations = [];
|
|
8
|
+
for (let i = 0; i < findings.length; i++) {
|
|
9
|
+
for (let j = i + 1; j < findings.length; j++) {
|
|
10
|
+
const a = findings[i];
|
|
11
|
+
const b = findings[j];
|
|
12
|
+
// Same line overlap
|
|
13
|
+
const aLines = a.lineNumbers !== undefined ? a.lineNumbers : [];
|
|
14
|
+
const bLines = b.lineNumbers !== undefined ? b.lineNumbers : [];
|
|
15
|
+
const shared = aLines.filter((ln) => bLines.includes(ln));
|
|
16
|
+
if (shared.length > 0) {
|
|
17
|
+
correlations.push({
|
|
18
|
+
finding1: { ruleId: a.ruleId, title: a.title },
|
|
19
|
+
finding2: { ruleId: b.ruleId, title: b.title },
|
|
20
|
+
relationship: "co-located",
|
|
21
|
+
sharedLines: shared,
|
|
22
|
+
});
|
|
23
|
+
continue;
|
|
24
|
+
}
|
|
25
|
+
// Same rule prefix (same domain)
|
|
26
|
+
const prefixA = a.ruleId.split("-")[0];
|
|
27
|
+
const prefixB = b.ruleId.split("-")[0];
|
|
28
|
+
if (prefixA === prefixB && a.severity === b.severity) {
|
|
29
|
+
correlations.push({
|
|
30
|
+
finding1: { ruleId: a.ruleId, title: a.title },
|
|
31
|
+
finding2: { ruleId: b.ruleId, title: b.title },
|
|
32
|
+
relationship: "same-domain-same-severity",
|
|
33
|
+
sharedLines: [],
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
return correlations;
|
|
39
|
+
}
|
|
40
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
41
|
+
export function runFindingCorrelationMap(argv) {
|
|
42
|
+
const fileIdx = argv.indexOf("--file");
|
|
43
|
+
const formatIdx = argv.indexOf("--format");
|
|
44
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
45
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
46
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
47
|
+
console.log(`
|
|
48
|
+
judges finding-correlation-map — Map finding correlations
|
|
49
|
+
|
|
50
|
+
Usage:
|
|
51
|
+
judges finding-correlation-map --file <review.json> [--format table|json]
|
|
52
|
+
|
|
53
|
+
Options:
|
|
54
|
+
--file <path> Review result JSON file
|
|
55
|
+
--format <fmt> Output format: table (default), json
|
|
56
|
+
--help, -h Show this help
|
|
57
|
+
`);
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
if (!filePath) {
|
|
61
|
+
console.error("Error: --file is required");
|
|
62
|
+
process.exitCode = 1;
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
if (!existsSync(filePath)) {
|
|
66
|
+
console.error(`Error: file not found: ${filePath}`);
|
|
67
|
+
process.exitCode = 1;
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
let verdict;
|
|
71
|
+
try {
|
|
72
|
+
verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
73
|
+
}
|
|
74
|
+
catch {
|
|
75
|
+
console.error(`Error: failed to parse review file: ${filePath}`);
|
|
76
|
+
process.exitCode = 1;
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
const correlations = findCorrelations(verdict.findings);
|
|
80
|
+
if (format === "json") {
|
|
81
|
+
console.log(JSON.stringify({ total: correlations.length, correlations }, null, 2));
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
console.log(`\nFinding Correlations: ${correlations.length} relationship(s)`);
|
|
85
|
+
console.log("═".repeat(65));
|
|
86
|
+
if (correlations.length === 0) {
|
|
87
|
+
console.log(" No correlations found between findings.");
|
|
88
|
+
console.log("═".repeat(65));
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
for (const c of correlations) {
|
|
92
|
+
console.log(`\n ${c.finding1.ruleId} ↔ ${c.finding2.ruleId}`);
|
|
93
|
+
console.log(` Relationship: ${c.relationship}`);
|
|
94
|
+
if (c.sharedLines.length > 0) {
|
|
95
|
+
console.log(` Shared lines: ${c.sharedLines.join(", ")}`);
|
|
96
|
+
}
|
|
97
|
+
console.log(` A: ${c.finding1.title}`);
|
|
98
|
+
console.log(` B: ${c.finding2.title}`);
|
|
99
|
+
}
|
|
100
|
+
console.log("\n" + "═".repeat(65));
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=finding-correlation-map.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-correlation-map.js","sourceRoot":"","sources":["../../src/commands/finding-correlation-map.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dedup-cross.d.ts","sourceRoot":"","sources":["../../src/commands/finding-dedup-cross.ts"],"names":[],"mappings":"AAAA;;GAEG;AAeH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA8FzD"}
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-dedup-cross — Deduplicate findings across multiple review files.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
6
|
+
export function runFindingDedupCross(argv) {
|
|
7
|
+
const filesIdx = argv.indexOf("--files");
|
|
8
|
+
const formatIdx = argv.indexOf("--format");
|
|
9
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
10
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
11
|
+
console.log(`
|
|
12
|
+
judges finding-dedup-cross — Deduplicate findings across reviews
|
|
13
|
+
|
|
14
|
+
Usage:
|
|
15
|
+
judges finding-dedup-cross --files <file1.json> <file2.json> ...
|
|
16
|
+
[--format table|json]
|
|
17
|
+
|
|
18
|
+
Options:
|
|
19
|
+
--files <paths> Review JSON files (space-separated)
|
|
20
|
+
--format <fmt> Output format: table (default), json
|
|
21
|
+
--help, -h Show this help
|
|
22
|
+
`);
|
|
23
|
+
return;
|
|
24
|
+
}
|
|
25
|
+
if (filesIdx < 0) {
|
|
26
|
+
console.error("Error: --files is required");
|
|
27
|
+
process.exitCode = 1;
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
// Collect file paths (everything after --files until next flag)
|
|
31
|
+
const filePaths = [];
|
|
32
|
+
for (let i = filesIdx + 1; i < argv.length; i++) {
|
|
33
|
+
if (argv[i].startsWith("--"))
|
|
34
|
+
break;
|
|
35
|
+
filePaths.push(argv[i]);
|
|
36
|
+
}
|
|
37
|
+
if (filePaths.length === 0) {
|
|
38
|
+
console.error("Error: no files specified after --files");
|
|
39
|
+
process.exitCode = 1;
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
const seen = new Map();
|
|
43
|
+
let totalDuplicates = 0;
|
|
44
|
+
for (const fp of filePaths) {
|
|
45
|
+
if (!existsSync(fp)) {
|
|
46
|
+
console.error(`Warning: file not found, skipping: ${fp}`);
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
let verdict;
|
|
50
|
+
try {
|
|
51
|
+
verdict = JSON.parse(readFileSync(fp, "utf-8"));
|
|
52
|
+
}
|
|
53
|
+
catch {
|
|
54
|
+
console.error(`Warning: failed to parse, skipping: ${fp}`);
|
|
55
|
+
continue;
|
|
56
|
+
}
|
|
57
|
+
for (const f of verdict.findings) {
|
|
58
|
+
const key = `${f.ruleId}|${f.title}|${f.severity}`;
|
|
59
|
+
const existing = seen.get(key);
|
|
60
|
+
if (existing !== undefined) {
|
|
61
|
+
if (!existing.seenIn.includes(fp)) {
|
|
62
|
+
existing.seenIn.push(fp);
|
|
63
|
+
}
|
|
64
|
+
totalDuplicates++;
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
seen.set(key, { ...f, seenIn: [fp] });
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
const result = {
|
|
72
|
+
unique: seen.size,
|
|
73
|
+
duplicates: totalDuplicates,
|
|
74
|
+
findings: [...seen.values()],
|
|
75
|
+
};
|
|
76
|
+
if (format === "json") {
|
|
77
|
+
console.log(JSON.stringify(result, null, 2));
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
console.log(`\nCross-Review Deduplication`);
|
|
81
|
+
console.log(` Files analyzed: ${filePaths.length}`);
|
|
82
|
+
console.log(` Unique findings: ${result.unique}`);
|
|
83
|
+
console.log(` Duplicates removed: ${result.duplicates}`);
|
|
84
|
+
console.log("═".repeat(65));
|
|
85
|
+
for (const f of result.findings) {
|
|
86
|
+
const dupLabel = f.seenIn.length > 1 ? ` (in ${f.seenIn.length} files)` : "";
|
|
87
|
+
console.log(` [${f.severity}] ${f.ruleId}: ${f.title}${dupLabel}`);
|
|
88
|
+
}
|
|
89
|
+
console.log("═".repeat(65));
|
|
90
|
+
}
|
|
91
|
+
//# sourceMappingURL=finding-dedup-cross.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dedup-cross.js","sourceRoot":"","sources":["../../src/commands/finding-dedup-cross.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-groupby-file.d.ts","sourceRoot":"","sources":["../../src/commands/finding-groupby-file.ts"],"names":[],"mappings":"AAAA;;GAEG;AA+BH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAwF1D"}
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-groupby-file — Group findings by source file path.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Helpers ────────────────────────────────────────────────────────────────
|
|
6
|
+
function inferFile(f) {
|
|
7
|
+
// Use provenance if available, otherwise group by rule prefix
|
|
8
|
+
if (f.provenance !== undefined && typeof f.provenance === "string" && f.provenance.length > 0) {
|
|
9
|
+
return f.provenance;
|
|
10
|
+
}
|
|
11
|
+
const prefix = f.ruleId.split("-")[0];
|
|
12
|
+
return `[${prefix}]`;
|
|
13
|
+
}
|
|
14
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
15
|
+
export function runFindingGroupbyFile(argv) {
|
|
16
|
+
const fileIdx = argv.indexOf("--file");
|
|
17
|
+
const formatIdx = argv.indexOf("--format");
|
|
18
|
+
const sortIdx = argv.indexOf("--sort");
|
|
19
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
20
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
21
|
+
const sortBy = sortIdx >= 0 ? argv[sortIdx + 1] : "count";
|
|
22
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
23
|
+
console.log(`
|
|
24
|
+
judges finding-groupby-file — Group findings by source file
|
|
25
|
+
|
|
26
|
+
Usage:
|
|
27
|
+
judges finding-groupby-file --file <review.json> [--sort count|name]
|
|
28
|
+
[--format table|json]
|
|
29
|
+
|
|
30
|
+
Options:
|
|
31
|
+
--file <path> Review result JSON file
|
|
32
|
+
--sort <by> Sort by: count (default), name
|
|
33
|
+
--format <fmt> Output format: table (default), json
|
|
34
|
+
--help, -h Show this help
|
|
35
|
+
`);
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
if (!filePath) {
|
|
39
|
+
console.error("Error: --file is required");
|
|
40
|
+
process.exitCode = 1;
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
if (!existsSync(filePath)) {
|
|
44
|
+
console.error(`Error: file not found: ${filePath}`);
|
|
45
|
+
process.exitCode = 1;
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
let verdict;
|
|
49
|
+
try {
|
|
50
|
+
verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
console.error(`Error: failed to parse review file: ${filePath}`);
|
|
54
|
+
process.exitCode = 1;
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
const groupMap = new Map();
|
|
58
|
+
for (const f of verdict.findings) {
|
|
59
|
+
const fileName = inferFile(f);
|
|
60
|
+
let group = groupMap.get(fileName);
|
|
61
|
+
if (group === undefined) {
|
|
62
|
+
group = { file: fileName, findings: [], count: 0 };
|
|
63
|
+
groupMap.set(fileName, group);
|
|
64
|
+
}
|
|
65
|
+
group.findings.push({
|
|
66
|
+
ruleId: f.ruleId,
|
|
67
|
+
severity: f.severity,
|
|
68
|
+
title: f.title,
|
|
69
|
+
lineNumbers: f.lineNumbers !== undefined ? f.lineNumbers : [],
|
|
70
|
+
});
|
|
71
|
+
group.count++;
|
|
72
|
+
}
|
|
73
|
+
const groups = [...groupMap.values()];
|
|
74
|
+
if (sortBy === "name") {
|
|
75
|
+
groups.sort((a, b) => a.file.localeCompare(b.file));
|
|
76
|
+
}
|
|
77
|
+
else {
|
|
78
|
+
groups.sort((a, b) => b.count - a.count);
|
|
79
|
+
}
|
|
80
|
+
if (format === "json") {
|
|
81
|
+
console.log(JSON.stringify(groups, null, 2));
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
console.log(`\nFindings by File: ${groups.length} group(s), ${verdict.findings.length} total`);
|
|
85
|
+
console.log("═".repeat(65));
|
|
86
|
+
for (const g of groups) {
|
|
87
|
+
console.log(`\n ${g.file} (${g.count} finding${g.count !== 1 ? "s" : ""})`);
|
|
88
|
+
for (const f of g.findings) {
|
|
89
|
+
const lines = f.lineNumbers.length > 0 ? ` L${f.lineNumbers.join(",")}` : "";
|
|
90
|
+
console.log(` [${f.severity}] ${f.ruleId}${lines}: ${f.title}`);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
console.log("\n" + "═".repeat(65));
|
|
94
|
+
}
|
|
95
|
+
//# sourceMappingURL=finding-groupby-file.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-groupby-file.js","sourceRoot":"","sources":["../../src/commands/finding-groupby-file.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-api-export.d.ts","sourceRoot":"","sources":["../../src/commands/review-api-export.ts"],"names":[],"mappings":"AAAA;;GAEG;AAgCH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAgGvD"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Review-api-export — Export review data in API-compatible JSON format.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, writeFileSync, existsSync, readdirSync } from "fs";
|
|
5
|
+
import { join } from "path";
|
|
6
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
7
|
+
export function runReviewApiExport(argv) {
|
|
8
|
+
const fileIdx = argv.indexOf("--file");
|
|
9
|
+
const dirIdx = argv.indexOf("--dir");
|
|
10
|
+
const outputIdx = argv.indexOf("--output");
|
|
11
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
12
|
+
const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
|
|
13
|
+
const outputPath = outputIdx >= 0 ? argv[outputIdx + 1] : undefined;
|
|
14
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
15
|
+
console.log(`
|
|
16
|
+
judges review-api-export — Export review data in API format
|
|
17
|
+
|
|
18
|
+
Usage:
|
|
19
|
+
judges review-api-export --file <review.json> [--output <file>]
|
|
20
|
+
judges review-api-export --dir <path> [--output <file>]
|
|
21
|
+
|
|
22
|
+
Options:
|
|
23
|
+
--file <path> Single review JSON file
|
|
24
|
+
--dir <path> Directory of review JSON files
|
|
25
|
+
--output <path> Write API export to file
|
|
26
|
+
--help, -h Show this help
|
|
27
|
+
`);
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
const verdicts = [];
|
|
31
|
+
if (filePath) {
|
|
32
|
+
if (!existsSync(filePath)) {
|
|
33
|
+
console.error(`Error: file not found: ${filePath}`);
|
|
34
|
+
process.exitCode = 1;
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
try {
|
|
38
|
+
verdicts.push(JSON.parse(readFileSync(filePath, "utf-8")));
|
|
39
|
+
}
|
|
40
|
+
catch {
|
|
41
|
+
console.error(`Error: failed to parse: ${filePath}`);
|
|
42
|
+
process.exitCode = 1;
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
else if (dirPath) {
|
|
47
|
+
if (!existsSync(dirPath)) {
|
|
48
|
+
console.error(`Error: directory not found: ${dirPath}`);
|
|
49
|
+
process.exitCode = 1;
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
const files = readdirSync(dirPath).filter((f) => typeof f === "string" && f.endsWith(".json"));
|
|
53
|
+
for (const file of files) {
|
|
54
|
+
try {
|
|
55
|
+
const v = JSON.parse(readFileSync(join(dirPath, file), "utf-8"));
|
|
56
|
+
if (v.overallVerdict !== undefined) {
|
|
57
|
+
verdicts.push(v);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
// skip
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
console.error("Error: --file or --dir is required");
|
|
67
|
+
process.exitCode = 1;
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
const apiExport = {
|
|
71
|
+
version: "1.0",
|
|
72
|
+
exportedAt: new Date().toISOString(),
|
|
73
|
+
reviews: verdicts.map((v) => ({
|
|
74
|
+
verdict: v.overallVerdict,
|
|
75
|
+
score: v.overallScore,
|
|
76
|
+
findingCount: v.findings.length,
|
|
77
|
+
criticalCount: v.criticalCount,
|
|
78
|
+
highCount: v.highCount,
|
|
79
|
+
timestamp: v.timestamp ?? new Date().toISOString(),
|
|
80
|
+
findings: v.findings.map((f) => ({
|
|
81
|
+
ruleId: f.ruleId,
|
|
82
|
+
severity: f.severity,
|
|
83
|
+
title: f.title,
|
|
84
|
+
description: f.description,
|
|
85
|
+
recommendation: f.recommendation,
|
|
86
|
+
lineNumbers: f.lineNumbers,
|
|
87
|
+
confidence: f.confidence,
|
|
88
|
+
})),
|
|
89
|
+
})),
|
|
90
|
+
};
|
|
91
|
+
const output = JSON.stringify(apiExport, null, 2);
|
|
92
|
+
if (outputPath) {
|
|
93
|
+
writeFileSync(outputPath, output);
|
|
94
|
+
console.log(`API export written to ${outputPath} (${verdicts.length} review(s))`);
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
console.log(output);
|
|
98
|
+
}
|
|
99
|
+
//# sourceMappingURL=review-api-export.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-api-export.js","sourceRoot":"","sources":["../../src/commands/review-api-export.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-merge-request.d.ts","sourceRoot":"","sources":["../../src/commands/review-merge-request.ts"],"names":[],"mappings":"AAAA;;GAEG;AAsDH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4D1D"}
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Review-merge-request — Format findings for merge/pull request comments.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, writeFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Formatting ─────────────────────────────────────────────────────────────
|
|
6
|
+
function toMarkdownComment(verdict) {
|
|
7
|
+
const lines = [];
|
|
8
|
+
lines.push(`## Judges Review Summary`);
|
|
9
|
+
lines.push(``);
|
|
10
|
+
lines.push(`**Verdict:** ${verdict.overallVerdict} | **Score:** ${verdict.overallScore} | **Findings:** ${verdict.findings.length}`);
|
|
11
|
+
lines.push(``);
|
|
12
|
+
if (verdict.criticalCount > 0 || verdict.highCount > 0) {
|
|
13
|
+
lines.push(`> ⚠️ ${verdict.criticalCount} critical, ${verdict.highCount} high severity finding(s)`);
|
|
14
|
+
lines.push(``);
|
|
15
|
+
}
|
|
16
|
+
const bySeverity = {};
|
|
17
|
+
for (const f of verdict.findings) {
|
|
18
|
+
if (bySeverity[f.severity] === undefined) {
|
|
19
|
+
bySeverity[f.severity] = [];
|
|
20
|
+
}
|
|
21
|
+
bySeverity[f.severity].push(f);
|
|
22
|
+
}
|
|
23
|
+
const order = ["critical", "high", "medium", "low", "info"];
|
|
24
|
+
for (const sev of order) {
|
|
25
|
+
const group = bySeverity[sev];
|
|
26
|
+
if (group === undefined || group.length === 0)
|
|
27
|
+
continue;
|
|
28
|
+
lines.push(`### ${sev.charAt(0).toUpperCase() + sev.slice(1)} (${group.length})`);
|
|
29
|
+
lines.push(``);
|
|
30
|
+
for (const f of group) {
|
|
31
|
+
const loc = f.lineNumbers !== undefined && f.lineNumbers.length > 0 ? ` (L${f.lineNumbers.join(",")})` : "";
|
|
32
|
+
lines.push(`- **${f.ruleId}**${loc}: ${f.title}`);
|
|
33
|
+
lines.push(` - ${f.recommendation}`);
|
|
34
|
+
}
|
|
35
|
+
lines.push(``);
|
|
36
|
+
}
|
|
37
|
+
lines.push(`---`);
|
|
38
|
+
lines.push(`*Generated by [Judges](https://github.com/KevinRabun/judges) at ${verdict.timestamp ?? new Date().toISOString()}*`);
|
|
39
|
+
return lines.join("\n");
|
|
40
|
+
}
|
|
41
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
42
|
+
export function runReviewMergeRequest(argv) {
|
|
43
|
+
const fileIdx = argv.indexOf("--file");
|
|
44
|
+
const outputIdx = argv.indexOf("--output");
|
|
45
|
+
const formatIdx = argv.indexOf("--format");
|
|
46
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
47
|
+
const outputPath = outputIdx >= 0 ? argv[outputIdx + 1] : undefined;
|
|
48
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "markdown";
|
|
49
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
50
|
+
console.log(`
|
|
51
|
+
judges review-merge-request — Format findings for MR/PR comments
|
|
52
|
+
|
|
53
|
+
Usage:
|
|
54
|
+
judges review-merge-request --file <review.json> [--output <file>]
|
|
55
|
+
[--format markdown|json]
|
|
56
|
+
|
|
57
|
+
Options:
|
|
58
|
+
--file <path> Review result JSON file
|
|
59
|
+
--output <path> Write formatted comment to file
|
|
60
|
+
--format <fmt> Output format: markdown (default), json
|
|
61
|
+
--help, -h Show this help
|
|
62
|
+
`);
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
if (!filePath) {
|
|
66
|
+
console.error("Error: --file is required");
|
|
67
|
+
process.exitCode = 1;
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
if (!existsSync(filePath)) {
|
|
71
|
+
console.error(`Error: file not found: ${filePath}`);
|
|
72
|
+
process.exitCode = 1;
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
let verdict;
|
|
76
|
+
try {
|
|
77
|
+
verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
78
|
+
}
|
|
79
|
+
catch {
|
|
80
|
+
console.error(`Error: failed to parse review file: ${filePath}`);
|
|
81
|
+
process.exitCode = 1;
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (format === "json") {
|
|
85
|
+
console.log(JSON.stringify({ comment: toMarkdownComment(verdict) }, null, 2));
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
const output = toMarkdownComment(verdict);
|
|
89
|
+
if (outputPath) {
|
|
90
|
+
writeFileSync(outputPath, output);
|
|
91
|
+
console.log(`MR comment written to ${outputPath}`);
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
console.log(output);
|
|
95
|
+
}
|
|
96
|
+
//# sourceMappingURL=review-merge-request.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-merge-request.js","sourceRoot":"","sources":["../../src/commands/review-merge-request.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AAG7D,+EAA+E;AAE/E,SAAS,iBAAiB,CAAC,OAAwB;IACjD,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,KAAK,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;IACvC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACf,KAAK,CAAC,IAAI,CACR,gBAAgB,OAAO,CAAC,cAAc,iBAAiB,OAAO,CAAC,YAAY,oBAAoB,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,CACzH,CAAC;IACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAEf,IAAI,OAAO,CAAC,aAAa,GAAG,CAAC,IAAI,OAAO,CAAC,SAAS,GAAG,CAAC,EAAE,CAAC;QACvD,KAAK,CAAC,IAAI,CAAC,QAAQ,OAAO,CAAC,aAAa,cAAc,OAAO,CAAC,SAAS,2BAA2B,CAAC,CAAC;QACpG,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACjB,CAAC;IAED,MAAM,UAAU,GAA8B,EAAE,CAAC;IACjD,KAAK,MAAM,CAAC,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;QACjC,IAAI,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,SAAS,EAAE,CAAC;YACzC,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;QAC9B,CAAC;QACD,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC;IAED,MAAM,KAAK,GAAG,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;IAC5D,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE,CAAC;QACxB,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC;QAC9B,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,SAAS;QAExD,KAAK,CAAC,IAAI,CAAC,OAAO,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;QAClF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACf,KAAK,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;YACtB,MAAM,GAAG,GAAG,CAAC,CAAC,WAAW,KAAK,SAAS,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YAC5G,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YAClD,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,cAAc,EAAE,CAAC,CAAC;QACxC,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAClB,KAAK,CAAC,IAAI,CACR,mEAAmE,OAAO,CAAC,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,GAAG,CACpH,CAAC;IAEF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,+EAA+E;AAE/E,MAAM,UAAU,qBAAqB,CAAC,IAAc;IAClD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACvC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,QAAQ,GAAG,OAAO,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAC9D,MAAM,UAAU,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IACpE,MAAM,MAAM,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC;IAEjE,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;;CAYf,CAAC,CAAC;QACC,OAAO;IACT,CAAC;IAED,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAC;QAC3C,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1B,OAAO,CAAC,KAAK,CAAC,0BAA0B,QAAQ,EAAE,CAAC,CAAC;QACpD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,OAAwB,CAAC;IAC7B,IAAI,CAAC;QACH,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAoB,CAAC;IAC3E,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,CAAC,KAAK,CAAC,uCAAuC,QAAQ,EAAE,CAAC,CAAC;QACjE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC9E,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;IAE1C,IAAI,UAAU,EAAE,CAAC;QACf,aAAa,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;QAClC,OAAO,CAAC,GAAG,CAAC,yBAAyB,UAAU,EAAE,CAAC,CAAC;QACnD,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-notification-config.d.ts","sourceRoot":"","sources":["../../src/commands/review-notification-config.ts"],"names":[],"mappings":"AAAA;;GAEG;AAoBH,wBAAgB,2BAA2B,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA2HhE"}
|