Files
openclaw/test/scripts/mcp-channels-harness.test.ts
2026-06-04 20:49:50 -04:00

101 lines
3.6 KiB
TypeScript

// Mcp Channels Harness tests cover mcp channels harness script behavior.
import { existsSync, mkdtempSync, readFileSync, rmSync, statSync } from "node:fs";
import { tmpdir } from "node:os";
import path from "node:path";
import { describe, expect, it, vi } from "vitest";
import {
connectMcpClientWithPairingReconnect,
createMcpClientTempState,
type McpClientTempState,
} from "../../scripts/e2e/mcp-client-temp-state.js";
describe("mcp-channels harness", () => {
it("creates unique client temp state and removes token files on cleanup", () => {
const tempRoot = mkdtempSync(path.join(tmpdir(), "openclaw-mcp-harness-test-"));
try {
const first = createMcpClientTempState({ gatewayToken: "first-token", tempRoot });
const second = createMcpClientTempState({ gatewayToken: "second-token", tempRoot });
expect(first.root).not.toBe(second.root);
expect(first.stateDir).toBe(path.join(first.root, "state"));
expect(readFileSync(first.tokenFile, "utf8")).toBe("first-token\n");
expect(statSync(first.tokenFile).mode & 0o777).toBe(0o600);
expect(readFileSync(second.tokenFile, "utf8")).toBe("second-token\n");
first.cleanup();
second.cleanup();
expect(existsSync(first.root)).toBe(false);
expect(existsSync(second.root)).toBe(false);
} finally {
rmSync(tempRoot, { force: true, recursive: true });
}
});
it("reuses one MCP temp state across the pairing reconnect path", async () => {
const tempState = createMcpClientTempState({ gatewayToken: "pairing-token" });
const firstHandle = {
cleanup: vi.fn(),
client: { close: vi.fn(async () => undefined) },
transport: { close: vi.fn(async () => undefined) },
};
const secondHandle = {
cleanup: vi.fn(),
client: { close: vi.fn(async () => undefined) },
transport: { close: vi.fn(async () => undefined) },
};
const connectCalls: McpClientTempState[] = [];
const connect = vi.fn(async (state: McpClientTempState) => {
connectCalls.push(state);
return connectCalls.length === 1 ? firstHandle : secondHandle;
});
try {
await expect(
connectMcpClientWithPairingReconnect({
connect,
maybeApprovePairing: async () => true,
tempState,
}),
).resolves.toBe(secondHandle);
expect(connect).toHaveBeenCalledTimes(2);
expect(connectCalls).toEqual([tempState, tempState]);
expect(firstHandle.client.close).toHaveBeenCalledOnce();
expect(firstHandle.transport.close).toHaveBeenCalledOnce();
expect(firstHandle.cleanup).toHaveBeenCalledOnce();
expect(secondHandle.cleanup).not.toHaveBeenCalled();
} finally {
tempState.cleanup();
}
});
it("cleans up the first MCP client when pairing approval fails", async () => {
const tempState = createMcpClientTempState({ gatewayToken: "pairing-token" });
const handle = {
cleanup: vi.fn(),
client: { close: vi.fn(async () => undefined) },
transport: { close: vi.fn(async () => undefined) },
};
const failure = new Error("pairing approval failed");
try {
await expect(
connectMcpClientWithPairingReconnect({
connect: async () => handle,
maybeApprovePairing: async () => {
throw failure;
},
tempState,
}),
).rejects.toBe(failure);
expect(handle.client.close).toHaveBeenCalledOnce();
expect(handle.transport.close).toHaveBeenCalledOnce();
expect(handle.cleanup).toHaveBeenCalledOnce();
} finally {
tempState.cleanup();
}
});
});