import { existsSync, readFileSync, readdirSync } from "node:fs"; import { join, posix, resolve } from "node:path"; 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/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/qa-channel/api.js": ["../dist/plugin-sdk/extensions/qa-channel/api.d.ts"], "@openclaw/discord/api.js": ["../dist/plugin-sdk/extensions/discord/api.d.ts"], "@openclaw/slack/api.js": ["../dist/plugin-sdk/extensions/slack/api.d.ts"], "@openclaw/whatsapp/api.js": ["../dist/plugin-sdk/extensions/whatsapp/api.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, "@openclaw/discord/api.js": _omitDiscord, "@openclaw/slack/api.js": _omitSlack, "@openclaw/whatsapp/api.js": _omitWhatsApp, ...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/qa-channel/api.js": ["../../dist/plugin-sdk/extensions/qa-channel/api.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; type ExtensionPackageBoundaryTsConfigJson = { extends?: unknown; compilerOptions?: { rootDir?: unknown; paths?: unknown; }; include?: unknown; exclude?: unknown; }; type ExtensionPackageBoundaryPackageJson = { devDependencies?: Record; }; // oxlint-disable-next-line typescript/no-unnecessary-type-parameters -- Boundary helper lets callers ascribe JSON file shape. function readJsonFile(filePath: string): T { return JSON.parse(readFileSync(filePath, "utf8")) as T; } function collectBundledExtensionIds(rootDir = resolve(".")): string[] { return readdirSync(join(rootDir, "extensions"), { withFileTypes: true }) .filter((entry) => entry.isDirectory()) .map((entry) => entry.name) .toSorted(); } function resolveExtensionTsconfigPath(extensionId: string, rootDir = resolve(".")): string { return join(rootDir, "extensions", extensionId, "tsconfig.json"); } 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), ), ); }