import fs from "node:fs" ;
import path from "node:path" ;
import { BUNDLED_PLUGIN_PATH_PREFIX } from "../bundled-plugin-paths.mjs" ;
import { pluginSdkEntrypoints } from "../plugin-sdk-entries.mjs" ;
import type { ConsumerScope, PublicEntrypoint, TopologyScope, UsageBucket } from "./types.js" ;
function isTestFile(relPath: string): boolean {
return (
relPath.startsWith("test/" ) ||
relPath.includes("/__tests__/" ) ||
relPath.includes(".test." ) ||
relPath.includes(".spec." ) ||
relPath.includes(".e2e." ) ||
relPath.includes(".suite." ) ||
relPath.includes("test-harness" ) ||
relPath.includes("test-support" ) ||
relPath.includes("test-helper" ) ||
relPath.includes("test-utils" )
);
}
function classifyScope(relPath: string): ConsumerScope {
if (relPath.startsWith(BUNDLED_PLUGIN_PATH_PREFIX)) {
return "extension" ;
}
if (relPath.startsWith("packages/" )) {
return "package" ;
}
if (relPath.startsWith("apps/" )) {
return "app" ;
}
if (relPath.startsWith("ui/" )) {
return "ui" ;
}
if (relPath.startsWith("scripts/" )) {
return "script" ;
}
if (relPath.startsWith("src/" )) {
return "src" ;
}
if (relPath.startsWith("test/" )) {
return "test" ;
}
return "other" ;
}
function classifyUsageBucketForRoots(internalRoots: string[], relPath: string): UsageBucket {
if (internalRoots.some((root) => relPath === root || relPath.startsWith(`${root}/`))) {
return "internal" ;
}
return isTestFile(relPath) ? "test" : "production" ;
}
function extractOwner(relPath: string): string | null {
const scope = classifyScope(relPath);
const parts = relPath.split("/" );
switch (scope) {
case "extension" :
return parts[1 ] ? `extension:${parts[1 ]}` : "extension" ;
case "package" :
return parts[1 ] ? `package :${parts[1 ]}` : "package" ;
case "app" :
return parts[1 ] ? `app:${parts[1 ]}` : "app" ;
case "src" :
return "src" ;
case "ui" :
return "ui" ;
case "script" :
return "scripts" ;
case "other" :
return parts[0 ] || "other" ;
case "test" :
return null ;
}
throw new Error("Unsupported topology scope" );
}
function extractExtensionId(relPath: string): string | null {
if (!relPath.startsWith(BUNDLED_PLUGIN_PATH_PREFIX)) {
return null ;
}
const parts = relPath.split("/" );
return parts[1 ] ?? null ;
}
function extractPackageOwner(relPath: string): string | null {
const owner = extractOwner(relPath);
return owner?.startsWith("extension:" ) ? null : owner;
}
function buildScopeFromEntrypoints(
id: string,
description: string,
entrypoints: PublicEntrypoint[],
): TopologyScope {
const internalRoots = [
...new Set(entrypoints.map((entrypoint) => path.posix.dirname(entrypoint.sourcePath))),
];
const publicSpecifiers = new Set(entrypoints.map((entrypoint) => entrypoint.importSpecifier));
return {
id,
description,
entrypoints,
importFilter(specifier: string) {
return publicSpecifiers.has(specifier);
},
classifyUsageBucket(relPath: string) {
return classifyUsageBucketForRoots(internalRoots, relPath);
},
classifyScope,
ownerForPath(relPath: string) {
return extractOwner(relPath);
},
extensionForPath(relPath: string) {
return extractExtensionId(relPath);
},
packageOwnerForPath(relPath: string) {
return extractPackageOwner(relPath);
},
};
}
export function createPluginSdkScope(_repoRoot: string): TopologyScope {
const entrypoints = pluginSdkEntrypoints.map((entrypoint) => ({
entrypoint,
sourcePath: `src/plugin-sdk/${entrypoint}.ts`,
importSpecifier:
entrypoint === "index" ? "openclaw/plugin-sdk" : `openclaw/plugin-sdk/${entrypoint}`,
}));
return buildScopeFromEntrypoints("plugin-sdk" , "OpenClaw plugin-sdk public surface" , entrypoints);
}
export function createFilesystemPublicSurfaceScope(
repoRoot: string,
options: {
id: string;
description?: string;
entrypointRoot: string;
importPrefix: string;
},
): TopologyScope {
const absoluteRoot = path.join(repoRoot, options.entrypointRoot);
const entries = fs
.readdirSync(absoluteRoot, { withFileTypes: true })
.filter((entry) => entry.isFile() && entry.name.endsWith(".ts" ))
.map((entry) => entry.name)
.toSorted();
const publicEntrypoints = entries.map((fileName) => {
const entrypoint = fileName.replace(/\.ts$/, "" );
return {
entrypoint,
sourcePath: path.posix.join(options.entrypointRoot, fileName),
importSpecifier:
entrypoint === "index" ? options.importPrefix : `${options.importPrefix}/${entrypoint}`,
};
});
return buildScopeFromEntrypoints(
options.id,
options.description ?? `Public surface rooted at ${options.entrypointRoot}`,
publicEntrypoints,
);
}
Messung V0.5 in Prozent C=95 H=98 G=96
¤ Dauer der Verarbeitung: 0.10 Sekunden
(vorverarbeitet am 2026-06-10)
¤
*© Formatika GbR, Deutschland