@kevinrabun/judges 3.82.0 → 3.84.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-category-stats.d.ts +5 -0
  6. package/dist/commands/finding-category-stats.d.ts.map +1 -0
  7. package/dist/commands/finding-category-stats.js +105 -0
  8. package/dist/commands/finding-category-stats.js.map +1 -0
  9. package/dist/commands/finding-cluster-analysis.d.ts +5 -0
  10. package/dist/commands/finding-cluster-analysis.d.ts.map +1 -0
  11. package/dist/commands/finding-cluster-analysis.js +134 -0
  12. package/dist/commands/finding-cluster-analysis.js.map +1 -0
  13. package/dist/commands/finding-compare-runs.d.ts +5 -0
  14. package/dist/commands/finding-compare-runs.d.ts.map +1 -0
  15. package/dist/commands/finding-compare-runs.js +106 -0
  16. package/dist/commands/finding-compare-runs.js.map +1 -0
  17. package/dist/commands/finding-evidence-chain.d.ts +5 -0
  18. package/dist/commands/finding-evidence-chain.d.ts.map +1 -0
  19. package/dist/commands/finding-evidence-chain.js +148 -0
  20. package/dist/commands/finding-evidence-chain.js.map +1 -0
  21. package/dist/commands/finding-fix-priority.d.ts +5 -0
  22. package/dist/commands/finding-fix-priority.d.ts.map +1 -0
  23. package/dist/commands/finding-fix-priority.js +99 -0
  24. package/dist/commands/finding-fix-priority.js.map +1 -0
  25. package/dist/commands/finding-hotfix-suggest.d.ts +8 -0
  26. package/dist/commands/finding-hotfix-suggest.d.ts.map +1 -0
  27. package/dist/commands/finding-hotfix-suggest.js +171 -0
  28. package/dist/commands/finding-hotfix-suggest.js.map +1 -0
  29. package/dist/commands/finding-noise-filter.d.ts +8 -0
  30. package/dist/commands/finding-noise-filter.d.ts.map +1 -0
  31. package/dist/commands/finding-noise-filter.js +141 -0
  32. package/dist/commands/finding-noise-filter.js.map +1 -0
  33. package/dist/commands/finding-trend-report.d.ts +5 -0
  34. package/dist/commands/finding-trend-report.d.ts.map +1 -0
  35. package/dist/commands/finding-trend-report.js +108 -0
  36. package/dist/commands/finding-trend-report.js.map +1 -0
  37. package/dist/commands/review-approval-gate.d.ts +8 -0
  38. package/dist/commands/review-approval-gate.d.ts.map +1 -0
  39. package/dist/commands/review-approval-gate.js +191 -0
  40. package/dist/commands/review-approval-gate.js.map +1 -0
  41. package/dist/commands/review-branch-compare.d.ts +5 -0
  42. package/dist/commands/review-branch-compare.d.ts.map +1 -0
  43. package/dist/commands/review-branch-compare.js +114 -0
  44. package/dist/commands/review-branch-compare.js.map +1 -0
  45. package/dist/commands/review-changelog-entry.d.ts +8 -0
  46. package/dist/commands/review-changelog-entry.d.ts.map +1 -0
  47. package/dist/commands/review-changelog-entry.js +110 -0
  48. package/dist/commands/review-changelog-entry.js.map +1 -0
  49. package/dist/commands/review-commit-hook.d.ts +8 -0
  50. package/dist/commands/review-commit-hook.d.ts.map +1 -0
  51. package/dist/commands/review-commit-hook.js +135 -0
  52. package/dist/commands/review-commit-hook.js.map +1 -0
  53. package/dist/commands/review-file-complexity.d.ts +5 -0
  54. package/dist/commands/review-file-complexity.d.ts.map +1 -0
  55. package/dist/commands/review-file-complexity.js +138 -0
  56. package/dist/commands/review-file-complexity.js.map +1 -0
  57. package/dist/commands/review-finding-link.d.ts +8 -0
  58. package/dist/commands/review-finding-link.d.ts.map +1 -0
  59. package/dist/commands/review-finding-link.js +116 -0
  60. package/dist/commands/review-finding-link.js.map +1 -0
  61. package/dist/commands/review-quota-check.d.ts +5 -0
  62. package/dist/commands/review-quota-check.d.ts.map +1 -0
  63. package/dist/commands/review-quota-check.js +98 -0
  64. package/dist/commands/review-quota-check.js.map +1 -0
  65. package/dist/commands/review-session-save.d.ts +5 -0
  66. package/dist/commands/review-session-save.d.ts.map +1 -0
  67. package/dist/commands/review-session-save.js +174 -0
  68. package/dist/commands/review-session-save.js.map +1 -0
  69. package/dist/commands/review-skip-list.d.ts +5 -0
  70. package/dist/commands/review-skip-list.d.ts.map +1 -0
  71. package/dist/commands/review-skip-list.js +136 -0
  72. package/dist/commands/review-skip-list.js.map +1 -0
  73. package/dist/commands/review-team-assign.d.ts +8 -0
  74. package/dist/commands/review-team-assign.d.ts.map +1 -0
  75. package/dist/commands/review-team-assign.js +212 -0
  76. package/dist/commands/review-team-assign.js.map +1 -0
  77. package/package.json +1 -1
  78. package/server.json +2 -2
@@ -0,0 +1,138 @@
1
+ /**
2
+ * Review-file-complexity — Analyze file complexity metrics.
3
+ */
4
+ import { readFileSync, existsSync } from "fs";
5
+ // ─── Helpers ────────────────────────────────────────────────────────────────
6
+ function analyzeComplexity(filePath) {
7
+ const content = readFileSync(filePath, "utf-8");
8
+ const lines = content.split("\n");
9
+ const totalLines = lines.length;
10
+ let codeLines = 0;
11
+ let blankLines = 0;
12
+ let commentLines = 0;
13
+ let functions = 0;
14
+ let maxIndentDepth = 0;
15
+ let totalLineLength = 0;
16
+ const funcPatterns = [/\bfunction\b/, /=>\s*[{(]/, /\bdef\b/, /\bfunc\b/, /\bfn\b/];
17
+ for (const line of lines) {
18
+ const trimmed = line.trim();
19
+ if (trimmed.length === 0) {
20
+ blankLines++;
21
+ continue;
22
+ }
23
+ if (trimmed.startsWith("//") || trimmed.startsWith("#") || trimmed.startsWith("*") || trimmed.startsWith("/*")) {
24
+ commentLines++;
25
+ continue;
26
+ }
27
+ codeLines++;
28
+ totalLineLength += trimmed.length;
29
+ // calculate indent depth
30
+ const indent = line.length - line.trimStart().length;
31
+ const depth = Math.floor(indent / 2);
32
+ if (depth > maxIndentDepth)
33
+ maxIndentDepth = depth;
34
+ // count functions
35
+ for (const p of funcPatterns) {
36
+ if (p.test(trimmed)) {
37
+ functions++;
38
+ break;
39
+ }
40
+ }
41
+ }
42
+ const avgLineLength = codeLines > 0 ? Math.round(totalLineLength / codeLines) : 0;
43
+ // Compute complexity score (0-100)
44
+ let score = 0;
45
+ if (totalLines > 300)
46
+ score += 15;
47
+ else if (totalLines > 150)
48
+ score += 8;
49
+ if (functions > 20)
50
+ score += 20;
51
+ else if (functions > 10)
52
+ score += 10;
53
+ if (maxIndentDepth > 8)
54
+ score += 25;
55
+ else if (maxIndentDepth > 5)
56
+ score += 15;
57
+ else if (maxIndentDepth > 3)
58
+ score += 5;
59
+ if (avgLineLength > 100)
60
+ score += 15;
61
+ else if (avgLineLength > 80)
62
+ score += 8;
63
+ if (commentLines === 0 && codeLines > 50)
64
+ score += 10;
65
+ if (codeLines > 0 && blankLines / codeLines < 0.05)
66
+ score += 5;
67
+ const risk = score >= 60 ? "critical" : score >= 40 ? "high" : score >= 20 ? "medium" : "low";
68
+ return {
69
+ filePath,
70
+ lines: totalLines,
71
+ codeLines,
72
+ blankLines,
73
+ commentLines,
74
+ functions,
75
+ maxIndentDepth,
76
+ avgLineLength,
77
+ complexityScore: score,
78
+ risk,
79
+ };
80
+ }
81
+ // ─── CLI ────────────────────────────────────────────────────────────────────
82
+ export function runReviewFileComplexity(argv) {
83
+ const formatIdx = argv.indexOf("--format");
84
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
85
+ if (argv.includes("--help") || argv.includes("-h")) {
86
+ console.log(`
87
+ judges review-file-complexity — Analyze file complexity
88
+
89
+ Usage:
90
+ judges review-file-complexity <file1> [file2 ...] [--format table|json]
91
+
92
+ Options:
93
+ --format <fmt> Output format: table (default), json
94
+ --help, -h Show this help
95
+ `);
96
+ return;
97
+ }
98
+ const files = argv.filter((a) => !a.startsWith("--") && (argv.indexOf(a) === 0 || argv[argv.indexOf(a) - 1] !== "--format"));
99
+ if (files.length === 0) {
100
+ console.error("Error: provide one or more file paths");
101
+ process.exitCode = 1;
102
+ return;
103
+ }
104
+ const results = [];
105
+ for (const f of files) {
106
+ if (!existsSync(f)) {
107
+ console.error(`Warning: not found: ${f}`);
108
+ continue;
109
+ }
110
+ try {
111
+ results.push(analyzeComplexity(f));
112
+ }
113
+ catch {
114
+ console.error(`Warning: cannot read: ${f}`);
115
+ }
116
+ }
117
+ if (results.length === 0) {
118
+ console.error("Error: no valid files");
119
+ process.exitCode = 1;
120
+ return;
121
+ }
122
+ if (format === "json") {
123
+ console.log(JSON.stringify(results, null, 2));
124
+ return;
125
+ }
126
+ console.log(`\nFile Complexity Analysis (${results.length} files)`);
127
+ console.log("═".repeat(80));
128
+ console.log(`${"File".padEnd(35)} ${"Lines".padEnd(7)} ${"Code".padEnd(7)} ${"Funcs".padEnd(7)} ${"Depth".padEnd(7)} ${"Score".padEnd(7)} Risk`);
129
+ console.log("─".repeat(80));
130
+ for (const r of results.sort((a, b) => b.complexityScore - a.complexityScore)) {
131
+ const name = r.filePath.length > 33 ? "…" + r.filePath.slice(-32) : r.filePath;
132
+ console.log(`${name.padEnd(35)} ${String(r.lines).padEnd(7)} ${String(r.codeLines).padEnd(7)} ${String(r.functions).padEnd(7)} ${String(r.maxIndentDepth).padEnd(7)} ${String(r.complexityScore).padEnd(7)} ${r.risk}`);
133
+ }
134
+ console.log("═".repeat(80));
135
+ const avgScore = Math.round(results.reduce((s, r) => s + r.complexityScore, 0) / results.length);
136
+ console.log(`\nAverage complexity score: ${avgScore}`);
137
+ }
138
+ //# sourceMappingURL=review-file-complexity.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-file-complexity.js","sourceRoot":"","sources":["../../src/commands/review-file-complexity.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Review-finding-link — Link related findings together.
3
+ *
4
+ * Identifies findings that share common patterns, lines, or rules
5
+ * and links them together for holistic remediation.
6
+ */
7
+ export declare function runReviewFindingLink(argv: string[]): void;
8
+ //# sourceMappingURL=review-finding-link.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-finding-link.d.ts","sourceRoot":"","sources":["../../src/commands/review-finding-link.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAwEH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAmEzD"}
@@ -0,0 +1,116 @@
1
+ /**
2
+ * Review-finding-link — Link related findings together.
3
+ *
4
+ * Identifies findings that share common patterns, lines, or rules
5
+ * and links them together for holistic remediation.
6
+ */
7
+ import { readFileSync, existsSync } from "fs";
8
+ // ─── Helpers ────────────────────────────────────────────────────────────────
9
+ function findLinks(verdict) {
10
+ const links = [];
11
+ const findings = verdict.findings;
12
+ for (let i = 0; i < findings.length; i++) {
13
+ for (let j = i + 1; j < findings.length; j++) {
14
+ const a = findings[i];
15
+ const b = findings[j];
16
+ let strength = 0;
17
+ const linkTypes = [];
18
+ // Same rule
19
+ if (a.ruleId === b.ruleId) {
20
+ strength += 3;
21
+ linkTypes.push("same-rule");
22
+ }
23
+ // Shared line numbers
24
+ if (a.lineNumbers && b.lineNumbers) {
25
+ const shared = a.lineNumbers.filter((ln) => b.lineNumbers.includes(ln));
26
+ if (shared.length > 0) {
27
+ strength += 2;
28
+ linkTypes.push("shared-lines");
29
+ }
30
+ }
31
+ // Similar titles (word overlap)
32
+ const wordsA = new Set(a.title.toLowerCase().split(/\s+/));
33
+ const wordsB = new Set(b.title.toLowerCase().split(/\s+/));
34
+ const commonWords = [...wordsA].filter((w) => wordsB.has(w) && w.length > 3);
35
+ if (commonWords.length >= 2) {
36
+ strength += 1;
37
+ linkTypes.push("similar-title");
38
+ }
39
+ // Same severity
40
+ if ((a.severity || "medium") === (b.severity || "medium")) {
41
+ strength += 0.5;
42
+ }
43
+ if (strength >= 2) {
44
+ links.push({
45
+ findingA: { ruleId: a.ruleId, title: a.title },
46
+ findingB: { ruleId: b.ruleId, title: b.title },
47
+ linkType: linkTypes.join(", "),
48
+ strength,
49
+ });
50
+ }
51
+ }
52
+ }
53
+ return links.sort((a, b) => b.strength - a.strength);
54
+ }
55
+ // ─── CLI ────────────────────────────────────────────────────────────────────
56
+ export function runReviewFindingLink(argv) {
57
+ const fileIdx = argv.indexOf("--file");
58
+ const formatIdx = argv.indexOf("--format");
59
+ const minIdx = argv.indexOf("--min-strength");
60
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
61
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
62
+ const minStrength = minIdx >= 0 ? parseFloat(argv[minIdx + 1]) : 2;
63
+ if (argv.includes("--help") || argv.includes("-h")) {
64
+ console.log(`
65
+ judges review-finding-link — Link related findings
66
+
67
+ Usage:
68
+ judges review-finding-link --file <verdict.json> [--format table|json]
69
+ [--min-strength <n>]
70
+
71
+ Options:
72
+ --file <path> Path to verdict JSON file (required)
73
+ --format <fmt> Output format: table (default), json
74
+ --min-strength <n> Minimum link strength (default: 2)
75
+ --help, -h Show this help
76
+ `);
77
+ return;
78
+ }
79
+ if (!filePath) {
80
+ console.error("Error: --file required");
81
+ process.exitCode = 1;
82
+ return;
83
+ }
84
+ if (!existsSync(filePath)) {
85
+ console.error(`Error: not found: ${filePath}`);
86
+ process.exitCode = 1;
87
+ return;
88
+ }
89
+ let verdict;
90
+ try {
91
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
92
+ }
93
+ catch {
94
+ console.error("Error: invalid JSON");
95
+ process.exitCode = 1;
96
+ return;
97
+ }
98
+ const links = findLinks(verdict).filter((l) => l.strength >= minStrength);
99
+ if (format === "json") {
100
+ console.log(JSON.stringify(links, null, 2));
101
+ return;
102
+ }
103
+ if (links.length === 0) {
104
+ console.log("No linked findings detected.");
105
+ return;
106
+ }
107
+ console.log(`\nLinked Findings (${links.length} links)`);
108
+ console.log("═".repeat(70));
109
+ for (const l of links) {
110
+ console.log(`\n [Strength: ${l.strength}] ${l.linkType}`);
111
+ console.log(` A: ${l.findingA.title} (${l.findingA.ruleId})`);
112
+ console.log(` B: ${l.findingB.title} (${l.findingB.ruleId})`);
113
+ }
114
+ console.log("\n" + "═".repeat(70));
115
+ }
116
+ //# sourceMappingURL=review-finding-link.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-finding-link.js","sourceRoot":"","sources":["../../src/commands/review-finding-link.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-quota-check — Check review quotas and rate limits.
3
+ */
4
+ export declare function runReviewQuotaCheck(argv: string[]): void;
5
+ //# sourceMappingURL=review-quota-check.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-quota-check.d.ts","sourceRoot":"","sources":["../../src/commands/review-quota-check.ts"],"names":[],"mappings":"AAAA;;GAEG;AAqEH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAiDxD"}
@@ -0,0 +1,98 @@
1
+ /**
2
+ * Review-quota-check — Check review quotas and rate limits.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ // ─── Helpers ────────────────────────────────────────────────────────────────
6
+ function todayStr() {
7
+ return new Date().toISOString().slice(0, 10);
8
+ }
9
+ function checkQuota(dir, limit) {
10
+ const today = todayStr();
11
+ const recentReviews = [];
12
+ let filesReviewed = 0;
13
+ if (!existsSync(dir)) {
14
+ return {
15
+ directory: dir,
16
+ filesReviewed: 0,
17
+ dailyLimit: limit,
18
+ remainingToday: limit,
19
+ isOverLimit: false,
20
+ todayDate: today,
21
+ recentReviews,
22
+ };
23
+ }
24
+ const files = readdirSync(dir);
25
+ for (const f of files) {
26
+ if (!f.endsWith(".json"))
27
+ continue;
28
+ const full = dir.endsWith("/") || dir.endsWith("\\") ? dir + f : dir + "/" + f;
29
+ try {
30
+ const raw = readFileSync(full, "utf-8");
31
+ const data = JSON.parse(raw);
32
+ const ts = data.timestamp || data.date || "";
33
+ if (typeof ts === "string" && ts.startsWith(today)) {
34
+ filesReviewed++;
35
+ recentReviews.push(f);
36
+ }
37
+ }
38
+ catch {
39
+ // skip invalid
40
+ }
41
+ }
42
+ return {
43
+ directory: dir,
44
+ filesReviewed,
45
+ dailyLimit: limit,
46
+ remainingToday: Math.max(0, limit - filesReviewed),
47
+ isOverLimit: filesReviewed >= limit,
48
+ todayDate: today,
49
+ recentReviews,
50
+ };
51
+ }
52
+ // ─── CLI ────────────────────────────────────────────────────────────────────
53
+ export function runReviewQuotaCheck(argv) {
54
+ const dirIdx = argv.indexOf("--dir");
55
+ const limitIdx = argv.indexOf("--limit");
56
+ const formatIdx = argv.indexOf("--format");
57
+ const dir = dirIdx >= 0 ? argv[dirIdx + 1] : ".judges/verdicts";
58
+ const limit = limitIdx >= 0 ? parseInt(argv[limitIdx + 1], 10) : 100;
59
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
60
+ if (argv.includes("--help") || argv.includes("-h")) {
61
+ console.log(`
62
+ judges review-quota-check — Check review quotas and limits
63
+
64
+ Usage:
65
+ judges review-quota-check [--dir <path>] [--limit <n>] [--format table|json]
66
+
67
+ Options:
68
+ --dir <path> Verdict directory (default: .judges/verdicts)
69
+ --limit <n> Daily review limit (default: 100)
70
+ --format <fmt> Output format: table (default), json
71
+ --help, -h Show this help
72
+ `);
73
+ return;
74
+ }
75
+ const status = checkQuota(dir, limit);
76
+ if (format === "json") {
77
+ console.log(JSON.stringify(status, null, 2));
78
+ return;
79
+ }
80
+ console.log(`\nReview Quota Status (${status.todayDate})`);
81
+ console.log("═".repeat(50));
82
+ console.log(` Directory: ${status.directory}`);
83
+ console.log(` Reviews today: ${status.filesReviewed}`);
84
+ console.log(` Daily limit: ${status.dailyLimit}`);
85
+ console.log(` Remaining: ${status.remainingToday}`);
86
+ console.log(` Status: ${status.isOverLimit ? "OVER LIMIT" : "OK"}`);
87
+ console.log("═".repeat(50));
88
+ if (status.recentReviews.length > 0) {
89
+ console.log("\nRecent reviews today:");
90
+ for (const r of status.recentReviews.slice(0, 10)) {
91
+ console.log(` - ${r}`);
92
+ }
93
+ if (status.recentReviews.length > 10) {
94
+ console.log(` ... and ${status.recentReviews.length - 10} more`);
95
+ }
96
+ }
97
+ }
98
+ //# sourceMappingURL=review-quota-check.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-quota-check.js","sourceRoot":"","sources":["../../src/commands/review-quota-check.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-session-save — Save and restore review sessions.
3
+ */
4
+ export declare function runReviewSessionSave(argv: string[]): void;
5
+ //# sourceMappingURL=review-session-save.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-session-save.d.ts","sourceRoot":"","sources":["../../src/commands/review-session-save.ts"],"names":[],"mappings":"AAAA;;GAEG;AAmDH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAuJzD"}
@@ -0,0 +1,174 @@
1
+ /**
2
+ * Review-session-save — Save and restore review sessions.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
5
+ // ─── Helpers ────────────────────────────────────────────────────────────────
6
+ function sessionDir() {
7
+ return ".judges/sessions";
8
+ }
9
+ function sessionPath(id) {
10
+ return `${sessionDir()}/${id}.json`;
11
+ }
12
+ function generateId() {
13
+ const now = new Date();
14
+ return `session-${now.toISOString().slice(0, 10)}-${now.getTime().toString(36)}`;
15
+ }
16
+ function loadSession(id) {
17
+ const p = sessionPath(id);
18
+ if (!existsSync(p))
19
+ return null;
20
+ try {
21
+ return JSON.parse(readFileSync(p, "utf-8"));
22
+ }
23
+ catch {
24
+ return null;
25
+ }
26
+ }
27
+ function saveSession(session) {
28
+ const dir = sessionDir();
29
+ if (!existsSync(dir))
30
+ mkdirSync(dir, { recursive: true });
31
+ session.updatedAt = new Date().toISOString();
32
+ writeFileSync(sessionPath(session.id), JSON.stringify(session, null, 2));
33
+ }
34
+ // ─── CLI ────────────────────────────────────────────────────────────────────
35
+ export function runReviewSessionSave(argv) {
36
+ const sub = argv[0];
37
+ if (argv.includes("--help") || argv.includes("-h") || !sub) {
38
+ console.log(`
39
+ judges review-session-save — Save and restore review sessions
40
+
41
+ Usage:
42
+ judges review-session-save create [--note <text>]
43
+ judges review-session-save add --id <session-id> --file <verdict.json>
44
+ judges review-session-save show --id <session-id> [--format table|json]
45
+ judges review-session-save complete --id <session-id>
46
+ judges review-session-save pause --id <session-id>
47
+ judges review-session-save resume --id <session-id>
48
+
49
+ Subcommands:
50
+ create Create a new review session
51
+ add Add a verdict file to a session
52
+ show Show session details
53
+ complete Mark session as completed
54
+ pause Mark session as paused
55
+ resume Resume a paused session
56
+
57
+ Options:
58
+ --id <id> Session ID
59
+ --file <path> Verdict file to add
60
+ --note <text> Session note
61
+ --format <fmt> Output format: table (default), json
62
+ --help, -h Show this help
63
+ `);
64
+ return;
65
+ }
66
+ const idIdx = argv.indexOf("--id");
67
+ const fileIdx = argv.indexOf("--file");
68
+ const noteIdx = argv.indexOf("--note");
69
+ const formatIdx = argv.indexOf("--format");
70
+ const id = idIdx >= 0 ? argv[idIdx + 1] : undefined;
71
+ const filePath = fileIdx >= 0 ? argv[fileIdx + 1] : undefined;
72
+ const note = noteIdx >= 0 ? argv[noteIdx + 1] : "";
73
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
74
+ if (sub === "create") {
75
+ const session = {
76
+ id: generateId(),
77
+ createdAt: new Date().toISOString(),
78
+ updatedAt: new Date().toISOString(),
79
+ files: [],
80
+ verdictPaths: [],
81
+ notes: note,
82
+ status: "in-progress",
83
+ };
84
+ saveSession(session);
85
+ console.log(`Created session: ${session.id}`);
86
+ return;
87
+ }
88
+ if (!id) {
89
+ console.error("Error: --id required");
90
+ process.exitCode = 1;
91
+ return;
92
+ }
93
+ if (sub === "add") {
94
+ if (!filePath) {
95
+ console.error("Error: --file required");
96
+ process.exitCode = 1;
97
+ return;
98
+ }
99
+ if (!existsSync(filePath)) {
100
+ console.error(`Error: not found: ${filePath}`);
101
+ process.exitCode = 1;
102
+ return;
103
+ }
104
+ const session = loadSession(id);
105
+ if (!session) {
106
+ console.error(`Error: session not found: ${id}`);
107
+ process.exitCode = 1;
108
+ return;
109
+ }
110
+ let verdict;
111
+ try {
112
+ verdict = JSON.parse(readFileSync(filePath, "utf-8"));
113
+ }
114
+ catch {
115
+ console.error("Error: invalid JSON");
116
+ process.exitCode = 1;
117
+ return;
118
+ }
119
+ session.verdictPaths.push(filePath);
120
+ session.files.push(`${filePath} (${verdict.findings.length} findings)`);
121
+ saveSession(session);
122
+ console.log(`Added ${filePath} to session ${id}`);
123
+ return;
124
+ }
125
+ if (sub === "show") {
126
+ const session = loadSession(id);
127
+ if (!session) {
128
+ console.error(`Error: session not found: ${id}`);
129
+ process.exitCode = 1;
130
+ return;
131
+ }
132
+ if (format === "json") {
133
+ console.log(JSON.stringify(session, null, 2));
134
+ return;
135
+ }
136
+ console.log(`\nReview Session: ${session.id}`);
137
+ console.log("═".repeat(50));
138
+ console.log(` Status: ${session.status}`);
139
+ console.log(` Created: ${session.createdAt}`);
140
+ console.log(` Updated: ${session.updatedAt}`);
141
+ console.log(` Verdicts: ${session.verdictPaths.length}`);
142
+ if (session.notes)
143
+ console.log(` Notes: ${session.notes}`);
144
+ console.log("═".repeat(50));
145
+ if (session.files.length > 0) {
146
+ console.log("\nFiles:");
147
+ for (const f of session.files) {
148
+ console.log(` - ${f}`);
149
+ }
150
+ }
151
+ return;
152
+ }
153
+ const statusMap = {
154
+ complete: "completed",
155
+ pause: "paused",
156
+ resume: "in-progress",
157
+ };
158
+ const newStatus = statusMap[sub];
159
+ if (newStatus) {
160
+ const session = loadSession(id);
161
+ if (!session) {
162
+ console.error(`Error: session not found: ${id}`);
163
+ process.exitCode = 1;
164
+ return;
165
+ }
166
+ session.status = newStatus;
167
+ saveSession(session);
168
+ console.log(`Session ${id} marked as ${newStatus}`);
169
+ return;
170
+ }
171
+ console.error(`Error: unknown subcommand: ${sub}`);
172
+ process.exitCode = 1;
173
+ }
174
+ //# sourceMappingURL=review-session-save.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-session-save.js","sourceRoot":"","sources":["../../src/commands/review-session-save.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-skip-list — Manage a list of files to skip during review.
3
+ */
4
+ export declare function runReviewSkipList(argv: string[]): void;
5
+ //# sourceMappingURL=review-skip-list.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-skip-list.d.ts","sourceRoot":"","sources":["../../src/commands/review-skip-list.ts"],"names":[],"mappings":"AAAA;;GAEG;AA+CH,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4FtD"}