Files
openclaw/src/plugins/bundled-plugin-scan.ts
2026-04-07 09:44:53 +01:00

130 lines
3.9 KiB
TypeScript

import fs from "node:fs";
import path from "node:path";
import { normalizeOptionalString } from "../shared/string-coerce.js";
import { normalizeTrimmedStringList } from "../shared/string-normalization.js";
const PUBLIC_SURFACE_SOURCE_EXTENSIONS = [".ts", ".mts", ".js", ".mjs", ".cts", ".cjs"] as const;
const RUNTIME_SIDECAR_ARTIFACTS = new Set([
"helper-api.js",
"light-runtime-api.js",
"runtime-api.js",
"thread-bindings-runtime.js",
]);
export { normalizeOptionalString as trimBundledPluginString };
export function normalizeBundledPluginStringList(value: unknown): string[] {
return normalizeTrimmedStringList(value);
}
export function rewriteBundledPluginEntryToBuiltPath(
entry: string | undefined,
): string | undefined {
if (!entry) {
return undefined;
}
const normalized = entry.replace(/^\.\//u, "");
return normalized.replace(/\.[^.]+$/u, ".js");
}
function isTopLevelPublicSurfaceSource(name: string): boolean {
if (
!PUBLIC_SURFACE_SOURCE_EXTENSIONS.includes(
path.extname(name) as (typeof PUBLIC_SURFACE_SOURCE_EXTENSIONS)[number],
)
) {
return false;
}
if (name.startsWith(".") || name.startsWith("test-") || name.includes(".test-")) {
return false;
}
if (name.endsWith(".d.ts")) {
return false;
}
if (/^config-api(\.[cm]?[jt]s)$/u.test(name)) {
return false;
}
return !/(\.test|\.spec)(\.[cm]?[jt]s)$/u.test(name);
}
export function deriveBundledPluginIdHint(params: {
entryPath: string;
manifestId: string;
packageName?: string;
hasMultipleExtensions: boolean;
}): string {
const base = path.basename(params.entryPath, path.extname(params.entryPath));
if (!params.hasMultipleExtensions) {
return params.manifestId;
}
const packageName = normalizeOptionalString(params.packageName);
if (!packageName) {
return `${params.manifestId}/${base}`;
}
const unscoped = packageName.includes("/")
? (packageName.split("/").pop() ?? packageName)
: packageName;
return `${unscoped}/${base}`;
}
export function collectBundledPluginPublicSurfaceArtifacts(params: {
pluginDir: string;
sourceEntry: string;
setupEntry?: string;
}): readonly string[] | undefined {
const excluded = new Set(
normalizeTrimmedStringList([params.sourceEntry, params.setupEntry]).map((entry) =>
path.basename(entry),
),
);
const artifacts = fs
.readdirSync(params.pluginDir, { withFileTypes: true })
.filter((entry) => entry.isFile())
.map((entry) => entry.name)
.filter(isTopLevelPublicSurfaceSource)
.filter((entry) => !excluded.has(entry))
.map((entry) => rewriteBundledPluginEntryToBuiltPath(entry))
.filter((entry): entry is string => typeof entry === "string" && entry.length > 0)
.toSorted((left, right) => left.localeCompare(right));
return artifacts.length > 0 ? artifacts : undefined;
}
export function collectBundledPluginRuntimeSidecarArtifacts(
publicSurfaceArtifacts: readonly string[] | undefined,
): readonly string[] | undefined {
if (!publicSurfaceArtifacts) {
return undefined;
}
const artifacts = publicSurfaceArtifacts.filter((artifact) =>
RUNTIME_SIDECAR_ARTIFACTS.has(artifact),
);
return artifacts.length > 0 ? artifacts : undefined;
}
export function resolveBundledPluginScanDir(params: {
packageRoot: string;
runningFromBuiltArtifact: boolean;
}): string | undefined {
const sourceDir = path.join(params.packageRoot, "extensions");
const runtimeDir = path.join(params.packageRoot, "dist-runtime", "extensions");
const builtDir = path.join(params.packageRoot, "dist", "extensions");
if (params.runningFromBuiltArtifact) {
if (fs.existsSync(builtDir)) {
return builtDir;
}
if (fs.existsSync(runtimeDir)) {
return runtimeDir;
}
}
if (fs.existsSync(sourceDir)) {
return sourceDir;
}
if (fs.existsSync(runtimeDir) && fs.existsSync(builtDir)) {
return runtimeDir;
}
if (fs.existsSync(builtDir)) {
return builtDir;
}
return undefined;
}