mirror of
https://github.com/openclaw/openclaw.git
synced 2026-07-01 18:33:38 +00:00
Merged via squash.
Prepared head SHA: 33de08d9a1
Co-authored-by: moeedahmed <5780040+moeedahmed@users.noreply.github.com>
Co-authored-by: vincentkoc <25068+vincentkoc@users.noreply.github.com>
Reviewed-by: @vincentkoc
226 lines
6.9 KiB
TypeScript
226 lines
6.9 KiB
TypeScript
// Tokenjuice tests cover index plugin behavior.
|
|
import fs from "node:fs";
|
|
import { createTestPluginApi } from "openclaw/plugin-sdk/plugin-test-api";
|
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const { tokenjuiceFactory, createTokenjuiceOpenClawEmbeddedExtension } = vi.hoisted(() => {
|
|
const tokenjuiceFactoryLocal = vi.fn();
|
|
const createTokenjuiceOpenClawEmbeddedExtensionLocal = vi.fn(() => tokenjuiceFactoryLocal);
|
|
return {
|
|
tokenjuiceFactory: tokenjuiceFactoryLocal,
|
|
createTokenjuiceOpenClawEmbeddedExtension: createTokenjuiceOpenClawEmbeddedExtensionLocal,
|
|
};
|
|
});
|
|
|
|
vi.mock("./runtime-api.js", () => ({
|
|
createTokenjuiceOpenClawEmbeddedExtension,
|
|
}));
|
|
|
|
import plugin from "./index.js";
|
|
import { createTokenjuiceAgentToolResultMiddleware } from "./tool-result-middleware.js";
|
|
|
|
describe("tokenjuice plugin", () => {
|
|
beforeEach(() => {
|
|
createTokenjuiceOpenClawEmbeddedExtension.mockClear();
|
|
tokenjuiceFactory.mockClear();
|
|
});
|
|
|
|
it("is opt-in by default", () => {
|
|
const manifest = JSON.parse(
|
|
fs.readFileSync(new URL("./openclaw.plugin.json", import.meta.url), "utf8"),
|
|
) as { enabledByDefault?: unknown };
|
|
|
|
expect(manifest.enabledByDefault).toBeUndefined();
|
|
});
|
|
|
|
it("registers tokenjuice tool result middleware for OpenClaw and Codex runtimes", () => {
|
|
const registerAgentToolResultMiddleware = vi.fn();
|
|
|
|
plugin.register(
|
|
createTestPluginApi({
|
|
id: "tokenjuice",
|
|
name: "tokenjuice",
|
|
source: "test",
|
|
config: {},
|
|
pluginConfig: {},
|
|
runtime: {} as never,
|
|
registerAgentToolResultMiddleware,
|
|
}),
|
|
);
|
|
|
|
expect(createTokenjuiceOpenClawEmbeddedExtension).toHaveBeenCalledTimes(1);
|
|
expect(tokenjuiceFactory).toHaveBeenCalledTimes(1);
|
|
const registration = registerAgentToolResultMiddleware.mock.calls[0];
|
|
expect(typeof registration?.[0]).toBe("function");
|
|
expect(registration?.[1]).toEqual({ runtimes: ["openclaw", "codex"] });
|
|
});
|
|
|
|
it("synthesises exec fields when bash provides metadata-only details (no status)", async () => {
|
|
let received:
|
|
| {
|
|
details: unknown;
|
|
}
|
|
| undefined;
|
|
tokenjuiceFactory.mockImplementationOnce(
|
|
(api: { on: (event: string, handler: unknown) => void }) => {
|
|
api.on("tool_result", async (event: typeof received) => {
|
|
received = event;
|
|
});
|
|
},
|
|
);
|
|
|
|
const middleware = createTokenjuiceAgentToolResultMiddleware();
|
|
await middleware(
|
|
{
|
|
toolCallId: "tool-call-tokenjuice-bash-meta",
|
|
toolName: "bash",
|
|
args: { command: "cat /tmp/out.txt", workdir: "/tmp/openclaw-tokenjuice-test" },
|
|
result: {
|
|
content: [{ type: "text", text: "file contents\n" }],
|
|
details: {
|
|
truncation: { reason: "max_bytes" },
|
|
fullOutputPath: "/tmp/out.txt",
|
|
},
|
|
},
|
|
isError: false,
|
|
},
|
|
{ runtime: "openclaw" },
|
|
);
|
|
|
|
expect(received?.details).toMatchObject({
|
|
status: "completed",
|
|
aggregated: "file contents\n",
|
|
exitCode: 0,
|
|
truncation: { reason: "max_bytes" },
|
|
fullOutputPath: "/tmp/out.txt",
|
|
});
|
|
});
|
|
|
|
it("passes through bash details that already have a status field unchanged", async () => {
|
|
let received:
|
|
| {
|
|
details: unknown;
|
|
}
|
|
| undefined;
|
|
tokenjuiceFactory.mockImplementationOnce(
|
|
(api: { on: (event: string, handler: unknown) => void }) => {
|
|
api.on("tool_result", async (event: typeof received) => {
|
|
received = event;
|
|
});
|
|
},
|
|
);
|
|
|
|
const existingDetails = {
|
|
status: "completed",
|
|
aggregated: "pre-built output",
|
|
exitCode: 0,
|
|
cwd: "/existing/cwd",
|
|
};
|
|
|
|
const middleware = createTokenjuiceAgentToolResultMiddleware();
|
|
await middleware(
|
|
{
|
|
toolCallId: "tool-call-tokenjuice-bash-existing",
|
|
toolName: "bash",
|
|
args: { command: "echo hi", workdir: "/tmp" },
|
|
result: {
|
|
content: [{ type: "text", text: "hi\n" }],
|
|
details: existingDetails,
|
|
},
|
|
isError: false,
|
|
},
|
|
{ runtime: "openclaw" },
|
|
);
|
|
|
|
expect(received?.details).toBe(existingDetails);
|
|
});
|
|
|
|
it.each([
|
|
["exit code", { exitCode: 7 }, "failed", 7],
|
|
["success flag", { success: false }, "failed", 1],
|
|
["ok flag", { ok: false }, "failed", 1],
|
|
["timeout flag", { timedOut: true }, "failed", 1],
|
|
["error value", { error: "command failed" }, "failed", 1],
|
|
["successful exit code", { exitCode: 0 }, "completed", 0],
|
|
["successful flag", { success: true }, "completed", 0],
|
|
])(
|
|
"adds a canonical status while preserving bash details with a %s",
|
|
async (_label, existingDetails, status, exitCode) => {
|
|
let received:
|
|
| {
|
|
details: unknown;
|
|
}
|
|
| undefined;
|
|
tokenjuiceFactory.mockImplementationOnce(
|
|
(api: { on: (event: string, handler: unknown) => void }) => {
|
|
api.on("tool_result", async (event: typeof received) => {
|
|
received = event;
|
|
});
|
|
},
|
|
);
|
|
|
|
const middleware = createTokenjuiceAgentToolResultMiddleware();
|
|
await middleware(
|
|
{
|
|
toolCallId: "tool-call-tokenjuice-bash-terminal",
|
|
toolName: "bash",
|
|
args: { command: "exit 7", workdir: "/tmp" },
|
|
result: {
|
|
content: [{ type: "text", text: "failed\n" }],
|
|
details: existingDetails,
|
|
},
|
|
isError: false,
|
|
},
|
|
{ runtime: "openclaw" },
|
|
);
|
|
|
|
expect(received?.details).toMatchObject({
|
|
...existingDetails,
|
|
status,
|
|
exitCode,
|
|
});
|
|
},
|
|
);
|
|
|
|
it("normalizes bash results without details before passing them to tokenjuice", async () => {
|
|
let received:
|
|
| {
|
|
toolName: string;
|
|
input: Record<string, unknown>;
|
|
content: unknown;
|
|
details: unknown;
|
|
isError?: boolean;
|
|
}
|
|
| undefined;
|
|
tokenjuiceFactory.mockImplementationOnce(
|
|
(api: { on: (event: string, handler: unknown) => void }) => {
|
|
api.on("tool_result", async (event: typeof received) => {
|
|
received = event;
|
|
return { content: [{ type: "text", text: "compacted" }] };
|
|
});
|
|
},
|
|
);
|
|
|
|
const middleware = createTokenjuiceAgentToolResultMiddleware();
|
|
const result = await middleware(
|
|
{
|
|
toolCallId: "tool-call-tokenjuice-bash",
|
|
toolName: "bash",
|
|
args: { command: "printf 'hello\\n'", workdir: "/tmp/openclaw-tokenjuice-test" },
|
|
result: { content: [{ type: "text", text: "hello\n" }], details: undefined },
|
|
isError: false,
|
|
},
|
|
{ runtime: "openclaw" },
|
|
);
|
|
|
|
expect(received?.toolName).toBe("bash");
|
|
expect(received?.details).toMatchObject({
|
|
status: "completed",
|
|
aggregated: "hello\n",
|
|
exitCode: 0,
|
|
});
|
|
expect(received?.details).not.toHaveProperty("cwd");
|
|
expect(result?.result.content).toEqual([{ type: "text", text: "compacted" }]);
|
|
});
|
|
});
|