mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 09:30:43 +00:00
120 lines
3.7 KiB
TypeScript
120 lines
3.7 KiB
TypeScript
import type { OpenClawPluginApi } from "openclaw/plugin-sdk/plugin-entry";
|
|
import { createTestPluginApi } from "openclaw/plugin-sdk/plugin-test-api";
|
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
import setupPlugin from "./setup-api.js";
|
|
|
|
const { createAcpxRuntimeServiceMock, tryDispatchAcpReplyHookMock } = vi.hoisted(() => ({
|
|
createAcpxRuntimeServiceMock: vi.fn(),
|
|
tryDispatchAcpReplyHookMock: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("./register.runtime.js", () => ({
|
|
createAcpxRuntimeService: createAcpxRuntimeServiceMock,
|
|
}));
|
|
|
|
vi.mock("openclaw/plugin-sdk/acp-runtime-backend", () => ({
|
|
tryDispatchAcpReplyHook: tryDispatchAcpReplyHookMock,
|
|
}));
|
|
|
|
import plugin from "./index.js";
|
|
|
|
type AcpxAutoEnableProbe = Parameters<OpenClawPluginApi["registerAutoEnableProbe"]>[0];
|
|
|
|
function registerAcpxAutoEnableProbe(): AcpxAutoEnableProbe {
|
|
const probes: AcpxAutoEnableProbe[] = [];
|
|
setupPlugin.register(
|
|
createTestPluginApi({
|
|
registerAutoEnableProbe(probe) {
|
|
probes.push(probe);
|
|
},
|
|
}),
|
|
);
|
|
const probe = probes[0];
|
|
if (!probe) {
|
|
throw new Error("expected ACPX setup plugin to register an auto-enable probe");
|
|
}
|
|
return probe;
|
|
}
|
|
|
|
describe("acpx plugin", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it("registers the runtime service and reply_dispatch hook", () => {
|
|
const service = { id: "acpx-service", start: vi.fn() };
|
|
createAcpxRuntimeServiceMock.mockReturnValue(service);
|
|
|
|
const api = {
|
|
pluginConfig: { stateDir: "/tmp/acpx" },
|
|
registerService: vi.fn(),
|
|
on: vi.fn(),
|
|
};
|
|
|
|
plugin.register(api as never);
|
|
|
|
expect(createAcpxRuntimeServiceMock).toHaveBeenCalledWith({
|
|
pluginConfig: api.pluginConfig,
|
|
});
|
|
expect(api.registerService).toHaveBeenCalledWith(service);
|
|
expect(api.on).toHaveBeenCalledWith("reply_dispatch", tryDispatchAcpReplyHookMock);
|
|
});
|
|
|
|
it("preserves the ACP reply_dispatch runtime path through the registered hook", async () => {
|
|
const service = { id: "acpx-service", start: vi.fn() };
|
|
createAcpxRuntimeServiceMock.mockReturnValue(service);
|
|
tryDispatchAcpReplyHookMock.mockResolvedValue({
|
|
handled: true,
|
|
queuedFinal: true,
|
|
counts: { tool: 1, block: 0, final: 1 },
|
|
});
|
|
|
|
const on = vi.fn();
|
|
const api = createTestPluginApi({
|
|
pluginConfig: { stateDir: "/tmp/acpx" },
|
|
registerService: vi.fn(),
|
|
on,
|
|
});
|
|
|
|
plugin.register(api);
|
|
|
|
const hook = on.mock.calls.find(([hookName]) => hookName === "reply_dispatch")?.[1];
|
|
if (!hook) {
|
|
throw new Error("expected reply_dispatch hook to be registered");
|
|
}
|
|
|
|
const event = {
|
|
ctx: { raw: "reply ctx" },
|
|
runId: "run-1",
|
|
sessionKey: "agent:test:session",
|
|
inboundAudio: false,
|
|
shouldRouteToOriginating: false,
|
|
shouldSendToolSummaries: true,
|
|
sendPolicy: "allow",
|
|
};
|
|
const ctx = {
|
|
cfg: {},
|
|
dispatcher: { dispatch: vi.fn(), getQueuedCounts: vi.fn(), getFailedCounts: vi.fn() },
|
|
recordProcessed: vi.fn(),
|
|
markIdle: vi.fn(),
|
|
};
|
|
|
|
await expect(hook(event, ctx)).resolves.toEqual({
|
|
handled: true,
|
|
queuedFinal: true,
|
|
counts: { tool: 1, block: 0, final: 1 },
|
|
});
|
|
expect(tryDispatchAcpReplyHookMock).toHaveBeenCalledWith(event, ctx);
|
|
});
|
|
|
|
it("declares setup auto-enable reasons for ACPX-owned ACP config", () => {
|
|
const probe = registerAcpxAutoEnableProbe();
|
|
|
|
expect(probe({ config: { acp: { enabled: true } }, env: {} })).toBe("ACP runtime configured");
|
|
expect(probe({ config: { acp: { backend: "acpx" } }, env: {} })).toBe("ACP runtime configured");
|
|
expect(probe({ config: { acp: { enabled: true, backend: "custom-runtime" } }, env: {} })).toBe(
|
|
null,
|
|
);
|
|
});
|
|
});
|