Files
openclaw/src/gateway/server-startup-plugins.test.ts
2026-05-10 17:06:15 +01:00

343 lines
11 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from "vitest";
import type { OpenClawConfig } from "../config/types.openclaw.js";
import type { PluginManifestRegistry } from "../plugins/manifest-registry.js";
import type { PluginMetadataSnapshot } from "../plugins/plugin-metadata-snapshot.js";
const applyPluginAutoEnable = vi.hoisted(() =>
vi.fn((params: { config: unknown }) => ({
config: params.config,
changes: [] as string[],
autoEnabledReasons: {} as Record<string, string[]>,
})),
);
const initSubagentRegistry = vi.hoisted(() => vi.fn());
const loadGatewayStartupPlugins = vi.hoisted(() =>
vi.fn((_params: unknown) => ({
pluginRegistry: { diagnostics: [], gatewayHandlers: {}, plugins: [] },
gatewayMethods: ["ping"],
})),
);
const pluginManifestRegistry = vi.hoisted(
(): PluginManifestRegistry => ({
plugins: [
{
id: "telegram",
origin: "bundled",
rootDir: "/package/dist/extensions/telegram",
source: "/package/dist/extensions/telegram/index.js",
manifestPath: "/package/dist/extensions/telegram/package.json",
channels: ["telegram"],
providers: [],
cliBackends: [],
skills: [],
hooks: [],
},
],
diagnostics: [],
}),
);
const pluginMetadataSnapshot = vi.hoisted(
(): PluginMetadataSnapshot => ({
policyHash: "policy",
index: {
version: 1,
hostContractVersion: "test",
compatRegistryVersion: "test",
migrationVersion: 1,
policyHash: "policy",
generatedAtMs: 0,
installRecords: {},
plugins: [],
diagnostics: [],
},
registryDiagnostics: [],
manifestRegistry: pluginManifestRegistry,
plugins: [],
diagnostics: [],
byPluginId: new Map(),
normalizePluginId: (pluginId) => pluginId,
owners: {
channels: new Map(),
channelConfigs: new Map(),
providers: new Map(),
modelCatalogProviders: new Map(),
cliBackends: new Map(),
setupProviders: new Map(),
commandAliases: new Map(),
contracts: new Map(),
},
metrics: {
registrySnapshotMs: 0,
manifestRegistryMs: 0,
ownerMapsMs: 0,
totalMs: 0,
indexPluginCount: 0,
manifestPluginCount: 0,
},
}),
);
const pluginLookUpTableMetrics = vi.hoisted(() => ({
registrySnapshotMs: 0,
manifestRegistryMs: 0,
startupPlanMs: 0,
ownerMapsMs: 0,
totalMs: 0,
indexPluginCount: 0,
manifestPluginCount: 0,
startupPluginCount: 1,
deferredChannelPluginCount: 0,
}));
const loadPluginLookUpTable = vi.hoisted(() =>
vi.fn((_params: unknown) => ({
manifestRegistry: pluginManifestRegistry,
startup: {
configuredDeferredChannelPluginIds: [],
pluginIds: ["telegram"],
},
metrics: pluginLookUpTableMetrics,
})),
);
const resolveOpenClawPackageRootSync = vi.hoisted(() => vi.fn((_params: unknown) => "/package"));
const runChannelPluginStartupMaintenance = vi.hoisted(() =>
vi.fn(async (_params: unknown) => undefined),
);
const runStartupSessionMigration = vi.hoisted(() => vi.fn(async (_params: unknown) => undefined));
vi.mock("../agents/agent-scope.js", () => ({
resolveAgentWorkspaceDir: () => "/workspace",
resolveDefaultAgentId: () => "default",
}));
vi.mock("../agents/subagent-registry.js", () => ({
initSubagentRegistry: () => initSubagentRegistry(),
}));
vi.mock("../channels/plugins/lifecycle-startup.js", () => ({
runChannelPluginStartupMaintenance: (params: unknown) =>
runChannelPluginStartupMaintenance(params),
}));
vi.mock("../config/plugin-auto-enable.js", () => ({
applyPluginAutoEnable: (params: { config: unknown }) => applyPluginAutoEnable(params),
}));
vi.mock("../infra/openclaw-root.js", () => ({
resolveOpenClawPackageRootSync: (params: unknown) => resolveOpenClawPackageRootSync(params),
}));
vi.mock("../plugins/plugin-lookup-table.js", () => ({
loadPluginLookUpTable: (params: unknown) => loadPluginLookUpTable(params),
}));
vi.mock("../plugins/registry.js", () => ({
createEmptyPluginRegistry: () => ({ diagnostics: [], gatewayHandlers: {}, plugins: [] }),
}));
vi.mock("../plugins/runtime.js", () => ({
getActivePluginRegistry: () => undefined,
setActivePluginRegistry: vi.fn(),
}));
vi.mock("./server-methods-list.js", () => ({
listGatewayMethods: () => ["ping"],
}));
vi.mock("./server-methods.js", () => ({
coreGatewayHandlers: {},
}));
vi.mock("./server-plugin-bootstrap.js", () => ({
loadGatewayStartupPlugins: (params: unknown) => loadGatewayStartupPlugins(params),
}));
vi.mock("./server-startup-session-migration.js", () => ({
runStartupSessionMigration: (params: unknown) => runStartupSessionMigration(params),
}));
function createLog() {
return {
info: vi.fn(),
warn: vi.fn(),
error: vi.fn(),
debug: vi.fn(),
};
}
function firstCallArg<T>(mock: { mock: { calls: unknown[][] } }, _type?: (value: T) => T): T {
const call = mock.mock.calls[0];
expect(call).toBeDefined();
return call?.[0] as T;
}
describe("prepareGatewayPluginBootstrap startup plugins", () => {
beforeEach(() => {
applyPluginAutoEnable.mockClear();
initSubagentRegistry.mockClear();
loadGatewayStartupPlugins.mockClear();
loadPluginLookUpTable.mockClear().mockReturnValue({
manifestRegistry: pluginManifestRegistry,
startup: {
configuredDeferredChannelPluginIds: [],
pluginIds: ["telegram"],
},
metrics: pluginLookUpTableMetrics,
});
resolveOpenClawPackageRootSync.mockClear().mockReturnValue("/package");
runChannelPluginStartupMaintenance.mockClear();
runStartupSessionMigration.mockClear();
});
it("derives startup activation from source config instead of runtime plugin defaults", async () => {
const sourceConfig = {
channels: {
telegram: {
botToken: "token",
},
},
plugins: {
allow: ["bench-plugin"],
},
} as OpenClawConfig;
const activationConfig = {
channels: {
telegram: {
botToken: "token",
enabled: true,
},
},
plugins: {
allow: ["bench-plugin"],
entries: {
"bench-plugin": {
enabled: true,
},
},
},
} as OpenClawConfig;
const runtimeConfig = {
channels: {
telegram: {
botToken: "token",
dmPolicy: "pairing",
groupPolicy: "allowlist",
},
},
plugins: {
allow: ["bench-plugin", "memory-core"],
entries: {
"bench-plugin": {
config: {
runtimeDefault: true,
},
},
"memory-core": {
config: {
dreaming: {
enabled: false,
},
},
},
},
},
} as OpenClawConfig;
applyPluginAutoEnable.mockReturnValueOnce({
config: activationConfig,
changes: [],
autoEnabledReasons: {},
});
const log = createLog();
const { prepareGatewayPluginBootstrap } = await import("./server-startup-plugins.js");
await prepareGatewayPluginBootstrap({
cfgAtStart: runtimeConfig,
activationSourceConfig: sourceConfig,
startupRuntimeConfig: runtimeConfig,
pluginMetadataSnapshot,
minimalTestGateway: false,
log,
});
expect(applyPluginAutoEnable).toHaveBeenCalledWith({
config: sourceConfig,
env: process.env,
manifestRegistry: pluginManifestRegistry,
});
const lookupInput = firstCallArg<{
activationSourceConfig?: OpenClawConfig;
metadataSnapshot?: PluginMetadataSnapshot;
config?: OpenClawConfig;
}>(loadPluginLookUpTable);
expect(lookupInput.activationSourceConfig).toBe(sourceConfig);
expect(lookupInput.metadataSnapshot).toBe(pluginMetadataSnapshot);
expect(lookupInput.config?.channels?.telegram?.enabled).toBe(true);
expect(lookupInput.config?.channels?.telegram?.dmPolicy).toBe("pairing");
expect(lookupInput.config?.channels?.telegram?.groupPolicy).toBe("allowlist");
expect(lookupInput.config?.plugins?.allow).toEqual(["bench-plugin"]);
expect(lookupInput.config?.plugins?.entries?.["bench-plugin"]?.enabled).toBe(true);
expect(lookupInput.config?.plugins?.entries?.["bench-plugin"]?.config).toEqual({
runtimeDefault: true,
});
expect(lookupInput.config?.plugins?.entries?.["memory-core"]?.config).toEqual({
dreaming: { enabled: false },
});
const startupInput = firstCallArg<{
activationSourceConfig?: OpenClawConfig;
cfg?: OpenClawConfig;
}>(loadGatewayStartupPlugins);
expect(startupInput.activationSourceConfig).toBe(sourceConfig);
expect(startupInput.cfg?.channels?.telegram?.enabled).toBe(true);
expect(startupInput.cfg?.channels?.telegram?.dmPolicy).toBe("pairing");
expect(startupInput.cfg?.channels?.telegram?.groupPolicy).toBe("allowlist");
expect(startupInput.cfg?.plugins?.allow).toEqual(["bench-plugin"]);
expect(startupInput.cfg?.plugins?.entries?.["bench-plugin"]?.enabled).toBe(true);
expect(startupInput.cfg?.plugins?.entries?.["bench-plugin"]?.config).toEqual({
runtimeDefault: true,
});
expect(startupInput.cfg?.plugins?.entries?.["memory-core"]?.config).toEqual({
dreaming: { enabled: false },
});
});
it("bypasses plugin lookup when plugins are globally disabled", async () => {
const cfg = {
channels: {
telegram: {
botToken: "token",
},
},
plugins: {
enabled: false,
allow: ["telegram"],
entries: {
telegram: { enabled: true },
},
},
} as OpenClawConfig;
const log = createLog();
const { prepareGatewayPluginBootstrap } = await import("./server-startup-plugins.js");
const result = await prepareGatewayPluginBootstrap({
cfgAtStart: cfg,
startupRuntimeConfig: cfg,
minimalTestGateway: false,
log,
});
expect(result.startupPluginIds).toEqual([]);
expect(result.deferredConfiguredChannelPluginIds).toEqual([]);
expect(result.pluginLookUpTable).toBeUndefined();
expect(result.baseGatewayMethods).toEqual(["ping"]);
expect(loadPluginLookUpTable).not.toHaveBeenCalled();
const startupInput = firstCallArg<{
cfg?: OpenClawConfig;
pluginIds?: string[];
pluginLookUpTable?: unknown;
preferSetupRuntimeForChannelPlugins?: boolean;
suppressPluginInfoLogs?: boolean;
}>(loadGatewayStartupPlugins);
expect(startupInput.cfg).toStrictEqual(cfg);
expect(startupInput.pluginIds).toEqual([]);
expect(startupInput.pluginLookUpTable).toBeUndefined();
expect(startupInput.preferSetupRuntimeForChannelPlugins).toBe(false);
expect(startupInput.suppressPluginInfoLogs).toBe(false);
});
});