@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.
- package/CHANGELOG.md +13 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +63 -0
- package/dist/cli.js.map +1 -1
- package/dist/commands/finding-cwe-lookup.d.ts +5 -0
- package/dist/commands/finding-cwe-lookup.d.ts.map +1 -0
- package/dist/commands/finding-cwe-lookup.js +149 -0
- package/dist/commands/finding-cwe-lookup.js.map +1 -0
- package/dist/commands/finding-duplicate-detect.d.ts +5 -0
- package/dist/commands/finding-duplicate-detect.d.ts.map +1 -0
- package/dist/commands/finding-duplicate-detect.js +114 -0
- package/dist/commands/finding-duplicate-detect.js.map +1 -0
- package/dist/commands/finding-patch-preview.d.ts +5 -0
- package/dist/commands/finding-patch-preview.d.ts.map +1 -0
- package/dist/commands/finding-patch-preview.js +104 -0
- package/dist/commands/finding-patch-preview.js.map +1 -0
- package/dist/commands/finding-priority-matrix.d.ts +5 -0
- package/dist/commands/finding-priority-matrix.d.ts.map +1 -0
- package/dist/commands/finding-priority-matrix.js +103 -0
- package/dist/commands/finding-priority-matrix.js.map +1 -0
- package/dist/commands/review-cicd-integrate.d.ts +5 -0
- package/dist/commands/review-cicd-integrate.d.ts.map +1 -0
- package/dist/commands/review-cicd-integrate.js +123 -0
- package/dist/commands/review-cicd-integrate.js.map +1 -0
- package/dist/commands/review-language-profile.d.ts +5 -0
- package/dist/commands/review-language-profile.d.ts.map +1 -0
- package/dist/commands/review-language-profile.js +73 -0
- package/dist/commands/review-language-profile.js.map +1 -0
- package/dist/commands/review-org-dashboard.d.ts +5 -0
- package/dist/commands/review-org-dashboard.d.ts.map +1 -0
- package/dist/commands/review-org-dashboard.js +69 -0
- package/dist/commands/review-org-dashboard.js.map +1 -0
- package/dist/commands/review-report-archive.d.ts +5 -0
- package/dist/commands/review-report-archive.d.ts.map +1 -0
- package/dist/commands/review-report-archive.js +101 -0
- package/dist/commands/review-report-archive.js.map +1 -0
- package/dist/commands/review-sla-config.d.ts +5 -0
- package/dist/commands/review-sla-config.d.ts.map +1 -0
- package/dist/commands/review-sla-config.js +89 -0
- package/dist/commands/review-sla-config.js.map +1 -0
- package/package.json +1 -1
- 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":"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"}
|
|
@@ -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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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
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.
|
|
10
|
+
"version": "3.98.0",
|
|
11
11
|
"packages": [
|
|
12
12
|
{
|
|
13
13
|
"registryType": "npm",
|
|
14
14
|
"identifier": "@kevinrabun/judges",
|
|
15
|
-
"version": "3.
|
|
15
|
+
"version": "3.98.0",
|
|
16
16
|
"transport": {
|
|
17
17
|
"type": "stdio"
|
|
18
18
|
}
|