mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 07:30:43 +00:00
164 lines
4.4 KiB
JavaScript
164 lines
4.4 KiB
JavaScript
#!/usr/bin/env node
|
|
import { spawnSync } from "node:child_process";
|
|
import { fileURLToPath } from "node:url";
|
|
import {
|
|
KNIP_OPTIONAL_UNUSED_FILE_ALLOWLIST,
|
|
KNIP_UNUSED_FILE_ALLOWLIST,
|
|
} from "./deadcode-unused-files.allowlist.mjs";
|
|
|
|
const KNIP_VERSION = "6.8.0";
|
|
const KNIP_ARGS = [
|
|
"--config",
|
|
"config/knip.config.ts",
|
|
"--production",
|
|
"--no-progress",
|
|
"--reporter",
|
|
"compact",
|
|
"--files",
|
|
"--no-config-hints",
|
|
];
|
|
|
|
function normalizeRepoPath(value) {
|
|
return value.replaceAll("\\", "/").replace(/^\.\//u, "");
|
|
}
|
|
|
|
function uniqueSorted(values) {
|
|
return [...new Set(values.map(normalizeRepoPath))].toSorted((left, right) =>
|
|
left.localeCompare(right),
|
|
);
|
|
}
|
|
|
|
function isLikelyRepoFilePath(value) {
|
|
return /^(apps|docs|extensions|packages|scripts|src|test|ui)\//u.test(normalizeRepoPath(value));
|
|
}
|
|
|
|
export function parseKnipCompactUnusedFiles(output) {
|
|
const files = [];
|
|
let inUnusedFilesSection = false;
|
|
let sawUnusedFilesSection = false;
|
|
|
|
for (const line of output.split(/\r?\n/u)) {
|
|
if (/^Unused files \(\d+\)$/u.test(line)) {
|
|
inUnusedFilesSection = true;
|
|
sawUnusedFilesSection = true;
|
|
continue;
|
|
}
|
|
if (inUnusedFilesSection && line.trim() === "") {
|
|
break;
|
|
}
|
|
|
|
const separatorIndex = line.lastIndexOf(": ");
|
|
if (separatorIndex === -1) {
|
|
continue;
|
|
}
|
|
if (sawUnusedFilesSection && !inUnusedFilesSection) {
|
|
continue;
|
|
}
|
|
const file = line.slice(separatorIndex + 2).trim();
|
|
if (isLikelyRepoFilePath(file)) {
|
|
files.push(file);
|
|
}
|
|
}
|
|
|
|
return uniqueSorted(files);
|
|
}
|
|
|
|
export function compareUnusedFilesToAllowlist(
|
|
actualFiles,
|
|
allowlistFiles,
|
|
optionalAllowlistFiles = [],
|
|
) {
|
|
const actual = uniqueSorted(actualFiles);
|
|
const allowed = uniqueSorted(allowlistFiles);
|
|
const optionalAllowed = uniqueSorted(optionalAllowlistFiles);
|
|
const allowedOrOptionalSet = new Set([...allowed, ...optionalAllowed]);
|
|
const actualSet = new Set(actual);
|
|
|
|
return {
|
|
actual,
|
|
allowed,
|
|
unexpected: actual.filter((file) => !allowedOrOptionalSet.has(file)),
|
|
stale: allowed.filter((file) => !actualSet.has(file)),
|
|
duplicateAllowedCount: allowlistFiles.length - new Set(allowlistFiles).size,
|
|
allowlistIsSorted:
|
|
JSON.stringify(allowlistFiles.map(normalizeRepoPath)) === JSON.stringify(allowed),
|
|
};
|
|
}
|
|
|
|
export function formatUnusedFileComparison(comparison) {
|
|
const lines = [];
|
|
if (!comparison.allowlistIsSorted) {
|
|
lines.push("deadcode unused-file allowlist is not sorted.");
|
|
}
|
|
if (comparison.duplicateAllowedCount > 0) {
|
|
lines.push(
|
|
`deadcode unused-file allowlist contains ${comparison.duplicateAllowedCount} duplicate entr${
|
|
comparison.duplicateAllowedCount === 1 ? "y" : "ies"
|
|
}.`,
|
|
);
|
|
}
|
|
if (comparison.unexpected.length > 0) {
|
|
lines.push("Unexpected unused files:");
|
|
lines.push(...comparison.unexpected.map((file) => ` ${file}`));
|
|
}
|
|
if (comparison.stale.length > 0) {
|
|
lines.push("Stale allowlist entries:");
|
|
lines.push(...comparison.stale.map((file) => ` ${file}`));
|
|
}
|
|
return lines.join("\n");
|
|
}
|
|
|
|
export function runKnipUnusedFiles() {
|
|
const result = spawnSync(
|
|
"pnpm",
|
|
["--config.minimum-release-age=0", "dlx", `knip@${KNIP_VERSION}`, ...KNIP_ARGS],
|
|
{
|
|
encoding: "utf8",
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
},
|
|
);
|
|
return {
|
|
status: result.status,
|
|
signal: result.signal,
|
|
output: `${result.stdout ?? ""}${result.stderr ?? ""}`,
|
|
};
|
|
}
|
|
|
|
export function checkUnusedFiles(
|
|
output,
|
|
allowlistFiles = KNIP_UNUSED_FILE_ALLOWLIST,
|
|
optionalAllowlistFiles = KNIP_OPTIONAL_UNUSED_FILE_ALLOWLIST,
|
|
) {
|
|
const actual = parseKnipCompactUnusedFiles(output);
|
|
const comparison = compareUnusedFilesToAllowlist(actual, allowlistFiles, optionalAllowlistFiles);
|
|
return {
|
|
ok:
|
|
comparison.allowlistIsSorted &&
|
|
comparison.duplicateAllowedCount === 0 &&
|
|
comparison.unexpected.length === 0 &&
|
|
comparison.stale.length === 0,
|
|
comparison,
|
|
message: formatUnusedFileComparison(comparison),
|
|
};
|
|
}
|
|
|
|
function main() {
|
|
const result = runKnipUnusedFiles();
|
|
const check = checkUnusedFiles(result.output);
|
|
if (!check.ok) {
|
|
if (check.message) {
|
|
console.error(check.message);
|
|
}
|
|
process.exitCode = 1;
|
|
return;
|
|
}
|
|
|
|
console.log(
|
|
`[deadcode] Knip unused-file allowlist matched ${check.comparison.actual.length} intentional entries.`,
|
|
);
|
|
}
|
|
|
|
if (process.argv[1] === fileURLToPath(import.meta.url)) {
|
|
main();
|
|
}
|