@kevinrabun/judges 3.89.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 (42) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/cli.d.ts.map +1 -1
  3. package/dist/cli.js +63 -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-cluster-group.d.ts +5 -0
  10. package/dist/commands/finding-cluster-group.d.ts.map +1 -0
  11. package/dist/commands/finding-cluster-group.js +106 -0
  12. package/dist/commands/finding-cluster-group.js.map +1 -0
  13. package/dist/commands/finding-evidence-collect.d.ts +5 -0
  14. package/dist/commands/finding-evidence-collect.d.ts.map +1 -0
  15. package/dist/commands/finding-evidence-collect.js +115 -0
  16. package/dist/commands/finding-evidence-collect.js.map +1 -0
  17. package/dist/commands/finding-resolution-tracker.d.ts +5 -0
  18. package/dist/commands/finding-resolution-tracker.d.ts.map +1 -0
  19. package/dist/commands/finding-resolution-tracker.js +164 -0
  20. package/dist/commands/finding-resolution-tracker.js.map +1 -0
  21. package/dist/commands/finding-trend-analysis.d.ts +5 -0
  22. package/dist/commands/finding-trend-analysis.d.ts.map +1 -0
  23. package/dist/commands/finding-trend-analysis.js +96 -0
  24. package/dist/commands/finding-trend-analysis.js.map +1 -0
  25. package/dist/commands/review-batch-mode.d.ts +5 -0
  26. package/dist/commands/review-batch-mode.d.ts.map +1 -0
  27. package/dist/commands/review-batch-mode.js +98 -0
  28. package/dist/commands/review-batch-mode.js.map +1 -0
  29. package/dist/commands/review-compliance-gate.d.ts +5 -0
  30. package/dist/commands/review-compliance-gate.d.ts.map +1 -0
  31. package/dist/commands/review-compliance-gate.js +152 -0
  32. package/dist/commands/review-compliance-gate.js.map +1 -0
  33. package/dist/commands/review-threshold-tune.d.ts +5 -0
  34. package/dist/commands/review-threshold-tune.d.ts.map +1 -0
  35. package/dist/commands/review-threshold-tune.js +136 -0
  36. package/dist/commands/review-threshold-tune.js.map +1 -0
  37. package/dist/commands/review-webhook-notify.d.ts +5 -0
  38. package/dist/commands/review-webhook-notify.d.ts.map +1 -0
  39. package/dist/commands/review-webhook-notify.js +146 -0
  40. package/dist/commands/review-webhook-notify.js.map +1 -0
  41. package/package.json +1 -1
  42. package/server.json +2 -2
@@ -0,0 +1,96 @@
1
+ /**
2
+ * Finding-trend-analysis — Analyze finding trends across multiple reports.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ // ─── Analysis ───────────────────────────────────────────────────────────────
6
+ function analyzeTrends(verdicts) {
7
+ const sorted = [...verdicts].sort((a, b) => a.timestamp.localeCompare(b.timestamp));
8
+ if (sorted.length < 2) {
9
+ return { points: sorted, direction: "stable", avgScoreChange: 0, avgFindingsChange: 0 };
10
+ }
11
+ const scoreChanges = [];
12
+ const findingsChanges = [];
13
+ for (let i = 1; i < sorted.length; i++) {
14
+ scoreChanges.push(sorted[i].score - sorted[i - 1].score);
15
+ findingsChanges.push(sorted[i].findingCount - sorted[i - 1].findingCount);
16
+ }
17
+ const avgScoreChange = scoreChanges.reduce((a, b) => a + b, 0) / scoreChanges.length;
18
+ const avgFindingsChange = findingsChanges.reduce((a, b) => a + b, 0) / findingsChanges.length;
19
+ let direction = "stable";
20
+ if (avgScoreChange > 2)
21
+ direction = "improving";
22
+ else if (avgScoreChange < -2)
23
+ direction = "degrading";
24
+ return {
25
+ points: sorted,
26
+ direction,
27
+ avgScoreChange: Math.round(avgScoreChange * 10) / 10,
28
+ avgFindingsChange: Math.round(avgFindingsChange * 10) / 10,
29
+ };
30
+ }
31
+ // ─── CLI ────────────────────────────────────────────────────────────────────
32
+ export function runFindingTrendAnalysis(argv) {
33
+ const dirIdx = argv.indexOf("--dir");
34
+ const formatIdx = argv.indexOf("--format");
35
+ const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
36
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
37
+ if (argv.includes("--help") || argv.includes("-h")) {
38
+ console.log(`
39
+ judges finding-trend-analysis — Analyze finding trends
40
+
41
+ Usage:
42
+ judges finding-trend-analysis --dir <verdicts-dir> [--format table|json]
43
+
44
+ Options:
45
+ --dir <path> Directory of verdict JSON files (required)
46
+ --format <fmt> Output format: table (default), json
47
+ --help, -h Show this help
48
+ `);
49
+ return;
50
+ }
51
+ if (!dirPath) {
52
+ console.error("Error: --dir required");
53
+ process.exitCode = 1;
54
+ return;
55
+ }
56
+ if (!existsSync(dirPath)) {
57
+ console.error(`Error: not found: ${dirPath}`);
58
+ process.exitCode = 1;
59
+ return;
60
+ }
61
+ const files = readdirSync(dirPath).filter((f) => f.endsWith(".json"));
62
+ const points = [];
63
+ for (const file of files) {
64
+ try {
65
+ const v = JSON.parse(readFileSync(`${dirPath}/${file}`, "utf-8"));
66
+ points.push({
67
+ timestamp: v.timestamp || file.replace(".json", ""),
68
+ score: v.overallScore,
69
+ findingCount: v.findings.length,
70
+ criticalCount: v.criticalCount,
71
+ highCount: v.highCount,
72
+ });
73
+ }
74
+ catch {
75
+ // skip
76
+ }
77
+ }
78
+ const analysis = analyzeTrends(points);
79
+ if (format === "json") {
80
+ console.log(JSON.stringify(analysis, null, 2));
81
+ return;
82
+ }
83
+ const icon = analysis.direction === "improving" ? "UP" : analysis.direction === "degrading" ? "DOWN" : "FLAT";
84
+ console.log(`\nFinding Trend Analysis — ${icon}`);
85
+ console.log("═".repeat(70));
86
+ console.log(` Direction: ${analysis.direction} | Avg Score Δ: ${analysis.avgScoreChange} | Avg Findings Δ: ${analysis.avgFindingsChange}`);
87
+ console.log("─".repeat(70));
88
+ console.log(`${"Timestamp".padEnd(24)} ${"Score".padEnd(8)} ${"Findings".padEnd(10)} ${"Critical".padEnd(10)} High`);
89
+ console.log("─".repeat(70));
90
+ for (const p of analysis.points) {
91
+ const ts = p.timestamp.length > 22 ? p.timestamp.slice(0, 22) + "…" : p.timestamp;
92
+ console.log(`${ts.padEnd(24)} ${String(p.score).padEnd(8)} ${String(p.findingCount).padEnd(10)} ${String(p.criticalCount).padEnd(10)} ${p.highCount}`);
93
+ }
94
+ console.log("═".repeat(70));
95
+ }
96
+ //# sourceMappingURL=finding-trend-analysis.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-trend-analysis.js","sourceRoot":"","sources":["../../src/commands/finding-trend-analysis.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-batch-mode — Run reviews on multiple files in batch.
3
+ */
4
+ export declare function runReviewBatchMode(argv: string[]): void;
5
+ //# sourceMappingURL=review-batch-mode.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-batch-mode.d.ts","sourceRoot":"","sources":["../../src/commands/review-batch-mode.ts"],"names":[],"mappings":"AAAA;;GAEG;AA8DH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAoEvD"}
@@ -0,0 +1,98 @@
1
+ /**
2
+ * Review-batch-mode — Run reviews on multiple files in batch.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync, writeFileSync } from "fs";
5
+ // ─── Analysis ───────────────────────────────────────────────────────────────
6
+ function processBatch(dir, pattern) {
7
+ const files = readdirSync(dir).filter((f) => f.endsWith(".json"));
8
+ const filtered = pattern ? files.filter((f) => f.includes(pattern)) : files;
9
+ const entries = [];
10
+ let totalScore = 0;
11
+ let passCount = 0;
12
+ for (const file of filtered) {
13
+ try {
14
+ const content = readFileSync(`${dir}/${file}`, "utf-8");
15
+ const verdict = JSON.parse(content);
16
+ entries.push({
17
+ file,
18
+ score: verdict.overallScore,
19
+ verdict: verdict.overallVerdict,
20
+ findingCount: verdict.findings.length,
21
+ criticalCount: verdict.criticalCount,
22
+ });
23
+ totalScore += verdict.overallScore;
24
+ if (verdict.overallVerdict === "pass")
25
+ passCount++;
26
+ }
27
+ catch {
28
+ // skip invalid files
29
+ }
30
+ }
31
+ return {
32
+ totalFiles: filtered.length,
33
+ processedFiles: entries.length,
34
+ avgScore: entries.length > 0 ? Math.round(totalScore / entries.length) : 0,
35
+ passRate: entries.length > 0 ? Math.round((passCount / entries.length) * 100) : 0,
36
+ entries: entries.sort((a, b) => a.score - b.score),
37
+ };
38
+ }
39
+ // ─── CLI ────────────────────────────────────────────────────────────────────
40
+ export function runReviewBatchMode(argv) {
41
+ const dirIdx = argv.indexOf("--dir");
42
+ const patternIdx = argv.indexOf("--pattern");
43
+ const outputIdx = argv.indexOf("--output");
44
+ const formatIdx = argv.indexOf("--format");
45
+ const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
46
+ const pattern = patternIdx >= 0 ? argv[patternIdx + 1] : "";
47
+ const outputPath = outputIdx >= 0 ? argv[outputIdx + 1] : undefined;
48
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
49
+ if (argv.includes("--help") || argv.includes("-h")) {
50
+ console.log(`
51
+ judges review-batch-mode — Batch review processing
52
+
53
+ Usage:
54
+ judges review-batch-mode --dir <verdicts-dir> [--pattern <filter>]
55
+ [--output <file>] [--format table|json]
56
+
57
+ Options:
58
+ --dir <path> Directory of verdict JSON files (required)
59
+ --pattern <str> Filter filenames containing this string
60
+ --output <path> Write results to file
61
+ --format <fmt> Output format: table (default), json
62
+ --help, -h Show this help
63
+ `);
64
+ return;
65
+ }
66
+ if (!dirPath) {
67
+ console.error("Error: --dir required");
68
+ process.exitCode = 1;
69
+ return;
70
+ }
71
+ if (!existsSync(dirPath)) {
72
+ console.error(`Error: not found: ${dirPath}`);
73
+ process.exitCode = 1;
74
+ return;
75
+ }
76
+ const result = processBatch(dirPath, pattern);
77
+ if (outputPath) {
78
+ writeFileSync(outputPath, JSON.stringify(result, null, 2));
79
+ console.log(`Batch results written to ${outputPath}`);
80
+ return;
81
+ }
82
+ if (format === "json") {
83
+ console.log(JSON.stringify(result, null, 2));
84
+ return;
85
+ }
86
+ console.log(`\nBatch Review Results`);
87
+ console.log("═".repeat(70));
88
+ console.log(` Files: ${result.processedFiles}/${result.totalFiles} | Avg Score: ${result.avgScore} | Pass Rate: ${result.passRate}%`);
89
+ console.log("─".repeat(70));
90
+ console.log(`${"File".padEnd(30)} ${"Score".padEnd(8)} ${"Verdict".padEnd(10)} ${"Findings".padEnd(10)} Critical`);
91
+ console.log("─".repeat(70));
92
+ for (const e of result.entries) {
93
+ const name = e.file.length > 28 ? e.file.slice(0, 28) + "…" : e.file;
94
+ console.log(`${name.padEnd(30)} ${String(e.score).padEnd(8)} ${e.verdict.padEnd(10)} ${String(e.findingCount).padEnd(10)} ${e.criticalCount}`);
95
+ }
96
+ console.log("═".repeat(70));
97
+ }
98
+ //# sourceMappingURL=review-batch-mode.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-batch-mode.js","sourceRoot":"","sources":["../../src/commands/review-batch-mode.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,IAAI,CAAC;AAqB1E,+EAA+E;AAE/E,SAAS,YAAY,CAAC,GAAW,EAAE,OAAe;IAChD,MAAM,KAAK,GAAI,WAAW,CAAC,GAAG,CAAyB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;IAC3F,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;IAE5E,MAAM,OAAO,GAAiB,EAAE,CAAC;IACjC,IAAI,UAAU,GAAG,CAAC,CAAC;IACnB,IAAI,SAAS,GAAG,CAAC,CAAC;IAElB,KAAK,MAAM,IAAI,IAAI,QAAQ,EAAE,CAAC;QAC5B,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,YAAY,CAAC,GAAG,GAAG,IAAI,IAAI,EAAE,EAAE,OAAO,CAAC,CAAC;YACxD,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAoB,CAAC;YACvD,OAAO,CAAC,IAAI,CAAC;gBACX,IAAI;gBACJ,KAAK,EAAE,OAAO,CAAC,YAAY;gBAC3B,OAAO,EAAE,OAAO,CAAC,cAAc;gBAC/B,YAAY,EAAE,OAAO,CAAC,QAAQ,CAAC,MAAM;gBACrC,aAAa,EAAE,OAAO,CAAC,aAAa;aACrC,CAAC,CAAC;YACH,UAAU,IAAI,OAAO,CAAC,YAAY,CAAC;YACnC,IAAI,OAAO,CAAC,cAAc,KAAK,MAAM;gBAAE,SAAS,EAAE,CAAC;QACrD,CAAC;QAAC,MAAM,CAAC;YACP,qBAAqB;QACvB,CAAC;IACH,CAAC;IAED,OAAO;QACL,UAAU,EAAE,QAAQ,CAAC,MAAM;QAC3B,cAAc,EAAE,OAAO,CAAC,MAAM;QAC9B,QAAQ,EAAE,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1E,QAAQ,EAAE,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QACjF,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC;KACnD,CAAC;AACJ,CAAC;AAED,+EAA+E;AAE/E,MAAM,UAAU,kBAAkB,CAAC,IAAc;IAC/C,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACrC,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IAC7C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAC3D,MAAM,OAAO,GAAG,UAAU,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAC5D,MAAM,UAAU,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IACpE,MAAM,MAAM,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAE9D,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;;;CAaf,CAAC,CAAC;QACC,OAAO;IACT,CAAC;IAED,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;QACvC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IACD,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;QACzB,OAAO,CAAC,KAAK,CAAC,qBAAqB,OAAO,EAAE,CAAC,CAAC;QAC9C,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,YAAY,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IAE9C,IAAI,UAAU,EAAE,CAAC;QACf,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC3D,OAAO,CAAC,GAAG,CAAC,4BAA4B,UAAU,EAAE,CAAC,CAAC;QACtD,OAAO;IACT,CAAC;IAED,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC7C,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;IACtC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC5B,OAAO,CAAC,GAAG,CACT,YAAY,MAAM,CAAC,cAAc,IAAI,MAAM,CAAC,UAAU,mBAAmB,MAAM,CAAC,QAAQ,mBAAmB,MAAM,CAAC,QAAQ,GAAG,CAC9H,CAAC;IACF,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC5B,OAAO,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC,WAAW,CAAC,CAAC;IACnH,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAE5B,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QAC/B,MAAM,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACrE,OAAO,CAAC,GAAG,CACT,GAAG,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,aAAa,EAAE,CAClI,CAAC;IACJ,CAAC;IACD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;AAC9B,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-compliance-gate — Gate reviews based on compliance policy.
3
+ */
4
+ export declare function runReviewComplianceGate(argv: string[]): void;
5
+ //# sourceMappingURL=review-compliance-gate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-compliance-gate.d.ts","sourceRoot":"","sources":["../../src/commands/review-compliance-gate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAkGH,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA8F5D"}
@@ -0,0 +1,152 @@
1
+ /**
2
+ * Review-compliance-gate — Gate reviews based on compliance policy.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
5
+ import { dirname } from "path";
6
+ // ─── Helpers ────────────────────────────────────────────────────────────────
7
+ function loadPolicy(policyPath) {
8
+ if (!existsSync(policyPath)) {
9
+ return {
10
+ version: 1,
11
+ name: "default",
12
+ rules: [
13
+ { id: "no-critical", description: "No critical findings", type: "max-critical", threshold: 0 },
14
+ { id: "max-high-5", description: "Max 5 high findings", type: "max-high", threshold: 5 },
15
+ { id: "min-score-50", description: "Minimum score of 50", type: "min-score", threshold: 50 },
16
+ ],
17
+ };
18
+ }
19
+ try {
20
+ return JSON.parse(readFileSync(policyPath, "utf-8"));
21
+ }
22
+ catch {
23
+ return { version: 1, name: "default", rules: [] };
24
+ }
25
+ }
26
+ function evaluateCompliance(verdict, policy) {
27
+ const results = [];
28
+ for (const rule of policy.rules) {
29
+ let passed = true;
30
+ let detail = "";
31
+ switch (rule.type) {
32
+ case "max-critical":
33
+ passed = verdict.criticalCount <= rule.threshold;
34
+ detail = `Critical: ${verdict.criticalCount} (max: ${rule.threshold})`;
35
+ break;
36
+ case "max-high":
37
+ passed = verdict.highCount <= rule.threshold;
38
+ detail = `High: ${verdict.highCount} (max: ${rule.threshold})`;
39
+ break;
40
+ case "min-score":
41
+ passed = verdict.overallScore >= rule.threshold;
42
+ detail = `Score: ${verdict.overallScore} (min: ${rule.threshold})`;
43
+ break;
44
+ case "required-judges": {
45
+ const activeJudges = new Set(verdict.evaluations.map((e) => e.judgeId));
46
+ const missing = (rule.requiredJudges || []).filter((j) => !activeJudges.has(j));
47
+ passed = missing.length === 0;
48
+ detail = passed ? "All required judges active" : `Missing: ${missing.join(", ")}`;
49
+ break;
50
+ }
51
+ case "no-unfixed":
52
+ passed = verdict.findings.every((f) => f.patch !== undefined && f.patch !== null);
53
+ detail = passed ? "All findings have fixes" : "Some findings lack fixes";
54
+ break;
55
+ }
56
+ results.push({ ruleId: rule.id, passed, detail });
57
+ }
58
+ return {
59
+ policy: policy.name,
60
+ passed: results.every((r) => r.passed),
61
+ results,
62
+ };
63
+ }
64
+ // ─── CLI ────────────────────────────────────────────────────────────────────
65
+ export function runReviewComplianceGate(argv) {
66
+ const fileIdx = argv.indexOf("--file");
67
+ const policyIdx = argv.indexOf("--policy");
68
+ const actionIdx = argv.indexOf("--action");
69
+ const formatIdx = argv.indexOf("--format");
70
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
71
+ const policyPath = policyIdx >= 0 ? argv[policyIdx + 1] : ".judges-compliance.json";
72
+ const action = actionIdx >= 0 ? argv[actionIdx + 1] : "check";
73
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
74
+ if (argv.includes("--help") || argv.includes("-h")) {
75
+ console.log(`
76
+ judges review-compliance-gate — Compliance gate for reviews
77
+
78
+ Usage:
79
+ judges review-compliance-gate --file <verdict.json> [--policy <path>]
80
+ [--action check|init|show] [--format table|json]
81
+
82
+ Options:
83
+ --file <path> Verdict JSON file (for check)
84
+ --policy <path> Policy file (default: .judges-compliance.json)
85
+ --action <act> Action: check (default), init, show
86
+ --format <fmt> Output format: table (default), json
87
+ --help, -h Show this help
88
+ `);
89
+ return;
90
+ }
91
+ const policy = loadPolicy(policyPath);
92
+ if (action === "init") {
93
+ const dir = dirname(policyPath);
94
+ if (!existsSync(dir)) {
95
+ mkdirSync(dir, { recursive: true });
96
+ }
97
+ writeFileSync(policyPath, JSON.stringify(policy, null, 2));
98
+ console.log(`Compliance policy initialized: ${policyPath}`);
99
+ return;
100
+ }
101
+ if (action === "show") {
102
+ if (format === "json") {
103
+ console.log(JSON.stringify(policy, null, 2));
104
+ return;
105
+ }
106
+ console.log(`\nCompliance Policy: ${policy.name}`);
107
+ console.log("═".repeat(60));
108
+ for (const r of policy.rules) {
109
+ console.log(` ${r.id.padEnd(20)} ${r.type.padEnd(18)} threshold=${r.threshold}`);
110
+ }
111
+ console.log("═".repeat(60));
112
+ return;
113
+ }
114
+ // check
115
+ if (!filePath) {
116
+ console.error("Error: --file required for check");
117
+ process.exitCode = 1;
118
+ return;
119
+ }
120
+ if (!existsSync(filePath)) {
121
+ console.error(`Error: not found: ${filePath}`);
122
+ process.exitCode = 1;
123
+ return;
124
+ }
125
+ let verdict;
126
+ try {
127
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
128
+ }
129
+ catch {
130
+ console.error("Error: invalid JSON");
131
+ process.exitCode = 1;
132
+ return;
133
+ }
134
+ const result = evaluateCompliance(verdict, policy);
135
+ if (format === "json") {
136
+ console.log(JSON.stringify(result, null, 2));
137
+ if (!result.passed)
138
+ process.exitCode = 1;
139
+ return;
140
+ }
141
+ console.log(`\nCompliance Gate: ${result.passed ? "PASS" : "FAIL"} (policy: ${result.policy})`);
142
+ console.log("═".repeat(60));
143
+ console.log(`${"Status".padEnd(8)} ${"Rule".padEnd(22)} Detail`);
144
+ console.log("─".repeat(60));
145
+ for (const r of result.results) {
146
+ console.log(`${(r.passed ? "PASS" : "FAIL").padEnd(8)} ${r.ruleId.padEnd(22)} ${r.detail}`);
147
+ }
148
+ console.log("═".repeat(60));
149
+ if (!result.passed)
150
+ process.exitCode = 1;
151
+ }
152
+ //# sourceMappingURL=review-compliance-gate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-compliance-gate.js","sourceRoot":"","sources":["../../src/commands/review-compliance-gate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AACxE,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAyB/B,+EAA+E;AAE/E,SAAS,UAAU,CAAC,UAAkB;IACpC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;QAC5B,OAAO;YACL,OAAO,EAAE,CAAC;YACV,IAAI,EAAE,SAAS;YACf,KAAK,EAAE;gBACL,EAAE,EAAE,EAAE,aAAa,EAAE,WAAW,EAAE,sBAAsB,EAAE,IAAI,EAAE,cAAc,EAAE,SAAS,EAAE,CAAC,EAAE;gBAC9F,EAAE,EAAE,EAAE,YAAY,EAAE,WAAW,EAAE,qBAAqB,EAAE,IAAI,EAAE,UAAU,EAAE,SAAS,EAAE,CAAC,EAAE;gBACxF,EAAE,EAAE,EAAE,cAAc,EAAE,WAAW,EAAE,qBAAqB,EAAE,IAAI,EAAE,WAAW,EAAE,SAAS,EAAE,EAAE,EAAE;aAC7F;SACF,CAAC;IACJ,CAAC;IACD,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC;IACvD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC;IACpD,CAAC;AACH,CAAC;AAED,SAAS,kBAAkB,CAAC,OAAwB,EAAE,MAAwB;IAC5E,MAAM,OAAO,GAAgC,EAAE,CAAC;IAEhD,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;QAChC,IAAI,MAAM,GAAG,IAAI,CAAC;QAClB,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,KAAK,cAAc;gBACjB,MAAM,GAAG,OAAO,CAAC,aAAa,IAAI,IAAI,CAAC,SAAS,CAAC;gBACjD,MAAM,GAAG,aAAa,OAAO,CAAC,aAAa,UAAU,IAAI,CAAC,SAAS,GAAG,CAAC;gBACvE,MAAM;YAER,KAAK,UAAU;gBACb,MAAM,GAAG,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC,SAAS,CAAC;gBAC7C,MAAM,GAAG,SAAS,OAAO,CAAC,SAAS,UAAU,IAAI,CAAC,SAAS,GAAG,CAAC;gBAC/D,MAAM;YAER,KAAK,WAAW;gBACd,MAAM,GAAG,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC,SAAS,CAAC;gBAChD,MAAM,GAAG,UAAU,OAAO,CAAC,YAAY,UAAU,IAAI,CAAC,SAAS,GAAG,CAAC;gBACnE,MAAM;YAER,KAAK,iBAAiB,CAAC,CAAC,CAAC;gBACvB,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;gBACxE,MAAM,OAAO,GAAG,CAAC,IAAI,CAAC,cAAc,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChF,MAAM,GAAG,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC;gBAC9B,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,4BAA4B,CAAC,CAAC,CAAC,YAAY,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBAClF,MAAM;YACR,CAAC;YAED,KAAK,YAAY;gBACf,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,KAAK,SAAS,IAAI,CAAC,CAAC,KAAK,KAAK,IAAI,CAAC,CAAC;gBAClF,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC,0BAA0B,CAAC;gBACzE,MAAM;QACV,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACpD,CAAC;IAED,OAAO;QACL,MAAM,EAAE,MAAM,CAAC,IAAI;QACnB,MAAM,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC;QACtC,OAAO;KACR,CAAC;AACJ,CAAC;AAED,+EAA+E;AAE/E,MAAM,UAAU,uBAAuB,CAAC,IAAc;IACpD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACvC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,QAAQ,GAAG,OAAO,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAC9D,MAAM,UAAU,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,yBAAyB,CAAC;IACpF,MAAM,MAAM,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAC9D,MAAM,MAAM,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAE9D,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;;;CAaf,CAAC,CAAC;QACC,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,UAAU,CAAC,UAAU,CAAC,CAAC;IAEtC,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,MAAM,GAAG,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;QAChC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACrB,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACtC,CAAC;QACD,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC3D,OAAO,CAAC,GAAG,CAAC,kCAAkC,UAAU,EAAE,CAAC,CAAC;QAC5D,OAAO;IACT,CAAC;IAED,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC7C,OAAO;QACT,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,wBAAwB,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC5B,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YAC7B,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC;QACpF,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC5B,OAAO;IACT,CAAC;IAED,QAAQ;IACR,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;QAClD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IACD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1B,OAAO,CAAC,KAAK,CAAC,qBAAqB,QAAQ,EAAE,CAAC,CAAC;QAC/C,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,OAAwB,CAAC;IAC7B,IAAI,CAAC;QACH,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC;IACxD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;QACrC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,kBAAkB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAEnD,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC7C,IAAI,CAAC,MAAM,CAAC,MAAM;YAAE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACzC,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,sBAAsB,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,aAAa,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC;IAChG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC5B,OAAO,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;IACjE,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAE5B,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;IAC9F,CAAC;IACD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAE5B,IAAI,CAAC,MAAM,CAAC,MAAM;QAAE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;AAC3C,CAAC"}
@@ -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"}