mirror of https://github.com/openclaw/openclaw.git
160 lines
4.6 KiB
TypeScript
160 lines
4.6 KiB
TypeScript
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import { withFetchPreconnect } from "../../test/helpers/extensions/fetch-mock.js";
|
|
import { createCodeExecutionTool } from "./code-execution.js";
|
|
|
|
function installCodeExecutionFetch(payload?: Record<string, unknown>) {
|
|
const mockFetch = vi.fn((_input?: unknown, _init?: unknown) =>
|
|
Promise.resolve({
|
|
ok: true,
|
|
json: () =>
|
|
Promise.resolve(
|
|
payload ?? {
|
|
output: [
|
|
{ type: "code_interpreter_call" },
|
|
{
|
|
type: "message",
|
|
content: [
|
|
{
|
|
type: "output_text",
|
|
text: "Mean: 42",
|
|
annotations: [{ type: "url_citation", url: "https://example.com/data.csv" }],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
citations: ["https://example.com/data.csv"],
|
|
},
|
|
),
|
|
} as Response),
|
|
);
|
|
global.fetch = withFetchPreconnect(mockFetch);
|
|
return mockFetch;
|
|
}
|
|
|
|
function parseFirstRequestBody(mockFetch: ReturnType<typeof installCodeExecutionFetch>) {
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
const requestBody = request?.body;
|
|
return JSON.parse(typeof requestBody === "string" ? requestBody : "{}") as Record<
|
|
string,
|
|
unknown
|
|
>;
|
|
}
|
|
|
|
afterEach(() => {
|
|
vi.restoreAllMocks();
|
|
});
|
|
|
|
describe("xai code_execution tool", () => {
|
|
it("enables code_execution when the xAI plugin web search key is configured", () => {
|
|
const tool = createCodeExecutionTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-plugin-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(tool?.name).toBe("code_execution");
|
|
});
|
|
|
|
it("uses the xAI Responses code_interpreter tool", async () => {
|
|
const mockFetch = installCodeExecutionFetch();
|
|
const tool = createCodeExecutionTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-config-test", // pragma: allowlist secret
|
|
},
|
|
codeExecution: {
|
|
model: "grok-4-1-fast",
|
|
maxTurns: 2,
|
|
timeoutSeconds: 45,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
const result = await tool?.execute?.("code-execution:1", {
|
|
task: "Calculate the mean of [40, 42, 44]",
|
|
});
|
|
|
|
expect(mockFetch).toHaveBeenCalled();
|
|
expect(String(mockFetch.mock.calls[0]?.[0])).toContain("api.x.ai/v1/responses");
|
|
const body = parseFirstRequestBody(mockFetch);
|
|
expect(body.model).toBe("grok-4-1-fast");
|
|
expect(body.max_turns).toBe(2);
|
|
expect(body.tools).toEqual([{ type: "code_interpreter" }]);
|
|
expect(
|
|
(result?.details as { usedCodeExecution?: boolean } | undefined)?.usedCodeExecution,
|
|
).toBe(true);
|
|
});
|
|
|
|
it("reuses the xAI plugin web search key for code_execution requests", async () => {
|
|
const mockFetch = installCodeExecutionFetch();
|
|
const tool = createCodeExecutionTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-plugin-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("code-execution:plugin-key", {
|
|
task: "Compute the standard deviation of [1, 2, 3]",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer xai-plugin-key",
|
|
);
|
|
});
|
|
|
|
it("reuses the legacy grok web search key for code_execution requests", async () => {
|
|
const mockFetch = installCodeExecutionFetch();
|
|
const tool = createCodeExecutionTool({
|
|
config: {
|
|
tools: {
|
|
web: {
|
|
search: {
|
|
grok: {
|
|
apiKey: "xai-legacy-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("code-execution:legacy-key", {
|
|
task: "Count rows in a two-column table",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer xai-legacy-key",
|
|
);
|
|
});
|
|
});
|