@kevinrabun/judges 3.102.0 → 3.104.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-annotation-export.d.ts +2 -0
  6. package/dist/commands/finding-annotation-export.d.ts.map +1 -0
  7. package/dist/commands/finding-annotation-export.js +98 -0
  8. package/dist/commands/finding-annotation-export.js.map +1 -0
  9. package/dist/commands/finding-cross-file-link.d.ts +2 -0
  10. package/dist/commands/finding-cross-file-link.d.ts.map +1 -0
  11. package/dist/commands/finding-cross-file-link.js +102 -0
  12. package/dist/commands/finding-cross-file-link.js.map +1 -0
  13. package/dist/commands/finding-dedup-smart.d.ts +2 -0
  14. package/dist/commands/finding-dedup-smart.d.ts.map +1 -0
  15. package/dist/commands/finding-dedup-smart.js +110 -0
  16. package/dist/commands/finding-dedup-smart.js.map +1 -0
  17. package/dist/commands/finding-hotspot-detect.d.ts +2 -0
  18. package/dist/commands/finding-hotspot-detect.d.ts.map +1 -0
  19. package/dist/commands/finding-hotspot-detect.js +121 -0
  20. package/dist/commands/finding-hotspot-detect.js.map +1 -0
  21. package/dist/commands/finding-merge-strategy.d.ts +2 -0
  22. package/dist/commands/finding-merge-strategy.d.ts.map +1 -0
  23. package/dist/commands/finding-merge-strategy.js +85 -0
  24. package/dist/commands/finding-merge-strategy.js.map +1 -0
  25. package/dist/commands/finding-similar-match.d.ts +2 -0
  26. package/dist/commands/finding-similar-match.d.ts.map +1 -0
  27. package/dist/commands/finding-similar-match.js +113 -0
  28. package/dist/commands/finding-similar-match.js.map +1 -0
  29. package/dist/commands/finding-trend-alert.d.ts +2 -0
  30. package/dist/commands/finding-trend-alert.d.ts.map +1 -0
  31. package/dist/commands/finding-trend-alert.js +127 -0
  32. package/dist/commands/finding-trend-alert.js.map +1 -0
  33. package/dist/commands/review-ai-feedback-loop.d.ts +2 -0
  34. package/dist/commands/review-ai-feedback-loop.d.ts.map +1 -0
  35. package/dist/commands/review-ai-feedback-loop.js +117 -0
  36. package/dist/commands/review-ai-feedback-loop.js.map +1 -0
  37. package/dist/commands/review-ci-insight.d.ts +2 -0
  38. package/dist/commands/review-ci-insight.d.ts.map +1 -0
  39. package/dist/commands/review-ci-insight.js +101 -0
  40. package/dist/commands/review-ci-insight.js.map +1 -0
  41. package/dist/commands/review-code-health-score.d.ts +2 -0
  42. package/dist/commands/review-code-health-score.d.ts.map +1 -0
  43. package/dist/commands/review-code-health-score.js +101 -0
  44. package/dist/commands/review-code-health-score.js.map +1 -0
  45. package/dist/commands/review-confidence-explain.d.ts +2 -0
  46. package/dist/commands/review-confidence-explain.d.ts.map +1 -0
  47. package/dist/commands/review-confidence-explain.js +100 -0
  48. package/dist/commands/review-confidence-explain.js.map +1 -0
  49. package/dist/commands/review-focus-area.d.ts +2 -0
  50. package/dist/commands/review-focus-area.d.ts.map +1 -0
  51. package/dist/commands/review-focus-area.js +97 -0
  52. package/dist/commands/review-focus-area.js.map +1 -0
  53. package/dist/commands/review-pr-size-check.d.ts +2 -0
  54. package/dist/commands/review-pr-size-check.d.ts.map +1 -0
  55. package/dist/commands/review-pr-size-check.js +99 -0
  56. package/dist/commands/review-pr-size-check.js.map +1 -0
  57. package/dist/commands/review-scope-suggest.d.ts +2 -0
  58. package/dist/commands/review-scope-suggest.d.ts.map +1 -0
  59. package/dist/commands/review-scope-suggest.js +113 -0
  60. package/dist/commands/review-scope-suggest.js.map +1 -0
  61. package/dist/commands/review-team-analytics.d.ts +2 -0
  62. package/dist/commands/review-team-analytics.d.ts.map +1 -0
  63. package/dist/commands/review-team-analytics.js +95 -0
  64. package/dist/commands/review-team-analytics.js.map +1 -0
  65. package/dist/commands/review-template-suggest.d.ts +2 -0
  66. package/dist/commands/review-template-suggest.d.ts.map +1 -0
  67. package/dist/commands/review-template-suggest.js +120 -0
  68. package/dist/commands/review-template-suggest.js.map +1 -0
  69. package/dist/commands/review-velocity-track.d.ts +2 -0
  70. package/dist/commands/review-velocity-track.d.ts.map +1 -0
  71. package/dist/commands/review-velocity-track.js +95 -0
  72. package/dist/commands/review-velocity-track.js.map +1 -0
  73. package/dist/commands/review-workload-balance.d.ts +2 -0
  74. package/dist/commands/review-workload-balance.d.ts.map +1 -0
  75. package/dist/commands/review-workload-balance.js +87 -0
  76. package/dist/commands/review-workload-balance.js.map +1 -0
  77. package/package.json +1 -1
  78. package/server.json +2 -2
@@ -0,0 +1,113 @@
1
+ import { readFileSync, existsSync } from "fs";
2
+ import { join } from "path";
3
+ function computeSimilarity(a, b) {
4
+ const wordsA = new Set(a
5
+ .toLowerCase()
6
+ .split(/\s+/)
7
+ .filter((w) => w.length > 2));
8
+ const wordsB = new Set(b
9
+ .toLowerCase()
10
+ .split(/\s+/)
11
+ .filter((w) => w.length > 2));
12
+ let overlap = 0;
13
+ for (const w of wordsA) {
14
+ if (wordsB.has(w))
15
+ overlap++;
16
+ }
17
+ const total = Math.max(wordsA.size, wordsB.size);
18
+ return total > 0 ? overlap / total : 0;
19
+ }
20
+ function findSimilar(findings, threshold) {
21
+ const groups = [];
22
+ const processed = new Set();
23
+ for (let i = 0; i < findings.length; i++) {
24
+ if (processed.has(i))
25
+ continue;
26
+ const anchor = findings[i];
27
+ const similar = [];
28
+ for (let j = i + 1; j < findings.length; j++) {
29
+ if (processed.has(j))
30
+ continue;
31
+ const other = findings[j];
32
+ const titleSim = computeSimilarity(anchor.title, other.title);
33
+ const descSim = computeSimilarity(anchor.description, other.description);
34
+ const combined = titleSim * 0.6 + descSim * 0.4;
35
+ if (combined >= threshold || anchor.ruleId === other.ruleId) {
36
+ processed.add(j);
37
+ similar.push({
38
+ ruleId: other.ruleId,
39
+ title: other.title,
40
+ similarity: Math.round(combined * 100),
41
+ });
42
+ }
43
+ }
44
+ processed.add(i);
45
+ if (similar.length > 0) {
46
+ groups.push({
47
+ anchor: anchor.ruleId,
48
+ anchorTitle: anchor.title,
49
+ severity: anchor.severity,
50
+ similarFindings: similar,
51
+ recommendation: similar.length >= 3
52
+ ? "Recurring pattern — consider a project-wide fix"
53
+ : "Similar findings found — review together",
54
+ });
55
+ }
56
+ }
57
+ return groups;
58
+ }
59
+ export function runFindingSimilarMatch(argv) {
60
+ if (argv.includes("--help") || argv.includes("-h")) {
61
+ console.log(`Usage: judges finding-similar-match [options]
62
+
63
+ Find similar findings across reviews.
64
+
65
+ Options:
66
+ --report <path> Path to verdict JSON file
67
+ --threshold <n> Similarity threshold 0-100 (default: 50)
68
+ --format <fmt> Output format: table (default) or json
69
+ -h, --help Show this help message`);
70
+ return;
71
+ }
72
+ const formatIdx = argv.indexOf("--format");
73
+ const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
74
+ const threshIdx = argv.indexOf("--threshold");
75
+ const threshold = threshIdx !== -1 && argv[threshIdx + 1] ? parseInt(argv[threshIdx + 1], 10) / 100 : 0.5;
76
+ const reportIdx = argv.indexOf("--report");
77
+ const reportPath = reportIdx !== -1 && argv[reportIdx + 1]
78
+ ? join(process.cwd(), argv[reportIdx + 1])
79
+ : join(process.cwd(), ".judges", "last-verdict.json");
80
+ if (!existsSync(reportPath)) {
81
+ console.log(`No report found at: ${reportPath}`);
82
+ console.log("Run a review first or provide --report.");
83
+ return;
84
+ }
85
+ const data = JSON.parse(readFileSync(reportPath, "utf-8"));
86
+ const findings = data.findings ?? [];
87
+ if (findings.length < 2) {
88
+ console.log("Need at least 2 findings for similarity analysis.");
89
+ return;
90
+ }
91
+ const groups = findSimilar(findings, threshold);
92
+ if (format === "json") {
93
+ console.log(JSON.stringify(groups, null, 2));
94
+ return;
95
+ }
96
+ console.log("\n=== Similar Findings ===\n");
97
+ console.log(`Analyzed: ${findings.length} findings`);
98
+ console.log(`Similar groups: ${groups.length}\n`);
99
+ if (groups.length === 0) {
100
+ console.log("No similar findings found at current threshold.");
101
+ return;
102
+ }
103
+ for (const g of groups) {
104
+ console.log(`[${g.severity.toUpperCase()}] ${g.anchor}: ${g.anchorTitle}`);
105
+ console.log(` Similar matches (${g.similarFindings.length}):`);
106
+ for (const s of g.similarFindings) {
107
+ console.log(` ${s.ruleId} (${s.similarity}% similar): ${s.title}`);
108
+ }
109
+ console.log(` → ${g.recommendation}`);
110
+ console.log();
111
+ }
112
+ }
113
+ //# sourceMappingURL=finding-similar-match.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-similar-match.js","sourceRoot":"","sources":["../../src/commands/finding-similar-match.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function runFindingTrendAlert(argv: string[]): void;
2
+ //# sourceMappingURL=finding-trend-alert.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-trend-alert.d.ts","sourceRoot":"","sources":["../../src/commands/finding-trend-alert.ts"],"names":[],"mappings":"AAgFA,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAgFzD"}
@@ -0,0 +1,127 @@
1
+ import { readFileSync, existsSync, readdirSync } from "fs";
2
+ import { join } from "path";
3
+ function detectTrends(currentFindings, baselineFindings) {
4
+ const alerts = [];
5
+ const currentCounts = new Map();
6
+ for (const f of currentFindings) {
7
+ const existing = currentCounts.get(f.ruleId);
8
+ if (existing !== undefined) {
9
+ existing.count += 1;
10
+ }
11
+ else {
12
+ currentCounts.set(f.ruleId, { count: 1, severity: f.severity });
13
+ }
14
+ }
15
+ const baselineCounts = new Map();
16
+ for (const f of baselineFindings) {
17
+ baselineCounts.set(f.ruleId, (baselineCounts.get(f.ruleId) ?? 0) + 1);
18
+ }
19
+ for (const [ruleId, data] of currentCounts) {
20
+ const baseline = baselineCounts.get(ruleId) ?? 0;
21
+ const delta = data.count - baseline;
22
+ if (delta <= 0)
23
+ continue;
24
+ let alertLevel;
25
+ let message;
26
+ if (delta >= 5 || (baseline === 0 && data.count >= 3)) {
27
+ alertLevel = "critical";
28
+ message =
29
+ baseline === 0
30
+ ? `New rule ${ruleId} appeared with ${data.count} findings`
31
+ : `Spike: ${ruleId} increased by ${delta} (${baseline} → ${data.count})`;
32
+ }
33
+ else if (delta >= 2) {
34
+ alertLevel = "warning";
35
+ message = `${ruleId} increased by ${delta} (${baseline} → ${data.count})`;
36
+ }
37
+ else {
38
+ alertLevel = "info";
39
+ message = `${ruleId} increased by ${delta}`;
40
+ }
41
+ alerts.push({
42
+ ruleId,
43
+ severity: data.severity,
44
+ currentCount: data.count,
45
+ baselineCount: baseline,
46
+ delta,
47
+ alertLevel,
48
+ message,
49
+ });
50
+ }
51
+ alerts.sort((a, b) => {
52
+ const order = { critical: 0, warning: 1, info: 2 };
53
+ return (order[a.alertLevel] ?? 3) - (order[b.alertLevel] ?? 3);
54
+ });
55
+ return alerts;
56
+ }
57
+ export function runFindingTrendAlert(argv) {
58
+ if (argv.includes("--help") || argv.includes("-h")) {
59
+ console.log(`Usage: judges finding-trend-alert [options]
60
+
61
+ Alert on emerging finding trends compared to baseline.
62
+
63
+ Options:
64
+ --current <path> Path to current verdict JSON
65
+ --baseline <path> Path to baseline verdict JSON
66
+ --dir <path> Directory with historical verdicts
67
+ --format <fmt> Output format: table (default) or json
68
+ -h, --help Show this help message`);
69
+ return;
70
+ }
71
+ const formatIdx = argv.indexOf("--format");
72
+ const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
73
+ const currentIdx = argv.indexOf("--current");
74
+ const currentPath = currentIdx !== -1 && argv[currentIdx + 1]
75
+ ? join(process.cwd(), argv[currentIdx + 1])
76
+ : join(process.cwd(), ".judges", "last-verdict.json");
77
+ let currentFindings = [];
78
+ if (existsSync(currentPath)) {
79
+ const data = JSON.parse(readFileSync(currentPath, "utf-8"));
80
+ currentFindings = data.findings ?? [];
81
+ }
82
+ let baselineFindings = [];
83
+ const baselineIdx = argv.indexOf("--baseline");
84
+ if (baselineIdx !== -1 && argv[baselineIdx + 1]) {
85
+ const bPath = join(process.cwd(), argv[baselineIdx + 1]);
86
+ if (existsSync(bPath)) {
87
+ const data = JSON.parse(readFileSync(bPath, "utf-8"));
88
+ baselineFindings = data.findings ?? [];
89
+ }
90
+ }
91
+ else {
92
+ const dirIdx = argv.indexOf("--dir");
93
+ const histDir = dirIdx !== -1 && argv[dirIdx + 1]
94
+ ? join(process.cwd(), argv[dirIdx + 1])
95
+ : join(process.cwd(), ".judges", "history");
96
+ if (existsSync(histDir)) {
97
+ const files = readdirSync(histDir).filter((f) => f.endsWith(".json")).sort();
98
+ if (files.length > 0) {
99
+ const latest = files[files.length - 1];
100
+ const data = JSON.parse(readFileSync(join(histDir, latest), "utf-8"));
101
+ baselineFindings = data.findings ?? [];
102
+ }
103
+ }
104
+ }
105
+ if (currentFindings.length === 0) {
106
+ console.log("No current findings found. Run a review first or provide --current.");
107
+ return;
108
+ }
109
+ const alerts = detectTrends(currentFindings, baselineFindings);
110
+ if (format === "json") {
111
+ console.log(JSON.stringify(alerts, null, 2));
112
+ return;
113
+ }
114
+ console.log("\n=== Finding Trend Alerts ===\n");
115
+ if (alerts.length === 0) {
116
+ console.log("No significant trends detected.");
117
+ return;
118
+ }
119
+ const critical = alerts.filter((a) => a.alertLevel === "critical").length;
120
+ const warning = alerts.filter((a) => a.alertLevel === "warning").length;
121
+ console.log(`Alerts: ${critical} critical, ${warning} warning, ${alerts.length - critical - warning} info\n`);
122
+ for (const alert of alerts) {
123
+ console.log(`[${alert.alertLevel.toUpperCase()}] ${alert.message}`);
124
+ console.log(` Severity: ${alert.severity} | Delta: +${alert.delta}`);
125
+ }
126
+ }
127
+ //# sourceMappingURL=finding-trend-alert.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-trend-alert.js","sourceRoot":"","sources":["../../src/commands/finding-trend-alert.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,IAAI,CAAC;AAC3D,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAmB5B,SAAS,YAAY,CAAC,eAA0B,EAAE,gBAA2B;IAC3E,MAAM,MAAM,GAAiB,EAAE,CAAC;IAEhC,MAAM,aAAa,GAAG,IAAI,GAAG,EAA+C,CAAC;IAC7E,KAAK,MAAM,CAAC,IAAI,eAAe,EAAE,CAAC;QAChC,MAAM,QAAQ,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC3B,QAAQ,CAAC,KAAK,IAAI,CAAC,CAAC;QACtB,CAAC;aAAM,CAAC;YACN,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;QAClE,CAAC;IACH,CAAC;IAED,MAAM,cAAc,GAAG,IAAI,GAAG,EAAkB,CAAC;IACjD,KAAK,MAAM,CAAC,IAAI,gBAAgB,EAAE,CAAC;QACjC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IACxE,CAAC;IAED,KAAK,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC,IAAI,aAAa,EAAE,CAAC;QAC3C,MAAM,QAAQ,GAAG,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACjD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC;QAEpC,IAAI,KAAK,IAAI,CAAC;YAAE,SAAS;QAEzB,IAAI,UAAkB,CAAC;QACvB,IAAI,OAAe,CAAC;QAEpB,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC,QAAQ,KAAK,CAAC,IAAI,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,EAAE,CAAC;YACtD,UAAU,GAAG,UAAU,CAAC;YACxB,OAAO;gBACL,QAAQ,KAAK,CAAC;oBACZ,CAAC,CAAC,YAAY,MAAM,kBAAkB,IAAI,CAAC,KAAK,WAAW;oBAC3D,CAAC,CAAC,UAAU,MAAM,iBAAiB,KAAK,KAAK,QAAQ,MAAM,IAAI,CAAC,KAAK,GAAG,CAAC;QAC/E,CAAC;aAAM,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC;YACtB,UAAU,GAAG,SAAS,CAAC;YACvB,OAAO,GAAG,GAAG,MAAM,iBAAiB,KAAK,KAAK,QAAQ,MAAM,IAAI,CAAC,KAAK,GAAG,CAAC;QAC5E,CAAC;aAAM,CAAC;YACN,UAAU,GAAG,MAAM,CAAC;YACpB,OAAO,GAAG,GAAG,MAAM,iBAAiB,KAAK,EAAE,CAAC;QAC9C,CAAC;QAED,MAAM,CAAC,IAAI,CAAC;YACV,MAAM;YACN,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,YAAY,EAAE,IAAI,CAAC,KAAK;YACxB,aAAa,EAAE,QAAQ;YACvB,KAAK;YACL,UAAU;YACV,OAAO;SACR,CAAC,CAAC;IACL,CAAC;IAED,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QACnB,MAAM,KAAK,GAA2B,EAAE,QAAQ,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;QAC3E,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;IACjE,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,oBAAoB,CAAC,IAAc;IACjD,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;8CAS8B,CAAC,CAAC;QAC5C,OAAO;IACT,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,MAAM,GAAG,SAAS,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAEvF,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IAC7C,MAAM,WAAW,GACf,UAAU,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;QACvC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC;QAC3C,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,mBAAmB,CAAC,CAAC;IAE1D,IAAI,eAAe,GAAc,EAAE,CAAC;IACpC,IAAI,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAoB,CAAC;QAC/E,eAAe,GAAG,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;IACxC,CAAC;IAED,IAAI,gBAAgB,GAAc,EAAE,CAAC;IACrC,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IAC/C,IAAI,WAAW,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,EAAE,CAAC;QAChD,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,CAAC;QACzD,IAAI,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;YACtB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,OAAO,CAAC,CAAoB,CAAC;YACzE,gBAAgB,GAAG,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;QACzC,CAAC;IACH,CAAC;SAAM,CAAC;QACN,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACrC,MAAM,OAAO,GACX,MAAM,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;YAC/B,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACvC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC;QAChD,IAAI,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,MAAM,KAAK,GAAI,WAAW,CAAC,OAAO,CAAyB,CAAC,MAAM,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAC9G,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;gBACvC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAoB,CAAC;gBACzF,gBAAgB,GAAG,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;YACzC,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjC,OAAO,CAAC,GAAG,CAAC,qEAAqE,CAAC,CAAC;QACnF,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,YAAY,CAAC,eAAe,EAAE,gBAAgB,CAAC,CAAC;IAE/D,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,kCAAkC,CAAC,CAAC;IAChD,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACxB,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;QAC/C,OAAO;IACT,CAAC;IAED,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,KAAK,UAAU,CAAC,CAAC,MAAM,CAAC;IAC1E,MAAM,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,KAAK,SAAS,CAAC,CAAC,MAAM,CAAC;IACxE,OAAO,CAAC,GAAG,CAAC,WAAW,QAAQ,cAAc,OAAO,aAAa,MAAM,CAAC,MAAM,GAAG,QAAQ,GAAG,OAAO,SAAS,CAAC,CAAC;IAE9G,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,OAAO,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,WAAW,EAAE,KAAK,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QACpE,OAAO,CAAC,GAAG,CAAC,eAAe,KAAK,CAAC,QAAQ,cAAc,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;IACxE,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function runReviewAiFeedbackLoop(argv: string[]): void;
2
+ //# sourceMappingURL=review-ai-feedback-loop.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-ai-feedback-loop.d.ts","sourceRoot":"","sources":["../../src/commands/review-ai-feedback-loop.ts"],"names":[],"mappings":"AAsDA,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAmG5D"}
@@ -0,0 +1,117 @@
1
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
2
+ import { join } from "path";
3
+ function computeStats(entries) {
4
+ const agree = entries.filter((e) => e.feedback === "agree").length;
5
+ const disagree = entries.filter((e) => e.feedback === "disagree").length;
6
+ const partial = entries.filter((e) => e.feedback === "partial").length;
7
+ const disagreedRules = new Map();
8
+ for (const e of entries) {
9
+ if (e.feedback === "disagree") {
10
+ disagreedRules.set(e.ruleId, (disagreedRules.get(e.ruleId) ?? 0) + 1);
11
+ }
12
+ }
13
+ const topDisagreed = [...disagreedRules.entries()]
14
+ .sort((a, b) => b[1] - a[1])
15
+ .slice(0, 10)
16
+ .map(([ruleId, count]) => ({ ruleId, count }));
17
+ return {
18
+ total: entries.length,
19
+ agree,
20
+ disagree,
21
+ partial,
22
+ agreementRate: entries.length > 0 ? agree / entries.length : 0,
23
+ topDisagreed,
24
+ };
25
+ }
26
+ export function runReviewAiFeedbackLoop(argv) {
27
+ if (argv.includes("--help") || argv.includes("-h")) {
28
+ console.log(`Usage: judges review-ai-feedback-loop [options]
29
+
30
+ Capture and analyze reviewer feedback on AI findings.
31
+
32
+ Options:
33
+ --add <ruleId> Add feedback for a specific rule
34
+ --feedback <type> Feedback type: agree, disagree, partial
35
+ --reason <text> Reason for feedback
36
+ --report <path> Path to verdict JSON to review
37
+ --stats Show feedback statistics
38
+ --format <fmt> Output format: table (default) or json
39
+ -h, --help Show this help message`);
40
+ return;
41
+ }
42
+ const formatIdx = argv.indexOf("--format");
43
+ const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
44
+ const feedbackDir = join(process.cwd(), ".judges", "feedback");
45
+ if (!existsSync(feedbackDir)) {
46
+ mkdirSync(feedbackDir, { recursive: true });
47
+ }
48
+ const feedbackFile = join(feedbackDir, "feedback-log.json");
49
+ let entries = [];
50
+ if (existsSync(feedbackFile)) {
51
+ entries = JSON.parse(readFileSync(feedbackFile, "utf-8"));
52
+ }
53
+ if (argv.includes("--stats")) {
54
+ const stats = computeStats(entries);
55
+ if (format === "json") {
56
+ console.log(JSON.stringify(stats, null, 2));
57
+ return;
58
+ }
59
+ console.log("\n=== AI Feedback Statistics ===\n");
60
+ console.log(`Total feedback entries: ${stats.total}`);
61
+ console.log(` Agree: ${stats.agree} (${(stats.agreementRate * 100).toFixed(1)}%)`);
62
+ console.log(` Disagree: ${stats.disagree}`);
63
+ console.log(` Partial: ${stats.partial}`);
64
+ if (stats.topDisagreed.length > 0) {
65
+ console.log("\nTop Disagreed Rules:");
66
+ for (const rule of stats.topDisagreed) {
67
+ console.log(` ${rule.ruleId}: ${rule.count} disagreements`);
68
+ }
69
+ }
70
+ return;
71
+ }
72
+ const addIdx = argv.indexOf("--add");
73
+ if (addIdx !== -1 && argv[addIdx + 1]) {
74
+ const ruleId = argv[addIdx + 1];
75
+ const fbIdx = argv.indexOf("--feedback");
76
+ const feedbackType = fbIdx !== -1 && argv[fbIdx + 1] ? argv[fbIdx + 1] : "agree";
77
+ const reasonIdx = argv.indexOf("--reason");
78
+ const reason = reasonIdx !== -1 && argv[reasonIdx + 1] ? argv[reasonIdx + 1] : "";
79
+ if (feedbackType !== "agree" && feedbackType !== "disagree" && feedbackType !== "partial") {
80
+ console.error("Invalid feedback type. Use: agree, disagree, or partial");
81
+ process.exitCode = 1;
82
+ return;
83
+ }
84
+ const entry = {
85
+ ruleId,
86
+ verdict: "reviewed",
87
+ feedback: feedbackType,
88
+ reason,
89
+ timestamp: new Date().toISOString(),
90
+ };
91
+ entries.push(entry);
92
+ writeFileSync(feedbackFile, JSON.stringify(entries, null, 2));
93
+ console.log(`Feedback recorded for ${ruleId}: ${feedbackType}`);
94
+ return;
95
+ }
96
+ const reportIdx = argv.indexOf("--report");
97
+ if (reportIdx !== -1 && argv[reportIdx + 1]) {
98
+ const reportPath = join(process.cwd(), argv[reportIdx + 1]);
99
+ if (!existsSync(reportPath)) {
100
+ console.error(`Report not found: ${reportPath}`);
101
+ process.exitCode = 1;
102
+ return;
103
+ }
104
+ const data = JSON.parse(readFileSync(reportPath, "utf-8"));
105
+ const findings = data.findings ?? [];
106
+ console.log(`\nFound ${findings.length} findings to review.`);
107
+ console.log("Use --add <ruleId> --feedback <type> to record feedback.\n");
108
+ for (const f of findings) {
109
+ const existingFb = entries.find((e) => e.ruleId === f.ruleId);
110
+ const fbStatus = existingFb !== undefined ? ` [${existingFb.feedback}]` : " [no feedback]";
111
+ console.log(` ${f.ruleId} (${f.severity})${fbStatus}: ${f.title}`);
112
+ }
113
+ return;
114
+ }
115
+ console.log("Use --stats, --add, or --report to interact with the feedback system.");
116
+ }
117
+ //# sourceMappingURL=review-ai-feedback-loop.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-ai-feedback-loop.js","sourceRoot":"","sources":["../../src/commands/review-ai-feedback-loop.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AACxE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AA0B5B,SAAS,YAAY,CAAC,OAAwB;IAC5C,MAAM,KAAK,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC,MAAM,CAAC;IACnE,MAAM,QAAQ,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,UAAU,CAAC,CAAC,MAAM,CAAC;IACzE,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC,MAAM,CAAC;IAEvE,MAAM,cAAc,GAAG,IAAI,GAAG,EAAkB,CAAC;IACjD,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE,CAAC;QACxB,IAAI,CAAC,CAAC,QAAQ,KAAK,UAAU,EAAE,CAAC;YAC9B,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC;IAED,MAAM,YAAY,GAAG,CAAC,GAAG,cAAc,CAAC,OAAO,EAAE,CAAC;SAC/C,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;SAC3B,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;SACZ,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;IAEjD,OAAO;QACL,KAAK,EAAE,OAAO,CAAC,MAAM;QACrB,KAAK;QACL,QAAQ;QACR,OAAO;QACP,aAAa,EAAE,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;QAC9D,YAAY;KACb,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,uBAAuB,CAAC,IAAc;IACpD,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;8CAW8B,CAAC,CAAC;QAC5C,OAAO;IACT,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,MAAM,GAAG,SAAS,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IACvF,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAE/D,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC7B,SAAS,CAAC,WAAW,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC9C,CAAC;IAED,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,EAAE,mBAAmB,CAAC,CAAC;IAC5D,IAAI,OAAO,GAAoB,EAAE,CAAC;IAClC,IAAI,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;QAC7B,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAoB,CAAC;IAC/E,CAAC;IAED,IAAI,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;QAC7B,MAAM,KAAK,GAAG,YAAY,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC5C,OAAO;QACT,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,oCAAoC,CAAC,CAAC;QAClD,OAAO,CAAC,GAAG,CAAC,2BAA2B,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;QACtD,OAAO,CAAC,GAAG,CAAC,YAAY,KAAK,CAAC,KAAK,KAAK,CAAC,KAAK,CAAC,aAAa,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACpF,OAAO,CAAC,GAAG,CAAC,eAAe,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7C,OAAO,CAAC,GAAG,CAAC,cAAc,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QAC3C,IAAI,KAAK,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClC,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;YACtC,KAAK,MAAM,IAAI,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;gBACtC,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,KAAK,gBAAgB,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;QACD,OAAO;IACT,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACrC,IAAI,MAAM,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC;QACtC,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAChC,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACzC,MAAM,YAAY,GAAG,KAAK,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;QACjF,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAC3C,MAAM,MAAM,GAAG,SAAS,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAElF,IAAI,YAAY,KAAK,OAAO,IAAI,YAAY,KAAK,UAAU,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;YAC1F,OAAO,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;YACzE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;QAED,MAAM,KAAK,GAAkB;YAC3B,MAAM;YACN,OAAO,EAAE,UAAU;YACnB,QAAQ,EAAE,YAAY;YACtB,MAAM;YACN,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SACpC,CAAC;QACF,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACpB,aAAa,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC9D,OAAO,CAAC,GAAG,CAAC,yBAAyB,MAAM,KAAK,YAAY,EAAE,CAAC,CAAC;QAChE,OAAO;IACT,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,IAAI,SAAS,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,EAAE,CAAC;QAC5C,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC;QAC5D,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC5B,OAAO,CAAC,KAAK,CAAC,qBAAqB,UAAU,EAAE,CAAC,CAAC;YACjD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;QACD,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC;QAC3D,MAAM,QAAQ,GAAc,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;QAChD,OAAO,CAAC,GAAG,CAAC,WAAW,QAAQ,CAAC,MAAM,sBAAsB,CAAC,CAAC;QAC9D,OAAO,CAAC,GAAG,CAAC,4DAA4D,CAAC,CAAC;QAC1E,KAAK,MAAM,CAAC,IAAI,QAAQ,EAAE,CAAC;YACzB,MAAM,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,MAAM,CAAC,CAAC;YAC9D,MAAM,QAAQ,GAAG,UAAU,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,UAAU,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,gBAAgB,CAAC;YAC3F,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,QAAQ,IAAI,QAAQ,KAAK,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;QACtE,CAAC;QACD,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,uEAAuE,CAAC,CAAC;AACvF,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function runReviewCiInsight(argv: string[]): void;
2
+ //# sourceMappingURL=review-ci-insight.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-ci-insight.d.ts","sourceRoot":"","sources":["../../src/commands/review-ci-insight.ts"],"names":[],"mappings":"AAsFA,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA0DvD"}
@@ -0,0 +1,101 @@
1
+ import { readFileSync, existsSync, readdirSync } from "fs";
2
+ import { join } from "path";
3
+ function analyzeCiData(verdicts) {
4
+ const insights = [];
5
+ if (verdicts.length === 0)
6
+ return insights;
7
+ const totalFindings = verdicts.reduce((sum, v) => sum + (v.findings?.length ?? 0), 0);
8
+ const avgFindings = totalFindings / verdicts.length;
9
+ const criticalTotal = verdicts.reduce((sum, v) => sum + (v.criticalCount ?? 0), 0);
10
+ const highTotal = verdicts.reduce((sum, v) => sum + (v.highCount ?? 0), 0);
11
+ const passRate = verdicts.filter((v) => v.overallVerdict === "pass").length / verdicts.length;
12
+ const scores = verdicts.map((v) => v.overallScore ?? 0);
13
+ const avgScore = scores.reduce((a, b) => a + b, 0) / scores.length;
14
+ const recentHalf = verdicts.slice(Math.floor(verdicts.length / 2));
15
+ const olderHalf = verdicts.slice(0, Math.floor(verdicts.length / 2));
16
+ const recentAvg = recentHalf.length > 0 ? recentHalf.reduce((s, v) => s + (v.overallScore ?? 0), 0) / recentHalf.length : 0;
17
+ const olderAvg = olderHalf.length > 0 ? olderHalf.reduce((s, v) => s + (v.overallScore ?? 0), 0) / olderHalf.length : 0;
18
+ const scoreTrend = recentAvg > olderAvg ? "improving" : recentAvg < olderAvg ? "declining" : "stable";
19
+ insights.push({
20
+ metric: "Average Findings per Review",
21
+ value: avgFindings.toFixed(1),
22
+ trend: avgFindings > 5 ? "high" : "normal",
23
+ recommendation: avgFindings > 5
24
+ ? "High finding density may slow CI — consider pre-commit hooks"
25
+ : "Finding density is manageable",
26
+ });
27
+ insights.push({
28
+ metric: "Review Pass Rate",
29
+ value: `${(passRate * 100).toFixed(1)}%`,
30
+ trend: passRate < 0.7 ? "concerning" : "healthy",
31
+ recommendation: passRate < 0.7 ? "Low pass rate — review coding guidelines or adjust thresholds" : "Pass rate is healthy",
32
+ });
33
+ insights.push({
34
+ metric: "Critical + High Findings",
35
+ value: `${criticalTotal + highTotal} total`,
36
+ trend: criticalTotal > 0 ? "attention" : "good",
37
+ recommendation: criticalTotal > 0 ? "Critical findings should block deployment" : "No critical findings — pipeline can proceed",
38
+ });
39
+ insights.push({
40
+ metric: "Quality Score Trend",
41
+ value: `${avgScore.toFixed(1)} avg`,
42
+ trend: scoreTrend,
43
+ recommendation: scoreTrend === "declining" ? "Quality declining — investigate recent changes" : "Quality trend is positive",
44
+ });
45
+ insights.push({
46
+ metric: "Reviews Analyzed",
47
+ value: `${verdicts.length}`,
48
+ trend: verdicts.length < 5 ? "limited data" : "sufficient",
49
+ recommendation: verdicts.length < 5 ? "Need more review runs for reliable insights" : "Sufficient data for trend analysis",
50
+ });
51
+ return insights;
52
+ }
53
+ export function runReviewCiInsight(argv) {
54
+ if (argv.includes("--help") || argv.includes("-h")) {
55
+ console.log(`Usage: judges review-ci-insight [options]
56
+
57
+ CI pipeline performance insights from review data.
58
+
59
+ Options:
60
+ --dir <path> Directory with verdict JSON files
61
+ --format <fmt> Output format: table (default) or json
62
+ -h, --help Show this help message`);
63
+ return;
64
+ }
65
+ const formatIdx = argv.indexOf("--format");
66
+ const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
67
+ const dirIdx = argv.indexOf("--dir");
68
+ const dirPath = dirIdx !== -1 && argv[dirIdx + 1]
69
+ ? join(process.cwd(), argv[dirIdx + 1])
70
+ : join(process.cwd(), ".judges", "history");
71
+ const verdicts = [];
72
+ if (existsSync(dirPath)) {
73
+ const files = readdirSync(dirPath).filter((f) => f.endsWith(".json")).sort();
74
+ for (const file of files) {
75
+ const content = JSON.parse(readFileSync(join(dirPath, file), "utf-8"));
76
+ verdicts.push(content);
77
+ }
78
+ }
79
+ const lastVerdict = join(process.cwd(), ".judges", "last-verdict.json");
80
+ if (existsSync(lastVerdict)) {
81
+ const data = JSON.parse(readFileSync(lastVerdict, "utf-8"));
82
+ verdicts.push(data);
83
+ }
84
+ if (verdicts.length === 0) {
85
+ console.log("No verdict data found. Run reviews first or provide --dir.");
86
+ return;
87
+ }
88
+ const insights = analyzeCiData(verdicts);
89
+ if (format === "json") {
90
+ console.log(JSON.stringify(insights, null, 2));
91
+ return;
92
+ }
93
+ console.log("\n=== CI Pipeline Insights ===\n");
94
+ console.log(`Data points: ${verdicts.length} review(s)\n`);
95
+ for (const insight of insights) {
96
+ console.log(`${insight.metric}: ${insight.value} [${insight.trend}]`);
97
+ console.log(` → ${insight.recommendation}`);
98
+ console.log();
99
+ }
100
+ }
101
+ //# sourceMappingURL=review-ci-insight.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-ci-insight.js","sourceRoot":"","sources":["../../src/commands/review-ci-insight.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function runReviewCodeHealthScore(argv: string[]): void;
2
+ //# sourceMappingURL=review-code-health-score.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-code-health-score.d.ts","sourceRoot":"","sources":["../../src/commands/review-code-health-score.ts"],"names":[],"mappings":"AAgFA,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA+C7D"}
@@ -0,0 +1,101 @@
1
+ import { readFileSync, existsSync } from "fs";
2
+ import { join } from "path";
3
+ function computeHealthScore(verdict) {
4
+ const findings = verdict.findings ?? [];
5
+ const total = findings.length;
6
+ const severityWeights = {
7
+ critical: 10,
8
+ high: 7,
9
+ medium: 4,
10
+ low: 2,
11
+ info: 0.5,
12
+ };
13
+ let severitySum = 0;
14
+ let confidenceSum = 0;
15
+ for (const f of findings) {
16
+ severitySum += severityWeights[f.severity] ?? 1;
17
+ confidenceSum += f.confidence ?? 0.5;
18
+ }
19
+ const maxSeverity = total * 10;
20
+ const severityScore = maxSeverity > 0 ? Math.max(0, 100 - (severitySum / maxSeverity) * 100) : 100;
21
+ const densityScore = total === 0 ? 100 : Math.max(0, 100 - total * 5);
22
+ const avgConfidence = total > 0 ? confidenceSum / total : 1;
23
+ const confidenceScore = avgConfidence * 100;
24
+ const passRateScore = verdict.overallVerdict === "pass" ? 100 : verdict.overallVerdict === "warning" ? 60 : 20;
25
+ const overall = Math.round(severityScore * 0.35 + densityScore * 0.25 + confidenceScore * 0.2 + passRateScore * 0.2);
26
+ let grade;
27
+ if (overall >= 90)
28
+ grade = "A";
29
+ else if (overall >= 80)
30
+ grade = "B";
31
+ else if (overall >= 70)
32
+ grade = "C";
33
+ else if (overall >= 60)
34
+ grade = "D";
35
+ else
36
+ grade = "F";
37
+ const recommendations = [];
38
+ if (severityScore < 50)
39
+ recommendations.push("Address critical/high severity findings urgently");
40
+ if (densityScore < 50)
41
+ recommendations.push("High finding density — consider breaking into smaller changes");
42
+ if (confidenceScore < 60)
43
+ recommendations.push("Low confidence findings — manual validation recommended");
44
+ if (passRateScore < 50)
45
+ recommendations.push("Review failing — resolve blockers before merge");
46
+ if (overall >= 90)
47
+ recommendations.push("Excellent code health — safe to proceed");
48
+ return {
49
+ overall,
50
+ grade,
51
+ breakdown: {
52
+ severityScore: Math.round(severityScore),
53
+ densityScore: Math.round(densityScore),
54
+ confidenceScore: Math.round(confidenceScore),
55
+ passRateScore: Math.round(passRateScore),
56
+ },
57
+ recommendations,
58
+ };
59
+ }
60
+ export function runReviewCodeHealthScore(argv) {
61
+ if (argv.includes("--help") || argv.includes("-h")) {
62
+ console.log(`Usage: judges review-code-health-score [options]
63
+
64
+ Compute aggregated code health score from review verdicts.
65
+
66
+ Options:
67
+ --report <path> Path to verdict JSON file
68
+ --format <fmt> Output format: table (default) or json
69
+ -h, --help Show this help message`);
70
+ return;
71
+ }
72
+ const formatIdx = argv.indexOf("--format");
73
+ const format = formatIdx !== -1 && argv[formatIdx + 1] ? argv[formatIdx + 1] : "table";
74
+ const reportIdx = argv.indexOf("--report");
75
+ const reportPath = reportIdx !== -1 && argv[reportIdx + 1]
76
+ ? join(process.cwd(), argv[reportIdx + 1])
77
+ : join(process.cwd(), ".judges", "last-verdict.json");
78
+ if (!existsSync(reportPath)) {
79
+ console.log(`No report found at: ${reportPath}`);
80
+ console.log("Run a review first or provide --report.");
81
+ return;
82
+ }
83
+ const verdict = JSON.parse(readFileSync(reportPath, "utf-8"));
84
+ const health = computeHealthScore(verdict);
85
+ if (format === "json") {
86
+ console.log(JSON.stringify(health, null, 2));
87
+ return;
88
+ }
89
+ console.log("\n=== Code Health Score ===\n");
90
+ console.log(`Overall: ${health.overall}/100 (Grade: ${health.grade})\n`);
91
+ console.log("Breakdown:");
92
+ console.log(` Severity: ${health.breakdown.severityScore}/100`);
93
+ console.log(` Density: ${health.breakdown.densityScore}/100`);
94
+ console.log(` Confidence: ${health.breakdown.confidenceScore}/100`);
95
+ console.log(` Pass Rate: ${health.breakdown.passRateScore}/100`);
96
+ console.log("\nRecommendations:");
97
+ for (const r of health.recommendations) {
98
+ console.log(` → ${r}`);
99
+ }
100
+ }
101
+ //# sourceMappingURL=review-code-health-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-code-health-score.js","sourceRoot":"","sources":["../../src/commands/review-code-health-score.ts"],"names":[],"mappings":"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"}