Files
openclaw/extensions/tokenjuice/index.test.ts
Moeed Ahmed ecd0d17243 Fix tokenjuice bash results without details (#93269)
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
2026-06-16 18:53:52 +08:00

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" }]);
});
});