@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.
Files changed (42) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/cli.d.ts.map +1 -1
  3. package/dist/cli.js +63 -0
  4. package/dist/cli.js.map +1 -1
  5. package/dist/commands/finding-correlation-map.d.ts +5 -0
  6. package/dist/commands/finding-correlation-map.d.ts.map +1 -0
  7. package/dist/commands/finding-correlation-map.js +102 -0
  8. package/dist/commands/finding-correlation-map.js.map +1 -0
  9. package/dist/commands/finding-dedup-cross.d.ts +5 -0
  10. package/dist/commands/finding-dedup-cross.d.ts.map +1 -0
  11. package/dist/commands/finding-dedup-cross.js +91 -0
  12. package/dist/commands/finding-dedup-cross.js.map +1 -0
  13. package/dist/commands/finding-groupby-file.d.ts +5 -0
  14. package/dist/commands/finding-groupby-file.d.ts.map +1 -0
  15. package/dist/commands/finding-groupby-file.js +95 -0
  16. package/dist/commands/finding-groupby-file.js.map +1 -0
  17. package/dist/commands/review-api-export.d.ts +5 -0
  18. package/dist/commands/review-api-export.d.ts.map +1 -0
  19. package/dist/commands/review-api-export.js +99 -0
  20. package/dist/commands/review-api-export.js.map +1 -0
  21. package/dist/commands/review-merge-request.d.ts +5 -0
  22. package/dist/commands/review-merge-request.d.ts.map +1 -0
  23. package/dist/commands/review-merge-request.js +96 -0
  24. package/dist/commands/review-merge-request.js.map +1 -0
  25. package/dist/commands/review-notification-config.d.ts +5 -0
  26. package/dist/commands/review-notification-config.d.ts.map +1 -0
  27. package/dist/commands/review-notification-config.js +123 -0
  28. package/dist/commands/review-notification-config.js.map +1 -0
  29. package/dist/commands/review-scope-select.d.ts +5 -0
  30. package/dist/commands/review-scope-select.d.ts.map +1 -0
  31. package/dist/commands/review-scope-select.js +99 -0
  32. package/dist/commands/review-scope-select.js.map +1 -0
  33. package/dist/commands/review-summary-dashboard.d.ts +5 -0
  34. package/dist/commands/review-summary-dashboard.d.ts.map +1 -0
  35. package/dist/commands/review-summary-dashboard.js +97 -0
  36. package/dist/commands/review-summary-dashboard.js.map +1 -0
  37. package/dist/commands/review-template-library.d.ts +5 -0
  38. package/dist/commands/review-template-library.d.ts.map +1 -0
  39. package/dist/commands/review-template-library.js +156 -0
  40. package/dist/commands/review-template-library.js.map +1 -0
  41. package/package.json +1 -1
  42. package/server.json +2 -2
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Finding-correlation-map — Map correlations between related findings.
3
+ */
4
+ export declare function runFindingCorrelationMap(argv: string[]): void;
5
+ //# sourceMappingURL=finding-correlation-map.d.ts.map
@@ -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,5 @@
1
+ /**
2
+ * Finding-dedup-cross — Deduplicate findings across multiple review files.
3
+ */
4
+ export declare function runFindingDedupCross(argv: string[]): void;
5
+ //# sourceMappingURL=finding-dedup-cross.d.ts.map
@@ -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,5 @@
1
+ /**
2
+ * Finding-groupby-file — Group findings by source file path.
3
+ */
4
+ export declare function runFindingGroupbyFile(argv: string[]): void;
5
+ //# sourceMappingURL=finding-groupby-file.d.ts.map
@@ -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,5 @@
1
+ /**
2
+ * Review-api-export — Export review data in API-compatible JSON format.
3
+ */
4
+ export declare function runReviewApiExport(argv: string[]): void;
5
+ //# sourceMappingURL=review-api-export.d.ts.map
@@ -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,5 @@
1
+ /**
2
+ * Review-merge-request — Format findings for merge/pull request comments.
3
+ */
4
+ export declare function runReviewMergeRequest(argv: string[]): void;
5
+ //# sourceMappingURL=review-merge-request.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-notification-config — Configure notification preferences for review results.
3
+ */
4
+ export declare function runReviewNotificationConfig(argv: string[]): void;
5
+ //# sourceMappingURL=review-notification-config.d.ts.map
@@ -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"}