@kevinrabun/judges 3.48.0 → 3.50.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 +24 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +112 -0
- package/dist/cli.js.map +1 -1
- package/dist/commands/api-audit.d.ts +9 -0
- package/dist/commands/api-audit.d.ts.map +1 -0
- package/dist/commands/api-audit.js +360 -0
- package/dist/commands/api-audit.js.map +1 -0
- package/dist/commands/compliance-map.d.ts +9 -0
- package/dist/commands/compliance-map.d.ts.map +1 -0
- package/dist/commands/compliance-map.js +375 -0
- package/dist/commands/compliance-map.js.map +1 -0
- package/dist/commands/exec-report.d.ts +9 -0
- package/dist/commands/exec-report.d.ts.map +1 -0
- package/dist/commands/exec-report.js +272 -0
- package/dist/commands/exec-report.js.map +1 -0
- package/dist/commands/guided-tour.d.ts +9 -0
- package/dist/commands/guided-tour.d.ts.map +1 -0
- package/dist/commands/guided-tour.js +288 -0
- package/dist/commands/guided-tour.js.map +1 -0
- package/dist/commands/iac-lint.d.ts +8 -0
- package/dist/commands/iac-lint.d.ts.map +1 -0
- package/dist/commands/iac-lint.js +313 -0
- package/dist/commands/iac-lint.js.map +1 -0
- package/dist/commands/incident-response.d.ts +8 -0
- package/dist/commands/incident-response.d.ts.map +1 -0
- package/dist/commands/incident-response.js +255 -0
- package/dist/commands/incident-response.js.map +1 -0
- package/dist/commands/learning-path.d.ts +9 -0
- package/dist/commands/learning-path.d.ts.map +1 -0
- package/dist/commands/learning-path.js +326 -0
- package/dist/commands/learning-path.js.map +1 -0
- package/dist/commands/license-scan.d.ts +9 -0
- package/dist/commands/license-scan.d.ts.map +1 -0
- package/dist/commands/license-scan.js +180 -0
- package/dist/commands/license-scan.js.map +1 -0
- package/dist/commands/org-policy.d.ts +8 -0
- package/dist/commands/org-policy.d.ts.map +1 -0
- package/dist/commands/org-policy.js +208 -0
- package/dist/commands/org-policy.js.map +1 -0
- package/dist/commands/perf-compare.d.ts +9 -0
- package/dist/commands/perf-compare.d.ts.map +1 -0
- package/dist/commands/perf-compare.js +246 -0
- package/dist/commands/perf-compare.js.map +1 -0
- package/dist/commands/pii-scan.d.ts +8 -0
- package/dist/commands/pii-scan.d.ts.map +1 -0
- package/dist/commands/pii-scan.js +300 -0
- package/dist/commands/pii-scan.js.map +1 -0
- package/dist/commands/predict.d.ts +8 -0
- package/dist/commands/predict.d.ts.map +1 -0
- package/dist/commands/predict.js +219 -0
- package/dist/commands/predict.js.map +1 -0
- package/dist/commands/risk-heatmap.d.ts +8 -0
- package/dist/commands/risk-heatmap.d.ts.map +1 -0
- package/dist/commands/risk-heatmap.js +224 -0
- package/dist/commands/risk-heatmap.js.map +1 -0
- package/dist/commands/sbom-export.d.ts +8 -0
- package/dist/commands/sbom-export.d.ts.map +1 -0
- package/dist/commands/sbom-export.js +162 -0
- package/dist/commands/sbom-export.js.map +1 -0
- package/dist/commands/secret-scan.d.ts +8 -0
- package/dist/commands/secret-scan.d.ts.map +1 -0
- package/dist/commands/secret-scan.js +245 -0
- package/dist/commands/secret-scan.js.map +1 -0
- package/dist/commands/test-correlate.d.ts +8 -0
- package/dist/commands/test-correlate.d.ts.map +1 -0
- package/dist/commands/test-correlate.js +222 -0
- package/dist/commands/test-correlate.js.map +1 -0
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -0,0 +1,300 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PII scan — detect personally-identifiable information patterns
|
|
3
|
+
* in source code: string literals, logs, config files.
|
|
4
|
+
*
|
|
5
|
+
* All analysis local.
|
|
6
|
+
*/
|
|
7
|
+
import { existsSync, readFileSync, readdirSync, mkdirSync, writeFileSync } from "fs";
|
|
8
|
+
import { join, extname } from "path";
|
|
9
|
+
// ─── Patterns ───────────────────────────────────────────────────────────────
|
|
10
|
+
const PII_PATTERNS = [
|
|
11
|
+
{
|
|
12
|
+
id: "ssn",
|
|
13
|
+
label: "Social Security Number",
|
|
14
|
+
regex: /\b\d{3}-\d{2}-\d{4}\b/g,
|
|
15
|
+
confidence: "high",
|
|
16
|
+
validate: (m) => {
|
|
17
|
+
const parts = m.split("-");
|
|
18
|
+
return parseInt(parts[0]) > 0 && parseInt(parts[0]) < 900;
|
|
19
|
+
},
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
id: "credit-card",
|
|
23
|
+
label: "Credit Card Number",
|
|
24
|
+
regex: /\b(?:4\d{3}|5[1-5]\d{2}|3[47]\d{2}|6(?:011|5\d{2}))[- ]?\d{4}[- ]?\d{4}[- ]?\d{1,4}\b/g,
|
|
25
|
+
confidence: "high",
|
|
26
|
+
validate: (m) => {
|
|
27
|
+
const digits = m.replace(/[- ]/g, "");
|
|
28
|
+
if (digits.length < 13 || digits.length > 19)
|
|
29
|
+
return false;
|
|
30
|
+
// Luhn check
|
|
31
|
+
let sum = 0;
|
|
32
|
+
let alt = false;
|
|
33
|
+
for (let i = digits.length - 1; i >= 0; i--) {
|
|
34
|
+
let n = parseInt(digits[i]);
|
|
35
|
+
if (alt) {
|
|
36
|
+
n *= 2;
|
|
37
|
+
if (n > 9)
|
|
38
|
+
n -= 9;
|
|
39
|
+
}
|
|
40
|
+
sum += n;
|
|
41
|
+
alt = !alt;
|
|
42
|
+
}
|
|
43
|
+
return sum % 10 === 0;
|
|
44
|
+
},
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
id: "email",
|
|
48
|
+
label: "Email Address",
|
|
49
|
+
regex: /\b[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}\b/g,
|
|
50
|
+
confidence: "medium",
|
|
51
|
+
validate: (m) => {
|
|
52
|
+
// Exclude common non-PII patterns
|
|
53
|
+
if (m.endsWith("@example.com") || m.endsWith("@test.com"))
|
|
54
|
+
return false;
|
|
55
|
+
if (m.startsWith("noreply@") || m.startsWith("info@"))
|
|
56
|
+
return false;
|
|
57
|
+
return true;
|
|
58
|
+
},
|
|
59
|
+
},
|
|
60
|
+
{
|
|
61
|
+
id: "phone-us",
|
|
62
|
+
label: "US Phone Number",
|
|
63
|
+
regex: /\b(?:\+?1[-.\s]?)?\(?\d{3}\)?[-.\s]?\d{3}[-.\s]?\d{4}\b/g,
|
|
64
|
+
confidence: "medium",
|
|
65
|
+
validate: (m) => {
|
|
66
|
+
const digits = m.replace(/\D/g, "");
|
|
67
|
+
return digits.length >= 10 && digits.length <= 11;
|
|
68
|
+
},
|
|
69
|
+
},
|
|
70
|
+
{
|
|
71
|
+
id: "ip-address",
|
|
72
|
+
label: "IP Address",
|
|
73
|
+
regex: /\b(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\b/g,
|
|
74
|
+
confidence: "low",
|
|
75
|
+
validate: (m) => {
|
|
76
|
+
// Skip loopback and private ranges that are typically non-PII
|
|
77
|
+
if (m.startsWith("127.") || m.startsWith("0.") || m === "255.255.255.255")
|
|
78
|
+
return false;
|
|
79
|
+
return true;
|
|
80
|
+
},
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
id: "date-of-birth",
|
|
84
|
+
label: "Date of Birth Pattern",
|
|
85
|
+
regex: /\b(?:dob|date_?of_?birth|birth_?date|birthday)\s*[:=]\s*["']?\d{1,4}[-/]\d{1,2}[-/]\d{1,4}["']?/gi,
|
|
86
|
+
confidence: "high",
|
|
87
|
+
},
|
|
88
|
+
{
|
|
89
|
+
id: "passport",
|
|
90
|
+
label: "Passport Number Pattern",
|
|
91
|
+
regex: /\b(?:passport(?:_?(?:no|num|number))?)\s*[:=]\s*["']?[A-Z0-9]{6,9}["']?/gi,
|
|
92
|
+
confidence: "high",
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
id: "drivers-license",
|
|
96
|
+
label: "Drivers License Pattern",
|
|
97
|
+
regex: /\b(?:drivers?_?(?:license|licence)(?:_?(?:no|num|number))?)\s*[:=]\s*["']?[A-Z0-9-]{5,15}["']?/gi,
|
|
98
|
+
confidence: "high",
|
|
99
|
+
},
|
|
100
|
+
{
|
|
101
|
+
id: "logging-pii",
|
|
102
|
+
label: "PII in Logging Statement",
|
|
103
|
+
regex: /(?:console\.(?:log|warn|error|info)|logger?\.\w+|print|println)\s*\(.*(?:email|phone|ssn|social|password|name|address|dob|birth)/gi,
|
|
104
|
+
confidence: "medium",
|
|
105
|
+
},
|
|
106
|
+
];
|
|
107
|
+
// ─── Scanner ────────────────────────────────────────────────────────────────
|
|
108
|
+
const SKIP_DIRS = new Set(["node_modules", ".git", "dist", "build", "coverage", "__pycache__"]);
|
|
109
|
+
const CODE_EXTENSIONS = new Set([
|
|
110
|
+
".ts",
|
|
111
|
+
".js",
|
|
112
|
+
".py",
|
|
113
|
+
".java",
|
|
114
|
+
".cs",
|
|
115
|
+
".go",
|
|
116
|
+
".rb",
|
|
117
|
+
".php",
|
|
118
|
+
".rs",
|
|
119
|
+
".swift",
|
|
120
|
+
".kt",
|
|
121
|
+
".scala",
|
|
122
|
+
".json",
|
|
123
|
+
".yaml",
|
|
124
|
+
".yml",
|
|
125
|
+
".xml",
|
|
126
|
+
".env",
|
|
127
|
+
".cfg",
|
|
128
|
+
".ini",
|
|
129
|
+
".conf",
|
|
130
|
+
]);
|
|
131
|
+
function collectSourceFiles(dir) {
|
|
132
|
+
const result = [];
|
|
133
|
+
function walk(d) {
|
|
134
|
+
let entries;
|
|
135
|
+
try {
|
|
136
|
+
entries = readdirSync(d);
|
|
137
|
+
}
|
|
138
|
+
catch {
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
for (const name of entries) {
|
|
142
|
+
if (SKIP_DIRS.has(name) || name.startsWith("."))
|
|
143
|
+
continue;
|
|
144
|
+
const full = join(d, name);
|
|
145
|
+
try {
|
|
146
|
+
const sub = readdirSync(full);
|
|
147
|
+
void sub;
|
|
148
|
+
walk(full);
|
|
149
|
+
}
|
|
150
|
+
catch {
|
|
151
|
+
if (CODE_EXTENSIONS.has(extname(name).toLowerCase()))
|
|
152
|
+
result.push(full);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
walk(dir);
|
|
157
|
+
return result;
|
|
158
|
+
}
|
|
159
|
+
function maskPii(text) {
|
|
160
|
+
if (text.length <= 4)
|
|
161
|
+
return "****";
|
|
162
|
+
return text.slice(0, 2) + "*".repeat(text.length - 4) + text.slice(-2);
|
|
163
|
+
}
|
|
164
|
+
function scanFile(filePath) {
|
|
165
|
+
let content;
|
|
166
|
+
try {
|
|
167
|
+
content = readFileSync(filePath, "utf-8");
|
|
168
|
+
}
|
|
169
|
+
catch {
|
|
170
|
+
return [];
|
|
171
|
+
}
|
|
172
|
+
const lines = content.split("\n");
|
|
173
|
+
const matches = [];
|
|
174
|
+
for (let lineIdx = 0; lineIdx < lines.length; lineIdx++) {
|
|
175
|
+
const line = lines[lineIdx];
|
|
176
|
+
// Skip comments that look like documentation/patterns
|
|
177
|
+
if (/^\s*(?:\/\/|#|\/?\*)\s*(?:example|test|sample|pattern|regex|format)/i.test(line))
|
|
178
|
+
continue;
|
|
179
|
+
for (const pattern of PII_PATTERNS) {
|
|
180
|
+
pattern.regex.lastIndex = 0;
|
|
181
|
+
let m;
|
|
182
|
+
while ((m = pattern.regex.exec(line)) !== null) {
|
|
183
|
+
if (pattern.validate && !pattern.validate(m[0]))
|
|
184
|
+
continue;
|
|
185
|
+
matches.push({
|
|
186
|
+
file: filePath,
|
|
187
|
+
line: lineIdx + 1,
|
|
188
|
+
column: m.index + 1,
|
|
189
|
+
type: pattern.id,
|
|
190
|
+
snippet: maskPii(m[0]),
|
|
191
|
+
confidence: pattern.confidence,
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
return matches;
|
|
197
|
+
}
|
|
198
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
199
|
+
export function runPiiScan(argv) {
|
|
200
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
201
|
+
console.log(`
|
|
202
|
+
judges pii-scan — Detect personally-identifiable information in source code
|
|
203
|
+
|
|
204
|
+
Usage:
|
|
205
|
+
judges pii-scan [dir]
|
|
206
|
+
judges pii-scan src/ --confidence high
|
|
207
|
+
judges pii-scan . --format json --output pii-report.json
|
|
208
|
+
|
|
209
|
+
Options:
|
|
210
|
+
--confidence <level> Filter by confidence (high, medium, low)
|
|
211
|
+
--type <types> Filter by PII type (comma-separated: ssn,credit-card,email,...)
|
|
212
|
+
--patterns List all PII detection patterns
|
|
213
|
+
--format json JSON output
|
|
214
|
+
--output <file> Write report to file
|
|
215
|
+
--help, -h Show this help
|
|
216
|
+
|
|
217
|
+
PII Types: ${PII_PATTERNS.map((p) => p.id).join(", ")}
|
|
218
|
+
`);
|
|
219
|
+
return;
|
|
220
|
+
}
|
|
221
|
+
const format = argv.find((_a, i) => argv[i - 1] === "--format") || "text";
|
|
222
|
+
const outputFile = argv.find((_a, i) => argv[i - 1] === "--output");
|
|
223
|
+
if (argv.includes("--patterns")) {
|
|
224
|
+
if (format === "json") {
|
|
225
|
+
console.log(JSON.stringify(PII_PATTERNS.map(({ regex: _r, validate: _v, ...rest }) => rest), null, 2));
|
|
226
|
+
}
|
|
227
|
+
else {
|
|
228
|
+
console.log(`\n PII Detection Patterns (${PII_PATTERNS.length})\n ──────────────────────────`);
|
|
229
|
+
for (const p of PII_PATTERNS) {
|
|
230
|
+
console.log(` [${p.confidence.toUpperCase().padEnd(6)}] ${p.id.padEnd(20)} — ${p.label}`);
|
|
231
|
+
}
|
|
232
|
+
console.log("");
|
|
233
|
+
}
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
236
|
+
const target = argv.find((a) => !a.startsWith("--") && !argv[argv.indexOf(a) - 1]?.startsWith("--")) || ".";
|
|
237
|
+
const confFilter = argv.find((_a, i) => argv[i - 1] === "--confidence");
|
|
238
|
+
const typeFilter = argv.find((_a, i) => argv[i - 1] === "--type");
|
|
239
|
+
if (!existsSync(target)) {
|
|
240
|
+
console.error(` Path not found: ${target}`);
|
|
241
|
+
return;
|
|
242
|
+
}
|
|
243
|
+
const files = collectSourceFiles(target);
|
|
244
|
+
let allMatches = [];
|
|
245
|
+
for (const f of files) {
|
|
246
|
+
allMatches.push(...scanFile(f));
|
|
247
|
+
}
|
|
248
|
+
if (confFilter) {
|
|
249
|
+
allMatches = allMatches.filter((m) => m.confidence === confFilter);
|
|
250
|
+
}
|
|
251
|
+
if (typeFilter) {
|
|
252
|
+
const allowed = typeFilter.split(",");
|
|
253
|
+
allMatches = allMatches.filter((m) => allowed.includes(m.type));
|
|
254
|
+
}
|
|
255
|
+
const report = {
|
|
256
|
+
matches: allMatches,
|
|
257
|
+
scannedFiles: files.length,
|
|
258
|
+
summary: {
|
|
259
|
+
total: allMatches.length,
|
|
260
|
+
byType: Object.fromEntries(PII_PATTERNS.map((p) => [p.id, allMatches.filter((m) => m.type === p.id).length]).filter(([, count]) => count > 0)),
|
|
261
|
+
byConfidence: {
|
|
262
|
+
high: allMatches.filter((m) => m.confidence === "high").length,
|
|
263
|
+
medium: allMatches.filter((m) => m.confidence === "medium").length,
|
|
264
|
+
low: allMatches.filter((m) => m.confidence === "low").length,
|
|
265
|
+
},
|
|
266
|
+
},
|
|
267
|
+
timestamp: new Date().toISOString(),
|
|
268
|
+
};
|
|
269
|
+
if (outputFile) {
|
|
270
|
+
const dir = join(".", ".judges-pii-scan");
|
|
271
|
+
if (!existsSync(dir))
|
|
272
|
+
mkdirSync(dir, { recursive: true });
|
|
273
|
+
writeFileSync(join(dir, outputFile), JSON.stringify(report, null, 2));
|
|
274
|
+
console.log(` Report saved to .judges-pii-scan/${outputFile}`);
|
|
275
|
+
}
|
|
276
|
+
if (format === "json") {
|
|
277
|
+
console.log(JSON.stringify(report, null, 2));
|
|
278
|
+
}
|
|
279
|
+
else {
|
|
280
|
+
console.log(`\n PII Scan — ${files.length} files scanned`);
|
|
281
|
+
console.log(` Found: ${allMatches.length} potential PII occurrences\n ──────────────────────────`);
|
|
282
|
+
if (allMatches.length === 0) {
|
|
283
|
+
console.log(` ✅ No PII detected\n`);
|
|
284
|
+
return;
|
|
285
|
+
}
|
|
286
|
+
for (const conf of ["high", "medium", "low"]) {
|
|
287
|
+
const items = allMatches.filter((m) => m.confidence === conf);
|
|
288
|
+
if (items.length === 0)
|
|
289
|
+
continue;
|
|
290
|
+
console.log(`\n ${conf.toUpperCase()} CONFIDENCE (${items.length})`);
|
|
291
|
+
for (const m of items) {
|
|
292
|
+
const piiDef = PII_PATTERNS.find((p) => p.id === m.type);
|
|
293
|
+
console.log(` ${m.file}:${m.line}:${m.column} — ${piiDef?.label || m.type}`);
|
|
294
|
+
console.log(` Masked: ${m.snippet}`);
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
console.log("");
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
//# sourceMappingURL=pii-scan.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pii-scan.js","sourceRoot":"","sources":["../../src/commands/pii-scan.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Predict — applies trend analysis to finding snapshots to
|
|
3
|
+
* forecast remediation timelines and regression-prone files.
|
|
4
|
+
*
|
|
5
|
+
* All data from local snapshot history.
|
|
6
|
+
*/
|
|
7
|
+
export declare function runPredict(argv: string[]): void;
|
|
8
|
+
//# sourceMappingURL=predict.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"predict.d.ts","sourceRoot":"","sources":["../../src/commands/predict.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAuJH,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAyG/C"}
|
|
@@ -0,0 +1,219 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Predict — applies trend analysis to finding snapshots to
|
|
3
|
+
* forecast remediation timelines and regression-prone files.
|
|
4
|
+
*
|
|
5
|
+
* All data from local snapshot history.
|
|
6
|
+
*/
|
|
7
|
+
import { existsSync, readFileSync, mkdirSync, writeFileSync } from "fs";
|
|
8
|
+
import { join } from "path";
|
|
9
|
+
// ─── Analysis ───────────────────────────────────────────────────────────────
|
|
10
|
+
function loadSnapshots() {
|
|
11
|
+
const paths = [
|
|
12
|
+
join(".judges-snapshots", "history.json"),
|
|
13
|
+
".judges-snapshots.json",
|
|
14
|
+
join(".judges-burndown", "snapshots.json"),
|
|
15
|
+
];
|
|
16
|
+
for (const p of paths) {
|
|
17
|
+
if (!existsSync(p))
|
|
18
|
+
continue;
|
|
19
|
+
try {
|
|
20
|
+
const data = JSON.parse(readFileSync(p, "utf-8"));
|
|
21
|
+
if (Array.isArray(data))
|
|
22
|
+
return data;
|
|
23
|
+
if (data.snapshots)
|
|
24
|
+
return data.snapshots;
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
/* skip */
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
return [];
|
|
31
|
+
}
|
|
32
|
+
function linearRegression(points) {
|
|
33
|
+
const n = points.length;
|
|
34
|
+
if (n < 2)
|
|
35
|
+
return { slope: 0, intercept: points[0]?.y || 0, r2: 0 };
|
|
36
|
+
const sumX = points.reduce((s, p) => s + p.x, 0);
|
|
37
|
+
const sumY = points.reduce((s, p) => s + p.y, 0);
|
|
38
|
+
const sumXY = points.reduce((s, p) => s + p.x * p.y, 0);
|
|
39
|
+
const sumX2 = points.reduce((s, p) => s + p.x * p.x, 0);
|
|
40
|
+
const sumY2 = points.reduce((s, p) => s + p.y * p.y, 0);
|
|
41
|
+
const denom = n * sumX2 - sumX * sumX;
|
|
42
|
+
if (denom === 0)
|
|
43
|
+
return { slope: 0, intercept: sumY / n, r2: 0 };
|
|
44
|
+
const slope = (n * sumXY - sumX * sumY) / denom;
|
|
45
|
+
const intercept = (sumY - slope * sumX) / n;
|
|
46
|
+
// R²
|
|
47
|
+
const yMean = sumY / n;
|
|
48
|
+
const ssTot = sumY2 - n * yMean * yMean;
|
|
49
|
+
const ssRes = points.reduce((s, p) => s + Math.pow(p.y - (slope * p.x + intercept), 2), 0);
|
|
50
|
+
const r2 = ssTot === 0 ? 0 : Math.max(0, 1 - ssRes / ssTot);
|
|
51
|
+
return { slope, intercept, r2 };
|
|
52
|
+
}
|
|
53
|
+
function predictMetric(snapshots, field) {
|
|
54
|
+
if (snapshots.length < 2) {
|
|
55
|
+
return {
|
|
56
|
+
metric: field,
|
|
57
|
+
currentValue: snapshots[0]?.[field] || 0,
|
|
58
|
+
trend: "stable",
|
|
59
|
+
ratePerDay: 0,
|
|
60
|
+
estimatedZeroDate: null,
|
|
61
|
+
confidence: 0,
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
const t0 = new Date(snapshots[0].timestamp).getTime();
|
|
65
|
+
const points = snapshots.map((s) => ({
|
|
66
|
+
x: (new Date(s.timestamp).getTime() - t0) / (1000 * 60 * 60 * 24), // days
|
|
67
|
+
y: s[field],
|
|
68
|
+
}));
|
|
69
|
+
const { slope, r2 } = linearRegression(points);
|
|
70
|
+
const current = points[points.length - 1].y;
|
|
71
|
+
let trend = "stable";
|
|
72
|
+
if (slope < -0.1)
|
|
73
|
+
trend = "decreasing";
|
|
74
|
+
else if (slope > 0.1)
|
|
75
|
+
trend = "increasing";
|
|
76
|
+
let estimatedZeroDate = null;
|
|
77
|
+
if (slope < 0 && current > 0) {
|
|
78
|
+
const daysToZero = -current / slope;
|
|
79
|
+
const zeroDate = new Date(Date.now() + daysToZero * 24 * 60 * 60 * 1000);
|
|
80
|
+
estimatedZeroDate = zeroDate.toISOString().split("T")[0];
|
|
81
|
+
}
|
|
82
|
+
return {
|
|
83
|
+
metric: field,
|
|
84
|
+
currentValue: current,
|
|
85
|
+
trend,
|
|
86
|
+
ratePerDay: Math.round(slope * 100) / 100,
|
|
87
|
+
estimatedZeroDate,
|
|
88
|
+
confidence: Math.round(r2 * 100),
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
function loadRegressionData() {
|
|
92
|
+
const paths = [".judges-regressions.json", join(".judges-regression-alert", "history.json")];
|
|
93
|
+
for (const p of paths) {
|
|
94
|
+
if (!existsSync(p))
|
|
95
|
+
continue;
|
|
96
|
+
try {
|
|
97
|
+
const data = JSON.parse(readFileSync(p, "utf-8"));
|
|
98
|
+
const files = Array.isArray(data) ? data : data.regressions || [];
|
|
99
|
+
const counts = new Map();
|
|
100
|
+
for (const r of files) {
|
|
101
|
+
const file = r.file || r.path || "unknown";
|
|
102
|
+
counts.set(file, (counts.get(file) || 0) + 1);
|
|
103
|
+
}
|
|
104
|
+
return [...counts.entries()]
|
|
105
|
+
.map(([file, count]) => ({
|
|
106
|
+
file,
|
|
107
|
+
regressionCount: count,
|
|
108
|
+
risk: count > 3 ? "high" : count > 1 ? "medium" : "low",
|
|
109
|
+
}))
|
|
110
|
+
.sort((a, b) => b.regressionCount - a.regressionCount);
|
|
111
|
+
}
|
|
112
|
+
catch {
|
|
113
|
+
/* skip */
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return [];
|
|
117
|
+
}
|
|
118
|
+
// ─── CLI ────────────────────────────────────────────────────────────────────
|
|
119
|
+
const STORE = ".judges-predictions";
|
|
120
|
+
export function runPredict(argv) {
|
|
121
|
+
if (argv.includes("--help") || argv.includes("-h")) {
|
|
122
|
+
console.log(`
|
|
123
|
+
judges predict — Forecast remediation timelines and regression risk
|
|
124
|
+
|
|
125
|
+
Usage:
|
|
126
|
+
judges predict
|
|
127
|
+
judges predict --metric critical
|
|
128
|
+
judges predict --regressions
|
|
129
|
+
judges predict --save
|
|
130
|
+
|
|
131
|
+
Options:
|
|
132
|
+
--metric <name> Predict specific metric (findings, critical, high, medium, low)
|
|
133
|
+
--regressions Show regression-prone files prediction
|
|
134
|
+
--save Save predictions to ${STORE}/
|
|
135
|
+
--format json JSON output
|
|
136
|
+
--help, -h Show this help
|
|
137
|
+
`);
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
const format = argv.find((_a, i) => argv[i - 1] === "--format") || "text";
|
|
141
|
+
const snapshots = loadSnapshots();
|
|
142
|
+
if (snapshots.length < 2 && !argv.includes("--regressions")) {
|
|
143
|
+
console.log(" Need at least 2 snapshots for predictions.");
|
|
144
|
+
console.log(" Run scans over time and they'll be recorded automatically.");
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
// Single metric
|
|
148
|
+
const metricName = argv.find((_a, i) => argv[i - 1] === "--metric");
|
|
149
|
+
if (metricName) {
|
|
150
|
+
const pred = predictMetric(snapshots, metricName);
|
|
151
|
+
if (format === "json") {
|
|
152
|
+
console.log(JSON.stringify(pred, null, 2));
|
|
153
|
+
}
|
|
154
|
+
else {
|
|
155
|
+
console.log(`\n Prediction: ${pred.metric}\n ──────────────────────────`);
|
|
156
|
+
console.log(` Current: ${pred.currentValue}`);
|
|
157
|
+
console.log(` Trend: ${pred.trend} (${pred.ratePerDay >= 0 ? "+" : ""}${pred.ratePerDay}/day)`);
|
|
158
|
+
console.log(` Confidence: ${pred.confidence}%`);
|
|
159
|
+
if (pred.estimatedZeroDate)
|
|
160
|
+
console.log(` Estimated zero: ${pred.estimatedZeroDate}`);
|
|
161
|
+
console.log("");
|
|
162
|
+
}
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
// Regressions
|
|
166
|
+
if (argv.includes("--regressions")) {
|
|
167
|
+
const regressions = loadRegressionData();
|
|
168
|
+
if (format === "json") {
|
|
169
|
+
console.log(JSON.stringify(regressions, null, 2));
|
|
170
|
+
}
|
|
171
|
+
else {
|
|
172
|
+
console.log(`\n Regression-Prone Files\n ──────────────────────────`);
|
|
173
|
+
if (regressions.length === 0) {
|
|
174
|
+
console.log(` No regression data found.\n`);
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
for (const r of regressions.slice(0, 15)) {
|
|
178
|
+
console.log(` [${r.risk.toUpperCase().padEnd(6)}] ${r.file} (${r.regressionCount} regressions)`);
|
|
179
|
+
}
|
|
180
|
+
console.log("");
|
|
181
|
+
}
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
// Full prediction
|
|
185
|
+
const metrics = ["findings", "critical", "high", "medium", "low"];
|
|
186
|
+
const predictions = metrics.map((m) => predictMetric(snapshots, m));
|
|
187
|
+
const regressions = loadRegressionData();
|
|
188
|
+
const report = {
|
|
189
|
+
predictions,
|
|
190
|
+
regressionRisk: regressions.slice(0, 10),
|
|
191
|
+
timestamp: new Date().toISOString(),
|
|
192
|
+
};
|
|
193
|
+
if (argv.includes("--save")) {
|
|
194
|
+
if (!existsSync(STORE))
|
|
195
|
+
mkdirSync(STORE, { recursive: true });
|
|
196
|
+
writeFileSync(join(STORE, "prediction-report.json"), JSON.stringify(report, null, 2));
|
|
197
|
+
console.log(` Saved to ${STORE}/prediction-report.json`);
|
|
198
|
+
}
|
|
199
|
+
if (format === "json") {
|
|
200
|
+
console.log(JSON.stringify(report, null, 2));
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
console.log(`\n Prediction Report (${snapshots.length} snapshots)`);
|
|
204
|
+
console.log(` ──────────────────────────`);
|
|
205
|
+
for (const p of predictions) {
|
|
206
|
+
const arrow = p.trend === "decreasing" ? "↓" : p.trend === "increasing" ? "↑" : "→";
|
|
207
|
+
const zero = p.estimatedZeroDate ? ` → zero by ${p.estimatedZeroDate}` : "";
|
|
208
|
+
console.log(` ${p.metric.padEnd(12)} ${String(p.currentValue).padEnd(6)} ${arrow} ${p.ratePerDay >= 0 ? "+" : ""}${p.ratePerDay}/day (${p.confidence}% conf)${zero}`);
|
|
209
|
+
}
|
|
210
|
+
if (regressions.length > 0) {
|
|
211
|
+
console.log(`\n Regression-Prone Files:`);
|
|
212
|
+
for (const r of regressions.slice(0, 5)) {
|
|
213
|
+
console.log(` [${r.risk.toUpperCase().padEnd(6)}] ${r.file}`);
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
console.log("");
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
//# sourceMappingURL=predict.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"predict.js","sourceRoot":"","sources":["../../src/commands/predict.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Risk heatmap — generates a file/directory risk heatmap
|
|
3
|
+
* combining finding density, severity, and test coverage.
|
|
4
|
+
*
|
|
5
|
+
* All data from local files.
|
|
6
|
+
*/
|
|
7
|
+
export declare function runRiskHeatmap(argv: string[]): void;
|
|
8
|
+
//# sourceMappingURL=risk-heatmap.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"risk-heatmap.d.ts","sourceRoot":"","sources":["../../src/commands/risk-heatmap.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAwJH,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA+FnD"}
|