@kevinrabun/judges 3.102.0 → 3.104.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 +26 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +126 -0
- package/dist/cli.js.map +1 -1
- package/dist/commands/finding-annotation-export.d.ts +2 -0
- package/dist/commands/finding-annotation-export.d.ts.map +1 -0
- package/dist/commands/finding-annotation-export.js +98 -0
- package/dist/commands/finding-annotation-export.js.map +1 -0
- package/dist/commands/finding-cross-file-link.d.ts +2 -0
- package/dist/commands/finding-cross-file-link.d.ts.map +1 -0
- package/dist/commands/finding-cross-file-link.js +102 -0
- package/dist/commands/finding-cross-file-link.js.map +1 -0
- package/dist/commands/finding-dedup-smart.d.ts +2 -0
- package/dist/commands/finding-dedup-smart.d.ts.map +1 -0
- package/dist/commands/finding-dedup-smart.js +110 -0
- package/dist/commands/finding-dedup-smart.js.map +1 -0
- package/dist/commands/finding-hotspot-detect.d.ts +2 -0
- package/dist/commands/finding-hotspot-detect.d.ts.map +1 -0
- package/dist/commands/finding-hotspot-detect.js +121 -0
- package/dist/commands/finding-hotspot-detect.js.map +1 -0
- package/dist/commands/finding-merge-strategy.d.ts +2 -0
- package/dist/commands/finding-merge-strategy.d.ts.map +1 -0
- package/dist/commands/finding-merge-strategy.js +85 -0
- package/dist/commands/finding-merge-strategy.js.map +1 -0
- package/dist/commands/finding-similar-match.d.ts +2 -0
- package/dist/commands/finding-similar-match.d.ts.map +1 -0
- package/dist/commands/finding-similar-match.js +113 -0
- package/dist/commands/finding-similar-match.js.map +1 -0
- package/dist/commands/finding-trend-alert.d.ts +2 -0
- package/dist/commands/finding-trend-alert.d.ts.map +1 -0
- package/dist/commands/finding-trend-alert.js +127 -0
- package/dist/commands/finding-trend-alert.js.map +1 -0
- package/dist/commands/review-ai-feedback-loop.d.ts +2 -0
- package/dist/commands/review-ai-feedback-loop.d.ts.map +1 -0
- package/dist/commands/review-ai-feedback-loop.js +117 -0
- package/dist/commands/review-ai-feedback-loop.js.map +1 -0
- package/dist/commands/review-ci-insight.d.ts +2 -0
- package/dist/commands/review-ci-insight.d.ts.map +1 -0
- package/dist/commands/review-ci-insight.js +101 -0
- package/dist/commands/review-ci-insight.js.map +1 -0
- package/dist/commands/review-code-health-score.d.ts +2 -0
- package/dist/commands/review-code-health-score.d.ts.map +1 -0
- package/dist/commands/review-code-health-score.js +101 -0
- package/dist/commands/review-code-health-score.js.map +1 -0
- package/dist/commands/review-confidence-explain.d.ts +2 -0
- package/dist/commands/review-confidence-explain.d.ts.map +1 -0
- package/dist/commands/review-confidence-explain.js +100 -0
- package/dist/commands/review-confidence-explain.js.map +1 -0
- package/dist/commands/review-focus-area.d.ts +2 -0
- package/dist/commands/review-focus-area.d.ts.map +1 -0
- package/dist/commands/review-focus-area.js +97 -0
- package/dist/commands/review-focus-area.js.map +1 -0
- package/dist/commands/review-pr-size-check.d.ts +2 -0
- package/dist/commands/review-pr-size-check.d.ts.map +1 -0
- package/dist/commands/review-pr-size-check.js +99 -0
- package/dist/commands/review-pr-size-check.js.map +1 -0
- package/dist/commands/review-scope-suggest.d.ts +2 -0
- package/dist/commands/review-scope-suggest.d.ts.map +1 -0
- package/dist/commands/review-scope-suggest.js +113 -0
- package/dist/commands/review-scope-suggest.js.map +1 -0
- package/dist/commands/review-team-analytics.d.ts +2 -0
- package/dist/commands/review-team-analytics.d.ts.map +1 -0
- package/dist/commands/review-team-analytics.js +95 -0
- package/dist/commands/review-team-analytics.js.map +1 -0
- package/dist/commands/review-template-suggest.d.ts +2 -0
- package/dist/commands/review-template-suggest.d.ts.map +1 -0
- package/dist/commands/review-template-suggest.js +120 -0
- package/dist/commands/review-template-suggest.js.map +1 -0
- package/dist/commands/review-velocity-track.d.ts +2 -0
- package/dist/commands/review-velocity-track.d.ts.map +1 -0
- package/dist/commands/review-velocity-track.js +95 -0
- package/dist/commands/review-velocity-track.js.map +1 -0
- package/dist/commands/review-workload-balance.d.ts +2 -0
- package/dist/commands/review-workload-balance.d.ts.map +1 -0
- package/dist/commands/review-workload-balance.js +87 -0
- package/dist/commands/review-workload-balance.js.map +1 -0
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-annotation-export.d.ts","sourceRoot":"","sources":["../../src/commands/finding-annotation-export.ts"],"names":[],"mappings":"AAyDA,wBAAgB,0BAA0B,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAyE/D"}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
function toAnnotations(findings) {
|
|
4
|
+
const annotations = [];
|
|
5
|
+
for (const f of findings) {
|
|
6
|
+
const startLine = f.lineNumbers !== undefined && f.lineNumbers.length > 0 ? f.lineNumbers[0] : 1;
|
|
7
|
+
const endLine = f.lineNumbers !== undefined && f.lineNumbers.length > 1 ? f.lineNumbers[f.lineNumbers.length - 1] : startLine;
|
|
8
|
+
annotations.push({
|
|
9
|
+
ruleId: f.ruleId,
|
|
10
|
+
severity: f.severity,
|
|
11
|
+
title: f.title,
|
|
12
|
+
message: f.description,
|
|
13
|
+
startLine,
|
|
14
|
+
endLine,
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
return annotations;
|
|
18
|
+
}
|
|
19
|
+
function formatGitHub(annotations) {
|
|
20
|
+
const lines = [];
|
|
21
|
+
for (const a of annotations) {
|
|
22
|
+
const level = a.severity === "critical" || a.severity === "high" ? "error" : "warning";
|
|
23
|
+
lines.push(`::${level} title=${a.ruleId}::${a.title} (line ${a.startLine}): ${a.message}`);
|
|
24
|
+
}
|
|
25
|
+
return lines.join("\n");
|
|
26
|
+
}
|
|
27
|
+
function formatProblemMatcher(annotations) {
|
|
28
|
+
const lines = [];
|
|
29
|
+
for (const a of annotations) {
|
|
30
|
+
lines.push(`${a.ruleId}:${a.startLine}:${a.endLine}: ${a.severity}: ${a.title} - ${a.message}`);
|
|
31
|
+
}
|
|
32
|
+
return lines.join("\n");
|
|
33
|
+
}
|
|
34
|
+
export function runFindingAnnotationExport(argv) {
|
|
35
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
36
|
+
console.log(`Usage: judges finding-annotation-export [options]
|
|
37
|
+
|
|
38
|
+
Export findings as code annotations for IDEs and CI systems.
|
|
39
|
+
|
|
40
|
+
Options:
|
|
41
|
+
--report <path> Path to verdict JSON file
|
|
42
|
+
--output <path> Output file path (default: stdout)
|
|
43
|
+
--type <type> Annotation type: github, problem-matcher, json (default: json)
|
|
44
|
+
--format <fmt> Output format: table (default) or json
|
|
45
|
+
-h, --help Show this help message`);
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
const formatIdx = argv.indexOf("--format");
|
|
49
|
+
const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
|
|
50
|
+
const reportIdx = argv.indexOf("--report");
|
|
51
|
+
const reportPath = reportIdx !== -1 && argv[reportIdx + 1]
|
|
52
|
+
? join(process.cwd(), argv[reportIdx + 1])
|
|
53
|
+
: join(process.cwd(), ".judges", "last-verdict.json");
|
|
54
|
+
if (!existsSync(reportPath)) {
|
|
55
|
+
console.log(`No report found at: ${reportPath}`);
|
|
56
|
+
console.log("Run a review first or provide --report.");
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const data = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
60
|
+
const findings = data.findings ?? [];
|
|
61
|
+
if (findings.length === 0) {
|
|
62
|
+
console.log("No findings to export.");
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
const annotations = toAnnotations(findings);
|
|
66
|
+
const typeIdx = argv.indexOf("--type");
|
|
67
|
+
const annotationType = typeIdx !== -1 && argv[typeIdx + 1] ? argv[typeIdx + 1] : "json";
|
|
68
|
+
let output;
|
|
69
|
+
if (annotationType === "github") {
|
|
70
|
+
output = formatGitHub(annotations);
|
|
71
|
+
}
|
|
72
|
+
else if (annotationType === "problem-matcher") {
|
|
73
|
+
output = formatProblemMatcher(annotations);
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
output = JSON.stringify(annotations, null, 2);
|
|
77
|
+
}
|
|
78
|
+
const outputIdx = argv.indexOf("--output");
|
|
79
|
+
if (outputIdx !== -1 && argv[outputIdx + 1]) {
|
|
80
|
+
const outPath = join(process.cwd(), argv[outputIdx + 1]);
|
|
81
|
+
const outDir = join(outPath, "..");
|
|
82
|
+
if (!existsSync(outDir)) {
|
|
83
|
+
mkdirSync(outDir, { recursive: true });
|
|
84
|
+
}
|
|
85
|
+
writeFileSync(outPath, output);
|
|
86
|
+
console.log(`Annotations exported to: ${outPath}`);
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
if (format === "json" || annotationType === "json") {
|
|
90
|
+
console.log(output);
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
console.log("\n=== Finding Annotations ===\n");
|
|
94
|
+
console.log(`Type: ${annotationType}`);
|
|
95
|
+
console.log(`Findings: ${annotations.length}\n`);
|
|
96
|
+
console.log(output);
|
|
97
|
+
}
|
|
98
|
+
//# sourceMappingURL=finding-annotation-export.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-annotation-export.js","sourceRoot":"","sources":["../../src/commands/finding-annotation-export.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-cross-file-link.d.ts","sourceRoot":"","sources":["../../src/commands/finding-cross-file-link.ts"],"names":[],"mappings":"AA4EA,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAsD5D"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { readFileSync, existsSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
function linkCrossFile(findings) {
|
|
4
|
+
const links = [];
|
|
5
|
+
const ruleGroups = new Map();
|
|
6
|
+
for (const f of findings) {
|
|
7
|
+
const prefix = f.ruleId.split("-").slice(0, 2).join("-");
|
|
8
|
+
const group = ruleGroups.get(prefix);
|
|
9
|
+
if (group !== undefined) {
|
|
10
|
+
group.push(f);
|
|
11
|
+
}
|
|
12
|
+
else {
|
|
13
|
+
ruleGroups.set(prefix, [f]);
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
for (const [prefix, group] of ruleGroups) {
|
|
17
|
+
if (group.length < 2)
|
|
18
|
+
continue;
|
|
19
|
+
const uniqueRules = [...new Set(group.map((f) => f.ruleId))];
|
|
20
|
+
const severities = group.map((f) => f.severity);
|
|
21
|
+
const highestSeverity = severities.includes("critical")
|
|
22
|
+
? "critical"
|
|
23
|
+
: severities.includes("high")
|
|
24
|
+
? "high"
|
|
25
|
+
: severities.includes("medium")
|
|
26
|
+
? "medium"
|
|
27
|
+
: "low";
|
|
28
|
+
let pattern;
|
|
29
|
+
if (uniqueRules.length === 1) {
|
|
30
|
+
pattern = "Same rule appearing in multiple locations";
|
|
31
|
+
}
|
|
32
|
+
else {
|
|
33
|
+
pattern = "Related rules from same family";
|
|
34
|
+
}
|
|
35
|
+
let recommendation;
|
|
36
|
+
if (group.length >= 5) {
|
|
37
|
+
recommendation = "Systemic issue — consider a shared utility or design pattern";
|
|
38
|
+
}
|
|
39
|
+
else if (uniqueRules.length > 1) {
|
|
40
|
+
recommendation = "Cross-cutting concern — review holistically";
|
|
41
|
+
}
|
|
42
|
+
else {
|
|
43
|
+
recommendation = "Repeated pattern — consider abstracting";
|
|
44
|
+
}
|
|
45
|
+
links.push({
|
|
46
|
+
ruleId: prefix,
|
|
47
|
+
severity: highestSeverity,
|
|
48
|
+
instances: group.length,
|
|
49
|
+
relatedRules: uniqueRules,
|
|
50
|
+
pattern,
|
|
51
|
+
recommendation,
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
links.sort((a, b) => b.instances - a.instances);
|
|
55
|
+
return links;
|
|
56
|
+
}
|
|
57
|
+
export function runFindingCrossFileLink(argv) {
|
|
58
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
59
|
+
console.log(`Usage: judges finding-cross-file-link [options]
|
|
60
|
+
|
|
61
|
+
Link related findings across files.
|
|
62
|
+
|
|
63
|
+
Options:
|
|
64
|
+
--report <path> Path to verdict JSON file
|
|
65
|
+
--format <fmt> Output format: table (default) or json
|
|
66
|
+
-h, --help Show this help message`);
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
const formatIdx = argv.indexOf("--format");
|
|
70
|
+
const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
|
|
71
|
+
const reportIdx = argv.indexOf("--report");
|
|
72
|
+
const reportPath = reportIdx !== -1 && argv[reportIdx + 1]
|
|
73
|
+
? join(process.cwd(), argv[reportIdx + 1])
|
|
74
|
+
: join(process.cwd(), ".judges", "last-verdict.json");
|
|
75
|
+
if (!existsSync(reportPath)) {
|
|
76
|
+
console.log(`No report found at: ${reportPath}`);
|
|
77
|
+
console.log("Run a review first or provide --report.");
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
const data = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
81
|
+
const findings = data.findings ?? [];
|
|
82
|
+
if (findings.length === 0) {
|
|
83
|
+
console.log("No findings to link.");
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const links = linkCrossFile(findings);
|
|
87
|
+
if (format === "json") {
|
|
88
|
+
console.log(JSON.stringify(links, null, 2));
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
console.log("\n=== Cross-File Finding Links ===\n");
|
|
92
|
+
console.log(`Findings analyzed: ${findings.length}`);
|
|
93
|
+
console.log(`Cross-file groups: ${links.length}\n`);
|
|
94
|
+
for (const link of links) {
|
|
95
|
+
console.log(`[${link.severity.toUpperCase()}] ${link.ruleId} — ${link.instances} instances`);
|
|
96
|
+
console.log(` Pattern: ${link.pattern}`);
|
|
97
|
+
console.log(` Rules: ${link.relatedRules.join(", ")}`);
|
|
98
|
+
console.log(` → ${link.recommendation}`);
|
|
99
|
+
console.log();
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=finding-cross-file-link.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-cross-file-link.js","sourceRoot":"","sources":["../../src/commands/finding-cross-file-link.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dedup-smart.d.ts","sourceRoot":"","sources":["../../src/commands/finding-dedup-smart.ts"],"names":[],"mappings":"AAwFA,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAwDzD"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import { readFileSync, existsSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
function smartDedup(findings) {
|
|
4
|
+
const groups = [];
|
|
5
|
+
const processed = new Set();
|
|
6
|
+
for (let i = 0; i < findings.length; i++) {
|
|
7
|
+
if (processed.has(i))
|
|
8
|
+
continue;
|
|
9
|
+
const f = findings[i];
|
|
10
|
+
const duplicates = [];
|
|
11
|
+
const lines = f.lineNumbers !== undefined ? [...f.lineNumbers] : [];
|
|
12
|
+
for (let j = i + 1; j < findings.length; j++) {
|
|
13
|
+
if (processed.has(j))
|
|
14
|
+
continue;
|
|
15
|
+
const other = findings[j];
|
|
16
|
+
const sameRule = f.ruleId === other.ruleId;
|
|
17
|
+
const sameSeverity = f.severity === other.severity;
|
|
18
|
+
const similarTitle = titleSimilarity(f.title, other.title) > 0.6;
|
|
19
|
+
const closeLines = linesOverlap(f.lineNumbers, other.lineNumbers);
|
|
20
|
+
if ((sameRule && sameSeverity) || (similarTitle && closeLines)) {
|
|
21
|
+
processed.add(j);
|
|
22
|
+
duplicates.push(other.ruleId);
|
|
23
|
+
if (other.lineNumbers !== undefined) {
|
|
24
|
+
lines.push(...other.lineNumbers);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
processed.add(i);
|
|
29
|
+
const uniqueLines = [...new Set(lines)].sort((a, b) => a - b);
|
|
30
|
+
groups.push({
|
|
31
|
+
canonical: f.ruleId,
|
|
32
|
+
severity: f.severity,
|
|
33
|
+
count: 1 + duplicates.length,
|
|
34
|
+
duplicateRules: duplicates,
|
|
35
|
+
lineRanges: uniqueLines,
|
|
36
|
+
recommendation: duplicates.length > 0 ? `${duplicates.length} duplicate(s) can be consolidated` : "Unique finding",
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
return groups;
|
|
40
|
+
}
|
|
41
|
+
function titleSimilarity(a, b) {
|
|
42
|
+
const wordsA = new Set(a.toLowerCase().split(/\s+/));
|
|
43
|
+
const wordsB = new Set(b.toLowerCase().split(/\s+/));
|
|
44
|
+
let overlap = 0;
|
|
45
|
+
for (const w of wordsA) {
|
|
46
|
+
if (wordsB.has(w))
|
|
47
|
+
overlap++;
|
|
48
|
+
}
|
|
49
|
+
const total = Math.max(wordsA.size, wordsB.size);
|
|
50
|
+
return total > 0 ? overlap / total : 0;
|
|
51
|
+
}
|
|
52
|
+
function linesOverlap(a, b) {
|
|
53
|
+
if (a === undefined || b === undefined)
|
|
54
|
+
return false;
|
|
55
|
+
if (a.length === 0 || b.length === 0)
|
|
56
|
+
return false;
|
|
57
|
+
const minA = Math.min(...a);
|
|
58
|
+
const maxA = Math.max(...a);
|
|
59
|
+
const minB = Math.min(...b);
|
|
60
|
+
const maxB = Math.max(...b);
|
|
61
|
+
return Math.abs(minA - minB) <= 5 || (minA <= maxB && minB <= maxA);
|
|
62
|
+
}
|
|
63
|
+
export function runFindingDedupSmart(argv) {
|
|
64
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
65
|
+
console.log(`Usage: judges finding-dedup-smart [options]
|
|
66
|
+
|
|
67
|
+
Smart deduplication of similar findings.
|
|
68
|
+
|
|
69
|
+
Options:
|
|
70
|
+
--report <path> Path to verdict JSON file
|
|
71
|
+
--format <fmt> Output format: table (default) or json
|
|
72
|
+
-h, --help Show this help message`);
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
const formatIdx = argv.indexOf("--format");
|
|
76
|
+
const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
|
|
77
|
+
const reportIdx = argv.indexOf("--report");
|
|
78
|
+
const reportPath = reportIdx !== -1 && argv[reportIdx + 1]
|
|
79
|
+
? join(process.cwd(), argv[reportIdx + 1])
|
|
80
|
+
: join(process.cwd(), ".judges", "last-verdict.json");
|
|
81
|
+
if (!existsSync(reportPath)) {
|
|
82
|
+
console.log(`No report found at: ${reportPath}`);
|
|
83
|
+
console.log("Run a review first or provide --report.");
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const data = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
87
|
+
const findings = data.findings ?? [];
|
|
88
|
+
if (findings.length === 0) {
|
|
89
|
+
console.log("No findings to deduplicate.");
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
const groups = smartDedup(findings);
|
|
93
|
+
const totalDuplicates = groups.reduce((sum, g) => sum + g.count - 1, 0);
|
|
94
|
+
if (format === "json") {
|
|
95
|
+
console.log(JSON.stringify({ groups, totalDuplicates, uniqueGroups: groups.length }, null, 2));
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
console.log("\n=== Smart Deduplication ===\n");
|
|
99
|
+
console.log(`Original findings: ${findings.length}`);
|
|
100
|
+
console.log(`Unique groups: ${groups.length}`);
|
|
101
|
+
console.log(`Duplicates removed: ${totalDuplicates}\n`);
|
|
102
|
+
for (const g of groups) {
|
|
103
|
+
console.log(`[${g.severity.toUpperCase()}] ${g.canonical} (${g.count} instance(s))`);
|
|
104
|
+
if (g.lineRanges.length > 0) {
|
|
105
|
+
console.log(` Lines: ${g.lineRanges.join(", ")}`);
|
|
106
|
+
}
|
|
107
|
+
console.log(` → ${g.recommendation}`);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=finding-dedup-smart.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dedup-smart.js","sourceRoot":"","sources":["../../src/commands/finding-dedup-smart.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-hotspot-detect.d.ts","sourceRoot":"","sources":["../../src/commands/finding-hotspot-detect.ts"],"names":[],"mappings":"AAoEA,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAqF5D"}
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import { readFileSync, existsSync, readdirSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
function detectHotspots(allFindings) {
|
|
4
|
+
const ruleGroups = new Map();
|
|
5
|
+
for (const f of allFindings) {
|
|
6
|
+
const group = ruleGroups.get(f.ruleId);
|
|
7
|
+
if (group !== undefined) {
|
|
8
|
+
group.push(f);
|
|
9
|
+
}
|
|
10
|
+
else {
|
|
11
|
+
ruleGroups.set(f.ruleId, [f]);
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
const hotspots = [];
|
|
15
|
+
for (const [ruleId, findings] of ruleGroups) {
|
|
16
|
+
if (findings.length < 2)
|
|
17
|
+
continue;
|
|
18
|
+
const lines = [];
|
|
19
|
+
for (const f of findings) {
|
|
20
|
+
if (f.lineNumbers !== undefined) {
|
|
21
|
+
lines.push(...f.lineNumbers);
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
const avgConf = findings.reduce((sum, f) => sum + (f.confidence ?? 0.5), 0) / findings.length;
|
|
25
|
+
const uniqueLines = [...new Set(lines)].sort((a, b) => a - b);
|
|
26
|
+
let recommendation;
|
|
27
|
+
if (findings.length >= 5) {
|
|
28
|
+
recommendation = "Critical hotspot — prioritize refactoring";
|
|
29
|
+
}
|
|
30
|
+
else if (findings.length >= 3) {
|
|
31
|
+
recommendation = "Frequent issue — add targeted tests";
|
|
32
|
+
}
|
|
33
|
+
else {
|
|
34
|
+
recommendation = "Recurring pattern — monitor";
|
|
35
|
+
}
|
|
36
|
+
hotspots.push({
|
|
37
|
+
ruleId,
|
|
38
|
+
occurrences: findings.length,
|
|
39
|
+
severity: findings[0].severity,
|
|
40
|
+
avgConfidence: avgConf,
|
|
41
|
+
lineRanges: uniqueLines.slice(0, 20),
|
|
42
|
+
recommendation,
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
hotspots.sort((a, b) => b.occurrences - a.occurrences);
|
|
46
|
+
return hotspots;
|
|
47
|
+
}
|
|
48
|
+
export function runFindingHotspotDetect(argv) {
|
|
49
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
50
|
+
console.log(`Usage: judges finding-hotspot-detect [options]
|
|
51
|
+
|
|
52
|
+
Detect code hotspots with recurring findings.
|
|
53
|
+
|
|
54
|
+
Options:
|
|
55
|
+
--dir <path> Directory with verdict JSON files
|
|
56
|
+
--report <path> Single verdict JSON file
|
|
57
|
+
--min <n> Minimum occurrences to report (default: 2)
|
|
58
|
+
--format <fmt> Output format: table (default) or json
|
|
59
|
+
-h, --help Show this help message`);
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
const formatIdx = argv.indexOf("--format");
|
|
63
|
+
const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
|
|
64
|
+
const minIdx = argv.indexOf("--min");
|
|
65
|
+
const minOccurrences = minIdx !== -1 && argv[minIdx + 1] ? parseInt(argv[minIdx + 1], 10) : 2;
|
|
66
|
+
const allFindings = [];
|
|
67
|
+
const dirIdx = argv.indexOf("--dir");
|
|
68
|
+
const dirPath = dirIdx !== -1 && argv[dirIdx + 1]
|
|
69
|
+
? join(process.cwd(), argv[dirIdx + 1])
|
|
70
|
+
: join(process.cwd(), ".judges", "history");
|
|
71
|
+
if (existsSync(dirPath)) {
|
|
72
|
+
const files = readdirSync(dirPath).filter((f) => f.endsWith(".json"));
|
|
73
|
+
for (const file of files) {
|
|
74
|
+
const data = JSON.parse(readFileSync(join(dirPath, file), "utf-8"));
|
|
75
|
+
if (data.findings !== undefined) {
|
|
76
|
+
allFindings.push(...data.findings);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
const reportIdx = argv.indexOf("--report");
|
|
81
|
+
if (reportIdx !== -1 && argv[reportIdx + 1]) {
|
|
82
|
+
const rPath = join(process.cwd(), argv[reportIdx + 1]);
|
|
83
|
+
if (existsSync(rPath)) {
|
|
84
|
+
const data = JSON.parse(readFileSync(rPath, "utf-8"));
|
|
85
|
+
if (data.findings !== undefined) {
|
|
86
|
+
allFindings.push(...data.findings);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
if (allFindings.length === 0) {
|
|
91
|
+
const defaultPath = join(process.cwd(), ".judges", "last-verdict.json");
|
|
92
|
+
if (existsSync(defaultPath)) {
|
|
93
|
+
const data = JSON.parse(readFileSync(defaultPath, "utf-8"));
|
|
94
|
+
if (data.findings !== undefined) {
|
|
95
|
+
allFindings.push(...data.findings);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
if (allFindings.length === 0) {
|
|
100
|
+
console.log("No findings data found. Run reviews first.");
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
const hotspots = detectHotspots(allFindings).filter((h) => h.occurrences >= minOccurrences);
|
|
104
|
+
if (format === "json") {
|
|
105
|
+
console.log(JSON.stringify(hotspots, null, 2));
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
console.log("\n=== Code Hotspots ===\n");
|
|
109
|
+
console.log(`Total findings analyzed: ${allFindings.length}`);
|
|
110
|
+
console.log(`Hotspots found: ${hotspots.length}\n`);
|
|
111
|
+
for (const h of hotspots) {
|
|
112
|
+
console.log(`[${h.severity.toUpperCase()}] ${h.ruleId} — ${h.occurrences} occurrences`);
|
|
113
|
+
console.log(` Confidence: ${(h.avgConfidence * 100).toFixed(0)}%`);
|
|
114
|
+
if (h.lineRanges.length > 0) {
|
|
115
|
+
console.log(` Lines: ${h.lineRanges.join(", ")}`);
|
|
116
|
+
}
|
|
117
|
+
console.log(` → ${h.recommendation}`);
|
|
118
|
+
console.log();
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
//# sourceMappingURL=finding-hotspot-detect.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-hotspot-detect.js","sourceRoot":"","sources":["../../src/commands/finding-hotspot-detect.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-merge-strategy.d.ts","sourceRoot":"","sources":["../../src/commands/finding-merge-strategy.ts"],"names":[],"mappings":"AAkDA,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA8D5D"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { readFileSync, existsSync, readdirSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
function detectMergeStrategy(branchFindings) {
|
|
4
|
+
const entries = [];
|
|
5
|
+
const seenRules = new Map();
|
|
6
|
+
for (const [branch, findings] of branchFindings) {
|
|
7
|
+
for (const f of findings) {
|
|
8
|
+
const existing = seenRules.get(f.ruleId);
|
|
9
|
+
if (existing !== undefined) {
|
|
10
|
+
existing.push(branch);
|
|
11
|
+
entries.push({
|
|
12
|
+
ruleId: f.ruleId,
|
|
13
|
+
severity: f.severity,
|
|
14
|
+
branch,
|
|
15
|
+
status: "duplicate",
|
|
16
|
+
strategy: "deduplicate — keep highest severity instance",
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
else {
|
|
20
|
+
seenRules.set(f.ruleId, [branch]);
|
|
21
|
+
entries.push({
|
|
22
|
+
ruleId: f.ruleId,
|
|
23
|
+
severity: f.severity,
|
|
24
|
+
branch,
|
|
25
|
+
status: "unique",
|
|
26
|
+
strategy: "carry forward to merged result",
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
return entries;
|
|
32
|
+
}
|
|
33
|
+
export function runFindingMergeStrategy(argv) {
|
|
34
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
35
|
+
console.log(`Usage: judges finding-merge-strategy [options]
|
|
36
|
+
|
|
37
|
+
Suggest merge strategies for findings across branches.
|
|
38
|
+
|
|
39
|
+
Options:
|
|
40
|
+
--dir <path> Directory containing per-branch verdict files
|
|
41
|
+
--format <fmt> Output format: table (default) or json
|
|
42
|
+
-h, --help Show this help message`);
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
const formatIdx = argv.indexOf("--format");
|
|
46
|
+
const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
|
|
47
|
+
const dirIdx = argv.indexOf("--dir");
|
|
48
|
+
const dirPath = dirIdx !== -1 && argv[dirIdx + 1]
|
|
49
|
+
? join(process.cwd(), argv[dirIdx + 1])
|
|
50
|
+
: join(process.cwd(), ".judges", "branches");
|
|
51
|
+
if (!existsSync(dirPath)) {
|
|
52
|
+
console.log(`No branch findings directory found at: ${dirPath}`);
|
|
53
|
+
console.log("Place per-branch verdict JSON files in .judges/branches/");
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
const files = readdirSync(dirPath).filter((f) => f.endsWith(".json"));
|
|
57
|
+
if (files.length === 0) {
|
|
58
|
+
console.log("No branch verdict files found.");
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
const branchFindings = new Map();
|
|
62
|
+
for (const file of files) {
|
|
63
|
+
const branchName = file.replace(/\.json$/, "");
|
|
64
|
+
const content = JSON.parse(readFileSync(join(dirPath, file), "utf-8"));
|
|
65
|
+
const findings = content.findings ?? [];
|
|
66
|
+
branchFindings.set(branchName, findings);
|
|
67
|
+
}
|
|
68
|
+
const entries = detectMergeStrategy(branchFindings);
|
|
69
|
+
if (format === "json") {
|
|
70
|
+
console.log(JSON.stringify(entries, null, 2));
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
console.log("\n=== Finding Merge Strategy ===\n");
|
|
74
|
+
console.log(`Branches analyzed: ${branchFindings.size}`);
|
|
75
|
+
console.log(`Total entries: ${entries.length}\n`);
|
|
76
|
+
const duplicates = entries.filter((e) => e.status === "duplicate").length;
|
|
77
|
+
const unique = entries.filter((e) => e.status === "unique").length;
|
|
78
|
+
console.log(` Unique: ${unique}`);
|
|
79
|
+
console.log(` Duplicates: ${duplicates}\n`);
|
|
80
|
+
for (const entry of entries) {
|
|
81
|
+
console.log(`[${entry.status.toUpperCase()}] ${entry.ruleId} (${entry.severity}) — ${entry.branch}`);
|
|
82
|
+
console.log(` Strategy: ${entry.strategy}`);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=finding-merge-strategy.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-merge-strategy.js","sourceRoot":"","sources":["../../src/commands/finding-merge-strategy.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-similar-match.d.ts","sourceRoot":"","sources":["../../src/commands/finding-similar-match.ts"],"names":[],"mappings":"AAsFA,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAiE3D"}
|