Files
openclaw/scripts/lib/run-extension-oxlint.mjs
2026-04-08 03:54:03 +01:00

141 lines
3.8 KiB
JavaScript

import { spawnSync } from "node:child_process";
import fs from "node:fs";
import os from "node:os";
import path from "node:path";
import {
acquireLocalHeavyCheckLockSync,
applyLocalOxlintPolicy,
} from "./local-heavy-check-runtime.mjs";
export function runExtensionOxlint(params) {
const repoRoot = process.cwd();
const oxlintPath = path.resolve("node_modules", ".bin", "oxlint");
const releaseLock = acquireLocalHeavyCheckLockSync({
cwd: repoRoot,
env: process.env,
toolName: params.toolName,
lockName: params.lockName,
});
const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), params.tempDirPrefix));
const tempConfigPath = path.join(tempDir, "oxlint.json");
try {
prepareExtensionPackageBoundaryArtifacts(repoRoot);
const extensionFiles = params.roots.flatMap((root) =>
collectTypeScriptFiles(path.resolve(repoRoot, root)),
);
if (extensionFiles.length === 0) {
console.error(params.emptyMessage);
process.exit(1);
}
writeTempOxlintConfig(repoRoot, tempConfigPath);
const baseArgs = ["-c", tempConfigPath, ...process.argv.slice(2), ...extensionFiles];
const { args: finalArgs, env } = applyLocalOxlintPolicy(baseArgs, process.env);
const result = spawnSync(oxlintPath, finalArgs, {
stdio: "inherit",
env,
shell: process.platform === "win32",
});
if (result.error) {
throw result.error;
}
process.exit(result.status ?? 1);
} finally {
fs.rmSync(tempDir, { recursive: true, force: true });
releaseLock();
}
}
function prepareExtensionPackageBoundaryArtifacts(repoRoot) {
const releaseLock = acquireLocalHeavyCheckLockSync({
cwd: repoRoot,
env: process.env,
toolName: "extension-package-boundary-artifacts",
lockName: "extension-package-boundary-artifacts",
});
try {
const result = spawnSync(
process.execPath,
[path.resolve(repoRoot, "scripts", "prepare-extension-package-boundary-artifacts.mjs")],
{
cwd: repoRoot,
stdio: "inherit",
env: process.env,
},
);
if (result.error) {
throw result.error;
}
if ((result.status ?? 1) !== 0) {
process.exit(result.status ?? 1);
}
} finally {
releaseLock();
}
}
function writeTempOxlintConfig(repoRoot, configPath) {
const config = JSON.parse(fs.readFileSync(path.resolve(repoRoot, ".oxlintrc.json"), "utf8"));
delete config.$schema;
if (Array.isArray(config.ignorePatterns)) {
const extensionsIgnorePattern = config.ignorePatterns.find((pattern) =>
isTopLevelExtensionsIgnorePattern(pattern),
);
if (extensionsIgnorePattern) {
throw new Error(
`Refusing to run extension oxlint with .oxlintrc.json ignore pattern ${JSON.stringify(
extensionsIgnorePattern,
)}. Remove the top-level extensions ignore so root and focused lint agree.`,
);
}
}
fs.writeFileSync(configPath, `${JSON.stringify(config, null, 2)}\n`, "utf8");
}
function isTopLevelExtensionsIgnorePattern(pattern) {
const normalized = pattern
.trim()
.replaceAll("\\", "/")
.replaceAll(/^\.?\//g, "");
return (
normalized === "extensions" || normalized === "extensions/" || normalized === "extensions/**"
);
}
function collectTypeScriptFiles(directoryPath) {
const entries = fs.readdirSync(directoryPath, { withFileTypes: true });
const files = [];
for (const entry of entries.toSorted((a, b) => a.name.localeCompare(b.name))) {
const entryPath = path.join(directoryPath, entry.name);
if (entry.isDirectory()) {
files.push(...collectTypeScriptFiles(entryPath));
continue;
}
if (!entry.isFile()) {
continue;
}
if (!entry.name.endsWith(".ts") && !entry.name.endsWith(".tsx")) {
continue;
}
files.push(path.relative(process.cwd(), entryPath).split(path.sep).join("/"));
}
return files;
}