mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-06 05:51:15 +08:00
101 lines
3.6 KiB
TypeScript
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();
|
|
}
|
|
});
|
|
});
|