@kevinrabun/judges 3.88.0 → 3.90.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (78) hide show
  1. package/CHANGELOG.md +26 -0
  2. package/dist/cli.d.ts.map +1 -1
  3. package/dist/cli.js +126 -0
  4. package/dist/cli.js.map +1 -1
  5. package/dist/commands/finding-auto-tag.d.ts +5 -0
  6. package/dist/commands/finding-auto-tag.d.ts.map +1 -0
  7. package/dist/commands/finding-auto-tag.js +114 -0
  8. package/dist/commands/finding-auto-tag.js.map +1 -0
  9. package/dist/commands/finding-category-map.d.ts +5 -0
  10. package/dist/commands/finding-category-map.d.ts.map +1 -0
  11. package/dist/commands/finding-category-map.js +104 -0
  12. package/dist/commands/finding-category-map.js.map +1 -0
  13. package/dist/commands/finding-cluster-group.d.ts +5 -0
  14. package/dist/commands/finding-cluster-group.d.ts.map +1 -0
  15. package/dist/commands/finding-cluster-group.js +106 -0
  16. package/dist/commands/finding-cluster-group.js.map +1 -0
  17. package/dist/commands/finding-dedup-report.d.ts +5 -0
  18. package/dist/commands/finding-dedup-report.d.ts.map +1 -0
  19. package/dist/commands/finding-dedup-report.js +102 -0
  20. package/dist/commands/finding-dedup-report.js.map +1 -0
  21. package/dist/commands/finding-evidence-collect.d.ts +5 -0
  22. package/dist/commands/finding-evidence-collect.d.ts.map +1 -0
  23. package/dist/commands/finding-evidence-collect.js +115 -0
  24. package/dist/commands/finding-evidence-collect.js.map +1 -0
  25. package/dist/commands/finding-false-positive-log.d.ts +5 -0
  26. package/dist/commands/finding-false-positive-log.d.ts.map +1 -0
  27. package/dist/commands/finding-false-positive-log.js +151 -0
  28. package/dist/commands/finding-false-positive-log.js.map +1 -0
  29. package/dist/commands/finding-fix-validation.d.ts +5 -0
  30. package/dist/commands/finding-fix-validation.d.ts.map +1 -0
  31. package/dist/commands/finding-fix-validation.js +116 -0
  32. package/dist/commands/finding-fix-validation.js.map +1 -0
  33. package/dist/commands/finding-regression-check.d.ts +5 -0
  34. package/dist/commands/finding-regression-check.d.ts.map +1 -0
  35. package/dist/commands/finding-regression-check.js +113 -0
  36. package/dist/commands/finding-regression-check.js.map +1 -0
  37. package/dist/commands/finding-resolution-tracker.d.ts +5 -0
  38. package/dist/commands/finding-resolution-tracker.d.ts.map +1 -0
  39. package/dist/commands/finding-resolution-tracker.js +164 -0
  40. package/dist/commands/finding-resolution-tracker.js.map +1 -0
  41. package/dist/commands/finding-trend-analysis.d.ts +5 -0
  42. package/dist/commands/finding-trend-analysis.d.ts.map +1 -0
  43. package/dist/commands/finding-trend-analysis.js +96 -0
  44. package/dist/commands/finding-trend-analysis.js.map +1 -0
  45. package/dist/commands/review-batch-mode.d.ts +5 -0
  46. package/dist/commands/review-batch-mode.d.ts.map +1 -0
  47. package/dist/commands/review-batch-mode.js +98 -0
  48. package/dist/commands/review-batch-mode.js.map +1 -0
  49. package/dist/commands/review-compliance-gate.d.ts +5 -0
  50. package/dist/commands/review-compliance-gate.d.ts.map +1 -0
  51. package/dist/commands/review-compliance-gate.js +152 -0
  52. package/dist/commands/review-compliance-gate.js.map +1 -0
  53. package/dist/commands/review-dashboard-data.d.ts +5 -0
  54. package/dist/commands/review-dashboard-data.d.ts.map +1 -0
  55. package/dist/commands/review-dashboard-data.js +143 -0
  56. package/dist/commands/review-dashboard-data.js.map +1 -0
  57. package/dist/commands/review-guardrail.d.ts +5 -0
  58. package/dist/commands/review-guardrail.d.ts.map +1 -0
  59. package/dist/commands/review-guardrail.js +156 -0
  60. package/dist/commands/review-guardrail.js.map +1 -0
  61. package/dist/commands/review-perf-profile.d.ts +5 -0
  62. package/dist/commands/review-perf-profile.d.ts.map +1 -0
  63. package/dist/commands/review-perf-profile.js +99 -0
  64. package/dist/commands/review-perf-profile.js.map +1 -0
  65. package/dist/commands/review-scope-limit.d.ts +5 -0
  66. package/dist/commands/review-scope-limit.d.ts.map +1 -0
  67. package/dist/commands/review-scope-limit.js +101 -0
  68. package/dist/commands/review-scope-limit.js.map +1 -0
  69. package/dist/commands/review-threshold-tune.d.ts +5 -0
  70. package/dist/commands/review-threshold-tune.d.ts.map +1 -0
  71. package/dist/commands/review-threshold-tune.js +136 -0
  72. package/dist/commands/review-threshold-tune.js.map +1 -0
  73. package/dist/commands/review-webhook-notify.d.ts +5 -0
  74. package/dist/commands/review-webhook-notify.d.ts.map +1 -0
  75. package/dist/commands/review-webhook-notify.js +146 -0
  76. package/dist/commands/review-webhook-notify.js.map +1 -0
  77. package/package.json +1 -1
  78. package/server.json +2 -2
@@ -0,0 +1,156 @@
1
+ /**
2
+ * Review-guardrail — Define and enforce review guardrails.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
5
+ import { dirname } from "path";
6
+ // ─── Helpers ────────────────────────────────────────────────────────────────
7
+ function loadGuardrails(configPath) {
8
+ if (!existsSync(configPath)) {
9
+ return {
10
+ version: 1,
11
+ guardrails: [
12
+ { id: "no-critical", type: "no-critical", threshold: 0, enabled: true },
13
+ { id: "min-score-60", type: "min-score", threshold: 60, enabled: true },
14
+ { id: "max-findings-50", type: "max-findings", threshold: 50, enabled: true },
15
+ ],
16
+ };
17
+ }
18
+ try {
19
+ return JSON.parse(readFileSync(configPath, "utf-8"));
20
+ }
21
+ catch {
22
+ return { version: 1, guardrails: [] };
23
+ }
24
+ }
25
+ function saveGuardrails(configPath, config) {
26
+ const dir = dirname(configPath);
27
+ if (!existsSync(dir)) {
28
+ mkdirSync(dir, { recursive: true });
29
+ }
30
+ writeFileSync(configPath, JSON.stringify(config, null, 2));
31
+ }
32
+ function evaluateGuardrails(verdict, guardrails) {
33
+ const results = [];
34
+ for (const g of guardrails) {
35
+ if (!g.enabled)
36
+ continue;
37
+ let passed = true;
38
+ let detail = "";
39
+ switch (g.type) {
40
+ case "no-critical":
41
+ passed = verdict.criticalCount === 0;
42
+ detail = passed ? "No critical findings" : `${verdict.criticalCount} critical findings found`;
43
+ break;
44
+ case "min-score":
45
+ passed = verdict.overallScore >= g.threshold;
46
+ detail = `Score: ${verdict.overallScore} (threshold: ${g.threshold})`;
47
+ break;
48
+ case "max-findings":
49
+ passed = verdict.findings.length <= g.threshold;
50
+ detail = `Findings: ${verdict.findings.length} (max: ${g.threshold})`;
51
+ break;
52
+ case "max-severity-count": {
53
+ const sev = (g.severity || "high").toLowerCase();
54
+ const count = verdict.findings.filter((f) => (f.severity || "medium").toLowerCase() === sev).length;
55
+ passed = count <= g.threshold;
56
+ detail = `${sev}: ${count} (max: ${g.threshold})`;
57
+ break;
58
+ }
59
+ }
60
+ results.push({ guardrailId: g.id, passed, detail });
61
+ }
62
+ return results;
63
+ }
64
+ // ─── CLI ────────────────────────────────────────────────────────────────────
65
+ export function runReviewGuardrail(argv) {
66
+ const actionIdx = argv.indexOf("--action");
67
+ const fileIdx = argv.indexOf("--file");
68
+ const configIdx = argv.indexOf("--config");
69
+ const formatIdx = argv.indexOf("--format");
70
+ const action = actionIdx >= 0 ? argv[actionIdx + 1] : "check";
71
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
72
+ const configPath = configIdx >= 0 ? argv[configIdx + 1] : ".judges-guardrails.json";
73
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
74
+ if (argv.includes("--help") || argv.includes("-h")) {
75
+ console.log(`
76
+ judges review-guardrail — Define and enforce review guardrails
77
+
78
+ Usage:
79
+ judges review-guardrail --action <action> [options]
80
+
81
+ Actions:
82
+ check Check verdict against guardrails (default)
83
+ list List configured guardrails
84
+ init Initialize default guardrails config
85
+
86
+ Options:
87
+ --action <act> Action: check, list, init
88
+ --file <path> Verdict JSON file (for check)
89
+ --config <path> Guardrails config (default: .judges-guardrails.json)
90
+ --format <fmt> Output format: table (default), json
91
+ --help, -h Show this help
92
+ `);
93
+ return;
94
+ }
95
+ const config = loadGuardrails(configPath);
96
+ if (action === "init") {
97
+ saveGuardrails(configPath, config);
98
+ console.log(`Guardrails config initialized: ${configPath}`);
99
+ return;
100
+ }
101
+ if (action === "list") {
102
+ if (format === "json") {
103
+ console.log(JSON.stringify(config, null, 2));
104
+ return;
105
+ }
106
+ console.log(`\nGuardrails (${config.guardrails.length})`);
107
+ console.log("═".repeat(60));
108
+ console.log(`${"ID".padEnd(22)} ${"Type".padEnd(22)} ${"Threshold".padEnd(12)} Enabled`);
109
+ console.log("─".repeat(60));
110
+ for (const g of config.guardrails) {
111
+ console.log(`${g.id.padEnd(22)} ${g.type.padEnd(22)} ${String(g.threshold).padEnd(12)} ${g.enabled}`);
112
+ }
113
+ console.log("═".repeat(60));
114
+ return;
115
+ }
116
+ // check
117
+ if (!filePath) {
118
+ console.error("Error: --file required for check");
119
+ process.exitCode = 1;
120
+ return;
121
+ }
122
+ if (!existsSync(filePath)) {
123
+ console.error(`Error: not found: ${filePath}`);
124
+ process.exitCode = 1;
125
+ return;
126
+ }
127
+ let verdict;
128
+ try {
129
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
130
+ }
131
+ catch {
132
+ console.error("Error: invalid JSON");
133
+ process.exitCode = 1;
134
+ return;
135
+ }
136
+ const results = evaluateGuardrails(verdict, config.guardrails);
137
+ const allPassed = results.every((r) => r.passed);
138
+ if (format === "json") {
139
+ console.log(JSON.stringify({ allPassed, results }, null, 2));
140
+ if (!allPassed)
141
+ process.exitCode = 1;
142
+ return;
143
+ }
144
+ console.log(`\nGuardrail Check: ${allPassed ? "PASS" : "FAIL"}`);
145
+ console.log("═".repeat(60));
146
+ console.log(`${"Status".padEnd(8)} ${"Guardrail".padEnd(25)} Detail`);
147
+ console.log("─".repeat(60));
148
+ for (const r of results) {
149
+ const icon = r.passed ? "PASS" : "FAIL";
150
+ console.log(`${icon.padEnd(8)} ${r.guardrailId.padEnd(25)} ${r.detail}`);
151
+ }
152
+ console.log("═".repeat(60));
153
+ if (!allPassed)
154
+ process.exitCode = 1;
155
+ }
156
+ //# sourceMappingURL=review-guardrail.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-guardrail.js","sourceRoot":"","sources":["../../src/commands/review-guardrail.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-perf-profile — Profile review performance and timing.
3
+ */
4
+ export declare function runReviewPerfProfile(argv: string[]): void;
5
+ //# sourceMappingURL=review-perf-profile.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-perf-profile.d.ts","sourceRoot":"","sources":["../../src/commands/review-perf-profile.ts"],"names":[],"mappings":"AAAA;;GAEG;AA8DH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA6DzD"}
@@ -0,0 +1,99 @@
1
+ /**
2
+ * Review-perf-profile — Profile review performance and timing.
3
+ */
4
+ import { readFileSync, existsSync } from "fs";
5
+ import { defaultRegistry } from "../judge-registry.js";
6
+ // ─── Analysis ───────────────────────────────────────────────────────────────
7
+ function profileReview(verdict, sourceFile) {
8
+ const metrics = [];
9
+ const judges = defaultRegistry.getJudges();
10
+ // findings per judge
11
+ const judgeFindings = new Map();
12
+ for (const f of verdict.findings) {
13
+ for (const j of judges) {
14
+ if (f.ruleId.startsWith(j.rulePrefix)) {
15
+ judgeFindings.set(j.id, (judgeFindings.get(j.id) || 0) + 1);
16
+ break;
17
+ }
18
+ }
19
+ }
20
+ // judge count
21
+ metrics.push({ component: "judges", metric: "total-judges", value: judges.length, unit: "count" });
22
+ metrics.push({ component: "judges", metric: "active-judges", value: judgeFindings.size, unit: "count" });
23
+ // findings metrics
24
+ metrics.push({ component: "findings", metric: "total-findings", value: verdict.findings.length, unit: "count" });
25
+ metrics.push({ component: "findings", metric: "critical", value: verdict.criticalCount, unit: "count" });
26
+ metrics.push({ component: "findings", metric: "high", value: verdict.highCount, unit: "count" });
27
+ // source file size
28
+ if (sourceFile && existsSync(sourceFile)) {
29
+ const content = readFileSync(sourceFile, "utf-8");
30
+ metrics.push({ component: "source", metric: "file-size", value: content.length, unit: "bytes" });
31
+ metrics.push({ component: "source", metric: "line-count", value: content.split("\n").length, unit: "lines" });
32
+ }
33
+ // evaluations count
34
+ metrics.push({ component: "evaluations", metric: "total", value: verdict.evaluations.length, unit: "count" });
35
+ // average score
36
+ if (verdict.evaluations.length > 0) {
37
+ const avgScore = verdict.evaluations.reduce((s, e) => s + e.score, 0) / verdict.evaluations.length;
38
+ metrics.push({ component: "evaluations", metric: "avg-score", value: Math.round(avgScore), unit: "score" });
39
+ }
40
+ return metrics;
41
+ }
42
+ // ─── CLI ────────────────────────────────────────────────────────────────────
43
+ export function runReviewPerfProfile(argv) {
44
+ const fileIdx = argv.indexOf("--file");
45
+ const sourceIdx = argv.indexOf("--source");
46
+ const formatIdx = argv.indexOf("--format");
47
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
48
+ const sourceFile = sourceIdx >= 0 ? argv[sourceIdx + 1] : undefined;
49
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
50
+ if (argv.includes("--help") || argv.includes("-h")) {
51
+ console.log(`
52
+ judges review-perf-profile — Profile review performance
53
+
54
+ Usage:
55
+ judges review-perf-profile --file <verdict.json> [--source <src.ts>]
56
+ [--format table|json]
57
+
58
+ Options:
59
+ --file <path> Path to verdict JSON file (required)
60
+ --source <path> Source file for size metrics
61
+ --format <fmt> Output format: table (default), json
62
+ --help, -h Show this help
63
+ `);
64
+ return;
65
+ }
66
+ if (!filePath) {
67
+ console.error("Error: --file required");
68
+ process.exitCode = 1;
69
+ return;
70
+ }
71
+ if (!existsSync(filePath)) {
72
+ console.error(`Error: not found: ${filePath}`);
73
+ process.exitCode = 1;
74
+ return;
75
+ }
76
+ let verdict;
77
+ try {
78
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
79
+ }
80
+ catch {
81
+ console.error("Error: invalid JSON");
82
+ process.exitCode = 1;
83
+ return;
84
+ }
85
+ const metrics = profileReview(verdict, sourceFile);
86
+ if (format === "json") {
87
+ console.log(JSON.stringify(metrics, null, 2));
88
+ return;
89
+ }
90
+ console.log(`\nReview Performance Profile`);
91
+ console.log("═".repeat(60));
92
+ console.log(`${"Component".padEnd(16)} ${"Metric".padEnd(20)} ${"Value".padEnd(12)} Unit`);
93
+ console.log("─".repeat(60));
94
+ for (const m of metrics) {
95
+ console.log(`${m.component.padEnd(16)} ${m.metric.padEnd(20)} ${String(m.value).padEnd(12)} ${m.unit}`);
96
+ }
97
+ console.log("═".repeat(60));
98
+ }
99
+ //# sourceMappingURL=review-perf-profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-perf-profile.js","sourceRoot":"","sources":["../../src/commands/review-perf-profile.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-scope-limit — Limit review scope to specific files, directories, or rules.
3
+ */
4
+ export declare function runReviewScopeLimit(argv: string[]): void;
5
+ //# sourceMappingURL=review-scope-limit.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-scope-limit.d.ts","sourceRoot":"","sources":["../../src/commands/review-scope-limit.ts"],"names":[],"mappings":"AAAA;;GAEG;AAiDH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA6ExD"}
@@ -0,0 +1,101 @@
1
+ /**
2
+ * Review-scope-limit — Limit review scope to specific files, directories, or rules.
3
+ */
4
+ import { readFileSync, existsSync } from "fs";
5
+ // ─── Analysis ───────────────────────────────────────────────────────────────
6
+ function applyScope(verdict, filters) {
7
+ const original = verdict.findings;
8
+ let filtered = [...original];
9
+ for (const filter of filters) {
10
+ if (filter.type === "rule-prefix") {
11
+ filtered = filtered.filter((f) => f.ruleId.startsWith(filter.value));
12
+ }
13
+ else if (filter.type === "severity") {
14
+ filtered = filtered.filter((f) => (f.severity || "medium").toLowerCase() === filter.value.toLowerCase());
15
+ }
16
+ }
17
+ return {
18
+ originalCount: original.length,
19
+ filteredCount: filtered.length,
20
+ removedCount: original.length - filtered.length,
21
+ filters,
22
+ findings: filtered.map((f) => ({
23
+ ruleId: f.ruleId,
24
+ title: f.title,
25
+ severity: (f.severity || "medium").toLowerCase(),
26
+ })),
27
+ };
28
+ }
29
+ // ─── CLI ────────────────────────────────────────────────────────────────────
30
+ export function runReviewScopeLimit(argv) {
31
+ const fileIdx = argv.indexOf("--file");
32
+ const prefixIdx = argv.indexOf("--prefix");
33
+ const severityIdx = argv.indexOf("--severity");
34
+ const formatIdx = argv.indexOf("--format");
35
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
36
+ const prefix = prefixIdx >= 0 ? argv[prefixIdx + 1] : undefined;
37
+ const severity = severityIdx >= 0 ? argv[severityIdx + 1] : undefined;
38
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
39
+ if (argv.includes("--help") || argv.includes("-h")) {
40
+ console.log(`
41
+ judges review-scope-limit — Limit review scope
42
+
43
+ Usage:
44
+ judges review-scope-limit --file <verdict.json> [--prefix <RULE-PREFIX>]
45
+ [--severity <level>] [--format table|json]
46
+
47
+ Options:
48
+ --file <path> Path to verdict JSON file (required)
49
+ --prefix <prefix> Filter by rule prefix (e.g., AUTH, CYBER)
50
+ --severity <level> Filter by severity (critical, high, medium, low)
51
+ --format <fmt> Output format: table (default), json
52
+ --help, -h Show this help
53
+ `);
54
+ return;
55
+ }
56
+ if (!filePath) {
57
+ console.error("Error: --file required");
58
+ process.exitCode = 1;
59
+ return;
60
+ }
61
+ if (!existsSync(filePath)) {
62
+ console.error(`Error: not found: ${filePath}`);
63
+ process.exitCode = 1;
64
+ return;
65
+ }
66
+ let verdict;
67
+ try {
68
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
69
+ }
70
+ catch {
71
+ console.error("Error: invalid JSON");
72
+ process.exitCode = 1;
73
+ return;
74
+ }
75
+ const filters = [];
76
+ if (prefix !== undefined)
77
+ filters.push({ type: "rule-prefix", value: prefix });
78
+ if (severity !== undefined)
79
+ filters.push({ type: "severity", value: severity });
80
+ const result = applyScope(verdict, filters);
81
+ if (format === "json") {
82
+ console.log(JSON.stringify(result, null, 2));
83
+ return;
84
+ }
85
+ console.log(`\nScope-Limited Review`);
86
+ console.log("═".repeat(65));
87
+ console.log(` Original: ${result.originalCount} → Filtered: ${result.filteredCount} (removed: ${result.removedCount})`);
88
+ if (filters.length > 0) {
89
+ console.log(` Filters: ${filters.map((f) => `${f.type}=${f.value}`).join(", ")}`);
90
+ }
91
+ console.log("─".repeat(65));
92
+ console.log(`${"Rule".padEnd(20)} ${"Severity".padEnd(10)} Title`);
93
+ console.log("─".repeat(65));
94
+ for (const f of result.findings) {
95
+ const rule = f.ruleId.length > 18 ? f.ruleId.slice(0, 18) + "…" : f.ruleId;
96
+ const title = f.title.length > 30 ? f.title.slice(0, 30) + "…" : f.title;
97
+ console.log(`${rule.padEnd(20)} ${f.severity.padEnd(10)} ${title}`);
98
+ }
99
+ console.log("═".repeat(65));
100
+ }
101
+ //# sourceMappingURL=review-scope-limit.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-scope-limit.js","sourceRoot":"","sources":["../../src/commands/review-scope-limit.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-threshold-tune — Tune review thresholds for optimal signal-to-noise.
3
+ */
4
+ export declare function runReviewThresholdTune(argv: string[]): void;
5
+ //# sourceMappingURL=review-threshold-tune.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-threshold-tune.d.ts","sourceRoot":"","sources":["../../src/commands/review-threshold-tune.ts"],"names":[],"mappings":"AAAA;;GAEG;AAuFH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAyE3D"}
@@ -0,0 +1,136 @@
1
+ /**
2
+ * Review-threshold-tune — Tune review thresholds for optimal signal-to-noise.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ // ─── Analysis ───────────────────────────────────────────────────────────────
6
+ function analyzeThresholds(verdicts) {
7
+ if (verdicts.length === 0)
8
+ return [];
9
+ const scores = verdicts.map((v) => v.overallScore);
10
+ const findingCounts = verdicts.map((v) => v.findings.length);
11
+ const criticals = verdicts.map((v) => v.criticalCount);
12
+ const avgScore = Math.round(scores.reduce((a, b) => a + b, 0) / scores.length);
13
+ const maxCritical = Math.max(...criticals);
14
+ const passRate = Math.round((verdicts.filter((v) => v.overallVerdict === "pass").length / verdicts.length) * 100);
15
+ const suggestions = [];
16
+ // score threshold
17
+ const p25Score = scores.sort((a, b) => a - b)[Math.floor(scores.length * 0.25)];
18
+ if (passRate < 30) {
19
+ suggestions.push({
20
+ metric: "min-score",
21
+ currentValue: 70,
22
+ suggestedValue: Math.max(30, p25Score - 5),
23
+ rationale: `Pass rate is ${passRate}%. Lowering threshold to increase adoption.`,
24
+ });
25
+ }
26
+ else if (passRate > 90) {
27
+ suggestions.push({
28
+ metric: "min-score",
29
+ currentValue: 70,
30
+ suggestedValue: Math.min(90, avgScore - 10),
31
+ rationale: `Pass rate is ${passRate}%. Raising threshold for higher quality.`,
32
+ });
33
+ }
34
+ // finding count threshold
35
+ const avgFindingsActual = Math.round(findingCounts.reduce((a, b) => a + b, 0) / findingCounts.length);
36
+ if (avgFindingsActual > 30) {
37
+ suggestions.push({
38
+ metric: "max-findings",
39
+ currentValue: 50,
40
+ suggestedValue: avgFindingsActual + 10,
41
+ rationale: `Average findings: ${avgFindingsActual}. Adjusting max to reduce noise.`,
42
+ });
43
+ }
44
+ // critical threshold
45
+ if (maxCritical > 5) {
46
+ suggestions.push({
47
+ metric: "max-critical",
48
+ currentValue: 0,
49
+ suggestedValue: 2,
50
+ rationale: `Max critical: ${maxCritical}. Allowing some criticals for gradual adoption.`,
51
+ });
52
+ }
53
+ // severity filter suggestion
54
+ const lowFindings = verdicts.reduce((sum, v) => sum + v.findings.filter((f) => (f.severity || "medium").toLowerCase() === "low").length, 0);
55
+ if (lowFindings > verdicts.length * 5) {
56
+ suggestions.push({
57
+ metric: "min-severity",
58
+ currentValue: 0,
59
+ suggestedValue: 1,
60
+ rationale: `${lowFindings} low-severity findings across ${verdicts.length} reports. Consider filtering.`,
61
+ });
62
+ }
63
+ return suggestions;
64
+ }
65
+ // ─── CLI ────────────────────────────────────────────────────────────────────
66
+ export function runReviewThresholdTune(argv) {
67
+ const dirIdx = argv.indexOf("--dir");
68
+ const fileIdx = argv.indexOf("--file");
69
+ const formatIdx = argv.indexOf("--format");
70
+ const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
71
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
72
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
73
+ if (argv.includes("--help") || argv.includes("-h")) {
74
+ console.log(`
75
+ judges review-threshold-tune — Tune review thresholds
76
+
77
+ Usage:
78
+ judges review-threshold-tune --dir <verdicts-dir> [--format table|json]
79
+ judges review-threshold-tune --file <verdict.json> [--format table|json]
80
+
81
+ Options:
82
+ --dir <path> Directory of verdict JSON files
83
+ --file <path> Single verdict JSON file
84
+ --format <fmt> Output format: table (default), json
85
+ --help, -h Show this help
86
+ `);
87
+ return;
88
+ }
89
+ const verdicts = [];
90
+ if (dirPath && existsSync(dirPath)) {
91
+ const files = readdirSync(dirPath).filter((f) => f.endsWith(".json"));
92
+ for (const file of files) {
93
+ try {
94
+ verdicts.push(JSON.parse(readFileSync(`${dirPath}/${file}`, "utf-8")));
95
+ }
96
+ catch {
97
+ // skip
98
+ }
99
+ }
100
+ }
101
+ else if (filePath && existsSync(filePath)) {
102
+ try {
103
+ verdicts.push(JSON.parse(readFileSync(filePath, "utf-8")));
104
+ }
105
+ catch {
106
+ console.error("Error: invalid JSON");
107
+ process.exitCode = 1;
108
+ return;
109
+ }
110
+ }
111
+ else {
112
+ console.error("Error: --dir or --file required");
113
+ process.exitCode = 1;
114
+ return;
115
+ }
116
+ const suggestions = analyzeThresholds(verdicts);
117
+ if (format === "json") {
118
+ console.log(JSON.stringify(suggestions, null, 2));
119
+ return;
120
+ }
121
+ console.log(`\nThreshold Tuning Suggestions (${verdicts.length} reports analyzed)`);
122
+ console.log("═".repeat(70));
123
+ if (suggestions.length === 0) {
124
+ console.log(" No threshold adjustments needed. Current settings look good.");
125
+ }
126
+ else {
127
+ console.log(`${"Metric".padEnd(18)} ${"Current".padEnd(10)} ${"Suggested".padEnd(12)} Rationale`);
128
+ console.log("─".repeat(70));
129
+ for (const s of suggestions) {
130
+ const rationale = s.rationale.length > 35 ? s.rationale.slice(0, 35) + "…" : s.rationale;
131
+ console.log(`${s.metric.padEnd(18)} ${String(s.currentValue).padEnd(10)} ${String(s.suggestedValue).padEnd(12)} ${rationale}`);
132
+ }
133
+ }
134
+ console.log("═".repeat(70));
135
+ }
136
+ //# sourceMappingURL=review-threshold-tune.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-threshold-tune.js","sourceRoot":"","sources":["../../src/commands/review-threshold-tune.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-webhook-notify — Configure and test webhook notifications for reviews.
3
+ */
4
+ export declare function runReviewWebhookNotify(argv: string[]): void;
5
+ //# sourceMappingURL=review-webhook-notify.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-webhook-notify.d.ts","sourceRoot":"","sources":["../../src/commands/review-webhook-notify.ts"],"names":[],"mappings":"AAAA;;GAEG;AAoEH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAmH3D"}