import type { OpenClawConfig } from "../../config/types.openclaw.js"; import { withActivatedPluginIds } from "../activation-context.js"; import { resolveChannelPluginIds, resolveConfiguredChannelPluginIds, } from "../channel-plugin-ids.js"; import { loadOpenClawPlugins } from "../loader.js"; import { hasExplicitPluginIdScope, hasNonEmptyPluginIdScope, normalizePluginIdScope, } from "../plugin-scope.js"; import { getActivePluginRegistry } from "../runtime.js"; import { buildPluginRuntimeLoadOptionsFromValues, resolvePluginRuntimeLoadContext, } from "./load-context.js"; let pluginRegistryLoaded: "none" | "configured-channels" | "channels" | "all" = "none"; export type PluginRegistryScope = "configured-channels" | "channels" | "all"; function scopeRank(scope: typeof pluginRegistryLoaded): number { switch (scope) { case "none": return 0; case "configured-channels": return 1; case "channels": return 2; case "all": return 3; } throw new Error("Unsupported plugin registry scope"); } function activeRegistrySatisfiesScope( scope: PluginRegistryScope, active: ReturnType, expectedChannelPluginIds: readonly string[], requestedPluginIds: readonly string[] | undefined, ): boolean { if (!active) { return false; } if (requestedPluginIds !== undefined) { if (requestedPluginIds.length === 0) { return false; } const activePluginIds = new Set( active.plugins.filter((plugin) => plugin.status === "loaded").map((plugin) => plugin.id), ); return requestedPluginIds.every((pluginId) => activePluginIds.has(pluginId)); } const activeChannelPluginIds = new Set(active.channels.map((entry) => entry.plugin.id)); switch (scope) { case "configured-channels": case "channels": return ( active.channels.length > 0 && expectedChannelPluginIds.every((pluginId) => activeChannelPluginIds.has(pluginId)) ); case "all": return false; } throw new Error("Unsupported plugin registry scope"); } function shouldForwardChannelScope(params: { scope: PluginRegistryScope; scopedLoad: boolean; }): boolean { return !params.scopedLoad && params.scope === "configured-channels"; } export function ensurePluginRegistryLoaded(options?: { scope?: PluginRegistryScope; config?: OpenClawConfig; activationSourceConfig?: OpenClawConfig; env?: NodeJS.ProcessEnv; workspaceDir?: string; onlyPluginIds?: string[]; }): void { const scope = options?.scope ?? "all"; const requestedPluginIds = normalizePluginIdScope(options?.onlyPluginIds); const scopedLoad = hasExplicitPluginIdScope(requestedPluginIds); const context = resolvePluginRuntimeLoadContext(options); const expectedChannelPluginIds = scopedLoad ? (requestedPluginIds ?? []) : scope === "configured-channels" ? resolveConfiguredChannelPluginIds({ config: context.config, activationSourceConfig: context.activationSourceConfig, workspaceDir: context.workspaceDir, env: context.env, }) : scope === "channels" ? resolveChannelPluginIds({ config: context.config, workspaceDir: context.workspaceDir, env: context.env, }) : []; const active = getActivePluginRegistry(); if ( !scopedLoad && scopeRank(pluginRegistryLoaded) >= scopeRank(scope) && activeRegistrySatisfiesScope(scope, active, expectedChannelPluginIds, undefined) ) { return; } if ( (pluginRegistryLoaded === "none" || scopedLoad) && activeRegistrySatisfiesScope(scope, active, expectedChannelPluginIds, requestedPluginIds) ) { if (!scopedLoad) { pluginRegistryLoaded = scope; } return; } const scopedConfig = !scopedLoad && scope === "configured-channels" && expectedChannelPluginIds.length > 0 ? (withActivatedPluginIds({ config: context.config, pluginIds: expectedChannelPluginIds, }) ?? context.config) : context.config; const scopedActivationSourceConfig = !scopedLoad && scope === "configured-channels" && expectedChannelPluginIds.length > 0 ? (withActivatedPluginIds({ config: context.activationSourceConfig, pluginIds: expectedChannelPluginIds, }) ?? context.activationSourceConfig) : context.activationSourceConfig; loadOpenClawPlugins( buildPluginRuntimeLoadOptionsFromValues( { ...context, config: scopedConfig, activationSourceConfig: scopedActivationSourceConfig, }, { throwOnLoadError: true, ...(hasExplicitPluginIdScope(requestedPluginIds) || shouldForwardChannelScope({ scope, scopedLoad }) || hasNonEmptyPluginIdScope(expectedChannelPluginIds) ? { onlyPluginIds: expectedChannelPluginIds } : {}), }, ), ); if (!scopedLoad) { pluginRegistryLoaded = scope; } } export const __testing = { resetPluginRegistryLoadedForTests(): void { pluginRegistryLoaded = "none"; }, };