import { existsSync, readFileSync, readdirSync } from "node:fs"; import { join, posix, resolve } from "node:path"; export const EXTENSION_PACKAGE_BOUNDARY_BASE_CONFIG = "extensions/tsconfig.package-boundary.base.json" as const; export const EXTENSION_PACKAGE_BOUNDARY_INCLUDE = ["./*.ts", "./src/**/*.ts"] as const; export const EXTENSION_PACKAGE_BOUNDARY_EXCLUDE = [ "./**/*.test.ts", "./dist/**", "./node_modules/**", "./src/test-support/**", "./src/**/*test-helpers.ts", "./src/**/*test-harness.ts", "./src/**/*test-support.ts", ] as const; export const EXTENSION_PACKAGE_BOUNDARY_BASE_PATHS = { "openclaw/extension-api": ["../src/extensionAPI.ts"], "openclaw/plugin-sdk": ["../dist/plugin-sdk/src/plugin-sdk/index.d.ts"], "openclaw/plugin-sdk/*": ["../dist/plugin-sdk/src/plugin-sdk/*.d.ts"], "openclaw/plugin-sdk/account-id": ["../dist/plugin-sdk/src/plugin-sdk/account-id.d.ts"], "openclaw/plugin-sdk/channel-entry-contract": [ "../packages/plugin-sdk/dist/src/plugin-sdk/channel-entry-contract.d.ts", ], "openclaw/plugin-sdk/browser-maintenance": [ "../packages/plugin-sdk/dist/extensions/browser/browser-maintenance.d.ts", ], "openclaw/plugin-sdk/browser-config-runtime": [ "../dist/plugin-sdk/src/plugin-sdk/browser-config-runtime.d.ts", ], "openclaw/plugin-sdk/browser-node-runtime": [ "../dist/plugin-sdk/src/plugin-sdk/browser-node-runtime.d.ts", ], "openclaw/plugin-sdk/browser-setup-tools": [ "../dist/plugin-sdk/src/plugin-sdk/browser-setup-tools.d.ts", ], "openclaw/plugin-sdk/browser-security-runtime": [ "../dist/plugin-sdk/src/plugin-sdk/browser-security-runtime.d.ts", ], "openclaw/plugin-sdk/channel-secret-basic-runtime": [ "../packages/plugin-sdk/dist/src/plugin-sdk/channel-secret-basic-runtime.d.ts", ], "openclaw/plugin-sdk/channel-secret-runtime": [ "../dist/plugin-sdk/src/plugin-sdk/channel-secret-runtime.d.ts", ], "openclaw/plugin-sdk/channel-secret-tts-runtime": [ "../packages/plugin-sdk/dist/src/plugin-sdk/channel-secret-tts-runtime.d.ts", ], "openclaw/plugin-sdk/channel-streaming": [ "../dist/plugin-sdk/src/plugin-sdk/channel-streaming.d.ts", ], "openclaw/plugin-sdk/error-runtime": ["../dist/plugin-sdk/src/plugin-sdk/error-runtime.d.ts"], "openclaw/plugin-sdk/provider-catalog-shared": [ "../packages/plugin-sdk/dist/src/plugin-sdk/provider-catalog-shared.d.ts", ], "openclaw/plugin-sdk/provider-entry": [ "../packages/plugin-sdk/dist/src/plugin-sdk/provider-entry.d.ts", ], "openclaw/plugin-sdk/secret-ref-runtime": [ "../dist/plugin-sdk/src/plugin-sdk/secret-ref-runtime.d.ts", ], "openclaw/plugin-sdk/ssrf-runtime": ["../dist/plugin-sdk/src/plugin-sdk/ssrf-runtime.d.ts"], "@openclaw/*.js": ["../packages/plugin-sdk/dist/extensions/*.d.ts", "../extensions/*"], "@openclaw/*": ["../packages/plugin-sdk/dist/extensions/*", "../extensions/*"], "@openclaw/plugin-sdk/*": ["../dist/plugin-sdk/src/plugin-sdk/*.d.ts"], } as const; function prefixExtensionPackageBoundaryPaths( paths: Record, prefix: string, ): Record { return Object.fromEntries( Object.entries(paths).map(([key, values]) => [ key, values.map((value) => posix.join(prefix, value)), ]), ); } export const EXTENSION_PACKAGE_BOUNDARY_XAI_PATHS = { ...prefixExtensionPackageBoundaryPaths( (({ "openclaw/plugin-sdk/channel-secret-basic-runtime": _omitBasic, "openclaw/plugin-sdk/channel-secret-tts-runtime": _omitTts, ...rest }) => rest)(EXTENSION_PACKAGE_BOUNDARY_BASE_PATHS), "../", ), "openclaw/plugin-sdk/channel-entry-contract": [ "../../dist/plugin-sdk/src/plugin-sdk/channel-entry-contract.d.ts", ], "openclaw/plugin-sdk/browser-maintenance": [ "../../dist/plugin-sdk/src/plugin-sdk/browser-maintenance.d.ts", ], "openclaw/plugin-sdk/cli-runtime": ["../../dist/plugin-sdk/src/plugin-sdk/cli-runtime.d.ts"], "openclaw/plugin-sdk/provider-catalog-shared": [ "../../dist/plugin-sdk/src/plugin-sdk/provider-catalog-shared.d.ts", ], "openclaw/plugin-sdk/provider-env-vars": [ "../../dist/plugin-sdk/src/plugin-sdk/provider-env-vars.d.ts", ], "openclaw/plugin-sdk/provider-entry": [ "../../dist/plugin-sdk/src/plugin-sdk/provider-entry.d.ts", ], "openclaw/plugin-sdk/provider-web-search-contract": [ "../../dist/plugin-sdk/src/plugin-sdk/provider-web-search-contract.d.ts", ], "@openclaw/*.js": ["../../packages/plugin-sdk/dist/extensions/*.d.ts", "../*"], "@openclaw/*": ["../*"], "@openclaw/plugin-sdk/*": ["../../dist/plugin-sdk/src/plugin-sdk/*.d.ts"], "@openclaw/anthropic-vertex/api.js": ["./.boundary-stubs/anthropic-vertex-api.d.ts"], "@openclaw/ollama/api.js": ["./.boundary-stubs/ollama-api.d.ts"], "@openclaw/ollama/runtime-api.js": ["./.boundary-stubs/ollama-runtime-api.d.ts"], "@openclaw/speech-core/runtime-api.js": ["./.boundary-stubs/speech-core-runtime-api.d.ts"], } as const; export type ExtensionPackageBoundaryTsConfigJson = { extends?: unknown; compilerOptions?: { rootDir?: unknown; paths?: unknown; }; include?: unknown; exclude?: unknown; }; export type ExtensionPackageBoundaryPackageJson = { devDependencies?: Record; }; function readJsonFile(filePath: string): T { return JSON.parse(readFileSync(filePath, "utf8")) as T; } export function collectBundledExtensionIds(rootDir = resolve(".")): string[] { return readdirSync(join(rootDir, "extensions"), { withFileTypes: true }) .filter((entry) => entry.isDirectory()) .map((entry) => entry.name) .toSorted(); } export function resolveExtensionTsconfigPath(extensionId: string, rootDir = resolve(".")): string { return join(rootDir, "extensions", extensionId, "tsconfig.json"); } export function resolveExtensionPackageJsonPath( extensionId: string, rootDir = resolve("."), ): string { return join(rootDir, "extensions", extensionId, "package.json"); } export function readExtensionPackageBoundaryTsconfig( extensionId: string, rootDir = resolve("."), ): ExtensionPackageBoundaryTsConfigJson { return readJsonFile( resolveExtensionTsconfigPath(extensionId, rootDir), ); } export function readExtensionPackageBoundaryPackageJson( extensionId: string, rootDir = resolve("."), ): ExtensionPackageBoundaryPackageJson { return readJsonFile( resolveExtensionPackageJsonPath(extensionId, rootDir), ); } export function isOptInExtensionPackageBoundaryTsconfig( tsconfig: ExtensionPackageBoundaryTsConfigJson, ): boolean { return tsconfig.extends === "../tsconfig.package-boundary.base.json"; } export function collectExtensionsWithTsconfig(rootDir = resolve(".")): string[] { return collectBundledExtensionIds(rootDir).filter((extensionId) => existsSync(resolveExtensionTsconfigPath(extensionId, rootDir)), ); } export function collectOptInExtensionPackageBoundaries(rootDir = resolve(".")): string[] { return collectExtensionsWithTsconfig(rootDir).filter((extensionId) => isOptInExtensionPackageBoundaryTsconfig( readExtensionPackageBoundaryTsconfig(extensionId, rootDir), ), ); } export function renderExtensionPackageBoundaryTsconfig(params?: { paths?: Record; }): { extends: "../tsconfig.package-boundary.base.json"; compilerOptions: { rootDir: "."; paths?: Record }; include: typeof EXTENSION_PACKAGE_BOUNDARY_INCLUDE; exclude: typeof EXTENSION_PACKAGE_BOUNDARY_EXCLUDE; } { return { extends: "../tsconfig.package-boundary.base.json", compilerOptions: { rootDir: ".", ...(params?.paths ? { paths: { ...EXTENSION_PACKAGE_BOUNDARY_BASE_PATHS, ...params.paths, }, } : {}), }, include: EXTENSION_PACKAGE_BOUNDARY_INCLUDE, exclude: EXTENSION_PACKAGE_BOUNDARY_EXCLUDE, }; }