@kevinrabun/judges 3.96.0 → 3.98.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-layer.d.ts +5 -0
- package/dist/commands/finding-annotation-layer.d.ts.map +1 -0
- package/dist/commands/finding-annotation-layer.js +129 -0
- package/dist/commands/finding-annotation-layer.js.map +1 -0
- package/dist/commands/finding-cwe-lookup.d.ts +5 -0
- package/dist/commands/finding-cwe-lookup.d.ts.map +1 -0
- package/dist/commands/finding-cwe-lookup.js +149 -0
- package/dist/commands/finding-cwe-lookup.js.map +1 -0
- package/dist/commands/finding-duplicate-detect.d.ts +5 -0
- package/dist/commands/finding-duplicate-detect.d.ts.map +1 -0
- package/dist/commands/finding-duplicate-detect.js +114 -0
- package/dist/commands/finding-duplicate-detect.js.map +1 -0
- package/dist/commands/finding-impact-rank.d.ts +5 -0
- package/dist/commands/finding-impact-rank.d.ts.map +1 -0
- package/dist/commands/finding-impact-rank.js +86 -0
- package/dist/commands/finding-impact-rank.js.map +1 -0
- package/dist/commands/finding-patch-preview.d.ts +5 -0
- package/dist/commands/finding-patch-preview.d.ts.map +1 -0
- package/dist/commands/finding-patch-preview.js +104 -0
- package/dist/commands/finding-patch-preview.js.map +1 -0
- package/dist/commands/finding-priority-matrix.d.ts +5 -0
- package/dist/commands/finding-priority-matrix.d.ts.map +1 -0
- package/dist/commands/finding-priority-matrix.js +103 -0
- package/dist/commands/finding-priority-matrix.js.map +1 -0
- package/dist/commands/finding-risk-score.d.ts +5 -0
- package/dist/commands/finding-risk-score.d.ts.map +1 -0
- package/dist/commands/finding-risk-score.js +96 -0
- package/dist/commands/finding-risk-score.js.map +1 -0
- package/dist/commands/finding-trend-forecast.d.ts +5 -0
- package/dist/commands/finding-trend-forecast.d.ts.map +1 -0
- package/dist/commands/finding-trend-forecast.js +107 -0
- package/dist/commands/finding-trend-forecast.js.map +1 -0
- package/dist/commands/review-cicd-integrate.d.ts +5 -0
- package/dist/commands/review-cicd-integrate.d.ts.map +1 -0
- package/dist/commands/review-cicd-integrate.js +123 -0
- package/dist/commands/review-cicd-integrate.js.map +1 -0
- package/dist/commands/review-compliance-map.d.ts +5 -0
- package/dist/commands/review-compliance-map.d.ts.map +1 -0
- package/dist/commands/review-compliance-map.js +111 -0
- package/dist/commands/review-compliance-map.js.map +1 -0
- package/dist/commands/review-gate-config.d.ts +5 -0
- package/dist/commands/review-gate-config.d.ts.map +1 -0
- package/dist/commands/review-gate-config.js +154 -0
- package/dist/commands/review-gate-config.js.map +1 -0
- package/dist/commands/review-language-profile.d.ts +5 -0
- package/dist/commands/review-language-profile.d.ts.map +1 -0
- package/dist/commands/review-language-profile.js +73 -0
- package/dist/commands/review-language-profile.js.map +1 -0
- package/dist/commands/review-org-dashboard.d.ts +5 -0
- package/dist/commands/review-org-dashboard.d.ts.map +1 -0
- package/dist/commands/review-org-dashboard.js +69 -0
- package/dist/commands/review-org-dashboard.js.map +1 -0
- package/dist/commands/review-policy-engine.d.ts +5 -0
- package/dist/commands/review-policy-engine.d.ts.map +1 -0
- package/dist/commands/review-policy-engine.js +136 -0
- package/dist/commands/review-policy-engine.js.map +1 -0
- package/dist/commands/review-report-archive.d.ts +5 -0
- package/dist/commands/review-report-archive.d.ts.map +1 -0
- package/dist/commands/review-report-archive.js +101 -0
- package/dist/commands/review-report-archive.js.map +1 -0
- package/dist/commands/review-rollout-plan.d.ts +5 -0
- package/dist/commands/review-rollout-plan.d.ts.map +1 -0
- package/dist/commands/review-rollout-plan.js +124 -0
- package/dist/commands/review-rollout-plan.js.map +1 -0
- package/dist/commands/review-sla-config.d.ts +5 -0
- package/dist/commands/review-sla-config.d.ts.map +1 -0
- package/dist/commands/review-sla-config.js +89 -0
- package/dist/commands/review-sla-config.js.map +1 -0
- package/dist/commands/review-webhook-dispatch.d.ts +5 -0
- package/dist/commands/review-webhook-dispatch.d.ts.map +1 -0
- package/dist/commands/review-webhook-dispatch.js +100 -0
- package/dist/commands/review-webhook-dispatch.js.map +1 -0
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-patch-preview — Preview how patches would modify source files.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
6
|
+
export function runFindingPatchPreview(argv) {
|
|
7
|
+
const reportIdx = argv.indexOf("--report");
|
|
8
|
+
const sourceIdx = argv.indexOf("--source");
|
|
9
|
+
const contextIdx = argv.indexOf("--context");
|
|
10
|
+
const formatIdx = argv.indexOf("--format");
|
|
11
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
12
|
+
const contextLines = contextIdx >= 0 ? parseInt(argv[contextIdx + 1], 10) : 3;
|
|
13
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
14
|
+
console.log(`
|
|
15
|
+
judges finding-patch-preview — Preview patch modifications
|
|
16
|
+
|
|
17
|
+
Usage:
|
|
18
|
+
judges finding-patch-preview --report <path> --source <path>
|
|
19
|
+
[--context <n>] [--format table|json]
|
|
20
|
+
|
|
21
|
+
Options:
|
|
22
|
+
--report <path> Report file with findings
|
|
23
|
+
--source <path> Source file to preview patches against
|
|
24
|
+
--context <n> Context lines around changes (default: 3)
|
|
25
|
+
--format <fmt> Output format: table (default), json
|
|
26
|
+
--help, -h Show this help
|
|
27
|
+
`);
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
if (reportIdx < 0 || sourceIdx < 0) {
|
|
31
|
+
console.error("Missing --report <path> and --source <path>");
|
|
32
|
+
process.exitCode = 1;
|
|
33
|
+
return;
|
|
34
|
+
}
|
|
35
|
+
const reportPath = argv[reportIdx + 1];
|
|
36
|
+
const sourcePath = argv[sourceIdx + 1];
|
|
37
|
+
if (!existsSync(reportPath)) {
|
|
38
|
+
console.error(`Report not found: ${reportPath}`);
|
|
39
|
+
process.exitCode = 1;
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
if (!existsSync(sourcePath)) {
|
|
43
|
+
console.error(`Source not found: ${sourcePath}`);
|
|
44
|
+
process.exitCode = 1;
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
const report = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
48
|
+
const findings = (report.findings ?? []).filter((f) => f.patch !== undefined && f.patch !== null);
|
|
49
|
+
const sourceContent = readFileSync(sourcePath, "utf-8");
|
|
50
|
+
const sourceLines = sourceContent.split("\n");
|
|
51
|
+
if (findings.length === 0) {
|
|
52
|
+
console.log("No findings with patches to preview.");
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
const previews = [];
|
|
56
|
+
for (const f of findings) {
|
|
57
|
+
const patchStr = String(f.patch);
|
|
58
|
+
const patchLines = patchStr.split("\n");
|
|
59
|
+
const firstLine = patchLines[0] ?? "";
|
|
60
|
+
const lineIdx = sourceLines.findIndex((l) => l.includes(firstLine.trim()));
|
|
61
|
+
const applicable = lineIdx >= 0;
|
|
62
|
+
const start = Math.max(0, lineIdx - contextLines);
|
|
63
|
+
const end = Math.min(sourceLines.length - 1, lineIdx + contextLines);
|
|
64
|
+
const affectedLines = [];
|
|
65
|
+
if (applicable) {
|
|
66
|
+
for (let i = start; i <= end; i++) {
|
|
67
|
+
affectedLines.push(i + 1);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
const snippet = applicable
|
|
71
|
+
? sourceLines
|
|
72
|
+
.slice(start, end + 1)
|
|
73
|
+
.map((l, i) => {
|
|
74
|
+
const lineNum = start + i + 1;
|
|
75
|
+
const marker = lineNum === lineIdx + 1 ? ">" : " ";
|
|
76
|
+
return `${marker} ${String(lineNum).padStart(4)} | ${l}`;
|
|
77
|
+
})
|
|
78
|
+
.join("\n")
|
|
79
|
+
: "(patch target not found in source)";
|
|
80
|
+
previews.push({
|
|
81
|
+
ruleId: f.ruleId,
|
|
82
|
+
title: f.title,
|
|
83
|
+
patchSnippet: snippet,
|
|
84
|
+
applicable,
|
|
85
|
+
affectedLines,
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
if (format === "json") {
|
|
89
|
+
console.log(JSON.stringify(previews, null, 2));
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
console.log(`\nPatch Preview — ${sourcePath}`);
|
|
93
|
+
console.log("═".repeat(70));
|
|
94
|
+
for (const p of previews) {
|
|
95
|
+
const status = p.applicable ? "APPLICABLE" : "NOT FOUND";
|
|
96
|
+
console.log(`\n [${status}] ${p.ruleId} — ${p.title}`);
|
|
97
|
+
console.log(" " + "─".repeat(60));
|
|
98
|
+
console.log(p.patchSnippet);
|
|
99
|
+
}
|
|
100
|
+
const applicableCount = previews.filter((p) => p.applicable).length;
|
|
101
|
+
console.log(`\n ${applicableCount}/${previews.length} patches applicable`);
|
|
102
|
+
console.log("═".repeat(70));
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=finding-patch-preview.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-patch-preview.js","sourceRoot":"","sources":["../../src/commands/finding-patch-preview.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-priority-matrix.d.ts","sourceRoot":"","sources":["../../src/commands/finding-priority-matrix.ts"],"names":[],"mappings":"AAAA;;GAEG;AAmCH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAgG7D"}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-priority-matrix — Create a priority matrix (urgency × impact) for findings.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Matrix model ───────────────────────────────────────────────────────────
|
|
6
|
+
const URGENCY = {
|
|
7
|
+
critical: 5,
|
|
8
|
+
high: 4,
|
|
9
|
+
medium: 3,
|
|
10
|
+
low: 2,
|
|
11
|
+
info: 1,
|
|
12
|
+
};
|
|
13
|
+
function classifyPriority(score) {
|
|
14
|
+
if (score >= 20)
|
|
15
|
+
return "P0-Immediate";
|
|
16
|
+
if (score >= 12)
|
|
17
|
+
return "P1-High";
|
|
18
|
+
if (score >= 6)
|
|
19
|
+
return "P2-Medium";
|
|
20
|
+
if (score >= 3)
|
|
21
|
+
return "P3-Low";
|
|
22
|
+
return "P4-Backlog";
|
|
23
|
+
}
|
|
24
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
25
|
+
export function runFindingPriorityMatrix(argv) {
|
|
26
|
+
const reportIdx = argv.indexOf("--report");
|
|
27
|
+
const priorityIdx = argv.indexOf("--priority");
|
|
28
|
+
const formatIdx = argv.indexOf("--format");
|
|
29
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
30
|
+
const priorityFilter = priorityIdx >= 0 ? argv[priorityIdx + 1] : "";
|
|
31
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
32
|
+
console.log(`
|
|
33
|
+
judges finding-priority-matrix — Create urgency × impact priority matrix
|
|
34
|
+
|
|
35
|
+
Usage:
|
|
36
|
+
judges finding-priority-matrix --report <path> [--priority <level>]
|
|
37
|
+
[--format table|json]
|
|
38
|
+
|
|
39
|
+
Options:
|
|
40
|
+
--report <path> Report file with findings
|
|
41
|
+
--priority <level> Filter: P0-Immediate, P1-High, P2-Medium, P3-Low, P4-Backlog
|
|
42
|
+
--format <fmt> Output format: table (default), json
|
|
43
|
+
--help, -h Show this help
|
|
44
|
+
`);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
if (reportIdx < 0) {
|
|
48
|
+
console.error("Missing --report <path>");
|
|
49
|
+
process.exitCode = 1;
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
const reportPath = argv[reportIdx + 1];
|
|
53
|
+
if (!existsSync(reportPath)) {
|
|
54
|
+
console.error(`Report not found: ${reportPath}`);
|
|
55
|
+
process.exitCode = 1;
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
const report = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
59
|
+
const findings = report.findings ?? [];
|
|
60
|
+
if (findings.length === 0) {
|
|
61
|
+
console.log("No findings to prioritize.");
|
|
62
|
+
return;
|
|
63
|
+
}
|
|
64
|
+
const entries = findings.map((f) => {
|
|
65
|
+
const urgency = URGENCY[f.severity] ?? 1;
|
|
66
|
+
const conf = f.confidence ?? 0.5;
|
|
67
|
+
const impact = Math.round(urgency * conf * 2);
|
|
68
|
+
const score = urgency * impact;
|
|
69
|
+
return {
|
|
70
|
+
ruleId: f.ruleId,
|
|
71
|
+
title: f.title,
|
|
72
|
+
severity: f.severity,
|
|
73
|
+
urgency,
|
|
74
|
+
impact,
|
|
75
|
+
priority: classifyPriority(score),
|
|
76
|
+
score,
|
|
77
|
+
};
|
|
78
|
+
});
|
|
79
|
+
entries.sort((a, b) => b.score - a.score);
|
|
80
|
+
const display = priorityFilter.length > 0 ? entries.filter((e) => e.priority === priorityFilter) : entries;
|
|
81
|
+
if (format === "json") {
|
|
82
|
+
console.log(JSON.stringify(display, null, 2));
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
console.log(`\nPriority Matrix`);
|
|
86
|
+
console.log("═".repeat(80));
|
|
87
|
+
console.log(` ${"Priority".padEnd(15)} ${"Score".padEnd(7)} ${"Urgency".padEnd(9)} ${"Impact".padEnd(8)} ${"Rule".padEnd(22)} Title`);
|
|
88
|
+
console.log(" " + "─".repeat(75));
|
|
89
|
+
for (const e of display) {
|
|
90
|
+
console.log(` ${e.priority.padEnd(15)} ${String(e.score).padEnd(7)} ${String(e.urgency).padEnd(9)} ${String(e.impact).padEnd(8)} ${e.ruleId.padEnd(22)} ${e.title}`);
|
|
91
|
+
}
|
|
92
|
+
// Summary counts
|
|
93
|
+
const counts = {};
|
|
94
|
+
for (const e of entries) {
|
|
95
|
+
counts[e.priority] = (counts[e.priority] ?? 0) + 1;
|
|
96
|
+
}
|
|
97
|
+
console.log(`\n Summary:`);
|
|
98
|
+
for (const [p, c] of Object.entries(counts).sort()) {
|
|
99
|
+
console.log(` ${p.padEnd(15)} ${c} finding(s)`);
|
|
100
|
+
}
|
|
101
|
+
console.log("═".repeat(80));
|
|
102
|
+
}
|
|
103
|
+
//# sourceMappingURL=finding-priority-matrix.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-priority-matrix.js","sourceRoot":"","sources":["../../src/commands/finding-priority-matrix.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-risk-score.d.ts","sourceRoot":"","sources":["../../src/commands/finding-risk-score.ts"],"names":[],"mappings":"AAAA;;GAEG;AAqDH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAuExD"}
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-risk-score — Calculate composite risk scores for findings.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── Scoring ────────────────────────────────────────────────────────────────
|
|
6
|
+
const SEVERITY_WEIGHT = {
|
|
7
|
+
critical: 10,
|
|
8
|
+
high: 7,
|
|
9
|
+
medium: 4,
|
|
10
|
+
low: 2,
|
|
11
|
+
info: 1,
|
|
12
|
+
};
|
|
13
|
+
function scoreFinding(f) {
|
|
14
|
+
const factors = [];
|
|
15
|
+
let score = SEVERITY_WEIGHT[f.severity] ?? 1;
|
|
16
|
+
factors.push(`severity(${f.severity})=${SEVERITY_WEIGHT[f.severity] ?? 1}`);
|
|
17
|
+
const conf = f.confidence ?? 0.5;
|
|
18
|
+
score *= conf;
|
|
19
|
+
factors.push(`confidence=${conf}`);
|
|
20
|
+
if (f.patch !== undefined && f.patch !== null) {
|
|
21
|
+
score *= 1.2;
|
|
22
|
+
factors.push("has-patch(+20%)");
|
|
23
|
+
}
|
|
24
|
+
if (f.isAbsenceBased === true) {
|
|
25
|
+
score *= 0.8;
|
|
26
|
+
factors.push("absence-based(-20%)");
|
|
27
|
+
}
|
|
28
|
+
return {
|
|
29
|
+
ruleId: f.ruleId,
|
|
30
|
+
severity: f.severity,
|
|
31
|
+
title: f.title,
|
|
32
|
+
riskScore: Math.round(score * 100) / 100,
|
|
33
|
+
factors,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
37
|
+
export function runFindingRiskScore(argv) {
|
|
38
|
+
const reportIdx = argv.indexOf("--report");
|
|
39
|
+
const topIdx = argv.indexOf("--top");
|
|
40
|
+
const formatIdx = argv.indexOf("--format");
|
|
41
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
42
|
+
const topN = topIdx >= 0 ? parseInt(argv[topIdx + 1], 10) : 0;
|
|
43
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
44
|
+
console.log(`
|
|
45
|
+
judges finding-risk-score — Calculate composite risk scores
|
|
46
|
+
|
|
47
|
+
Usage:
|
|
48
|
+
judges finding-risk-score --report <path> [--top <n>] [--format table|json]
|
|
49
|
+
|
|
50
|
+
Options:
|
|
51
|
+
--report <path> Report file with findings
|
|
52
|
+
--top <n> Show only top N riskiest findings
|
|
53
|
+
--format <fmt> Output format: table (default), json
|
|
54
|
+
--help, -h Show this help
|
|
55
|
+
|
|
56
|
+
Risk score factors: severity weight * confidence * patch bonus * absence penalty
|
|
57
|
+
`);
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
if (reportIdx < 0) {
|
|
61
|
+
console.error("Missing --report <path>");
|
|
62
|
+
process.exitCode = 1;
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
const reportPath = argv[reportIdx + 1];
|
|
66
|
+
if (!existsSync(reportPath)) {
|
|
67
|
+
console.error(`Report not found: ${reportPath}`);
|
|
68
|
+
process.exitCode = 1;
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
const report = JSON.parse(readFileSync(reportPath, "utf-8"));
|
|
72
|
+
const findings = report.findings ?? [];
|
|
73
|
+
if (findings.length === 0) {
|
|
74
|
+
console.log("No findings to score.");
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
const scored = findings.map(scoreFinding);
|
|
78
|
+
scored.sort((a, b) => b.riskScore - a.riskScore);
|
|
79
|
+
const display = topN > 0 ? scored.slice(0, topN) : scored;
|
|
80
|
+
if (format === "json") {
|
|
81
|
+
console.log(JSON.stringify(display, null, 2));
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
console.log(`\nFinding Risk Scores`);
|
|
85
|
+
console.log("═".repeat(70));
|
|
86
|
+
console.log(` ${"Rank".padEnd(6)} ${"Risk".padEnd(8)} ${"Severity".padEnd(10)} ${"Rule".padEnd(25)} Title`);
|
|
87
|
+
console.log(" " + "─".repeat(65));
|
|
88
|
+
for (let i = 0; i < display.length; i++) {
|
|
89
|
+
const s = display[i];
|
|
90
|
+
console.log(` ${String(i + 1).padEnd(6)} ${String(s.riskScore).padEnd(8)} ${s.severity.padEnd(10)} ${s.ruleId.padEnd(25)} ${s.title}`);
|
|
91
|
+
}
|
|
92
|
+
const totalRisk = scored.reduce((sum, s) => sum + s.riskScore, 0);
|
|
93
|
+
console.log(`\n Total risk: ${Math.round(totalRisk * 100) / 100} across ${scored.length} findings`);
|
|
94
|
+
console.log("═".repeat(70));
|
|
95
|
+
}
|
|
96
|
+
//# sourceMappingURL=finding-risk-score.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-risk-score.js","sourceRoot":"","sources":["../../src/commands/finding-risk-score.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-trend-forecast.d.ts","sourceRoot":"","sources":["../../src/commands/finding-trend-forecast.ts"],"names":[],"mappings":"AAAA;;GAEG;AAyBH,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA2G5D"}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Finding-trend-forecast — Forecast finding trends from historical data.
|
|
3
|
+
*/
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
6
|
+
export function runFindingTrendForecast(argv) {
|
|
7
|
+
const historyIdx = argv.indexOf("--history");
|
|
8
|
+
const periodsIdx = argv.indexOf("--periods");
|
|
9
|
+
const formatIdx = argv.indexOf("--format");
|
|
10
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
11
|
+
const forecastPeriods = periodsIdx >= 0 ? parseInt(argv[periodsIdx + 1], 10) : 1;
|
|
12
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
13
|
+
console.log(`
|
|
14
|
+
judges finding-trend-forecast — Forecast finding trends
|
|
15
|
+
|
|
16
|
+
Usage:
|
|
17
|
+
judges finding-trend-forecast --history <path> [--periods <n>]
|
|
18
|
+
[--format table|json]
|
|
19
|
+
|
|
20
|
+
Options:
|
|
21
|
+
--history <path> History file with periodic data
|
|
22
|
+
--periods <n> Number of periods to forecast (default: 1)
|
|
23
|
+
--format <fmt> Output format: table (default), json
|
|
24
|
+
--help, -h Show this help
|
|
25
|
+
|
|
26
|
+
History file format (JSON array):
|
|
27
|
+
[{"period":"2026-W10","totalFindings":50,"criticalCount":2,"highCount":8,"passRate":85}, ...]
|
|
28
|
+
`);
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
if (historyIdx < 0) {
|
|
32
|
+
console.error("Missing --history <path>");
|
|
33
|
+
process.exitCode = 1;
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
const historyPath = argv[historyIdx + 1];
|
|
37
|
+
if (!existsSync(historyPath)) {
|
|
38
|
+
console.error(`History file not found: ${historyPath}`);
|
|
39
|
+
process.exitCode = 1;
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
const entries = JSON.parse(readFileSync(historyPath, "utf-8"));
|
|
43
|
+
if (entries.length < 2) {
|
|
44
|
+
console.log("Need at least 2 historical data points for forecasting.");
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
// Simple linear regression on totalFindings
|
|
48
|
+
const forecasts = [];
|
|
49
|
+
const n = entries.length;
|
|
50
|
+
for (let p = 0; p < forecastPeriods; p++) {
|
|
51
|
+
const xValues = entries.map((_, i) => i);
|
|
52
|
+
const yTotal = entries.map((e) => e.totalFindings);
|
|
53
|
+
const yCritical = entries.map((e) => e.criticalCount);
|
|
54
|
+
const yHigh = entries.map((e) => e.highCount);
|
|
55
|
+
const slopeTotal = linearSlope(xValues, yTotal);
|
|
56
|
+
const slopeCritical = linearSlope(xValues, yCritical);
|
|
57
|
+
const slopeHigh = linearSlope(xValues, yHigh);
|
|
58
|
+
const predictedFindings = Math.max(0, Math.round(yTotal[n - 1] + slopeTotal * (p + 1)));
|
|
59
|
+
const predictedCritical = Math.max(0, Math.round(yCritical[n - 1] + slopeCritical * (p + 1)));
|
|
60
|
+
const predictedHigh = Math.max(0, Math.round(yHigh[n - 1] + slopeHigh * (p + 1)));
|
|
61
|
+
let trend;
|
|
62
|
+
if (slopeTotal < -1)
|
|
63
|
+
trend = "improving";
|
|
64
|
+
else if (slopeTotal > 1)
|
|
65
|
+
trend = "declining";
|
|
66
|
+
else
|
|
67
|
+
trend = "stable";
|
|
68
|
+
const confidence = Math.max(0.3, Math.min(0.95, 1 - p * 0.15));
|
|
69
|
+
forecasts.push({
|
|
70
|
+
nextPeriod: `${entries[n - 1].period}+${p + 1}`,
|
|
71
|
+
predictedFindings,
|
|
72
|
+
predictedCritical,
|
|
73
|
+
predictedHigh,
|
|
74
|
+
trend,
|
|
75
|
+
confidence: Math.round(confidence * 100) / 100,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
if (format === "json") {
|
|
79
|
+
console.log(JSON.stringify(forecasts, null, 2));
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
console.log(`\nFinding Trend Forecast`);
|
|
83
|
+
console.log("═".repeat(70));
|
|
84
|
+
// Historical
|
|
85
|
+
console.log(" Historical:");
|
|
86
|
+
for (const e of entries) {
|
|
87
|
+
console.log(` ${e.period.padEnd(15)} ${String(e.totalFindings).padEnd(8)} findings (${e.criticalCount} critical, ${e.highCount} high)`);
|
|
88
|
+
}
|
|
89
|
+
console.log("\n Forecast:");
|
|
90
|
+
for (const f of forecasts) {
|
|
91
|
+
console.log(` ${f.nextPeriod.padEnd(15)} ${String(f.predictedFindings).padEnd(8)} findings (${f.predictedCritical} critical, ${f.predictedHigh} high)`);
|
|
92
|
+
console.log(`${"".padEnd(20)} Trend: ${f.trend}, Confidence: ${f.confidence}`);
|
|
93
|
+
}
|
|
94
|
+
console.log("═".repeat(70));
|
|
95
|
+
}
|
|
96
|
+
function linearSlope(x, y) {
|
|
97
|
+
const n = x.length;
|
|
98
|
+
const sumX = x.reduce((a, b) => a + b, 0);
|
|
99
|
+
const sumY = y.reduce((a, b) => a + b, 0);
|
|
100
|
+
const sumXY = x.reduce((a, xi, i) => a + xi * y[i], 0);
|
|
101
|
+
const sumX2 = x.reduce((a, xi) => a + xi * xi, 0);
|
|
102
|
+
const denom = n * sumX2 - sumX * sumX;
|
|
103
|
+
if (denom === 0)
|
|
104
|
+
return 0;
|
|
105
|
+
return (n * sumXY - sumX * sumY) / denom;
|
|
106
|
+
}
|
|
107
|
+
//# sourceMappingURL=finding-trend-forecast.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"finding-trend-forecast.js","sourceRoot":"","sources":["../../src/commands/finding-trend-forecast.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-cicd-integrate.d.ts","sourceRoot":"","sources":["../../src/commands/review-cicd-integrate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAiEH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAiE3D"}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Review-cicd-integrate — Generate CI/CD integration configs for Judges.
|
|
3
|
+
*/
|
|
4
|
+
import { writeFileSync } from "fs";
|
|
5
|
+
// ─── Templates ──────────────────────────────────────────────────────────────
|
|
6
|
+
const TEMPLATES = {
|
|
7
|
+
"github-actions": {
|
|
8
|
+
filename: ".github/workflows/judges.yml",
|
|
9
|
+
content: `name: Judges Code Review
|
|
10
|
+
on: [pull_request]
|
|
11
|
+
jobs:
|
|
12
|
+
judges-review:
|
|
13
|
+
runs-on: ubuntu-latest
|
|
14
|
+
steps:
|
|
15
|
+
- uses: actions/checkout@v4
|
|
16
|
+
- uses: actions/setup-node@v4
|
|
17
|
+
with:
|
|
18
|
+
node-version: '20'
|
|
19
|
+
- run: npx @kevinrabun/judges eval --file \${{ github.event.pull_request.head.sha }} --format sarif --fail-on-findings
|
|
20
|
+
`,
|
|
21
|
+
},
|
|
22
|
+
"gitlab-ci": {
|
|
23
|
+
filename: ".gitlab-ci.yml",
|
|
24
|
+
content: `judges-review:
|
|
25
|
+
stage: test
|
|
26
|
+
image: node:20
|
|
27
|
+
script:
|
|
28
|
+
- npx @kevinrabun/judges eval --format sarif --fail-on-findings
|
|
29
|
+
only:
|
|
30
|
+
- merge_requests
|
|
31
|
+
`,
|
|
32
|
+
},
|
|
33
|
+
"azure-pipelines": {
|
|
34
|
+
filename: "azure-pipelines-judges.yml",
|
|
35
|
+
content: `trigger:
|
|
36
|
+
- main
|
|
37
|
+
pool:
|
|
38
|
+
vmImage: 'ubuntu-latest'
|
|
39
|
+
steps:
|
|
40
|
+
- task: NodeTool@0
|
|
41
|
+
inputs:
|
|
42
|
+
versionSpec: '20.x'
|
|
43
|
+
- script: npx @kevinrabun/judges eval --format sarif --fail-on-findings
|
|
44
|
+
displayName: 'Run Judges Review'
|
|
45
|
+
`,
|
|
46
|
+
},
|
|
47
|
+
jenkins: {
|
|
48
|
+
filename: "Jenkinsfile-judges",
|
|
49
|
+
content: `pipeline {
|
|
50
|
+
agent any
|
|
51
|
+
stages {
|
|
52
|
+
stage('Judges Review') {
|
|
53
|
+
steps {
|
|
54
|
+
sh 'npx @kevinrabun/judges eval --format sarif --fail-on-findings'
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
`,
|
|
60
|
+
},
|
|
61
|
+
};
|
|
62
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
63
|
+
export function runReviewCicdIntegrate(argv) {
|
|
64
|
+
const platformIdx = argv.indexOf("--platform");
|
|
65
|
+
const outIdx = argv.indexOf("--out");
|
|
66
|
+
const formatIdx = argv.indexOf("--format");
|
|
67
|
+
const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
|
|
68
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
69
|
+
console.log(`
|
|
70
|
+
judges review-cicd-integrate — Generate CI/CD integration configs
|
|
71
|
+
|
|
72
|
+
Usage:
|
|
73
|
+
judges review-cicd-integrate --platform <name> [--out <path>]
|
|
74
|
+
[--format table|json]
|
|
75
|
+
|
|
76
|
+
Options:
|
|
77
|
+
--platform <name> CI/CD platform: github-actions, gitlab-ci, azure-pipelines, jenkins
|
|
78
|
+
--out <path> Write config to file (default: print to stdout)
|
|
79
|
+
--format <fmt> Output format: table (default), json
|
|
80
|
+
--help, -h Show this help
|
|
81
|
+
`);
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
// List platforms
|
|
85
|
+
if (platformIdx < 0) {
|
|
86
|
+
const platforms = Object.keys(TEMPLATES);
|
|
87
|
+
if (format === "json") {
|
|
88
|
+
console.log(JSON.stringify({ platforms }, null, 2));
|
|
89
|
+
}
|
|
90
|
+
else {
|
|
91
|
+
console.log(`\nAvailable CI/CD Platforms:`);
|
|
92
|
+
console.log("═".repeat(40));
|
|
93
|
+
for (const p of platforms) {
|
|
94
|
+
const t = TEMPLATES[p];
|
|
95
|
+
console.log(` ${p.padEnd(20)} → ${t.filename}`);
|
|
96
|
+
}
|
|
97
|
+
console.log("\nUse --platform <name> to generate config.");
|
|
98
|
+
console.log("═".repeat(40));
|
|
99
|
+
}
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
const platform = argv[platformIdx + 1];
|
|
103
|
+
const template = TEMPLATES[platform];
|
|
104
|
+
if (template === undefined) {
|
|
105
|
+
console.error(`Unknown platform: ${platform}`);
|
|
106
|
+
console.error(`Available: ${Object.keys(TEMPLATES).join(", ")}`);
|
|
107
|
+
process.exitCode = 1;
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
if (outIdx >= 0) {
|
|
111
|
+
const outPath = argv[outIdx + 1];
|
|
112
|
+
writeFileSync(outPath, template.content);
|
|
113
|
+
console.log(`Config written to: ${outPath}`);
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
if (format === "json") {
|
|
117
|
+
console.log(JSON.stringify(template, null, 2));
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
console.log(`\n--- ${template.filename} ---`);
|
|
121
|
+
console.log(template.content);
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=review-cicd-integrate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-cicd-integrate.js","sourceRoot":"","sources":["../../src/commands/review-cicd-integrate.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"review-compliance-map.d.ts","sourceRoot":"","sources":["../../src/commands/review-compliance-map.ts"],"names":[],"mappings":"AAAA;;GAEG;AAwCH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4F3D"}
|