@kevinrabun/judges 3.94.0 → 3.95.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-correlation-map.d.ts +5 -0
  6. package/dist/commands/finding-correlation-map.d.ts.map +1 -0
  7. package/dist/commands/finding-correlation-map.js +102 -0
  8. package/dist/commands/finding-correlation-map.js.map +1 -0
  9. package/dist/commands/finding-dedup-cross.d.ts +5 -0
  10. package/dist/commands/finding-dedup-cross.d.ts.map +1 -0
  11. package/dist/commands/finding-dedup-cross.js +91 -0
  12. package/dist/commands/finding-dedup-cross.js.map +1 -0
  13. package/dist/commands/finding-groupby-file.d.ts +5 -0
  14. package/dist/commands/finding-groupby-file.d.ts.map +1 -0
  15. package/dist/commands/finding-groupby-file.js +95 -0
  16. package/dist/commands/finding-groupby-file.js.map +1 -0
  17. package/dist/commands/review-api-export.d.ts +5 -0
  18. package/dist/commands/review-api-export.d.ts.map +1 -0
  19. package/dist/commands/review-api-export.js +99 -0
  20. package/dist/commands/review-api-export.js.map +1 -0
  21. package/dist/commands/review-merge-request.d.ts +5 -0
  22. package/dist/commands/review-merge-request.d.ts.map +1 -0
  23. package/dist/commands/review-merge-request.js +96 -0
  24. package/dist/commands/review-merge-request.js.map +1 -0
  25. package/dist/commands/review-notification-config.d.ts +5 -0
  26. package/dist/commands/review-notification-config.d.ts.map +1 -0
  27. package/dist/commands/review-notification-config.js +123 -0
  28. package/dist/commands/review-notification-config.js.map +1 -0
  29. package/dist/commands/review-scope-select.d.ts +5 -0
  30. package/dist/commands/review-scope-select.d.ts.map +1 -0
  31. package/dist/commands/review-scope-select.js +99 -0
  32. package/dist/commands/review-scope-select.js.map +1 -0
  33. package/dist/commands/review-summary-dashboard.d.ts +5 -0
  34. package/dist/commands/review-summary-dashboard.d.ts.map +1 -0
  35. package/dist/commands/review-summary-dashboard.js +97 -0
  36. package/dist/commands/review-summary-dashboard.js.map +1 -0
  37. package/dist/commands/review-template-library.d.ts +5 -0
  38. package/dist/commands/review-template-library.d.ts.map +1 -0
  39. package/dist/commands/review-template-library.js +156 -0
  40. package/dist/commands/review-template-library.js.map +1 -0
  41. package/package.json +1 -1
  42. package/server.json +2 -2
@@ -0,0 +1,123 @@
1
+ /**
2
+ * Review-notification-config — Configure notification preferences for review results.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ // ─── CLI ────────────────────────────────────────────────────────────────────
6
+ export function runReviewNotificationConfig(argv) {
7
+ const fileIdx = argv.indexOf("--file");
8
+ const enableIdx = argv.indexOf("--enable");
9
+ const disableIdx = argv.indexOf("--disable");
10
+ const severityIdx = argv.indexOf("--severity");
11
+ const digestIdx = argv.indexOf("--digest");
12
+ const formatIdx = argv.indexOf("--format");
13
+ const configPath = fileIdx >= 0 ? argv[fileIdx + 1] : ".judges-notifications.json";
14
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
15
+ if (argv.includes("--help") || argv.includes("-h")) {
16
+ console.log(`
17
+ judges review-notification-config — Configure notification preferences
18
+
19
+ Usage:
20
+ judges review-notification-config [--file <config>] [--enable <channel>]
21
+ [--disable <channel>] [--severity <level>]
22
+ [--digest daily|weekly] [--format table|json]
23
+
24
+ Options:
25
+ --file <path> Config file (default: .judges-notifications.json)
26
+ --enable <channel> Enable: console, file, slack, email
27
+ --disable <channel> Disable a channel
28
+ --severity <level> Set min severity for notifications
29
+ --digest <freq> Enable digest: daily, weekly
30
+ --format <fmt> Output format: table (default), json
31
+ --help, -h Show this help
32
+ `);
33
+ return;
34
+ }
35
+ // Load or init config
36
+ let config;
37
+ if (existsSync(configPath)) {
38
+ config = JSON.parse(readFileSync(configPath, "utf-8"));
39
+ }
40
+ else {
41
+ config = {
42
+ channels: [
43
+ { type: "console", enabled: true, minSeverity: "medium" },
44
+ { type: "file", enabled: false, minSeverity: "low" },
45
+ { type: "slack", enabled: false, minSeverity: "high" },
46
+ { type: "email", enabled: false, minSeverity: "critical" },
47
+ ],
48
+ digest: false,
49
+ digestFrequency: "daily",
50
+ };
51
+ }
52
+ let changed = false;
53
+ // Enable channel
54
+ if (enableIdx >= 0) {
55
+ const channel = argv[enableIdx + 1];
56
+ const ch = config.channels.find((c) => c.type === channel);
57
+ if (ch) {
58
+ ch.enabled = true;
59
+ changed = true;
60
+ console.log(`Enabled ${channel} notifications.`);
61
+ }
62
+ else {
63
+ config.channels.push({ type: channel, enabled: true, minSeverity: "medium" });
64
+ changed = true;
65
+ console.log(`Added and enabled ${channel} notifications.`);
66
+ }
67
+ }
68
+ // Disable channel
69
+ if (disableIdx >= 0) {
70
+ const channel = argv[disableIdx + 1];
71
+ const ch = config.channels.find((c) => c.type === channel);
72
+ if (ch) {
73
+ ch.enabled = false;
74
+ changed = true;
75
+ console.log(`Disabled ${channel} notifications.`);
76
+ }
77
+ else {
78
+ console.error(`Channel not found: ${channel}`);
79
+ }
80
+ }
81
+ // Set severity
82
+ if (severityIdx >= 0) {
83
+ const sev = argv[severityIdx + 1];
84
+ for (const ch of config.channels) {
85
+ if (ch.enabled) {
86
+ ch.minSeverity = sev;
87
+ }
88
+ }
89
+ changed = true;
90
+ console.log(`Set min severity to ${sev} for enabled channels.`);
91
+ }
92
+ // Digest
93
+ if (digestIdx >= 0) {
94
+ const freq = argv[digestIdx + 1];
95
+ config.digest = true;
96
+ config.digestFrequency = freq;
97
+ changed = true;
98
+ console.log(`Enabled ${freq} digest.`);
99
+ }
100
+ if (changed) {
101
+ writeFileSync(configPath, JSON.stringify(config, null, 2));
102
+ console.log(`Config saved to ${configPath}`);
103
+ return;
104
+ }
105
+ // Display current config
106
+ if (format === "json") {
107
+ console.log(JSON.stringify(config, null, 2));
108
+ return;
109
+ }
110
+ console.log(`\nNotification Configuration`);
111
+ console.log("═".repeat(55));
112
+ console.log(` Digest: ${config.digest ? `enabled (${config.digestFrequency})` : "disabled"}`);
113
+ console.log(`\n Channels:`);
114
+ for (const ch of config.channels) {
115
+ const status = ch.enabled ? "ON " : "OFF";
116
+ console.log(` [${status}] ${ch.type.padEnd(10)} min: ${ch.minSeverity}`);
117
+ }
118
+ if (config.quietHours) {
119
+ console.log(`\n Quiet Hours: ${config.quietHours.start} – ${config.quietHours.end}`);
120
+ }
121
+ console.log("═".repeat(55));
122
+ }
123
+ //# sourceMappingURL=review-notification-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-notification-config.js","sourceRoot":"","sources":["../../src/commands/review-notification-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-scope-select — Select review scope by path patterns and exclusions.
3
+ */
4
+ export declare function runReviewScopeSelect(argv: string[]): void;
5
+ //# sourceMappingURL=review-scope-select.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-scope-select.d.ts","sourceRoot":"","sources":["../../src/commands/review-scope-select.ts"],"names":[],"mappings":"AAAA;;GAEG;AAoCH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4EzD"}
@@ -0,0 +1,99 @@
1
+ /**
2
+ * Review-scope-select — Select review scope by path patterns and exclusions.
3
+ */
4
+ import { existsSync, readdirSync, statSync } from "fs";
5
+ import { join, relative } from "path";
6
+ // ─── Helpers ────────────────────────────────────────────────────────────────
7
+ function collectFiles(dir, extensions) {
8
+ const results = [];
9
+ function walk(d) {
10
+ const entries = readdirSync(d);
11
+ for (const entry of entries) {
12
+ if (typeof entry !== "string")
13
+ continue;
14
+ const full = join(d, entry);
15
+ try {
16
+ const stat = statSync(full);
17
+ if (stat.isDirectory()) {
18
+ if (entry !== "node_modules" && entry !== ".git" && entry !== "dist") {
19
+ walk(full);
20
+ }
21
+ }
22
+ else if (extensions.some((ext) => entry.endsWith(ext))) {
23
+ results.push(full);
24
+ }
25
+ }
26
+ catch {
27
+ // skip inaccessible
28
+ }
29
+ }
30
+ }
31
+ walk(dir);
32
+ return results;
33
+ }
34
+ // ─── CLI ────────────────────────────────────────────────────────────────────
35
+ export function runReviewScopeSelect(argv) {
36
+ const dirIdx = argv.indexOf("--dir");
37
+ const extIdx = argv.indexOf("--ext");
38
+ const excludeIdx = argv.indexOf("--exclude");
39
+ const formatIdx = argv.indexOf("--format");
40
+ const limitIdx = argv.indexOf("--limit");
41
+ const projectDir = dirIdx >= 0 ? argv[dirIdx + 1] : process.cwd();
42
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
43
+ if (argv.includes("--help") || argv.includes("-h")) {
44
+ console.log(`
45
+ judges review-scope-select — Select review scope by path/extension
46
+
47
+ Usage:
48
+ judges review-scope-select [--dir <path>] [--ext .ts,.js,...]
49
+ [--exclude <pattern>] [--limit <n>]
50
+ [--format table|json]
51
+
52
+ Options:
53
+ --dir <path> Project directory (default: cwd)
54
+ --ext <extensions> Comma-separated file extensions (default: .ts,.js,.py,.java,.go)
55
+ --exclude <pattern> Exclude paths containing pattern
56
+ --limit <n> Maximum files to list
57
+ --format <fmt> Output format: table (default), json
58
+ --help, -h Show this help
59
+ `);
60
+ return;
61
+ }
62
+ if (!existsSync(projectDir)) {
63
+ console.error(`Error: directory not found: ${projectDir}`);
64
+ process.exitCode = 1;
65
+ return;
66
+ }
67
+ const extensions = extIdx >= 0 ? argv[extIdx + 1].split(",") : [".ts", ".js", ".py", ".java", ".go"];
68
+ const excludePattern = excludeIdx >= 0 ? argv[excludeIdx + 1] : undefined;
69
+ const limit = limitIdx >= 0 ? parseInt(argv[limitIdx + 1], 10) : undefined;
70
+ let files = collectFiles(projectDir, extensions);
71
+ if (excludePattern) {
72
+ files = files.filter((f) => !f.includes(excludePattern));
73
+ }
74
+ if (limit !== undefined && limit > 0) {
75
+ files = files.slice(0, limit);
76
+ }
77
+ const relativePaths = files.map((f) => relative(projectDir, f));
78
+ if (format === "json") {
79
+ console.log(JSON.stringify({ count: relativePaths.length, files: relativePaths }, null, 2));
80
+ return;
81
+ }
82
+ console.log(`\nReview Scope: ${relativePaths.length} file(s)`);
83
+ console.log("═".repeat(55));
84
+ const byExt = {};
85
+ for (const f of relativePaths) {
86
+ const ext = f.substring(f.lastIndexOf("."));
87
+ byExt[ext] = (byExt[ext] ?? 0) + 1;
88
+ }
89
+ console.log(" Extensions:");
90
+ for (const [ext, count] of Object.entries(byExt)) {
91
+ console.log(` ${ext.padEnd(8)} ${count} file(s)`);
92
+ }
93
+ console.log("\n Files:");
94
+ for (const f of relativePaths) {
95
+ console.log(` ${f}`);
96
+ }
97
+ console.log("═".repeat(55));
98
+ }
99
+ //# sourceMappingURL=review-scope-select.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-scope-select.js","sourceRoot":"","sources":["../../src/commands/review-scope-select.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-summary-dashboard — Aggregate review dashboard with key metrics.
3
+ */
4
+ export declare function runReviewSummaryDashboard(argv: string[]): void;
5
+ //# sourceMappingURL=review-summary-dashboard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-summary-dashboard.d.ts","sourceRoot":"","sources":["../../src/commands/review-summary-dashboard.ts"],"names":[],"mappings":"AAAA;;GAEG;AAmBH,wBAAgB,yBAAyB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAsG9D"}
@@ -0,0 +1,97 @@
1
+ /**
2
+ * Review-summary-dashboard — Aggregate review dashboard with key metrics.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ import { join } from "path";
6
+ // ─── CLI ────────────────────────────────────────────────────────────────────
7
+ export function runReviewSummaryDashboard(argv) {
8
+ const dirIdx = argv.indexOf("--dir");
9
+ const formatIdx = argv.indexOf("--format");
10
+ const reviewDir = dirIdx >= 0 ? argv[dirIdx + 1] : process.cwd();
11
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
12
+ if (argv.includes("--help") || argv.includes("-h")) {
13
+ console.log(`
14
+ judges review-summary-dashboard — Aggregate review dashboard
15
+
16
+ Usage:
17
+ judges review-summary-dashboard [--dir <path>] [--format table|json]
18
+
19
+ Options:
20
+ --dir <path> Directory with review JSON files (default: cwd)
21
+ --format <fmt> Output format: table (default), json
22
+ --help, -h Show this help
23
+ `);
24
+ return;
25
+ }
26
+ if (!existsSync(reviewDir)) {
27
+ console.error(`Error: directory not found: ${reviewDir}`);
28
+ process.exitCode = 1;
29
+ return;
30
+ }
31
+ const files = readdirSync(reviewDir).filter((f) => typeof f === "string" && f.endsWith(".json"));
32
+ const verdicts = [];
33
+ for (const file of files) {
34
+ try {
35
+ const v = JSON.parse(readFileSync(join(reviewDir, file), "utf-8"));
36
+ if (v.overallVerdict !== undefined && v.findings !== undefined) {
37
+ verdicts.push(v);
38
+ }
39
+ }
40
+ catch {
41
+ // skip non-review files
42
+ }
43
+ }
44
+ if (verdicts.length === 0) {
45
+ console.log("No review files found. Run reviews and save as JSON first.");
46
+ return;
47
+ }
48
+ const dashboard = {
49
+ totalReviews: verdicts.length,
50
+ totalFindings: 0,
51
+ verdictBreakdown: {},
52
+ severityBreakdown: {},
53
+ averageScore: 0,
54
+ topRules: [],
55
+ };
56
+ let scoreSum = 0;
57
+ const ruleCounts = {};
58
+ for (const v of verdicts) {
59
+ scoreSum += v.overallScore;
60
+ dashboard.verdictBreakdown[v.overallVerdict] = (dashboard.verdictBreakdown[v.overallVerdict] ?? 0) + 1;
61
+ for (const f of v.findings) {
62
+ dashboard.totalFindings++;
63
+ dashboard.severityBreakdown[f.severity] = (dashboard.severityBreakdown[f.severity] ?? 0) + 1;
64
+ ruleCounts[f.ruleId] = (ruleCounts[f.ruleId] ?? 0) + 1;
65
+ }
66
+ }
67
+ dashboard.averageScore = Math.round((scoreSum / verdicts.length) * 100) / 100;
68
+ dashboard.topRules = Object.entries(ruleCounts)
69
+ .sort((a, b) => b[1] - a[1])
70
+ .slice(0, 10)
71
+ .map(([ruleId, count]) => ({ ruleId, count }));
72
+ if (format === "json") {
73
+ console.log(JSON.stringify(dashboard, null, 2));
74
+ return;
75
+ }
76
+ console.log(`\nReview Dashboard`);
77
+ console.log("═".repeat(55));
78
+ console.log(` Reviews: ${dashboard.totalReviews}`);
79
+ console.log(` Findings: ${dashboard.totalFindings}`);
80
+ console.log(` Avg Score: ${dashboard.averageScore}`);
81
+ console.log(`\n Verdict Breakdown:`);
82
+ for (const [v, c] of Object.entries(dashboard.verdictBreakdown)) {
83
+ console.log(` ${v.padEnd(12)} ${c}`);
84
+ }
85
+ console.log(`\n Severity Breakdown:`);
86
+ for (const [s, c] of Object.entries(dashboard.severityBreakdown)) {
87
+ console.log(` ${s.padEnd(12)} ${c}`);
88
+ }
89
+ if (dashboard.topRules.length > 0) {
90
+ console.log(`\n Top Rules:`);
91
+ for (const r of dashboard.topRules) {
92
+ console.log(` ${r.ruleId.padEnd(25)} ${r.count}`);
93
+ }
94
+ }
95
+ console.log("═".repeat(55));
96
+ }
97
+ //# sourceMappingURL=review-summary-dashboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-summary-dashboard.js","sourceRoot":"","sources":["../../src/commands/review-summary-dashboard.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-template-library — Library of reusable review templates.
3
+ */
4
+ export declare function runReviewTemplateLibrary(argv: string[]): void;
5
+ //# sourceMappingURL=review-template-library.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-template-library.d.ts","sourceRoot":"","sources":["../../src/commands/review-template-library.ts"],"names":[],"mappings":"AAAA;;GAEG;AAiEH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAiH7D"}
@@ -0,0 +1,156 @@
1
+ /**
2
+ * Review-template-library — Library of reusable review templates.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync, readdirSync } from "fs";
5
+ import { join } from "path";
6
+ // ─── Built-in Templates ─────────────────────────────────────────────────────
7
+ function getBuiltinTemplates() {
8
+ return [
9
+ {
10
+ name: "ai-code-review",
11
+ description: "Optimized for reviewing AI-generated code",
12
+ preset: "strict",
13
+ minSeverity: "medium",
14
+ disabledRules: [],
15
+ tags: ["ai", "code-generation", "copilot"],
16
+ },
17
+ {
18
+ name: "security-audit",
19
+ description: "Deep security review with all severity levels",
20
+ preset: "security-focused",
21
+ minSeverity: "low",
22
+ disabledRules: [],
23
+ tags: ["security", "audit", "compliance"],
24
+ },
25
+ {
26
+ name: "quick-scan",
27
+ description: "Fast scan for critical issues only",
28
+ preset: "default",
29
+ minSeverity: "critical",
30
+ disabledRules: [],
31
+ tags: ["quick", "ci", "fast"],
32
+ },
33
+ {
34
+ name: "pr-review",
35
+ description: "Balanced review for pull requests",
36
+ preset: "default",
37
+ minSeverity: "medium",
38
+ disabledRules: [],
39
+ tags: ["pr", "pull-request", "review"],
40
+ },
41
+ {
42
+ name: "onboarding",
43
+ description: "Gentle review for new team members",
44
+ preset: "default",
45
+ minSeverity: "high",
46
+ disabledRules: [],
47
+ tags: ["onboarding", "learning", "gentle"],
48
+ },
49
+ ];
50
+ }
51
+ // ─── CLI ────────────────────────────────────────────────────────────────────
52
+ export function runReviewTemplateLibrary(argv) {
53
+ const dirIdx = argv.indexOf("--dir");
54
+ const nameIdx = argv.indexOf("--name");
55
+ const exportIdx = argv.indexOf("--export");
56
+ const importIdx = argv.indexOf("--import");
57
+ const tagIdx = argv.indexOf("--tag");
58
+ const formatIdx = argv.indexOf("--format");
59
+ const templateDir = dirIdx >= 0 ? argv[dirIdx + 1] : join(process.cwd(), ".judges-templates");
60
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
61
+ if (argv.includes("--help") || argv.includes("-h")) {
62
+ console.log(`
63
+ judges review-template-library — Reusable review templates
64
+
65
+ Usage:
66
+ judges review-template-library [--dir <path>] [--name <template>]
67
+ [--tag <tag>] [--export <file>]
68
+ [--import <file>] [--format table|json]
69
+
70
+ Options:
71
+ --dir <path> Template directory (default: .judges-templates/)
72
+ --name <template> Show or use specific template
73
+ --tag <tag> Filter by tag
74
+ --export <file> Export template to file
75
+ --import <file> Import template from file
76
+ --format <fmt> Output format: table (default), json
77
+ --help, -h Show this help
78
+ `);
79
+ return;
80
+ }
81
+ // Import a template
82
+ if (importIdx >= 0) {
83
+ const importPath = argv[importIdx + 1];
84
+ if (!existsSync(importPath)) {
85
+ console.error(`Error: file not found: ${importPath}`);
86
+ process.exitCode = 1;
87
+ return;
88
+ }
89
+ try {
90
+ const template = JSON.parse(readFileSync(importPath, "utf-8"));
91
+ const outPath = join(templateDir, `${template.name}.json`);
92
+ writeFileSync(outPath, JSON.stringify(template, null, 2));
93
+ console.log(`Imported template "${template.name}" to ${outPath}`);
94
+ }
95
+ catch {
96
+ console.error(`Error: failed to parse template file: ${importPath}`);
97
+ process.exitCode = 1;
98
+ }
99
+ return;
100
+ }
101
+ // Collect all templates (builtin + user)
102
+ let templates = getBuiltinTemplates();
103
+ if (existsSync(templateDir)) {
104
+ const files = readdirSync(templateDir);
105
+ for (const file of files) {
106
+ if (typeof file === "string" && file.endsWith(".json")) {
107
+ try {
108
+ const t = JSON.parse(readFileSync(join(templateDir, file), "utf-8"));
109
+ if (t.name !== undefined) {
110
+ templates = templates.filter((bt) => bt.name !== t.name);
111
+ templates.push(t);
112
+ }
113
+ }
114
+ catch {
115
+ // skip invalid
116
+ }
117
+ }
118
+ }
119
+ }
120
+ // Filter by tag
121
+ if (tagIdx >= 0) {
122
+ const tag = argv[tagIdx + 1];
123
+ templates = templates.filter((t) => t.tags.includes(tag));
124
+ }
125
+ // Show specific template
126
+ const templateName = nameIdx >= 0 ? argv[nameIdx + 1] : undefined;
127
+ if (templateName) {
128
+ const template = templates.find((t) => t.name === templateName);
129
+ if (!template) {
130
+ console.error(`Error: template not found: ${templateName}`);
131
+ process.exitCode = 1;
132
+ return;
133
+ }
134
+ if (exportIdx >= 0) {
135
+ const exportPath = argv[exportIdx + 1];
136
+ writeFileSync(exportPath, JSON.stringify(template, null, 2));
137
+ console.log(`Exported "${template.name}" to ${exportPath}`);
138
+ return;
139
+ }
140
+ console.log(JSON.stringify(template, null, 2));
141
+ return;
142
+ }
143
+ // List all templates
144
+ if (format === "json") {
145
+ console.log(JSON.stringify(templates, null, 2));
146
+ return;
147
+ }
148
+ console.log(`\nReview Template Library: ${templates.length} template(s)`);
149
+ console.log("═".repeat(65));
150
+ for (const t of templates) {
151
+ console.log(` ${t.name.padEnd(20)} ${t.description}`);
152
+ console.log(` ${"".padEnd(20)} preset: ${t.preset} minSeverity: ${t.minSeverity} tags: ${t.tags.join(", ")}`);
153
+ }
154
+ console.log("═".repeat(65));
155
+ }
156
+ //# sourceMappingURL=review-template-library.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-template-library.js","sourceRoot":"","sources":["../../src/commands/review-template-library.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,IAAI,CAAC;AAC1E,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAa5B,+EAA+E;AAE/E,SAAS,mBAAmB;IAC1B,OAAO;QACL;YACE,IAAI,EAAE,gBAAgB;YACtB,WAAW,EAAE,2CAA2C;YACxD,MAAM,EAAE,QAAQ;YAChB,WAAW,EAAE,QAAQ;YACrB,aAAa,EAAE,EAAE;YACjB,IAAI,EAAE,CAAC,IAAI,EAAE,iBAAiB,EAAE,SAAS,CAAC;SAC3C;QACD;YACE,IAAI,EAAE,gBAAgB;YACtB,WAAW,EAAE,+CAA+C;YAC5D,MAAM,EAAE,kBAAkB;YAC1B,WAAW,EAAE,KAAK;YAClB,aAAa,EAAE,EAAE;YACjB,IAAI,EAAE,CAAC,UAAU,EAAE,OAAO,EAAE,YAAY,CAAC;SAC1C;QACD;YACE,IAAI,EAAE,YAAY;YAClB,WAAW,EAAE,oCAAoC;YACjD,MAAM,EAAE,SAAS;YACjB,WAAW,EAAE,UAAU;YACvB,aAAa,EAAE,EAAE;YACjB,IAAI,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,MAAM,CAAC;SAC9B;QACD;YACE,IAAI,EAAE,WAAW;YACjB,WAAW,EAAE,mCAAmC;YAChD,MAAM,EAAE,SAAS;YACjB,WAAW,EAAE,QAAQ;YACrB,aAAa,EAAE,EAAE;YACjB,IAAI,EAAE,CAAC,IAAI,EAAE,cAAc,EAAE,QAAQ,CAAC;SACvC;QACD;YACE,IAAI,EAAE,YAAY;YAClB,WAAW,EAAE,oCAAoC;YACjD,MAAM,EAAE,SAAS;YACjB,WAAW,EAAE,MAAM;YACnB,aAAa,EAAE,EAAE;YACjB,IAAI,EAAE,CAAC,YAAY,EAAE,UAAU,EAAE,QAAQ,CAAC;SAC3C;KACF,CAAC;AACJ,CAAC;AAED,+EAA+E;AAE/E,MAAM,UAAU,wBAAwB,CAAC,IAAc;IACrD,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACrC,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACvC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACrC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,mBAAmB,CAAC,CAAC;IAC9F,MAAM,MAAM,GAAG,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAE9D,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;;;;;;CAgBf,CAAC,CAAC;QACC,OAAO;IACT,CAAC;IAED,oBAAoB;IACpB,IAAI,SAAS,IAAI,CAAC,EAAE,CAAC;QACnB,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;QACvC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC5B,OAAO,CAAC,KAAK,CAAC,0BAA0B,UAAU,EAAE,CAAC,CAAC;YACtD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;QACD,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAmB,CAAC;YACjF,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,EAAE,GAAG,QAAQ,CAAC,IAAI,OAAO,CAAC,CAAC;YAC3D,aAAa,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC1D,OAAO,CAAC,GAAG,CAAC,sBAAsB,QAAQ,CAAC,IAAI,QAAQ,OAAO,EAAE,CAAC,CAAC;QACpE,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,KAAK,CAAC,yCAAyC,UAAU,EAAE,CAAC,CAAC;YACrE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACvB,CAAC;QACD,OAAO;IACT,CAAC;IAED,yCAAyC;IACzC,IAAI,SAAS,GAAG,mBAAmB,EAAE,CAAC;IAEtC,IAAI,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5B,MAAM,KAAK,GAAG,WAAW,CAAC,WAAW,CAAwB,CAAC;QAC9D,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;gBACvD,IAAI,CAAC;oBACH,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,EAAE,OAAO,CAAC,CAAmB,CAAC;oBACvF,IAAI,CAAC,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;wBACzB,SAAS,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,IAAI,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC;wBACzD,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;oBACpB,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,eAAe;gBACjB,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,gBAAgB;IAChB,IAAI,MAAM,IAAI,CAAC,EAAE,CAAC;QAChB,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAC7B,SAAS,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED,yBAAyB;IACzB,MAAM,YAAY,GAAG,OAAO,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAClE,IAAI,YAAY,EAAE,CAAC;QACjB,MAAM,QAAQ,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,YAAY,CAAC,CAAC;QAChE,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO,CAAC,KAAK,CAAC,8BAA8B,YAAY,EAAE,CAAC,CAAC;YAC5D,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;QAED,IAAI,SAAS,IAAI,CAAC,EAAE,CAAC;YACnB,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;YACvC,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAC7D,OAAO,CAAC,GAAG,CAAC,aAAa,QAAQ,CAAC,IAAI,QAAQ,UAAU,EAAE,CAAC,CAAC;YAC5D,OAAO;QACT,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,OAAO;IACT,CAAC;IAED,qBAAqB;IACrB,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAChD,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,8BAA8B,SAAS,CAAC,MAAM,cAAc,CAAC,CAAC;IAC1E,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAE5B,KAAK,MAAM,CAAC,IAAI,SAAS,EAAE,CAAC;QAC1B,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;QACvD,OAAO,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC,MAAM,kBAAkB,CAAC,CAAC,WAAW,WAAW,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACnH,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;AAC9B,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kevinrabun/judges",
3
- "version": "3.94.0",
3
+ "version": "3.95.0",
4
4
  "description": "45 specialized judges that evaluate AI-generated code for security, cost, and quality.",
5
5
  "mcpName": "io.github.KevinRabun/judges",
6
6
  "type": "module",
package/server.json CHANGED
@@ -7,12 +7,12 @@
7
7
  "url": "https://github.com/kevinrabun/judges",
8
8
  "source": "github"
9
9
  },
10
- "version": "3.94.0",
10
+ "version": "3.95.0",
11
11
  "packages": [
12
12
  {
13
13
  "registryType": "npm",
14
14
  "identifier": "@kevinrabun/judges",
15
- "version": "3.94.0",
15
+ "version": "3.95.0",
16
16
  "transport": {
17
17
  "type": "stdio"
18
18
  }