@kevinrabun/judges 3.87.0 → 3.88.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 +16 -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-age-analysis.d.ts +5 -0
- package/dist/commands/finding-age-analysis.d.ts.map +1 -0
- package/dist/commands/finding-age-analysis.js +145 -0
- package/dist/commands/finding-age-analysis.js.map +1 -0
- package/dist/commands/finding-correlation.d.ts +5 -0
- package/dist/commands/finding-correlation.d.ts.map +1 -0
- package/dist/commands/finding-correlation.js +104 -0
- package/dist/commands/finding-correlation.js.map +1 -0
- package/dist/commands/finding-dependency-tree.d.ts +5 -0
- package/dist/commands/finding-dependency-tree.d.ts.map +1 -0
- package/dist/commands/finding-dependency-tree.js +117 -0
- package/dist/commands/finding-dependency-tree.js.map +1 -0
- package/dist/commands/finding-suppression-audit.d.ts +5 -0
- package/dist/commands/finding-suppression-audit.d.ts.map +1 -0
- package/dist/commands/finding-suppression-audit.js +138 -0
- package/dist/commands/finding-suppression-audit.js.map +1 -0
- package/dist/commands/review-ci-integration.d.ts +5 -0
- package/dist/commands/review-ci-integration.d.ts.map +1 -0
- package/dist/commands/review-ci-integration.js +126 -0
- package/dist/commands/review-ci-integration.js.map +1 -0
- package/dist/commands/review-comparative.d.ts +5 -0
- package/dist/commands/review-comparative.d.ts.map +1 -0
- package/dist/commands/review-comparative.js +150 -0
- package/dist/commands/review-comparative.js.map +1 -0
- package/dist/commands/review-custom-rule.d.ts +5 -0
- package/dist/commands/review-custom-rule.d.ts.map +1 -0
- package/dist/commands/review-custom-rule.js +170 -0
- package/dist/commands/review-custom-rule.js.map +1 -0
- package/dist/commands/review-notification.d.ts +5 -0
- package/dist/commands/review-notification.d.ts.map +1 -0
- package/dist/commands/review-notification.js +127 -0
- package/dist/commands/review-notification.js.map +1 -0
- package/dist/commands/review-template-export.d.ts +5 -0
- package/dist/commands/review-template-export.d.ts.map +1 -0
- package/dist/commands/review-template-export.js +147 -0
- package/dist/commands/review-template-export.js.map +1 -0
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-age-analysis.d.ts","sourceRoot":"","sources":["../../src/commands/finding-age-analysis.ts"],"names":[],"mappings":"AAAA;;GAEG;AAmFH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAyF1D"}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-age-analysis — Analyze the age and lifecycle of findings over time.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync, readdirSync } from "fs";
|
|
5
|
+
// ─── Analysis ───────────────────────────────────────────────────────────────
|
|
6
|
+
function analyzeAge(verdicts) {
|
|
7
|
+
const ruleOccurrences = new Map();
|
|
8
|
+
const ruleFirstSeen = new Map();
|
|
9
|
+
const now = Date.now();
|
|
10
|
+
const timestamps = verdicts.map((v) => v.timestamp).sort();
|
|
11
|
+
for (const v of verdicts) {
|
|
12
|
+
for (const f of v.verdict.findings) {
|
|
13
|
+
ruleOccurrences.set(f.ruleId, (ruleOccurrences.get(f.ruleId) || 0) + 1);
|
|
14
|
+
if (!ruleFirstSeen.has(f.ruleId)) {
|
|
15
|
+
ruleFirstSeen.set(f.ruleId, v.timestamp);
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
// bucket by age
|
|
20
|
+
const buckets = [
|
|
21
|
+
{ label: "< 1 day", count: 0, ruleIds: [] },
|
|
22
|
+
{ label: "1-7 days", count: 0, ruleIds: [] },
|
|
23
|
+
{ label: "1-4 weeks", count: 0, ruleIds: [] },
|
|
24
|
+
{ label: "> 4 weeks", count: 0, ruleIds: [] },
|
|
25
|
+
];
|
|
26
|
+
for (const [ruleId, firstSeen] of ruleFirstSeen) {
|
|
27
|
+
const age = now - new Date(firstSeen).getTime();
|
|
28
|
+
const days = age / (1000 * 60 * 60 * 24);
|
|
29
|
+
if (days < 1) {
|
|
30
|
+
buckets[0].count++;
|
|
31
|
+
buckets[0].ruleIds.push(ruleId);
|
|
32
|
+
}
|
|
33
|
+
else if (days < 7) {
|
|
34
|
+
buckets[1].count++;
|
|
35
|
+
buckets[1].ruleIds.push(ruleId);
|
|
36
|
+
}
|
|
37
|
+
else if (days < 28) {
|
|
38
|
+
buckets[2].count++;
|
|
39
|
+
buckets[2].ruleIds.push(ruleId);
|
|
40
|
+
}
|
|
41
|
+
else {
|
|
42
|
+
buckets[3].count++;
|
|
43
|
+
buckets[3].ruleIds.push(ruleId);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
// recurring rules
|
|
47
|
+
const recurring = [...ruleOccurrences.entries()]
|
|
48
|
+
.filter(([, count]) => count > 1)
|
|
49
|
+
.sort((a, b) => b[1] - a[1])
|
|
50
|
+
.map(([ruleId, occurrences]) => ({ ruleId, occurrences }));
|
|
51
|
+
return {
|
|
52
|
+
totalFindings: ruleFirstSeen.size,
|
|
53
|
+
oldestTimestamp: timestamps[0] || "",
|
|
54
|
+
newestTimestamp: timestamps[timestamps.length - 1] || "",
|
|
55
|
+
buckets,
|
|
56
|
+
recurringRules: recurring,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
60
|
+
export function runFindingAgeAnalysis(argv) {
|
|
61
|
+
const dirIdx = argv.indexOf("--dir");
|
|
62
|
+
const fileIdx = argv.indexOf("--file");
|
|
63
|
+
const formatIdx = argv.indexOf("--format");
|
|
64
|
+
const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
|
|
65
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
66
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
67
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
68
|
+
console.log(`
|
|
69
|
+
judges finding-age-analysis — Analyze finding age over time
|
|
70
|
+
|
|
71
|
+
Usage:
|
|
72
|
+
judges finding-age-analysis --dir <verdicts-dir> [--format table|json]
|
|
73
|
+
judges finding-age-analysis --file <verdict.json> [--format table|json]
|
|
74
|
+
|
|
75
|
+
Options:
|
|
76
|
+
--dir <path> Directory of verdict JSON files
|
|
77
|
+
--file <path> Single verdict JSON file
|
|
78
|
+
--format <fmt> Output format: table (default), json
|
|
79
|
+
--help, -h Show this help
|
|
80
|
+
`);
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
const verdicts = [];
|
|
84
|
+
if (dirPath && existsSync(dirPath)) {
|
|
85
|
+
const files = readdirSync(dirPath);
|
|
86
|
+
for (const file of files) {
|
|
87
|
+
if (!file.endsWith(".json"))
|
|
88
|
+
continue;
|
|
89
|
+
try {
|
|
90
|
+
const content = readFileSync(`${dirPath}/${file}`, "utf-8");
|
|
91
|
+
const verdict = JSON.parse(content);
|
|
92
|
+
verdicts.push({ verdict, timestamp: verdict.timestamp || file.replace(".json", "") });
|
|
93
|
+
}
|
|
94
|
+
catch {
|
|
95
|
+
// skip invalid files
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
else if (filePath && existsSync(filePath)) {
|
|
100
|
+
try {
|
|
101
|
+
const verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
102
|
+
verdicts.push({ verdict, timestamp: verdict.timestamp || new Date().toISOString() });
|
|
103
|
+
}
|
|
104
|
+
catch {
|
|
105
|
+
console.error("Error: invalid JSON");
|
|
106
|
+
process.exitCode = 1;
|
|
107
|
+
return;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
console.error("Error: --dir or --file required");
|
|
112
|
+
process.exitCode = 1;
|
|
113
|
+
return;
|
|
114
|
+
}
|
|
115
|
+
if (verdicts.length === 0) {
|
|
116
|
+
console.log("No verdict files found");
|
|
117
|
+
return;
|
|
118
|
+
}
|
|
119
|
+
const report = analyzeAge(verdicts);
|
|
120
|
+
if (format === "json") {
|
|
121
|
+
console.log(JSON.stringify(report, null, 2));
|
|
122
|
+
return;
|
|
123
|
+
}
|
|
124
|
+
console.log(`\nFinding Age Analysis`);
|
|
125
|
+
console.log("═".repeat(60));
|
|
126
|
+
console.log(` Total unique findings: ${report.totalFindings}`);
|
|
127
|
+
console.log(` Reports analyzed: ${verdicts.length}`);
|
|
128
|
+
console.log(` Range: ${report.oldestTimestamp} → ${report.newestTimestamp}`);
|
|
129
|
+
console.log("─".repeat(60));
|
|
130
|
+
console.log(`${"Age Bucket".padEnd(16)} ${"Count".padEnd(8)} Rules`);
|
|
131
|
+
console.log("─".repeat(60));
|
|
132
|
+
for (const b of report.buckets) {
|
|
133
|
+
const rules = b.ruleIds.slice(0, 3).join(", ");
|
|
134
|
+
const more = b.ruleIds.length > 3 ? ` +${b.ruleIds.length - 3} more` : "";
|
|
135
|
+
console.log(`${b.label.padEnd(16)} ${String(b.count).padEnd(8)} ${rules}${more}`);
|
|
136
|
+
}
|
|
137
|
+
if (report.recurringRules.length > 0) {
|
|
138
|
+
console.log(`\n Recurring Rules (${report.recurringRules.length}):`);
|
|
139
|
+
for (const r of report.recurringRules.slice(0, 10)) {
|
|
140
|
+
console.log(` ${r.ruleId.padEnd(20)} × ${r.occurrences}`);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
console.log("═".repeat(60));
|
|
144
|
+
}
|
|
145
|
+
//# sourceMappingURL=finding-age-analysis.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-age-analysis.js","sourceRoot":"","sources":["../../src/commands/finding-age-analysis.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-correlation.d.ts","sourceRoot":"","sources":["../../src/commands/finding-correlation.ts"],"names":[],"mappings":"AAAA;;GAEG;AA6DH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAuE1D"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-correlation — Find correlations between findings across reports.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync, readdirSync } from "fs";
|
|
5
|
+
// ─── Analysis ───────────────────────────────────────────────────────────────
|
|
6
|
+
function computeCorrelations(verdicts) {
|
|
7
|
+
// count co-occurrences of rule pairs across reports
|
|
8
|
+
const pairCounts = new Map();
|
|
9
|
+
const ruleCounts = new Map();
|
|
10
|
+
const totalReports = verdicts.length;
|
|
11
|
+
for (const v of verdicts) {
|
|
12
|
+
const rulesInReport = new Set(v.findings.map((f) => f.ruleId));
|
|
13
|
+
const ruleList = [...rulesInReport].sort();
|
|
14
|
+
for (const rule of ruleList) {
|
|
15
|
+
ruleCounts.set(rule, (ruleCounts.get(rule) || 0) + 1);
|
|
16
|
+
}
|
|
17
|
+
for (let i = 0; i < ruleList.length; i++) {
|
|
18
|
+
for (let j = i + 1; j < ruleList.length; j++) {
|
|
19
|
+
const key = `${ruleList[i]}|||${ruleList[j]}`;
|
|
20
|
+
pairCounts.set(key, (pairCounts.get(key) || 0) + 1);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
const correlations = [];
|
|
25
|
+
for (const [key, coOccurrences] of pairCounts) {
|
|
26
|
+
const [ruleA, ruleB] = key.split("|||");
|
|
27
|
+
const countA = ruleCounts.get(ruleA) || 0;
|
|
28
|
+
const countB = ruleCounts.get(ruleB) || 0;
|
|
29
|
+
// Jaccard similarity as correlation proxy
|
|
30
|
+
const union = countA + countB - coOccurrences;
|
|
31
|
+
const correlation = union > 0 ? coOccurrences / union : 0;
|
|
32
|
+
if (coOccurrences >= 2) {
|
|
33
|
+
correlations.push({ ruleA, ruleB, coOccurrences, totalReports, correlation });
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
correlations.sort((a, b) => b.correlation - a.correlation);
|
|
37
|
+
return correlations;
|
|
38
|
+
}
|
|
39
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
40
|
+
export function runFindingCorrelation(argv) {
|
|
41
|
+
const dirIdx = argv.indexOf("--dir");
|
|
42
|
+
const formatIdx = argv.indexOf("--format");
|
|
43
|
+
const limitIdx = argv.indexOf("--limit");
|
|
44
|
+
const minIdx = argv.indexOf("--min-correlation");
|
|
45
|
+
const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
|
|
46
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
47
|
+
const limit = limitIdx >= 0 ? parseInt(argv[limitIdx + 1], 10) : 20;
|
|
48
|
+
const minCorrelation = minIdx >= 0 ? parseFloat(argv[minIdx + 1]) : 0.3;
|
|
49
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
50
|
+
console.log(`
|
|
51
|
+
judges finding-correlation — Find correlations between findings
|
|
52
|
+
|
|
53
|
+
Usage:
|
|
54
|
+
judges finding-correlation --dir <verdicts-dir> [--format table|json]
|
|
55
|
+
[--limit <n>] [--min-correlation <0-1>]
|
|
56
|
+
|
|
57
|
+
Options:
|
|
58
|
+
--dir <path> Directory of verdict JSON files (required)
|
|
59
|
+
--format <fmt> Output format: table (default), json
|
|
60
|
+
--limit <n> Max results (default: 20)
|
|
61
|
+
--min-correlation <n> Minimum correlation threshold (default: 0.3)
|
|
62
|
+
--help, -h Show this help
|
|
63
|
+
`);
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
if (!dirPath || !existsSync(dirPath)) {
|
|
67
|
+
console.error("Error: --dir required and must exist");
|
|
68
|
+
process.exitCode = 1;
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
const verdicts = [];
|
|
72
|
+
const files = readdirSync(dirPath);
|
|
73
|
+
for (const file of files) {
|
|
74
|
+
if (!file.endsWith(".json"))
|
|
75
|
+
continue;
|
|
76
|
+
try {
|
|
77
|
+
verdicts.push(JSON.parse(readFileSync(`${dirPath}/${file}`, "utf-8")));
|
|
78
|
+
}
|
|
79
|
+
catch {
|
|
80
|
+
// skip invalid files
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
if (verdicts.length < 2) {
|
|
84
|
+
console.log("Need at least 2 verdict reports for correlation analysis");
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
let correlations = computeCorrelations(verdicts);
|
|
88
|
+
correlations = correlations.filter((c) => c.correlation >= minCorrelation).slice(0, limit);
|
|
89
|
+
if (format === "json") {
|
|
90
|
+
console.log(JSON.stringify(correlations, null, 2));
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
console.log(`\nFinding Correlations (${verdicts.length} reports analyzed)`);
|
|
94
|
+
console.log("═".repeat(75));
|
|
95
|
+
console.log(`${"Rule A".padEnd(20)} ${"Rule B".padEnd(20)} ${"Co-occur".padEnd(10)} ${"Correlation".padEnd(14)}`);
|
|
96
|
+
console.log("─".repeat(75));
|
|
97
|
+
for (const c of correlations) {
|
|
98
|
+
const rA = c.ruleA.length > 18 ? c.ruleA.slice(0, 18) + "…" : c.ruleA;
|
|
99
|
+
const rB = c.ruleB.length > 18 ? c.ruleB.slice(0, 18) + "…" : c.ruleB;
|
|
100
|
+
console.log(`${rA.padEnd(20)} ${rB.padEnd(20)} ${String(c.coOccurrences).padEnd(10)} ${c.correlation.toFixed(3).padEnd(14)}`);
|
|
101
|
+
}
|
|
102
|
+
console.log("═".repeat(75));
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=finding-correlation.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-correlation.js","sourceRoot":"","sources":["../../src/commands/finding-correlation.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dependency-tree.d.ts","sourceRoot":"","sources":["../../src/commands/finding-dependency-tree.ts"],"names":[],"mappings":"AAAA;;GAEG;AAgFH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA8D7D"}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-dependency-tree — Visualize dependency relationships among findings.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Analysis ───────────────────────────────────────────────────────────────
|
|
6
|
+
function buildDependencyTree(verdict) {
|
|
7
|
+
const nodes = new Map();
|
|
8
|
+
const findings = verdict.findings;
|
|
9
|
+
// create nodes
|
|
10
|
+
for (const f of findings) {
|
|
11
|
+
nodes.set(f.ruleId, {
|
|
12
|
+
ruleId: f.ruleId,
|
|
13
|
+
title: f.title,
|
|
14
|
+
severity: (f.severity || "medium").toLowerCase(),
|
|
15
|
+
children: [],
|
|
16
|
+
depth: 0,
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
// build relationships — findings sharing line numbers are related
|
|
20
|
+
for (let i = 0; i < findings.length; i++) {
|
|
21
|
+
for (let j = i + 1; j < findings.length; j++) {
|
|
22
|
+
const a = findings[i];
|
|
23
|
+
const b = findings[j];
|
|
24
|
+
const aLines = new Set(a.lineNumbers || []);
|
|
25
|
+
const bLines = b.lineNumbers || [];
|
|
26
|
+
if (bLines.some((ln) => aLines.has(ln))) {
|
|
27
|
+
const nodeA = nodes.get(a.ruleId);
|
|
28
|
+
const nodeB = nodes.get(b.ruleId);
|
|
29
|
+
if (nodeA !== undefined && !nodeA.children.includes(b.ruleId)) {
|
|
30
|
+
nodeA.children.push(b.ruleId);
|
|
31
|
+
}
|
|
32
|
+
if (nodeB !== undefined && !nodeB.children.includes(a.ruleId)) {
|
|
33
|
+
nodeB.children.push(a.ruleId);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
// compute depths via BFS
|
|
39
|
+
const roots = [...nodes.values()].filter((n) => ![...nodes.values()].some((other) => other.children.includes(n.ruleId)));
|
|
40
|
+
for (const root of roots) {
|
|
41
|
+
const queue = [{ id: root.ruleId, depth: 0 }];
|
|
42
|
+
const visited = new Set();
|
|
43
|
+
while (queue.length > 0) {
|
|
44
|
+
const item = queue.shift();
|
|
45
|
+
if (visited.has(item.id))
|
|
46
|
+
continue;
|
|
47
|
+
visited.add(item.id);
|
|
48
|
+
const node = nodes.get(item.id);
|
|
49
|
+
if (node !== undefined) {
|
|
50
|
+
node.depth = Math.max(node.depth, item.depth);
|
|
51
|
+
for (const child of node.children) {
|
|
52
|
+
queue.push({ id: child, depth: item.depth + 1 });
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
return [...nodes.values()].sort((a, b) => a.depth - b.depth);
|
|
58
|
+
}
|
|
59
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
60
|
+
export function runFindingDependencyTree(argv) {
|
|
61
|
+
const fileIdx = argv.indexOf("--file");
|
|
62
|
+
const formatIdx = argv.indexOf("--format");
|
|
63
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
64
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "tree";
|
|
65
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
66
|
+
console.log(`
|
|
67
|
+
judges finding-dependency-tree — Visualize finding dependencies
|
|
68
|
+
|
|
69
|
+
Usage:
|
|
70
|
+
judges finding-dependency-tree --file <verdict.json> [--format tree|json]
|
|
71
|
+
|
|
72
|
+
Options:
|
|
73
|
+
--file <path> Path to verdict JSON file (required)
|
|
74
|
+
--format <fmt> Output format: tree (default), json
|
|
75
|
+
--help, -h Show this help
|
|
76
|
+
`);
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
if (!filePath) {
|
|
80
|
+
console.error("Error: --file required");
|
|
81
|
+
process.exitCode = 1;
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (!existsSync(filePath)) {
|
|
85
|
+
console.error(`Error: not found: ${filePath}`);
|
|
86
|
+
process.exitCode = 1;
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
let verdict;
|
|
90
|
+
try {
|
|
91
|
+
verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
92
|
+
}
|
|
93
|
+
catch {
|
|
94
|
+
console.error("Error: invalid JSON");
|
|
95
|
+
process.exitCode = 1;
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
const nodes = buildDependencyTree(verdict);
|
|
99
|
+
if (format === "json") {
|
|
100
|
+
console.log(JSON.stringify(nodes, null, 2));
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
console.log(`\nFinding Dependency Tree (${nodes.length} nodes)`);
|
|
104
|
+
console.log("═".repeat(70));
|
|
105
|
+
for (const node of nodes) {
|
|
106
|
+
const indent = " ".repeat(node.depth);
|
|
107
|
+
const connector = node.depth > 0 ? "├─ " : "";
|
|
108
|
+
const title = node.title.length > 35 ? node.title.slice(0, 35) + "…" : node.title;
|
|
109
|
+
console.log(`${indent}${connector}[${node.severity.toUpperCase()}] ${node.ruleId}`);
|
|
110
|
+
console.log(`${indent} ${title}`);
|
|
111
|
+
if (node.children.length > 0) {
|
|
112
|
+
console.log(`${indent} └─ related: ${node.children.join(", ")}`);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
console.log("═".repeat(70));
|
|
116
|
+
}
|
|
117
|
+
//# sourceMappingURL=finding-dependency-tree.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-dependency-tree.js","sourceRoot":"","sources":["../../src/commands/finding-dependency-tree.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-suppression-audit.d.ts","sourceRoot":"","sources":["../../src/commands/finding-suppression-audit.ts"],"names":[],"mappings":"AAAA;;GAEG;AAkGH,wBAAgB,0BAA0B,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA2E/D"}
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-suppression-audit — Audit suppressed/ignored findings for review.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Analysis ───────────────────────────────────────────────────────────────
|
|
6
|
+
function auditSuppressions(verdict, sourceFile, disabledRules, minSeverity) {
|
|
7
|
+
const records = [];
|
|
8
|
+
const disabled = new Set(disabledRules || []);
|
|
9
|
+
// check source for inline suppressions
|
|
10
|
+
let sourceLines = [];
|
|
11
|
+
if (sourceFile && existsSync(sourceFile)) {
|
|
12
|
+
sourceLines = readFileSync(sourceFile, "utf-8").split("\n");
|
|
13
|
+
}
|
|
14
|
+
// inline comment suppressions
|
|
15
|
+
const suppressionPatterns = [
|
|
16
|
+
/eslint-disable/,
|
|
17
|
+
/noqa/,
|
|
18
|
+
/noinspection/,
|
|
19
|
+
/@SuppressWarnings/,
|
|
20
|
+
/NOSONAR/,
|
|
21
|
+
/judges-ignore/,
|
|
22
|
+
/# type:\s*ignore/,
|
|
23
|
+
];
|
|
24
|
+
for (let i = 0; i < sourceLines.length; i++) {
|
|
25
|
+
const line = sourceLines[i];
|
|
26
|
+
for (const pat of suppressionPatterns) {
|
|
27
|
+
if (pat.test(line)) {
|
|
28
|
+
records.push({
|
|
29
|
+
ruleId: `LINE-${i + 1}`,
|
|
30
|
+
title: `Inline suppression at line ${i + 1}`,
|
|
31
|
+
severity: "medium",
|
|
32
|
+
suppressionType: "inline-comment",
|
|
33
|
+
reason: line.trim().slice(0, 80),
|
|
34
|
+
risk: "medium",
|
|
35
|
+
});
|
|
36
|
+
break;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
// config-disabled rules
|
|
41
|
+
for (const ruleId of disabled) {
|
|
42
|
+
const finding = verdict.findings.find((f) => f.ruleId === ruleId);
|
|
43
|
+
records.push({
|
|
44
|
+
ruleId,
|
|
45
|
+
title: finding ? finding.title : `Disabled rule: ${ruleId}`,
|
|
46
|
+
severity: finding ? (finding.severity || "medium").toLowerCase() : "unknown",
|
|
47
|
+
suppressionType: "config-disabled",
|
|
48
|
+
reason: "Rule disabled in configuration",
|
|
49
|
+
risk: "high",
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
// severity-filtered findings
|
|
53
|
+
if (minSeverity !== undefined) {
|
|
54
|
+
const sevOrder = { critical: 0, high: 1, medium: 2, low: 3, info: 4 };
|
|
55
|
+
const threshold = sevOrder[minSeverity] || 2;
|
|
56
|
+
for (const f of verdict.findings) {
|
|
57
|
+
const fSev = (f.severity || "medium").toLowerCase();
|
|
58
|
+
if ((sevOrder[fSev] || 2) > threshold) {
|
|
59
|
+
records.push({
|
|
60
|
+
ruleId: f.ruleId,
|
|
61
|
+
title: f.title,
|
|
62
|
+
severity: fSev,
|
|
63
|
+
suppressionType: "severity-filter",
|
|
64
|
+
reason: `Below minimum severity: ${minSeverity}`,
|
|
65
|
+
risk: "low",
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return records;
|
|
71
|
+
}
|
|
72
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
73
|
+
export function runFindingSuppressionAudit(argv) {
|
|
74
|
+
const fileIdx = argv.indexOf("--file");
|
|
75
|
+
const sourceIdx = argv.indexOf("--source");
|
|
76
|
+
const disabledIdx = argv.indexOf("--disabled");
|
|
77
|
+
const formatIdx = argv.indexOf("--format");
|
|
78
|
+
const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
|
|
79
|
+
const sourceFile = sourceIdx >= 0 ? argv[sourceIdx + 1] : undefined;
|
|
80
|
+
const disabledStr = disabledIdx >= 0 ? argv[disabledIdx + 1] : undefined;
|
|
81
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
82
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
83
|
+
console.log(`
|
|
84
|
+
judges finding-suppression-audit — Audit suppressed findings
|
|
85
|
+
|
|
86
|
+
Usage:
|
|
87
|
+
judges finding-suppression-audit --file <verdict.json> [--source <src.ts>]
|
|
88
|
+
[--disabled <RULE1,RULE2>] [--format table|json]
|
|
89
|
+
|
|
90
|
+
Options:
|
|
91
|
+
--file <path> Path to verdict JSON file (required)
|
|
92
|
+
--source <path> Source file to check for inline suppressions
|
|
93
|
+
--disabled <rules> Comma-separated disabled rule IDs
|
|
94
|
+
--format <fmt> Output format: table (default), json
|
|
95
|
+
--help, -h Show this help
|
|
96
|
+
`);
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
if (!filePath) {
|
|
100
|
+
console.error("Error: --file required");
|
|
101
|
+
process.exitCode = 1;
|
|
102
|
+
return;
|
|
103
|
+
}
|
|
104
|
+
if (!existsSync(filePath)) {
|
|
105
|
+
console.error(`Error: not found: ${filePath}`);
|
|
106
|
+
process.exitCode = 1;
|
|
107
|
+
return;
|
|
108
|
+
}
|
|
109
|
+
let verdict;
|
|
110
|
+
try {
|
|
111
|
+
verdict = JSON.parse(readFileSync(filePath, "utf-8"));
|
|
112
|
+
}
|
|
113
|
+
catch {
|
|
114
|
+
console.error("Error: invalid JSON");
|
|
115
|
+
process.exitCode = 1;
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
const disabled = disabledStr ? disabledStr.split(",").map((s) => s.trim()) : [];
|
|
119
|
+
const records = auditSuppressions(verdict, sourceFile, disabled);
|
|
120
|
+
if (format === "json") {
|
|
121
|
+
console.log(JSON.stringify(records, null, 2));
|
|
122
|
+
return;
|
|
123
|
+
}
|
|
124
|
+
const highRisk = records.filter((r) => r.risk === "high").length;
|
|
125
|
+
const medRisk = records.filter((r) => r.risk === "medium").length;
|
|
126
|
+
console.log(`\nSuppression Audit (${records.length} suppressions)`);
|
|
127
|
+
console.log("═".repeat(75));
|
|
128
|
+
console.log(` High risk: ${highRisk} | Medium risk: ${medRisk} | Low risk: ${records.length - highRisk - medRisk}`);
|
|
129
|
+
console.log("─".repeat(75));
|
|
130
|
+
console.log(`${"Risk".padEnd(8)} ${"Type".padEnd(18)} ${"Rule".padEnd(18)} ${"Severity".padEnd(10)} Reason`);
|
|
131
|
+
console.log("─".repeat(75));
|
|
132
|
+
for (const r of records) {
|
|
133
|
+
const reason = r.reason.length > 25 ? r.reason.slice(0, 25) + "…" : r.reason;
|
|
134
|
+
console.log(`${r.risk.padEnd(8)} ${r.suppressionType.padEnd(18)} ${r.ruleId.padEnd(18)} ${r.severity.padEnd(10)} ${reason}`);
|
|
135
|
+
}
|
|
136
|
+
console.log("═".repeat(75));
|
|
137
|
+
}
|
|
138
|
+
//# sourceMappingURL=finding-suppression-audit.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-suppression-audit.js","sourceRoot":"","sources":["../../src/commands/finding-suppression-audit.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-ci-integration.d.ts","sourceRoot":"","sources":["../../src/commands/review-ci-integration.ts"],"names":[],"mappings":"AAAA;;GAEG;AA8EH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAwD3D"}
|