mirror of
http://192.168.0.88:13333/lywsvip/openclaw-zero-token.git
synced 2026-05-15 22:39:06 +08:00
Major upgrade from e26988a38 to upstream v2026.3.28 (f9b107928).
Key changes:
- Upstream src/, ui/, extensions/ (89 bundled extensions)
- Zero-token web providers preserved in src/zero-token/
- AskOnce plugin restored and registered as CLI command
- Added missing packages: @anthropic-ai/vertex-sdk, @modelcontextprotocol/sdk
- Fixed tsconfig rootDir, skipLibCheck for plugin-sdk DTS build
- Added askonce to bundled plugin metadata and package.json exports
- Fixed AskOnce CLI command registration (missing commands metadata)
- Restored AskOnce adapter imports (correct 5-level relative paths)
- Removed stale migration artifacts from root directory
369 lines
12 KiB
TypeScript
369 lines
12 KiB
TypeScript
import { beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import type { ChannelPluginCatalogEntry } from "../channels/plugins/catalog.js";
|
|
import type { ChannelPlugin } from "../channels/plugins/types.js";
|
|
import { setActivePluginRegistry } from "../plugins/runtime.js";
|
|
import { createChannelTestPluginBase, createTestRegistry } from "../test-utils/channel-plugins.js";
|
|
import {
|
|
ensureChannelSetupPluginInstalled,
|
|
loadChannelSetupPluginRegistrySnapshotForChannel,
|
|
} from "./channel-setup/plugin-install.js";
|
|
import { setDefaultChannelPluginRegistryForTests } from "./channel-test-helpers.js";
|
|
import { configMocks, offsetMocks } from "./channels.mock-harness.js";
|
|
import {
|
|
createMSTeamsCatalogEntry,
|
|
createMSTeamsSetupPlugin,
|
|
} from "./channels.plugin-install.test-helpers.js";
|
|
import { baseConfigSnapshot, createTestRuntime } from "./test-runtime-config-helpers.js";
|
|
|
|
const catalogMocks = vi.hoisted(() => ({
|
|
listChannelPluginCatalogEntries: vi.fn((): ChannelPluginCatalogEntry[] => []),
|
|
}));
|
|
|
|
const manifestRegistryMocks = vi.hoisted(() => ({
|
|
loadPluginManifestRegistry: vi.fn(() => ({ plugins: [], diagnostics: [] })),
|
|
}));
|
|
|
|
vi.mock("../channels/plugins/catalog.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../channels/plugins/catalog.js")>();
|
|
return {
|
|
...actual,
|
|
listChannelPluginCatalogEntries: catalogMocks.listChannelPluginCatalogEntries,
|
|
};
|
|
});
|
|
|
|
vi.mock("../plugins/manifest-registry.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../plugins/manifest-registry.js")>();
|
|
return {
|
|
...actual,
|
|
loadPluginManifestRegistry: manifestRegistryMocks.loadPluginManifestRegistry,
|
|
};
|
|
});
|
|
|
|
vi.mock("./channel-setup/plugin-install.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("./channel-setup/plugin-install.js")>();
|
|
const { createMockChannelSetupPluginInstallModule } =
|
|
await import("./channels.plugin-install.test-helpers.js");
|
|
return createMockChannelSetupPluginInstallModule(actual);
|
|
});
|
|
|
|
const runtime = createTestRuntime();
|
|
let channelsAddCommand: typeof import("./channels.js").channelsAddCommand;
|
|
|
|
function registerMSTeamsSetupPlugin(pluginId = "@openclaw/msteams-plugin"): void {
|
|
vi.mocked(loadChannelSetupPluginRegistrySnapshotForChannel).mockReturnValue(
|
|
createTestRegistry([{ pluginId, plugin: createMSTeamsSetupPlugin(), source: "test" }]),
|
|
);
|
|
}
|
|
|
|
type SignalAfterAccountConfigWritten = NonNullable<
|
|
NonNullable<ChannelPlugin["setup"]>["afterAccountConfigWritten"]
|
|
>;
|
|
|
|
function createSignalPlugin(
|
|
afterAccountConfigWritten: SignalAfterAccountConfigWritten,
|
|
): ChannelPlugin {
|
|
return {
|
|
...createChannelTestPluginBase({
|
|
id: "signal",
|
|
label: "Signal",
|
|
}),
|
|
setup: {
|
|
applyAccountConfig: ({ cfg, accountId, input }) => ({
|
|
...cfg,
|
|
channels: {
|
|
...cfg.channels,
|
|
signal: {
|
|
enabled: true,
|
|
accounts: {
|
|
[accountId]: {
|
|
signalNumber: input.signalNumber,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
afterAccountConfigWritten,
|
|
},
|
|
} as ChannelPlugin;
|
|
}
|
|
|
|
async function runSignalAddCommand(afterAccountConfigWritten: SignalAfterAccountConfigWritten) {
|
|
const plugin = createSignalPlugin(afterAccountConfigWritten);
|
|
setActivePluginRegistry(createTestRegistry([{ pluginId: "signal", plugin, source: "test" }]));
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({ ...baseConfigSnapshot });
|
|
await channelsAddCommand(
|
|
{ channel: "signal", account: "ops", signalNumber: "+15550001" },
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
}
|
|
|
|
describe("channelsAddCommand", () => {
|
|
beforeAll(async () => {
|
|
({ channelsAddCommand } = await import("./channels.js"));
|
|
});
|
|
|
|
beforeEach(async () => {
|
|
configMocks.readConfigFileSnapshot.mockClear();
|
|
configMocks.writeConfigFile.mockClear();
|
|
offsetMocks.deleteTelegramUpdateOffset.mockClear();
|
|
runtime.log.mockClear();
|
|
runtime.error.mockClear();
|
|
runtime.exit.mockClear();
|
|
catalogMocks.listChannelPluginCatalogEntries.mockClear();
|
|
catalogMocks.listChannelPluginCatalogEntries.mockReturnValue([]);
|
|
manifestRegistryMocks.loadPluginManifestRegistry.mockClear();
|
|
manifestRegistryMocks.loadPluginManifestRegistry.mockReturnValue({
|
|
plugins: [],
|
|
diagnostics: [],
|
|
});
|
|
vi.mocked(ensureChannelSetupPluginInstalled).mockClear();
|
|
vi.mocked(ensureChannelSetupPluginInstalled).mockImplementation(async ({ cfg }) => ({
|
|
cfg,
|
|
installed: true,
|
|
}));
|
|
vi.mocked(loadChannelSetupPluginRegistrySnapshotForChannel).mockClear();
|
|
vi.mocked(loadChannelSetupPluginRegistrySnapshotForChannel).mockReturnValue(
|
|
createTestRegistry(),
|
|
);
|
|
setDefaultChannelPluginRegistryForTests();
|
|
});
|
|
|
|
it("clears telegram update offsets when the token changes", async () => {
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({
|
|
...baseConfigSnapshot,
|
|
config: {
|
|
channels: {
|
|
telegram: { botToken: "old-token", enabled: true },
|
|
},
|
|
},
|
|
});
|
|
|
|
await channelsAddCommand(
|
|
{ channel: "telegram", account: "default", token: "new-token" },
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
|
|
expect(offsetMocks.deleteTelegramUpdateOffset).toHaveBeenCalledTimes(1);
|
|
expect(offsetMocks.deleteTelegramUpdateOffset).toHaveBeenCalledWith({ accountId: "default" });
|
|
});
|
|
|
|
it("does not clear telegram update offsets when the token is unchanged", async () => {
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({
|
|
...baseConfigSnapshot,
|
|
config: {
|
|
channels: {
|
|
telegram: { botToken: "same-token", enabled: true },
|
|
},
|
|
},
|
|
});
|
|
|
|
await channelsAddCommand(
|
|
{ channel: "telegram", account: "default", token: "same-token" },
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
|
|
expect(offsetMocks.deleteTelegramUpdateOffset).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("falls back to a scoped snapshot after installing an external channel plugin", async () => {
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({ ...baseConfigSnapshot });
|
|
setActivePluginRegistry(createTestRegistry());
|
|
const catalogEntry = createMSTeamsCatalogEntry();
|
|
catalogMocks.listChannelPluginCatalogEntries.mockReturnValue([catalogEntry]);
|
|
registerMSTeamsSetupPlugin("msteams");
|
|
|
|
await channelsAddCommand(
|
|
{
|
|
channel: "msteams",
|
|
account: "default",
|
|
token: "tenant-scoped",
|
|
},
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
|
|
expect(ensureChannelSetupPluginInstalled).toHaveBeenCalledWith(
|
|
expect.objectContaining({ entry: catalogEntry }),
|
|
);
|
|
expect(loadChannelSetupPluginRegistrySnapshotForChannel).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channel: "msteams",
|
|
pluginId: "@openclaw/msteams-plugin",
|
|
}),
|
|
);
|
|
expect(configMocks.writeConfigFile).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channels: {
|
|
msteams: {
|
|
enabled: true,
|
|
tenantId: "tenant-scoped",
|
|
},
|
|
},
|
|
}),
|
|
);
|
|
expect(runtime.error).not.toHaveBeenCalled();
|
|
expect(runtime.exit).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("uses the installed external channel snapshot without reinstalling", async () => {
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({ ...baseConfigSnapshot });
|
|
setActivePluginRegistry(createTestRegistry());
|
|
const catalogEntry = createMSTeamsCatalogEntry();
|
|
catalogMocks.listChannelPluginCatalogEntries.mockReturnValue([catalogEntry]);
|
|
manifestRegistryMocks.loadPluginManifestRegistry.mockReturnValue({
|
|
plugins: [
|
|
{
|
|
id: "@openclaw/msteams-plugin",
|
|
channels: ["msteams"],
|
|
} as never,
|
|
],
|
|
diagnostics: [],
|
|
});
|
|
registerMSTeamsSetupPlugin("msteams");
|
|
|
|
await channelsAddCommand(
|
|
{
|
|
channel: "msteams",
|
|
account: "default",
|
|
token: "tenant-installed",
|
|
},
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
|
|
expect(ensureChannelSetupPluginInstalled).not.toHaveBeenCalled();
|
|
expect(loadChannelSetupPluginRegistrySnapshotForChannel).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channel: "msteams",
|
|
pluginId: "@openclaw/msteams-plugin",
|
|
}),
|
|
);
|
|
expect(configMocks.writeConfigFile).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channels: {
|
|
msteams: {
|
|
enabled: true,
|
|
tenantId: "tenant-installed",
|
|
},
|
|
},
|
|
}),
|
|
);
|
|
});
|
|
|
|
it("uses the installed plugin id when channel and plugin ids differ", async () => {
|
|
configMocks.readConfigFileSnapshot.mockResolvedValue({ ...baseConfigSnapshot });
|
|
setActivePluginRegistry(createTestRegistry());
|
|
const catalogEntry: ChannelPluginCatalogEntry = {
|
|
id: "msteams",
|
|
pluginId: "@openclaw/msteams-plugin",
|
|
meta: {
|
|
id: "msteams",
|
|
label: "Microsoft Teams",
|
|
selectionLabel: "Microsoft Teams",
|
|
docsPath: "/channels/msteams",
|
|
blurb: "teams channel",
|
|
},
|
|
install: {
|
|
npmSpec: "@openclaw/msteams",
|
|
},
|
|
};
|
|
catalogMocks.listChannelPluginCatalogEntries.mockReturnValue([catalogEntry]);
|
|
vi.mocked(ensureChannelSetupPluginInstalled).mockImplementation(async ({ cfg }) => ({
|
|
cfg,
|
|
installed: true,
|
|
pluginId: "@vendor/teams-runtime",
|
|
}));
|
|
vi.mocked(loadChannelSetupPluginRegistrySnapshotForChannel).mockReturnValue(
|
|
createTestRegistry([
|
|
{
|
|
pluginId: "@vendor/teams-runtime",
|
|
plugin: {
|
|
...createChannelTestPluginBase({
|
|
id: "msteams",
|
|
label: "Microsoft Teams",
|
|
docsPath: "/channels/msteams",
|
|
}),
|
|
setup: {
|
|
applyAccountConfig: vi.fn(({ cfg, input }) => ({
|
|
...cfg,
|
|
channels: {
|
|
...cfg.channels,
|
|
msteams: {
|
|
enabled: true,
|
|
tenantId: input.token,
|
|
},
|
|
},
|
|
})),
|
|
},
|
|
},
|
|
source: "test",
|
|
},
|
|
]),
|
|
);
|
|
|
|
await channelsAddCommand(
|
|
{
|
|
channel: "msteams",
|
|
account: "default",
|
|
token: "tenant-scoped",
|
|
},
|
|
runtime,
|
|
{ hasFlags: true },
|
|
);
|
|
|
|
expect(loadChannelSetupPluginRegistrySnapshotForChannel).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channel: "msteams",
|
|
pluginId: "@vendor/teams-runtime",
|
|
}),
|
|
);
|
|
expect(runtime.error).not.toHaveBeenCalled();
|
|
expect(runtime.exit).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("runs post-setup hooks after writing config", async () => {
|
|
const afterAccountConfigWritten = vi.fn().mockResolvedValue(undefined);
|
|
await runSignalAddCommand(afterAccountConfigWritten);
|
|
|
|
expect(configMocks.writeConfigFile).toHaveBeenCalledTimes(1);
|
|
expect(afterAccountConfigWritten).toHaveBeenCalledTimes(1);
|
|
expect(configMocks.writeConfigFile.mock.invocationCallOrder[0]).toBeLessThan(
|
|
afterAccountConfigWritten.mock.invocationCallOrder[0] ?? Number.POSITIVE_INFINITY,
|
|
);
|
|
expect(afterAccountConfigWritten).toHaveBeenCalledWith({
|
|
previousCfg: baseConfigSnapshot.config,
|
|
cfg: expect.objectContaining({
|
|
channels: {
|
|
signal: {
|
|
enabled: true,
|
|
accounts: {
|
|
ops: {
|
|
signalNumber: "+15550001",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
accountId: "ops",
|
|
input: expect.objectContaining({
|
|
signalNumber: "+15550001",
|
|
}),
|
|
runtime,
|
|
});
|
|
});
|
|
|
|
it("keeps the saved config when a post-setup hook fails", async () => {
|
|
const afterAccountConfigWritten = vi.fn().mockRejectedValue(new Error("hook failed"));
|
|
await runSignalAddCommand(afterAccountConfigWritten);
|
|
|
|
expect(configMocks.writeConfigFile).toHaveBeenCalledTimes(1);
|
|
expect(runtime.exit).not.toHaveBeenCalled();
|
|
expect(runtime.error).toHaveBeenCalledWith(
|
|
'Channel signal post-setup warning for "ops": hook failed',
|
|
);
|
|
});
|
|
});
|