mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-18 22:57:28 +00:00
101 lines
3.1 KiB
TypeScript
101 lines
3.1 KiB
TypeScript
import type { AgentTool } from "@mariozechner/pi-agent-core";
|
|
import { Type } from "@sinclair/typebox";
|
|
import { describe, expect, it } from "vitest";
|
|
import { toToolDefinitions } from "./pi-tool-definition-adapter.js";
|
|
|
|
type ToolExecute = ReturnType<typeof toToolDefinitions>[number]["execute"];
|
|
const extensionContext = {} as Parameters<ToolExecute>[4];
|
|
|
|
async function executeThrowingTool(name: string, callId: string) {
|
|
const tool = {
|
|
name,
|
|
label: name === "bash" ? "Bash" : "Boom",
|
|
description: "throws",
|
|
parameters: Type.Object({}),
|
|
execute: async () => {
|
|
throw new Error("nope");
|
|
},
|
|
} satisfies AgentTool;
|
|
|
|
const defs = toToolDefinitions([tool]);
|
|
const def = defs[0];
|
|
if (!def) {
|
|
throw new Error("missing tool definition");
|
|
}
|
|
return await def.execute(callId, {}, undefined, undefined, extensionContext);
|
|
}
|
|
|
|
async function executeTool(tool: AgentTool, callId: string) {
|
|
const defs = toToolDefinitions([tool]);
|
|
const def = defs[0];
|
|
if (!def) {
|
|
throw new Error("missing tool definition");
|
|
}
|
|
return await def.execute(callId, {}, undefined, undefined, extensionContext);
|
|
}
|
|
|
|
describe("pi tool definition adapter", () => {
|
|
it("wraps tool errors into a tool result", async () => {
|
|
const result = await executeThrowingTool("boom", "call1");
|
|
|
|
expect(result.details).toMatchObject({
|
|
status: "error",
|
|
tool: "boom",
|
|
});
|
|
expect(result.details).toMatchObject({ error: "nope" });
|
|
expect(JSON.stringify(result.details)).not.toContain("\n at ");
|
|
});
|
|
|
|
it("normalizes exec tool aliases in error results", async () => {
|
|
const result = await executeThrowingTool("bash", "call2");
|
|
|
|
expect(result.details).toMatchObject({
|
|
status: "error",
|
|
tool: "exec",
|
|
error: "nope",
|
|
});
|
|
});
|
|
|
|
it("coerces details-only tool results to include content", async () => {
|
|
const tool = {
|
|
name: "memory_query",
|
|
label: "Memory Query",
|
|
description: "returns details only",
|
|
parameters: Type.Object({}),
|
|
execute: (async () => ({
|
|
details: {
|
|
hits: [{ id: "a1", score: 0.9 }],
|
|
},
|
|
})) as unknown as AgentTool["execute"],
|
|
} satisfies AgentTool;
|
|
|
|
const result = await executeTool(tool, "call3");
|
|
expect(result.details).toEqual({
|
|
hits: [{ id: "a1", score: 0.9 }],
|
|
});
|
|
expect(result.content[0]).toMatchObject({ type: "text" });
|
|
expect((result.content[0] as { text?: string }).text).toContain('"hits"');
|
|
});
|
|
|
|
it("coerces non-standard object results to include content", async () => {
|
|
const tool = {
|
|
name: "memory_query_raw",
|
|
label: "Memory Query Raw",
|
|
description: "returns plain object",
|
|
parameters: Type.Object({}),
|
|
execute: (async () => ({
|
|
count: 2,
|
|
ids: ["m1", "m2"],
|
|
})) as unknown as AgentTool["execute"],
|
|
} satisfies AgentTool;
|
|
|
|
const result = await executeTool(tool, "call4");
|
|
expect(result.details).toEqual({
|
|
count: 2,
|
|
ids: ["m1", "m2"],
|
|
});
|
|
expect(result.content[0]).toMatchObject({ type: "text" });
|
|
expect((result.content[0] as { text?: string }).text).toContain('"count"');
|
|
});
|
|
});
|