@kevinrabun/judges 3.97.0 → 3.98.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-cwe-lookup.d.ts +5 -0
  6. package/dist/commands/finding-cwe-lookup.d.ts.map +1 -0
  7. package/dist/commands/finding-cwe-lookup.js +149 -0
  8. package/dist/commands/finding-cwe-lookup.js.map +1 -0
  9. package/dist/commands/finding-duplicate-detect.d.ts +5 -0
  10. package/dist/commands/finding-duplicate-detect.d.ts.map +1 -0
  11. package/dist/commands/finding-duplicate-detect.js +114 -0
  12. package/dist/commands/finding-duplicate-detect.js.map +1 -0
  13. package/dist/commands/finding-patch-preview.d.ts +5 -0
  14. package/dist/commands/finding-patch-preview.d.ts.map +1 -0
  15. package/dist/commands/finding-patch-preview.js +104 -0
  16. package/dist/commands/finding-patch-preview.js.map +1 -0
  17. package/dist/commands/finding-priority-matrix.d.ts +5 -0
  18. package/dist/commands/finding-priority-matrix.d.ts.map +1 -0
  19. package/dist/commands/finding-priority-matrix.js +103 -0
  20. package/dist/commands/finding-priority-matrix.js.map +1 -0
  21. package/dist/commands/review-cicd-integrate.d.ts +5 -0
  22. package/dist/commands/review-cicd-integrate.d.ts.map +1 -0
  23. package/dist/commands/review-cicd-integrate.js +123 -0
  24. package/dist/commands/review-cicd-integrate.js.map +1 -0
  25. package/dist/commands/review-language-profile.d.ts +5 -0
  26. package/dist/commands/review-language-profile.d.ts.map +1 -0
  27. package/dist/commands/review-language-profile.js +73 -0
  28. package/dist/commands/review-language-profile.js.map +1 -0
  29. package/dist/commands/review-org-dashboard.d.ts +5 -0
  30. package/dist/commands/review-org-dashboard.d.ts.map +1 -0
  31. package/dist/commands/review-org-dashboard.js +69 -0
  32. package/dist/commands/review-org-dashboard.js.map +1 -0
  33. package/dist/commands/review-report-archive.d.ts +5 -0
  34. package/dist/commands/review-report-archive.d.ts.map +1 -0
  35. package/dist/commands/review-report-archive.js +101 -0
  36. package/dist/commands/review-report-archive.js.map +1 -0
  37. package/dist/commands/review-sla-config.d.ts +5 -0
  38. package/dist/commands/review-sla-config.d.ts.map +1 -0
  39. package/dist/commands/review-sla-config.js +89 -0
  40. package/dist/commands/review-sla-config.js.map +1 -0
  41. package/package.json +1 -1
  42. package/server.json +2 -2
@@ -0,0 +1,123 @@
1
+ /**
2
+ * Review-cicd-integrate — Generate CI/CD integration configs for Judges.
3
+ */
4
+ import { writeFileSync } from "fs";
5
+ // ─── Templates ──────────────────────────────────────────────────────────────
6
+ const TEMPLATES = {
7
+ "github-actions": {
8
+ filename: ".github/workflows/judges.yml",
9
+ content: `name: Judges Code Review
10
+ on: [pull_request]
11
+ jobs:
12
+ judges-review:
13
+ runs-on: ubuntu-latest
14
+ steps:
15
+ - uses: actions/checkout@v4
16
+ - uses: actions/setup-node@v4
17
+ with:
18
+ node-version: '20'
19
+ - run: npx @kevinrabun/judges eval --file \${{ github.event.pull_request.head.sha }} --format sarif --fail-on-findings
20
+ `,
21
+ },
22
+ "gitlab-ci": {
23
+ filename: ".gitlab-ci.yml",
24
+ content: `judges-review:
25
+ stage: test
26
+ image: node:20
27
+ script:
28
+ - npx @kevinrabun/judges eval --format sarif --fail-on-findings
29
+ only:
30
+ - merge_requests
31
+ `,
32
+ },
33
+ "azure-pipelines": {
34
+ filename: "azure-pipelines-judges.yml",
35
+ content: `trigger:
36
+ - main
37
+ pool:
38
+ vmImage: 'ubuntu-latest'
39
+ steps:
40
+ - task: NodeTool@0
41
+ inputs:
42
+ versionSpec: '20.x'
43
+ - script: npx @kevinrabun/judges eval --format sarif --fail-on-findings
44
+ displayName: 'Run Judges Review'
45
+ `,
46
+ },
47
+ jenkins: {
48
+ filename: "Jenkinsfile-judges",
49
+ content: `pipeline {
50
+ agent any
51
+ stages {
52
+ stage('Judges Review') {
53
+ steps {
54
+ sh 'npx @kevinrabun/judges eval --format sarif --fail-on-findings'
55
+ }
56
+ }
57
+ }
58
+ }
59
+ `,
60
+ },
61
+ };
62
+ // ─── CLI ────────────────────────────────────────────────────────────────────
63
+ export function runReviewCicdIntegrate(argv) {
64
+ const platformIdx = argv.indexOf("--platform");
65
+ const outIdx = argv.indexOf("--out");
66
+ const formatIdx = argv.indexOf("--format");
67
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
68
+ if (argv.includes("--help") || argv.includes("-h")) {
69
+ console.log(`
70
+ judges review-cicd-integrate — Generate CI/CD integration configs
71
+
72
+ Usage:
73
+ judges review-cicd-integrate --platform <name> [--out <path>]
74
+ [--format table|json]
75
+
76
+ Options:
77
+ --platform <name> CI/CD platform: github-actions, gitlab-ci, azure-pipelines, jenkins
78
+ --out <path> Write config to file (default: print to stdout)
79
+ --format <fmt> Output format: table (default), json
80
+ --help, -h Show this help
81
+ `);
82
+ return;
83
+ }
84
+ // List platforms
85
+ if (platformIdx < 0) {
86
+ const platforms = Object.keys(TEMPLATES);
87
+ if (format === "json") {
88
+ console.log(JSON.stringify({ platforms }, null, 2));
89
+ }
90
+ else {
91
+ console.log(`\nAvailable CI/CD Platforms:`);
92
+ console.log("═".repeat(40));
93
+ for (const p of platforms) {
94
+ const t = TEMPLATES[p];
95
+ console.log(` ${p.padEnd(20)} → ${t.filename}`);
96
+ }
97
+ console.log("\nUse --platform <name> to generate config.");
98
+ console.log("═".repeat(40));
99
+ }
100
+ return;
101
+ }
102
+ const platform = argv[platformIdx + 1];
103
+ const template = TEMPLATES[platform];
104
+ if (template === undefined) {
105
+ console.error(`Unknown platform: ${platform}`);
106
+ console.error(`Available: ${Object.keys(TEMPLATES).join(", ")}`);
107
+ process.exitCode = 1;
108
+ return;
109
+ }
110
+ if (outIdx >= 0) {
111
+ const outPath = argv[outIdx + 1];
112
+ writeFileSync(outPath, template.content);
113
+ console.log(`Config written to: ${outPath}`);
114
+ return;
115
+ }
116
+ if (format === "json") {
117
+ console.log(JSON.stringify(template, null, 2));
118
+ return;
119
+ }
120
+ console.log(`\n--- ${template.filename} ---`);
121
+ console.log(template.content);
122
+ }
123
+ //# sourceMappingURL=review-cicd-integrate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-cicd-integrate.js","sourceRoot":"","sources":["../../src/commands/review-cicd-integrate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,IAAI,CAAC;AAEnC,+EAA+E;AAE/E,MAAM,SAAS,GAA0D;IACvE,gBAAgB,EAAE;QAChB,QAAQ,EAAE,8BAA8B;QACxC,OAAO,EAAE;;;;;;;;;;;CAWZ;KACE;IACD,WAAW,EAAE;QACX,QAAQ,EAAE,gBAAgB;QAC1B,OAAO,EAAE;;;;;;;CAOZ;KACE;IACD,iBAAiB,EAAE;QACjB,QAAQ,EAAE,4BAA4B;QACtC,OAAO,EAAE;;;;;;;;;;CAUZ;KACE;IACD,OAAO,EAAE;QACP,QAAQ,EAAE,oBAAoB;QAC9B,OAAO,EAAE;;;;;;;;;;CAUZ;KACE;CACF,CAAC;AAEF,+EAA+E;AAE/E,MAAM,UAAU,sBAAsB,CAAC,IAAc;IACnD,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACrC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,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;;;;;;;;;;;;CAYf,CAAC,CAAC;QACC,OAAO;IACT,CAAC;IAED,iBAAiB;IACjB,IAAI,WAAW,GAAG,CAAC,EAAE,CAAC;QACpB,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACtD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAC;YAC5C,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5B,KAAK,MAAM,CAAC,IAAI,SAAS,EAAE,CAAC;gBAC1B,MAAM,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;gBACvB,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;YACnD,CAAC;YACD,OAAO,CAAC,GAAG,CAAC,6CAA6C,CAAC,CAAC;YAC3D,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9B,CAAC;QACD,OAAO;IACT,CAAC;IAED,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC;IACvC,MAAM,QAAQ,GAAG,SAAS,CAAC,QAAQ,CAAC,CAAC;IAErC,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;QAC3B,OAAO,CAAC,KAAK,CAAC,qBAAqB,QAAQ,EAAE,CAAC,CAAC;QAC/C,OAAO,CAAC,KAAK,CAAC,cAAc,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QACrB,OAAO;IACT,CAAC;IAED,IAAI,MAAM,IAAI,CAAC,EAAE,CAAC;QAChB,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QACjC,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,CAAC;QACzC,OAAO,CAAC,GAAG,CAAC,sBAAsB,OAAO,EAAE,CAAC,CAAC;QAC7C,OAAO;IACT,CAAC;IAED,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,SAAS,QAAQ,CAAC,QAAQ,MAAM,CAAC,CAAC;IAC9C,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;AAChC,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-language-profile — Analyze review findings distribution by language.
3
+ */
4
+ export declare function runReviewLanguageProfile(argv: string[]): void;
5
+ //# sourceMappingURL=review-language-profile.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-language-profile.d.ts","sourceRoot":"","sources":["../../src/commands/review-language-profile.ts"],"names":[],"mappings":"AAAA;;GAEG;AAiBH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAoF7D"}
@@ -0,0 +1,73 @@
1
+ /**
2
+ * Review-language-profile — Analyze review findings distribution by language.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ import { join } from "path";
6
+ // ─── CLI ────────────────────────────────────────────────────────────────────
7
+ export function runReviewLanguageProfile(argv) {
8
+ const dirIdx = argv.indexOf("--dir");
9
+ const dir = dirIdx >= 0 ? argv[dirIdx + 1] : ".judges/reports";
10
+ const formatIdx = argv.indexOf("--format");
11
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
12
+ if (argv.includes("--help") || argv.includes("-h")) {
13
+ console.log(`
14
+ judges review-language-profile — Analyze findings by language
15
+
16
+ Usage:
17
+ judges review-language-profile [--dir <path>] [--format table|json]
18
+
19
+ Options:
20
+ --dir <path> Reports directory (default: .judges/reports)
21
+ --format <fmt> Output format: table (default), json
22
+ --help, -h Show this help
23
+ `);
24
+ return;
25
+ }
26
+ if (!existsSync(dir)) {
27
+ console.log(`Reports directory not found: ${dir}`);
28
+ console.log("Run reviews first to generate report data.");
29
+ return;
30
+ }
31
+ const files = readdirSync(dir).filter((f) => f.endsWith(".json"));
32
+ const langMap = {};
33
+ for (const file of files) {
34
+ const content = JSON.parse(readFileSync(join(dir, file), "utf-8"));
35
+ const lang = content.language ?? "unknown";
36
+ if (langMap[lang] === undefined) {
37
+ langMap[lang] = { files: 0, findings: 0, rules: {} };
38
+ }
39
+ langMap[lang].files++;
40
+ const findings = content.findings ?? [];
41
+ langMap[lang].findings += findings.length;
42
+ for (const f of findings) {
43
+ langMap[lang].rules[f.ruleId] = (langMap[lang].rules[f.ruleId] ?? 0) + 1;
44
+ }
45
+ }
46
+ const stats = Object.entries(langMap).map(([language, data]) => {
47
+ const sortedRules = Object.entries(data.rules)
48
+ .sort(([, a], [, b]) => b - a)
49
+ .slice(0, 3)
50
+ .map(([rule]) => rule);
51
+ return {
52
+ language,
53
+ fileCount: data.files,
54
+ findingCount: data.findings,
55
+ avgFindings: data.files > 0 ? Math.round((data.findings / data.files) * 10) / 10 : 0,
56
+ topRules: sortedRules,
57
+ };
58
+ });
59
+ stats.sort((a, b) => b.findingCount - a.findingCount);
60
+ if (format === "json") {
61
+ console.log(JSON.stringify(stats, null, 2));
62
+ return;
63
+ }
64
+ console.log(`\nLanguage Profile`);
65
+ console.log("═".repeat(70));
66
+ console.log(` ${"Language".padEnd(15)} ${"Files".padEnd(8)} ${"Findings".padEnd(10)} ${"Avg".padEnd(8)} Top Rules`);
67
+ console.log(" " + "─".repeat(65));
68
+ for (const s of stats) {
69
+ console.log(` ${s.language.padEnd(15)} ${String(s.fileCount).padEnd(8)} ${String(s.findingCount).padEnd(10)} ${String(s.avgFindings).padEnd(8)} ${s.topRules.join(", ")}`);
70
+ }
71
+ console.log("═".repeat(70));
72
+ }
73
+ //# sourceMappingURL=review-language-profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-language-profile.js","sourceRoot":"","sources":["../../src/commands/review-language-profile.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-org-dashboard — Organization-wide review dashboard.
3
+ */
4
+ export declare function runReviewOrgDashboard(argv: string[]): void;
5
+ //# sourceMappingURL=review-org-dashboard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-org-dashboard.d.ts","sourceRoot":"","sources":["../../src/commands/review-org-dashboard.ts"],"names":[],"mappings":"AAAA;;GAEG;AAyBH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAgF1D"}
@@ -0,0 +1,69 @@
1
+ /**
2
+ * Review-org-dashboard — Organization-wide review dashboard.
3
+ */
4
+ import { readFileSync, existsSync, readdirSync } from "fs";
5
+ import { join } from "path";
6
+ // ─── CLI ────────────────────────────────────────────────────────────────────
7
+ export function runReviewOrgDashboard(argv) {
8
+ const dirIdx = argv.indexOf("--dir");
9
+ const dir = dirIdx >= 0 ? argv[dirIdx + 1] : ".judges/org-reports";
10
+ const formatIdx = argv.indexOf("--format");
11
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
12
+ if (argv.includes("--help") || argv.includes("-h")) {
13
+ console.log(`
14
+ judges review-org-dashboard — Organization-wide review dashboard
15
+
16
+ Usage:
17
+ judges review-org-dashboard [--dir <path>] [--format table|json]
18
+
19
+ Options:
20
+ --dir <path> Org reports directory (default: .judges/org-reports)
21
+ --format <fmt> Output format: table (default), json
22
+ --help, -h Show this help
23
+
24
+ Expects JSON files in the directory, each representing a repo summary:
25
+ {"repo":"my-app","reviews":25,"findings":42,"avgScore":7.5,"lastReview":"2026-03-10"}
26
+ `);
27
+ return;
28
+ }
29
+ if (!existsSync(dir)) {
30
+ console.log(`Org reports directory not found: ${dir}`);
31
+ console.log("Create the directory and add repo summary files.");
32
+ return;
33
+ }
34
+ const files = readdirSync(dir).filter((f) => f.endsWith(".json"));
35
+ if (files.length === 0) {
36
+ console.log("No org report files found.");
37
+ return;
38
+ }
39
+ const repoStats = [];
40
+ for (const file of files) {
41
+ const content = JSON.parse(readFileSync(join(dir, file), "utf-8"));
42
+ repoStats.push(content);
43
+ }
44
+ repoStats.sort((a, b) => b.findings - a.findings);
45
+ const summary = {
46
+ totalRepos: repoStats.length,
47
+ totalReviews: repoStats.reduce((s, r) => s + r.reviews, 0),
48
+ totalFindings: repoStats.reduce((s, r) => s + r.findings, 0),
49
+ avgScore: repoStats.length > 0
50
+ ? Math.round((repoStats.reduce((s, r) => s + r.avgScore, 0) / repoStats.length) * 10) / 10
51
+ : 0,
52
+ repoStats,
53
+ };
54
+ if (format === "json") {
55
+ console.log(JSON.stringify(summary, null, 2));
56
+ return;
57
+ }
58
+ console.log(`\nOrganization Dashboard`);
59
+ console.log("═".repeat(75));
60
+ console.log(` Repos: ${summary.totalRepos} | Reviews: ${summary.totalReviews} | Findings: ${summary.totalFindings} | Avg Score: ${summary.avgScore}`);
61
+ console.log("");
62
+ console.log(` ${"Repo".padEnd(25)} ${"Reviews".padEnd(10)} ${"Findings".padEnd(10)} ${"Score".padEnd(8)} Last Review`);
63
+ console.log(" " + "─".repeat(70));
64
+ for (const r of repoStats) {
65
+ console.log(` ${r.repo.padEnd(25)} ${String(r.reviews).padEnd(10)} ${String(r.findings).padEnd(10)} ${String(r.avgScore).padEnd(8)} ${r.lastReview}`);
66
+ }
67
+ console.log("═".repeat(75));
68
+ }
69
+ //# sourceMappingURL=review-org-dashboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-org-dashboard.js","sourceRoot":"","sources":["../../src/commands/review-org-dashboard.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-report-archive — Archive and manage historical review reports.
3
+ */
4
+ export declare function runReviewReportArchive(argv: string[]): void;
5
+ //# sourceMappingURL=review-report-archive.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-report-archive.d.ts","sourceRoot":"","sources":["../../src/commands/review-report-archive.ts"],"names":[],"mappings":"AAAA;;GAEG;AAsBH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAgH3D"}
@@ -0,0 +1,101 @@
1
+ /**
2
+ * Review-report-archive — Archive and manage historical review reports.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync, mkdirSync, copyFileSync } from "fs";
5
+ import { join, basename } from "path";
6
+ // ─── CLI ────────────────────────────────────────────────────────────────────
7
+ export function runReviewReportArchive(argv) {
8
+ const archiveIdx = argv.indexOf("--archive-dir");
9
+ const archiveDir = archiveIdx >= 0 ? argv[archiveIdx + 1] : ".judges/archive";
10
+ const formatIdx = argv.indexOf("--format");
11
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
12
+ if (argv.includes("--help") || argv.includes("-h")) {
13
+ console.log(`
14
+ judges review-report-archive — Archive review reports
15
+
16
+ Usage:
17
+ judges review-report-archive [--archive-dir <path>] [--add <report>]
18
+ [--prune <days>] [--format table|json]
19
+
20
+ Options:
21
+ --archive-dir <path> Archive directory (default: .judges/archive)
22
+ --add <report> Archive a report file
23
+ --prune <days> Remove entries older than N days
24
+ --format <fmt> Output format: table (default), json
25
+ --help, -h Show this help
26
+ `);
27
+ return;
28
+ }
29
+ if (!existsSync(archiveDir)) {
30
+ mkdirSync(archiveDir, { recursive: true });
31
+ }
32
+ const indexPath = join(archiveDir, "index.json");
33
+ let index;
34
+ if (existsSync(indexPath)) {
35
+ index = JSON.parse(readFileSync(indexPath, "utf-8"));
36
+ }
37
+ else {
38
+ index = { entries: [], lastUpdated: new Date().toISOString().split("T")[0] };
39
+ }
40
+ // Add report to archive
41
+ const addIdx = argv.indexOf("--add");
42
+ if (addIdx >= 0) {
43
+ const reportPath = argv[addIdx + 1];
44
+ if (!existsSync(reportPath)) {
45
+ console.error(`Report not found: ${reportPath}`);
46
+ process.exitCode = 1;
47
+ return;
48
+ }
49
+ const report = JSON.parse(readFileSync(reportPath, "utf-8"));
50
+ const ts = new Date().toISOString().replace(/[:.]/g, "-");
51
+ const archiveFilename = `${ts}_${basename(reportPath)}`;
52
+ const destPath = join(archiveDir, archiveFilename);
53
+ copyFileSync(reportPath, destPath);
54
+ const entry = {
55
+ filename: archiveFilename,
56
+ archivedAt: new Date().toISOString().split("T")[0],
57
+ originalPath: reportPath,
58
+ findings: report.findings?.length ?? 0,
59
+ verdict: report.overallVerdict ?? "unknown",
60
+ };
61
+ index.entries.push(entry);
62
+ index.lastUpdated = new Date().toISOString().split("T")[0];
63
+ writeFileSync(indexPath, JSON.stringify(index, null, 2));
64
+ console.log(`Archived: ${archiveFilename}`);
65
+ return;
66
+ }
67
+ // Prune old entries
68
+ const pruneIdx = argv.indexOf("--prune");
69
+ if (pruneIdx >= 0) {
70
+ const days = parseInt(argv[pruneIdx + 1], 10);
71
+ const cutoff = new Date();
72
+ cutoff.setDate(cutoff.getDate() - days);
73
+ const cutoffStr = cutoff.toISOString().split("T")[0];
74
+ const before = index.entries.length;
75
+ index.entries = index.entries.filter((e) => e.archivedAt >= cutoffStr);
76
+ index.lastUpdated = new Date().toISOString().split("T")[0];
77
+ writeFileSync(indexPath, JSON.stringify(index, null, 2));
78
+ console.log(`Pruned ${before - index.entries.length} entries older than ${days} days.`);
79
+ return;
80
+ }
81
+ // List archive
82
+ if (format === "json") {
83
+ console.log(JSON.stringify(index, null, 2));
84
+ return;
85
+ }
86
+ console.log(`\nReport Archive`);
87
+ console.log("═".repeat(70));
88
+ if (index.entries.length === 0) {
89
+ console.log(" No archived reports. Use --add <report> to archive one.");
90
+ }
91
+ else {
92
+ console.log(` ${"Date".padEnd(12)} ${"Verdict".padEnd(10)} ${"Findings".padEnd(10)} Filename`);
93
+ console.log(" " + "─".repeat(65));
94
+ for (const e of index.entries) {
95
+ console.log(` ${e.archivedAt.padEnd(12)} ${e.verdict.padEnd(10)} ${String(e.findings).padEnd(10)} ${e.filename}`);
96
+ }
97
+ }
98
+ console.log(`\n Total archived: ${index.entries.length}`);
99
+ console.log("═".repeat(70));
100
+ }
101
+ //# sourceMappingURL=review-report-archive.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-report-archive.js","sourceRoot":"","sources":["../../src/commands/review-report-archive.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Review-sla-config — Configure SLA targets for review resolution.
3
+ */
4
+ export declare function runReviewSlaConfig(argv: string[]): void;
5
+ //# sourceMappingURL=review-sla-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-sla-config.d.ts","sourceRoot":"","sources":["../../src/commands/review-sla-config.ts"],"names":[],"mappings":"AAAA;;GAEG;AA6BH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CAqFvD"}
@@ -0,0 +1,89 @@
1
+ /**
2
+ * Review-sla-config — Configure SLA targets for review resolution.
3
+ */
4
+ import { readFileSync, writeFileSync, existsSync } from "fs";
5
+ const DEFAULT_SLAS = [
6
+ { severity: "critical", maxResolutionHours: 4, escalateAfterHours: 2, notifyOnBreach: true },
7
+ { severity: "high", maxResolutionHours: 24, escalateAfterHours: 12, notifyOnBreach: true },
8
+ { severity: "medium", maxResolutionHours: 72, escalateAfterHours: 48, notifyOnBreach: false },
9
+ { severity: "low", maxResolutionHours: 168, escalateAfterHours: 120, notifyOnBreach: false },
10
+ { severity: "info", maxResolutionHours: 336, escalateAfterHours: 240, notifyOnBreach: false },
11
+ ];
12
+ // ─── CLI ────────────────────────────────────────────────────────────────────
13
+ export function runReviewSlaConfig(argv) {
14
+ const storeIdx = argv.indexOf("--store");
15
+ const storePath = storeIdx >= 0 ? argv[storeIdx + 1] : ".judges-sla.json";
16
+ const formatIdx = argv.indexOf("--format");
17
+ const format = formatIdx >= 0 ? argv[formatIdx + 1] : "table";
18
+ if (argv.includes("--help") || argv.includes("-h")) {
19
+ console.log(`
20
+ judges review-sla-config — Configure SLA targets for review resolution
21
+
22
+ Usage:
23
+ judges review-sla-config [--store <path>] [--init] [--set <json>]
24
+ [--format table|json]
25
+
26
+ Options:
27
+ --store <path> SLA config file (default: .judges-sla.json)
28
+ --init Initialize with default SLA targets
29
+ --set <json> Set SLA target (JSON with severity, maxResolutionHours, etc.)
30
+ --format <fmt> Output format: table (default), json
31
+ --help, -h Show this help
32
+ `);
33
+ return;
34
+ }
35
+ // Init with defaults
36
+ if (argv.includes("--init")) {
37
+ const store = {
38
+ targets: DEFAULT_SLAS,
39
+ lastUpdated: new Date().toISOString().split("T")[0],
40
+ };
41
+ writeFileSync(storePath, JSON.stringify(store, null, 2));
42
+ console.log(`SLA config initialized with defaults at: ${storePath}`);
43
+ return;
44
+ }
45
+ let store;
46
+ if (existsSync(storePath)) {
47
+ store = JSON.parse(readFileSync(storePath, "utf-8"));
48
+ }
49
+ else {
50
+ store = { targets: [], lastUpdated: new Date().toISOString().split("T")[0] };
51
+ }
52
+ // Set SLA target
53
+ const setIdx = argv.indexOf("--set");
54
+ if (setIdx >= 0) {
55
+ const target = JSON.parse(argv[setIdx + 1]);
56
+ const existingIdx = store.targets.findIndex((t) => t.severity === target.severity);
57
+ if (existingIdx >= 0) {
58
+ store.targets[existingIdx] = target;
59
+ }
60
+ else {
61
+ store.targets.push(target);
62
+ }
63
+ store.lastUpdated = new Date().toISOString().split("T")[0];
64
+ writeFileSync(storePath, JSON.stringify(store, null, 2));
65
+ console.log(`SLA target for "${target.severity}" saved.`);
66
+ return;
67
+ }
68
+ // Display
69
+ if (format === "json") {
70
+ console.log(JSON.stringify(store, null, 2));
71
+ return;
72
+ }
73
+ console.log(`\nSLA Configuration`);
74
+ console.log("═".repeat(65));
75
+ if (store.targets.length === 0) {
76
+ console.log(" No SLA targets configured. Use --init for defaults or --set to add.");
77
+ }
78
+ else {
79
+ console.log(` ${"Severity".padEnd(12)} ${"Max Resolution".padEnd(16)} ${"Escalate After".padEnd(16)} Notify`);
80
+ console.log(" " + "─".repeat(55));
81
+ for (const t of store.targets) {
82
+ const maxRes = t.maxResolutionHours < 24 ? `${t.maxResolutionHours}h` : `${Math.round(t.maxResolutionHours / 24)}d`;
83
+ const escalate = t.escalateAfterHours < 24 ? `${t.escalateAfterHours}h` : `${Math.round(t.escalateAfterHours / 24)}d`;
84
+ console.log(` ${t.severity.padEnd(12)} ${maxRes.padEnd(16)} ${escalate.padEnd(16)} ${t.notifyOnBreach ? "Yes" : "No"}`);
85
+ }
86
+ }
87
+ console.log("═".repeat(65));
88
+ }
89
+ //# sourceMappingURL=review-sla-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"review-sla-config.js","sourceRoot":"","sources":["../../src/commands/review-sla-config.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kevinrabun/judges",
3
- "version": "3.97.0",
3
+ "version": "3.98.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.97.0",
10
+ "version": "3.98.0",
11
11
  "packages": [
12
12
  {
13
13
  "registryType": "npm",
14
14
  "identifier": "@kevinrabun/judges",
15
- "version": "3.97.0",
15
+ "version": "3.98.0",
16
16
  "transport": {
17
17
  "type": "stdio"
18
18
  }