mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-11 17:22:58 +00:00
181 lines
5.3 KiB
TypeScript
181 lines
5.3 KiB
TypeScript
import { mkdtemp, mkdir, rm, symlink, writeFile } from "node:fs/promises";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import { createCanvasTool } from "./tool.js";
|
|
|
|
const mocks = vi.hoisted(() => ({
|
|
callGatewayTool: vi.fn(),
|
|
imageResultFromFile: vi.fn(async (params) => ({ content: [], details: params })),
|
|
listNodes: vi.fn(async () => []),
|
|
resolveNodeIdFromList: vi.fn(() => "node-1"),
|
|
}));
|
|
|
|
vi.mock("openclaw/plugin-sdk/agent-harness-runtime", () => ({
|
|
callGatewayTool: mocks.callGatewayTool,
|
|
listNodes: mocks.listNodes,
|
|
resolveNodeIdFromList: mocks.resolveNodeIdFromList,
|
|
}));
|
|
|
|
vi.mock("openclaw/plugin-sdk/channel-actions", async (importOriginal) => ({
|
|
...(await importOriginal<typeof import("openclaw/plugin-sdk/channel-actions")>()),
|
|
imageResultFromFile: mocks.imageResultFromFile,
|
|
}));
|
|
|
|
describe("Canvas tool", () => {
|
|
let tempRoot: string | undefined;
|
|
|
|
beforeEach(() => {
|
|
mocks.callGatewayTool.mockReset();
|
|
mocks.imageResultFromFile.mockClear();
|
|
mocks.listNodes.mockClear();
|
|
mocks.listNodes.mockResolvedValue([]);
|
|
mocks.resolveNodeIdFromList.mockClear();
|
|
mocks.resolveNodeIdFromList.mockReturnValue("node-1");
|
|
});
|
|
|
|
afterEach(async () => {
|
|
if (tempRoot) {
|
|
await rm(tempRoot, { recursive: true, force: true });
|
|
tempRoot = undefined;
|
|
}
|
|
});
|
|
|
|
it.skipIf(process.platform === "win32")(
|
|
"rejects jsonlPath symlinks that resolve outside the workspace",
|
|
async () => {
|
|
tempRoot = await mkdtemp(path.join(os.tmpdir(), "openclaw-canvas-tool-"));
|
|
const workspaceDir = path.join(tempRoot, "workspace");
|
|
await mkdir(workspaceDir);
|
|
const outsidePath = path.join(tempRoot, "outside.jsonl");
|
|
await writeFile(outsidePath, '{"secret":true}\n');
|
|
await symlink(outsidePath, path.join(workspaceDir, "events.jsonl"));
|
|
|
|
const tool = createCanvasTool({ workspaceDir });
|
|
|
|
await expect(
|
|
tool.execute("tool-call-1", {
|
|
action: "a2ui_push",
|
|
jsonlPath: "events.jsonl",
|
|
}),
|
|
).rejects.toThrow("jsonlPath outside workspace");
|
|
expect(mocks.callGatewayTool).not.toHaveBeenCalled();
|
|
},
|
|
);
|
|
|
|
it("applies configured image limits to canvas snapshots", async () => {
|
|
mocks.callGatewayTool.mockResolvedValue({
|
|
payload: {
|
|
format: "png",
|
|
base64: Buffer.from("not-a-real-png").toString("base64"),
|
|
},
|
|
});
|
|
const tool = createCanvasTool({
|
|
config: {
|
|
agents: {
|
|
defaults: {
|
|
imageMaxDimensionPx: 1600.9,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool.execute("tool-call-1", { action: "snapshot" });
|
|
|
|
expect(mocks.imageResultFromFile).toHaveBeenCalledTimes(1);
|
|
const imageResultParams = mocks.imageResultFromFile.mock.calls[0]?.[0] as
|
|
| {
|
|
label?: string;
|
|
path?: string;
|
|
details?: unknown;
|
|
imageSanitization?: unknown;
|
|
}
|
|
| undefined;
|
|
expect(imageResultParams?.label).toBe("canvas:snapshot");
|
|
expect(imageResultParams?.path).toMatch(/openclaw-canvas-snapshot-.*\.png$/);
|
|
expect(imageResultParams?.details).toEqual({ format: "png" });
|
|
expect(imageResultParams?.imageSanitization).toEqual({ maxDimensionPx: 1600 });
|
|
});
|
|
|
|
it("normalizes numeric string params before invoking node canvas commands", async () => {
|
|
mocks.callGatewayTool.mockResolvedValue({
|
|
payload: {
|
|
format: "png",
|
|
base64: Buffer.from("not-a-real-png").toString("base64"),
|
|
},
|
|
});
|
|
const tool = createCanvasTool();
|
|
|
|
await tool.execute("tool-call-1", {
|
|
action: "present",
|
|
timeoutMs: "1500",
|
|
x: "10.5",
|
|
y: "-2",
|
|
width: "640",
|
|
height: "480",
|
|
});
|
|
|
|
expect(mocks.callGatewayTool).toHaveBeenLastCalledWith(
|
|
"node.invoke",
|
|
{ timeoutMs: 1500 },
|
|
expect.objectContaining({
|
|
command: "canvas.present",
|
|
params: {
|
|
placement: {
|
|
x: 10.5,
|
|
y: -2,
|
|
width: 640,
|
|
height: 480,
|
|
},
|
|
},
|
|
}),
|
|
);
|
|
|
|
await tool.execute("tool-call-2", {
|
|
action: "snapshot",
|
|
maxWidth: "800",
|
|
quality: "0.75",
|
|
});
|
|
|
|
expect(mocks.callGatewayTool).toHaveBeenLastCalledWith(
|
|
"node.invoke",
|
|
{},
|
|
expect.objectContaining({
|
|
command: "canvas.snapshot",
|
|
params: {
|
|
format: "png",
|
|
maxWidth: 800,
|
|
quality: 0.75,
|
|
},
|
|
}),
|
|
);
|
|
});
|
|
|
|
it("rejects malformed numeric canvas params before invoking node commands", async () => {
|
|
const tool = createCanvasTool();
|
|
|
|
await expect(
|
|
tool.execute("tool-call-1", {
|
|
action: "snapshot",
|
|
maxWidth: "800px",
|
|
}),
|
|
).rejects.toThrow("maxWidth must be a positive integer");
|
|
expect(mocks.callGatewayTool).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("rejects node-controlled snapshot formats before creating image results", async () => {
|
|
mocks.callGatewayTool.mockResolvedValue({
|
|
payload: {
|
|
format: "/../../target.sh",
|
|
base64: Buffer.from("not-a-real-png").toString("base64"),
|
|
},
|
|
});
|
|
const tool = createCanvasTool();
|
|
|
|
await expect(tool.execute("tool-call-1", { action: "snapshot" })).rejects.toThrow(
|
|
/invalid canvas\.snapshot payload/i,
|
|
);
|
|
expect(mocks.imageResultFromFile).not.toHaveBeenCalled();
|
|
});
|
|
});
|