Files
openclaw/scripts/lib/ts-topology/scope.ts

161 lines
4.6 KiB
TypeScript

import fs from "node:fs";
import path from "node:path";
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("extensions/")) {
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;
}
}
function extractExtensionId(relPath: string): string | null {
if (!relPath.startsWith("extensions/")) {
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,
);
}