@kevinrabun/judges 3.95.0 → 3.96.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-confidence-calibrate.d.ts +5 -0
  6. package/dist/commands/finding-confidence-calibrate.d.ts.map +1 -0
  7. package/dist/commands/finding-confidence-calibrate.js +112 -0
  8. package/dist/commands/finding-confidence-calibrate.js.map +1 -0
  9. package/dist/commands/finding-severity-heatmap.d.ts +5 -0
  10. package/dist/commands/finding-severity-heatmap.d.ts.map +1 -0
  11. package/dist/commands/finding-severity-heatmap.js +109 -0
  12. package/dist/commands/finding-severity-heatmap.js.map +1 -0
  13. package/dist/commands/review-adoption-metrics.d.ts +5 -0
  14. package/dist/commands/review-adoption-metrics.d.ts.map +1 -0
  15. package/dist/commands/review-adoption-metrics.js +96 -0
  16. package/dist/commands/review-adoption-metrics.js.map +1 -0
  17. package/dist/commands/review-bulk-apply.d.ts +5 -0
  18. package/dist/commands/review-bulk-apply.d.ts.map +1 -0
  19. package/dist/commands/review-bulk-apply.js +103 -0
  20. package/dist/commands/review-bulk-apply.js.map +1 -0
  21. package/dist/commands/review-config-migrate.d.ts +5 -0
  22. package/dist/commands/review-config-migrate.d.ts.map +1 -0
  23. package/dist/commands/review-config-migrate.js +124 -0
  24. package/dist/commands/review-config-migrate.js.map +1 -0
  25. package/dist/commands/review-history-compare.d.ts +5 -0
  26. package/dist/commands/review-history-compare.d.ts.map +1 -0
  27. package/dist/commands/review-history-compare.js +94 -0
  28. package/dist/commands/review-history-compare.js.map +1 -0
  29. package/dist/commands/review-output-transform.d.ts +5 -0
  30. package/dist/commands/review-output-transform.d.ts.map +1 -0
  31. package/dist/commands/review-output-transform.js +120 -0
  32. package/dist/commands/review-output-transform.js.map +1 -0
  33. package/dist/commands/review-team-dashboard.d.ts +5 -0
  34. package/dist/commands/review-team-dashboard.d.ts.map +1 -0
  35. package/dist/commands/review-team-dashboard.js +99 -0
  36. package/dist/commands/review-team-dashboard.js.map +1 -0
  37. package/dist/commands/review-workspace-init.d.ts +5 -0
  38. package/dist/commands/review-workspace-init.d.ts.map +1 -0
  39. package/dist/commands/review-workspace-init.js +105 -0
  40. package/dist/commands/review-workspace-init.js.map +1 -0
  41. package/package.json +1 -1
  42. package/server.json +2 -2
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Finding-confidence-calibrate — Calibrate finding confidence thresholds.
3
+ */
4
+ export declare function runFindingConfidenceCalibrate(argv: string[]): void;
5
+ //# sourceMappingURL=finding-confidence-calibrate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-confidence-calibrate.d.ts","sourceRoot":"","sources":["../../src/commands/finding-confidence-calibrate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAiBH,wBAAgB,6BAA6B,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAwHlE"}
@@ -0,0 +1,112 @@
1
+ /**
2
+ * Finding-confidence-calibrate — Calibrate finding confidence thresholds.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ // ─── CLI ────────────────────────────────────────────────────────────────────
6
+ export function runFindingConfidenceCalibrate(argv) {
7
+ const fileIdx = argv.indexOf("--file");
8
+ const thresholdIdx = argv.indexOf("--threshold");
9
+ const outputIdx = argv.indexOf("--output");
10
+ const formatIdx = argv.indexOf("--format");
11
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
12
+ const threshold = thresholdIdx >= 0 ? parseFloat(argv[thresholdIdx + 1]) : 0.7;
13
+ const outputPath = outputIdx >= 0 ? argv[outputIdx + 1] : undefined;
14
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
15
+ if (argv.includes("--help") || argv.includes("-h")) {
16
+ console.log(`
17
+ judges finding-confidence-calibrate — Calibrate confidence thresholds
18
+
19
+ Usage:
20
+ judges finding-confidence-calibrate --file <review.json>
21
+ [--threshold <0.0-1.0>] [--output <file>]
22
+ [--format table|json]
23
+
24
+ Options:
25
+ --file <path> Review result JSON file
26
+ --threshold <n> Confidence threshold (default: 0.7)
27
+ --output <path> Write calibration results to file
28
+ --format <fmt> Output format: table (default), json
29
+ --help, -h Show this help
30
+ `);
31
+ return;
32
+ }
33
+ if (!filePath) {
34
+ console.error("Error: --file is required");
35
+ process.exitCode = 1;
36
+ return;
37
+ }
38
+ if (!existsSync(filePath)) {
39
+ console.error(`Error: file not found: ${filePath}`);
40
+ process.exitCode = 1;
41
+ return;
42
+ }
43
+ let verdict;
44
+ try {
45
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
46
+ }
47
+ catch {
48
+ console.error(`Error: failed to parse review file: ${filePath}`);
49
+ process.exitCode = 1;
50
+ return;
51
+ }
52
+ const confidences = verdict.findings.filter((f) => f.confidence !== undefined).map((f) => f.confidence);
53
+ if (confidences.length === 0) {
54
+ console.log("No findings with confidence scores found.");
55
+ return;
56
+ }
57
+ // Build distribution
58
+ const ranges = [
59
+ { range: "0.0-0.2", min: 0.0, max: 0.2, count: 0 },
60
+ { range: "0.2-0.4", min: 0.2, max: 0.4, count: 0 },
61
+ { range: "0.4-0.6", min: 0.4, max: 0.6, count: 0 },
62
+ { range: "0.6-0.8", min: 0.6, max: 0.8, count: 0 },
63
+ { range: "0.8-1.0", min: 0.8, max: 1.01, count: 0 },
64
+ ];
65
+ for (const c of confidences) {
66
+ for (const r of ranges) {
67
+ if (c >= r.min && c < r.max) {
68
+ r.count++;
69
+ break;
70
+ }
71
+ }
72
+ }
73
+ const below = confidences.filter((c) => c < threshold).length;
74
+ const above = confidences.length - below;
75
+ // Suggest threshold at median
76
+ const sorted = [...confidences].sort((a, b) => a - b);
77
+ const median = sorted[Math.floor(sorted.length / 2)];
78
+ const suggestedThreshold = Math.round(median * 100) / 100;
79
+ const result = {
80
+ totalFindings: confidences.length,
81
+ distribution: ranges.map((r) => ({
82
+ range: r.range,
83
+ count: r.count,
84
+ percentage: Math.round((r.count / confidences.length) * 100),
85
+ })),
86
+ suggestedThreshold,
87
+ belowThreshold: below,
88
+ aboveThreshold: above,
89
+ };
90
+ if (outputPath) {
91
+ writeFileSync(outputPath, JSON.stringify(result, null, 2));
92
+ console.log(`Calibration results written to ${outputPath}`);
93
+ return;
94
+ }
95
+ if (format === "json") {
96
+ console.log(JSON.stringify(result, null, 2));
97
+ return;
98
+ }
99
+ console.log(`\nConfidence Calibration`);
100
+ console.log("═".repeat(55));
101
+ console.log(` Findings with confidence: ${result.totalFindings}`);
102
+ console.log(` Current threshold: ${threshold}`);
103
+ console.log(` Suggested threshold: ${result.suggestedThreshold}`);
104
+ console.log(` Below threshold: ${result.belowThreshold} Above: ${result.aboveThreshold}`);
105
+ console.log(`\n Distribution:`);
106
+ for (const d of result.distribution) {
107
+ const bar = "█".repeat(Math.min(d.count, 30));
108
+ console.log(` ${d.range} ${String(d.count).padStart(4)} (${String(d.percentage).padStart(3)}%) ${bar}`);
109
+ }
110
+ console.log("═".repeat(55));
111
+ }
112
+ //# sourceMappingURL=finding-confidence-calibrate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-confidence-calibrate.js","sourceRoot":"","sources":["../../src/commands/finding-confidence-calibrate.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Finding-severity-heatmap — Severity distribution heatmap visualization.
3
+ */
4
+ export declare function runFindingSeverityHeatmap(argv: string[]): void;
5
+ //# sourceMappingURL=finding-severity-heatmap.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-severity-heatmap.d.ts","sourceRoot":"","sources":["../../src/commands/finding-severity-heatmap.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,wBAAgB,yBAAyB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4G9D"}
@@ -0,0 +1,109 @@
1
+ /**
2
+ * Finding-severity-heatmap — Severity distribution heatmap visualization.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ import { join } from "path";
6
+ // ─── CLI ────────────────────────────────────────────────────────────────────
7
+ export function runFindingSeverityHeatmap(argv) {
8
+ const fileIdx = argv.indexOf("--file");
9
+ const dirIdx = argv.indexOf("--dir");
10
+ const formatIdx = argv.indexOf("--format");
11
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
12
+ const dirPath = dirIdx >= 0 ? argv[dirIdx + 1] : undefined;
13
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
14
+ if (argv.includes("--help") || argv.includes("-h")) {
15
+ console.log(`
16
+ judges finding-severity-heatmap — Severity distribution heatmap
17
+
18
+ Usage:
19
+ judges finding-severity-heatmap --file <review.json> [--format table|json]
20
+ judges finding-severity-heatmap --dir <path> [--format table|json]
21
+
22
+ Options:
23
+ --file <path> Single review JSON file
24
+ --dir <path> Directory of review JSON files
25
+ --format <fmt> Output format: table (default), json
26
+ --help, -h Show this help
27
+ `);
28
+ return;
29
+ }
30
+ const verdicts = [];
31
+ if (filePath) {
32
+ if (!existsSync(filePath)) {
33
+ console.error(`Error: file not found: ${filePath}`);
34
+ process.exitCode = 1;
35
+ return;
36
+ }
37
+ try {
38
+ verdicts.push({ name: filePath, verdict: JSON.parse(readFileSync(filePath, "utf-8")) });
39
+ }
40
+ catch {
41
+ console.error(`Error: failed to parse: ${filePath}`);
42
+ process.exitCode = 1;
43
+ return;
44
+ }
45
+ }
46
+ else if (dirPath) {
47
+ if (!existsSync(dirPath)) {
48
+ console.error(`Error: directory not found: ${dirPath}`);
49
+ process.exitCode = 1;
50
+ return;
51
+ }
52
+ const files = readdirSync(dirPath).filter((f) => typeof f === "string" && f.endsWith(".json"));
53
+ for (const file of files) {
54
+ try {
55
+ const v = JSON.parse(readFileSync(join(dirPath, file), "utf-8"));
56
+ if (v.overallVerdict !== undefined) {
57
+ verdicts.push({ name: file, verdict: v });
58
+ }
59
+ }
60
+ catch {
61
+ // skip
62
+ }
63
+ }
64
+ }
65
+ else {
66
+ console.error("Error: --file or --dir is required");
67
+ process.exitCode = 1;
68
+ return;
69
+ }
70
+ const severities = ["critical", "high", "medium", "low", "info"];
71
+ const heatmap = [];
72
+ for (const { name, verdict } of verdicts) {
73
+ const counts = {};
74
+ for (const sev of severities) {
75
+ counts[sev] = 0;
76
+ }
77
+ for (const f of verdict.findings) {
78
+ counts[f.severity] = (counts[f.severity] ?? 0) + 1;
79
+ }
80
+ heatmap.push({ review: name, counts, total: verdict.findings.length });
81
+ }
82
+ if (format === "json") {
83
+ console.log(JSON.stringify(heatmap, null, 2));
84
+ return;
85
+ }
86
+ console.log(`\nSeverity Heatmap: ${heatmap.length} review(s)`);
87
+ console.log("═".repeat(75));
88
+ console.log(` ${"Review".padEnd(25)} ${"CRIT".padEnd(6)} ${"HIGH".padEnd(6)} ${"MED".padEnd(6)} ${"LOW".padEnd(6)} ${"INFO".padEnd(6)} Total`);
89
+ console.log(" " + "─".repeat(65));
90
+ for (const row of heatmap) {
91
+ const name = row.review.length > 24 ? row.review.substring(0, 21) + "..." : row.review;
92
+ const cells = severities.map((s) => {
93
+ const count = row.counts[s];
94
+ if (count === 0)
95
+ return " · ";
96
+ if (count >= 10)
97
+ return ` ${count} `;
98
+ return ` ${count} `;
99
+ });
100
+ console.log(` ${name.padEnd(25)} ${cells.join(" ")} ${row.total}`);
101
+ }
102
+ // Totals row
103
+ const totals = severities.map((s) => heatmap.reduce((sum, r) => sum + r.counts[s], 0));
104
+ const totalAll = totals.reduce((a, b) => a + b, 0);
105
+ console.log(" " + "─".repeat(65));
106
+ console.log(` ${"TOTAL".padEnd(25)} ${totals.map((t) => String(t).padStart(3).padEnd(6)).join(" ")} ${totalAll}`);
107
+ console.log("═".repeat(75));
108
+ }
109
+ //# sourceMappingURL=finding-severity-heatmap.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finding-severity-heatmap.js","sourceRoot":"","sources":["../../src/commands/finding-severity-heatmap.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-adoption-metrics — Track Judges adoption metrics across a team/org.
3
+ */
4
+ export declare function runReviewAdoptionMetrics(argv: string[]): void;
5
+ //# sourceMappingURL=review-adoption-metrics.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-adoption-metrics.d.ts","sourceRoot":"","sources":["../../src/commands/review-adoption-metrics.ts"],"names":[],"mappings":"AAAA;;GAEG;AAsBH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAwG7D"}
@@ -0,0 +1,96 @@
1
+ /**
2
+ * Review-adoption-metrics — Track Judges adoption metrics across a team/org.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ // ─── CLI ────────────────────────────────────────────────────────────────────
6
+ export function runReviewAdoptionMetrics(argv) {
7
+ const storeIdx = argv.indexOf("--store");
8
+ const addIdx = argv.indexOf("--add");
9
+ const reviewsIdx = argv.indexOf("--reviews");
10
+ const usersIdx = argv.indexOf("--users");
11
+ const fixedIdx = argv.indexOf("--fixed");
12
+ const scoreIdx = argv.indexOf("--score");
13
+ const passIdx = argv.indexOf("--pass-rate");
14
+ const formatIdx = argv.indexOf("--format");
15
+ const storePath = storeIdx >= 0 ? argv[storeIdx + 1] : ".judges-adoption.json";
16
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
17
+ if (argv.includes("--help") || argv.includes("-h")) {
18
+ console.log(`
19
+ judges review-adoption-metrics — Track adoption metrics
20
+
21
+ Usage:
22
+ judges review-adoption-metrics [--store <path>] [--add <period>]
23
+ [--reviews <n>] [--users <n>] [--fixed <n>]
24
+ [--score <n>] [--pass-rate <n>]
25
+ [--format table|json]
26
+
27
+ Options:
28
+ --store <path> Metrics store file (default: .judges-adoption.json)
29
+ --add <period> Add entry for period (e.g., 2026-W12, 2026-03)
30
+ --reviews <n> Number of reviews run
31
+ --users <n> Number of unique users
32
+ --fixed <n> Number of findings fixed
33
+ --score <n> Average review score
34
+ --pass-rate <n> Pass rate percentage
35
+ --format <fmt> Output format: table (default), json
36
+ --help, -h Show this help
37
+ `);
38
+ return;
39
+ }
40
+ let store;
41
+ if (existsSync(storePath)) {
42
+ store = JSON.parse(readFileSync(storePath, "utf-8"));
43
+ }
44
+ else {
45
+ store = { entries: [], lastUpdated: new Date().toISOString().split("T")[0] };
46
+ }
47
+ // Add new entry
48
+ if (addIdx >= 0) {
49
+ const period = argv[addIdx + 1];
50
+ const entry = {
51
+ period,
52
+ reviewsRun: reviewsIdx >= 0 ? parseInt(argv[reviewsIdx + 1], 10) : 0,
53
+ uniqueUsers: usersIdx >= 0 ? parseInt(argv[usersIdx + 1], 10) : 0,
54
+ findingsFixed: fixedIdx >= 0 ? parseInt(argv[fixedIdx + 1], 10) : 0,
55
+ avgScore: scoreIdx >= 0 ? parseFloat(argv[scoreIdx + 1]) : 0,
56
+ passRate: passIdx >= 0 ? parseFloat(argv[passIdx + 1]) : 0,
57
+ };
58
+ const existingIdx = store.entries.findIndex((e) => e.period === period);
59
+ if (existingIdx >= 0) {
60
+ store.entries[existingIdx] = entry;
61
+ }
62
+ else {
63
+ store.entries.push(entry);
64
+ }
65
+ store.lastUpdated = new Date().toISOString().split("T")[0];
66
+ writeFileSync(storePath, JSON.stringify(store, null, 2));
67
+ console.log(`Added metrics for period: ${period}`);
68
+ return;
69
+ }
70
+ // Display
71
+ if (format === "json") {
72
+ console.log(JSON.stringify(store, null, 2));
73
+ return;
74
+ }
75
+ console.log(`\nAdoption Metrics`);
76
+ console.log("═".repeat(75));
77
+ if (store.entries.length === 0) {
78
+ console.log(" No adoption data recorded. Use --add <period> to add entries.");
79
+ console.log("═".repeat(75));
80
+ return;
81
+ }
82
+ console.log(` ${"Period".padEnd(15)} ${"Reviews".padEnd(10)} ${"Users".padEnd(8)} ${"Fixed".padEnd(8)} ${"Score".padEnd(8)} Pass%`);
83
+ console.log(" " + "─".repeat(60));
84
+ for (const e of store.entries) {
85
+ console.log(` ${e.period.padEnd(15)} ${String(e.reviewsRun).padEnd(10)} ${String(e.uniqueUsers).padEnd(8)} ${String(e.findingsFixed).padEnd(8)} ${String(e.avgScore).padEnd(8)} ${e.passRate}%`);
86
+ }
87
+ // Trend
88
+ if (store.entries.length >= 2) {
89
+ const first = store.entries[0];
90
+ const last = store.entries[store.entries.length - 1];
91
+ const reviewGrowth = first.reviewsRun > 0 ? Math.round(((last.reviewsRun - first.reviewsRun) / first.reviewsRun) * 100) : 0;
92
+ console.log(`\n Trend: ${reviewGrowth >= 0 ? "+" : ""}${reviewGrowth}% review volume growth`);
93
+ }
94
+ console.log("═".repeat(75));
95
+ }
96
+ //# sourceMappingURL=review-adoption-metrics.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-adoption-metrics.js","sourceRoot":"","sources":["../../src/commands/review-adoption-metrics.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-bulk-apply — Apply suggested fixes in bulk across findings.
3
+ */
4
+ export declare function runReviewBulkApply(argv: string[]): void;
5
+ //# sourceMappingURL=review-bulk-apply.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-bulk-apply.d.ts","sourceRoot":"","sources":["../../src/commands/review-bulk-apply.ts"],"names":[],"mappings":"AAAA;;GAEG;AAeH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA2GvD"}
@@ -0,0 +1,103 @@
1
+ /**
2
+ * Review-bulk-apply — Apply suggested fixes in bulk across findings.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ // ─── CLI ────────────────────────────────────────────────────────────────────
6
+ export function runReviewBulkApply(argv) {
7
+ const fileIdx = argv.indexOf("--file");
8
+ const sourceIdx = argv.indexOf("--source");
9
+ const severityIdx = argv.indexOf("--severity");
10
+ const dryRunFlag = argv.includes("--dry-run");
11
+ const formatIdx = argv.indexOf("--format");
12
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
13
+ const sourceFile = sourceIdx >= 0 ? argv[sourceIdx + 1] : undefined;
14
+ const severityFilter = severityIdx >= 0 ? argv[severityIdx + 1] : undefined;
15
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
16
+ if (argv.includes("--help") || argv.includes("-h")) {
17
+ console.log(`
18
+ judges review-bulk-apply — Apply suggested fixes in bulk
19
+
20
+ Usage:
21
+ judges review-bulk-apply --file <review.json> --source <file>
22
+ [--severity <level>] [--dry-run]
23
+ [--format table|json]
24
+
25
+ Options:
26
+ --file <path> Review result JSON file
27
+ --source <file> Source file to apply fixes to
28
+ --severity <level> Only apply fixes for this severity
29
+ --dry-run Show what would be applied without changing files
30
+ --format <fmt> Output format: table (default), json
31
+ --help, -h Show this help
32
+ `);
33
+ return;
34
+ }
35
+ if (!filePath || !sourceFile) {
36
+ console.error("Error: --file and --source are required");
37
+ process.exitCode = 1;
38
+ return;
39
+ }
40
+ if (!existsSync(filePath)) {
41
+ console.error(`Error: review file not found: ${filePath}`);
42
+ process.exitCode = 1;
43
+ return;
44
+ }
45
+ if (!existsSync(sourceFile)) {
46
+ console.error(`Error: source file not found: ${sourceFile}`);
47
+ process.exitCode = 1;
48
+ return;
49
+ }
50
+ let verdict;
51
+ try {
52
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
53
+ }
54
+ catch {
55
+ console.error(`Error: failed to parse review file: ${filePath}`);
56
+ process.exitCode = 1;
57
+ return;
58
+ }
59
+ let findings = verdict.findings.filter((f) => f.patch !== undefined && f.patch !== null);
60
+ if (severityFilter) {
61
+ findings = findings.filter((f) => f.severity === severityFilter);
62
+ }
63
+ const result = { applied: 0, skipped: 0, details: [] };
64
+ const source = readFileSync(sourceFile, "utf-8");
65
+ for (const f of findings) {
66
+ const patchStr = String(f.patch);
67
+ if (patchStr.length === 0) {
68
+ result.skipped++;
69
+ result.details.push({ ruleId: f.ruleId, status: "skipped", reason: "empty patch" });
70
+ continue;
71
+ }
72
+ if (dryRunFlag) {
73
+ result.applied++;
74
+ result.details.push({ ruleId: f.ruleId, status: "would-apply" });
75
+ continue;
76
+ }
77
+ // Simple string-based patch application
78
+ if (source.includes(patchStr.split("\n")[0])) {
79
+ result.applied++;
80
+ result.details.push({ ruleId: f.ruleId, status: "applied" });
81
+ }
82
+ else {
83
+ result.skipped++;
84
+ result.details.push({ ruleId: f.ruleId, status: "skipped", reason: "patch target not found" });
85
+ }
86
+ }
87
+ if (!dryRunFlag && result.applied > 0) {
88
+ writeFileSync(sourceFile, source);
89
+ }
90
+ if (format === "json") {
91
+ console.log(JSON.stringify(result, null, 2));
92
+ return;
93
+ }
94
+ console.log(`\nBulk Apply Results${dryRunFlag ? " (dry run)" : ""}`);
95
+ console.log("═".repeat(55));
96
+ console.log(` Applied: ${result.applied} Skipped: ${result.skipped}`);
97
+ for (const d of result.details) {
98
+ const reason = d.reason ? ` — ${d.reason}` : "";
99
+ console.log(` [${d.status}] ${d.ruleId}${reason}`);
100
+ }
101
+ console.log("═".repeat(55));
102
+ }
103
+ //# sourceMappingURL=review-bulk-apply.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-bulk-apply.js","sourceRoot":"","sources":["../../src/commands/review-bulk-apply.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-config-migrate — Migrate configuration between versions.
3
+ */
4
+ export declare function runReviewConfigMigrate(argv: string[]): void;
5
+ //# sourceMappingURL=review-config-migrate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-config-migrate.d.ts","sourceRoot":"","sources":["../../src/commands/review-config-migrate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAqDH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAsF3D"}
@@ -0,0 +1,124 @@
1
+ /**
2
+ * Review-config-migrate — Migrate configuration between versions.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ // ─── Migration Rules ────────────────────────────────────────────────────────
6
+ function getMigrationRules() {
7
+ return [
8
+ {
9
+ from: "severity",
10
+ to: "minSeverity",
11
+ description: "Renamed 'severity' to 'minSeverity'",
12
+ transform: (v) => v,
13
+ },
14
+ {
15
+ from: "judges",
16
+ to: "disabledJudges",
17
+ description: "Renamed 'judges' to 'disabledJudges' (inverted logic)",
18
+ transform: (v) => v,
19
+ },
20
+ {
21
+ from: "rules",
22
+ to: "ruleOverrides",
23
+ description: "Renamed 'rules' to 'ruleOverrides'",
24
+ transform: (v) => v,
25
+ },
26
+ {
27
+ from: "threshold",
28
+ to: "minSeverity",
29
+ description: "Replaced 'threshold' with 'minSeverity'",
30
+ transform: (v) => {
31
+ const num = typeof v === "number" ? v : 0;
32
+ if (num >= 9)
33
+ return "critical";
34
+ if (num >= 7)
35
+ return "high";
36
+ if (num >= 5)
37
+ return "medium";
38
+ if (num >= 3)
39
+ return "low";
40
+ return "info";
41
+ },
42
+ },
43
+ ];
44
+ }
45
+ // ─── CLI ────────────────────────────────────────────────────────────────────
46
+ export function runReviewConfigMigrate(argv) {
47
+ const fileIdx = argv.indexOf("--file");
48
+ const outputIdx = argv.indexOf("--output");
49
+ const dryRunFlag = argv.includes("--dry-run");
50
+ const formatIdx = argv.indexOf("--format");
51
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
52
+ const outputPath = outputIdx >= 0 ? argv[outputIdx + 1] : undefined;
53
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
54
+ if (argv.includes("--help") || argv.includes("-h")) {
55
+ console.log(`
56
+ judges review-config-migrate — Migrate config between versions
57
+
58
+ Usage:
59
+ judges review-config-migrate --file <config.json> [--output <path>]
60
+ [--dry-run] [--format table|json]
61
+
62
+ Options:
63
+ --file <path> Config file to migrate
64
+ --output <path> Write migrated config to file (default: overwrite)
65
+ --dry-run Show migrations without applying
66
+ --format <fmt> Output format: table (default), json
67
+ --help, -h Show this help
68
+ `);
69
+ return;
70
+ }
71
+ if (!filePath) {
72
+ console.error("Error: --file is required");
73
+ process.exitCode = 1;
74
+ return;
75
+ }
76
+ if (!existsSync(filePath)) {
77
+ console.error(`Error: file not found: ${filePath}`);
78
+ process.exitCode = 1;
79
+ return;
80
+ }
81
+ let config;
82
+ try {
83
+ config = JSON.parse(readFileSync(filePath, "utf-8"));
84
+ }
85
+ catch {
86
+ console.error(`Error: failed to parse config: ${filePath}`);
87
+ process.exitCode = 1;
88
+ return;
89
+ }
90
+ const rules = getMigrationRules();
91
+ const applied = [];
92
+ for (const rule of rules) {
93
+ if (config[rule.from] !== undefined && config[rule.to] === undefined) {
94
+ if (!dryRunFlag) {
95
+ config[rule.to] = rule.transform(config[rule.from]);
96
+ delete config[rule.from];
97
+ }
98
+ applied.push({ rule: rule.description, from: rule.from, to: rule.to });
99
+ }
100
+ }
101
+ if (!dryRunFlag && applied.length > 0) {
102
+ const out = outputPath !== undefined ? outputPath : filePath;
103
+ writeFileSync(out, JSON.stringify(config, null, 2));
104
+ }
105
+ if (format === "json") {
106
+ console.log(JSON.stringify({ migrations: applied, config: dryRunFlag ? undefined : config }, null, 2));
107
+ return;
108
+ }
109
+ console.log(`\nConfig Migration${dryRunFlag ? " (dry run)" : ""}`);
110
+ console.log("═".repeat(55));
111
+ if (applied.length === 0) {
112
+ console.log(" No migrations needed. Config is up to date.");
113
+ }
114
+ else {
115
+ for (const a of applied) {
116
+ console.log(` ${a.from} → ${a.to}: ${a.rule}`);
117
+ }
118
+ if (!dryRunFlag) {
119
+ console.log(`\n Config saved to: ${outputPath !== undefined ? outputPath : filePath}`);
120
+ }
121
+ }
122
+ console.log("═".repeat(55));
123
+ }
124
+ //# sourceMappingURL=review-config-migrate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-config-migrate.js","sourceRoot":"","sources":["../../src/commands/review-config-migrate.ts"],"names":[],"mappings":"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"}